Merge pull request #8905 from JosJuice/jni-encoding

Android: Use correct encoding when converting strings
This commit is contained in:
LC 2020-07-18 22:13:14 -04:00 committed by GitHub
commit 487cd7abd9
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
15 changed files with 63 additions and 42 deletions

View File

@ -5,25 +5,28 @@
#include "jni/AndroidCommon/AndroidCommon.h" #include "jni/AndroidCommon/AndroidCommon.h"
#include <string> #include <string>
#include <string_view>
#include <vector> #include <vector>
#include <jni.h> #include <jni.h>
#include "Common/StringUtil.h"
std::string GetJString(JNIEnv* env, jstring jstr) std::string GetJString(JNIEnv* env, jstring jstr)
{ {
std::string result = ""; const jchar* jchars = env->GetStringChars(jstr, nullptr);
if (!jstr) const jsize length = env->GetStringLength(jstr);
return result; const std::u16string_view string_view(reinterpret_cast<const char16_t*>(jchars), length);
const std::string converted_string = UTF16ToUTF8(string_view);
const char* s = env->GetStringUTFChars(jstr, nullptr); env->ReleaseStringChars(jstr, jchars);
result = s; return converted_string;
env->ReleaseStringUTFChars(jstr, s);
return result;
} }
jstring ToJString(JNIEnv* env, const std::string& str) jstring ToJString(JNIEnv* env, const std::string& str)
{ {
return env->NewStringUTF(str.c_str()); const std::u16string converted_string = UTF8ToUTF16(str);
return env->NewString(reinterpret_cast<const jchar*>(converted_string.data()),
converted_string.size());
} }
std::vector<std::string> JStringArrayToVector(JNIEnv* env, jobjectArray array) std::vector<std::string> JStringArrayToVector(JNIEnv* env, jobjectArray array)

View File

