summaryrefslogtreecommitdiffstats
path: root/osdep/io.c
diff options
context:
space:
mode:
authorJames Ross-Gowan <rossymiles@gmail.com>2014-09-05 01:36:15 +1000
committerwm4 <wm4@nowhere>2014-09-05 17:51:44 +0200
commit5c3f3fd3dab3e399c473790f55272c5d6aec2764 (patch)
tree0b3cd4e30dd66ca3347c5c52682bfa2eee36f2ef /osdep/io.c
parent15a882d2ed7e9eee5c2b1166189698291e76b19b (diff)
downloadmpv-5c3f3fd3dab3e399c473790f55272c5d6aec2764.tar.bz2
mpv-5c3f3fd3dab3e399c473790f55272c5d6aec2764.tar.xz
win32: add tmpfile() replacement
The Windows version of tmpfile is actually pretty broken. It tries to create the file in the root directory of the current drive, which means on Vista and up, it normally fails due to insufficient permissions. Replace it with a version that uses GetTempPath. Also remove the Windows-specific note about automatic deletion of the cache file. FILE_FLAG_DELETE_ON_CLOSE is available in NT, and it should be pretty reliable.
Diffstat (limited to 'osdep/io.c')
-rw-r--r--osdep/io.c35
1 files changed, 35 insertions, 0 deletions
diff --git a/osdep/io.c b/osdep/io.c
index e0caa56f79..819cdcbf69 100644
--- a/osdep/io.c
+++ b/osdep/io.c
@@ -310,6 +310,41 @@ int mp_mkdir(const char *path, int mode)
return res;
}
+FILE *mp_tmpfile(void)
+{
+ // Reserve a file name in the format %TMP%\mpvXXXX.TMP
+ wchar_t tmp_path[MAX_PATH + 1];
+ if (!GetTempPathW(MAX_PATH + 1, tmp_path))
+ return NULL;
+ wchar_t tmp_name[MAX_PATH + 1];
+ if (!GetTempFileNameW(tmp_path, L"mpv", 0, tmp_name))
+ return NULL;
+
+ // Create the file. FILE_ATTRIBUTE_TEMPORARY indicates the file will be
+ // short-lived. Windows should avoid flushing it to disk while there is
+ // sufficient cache.
+ HANDLE file = CreateFileW(tmp_name, GENERIC_READ | GENERIC_WRITE | DELETE,
+ FILE_SHARE_READ | FILE_SHARE_DELETE, NULL, CREATE_ALWAYS,
+ FILE_ATTRIBUTE_TEMPORARY | FILE_FLAG_DELETE_ON_CLOSE, NULL);
+ if (file == INVALID_HANDLE_VALUE) {
+ DeleteFileW(tmp_name);
+ return NULL;
+ }
+
+ int fd = _open_osfhandle((intptr_t)file, 0);
+ if (fd < 0) {
+ CloseHandle(file);
+ return NULL;
+ }
+ FILE *fp = fdopen(fd, "w+b");
+ if (!fp) {
+ close(fd);
+ return NULL;
+ }
+
+ return fp;
+}
+
static char **utf8_environ;
static void *utf8_environ_ctx;