2017-10-05 23:32:57 -07:00
|
|
|
/* mz_os_win32.c -- System functions for Windows
|
2021-01-23 16:18:11 -08:00
|
|
|
part of the minizip-ng project
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2023-02-16 13:14:21 -08:00
|
|
|
Copyright (C) Nathan Moinvaziri
|
2021-01-23 16:18:11 -08:00
|
|
|
https://github.com/zlib-ng/minizip-ng
|
2017-10-05 23:32:57 -07:00
|
|
|
|
|
|
|
This program is distributed under the terms of the same license as zlib.
|
|
|
|
See the accompanying LICENSE file for the full text of the license.
|
|
|
|
*/
|
|
|
|
|
2017-10-16 07:37:11 -07:00
|
|
|
#include "mz.h"
|
2017-10-16 14:50:31 -07:00
|
|
|
#include "mz_os.h"
|
2018-10-24 09:50:16 -07:00
|
|
|
#include "mz_strm_os.h"
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2018-11-19 21:34:35 -08:00
|
|
|
#include <windows.h>
|
2022-11-29 15:53:50 -08:00
|
|
|
#include <winioctl.h>
|
2018-11-19 21:34:35 -08:00
|
|
|
|
2017-10-05 23:32:57 -07:00
|
|
|
/***************************************************************************/
|
|
|
|
|
2019-09-25 11:49:17 -07:00
|
|
|
#ifndef SYMBOLIC_LINK_FLAG_DIRECTORY
|
|
|
|
# define SYMBOLIC_LINK_FLAG_DIRECTORY 0x1
|
|
|
|
#endif
|
|
|
|
|
2023-04-19 09:15:06 -07:00
|
|
|
#ifndef _WIN32_WINNT_WIN8
|
|
|
|
# define _WIN32_WINNT_WIN8 0x0602
|
|
|
|
#endif
|
|
|
|
|
2017-10-16 20:34:01 -07:00
|
|
|
/***************************************************************************/
|
|
|
|
|
2017-10-16 14:50:31 -07:00
|
|
|
typedef struct DIR_int_s {
|
|
|
|
void *find_handle;
|
|
|
|
WIN32_FIND_DATAW find_data;
|
|
|
|
struct dirent entry;
|
|
|
|
uint8_t end;
|
|
|
|
} DIR_int;
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
wchar_t *mz_os_unicode_string_create(const char *string, int32_t encoding) {
|
2018-10-23 09:04:04 -07:00
|
|
|
wchar_t *string_wide = NULL;
|
|
|
|
uint32_t string_wide_size = 0;
|
2017-10-23 18:27:18 -07:00
|
|
|
|
2018-10-30 13:59:17 -07:00
|
|
|
string_wide_size = MultiByteToWideChar(encoding, 0, string, -1, NULL, 0);
|
|
|
|
if (string_wide_size == 0)
|
|
|
|
return NULL;
|
2023-02-24 14:35:31 -08:00
|
|
|
string_wide = (wchar_t *)calloc(string_wide_size + 1, sizeof(wchar_t));
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!string_wide)
|
2018-12-01 08:59:21 -08:00
|
|
|
return NULL;
|
2017-10-23 18:27:18 -07:00
|
|
|
|
2018-10-30 13:59:17 -07:00
|
|
|
MultiByteToWideChar(encoding, 0, string, -1, string_wide, string_wide_size);
|
2018-10-23 09:04:04 -07:00
|
|
|
return string_wide;
|
2017-10-23 18:27:18 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
void mz_os_unicode_string_delete(wchar_t **string) {
|
2023-02-19 11:17:54 -08:00
|
|
|
if (string) {
|
2023-02-24 14:23:04 -08:00
|
|
|
free(*string);
|
2018-10-23 09:04:04 -07:00
|
|
|
*string = NULL;
|
2017-10-23 18:27:18 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
uint8_t *mz_os_utf8_string_create(const char *string, int32_t encoding) {
|
2018-10-30 13:59:17 -07:00
|
|
|
wchar_t *string_wide = NULL;
|
|
|
|
uint8_t *string_utf8 = NULL;
|
|
|
|
uint32_t string_utf8_size = 0;
|
|
|
|
|
|
|
|
string_wide = mz_os_unicode_string_create(string, encoding);
|
2020-06-14 15:19:14 -07:00
|
|
|
if (string_wide) {
|
2018-10-30 13:59:17 -07:00
|
|
|
string_utf8_size = WideCharToMultiByte(CP_UTF8, 0, string_wide, -1, NULL, 0, NULL, NULL);
|
2023-02-24 14:35:31 -08:00
|
|
|
string_utf8 = (uint8_t *)calloc(string_utf8_size + 1, sizeof(wchar_t));
|
2018-10-30 13:59:17 -07:00
|
|
|
|
2023-02-24 14:35:31 -08:00
|
|
|
if (string_utf8)
|
2018-11-10 09:03:55 -08:00
|
|
|
WideCharToMultiByte(CP_UTF8, 0, string_wide, -1, (char *)string_utf8, string_utf8_size, NULL, NULL);
|
2018-10-30 13:59:17 -07:00
|
|
|
|
|
|
|
mz_os_unicode_string_delete(&string_wide);
|
|
|
|
}
|
|
|
|
|
|
|
|
return string_utf8;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
uint8_t *mz_os_utf8_string_create_from_unicode(const wchar_t *string, int32_t encoding) {
|
2019-05-02 21:07:39 -07:00
|
|
|
uint8_t *string_utf8 = NULL;
|
|
|
|
uint32_t string_utf8_size = 0;
|
|
|
|
|
2020-06-19 21:33:52 -07:00
|
|
|
MZ_UNUSED(encoding);
|
|
|
|
|
2019-05-02 21:07:39 -07:00
|
|
|
string_utf8_size = WideCharToMultiByte(CP_UTF8, 0, string, -1, NULL, 0, NULL, NULL);
|
2023-02-24 14:35:31 -08:00
|
|
|
string_utf8 = (uint8_t *)calloc(string_utf8_size + 1, sizeof(wchar_t));
|
2019-05-02 21:07:39 -07:00
|
|
|
|
2023-02-24 14:35:31 -08:00
|
|
|
if (string_utf8)
|
2019-05-02 21:07:39 -07:00
|
|
|
WideCharToMultiByte(CP_UTF8, 0, string, -1, (char *)string_utf8, string_utf8_size, NULL, NULL);
|
|
|
|
|
|
|
|
return string_utf8;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
void mz_os_utf8_string_delete(uint8_t **string) {
|
2023-02-19 11:17:54 -08:00
|
|
|
if (string) {
|
2023-02-24 14:23:04 -08:00
|
|
|
free(*string);
|
2018-10-30 13:59:17 -07:00
|
|
|
*string = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-23 09:04:04 -07:00
|
|
|
/***************************************************************************/
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_rand(uint8_t *buf, int32_t size) {
|
2018-10-25 19:35:50 -07:00
|
|
|
unsigned __int64 pentium_tsc[1];
|
|
|
|
int32_t len = 0;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
for (len = 0; len < (int)size; len += 1) {
|
2018-10-25 19:35:50 -07:00
|
|
|
if (len % 8 == 0)
|
|
|
|
QueryPerformanceCounter((LARGE_INTEGER *)pentium_tsc);
|
2023-02-19 10:41:34 -08:00
|
|
|
buf[len] = ((unsigned char *)pentium_tsc)[len % 8];
|
2018-10-25 19:35:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_rename(const char *source_path, const char *target_path) {
|
2018-07-31 10:06:12 -07:00
|
|
|
wchar_t *source_path_wide = NULL;
|
|
|
|
wchar_t *target_path_wide = NULL;
|
|
|
|
int32_t result = 0;
|
2018-10-30 13:59:17 -07:00
|
|
|
int32_t err = MZ_OK;
|
2018-07-31 10:06:12 -07:00
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!source_path || !target_path)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
2018-07-31 10:06:12 -07:00
|
|
|
|
2018-10-30 13:59:17 -07:00
|
|
|
source_path_wide = mz_os_unicode_string_create(source_path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!source_path_wide) {
|
2018-10-30 13:59:17 -07:00
|
|
|
err = MZ_PARAM_ERROR;
|
2020-06-14 15:19:14 -07:00
|
|
|
} else {
|
2018-10-30 13:59:17 -07:00
|
|
|
target_path_wide = mz_os_unicode_string_create(target_path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!target_path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
err = MZ_PARAM_ERROR;
|
|
|
|
}
|
2018-07-31 10:06:12 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (err == MZ_OK) {
|
2023-02-17 09:47:02 -08:00
|
|
|
#if _WIN32_WINNT >= _WIN32_WINNT_WINXP
|
2020-12-01 12:55:29 +01:00
|
|
|
result = MoveFileExW(source_path_wide, target_path_wide, MOVEFILE_WRITE_THROUGH);
|
|
|
|
#else
|
2018-10-30 13:59:17 -07:00
|
|
|
result = MoveFileW(source_path_wide, target_path_wide);
|
2020-12-01 12:55:29 +01:00
|
|
|
#endif
|
2018-10-30 13:59:17 -07:00
|
|
|
if (result == 0)
|
|
|
|
err = MZ_EXIST_ERROR;
|
|
|
|
}
|
2018-07-31 10:06:12 -07:00
|
|
|
|
2018-10-30 13:59:17 -07:00
|
|
|
if (target_path_wide)
|
|
|
|
mz_os_unicode_string_delete(&target_path_wide);
|
|
|
|
if (source_path_wide)
|
|
|
|
mz_os_unicode_string_delete(&source_path_wide);
|
2019-07-04 10:32:02 -07:00
|
|
|
|
2018-10-30 13:59:17 -07:00
|
|
|
return err;
|
2018-07-31 10:06:12 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_unlink(const char *path) {
|
2018-07-31 10:06:12 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
|
|
|
int32_t result = 0;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
2018-07-31 10:06:12 -07:00
|
|
|
|
2019-05-02 21:07:39 -07:00
|
|
|
if (mz_os_is_dir(path) == MZ_OK)
|
|
|
|
result = RemoveDirectoryW(path_wide);
|
|
|
|
else
|
|
|
|
result = DeleteFileW(path_wide);
|
|
|
|
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
2018-07-31 10:06:12 -07:00
|
|
|
|
|
|
|
if (result == 0)
|
|
|
|
return MZ_EXIST_ERROR;
|
|
|
|
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_file_exists(const char *path) {
|
2017-10-23 18:27:18 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
|
|
|
DWORD attribs = 0;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
2017-10-23 18:27:18 -07:00
|
|
|
|
|
|
|
attribs = GetFileAttributesW(path_wide);
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
2017-10-23 18:27:18 -07:00
|
|
|
|
|
|
|
if (attribs == 0xFFFFFFFF)
|
|
|
|
return MZ_EXIST_ERROR;
|
|
|
|
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int64_t mz_os_get_file_size(const char *path) {
|
2017-10-23 18:27:18 -07:00
|
|
|
HANDLE handle = NULL;
|
|
|
|
LARGE_INTEGER large_size;
|
|
|
|
wchar_t *path_wide = NULL;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
2023-02-17 09:47:02 -08:00
|
|
|
#if _WIN32_WINNT >= _WIN32_WINNT_WIN8
|
2020-12-01 12:55:29 +01:00
|
|
|
handle = CreateFile2(path_wide, GENERIC_READ, 0, OPEN_EXISTING, NULL);
|
2017-10-23 18:27:18 -07:00
|
|
|
#else
|
|
|
|
handle = CreateFileW(path_wide, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
|
|
|
|
#endif
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
2017-10-23 18:27:18 -07:00
|
|
|
|
|
|
|
large_size.QuadPart = 0;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (handle != INVALID_HANDLE_VALUE) {
|
2017-10-23 18:27:18 -07:00
|
|
|
GetFileSizeEx(handle, &large_size);
|
|
|
|
CloseHandle(handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
return large_size.QuadPart;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
static void mz_os_file_to_unix_time(FILETIME file_time, time_t *unix_time) {
|
2017-10-22 14:32:37 -07:00
|
|
|
uint64_t quad_file_time = 0;
|
|
|
|
quad_file_time = file_time.dwLowDateTime;
|
|
|
|
quad_file_time |= ((uint64_t)file_time.dwHighDateTime << 32);
|
|
|
|
*unix_time = (time_t)((quad_file_time - 116444736000000000LL) / 10000000);
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
static void mz_os_unix_to_file_time(time_t unix_time, FILETIME *file_time) {
|
2017-10-22 14:32:37 -07:00
|
|
|
uint64_t quad_file_time = 0;
|
|
|
|
quad_file_time = ((uint64_t)unix_time * 10000000) + 116444736000000000LL;
|
|
|
|
file_time->dwHighDateTime = (quad_file_time >> 32);
|
|
|
|
file_time->dwLowDateTime = (uint32_t)(quad_file_time);
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_get_file_date(const char *path, time_t *modified_date, time_t *accessed_date, time_t *creation_date) {
|
2017-10-05 23:32:57 -07:00
|
|
|
WIN32_FIND_DATAW ff32;
|
2017-10-22 14:32:37 -07:00
|
|
|
HANDLE handle = NULL;
|
2017-10-05 23:32:57 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
2017-10-20 07:59:39 -07:00
|
|
|
int32_t err = MZ_INTERNAL_ERROR;
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
2017-10-05 23:32:57 -07:00
|
|
|
handle = FindFirstFileW(path_wide, &ff32);
|
2023-02-24 14:23:04 -08:00
|
|
|
free(path_wide);
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (handle != INVALID_HANDLE_VALUE) {
|
2023-02-19 11:17:54 -08:00
|
|
|
if (modified_date)
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_file_to_unix_time(ff32.ftLastWriteTime, modified_date);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (accessed_date)
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_file_to_unix_time(ff32.ftLastAccessTime, accessed_date);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (creation_date)
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_file_to_unix_time(ff32.ftCreationTime, creation_date);
|
2017-10-22 14:32:37 -07:00
|
|
|
|
2017-10-05 23:32:57 -07:00
|
|
|
FindClose(handle);
|
|
|
|
err = MZ_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_set_file_date(const char *path, time_t modified_date, time_t accessed_date, time_t creation_date) {
|
2017-10-05 23:32:57 -07:00
|
|
|
HANDLE handle = NULL;
|
2017-10-22 14:32:37 -07:00
|
|
|
FILETIME ftm_creation, ftm_accessed, ftm_modified;
|
2017-10-05 23:32:57 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
2017-10-20 07:59:39 -07:00
|
|
|
int32_t err = MZ_OK;
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2023-02-17 09:47:02 -08:00
|
|
|
#if _WIN32_WINNT >= _WIN32_WINNT_WIN8
|
2020-12-01 12:55:29 +01:00
|
|
|
handle = CreateFile2(path_wide, GENERIC_READ | GENERIC_WRITE, 0, OPEN_EXISTING, NULL);
|
2017-10-05 23:32:57 -07:00
|
|
|
#else
|
|
|
|
handle = CreateFileW(path_wide, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, NULL);
|
|
|
|
#endif
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (handle != INVALID_HANDLE_VALUE) {
|
2017-10-22 14:32:37 -07:00
|
|
|
GetFileTime(handle, &ftm_creation, &ftm_accessed, &ftm_modified);
|
|
|
|
|
|
|
|
if (modified_date != 0)
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_unix_to_file_time(modified_date, &ftm_modified);
|
2017-10-22 14:32:37 -07:00
|
|
|
if (accessed_date != 0)
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_unix_to_file_time(accessed_date, &ftm_accessed);
|
2017-10-22 14:32:37 -07:00
|
|
|
if (creation_date != 0)
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_unix_to_file_time(creation_date, &ftm_creation);
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2017-10-22 14:32:37 -07:00
|
|
|
if (SetFileTime(handle, &ftm_creation, &ftm_accessed, &ftm_modified) == 0)
|
2017-10-05 23:32:57 -07:00
|
|
|
err = MZ_INTERNAL_ERROR;
|
|
|
|
|
|
|
|
CloseHandle(handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_get_file_attribs(const char *path, uint32_t *attributes) {
|
2018-05-01 13:45:08 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
|
|
|
int32_t err = MZ_OK;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path || !attributes)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
2018-05-01 13:45:08 -07:00
|
|
|
*attributes = GetFileAttributesW(path_wide);
|
2018-10-30 13:59:17 -07:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
2018-05-01 13:45:08 -07:00
|
|
|
|
|
|
|
if (*attributes == INVALID_FILE_ATTRIBUTES)
|
|
|
|
err = MZ_INTERNAL_ERROR;
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_set_file_attribs(const char *path, uint32_t attributes) {
|
2018-05-01 13:45:08 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
|
|
|
int32_t err = MZ_OK;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
2018-05-01 13:45:08 -07:00
|
|
|
if (SetFileAttributesW(path_wide, attributes) == 0)
|
|
|
|
err = MZ_INTERNAL_ERROR;
|
2018-10-30 13:59:17 -07:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
2018-05-01 13:45:08 -07:00
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_make_dir(const char *path) {
|
2017-10-05 23:32:57 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
2018-11-21 14:53:36 -08:00
|
|
|
int32_t err = MZ_OK;
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
2019-03-09 20:12:48 -08:00
|
|
|
|
|
|
|
/* Don't try to create a drive letter */
|
|
|
|
if ((path[0] != 0) && (strlen(path) <= 3) && (path[1] == ':'))
|
|
|
|
return mz_os_is_dir(path);
|
|
|
|
|
2018-10-30 13:59:17 -07:00
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (CreateDirectoryW(path_wide, NULL) == 0) {
|
2018-11-21 14:53:36 -08:00
|
|
|
if (GetLastError() != ERROR_ALREADY_EXISTS)
|
|
|
|
err = MZ_INTERNAL_ERROR;
|
|
|
|
}
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2018-11-21 14:53:36 -08:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2018-11-21 14:53:36 -08:00
|
|
|
return err;
|
2017-10-05 23:32:57 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
DIR *mz_os_open_dir(const char *path) {
|
2017-10-05 23:32:57 -07:00
|
|
|
WIN32_FIND_DATAW find_data;
|
2017-10-05 23:55:59 -07:00
|
|
|
DIR_int *dir_int = NULL;
|
2017-10-05 23:32:57 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
|
|
|
char fixed_path[320];
|
2017-10-06 00:08:29 -07:00
|
|
|
void *handle = NULL;
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2018-10-30 13:59:17 -07:00
|
|
|
return NULL;
|
|
|
|
|
2019-05-08 23:41:47 -07:00
|
|
|
strncpy(fixed_path, path, sizeof(fixed_path) - 1);
|
|
|
|
fixed_path[sizeof(fixed_path) - 1] = 0;
|
|
|
|
|
|
|
|
mz_path_append_slash(fixed_path, sizeof(fixed_path), MZ_PATH_SLASH_PLATFORM);
|
2017-10-16 14:50:31 -07:00
|
|
|
mz_path_combine(fixed_path, "*", sizeof(fixed_path));
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2018-10-30 13:59:17 -07:00
|
|
|
path_wide = mz_os_unicode_string_create(fixed_path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
return NULL;
|
|
|
|
|
2017-10-05 23:32:57 -07:00
|
|
|
handle = FindFirstFileW(path_wide, &find_data);
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2017-10-05 23:55:59 -07:00
|
|
|
if (handle == INVALID_HANDLE_VALUE)
|
2017-10-05 23:32:57 -07:00
|
|
|
return NULL;
|
|
|
|
|
2023-02-24 14:23:04 -08:00
|
|
|
dir_int = (DIR_int *)malloc(sizeof(DIR_int));
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!dir_int) {
|
2022-08-03 14:55:05 -04:00
|
|
|
FindClose(handle);
|
2018-12-01 08:59:21 -08:00
|
|
|
return NULL;
|
2022-08-03 14:55:05 -04:00
|
|
|
}
|
2017-10-05 23:55:59 -07:00
|
|
|
dir_int->find_handle = handle;
|
|
|
|
dir_int->end = 0;
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2017-10-05 23:55:59 -07:00
|
|
|
memcpy(&dir_int->find_data, &find_data, sizeof(dir_int->find_data));
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2017-10-06 00:08:29 -07:00
|
|
|
return (DIR *)dir_int;
|
2017-10-05 23:32:57 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
struct dirent* mz_os_read_dir(DIR *dir) {
|
2017-10-05 23:55:59 -07:00
|
|
|
DIR_int *dir_int;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!dir)
|
2017-10-05 23:32:57 -07:00
|
|
|
return NULL;
|
|
|
|
|
2017-10-05 23:55:59 -07:00
|
|
|
dir_int = (DIR_int *)dir;
|
|
|
|
if (dir_int->end)
|
2017-10-05 23:32:57 -07:00
|
|
|
return NULL;
|
|
|
|
|
2018-04-24 10:02:39 +00:00
|
|
|
WideCharToMultiByte(CP_UTF8, 0, dir_int->find_data.cFileName, -1,
|
2017-10-05 23:55:59 -07:00
|
|
|
dir_int->entry.d_name, sizeof(dir_int->entry.d_name), NULL, NULL);
|
2018-04-24 10:02:39 +00:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (FindNextFileW(dir_int->find_handle, &dir_int->find_data) == 0) {
|
2017-10-05 23:55:59 -07:00
|
|
|
if (GetLastError() != ERROR_NO_MORE_FILES)
|
|
|
|
return NULL;
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2017-10-05 23:55:59 -07:00
|
|
|
dir_int->end = 1;
|
|
|
|
}
|
2018-04-24 10:02:39 +00:00
|
|
|
|
2017-10-05 23:55:59 -07:00
|
|
|
return &dir_int->entry;
|
2017-10-05 23:32:57 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_close_dir(DIR *dir) {
|
2017-10-05 23:55:59 -07:00
|
|
|
DIR_int *dir_int;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!dir)
|
2017-10-05 23:32:57 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
2017-10-05 23:55:59 -07:00
|
|
|
dir_int = (DIR_int *)dir;
|
|
|
|
if (dir_int->find_handle != INVALID_HANDLE_VALUE)
|
|
|
|
FindClose(dir_int->find_handle);
|
2023-02-24 14:23:04 -08:00
|
|
|
free(dir_int);
|
2017-10-05 23:32:57 -07:00
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_is_dir(const char *path) {
|
2017-10-05 23:32:57 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
2018-05-09 09:45:15 -07:00
|
|
|
uint32_t attribs = 0;
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2018-10-30 13:59:17 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
2017-10-05 23:32:57 -07:00
|
|
|
attribs = GetFileAttributesW(path_wide);
|
2018-10-24 09:50:16 -07:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (attribs != 0xFFFFFFFF) {
|
2017-10-23 18:27:18 -07:00
|
|
|
if (attribs & FILE_ATTRIBUTE_DIRECTORY)
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
2017-10-05 23:32:57 -07:00
|
|
|
|
|
|
|
return MZ_EXIST_ERROR;
|
2018-05-02 19:59:38 +00:00
|
|
|
}
|
2018-09-18 12:06:54 +02:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_is_symlink(const char *path) {
|
2019-05-02 21:07:39 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
|
|
|
uint32_t attribs = 0;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2019-05-02 21:07:39 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2019-05-02 21:07:39 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
|
|
|
attribs = GetFileAttributesW(path_wide);
|
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (attribs != 0xFFFFFFFF) {
|
2019-05-02 21:07:39 -07:00
|
|
|
if (attribs & FILE_ATTRIBUTE_REPARSE_POINT)
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return MZ_EXIST_ERROR;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_make_symlink(const char *path, const char *target_path) {
|
2019-05-02 21:07:39 -07:00
|
|
|
typedef BOOLEAN (WINAPI *LPCREATESYMBOLICLINKW)(LPCWSTR, LPCWSTR, DWORD);
|
2023-02-17 09:47:02 -08:00
|
|
|
MEMORY_BASIC_INFORMATION mbi;
|
2019-05-02 21:07:39 -07:00
|
|
|
LPCREATESYMBOLICLINKW create_symbolic_link_w = NULL;
|
2020-02-08 22:13:10 -05:00
|
|
|
HMODULE kernel32_mod = NULL;
|
2019-05-02 21:07:39 -07:00
|
|
|
wchar_t *path_wide = NULL;
|
|
|
|
wchar_t *target_path_wide = NULL;
|
|
|
|
int32_t err = MZ_OK;
|
|
|
|
int32_t flags = 0;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2019-05-02 21:07:39 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
2023-02-17 09:47:02 -08:00
|
|
|
// Use VirtualQuery instead of GetModuleHandleW for UWP
|
2020-12-03 11:01:58 +01:00
|
|
|
memset(&mbi, 0, sizeof(mbi));
|
2020-12-01 12:55:29 +01:00
|
|
|
VirtualQuery(VirtualQuery, &mbi, sizeof(mbi));
|
|
|
|
kernel32_mod = (HMODULE)mbi.AllocationBase;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!kernel32_mod)
|
2019-05-02 21:07:39 -07:00
|
|
|
return MZ_SUPPORT_ERROR;
|
|
|
|
|
|
|
|
create_symbolic_link_w = (LPCREATESYMBOLICLINKW)GetProcAddress(kernel32_mod, "CreateSymbolicLinkW");
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!create_symbolic_link_w) {
|
2019-05-02 21:07:39 -07:00
|
|
|
return MZ_SUPPORT_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide) {
|
2019-05-02 21:07:39 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
target_path_wide = mz_os_unicode_string_create(target_path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (target_path_wide) {
|
2019-05-02 21:07:39 -07:00
|
|
|
if (mz_path_has_slash(target_path) == MZ_OK)
|
|
|
|
flags |= SYMBOLIC_LINK_FLAG_DIRECTORY;
|
|
|
|
|
|
|
|
if (create_symbolic_link_w(path_wide, target_path_wide, flags) == FALSE)
|
|
|
|
err = MZ_SYMLINK_ERROR;
|
|
|
|
|
|
|
|
mz_os_unicode_string_delete(&target_path_wide);
|
2020-06-14 15:19:14 -07:00
|
|
|
} else {
|
2019-05-02 21:07:39 -07:00
|
|
|
err = MZ_PARAM_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_os_read_symlink(const char *path, char *target_path, int32_t max_target_path) {
|
2019-05-02 21:11:31 -07:00
|
|
|
typedef struct _REPARSE_DATA_BUFFER {
|
|
|
|
ULONG ReparseTag;
|
|
|
|
USHORT ReparseDataLength;
|
|
|
|
USHORT Reserved;
|
|
|
|
union {
|
|
|
|
struct {
|
|
|
|
USHORT SubstituteNameOffset;
|
|
|
|
USHORT SubstituteNameLength;
|
|
|
|
USHORT PrintNameOffset;
|
|
|
|
USHORT PrintNameLength;
|
|
|
|
ULONG Flags;
|
|
|
|
WCHAR PathBuffer[1];
|
|
|
|
} SymbolicLinkReparseBuffer;
|
|
|
|
struct {
|
|
|
|
USHORT SubstituteNameOffset;
|
|
|
|
USHORT SubstituteNameLength;
|
|
|
|
USHORT PrintNameOffset;
|
|
|
|
USHORT PrintNameLength;
|
|
|
|
WCHAR PathBuffer[1];
|
|
|
|
} MountPointReparseBuffer;
|
|
|
|
struct {
|
|
|
|
UCHAR DataBuffer[1];
|
|
|
|
} GenericReparseBuffer;
|
|
|
|
};
|
2020-06-19 21:33:52 -07:00
|
|
|
} REPARSE_DATA_BUFFER;
|
2019-05-02 21:07:39 -07:00
|
|
|
REPARSE_DATA_BUFFER *reparse_data = NULL;
|
2019-09-25 11:50:30 -07:00
|
|
|
DWORD length = 0;
|
2019-05-02 21:07:39 -07:00
|
|
|
HANDLE handle = NULL;
|
|
|
|
wchar_t *path_wide = NULL;
|
|
|
|
wchar_t *target_path_wide = NULL;
|
|
|
|
uint8_t buffer[MAXIMUM_REPARSE_DATA_BUFFER_SIZE];
|
|
|
|
int32_t target_path_len = 0;
|
|
|
|
int32_t target_path_idx = 0;
|
|
|
|
int32_t err = MZ_OK;
|
|
|
|
uint8_t *target_path_utf8 = NULL;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2019-05-02 21:07:39 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
path_wide = mz_os_unicode_string_create(path, MZ_ENCODING_UTF8);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path_wide)
|
2019-05-02 21:07:39 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
2023-02-17 09:47:02 -08:00
|
|
|
#if _WIN32_WINNT >= _WIN32_WINNT_WIN8
|
2020-12-03 11:01:58 +01:00
|
|
|
CREATEFILE2_EXTENDED_PARAMETERS extended_params;
|
|
|
|
memset(&extended_params, 0, sizeof(extended_params));
|
|
|
|
extended_params.dwSize = sizeof(extended_params);
|
|
|
|
extended_params.dwFileAttributes = FILE_ATTRIBUTE_NORMAL;
|
|
|
|
extended_params.dwFileFlags = FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OPEN_REPARSE_POINT;
|
|
|
|
extended_params.dwSecurityQosFlags = SECURITY_ANONYMOUS;
|
|
|
|
extended_params.lpSecurityAttributes = NULL;
|
|
|
|
extended_params.hTemplateFile = NULL;
|
|
|
|
handle = CreateFile2(path_wide, FILE_READ_EA, FILE_SHARE_READ | FILE_SHARE_WRITE, OPEN_EXISTING, &extended_params);
|
2020-12-01 12:55:29 +01:00
|
|
|
#else
|
2019-07-04 10:32:02 -07:00
|
|
|
handle = CreateFileW(path_wide, FILE_READ_EA, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING,
|
2019-05-02 21:07:39 -07:00
|
|
|
FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OPEN_REPARSE_POINT, NULL);
|
2020-12-01 12:55:29 +01:00
|
|
|
#endif
|
2019-05-02 21:07:39 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (handle == INVALID_HANDLE_VALUE) {
|
2019-05-05 20:30:23 -07:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
|
|
|
return MZ_OPEN_ERROR;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (DeviceIoControl(handle, FSCTL_GET_REPARSE_POINT, NULL, 0, buffer, sizeof(buffer), &length, NULL) == TRUE) {
|
2019-05-05 20:30:23 -07:00
|
|
|
reparse_data = (REPARSE_DATA_BUFFER *)buffer;
|
2019-07-04 10:32:02 -07:00
|
|
|
if ((IsReparseTagMicrosoft(reparse_data->ReparseTag)) &&
|
2020-06-14 15:19:14 -07:00
|
|
|
(reparse_data->ReparseTag == IO_REPARSE_TAG_SYMLINK)) {
|
2019-05-05 20:30:23 -07:00
|
|
|
target_path_len = max_target_path * sizeof(wchar_t);
|
|
|
|
if (target_path_len > reparse_data->SymbolicLinkReparseBuffer.PrintNameLength)
|
|
|
|
target_path_len = reparse_data->SymbolicLinkReparseBuffer.PrintNameLength;
|
|
|
|
|
2023-02-24 14:23:04 -08:00
|
|
|
target_path_wide = (wchar_t *)malloc(target_path_len + sizeof(wchar_t));
|
2020-06-14 15:19:14 -07:00
|
|
|
if (target_path_wide) {
|
2019-05-05 20:30:23 -07:00
|
|
|
target_path_idx = reparse_data->SymbolicLinkReparseBuffer.PrintNameOffset / sizeof(wchar_t);
|
|
|
|
memcpy(target_path_wide, &reparse_data->SymbolicLinkReparseBuffer.PathBuffer[target_path_idx],
|
|
|
|
target_path_len);
|
|
|
|
|
|
|
|
target_path_wide[target_path_len / sizeof(wchar_t)] = 0;
|
|
|
|
target_path_utf8 = mz_os_utf8_string_create_from_unicode(target_path_wide, MZ_ENCODING_UTF8);
|
2019-05-02 21:07:39 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (target_path_utf8) {
|
fix Windows-specific compiler error and warnings
Exprienced in a mingw-w64 + clang-9 cross-build environment:
```
mz_os_win32.c:560:22: error: use of undeclared identifier 'SYMBOLIC_LINK_FLAG_DIRECTORY'
flags |= SYMBOLIC_LINK_FLAG_DIRECTORY;
^
mz_os_win32.c:632:91: warning: incompatible pointer types passing 'int32_t *' (aka 'int *') to parameter of type 'LPDWORD' (aka 'unsigned long *')
[-Wincompatible-pointer-types]
if (DeviceIoControl(handle, FSCTL_GET_REPARSE_POINT, NULL, 0, buffer, sizeof(buffer), &length, NULL) == TRUE)
^~~~~~~
/usr/local/opt/mingw-w64/toolchain-x86_64/x86_64-w64-mingw32/include/ioapiset.h:22:175: note: passing argument to parameter 'lpBytesReturned' here
WINBASEAPI WINBOOL WINAPI DeviceIoControl (HANDLE hDevice, DWORD dwIoControlCode, LPVOID lpInBuffer, DWORD nInBufferSize, LPVOID lpOutBuffer, DWORD nOutBufferSize, LPDWORD lpBytesReturned, LPOVE...
^
mz_os_win32.c:654:42: warning: passing 'uint8_t *' (aka 'unsigned char *') to parameter of type 'const char *' converts between pointers to integer types with
different sign [-Wpointer-sign]
strncpy(target_path, target_path_utf8, max_target_path - 1);
^~~~~~~~~~~~~~~~
/usr/local/opt/mingw-w64/toolchain-x86_64/x86_64-w64-mingw32/include/string.h:89:69: note: passing argument to parameter '_Source' here
char *strncpy(char * __restrict__ _Dest,const char * __restrict__ _Source,size_t _Count) __MINGW_ATTRIB_DEPRECATED_SEC_WARN;
^
mz_os_win32.c:657:38: warning: passing 'uint8_t *' (aka 'unsigned char *') to parameter of type 'const char *' converts between pointers to integer types with
different sign [-Wpointer-sign]
if (mz_os_is_dir(target_path_utf8) == MZ_OK)
^~~~~~~~~~~~~~~~
mz_os_win32.c:473:34: note: passing argument to parameter 'path' here
int32_t mz_os_is_dir(const char *path)
^
3 warnings and 1 error generated.
```
2019-09-25 14:16:11 +00:00
|
|
|
strncpy(target_path, (const char *)target_path_utf8, max_target_path - 1);
|
2019-05-08 23:41:47 -07:00
|
|
|
target_path[max_target_path - 1] = 0;
|
|
|
|
/* Ensure directories have slash at the end so we can recreate them later */
|
fix Windows-specific compiler error and warnings
Exprienced in a mingw-w64 + clang-9 cross-build environment:
```
mz_os_win32.c:560:22: error: use of undeclared identifier 'SYMBOLIC_LINK_FLAG_DIRECTORY'
flags |= SYMBOLIC_LINK_FLAG_DIRECTORY;
^
mz_os_win32.c:632:91: warning: incompatible pointer types passing 'int32_t *' (aka 'int *') to parameter of type 'LPDWORD' (aka 'unsigned long *')
[-Wincompatible-pointer-types]
if (DeviceIoControl(handle, FSCTL_GET_REPARSE_POINT, NULL, 0, buffer, sizeof(buffer), &length, NULL) == TRUE)
^~~~~~~
/usr/local/opt/mingw-w64/toolchain-x86_64/x86_64-w64-mingw32/include/ioapiset.h:22:175: note: passing argument to parameter 'lpBytesReturned' here
WINBASEAPI WINBOOL WINAPI DeviceIoControl (HANDLE hDevice, DWORD dwIoControlCode, LPVOID lpInBuffer, DWORD nInBufferSize, LPVOID lpOutBuffer, DWORD nOutBufferSize, LPDWORD lpBytesReturned, LPOVE...
^
mz_os_win32.c:654:42: warning: passing 'uint8_t *' (aka 'unsigned char *') to parameter of type 'const char *' converts between pointers to integer types with
different sign [-Wpointer-sign]
strncpy(target_path, target_path_utf8, max_target_path - 1);
^~~~~~~~~~~~~~~~
/usr/local/opt/mingw-w64/toolchain-x86_64/x86_64-w64-mingw32/include/string.h:89:69: note: passing argument to parameter '_Source' here
char *strncpy(char * __restrict__ _Dest,const char * __restrict__ _Source,size_t _Count) __MINGW_ATTRIB_DEPRECATED_SEC_WARN;
^
mz_os_win32.c:657:38: warning: passing 'uint8_t *' (aka 'unsigned char *') to parameter of type 'const char *' converts between pointers to integer types with
different sign [-Wpointer-sign]
if (mz_os_is_dir(target_path_utf8) == MZ_OK)
^~~~~~~~~~~~~~~~
mz_os_win32.c:473:34: note: passing argument to parameter 'path' here
int32_t mz_os_is_dir(const char *path)
^
3 warnings and 1 error generated.
```
2019-09-25 14:16:11 +00:00
|
|
|
if (mz_os_is_dir((const char *)target_path_utf8) == MZ_OK)
|
2019-05-08 23:41:47 -07:00
|
|
|
mz_path_append_slash(target_path, max_target_path, MZ_PATH_SLASH_PLATFORM);
|
2019-05-05 20:30:23 -07:00
|
|
|
mz_os_utf8_string_delete(&target_path_utf8);
|
2020-06-14 15:19:14 -07:00
|
|
|
} else {
|
2019-05-02 21:07:39 -07:00
|
|
|
err = MZ_MEM_ERROR;
|
|
|
|
}
|
2019-05-05 20:30:23 -07:00
|
|
|
|
2023-02-24 14:23:04 -08:00
|
|
|
free(target_path_wide);
|
2020-06-14 15:19:14 -07:00
|
|
|
} else {
|
2019-05-05 20:30:23 -07:00
|
|
|
err = MZ_MEM_ERROR;
|
2019-05-02 21:07:39 -07:00
|
|
|
}
|
|
|
|
}
|
2020-06-14 15:19:14 -07:00
|
|
|
} else {
|
2019-05-05 20:30:23 -07:00
|
|
|
err = MZ_INTERNAL_ERROR;
|
2019-05-02 21:07:39 -07:00
|
|
|
}
|
|
|
|
|
2019-05-05 20:30:23 -07:00
|
|
|
CloseHandle(handle);
|
2019-05-02 21:07:39 -07:00
|
|
|
mz_os_unicode_string_delete(&path_wide);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
uint64_t mz_os_ms_time(void) {
|
2018-09-18 12:06:54 +02:00
|
|
|
SYSTEMTIME system_time;
|
|
|
|
FILETIME file_time;
|
2018-09-18 20:25:48 -07:00
|
|
|
uint64_t quad_file_time = 0;
|
|
|
|
|
|
|
|
GetSystemTime(&system_time);
|
2018-09-18 12:06:54 +02:00
|
|
|
SystemTimeToFileTime(&system_time, &file_time);
|
|
|
|
|
2018-09-18 20:25:48 -07:00
|
|
|
quad_file_time = file_time.dwLowDateTime;
|
|
|
|
quad_file_time |= ((uint64_t)file_time.dwHighDateTime << 32);
|
|
|
|
|
|
|
|
return quad_file_time / 10000 - 11644473600000LL;
|
2018-09-18 12:06:54 +02:00
|
|
|
}
|