qga: Support Unicode paths in guest-file-open on win32
Currently, the win32 port of QEMU Guest Agent does not properly handle Unicode paths. The JSON decoder produces a valid UTF-8 path string, but this is passed directly to CreateFileA, which is expecting an ANSI string and not UTF-8. This leads to mangled filenames. This patch follows the example of qmp_guest_set_user_password() and uses g_utf8_to_utf16() to convert the string to UTF-16 and calls CreateFileW() explicitly. Signed-off-by: Jonathon Reinhart <jreinhart@cc-sw.com> Signed-off-by: Michael Roth <mdroth@linux.vnet.ibm.com>
This commit is contained in:
parent
a2e002ff79
commit
bad0227d3a
@ -160,13 +160,15 @@ static void handle_set_nonblocking(HANDLE fh)
|
|||||||
int64_t qmp_guest_file_open(const char *path, bool has_mode,
|
int64_t qmp_guest_file_open(const char *path, bool has_mode,
|
||||||
const char *mode, Error **errp)
|
const char *mode, Error **errp)
|
||||||
{
|
{
|
||||||
int64_t fd;
|
int64_t fd = -1;
|
||||||
HANDLE fh;
|
HANDLE fh;
|
||||||
HANDLE templ_file = NULL;
|
HANDLE templ_file = NULL;
|
||||||
DWORD share_mode = FILE_SHARE_READ;
|
DWORD share_mode = FILE_SHARE_READ;
|
||||||
DWORD flags_and_attr = FILE_ATTRIBUTE_NORMAL;
|
DWORD flags_and_attr = FILE_ATTRIBUTE_NORMAL;
|
||||||
LPSECURITY_ATTRIBUTES sa_attr = NULL;
|
LPSECURITY_ATTRIBUTES sa_attr = NULL;
|
||||||
OpenFlags *guest_flags;
|
OpenFlags *guest_flags;
|
||||||
|
GError *gerr = NULL;
|
||||||
|
wchar_t *w_path = NULL;
|
||||||
|
|
||||||
if (!has_mode) {
|
if (!has_mode) {
|
||||||
mode = "r";
|
mode = "r";
|
||||||
@ -175,16 +177,21 @@ int64_t qmp_guest_file_open(const char *path, bool has_mode,
|
|||||||
guest_flags = find_open_flag(mode);
|
guest_flags = find_open_flag(mode);
|
||||||
if (guest_flags == NULL) {
|
if (guest_flags == NULL) {
|
||||||
error_setg(errp, "invalid file open mode");
|
error_setg(errp, "invalid file open mode");
|
||||||
return -1;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
fh = CreateFile(path, guest_flags->desired_access, share_mode, sa_attr,
|
w_path = g_utf8_to_utf16(path, -1, NULL, NULL, &gerr);
|
||||||
|
if (!w_path) {
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
|
||||||
|
fh = CreateFileW(w_path, guest_flags->desired_access, share_mode, sa_attr,
|
||||||
guest_flags->creation_disposition, flags_and_attr,
|
guest_flags->creation_disposition, flags_and_attr,
|
||||||
templ_file);
|
templ_file);
|
||||||
if (fh == INVALID_HANDLE_VALUE) {
|
if (fh == INVALID_HANDLE_VALUE) {
|
||||||
error_setg_win32(errp, GetLastError(), "failed to open file '%s'",
|
error_setg_win32(errp, GetLastError(), "failed to open file '%s'",
|
||||||
path);
|
path);
|
||||||
return -1;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* set fd non-blocking to avoid common use cases (like reading from a
|
/* set fd non-blocking to avoid common use cases (like reading from a
|
||||||
@ -196,10 +203,17 @@ int64_t qmp_guest_file_open(const char *path, bool has_mode,
|
|||||||
if (fd < 0) {
|
if (fd < 0) {
|
||||||
CloseHandle(fh);
|
CloseHandle(fh);
|
||||||
error_setg(errp, "failed to add handle to qmp handle table");
|
error_setg(errp, "failed to add handle to qmp handle table");
|
||||||
return -1;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
slog("guest-file-open, handle: % " PRId64, fd);
|
slog("guest-file-open, handle: % " PRId64, fd);
|
||||||
|
|
||||||
|
done:
|
||||||
|
if (gerr) {
|
||||||
|
error_setg(errp, QERR_QGA_COMMAND_FAILED, gerr->message);
|
||||||
|
g_error_free(gerr);
|
||||||
|
}
|
||||||
|
g_free(w_path);
|
||||||
return fd;
|
return fd;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user