Skip to content

Commit

Permalink
Fea #66, 添加FindNLSStringEx(让.NET 8/9 支持 Windows XP)
Browse files Browse the repository at this point in the history
  • Loading branch information
mingkuang-Chuyu committed Sep 9, 2024
1 parent 48a5da3 commit 8a5483e
Show file tree
Hide file tree
Showing 4 changed files with 505 additions and 262 deletions.
1 change: 1 addition & 0 deletions ThunksList.md
Original file line number Diff line number Diff line change
Expand Up @@ -530,6 +530,7 @@
| GetCalendarInfoEx | 不存在时,调用GetCalendarInfoW。
| GetNLSVersionEx | 不存在时,返回一个假版本。
| IsNLSDefinedString | 不存在时,调用GetStringTypeW。
| FindNLSStringEx | 调用 CompareStringW。
| SetProcessWorkingSetSizeEx | 不存在时,调用SetProcessWorkingSetSize。
| GetProcessWorkingSetSizeEx | 不存在时,调用GetProcessWorkingSetSize。
| GetTimeZoneInformationForYear | 不存在时,直接读取`Time Zones`注册表。
Expand Down
141 changes: 141 additions & 0 deletions src/Thunks/api-ms-win-core-localization.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -2537,4 +2537,145 @@

}
#endif


#if (YY_Thunks_Target < __WindowsNT6)

// 最低受支持的客户端 Windows Vista [桌面应用 | UWP 应用]
// 最低受支持的服务器 Windows Server 2008[桌面应用 | UWP 应用]
__DEFINE_THUNK(
kernel32,
40,
int,
WINAPI,
FindNLSStringEx,
_In_opt_ LPCWSTR _szLocaleName,
_In_ DWORD _fFindNLSStringFlags,
_In_reads_(_cchSource) LPCWSTR _szStringSource,
_In_ int _cchSource,
_In_reads_(_cchValue) LPCWSTR _szStringValue,
_In_ int _cchValue,
_Out_opt_ LPINT _pcchFound,
_In_opt_ LPNLSVERSIONINFO _pVersionInformation,
_In_opt_ LPVOID _pReserved,
_In_opt_ LPARAM _hSortHandle
)
{
if (auto const _pfnFindNLSStringEx = try_get_FindNLSStringEx())
{
return _pfnFindNLSStringEx(_szLocaleName, _fFindNLSStringFlags, _szStringSource, _cchSource, _szStringValue, _cchValue, _pcchFound, _pVersionInformation, _pReserved, _hSortHandle);
}

__WarningMessage__("FindNLSStringEx 暂时只支持搜索 _cchValue 的子字符串。");

const DWORD _fFindFlags = _fFindNLSStringFlags & (FIND_STARTSWITH | FIND_ENDSWITH | FIND_FROMSTART | FIND_FROMEND);
if (_fFindFlags & (_fFindFlags - 1))
{
SetLastError(ERROR_INVALID_FLAGS);
return -1;
}

if (_pVersionInformation || _pReserved|| _hSortHandle
|| _szStringSource == nullptr || _cchSource == 0 || _cchSource < -1
|| _szStringValue == nullptr || _cchValue == 0 || _cchValue < -1)
{
SetLastError(ERROR_INVALID_PARAMETER);
return -1;
}

const auto _Locale = LocaleNameToLCID(_szLocaleName, 0);
if (_Locale == 0)
{
SetLastError(ERROR_INVALID_PARAMETER);
return -1;
}

if (_cchSource == -1)
{
_cchSource = wcslen(_szStringSource);
}

if (_cchValue == -1)
{
_cchValue = wcslen(_szStringValue);
}

if (_cchSource < _cchValue)
return -1;

const DWORD _fCmpFlags = _fFindNLSStringFlags & ~(FIND_STARTSWITH | FIND_ENDSWITH | FIND_FROMSTART | FIND_FROMEND);
if ((_fFindNLSStringFlags & (FIND_ENDSWITH | FIND_FROMEND)) == 0)
{
// 从头开始搜索
if (_fFindNLSStringFlags & FIND_STARTSWITH)
{
const auto _nResult = CompareStringW(_Locale, _fCmpFlags, _szStringSource, _cchValue, _szStringValue, _cchValue);
if (_nResult == CSTR_EQUAL)
{
if (_pcchFound)
*_pcchFound = _cchValue;

return 0;
}
}
else
{
auto _szStr = _szStringSource;
auto _szStrEnd = _szStringSource + _cchSource - _cchValue + 1;
for (; _szStr != _szStrEnd;++_szStr)
{
const auto _nResult = CompareStringW(_Locale, _fCmpFlags, _szStr, _cchValue, _szStringValue, _cchValue);
if (_nResult == 0)
return -1;

if (_nResult == CSTR_EQUAL)
{
if (_pcchFound)
*_pcchFound = _cchValue;

return _szStr - _szStringSource;
}
}
}
}
else
{
// 反向搜索
auto _szStr = _szStringSource + _cchSource - _cchValue;
if (_fFindNLSStringFlags & FIND_ENDSWITH)
{
const auto _nResult = CompareStringW(_Locale, _fCmpFlags, _szStr, _cchValue, _szStringValue, _cchValue);
if (_nResult == CSTR_EQUAL)
{
if (_pcchFound)
*_pcchFound = _cchValue;

return _szStr - _szStringSource;
}
}
else
{
for(;; --_szStr)
{
const auto _nResult = CompareStringW(_Locale, _fCmpFlags, _szStr, _cchValue, _szStringValue, _cchValue);
if (_nResult == 0)
return -1;

if (_nResult == CSTR_EQUAL)
{
if (_pcchFound)
*_pcchFound = _cchValue;

return _szStr - _szStringSource;
}

if (_szStr == _szStringSource)
break;
}
}
}

return -1;
}
#endif
} //namespace YY
52 changes: 31 additions & 21 deletions src/Thunks/api-ms-win-core-string.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -265,25 +265,11 @@ namespace YY::Thunks
#endif


