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:
Jonathon Reinhart 2018-08-01 19:10:59 -04:00 committed by Michael Roth
parent a2e002ff79
commit bad0227d3a

View File

@ -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;
} }