@ -114,7 +114,7 @@ bool Exists(const std::string& path)
bool IsDirectory(const std::string& path) bool IsDirectory(const std::string& path)
{ {
#ifdef _WIN32 #ifdef _WIN32
return PathIsDirectory(UTF8ToUTF16(path).c_str()); return PathIsDirectory(UTF8ToWString(path).c_str());
#else #else
return FileInfo(path).IsDirectory(); return FileInfo(path).IsDirectory();
#endif #endif

View File

@ -17,5 +17,5 @@ ConsoleListener::~ConsoleListener()
void ConsoleListener::Log([[maybe_unused]] Common::Log::LOG_LEVELS level, const char* text) void ConsoleListener::Log([[maybe_unused]] Common::Log::LOG_LEVELS level, const char* text)
{ {
::OutputDebugStringW(UTF8ToUTF16(text).c_str()); ::OutputDebugStringW(UTF8ToWString(text).c_str());
} }

View File

@ -5,6 +5,7 @@
#include "Common/StringUtil.h" #include "Common/StringUtil.h"
#include <algorithm> #include <algorithm>
#include <codecvt>
#include <cstdarg> #include <cstdarg>
#include <cstddef> #include <cstddef>
#include <cstdio> #include <cstdio>
@ -17,6 +18,7 @@
#include <locale> #include <locale>
#include <sstream> #include <sstream>
#include <string> #include <string>
#include <type_traits>
#include <vector> #include <vector>
#include <fmt/format.h> #include <fmt/format.h>
@ -32,7 +34,6 @@
constexpr u32 CODEPAGE_SHIFT_JIS = 932; constexpr u32 CODEPAGE_SHIFT_JIS = 932;
constexpr u32 CODEPAGE_WINDOWS_1252 = 1252; constexpr u32 CODEPAGE_WINDOWS_1252 = 1252;
#else #else
#include <codecvt>
#include <errno.h> #include <errno.h>
#include <iconv.h> #include <iconv.h>
#include <locale.h> #include <locale.h>
@ -463,29 +464,29 @@ std::string UTF16ToCP(u32 code_page, std::wstring_view input)
return output; return output;
} }
std::wstring UTF8ToUTF16(std::string_view input) std::wstring UTF8ToWString(std::string_view input)
{ {
return CPToUTF16(CP_UTF8, input); return CPToUTF16(CP_UTF8, input);
} }
std::string UTF16ToUTF8(std::wstring_view input) std::string WStringToUTF8(std::wstring_view input)
{ {
return UTF16ToCP(CP_UTF8, input); return UTF16ToCP(CP_UTF8, input);
} }
std::string SHIFTJISToUTF8(std::string_view input) std::string SHIFTJISToUTF8(std::string_view input)
{ {
return UTF16ToUTF8(CPToUTF16(CODEPAGE_SHIFT_JIS, input)); return WStringToUTF8(CPToUTF16(CODEPAGE_SHIFT_JIS, input));
} }
std::string UTF8ToSHIFTJIS(std::string_view input) std::string UTF8ToSHIFTJIS(std::string_view input)
{ {
return UTF16ToCP(CODEPAGE_SHIFT_JIS, UTF8ToUTF16(input)); return UTF16ToCP(CODEPAGE_SHIFT_JIS, UTF8ToWString(input));
} }
std::string CP1252ToUTF8(std::string_view input) std::string CP1252ToUTF8(std::string_view input)
{ {
return UTF16ToUTF8(CPToUTF16(CODEPAGE_WINDOWS_1252, input)); return WStringToUTF8(CPToUTF16(CODEPAGE_WINDOWS_1252, input));
} }
std::string UTF16BEToUTF8(const char16_t* str, size_t max_size) std::string UTF16BEToUTF8(const char16_t* str, size_t max_size)
@ -493,7 +494,7 @@ std::string UTF16BEToUTF8(const char16_t* str, size_t max_size)
const char16_t* str_end = std::find(str, str + max_size, '\0'); const char16_t* str_end = std::find(str, str + max_size, '\0');
std::wstring result(static_cast<size_t>(str_end - str), '\0'); std::wstring result(static_cast<size_t>(str_end - str), '\0');
std::transform(str, str_end, result.begin(), static_cast<u16 (&)(u16)>(Common::swap16)); std::transform(str, str_end, result.begin(), static_cast<u16 (&)(u16)>(Common::swap16));
return UTF16ToUTF8(result); return WStringToUTF8(result);
} }
#else #else
@ -578,9 +579,12 @@ std::string UTF8ToSHIFTJIS(std::string_view input)
return CodeTo("SJIS", "UTF-8", input); return CodeTo("SJIS", "UTF-8", input);
} }
std::string UTF16ToUTF8(std::wstring_view input) std::string WStringToUTF8(std::wstring_view input)
{ {
std::wstring_convert<std::codecvt_utf8<wchar_t>, wchar_t> converter; using codecvt = std::conditional_t<sizeof(wchar_t) == 2, std::codecvt_utf8_utf16<wchar_t>,
std::codecvt_utf8<wchar_t>>;
std::wstring_convert<codecvt, wchar_t> converter;
return converter.to_bytes(input.data(), input.data() + input.size()); return converter.to_bytes(input.data(), input.data() + input.size());
} }
@ -592,12 +596,24 @@ std::string UTF16BEToUTF8(const char16_t* str, size_t max_size)
#endif #endif
std::string UTF16ToUTF8(std::u16string_view input)
{
std::wstring_convert<std::codecvt_utf8_utf16<char16_t>, char16_t> converter;
return converter.to_bytes(input.data(), input.data() + input.size());
}
std::u16string UTF8ToUTF16(std::string_view input)
{
std::wstring_convert<std::codecvt_utf8_utf16<char16_t>, char16_t> converter;
return converter.from_bytes(input.data(), input.data() + input.size());
}
#ifdef HAS_STD_FILESYSTEM #ifdef HAS_STD_FILESYSTEM
// This is a replacement for path::u8path, which is deprecated starting with C++20. // This is a replacement for path::u8path, which is deprecated starting with C++20.
std::filesystem::path StringToPath(std::string_view path) std::filesystem::path StringToPath(std::string_view path)
{ {
#ifdef _MSC_VER #ifdef _MSC_VER
return std::filesystem::path(UTF8ToUTF16(path)); return std::filesystem::path(UTF8ToWString(path));
#else #else
return std::filesystem::path(path); return std::filesystem::path(path);
#endif #endif
@ -608,7 +624,7 @@ std::filesystem::path StringToPath(std::string_view path)
std::string PathToString(const std::filesystem::path& path) std::string PathToString(const std::filesystem::path& path)
{ {
#ifdef _MSC_VER #ifdef _MSC_VER
return UTF16ToUTF8(path.native()); return WStringToUTF8(path.native());
#else #else
return path.native(); return path.native();
#endif #endif

View File

@ -172,22 +172,24 @@ size_t StringUTF8CodePointCount(const std::string& str);
std::string CP1252ToUTF8(std::string_view str); std::string CP1252ToUTF8(std::string_view str);
std::string SHIFTJISToUTF8(std::string_view str); std::string SHIFTJISToUTF8(std::string_view str);
std::string UTF8ToSHIFTJIS(std::string_view str); std::string UTF8ToSHIFTJIS(std::string_view str);
std::string UTF16ToUTF8(std::wstring_view str); std::string WStringToUTF8(std::wstring_view str);
std::string UTF16BEToUTF8(const char16_t* str, size_t max_size); // Stops at \0 std::string UTF16BEToUTF8(const char16_t* str, size_t max_size); // Stops at \0
std::string UTF16ToUTF8(std::u16string_view str);
std::u16string UTF8ToUTF16(std::string_view str);
#ifdef _WIN32 #ifdef _WIN32
std::wstring UTF8ToUTF16(std::string_view str); std::wstring UTF8ToWString(std::string_view str);
#ifdef _UNICODE #ifdef _UNICODE
inline std::string TStrToUTF8(std::wstring_view str) inline std::string TStrToUTF8(std::wstring_view str)
{ {
return UTF16ToUTF8(str); return WStringToUTF8(str);
} }
inline std::wstring UTF8ToTStr(std::string_view str) inline std::wstring UTF8ToTStr(std::string_view str)
{ {
return UTF8ToUTF16(str); return UTF8ToWString(str);
} }
#else #else
inline std::string TStrToUTF8(std::string_view str) inline std::string TStrToUTF8(std::string_view str)
@ -221,7 +223,7 @@ std::string ThousandSeparate(I value, int spaces = 0)
stream << std::setw(spaces) << value; stream << std::setw(spaces) << value;
#ifdef _WIN32 #ifdef _WIN32
return UTF16ToUTF8(stream.str()); return WStringToUTF8(stream.str());
#else #else
return stream.str(); return stream.str();
#endif #endif

View File

@ -217,7 +217,7 @@ std::string Timer::GetTimeFormatted()
#ifdef _WIN32 #ifdef _WIN32
struct timeb tp; struct timeb tp;
(void)::ftime(&tp); (void)::ftime(&tp);
return UTF16ToUTF8(tmp) + fmt::format(":{:03}", tp.millitm); return WStringToUTF8(tmp) + fmt::format(":{:03}", tp.millitm);
#elif defined __APPLE__ #elif defined __APPLE__
struct timeval t; struct timeval t;
(void)gettimeofday(&t, nullptr); (void)gettimeofday(&t, nullptr);
@ -255,7 +255,7 @@ std::string Timer::GetDateTimeFormatted(double time)
#ifdef _WIN32 #ifdef _WIN32
wchar_t tmp[32] = {}; wchar_t tmp[32] = {};
wcsftime(tmp, sizeof(tmp), L"%x %X", localTime); wcsftime(tmp, sizeof(tmp), L"%x %X", localTime);
return UTF16ToUTF8(tmp); return WStringToUTF8(tmp);
#else #else
char tmp[32] = {}; char tmp[32] = {};
strftime(tmp, sizeof(tmp), "%x %X", localTime); strftime(tmp, sizeof(tmp), "%x %X", localTime);

View File

@ -556,7 +556,7 @@ void WiimoteScannerWindows::FindWiimotes(std::vector<Wiimote*>& found_wiimotes,
WinWriteMethod write_method = GetInitialWriteMethod(IsUsingToshibaStack); WinWriteMethod write_method = GetInitialWriteMethod(IsUsingToshibaStack);
if (!IsNewWiimote(UTF16ToUTF8(device_path)) || !IsWiimote(device_path, write_method)) if (!IsNewWiimote(WStringToUTF8(device_path)) || !IsWiimote(device_path, write_method))
{ {
free(detail_data); free(detail_data);
continue; continue;
@ -608,7 +608,7 @@ bool WiimoteWindows::ConnectInternal()
if (IsConnected()) if (IsConnected())
return true; return true;
if (!IsNewWiimote(UTF16ToUTF8(m_devicepath))) if (!IsNewWiimote(WStringToUTF8(m_devicepath)))
return false; return false;
auto const open_flags = FILE_SHARE_READ | FILE_SHARE_WRITE; auto const open_flags = FILE_SHARE_READ | FILE_SHARE_WRITE;
@ -886,7 +886,7 @@ void ProcessWiimotes(bool new_scan, const T& callback)
DEBUG_LOG(WIIMOTE, "Authenticated %i connected %i remembered %i ", btdi.fAuthenticated, DEBUG_LOG(WIIMOTE, "Authenticated %i connected %i remembered %i ", btdi.fAuthenticated,
btdi.fConnected, btdi.fRemembered); btdi.fConnected, btdi.fRemembered);
if (IsValidDeviceName(UTF16ToUTF8(btdi.szName))) if (IsValidDeviceName(WStringToUTF8(btdi.szName)))
{ {
callback(hRadio, radioInfo, btdi); callback(hRadio, radioInfo, btdi);
} }

View File

@ -26,7 +26,7 @@ class WiimoteWindows final : public Wiimote
public: public:
WiimoteWindows(const std::basic_string<TCHAR>& path, WinWriteMethod initial_write_method); WiimoteWindows(const std::basic_string<TCHAR>& path, WinWriteMethod initial_write_method);
~WiimoteWindows() override; ~WiimoteWindows() override;
std::string GetId() const override { return UTF16ToUTF8(m_devicepath); } std::string GetId() const override { return WStringToUTF8(m_devicepath); }
protected: protected:
bool ConnectInternal() override; bool ConnectInternal() override;

View File

@ -61,7 +61,7 @@ void WiimoteScannerHidapi::FindWiimotes(std::vector<Wiimote*>& wiimotes, Wiimote
hid_device_info* list = hid_enumerate(0x0, 0x0); hid_device_info* list = hid_enumerate(0x0, 0x0);
for (hid_device_info* device = list; device; device = device->next) for (hid_device_info* device = list; device; device = device->next)
{ {
const std::string name = device->product_string ? UTF16ToUTF8(device->product_string) : ""; const std::string name = device->product_string ? WStringToUTF8(device->product_string) : "";
const bool is_wiimote = const bool is_wiimote =
IsValidDeviceName(name) || (device->vendor_id == 0x057e && IsValidDeviceName(name) || (device->vendor_id == 0x057e &&
(device->product_id == 0x0306 || device->product_id == 0x0330)); (device->product_id == 0x0306 || device->product_id == 0x0330));

View File

@ -113,7 +113,7 @@ bool PlatformWin32::Init()
void PlatformWin32::SetTitle(const std::string& string) void PlatformWin32::SetTitle(const std::string& string)
{ {
SetWindowTextW(m_hwnd, UTF8ToUTF16(string).c_str()); SetWindowTextW(m_hwnd, UTF8ToWString(string).c_str());
} }
void PlatformWin32::MainLoop() void PlatformWin32::MainLoop()

View File

@ -38,7 +38,7 @@ std::string GetDeviceName(const LPDIRECTINPUTDEVICE8 device)
std::string result; std::string result;
if (SUCCEEDED(device->GetProperty(DIPROP_PRODUCTNAME, &str.diph))) if (SUCCEEDED(device->GetProperty(DIPROP_PRODUCTNAME, &str.diph)))
{ {
result = StripSpaces(UTF16ToUTF8(str.wsz)); result = StripSpaces(WStringToUTF8(str.wsz));
} }
else else
{ {

View File

@ -237,7 +237,7 @@ void AutoUpdateChecker::TriggerUpdate(const AutoUpdateChecker::NewVersionInforma
STARTUPINFO sinfo = {sizeof(sinfo)}; STARTUPINFO sinfo = {sizeof(sinfo)};
sinfo.dwFlags = STARTF_FORCEOFFFEEDBACK; // No hourglass cursor after starting the process. sinfo.dwFlags = STARTF_FORCEOFFFEEDBACK; // No hourglass cursor after starting the process.
PROCESS_INFORMATION pinfo; PROCESS_INFORMATION pinfo;
if (CreateProcessW(UTF8ToUTF16(reloc_updater_path).c_str(), UTF8ToUTF16(command_line).data(), if (CreateProcessW(UTF8ToWString(reloc_updater_path).c_str(), UTF8ToWString(command_line).data(),
nullptr, nullptr, FALSE, 0, nullptr, nullptr, &sinfo, &pinfo)) nullptr, nullptr, FALSE, 0, nullptr, nullptr, &sinfo, &pinfo))
{ {
CloseHandle(pinfo.hThread); CloseHandle(pinfo.hThread);

View File

@ -110,7 +110,7 @@ std::vector<std::string> GetAdapterNames()
std::string name; std::string name;
DXGI_ADAPTER_DESC desc; DXGI_ADAPTER_DESC desc;
if (SUCCEEDED(adapter->GetDesc(&desc))) if (SUCCEEDED(adapter->GetDesc(&desc)))
name = UTF16ToUTF8(desc.Description); name = WStringToUTF8(desc.Description);
adapters.push_back(std::move(name)); adapters.push_back(std::move(name));
} }

View File

@ -28,7 +28,7 @@ std::vector<std::string> CommandLineToUtf8Argv(PCWSTR command_line)
std::vector<std::string> argv(nargs); std::vector<std::string> argv(nargs);
for (int i = 0; i < nargs; ++i) for (int i = 0; i < nargs; ++i)
{ {
argv[i] = UTF16ToUTF8(tokenized[i]); argv[i] = WStringToUTF8(tokenized[i]);
} }
LocalFree(tokenized); LocalFree(tokenized);

View File

@ -180,7 +180,7 @@ void ResetCurrentProgress()
void Error(const std::string& text) void Error(const std::string& text)
{ {
auto wide_text = UTF8ToUTF16(text); auto wide_text = UTF8ToWString(text);
MessageBox(nullptr, MessageBox(nullptr,
(L"A fatal error occured and the updater cannot continue:\n " + wide_text).c_str(), (L"A fatal error occured and the updater cannot continue:\n " + wide_text).c_str(),
@ -200,7 +200,7 @@ void SetCurrentProgress(int current, int total)
void SetDescription(const std::string& text) void SetDescription(const std::string& text)
{ {
SetWindowText(label_handle, UTF8ToUTF16(text).c_str()); SetWindowText(label_handle, UTF8ToWString(text).c_str());
} }
void MessageLoop() void MessageLoop()
@ -256,7 +256,7 @@ void LaunchApplication(std::string path)
{ {
// Hack: Launching the updater over the explorer ensures that admin priviliges are dropped. Why? // Hack: Launching the updater over the explorer ensures that admin priviliges are dropped. Why?
// Ask Microsoft. // Ask Microsoft.
ShellExecuteW(nullptr, nullptr, L"explorer.exe", UTF8ToUTF16(path).c_str(), nullptr, SW_SHOW); ShellExecuteW(nullptr, nullptr, L"explorer.exe", UTF8ToWString(path).c_str(), nullptr, SW_SHOW);
} }
void Sleep(int sleep) void Sleep(int sleep)