#if defined(YY_Thunks_Implemented) && (YY_Thunks_Target < __WindowsNT5_1)
static bool IsSupportLOCALE_INVARIANT()
{
// 特意不考虑多线程支持,因为这里产生写入竞争并没有关系。
static int s_iStat = 0;
if (s_iStat == 0)
{
s_iStat = internal::GetSystemVersion() >= internal::MakeVersion(5, 1) ? 1 : -1;
}
return s_iStat == 1;
}
#endif


#if (YY_Thunks_Target < __WindowsNT5_1)
#if (YY_Thunks_Target < __WindowsNT6)

// Windows 2000 不支持 LOCALE_INVARIANT 参数特殊处理一下。
// Minimum supported client Windows 2000 Professional [desktop apps only]
// Minimum supported server Windows 2000 Server [desktop apps only]
// 虽然Windows 2000已经支持,但是几个新标记老系统不支持,特殊处理一下。
__DEFINE_THUNK(
kernel32,
24,
Expand All @@ -305,21 +291,33 @@ namespace YY::Thunks
return 0;
}

if (Locale == LOCALE_INVARIANT && IsSupportLOCALE_INVARIANT() == false)
#if (YY_Thunks_Target < __WindowsNT5_1)
// Windows 2000 不支持 LOCALE_INVARIANT 参数特殊处理一下。
// 网友 海好蓝 提供
if (Locale == LOCALE_INVARIANT && internal::GetSystemVersion() < internal::MakeVersion(5, 1))
{
Locale = MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), SORT_DEFAULT);
}
#endif

#if (YY_Thunks_Target < __WindowsNT6)
// Windows XP不支持 NORM_LINGUISTIC_CASING
// 网友 海好蓝 反馈
if (internal::GetSystemVersion() < internal::MakeVersion(6, 0))
{
dwCmpFlags &= ~NORM_LINGUISTIC_CASING;
}
#endif
return _pfnCompareStringA(Locale, dwCmpFlags, lpString1, cchCount1, lpString2, cchCount2);
}
#endif


#if (YY_Thunks_Target < __WindowsNT5_1)
#if (YY_Thunks_Target < __WindowsNT6)

// Windows 2000 不支持 LOCALE_INVARIANT 参数特殊处理一下。
// Minimum supported client Windows 2000 Professional [desktop apps only]
// Minimum supported server Windows 2000 Server [desktop apps only]
// 虽然Windows 2000已经支持,但是几个新标记老系统不支持,特殊处理一下。
__DEFINE_THUNK(
kernel32,
24,
Expand All @@ -334,18 +332,30 @@ namespace YY::Thunks
_In_ int cchCount2
)
{
// 网友 海好蓝 提供
const auto _pfnCompareStringW = try_get_CompareStringW();
if (!_pfnCompareStringW)
{
SetLastError(ERROR_NOT_SUPPORTED);
return 0;
}

if (Locale == LOCALE_INVARIANT && IsSupportLOCALE_INVARIANT() == false)
#if (YY_Thunks_Target < __WindowsNT5_1)
// Windows 2000 不支持 LOCALE_INVARIANT 参数特殊处理一下。
// 网友 海好蓝 提供
if (Locale == LOCALE_INVARIANT && internal::GetSystemVersion() < internal::MakeVersion(5, 1))
{
Locale = MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), SORT_DEFAULT);
}
#endif

#if (YY_Thunks_Target < __WindowsNT6)
// Windows XP不支持 NORM_LINGUISTIC_CASING
// 网友 海好蓝 反馈
if (internal::GetSystemVersion() < internal::MakeVersion(6, 0))
{
dwCmpFlags &= ~NORM_LINGUISTIC_CASING;
}
#endif

return _pfnCompareStringW(Locale, dwCmpFlags, lpString1, cchCount1, lpString2, cchCount2);
}
Expand Down
Loading

0 comments on commit 8a5483e

Please sign in to comment.