众所周知Windows XP每个SP都会新增大量API,而Windows 10每次更新又会新增大量API,这使得兼容不同版本的Windows需要大量的判断。
甚至大量开源代码已经不再兼容一些早期的Windows XP版本,比如 Windows XP RTM。难道就没有一种快速高效的方案解决无法定位程序输入点的问题吗?
YY-Thunks(鸭船),存在的目的就是抹平不同系统的差异,编译时单纯添加一个obj即可自动解决这些兼容性问题。让你兼容Windows更轻松!
[ 鸭船交流群 633710173 ]
使用LoadLibrary
以及GetProcAddress
动态加载API,不存在时做出补偿措施,最大限度模拟原始API行为,让你的程序正常运行。
- 更快!更安全!
鸭船
内建2级缓存以及按需加载机制,同时自动加密所有函数指针,防止内存爆破攻击。最大程度减少不需要的、不必要的LoadLibrary以及GetProcAddress
调用以及潜在安全风险。 - 轻松兼容Windows XP,让你安心专注于业务逻辑。
- 完全的开放代码,广泛的接受用户意见,希望大家能踊跃的 pull requests,为
鸭船
添砖加瓦。
- 下载YY-Thunks-Binary,然后解压到你的工程目录。
- 【链接器】-【输入】-【附加依赖项】,添加
objs\$(PlatformShortName)\YY_Thunks_for_WinXP.obj
。 - 重新编译代码。
全平台ABI兼容。
- 所有Visual Studio版本均支持(比如:VC6.0、VS2008、VS2010、VS2015、VS2017、VS2019等等)。
- 所有运行库模式均支持(比如:
/MD
、/MT
、/MDd
、/MTd
)。
此表展示了YY-Thunks(鸭船)可以解决的函数不存在问题,欢迎大家扩充!
开头带
*
的函数并不建议使用,仅用于编译通过处理,如果使用可能导致老版本系统无法充分发挥性能。
函数 | Fallback |
---|---|
DecodePointer | 不存在时,返回指针本身。 |
EncodePointer | 不存在时,返回指针本身。 |
RegDeleteKeyExW(A) | 不存在时,调用RegDeleteKeyW(A)。 |
Wow64DisableWow64FsRedirection | 不存在时,返回FALSE,并设置 LastError = ERROR_INVALID_FUNCTION。 |
Wow64RevertWow64FsRedirection | 不存在时,返回FALSE,并设置 LastError = ERROR_INVALID_FUNCTION。 |
Wow64EnableWow64FsRedirection | 不存在时,返回FALSE,并设置 LastError = ERROR_INVALID_FUNCTION。 |
IsWow64Process | 不存在时,返回TRUE,并设置 *Wow64Process = FALSE 。 |
IsWow64Message | 不存在时,返回FALSE。 |
RegSetKeyValueW(A) | 调用RegCreateKeyExW(A)以及RegSetValueExW(A)。 |
RegDeleteKeyValueW(A) | 调用RegOpenKeyExW(A)以及RegDeleteValueW(A)。 |
RegDeleteTreeW(A) | 调用SHDeleteKeyW(A)。 |
RegGetValueW(A) | 不存在时,调用RegQueryValueExW(A)。 |
RegCopyTreeW(A) | 不存在时,调用SHCopyKeyW(A)。 |
IsWow64Process2 | 不存在时,调用IsWow64Process。 |
IsWow64GuestMachineSupported | 不存在时,调用GetNativeSystemInfo。 |
GetTickCount64 | 不存在时,调用GetTickCount。 |
GetSystemTimePreciseAsFileTime | 不存在时,调用GetSystemTimeAsFileTime。 |
InitializeCriticalSectionEx | 不存在时,调用InitializeCriticalSectionAndSpinCount。 |
InitOnceInitialize | 初始化为 INIT_ONCE_STATIC_INIT。 |
InitOnceBeginInitialize | 不存在时,调用NtWaitForKeyedEvent。 |
InitOnceComplete | 不存在时,调用NtReleaseKeyedEvent。 |
InitOnceExecuteOnce | 不存在时,调用NtWaitForKeyedEvent以及NtReleaseKeyedEvent。 |
LocaleNameToLCID | 不存在时,查LocaleNameToLcidTable。 |
LCIDToLocaleName | 不存在时,查LcidToLocaleNameTable。 |
GetLocaleInfoEx | 不存在时,调用GetLocaleInfoW。 |
GetDateFormatEx | 不存在时,调用GetDateFormatW。 |
GetTimeFormatEx | 不存在时,调用GetTimeFormatW。 |
GetNumberFormatEx | 不存在时,调用GetNumberFormatW。 |
GetCurrencyFormatEx | 不存在时,调用GetCurrencyFormatW。 |
GetUserDefaultLocaleName | 不存在时,调用LCIDToLocaleName。 |
GetSystemDefaultLocaleName | 不存在时,调用LCIDToLocaleName。 |
EnumCalendarInfoExEx | 不存在时,调用EnumCalendarInfoExW。 |
EnumDateFormatsExEx | 不存在时,调用EnumDateFormatsExW。 |
GetFileInformationByHandleEx | 不存在时,调用NtQueryInformationFile 或者 NtQueryDirectoryFile。 |
SetFileInformationByHandle | 不存在时,调用NtSetInformationFile。 |
GetFinalPathNameByHandleW(A) | 不存在时,调用NtQueryObject以及NtQueryInformationFile。 |
GetLogicalProcessorInformation | 不存在时,返回FALSE,并设置 LastError = ERROR_INVALID_FUNCTION。 |
GetLogicalProcessorInformationEx | 不存在时,调用GetLogicalProcessorInformation。 |
InetPtonW(inet_pton) | 不存在时,类似于sscanf手工分析字符串。 |
InetNtopW(inet_ntop) | 不存在时,类似于sprintf手工生成字符串。 |
WSAPoll | 不存在时,调用select。 |
GetNumaHighestNodeNumber | 不存在时,返回0。 |
RaiseFailFastException | 不存在时,调用TerminateProcess。 |
GetThreadId | 不存在时,调用NtQueryInformationThread。 |
GetProcessIdOfThread | 不存在时,调用NtQueryInformationThread。 |
GetProcessId | 不存在时,调用NtQueryInformationProcess。 |
QueryThreadCycleTime | 不存在时,调用GetThreadTimes。 |
QueryProcessCycleTime | 不存在时,调用GetProcessTimes。 |
K32EnumProcessModules | 调用EnumProcessModules。 |
K32EnumProcessModulesEx | 调用EnumProcessModulesEx。 |
K32GetModuleBaseNameW(A) | 调用GetModuleBaseNameW(A)。 |
K32GetModuleFileNameExW(A) | 调用K32GetModuleFileNameExW(A)。 |
K32EmptyWorkingSet | 调用EmptyWorkingSet。 |
K32QueryWorkingSet | 调用QueryWorkingSet。 |
K32QueryWorkingSetEx | 调用QueryWorkingSetEx。 |
K32InitializeProcessForWsWatch | 调用InitializeProcessForWsWatch。 |
K32GetWsChanges | 调用GetWsChanges。 |
K32GetWsChangesEx | 调用GetWsChangesEx。 |
K32GetMappedFileNameW(A) | 调用GetMappedFileNameW(A)。 |
K32EnumDeviceDrivers | 调用EnumDeviceDrivers。 |
K32GetDeviceDriverBaseNameW(A) | 调用GetDeviceDriverBaseNameW(A)。 |
K32GetDeviceDriverFileNameW(A) | 调用GetDeviceDriverFileNameW(A)。 |
K32GetPerformanceInfo | 调用GetPerformanceInfo。 |
K32EnumPageFilesW(A) | 调用EnumPageFilesW(A)。 |
K32GetProcessImageFileNameW(A) | 调用GetProcessImageFileNameW(A)。 |
EnumProcessModulesEx | 不存在时,调用EnumProcessModules。 |
GetWsChangesEx | 不存在时,调用GetWsChanges。 |
QueryFullProcessImageNameW(A) | 不存在时,调用GetProcessImageFileNameW(A) 或者 GetModuleFileNameExW(A)。 |
CreateFile2 | 不存在时,调用CreateFileW。 |
CreateEventExW(A) | 不存在时,调用CreateEventW(A)。 |
CreateMutexExW(A) | 不存在时,调用CreateMutexW(A)。 |
CreateSemaphoreExW | 不存在时,调用CreateSemaphoreW。 |
CreateWaitableTimerExW | 不存在时,调用CreateWaitableTimerW。 |
GetFileVersionInfoExW(A) | 不存在时,调用GetFileVersionInfoW(A)。 |
GetFileVersionInfoSizeExW(A) | 不存在时,调用GetFileVersionInfoSizeW(A)。 |
InterlockedCompareExchange64 | 调用内部函数_InterlockedCompareExchange64。 |
SetThreadErrorMode | 不存在时,调用SetErrorMode。 |
GetThreadErrorMode | 不存在时,调用GetErrorMode。 |
GetErrorMode | 不存在时,调用NtQueryInformationProcess。 |
InitializeSRWLock | 初始化为 RTL_SRWLOCK_INIT。 |
AcquireSRWLockExclusive | 不存在时,调用NtWaitForKeyedEvent。 |
TryAcquireSRWLockExclusive | 不存在时,调用InterlockedBitTestAndSet(64)。 |
ReleaseSRWLockExclusive | 不存在时,调用NtReleaseKeyedEvent。 |
AcquireSRWLockShared | 不存在时,调用NtWaitForKeyedEvent。 |
TryAcquireSRWLockShared | 不存在时,调用InterlockedCompareExchange。 |
ReleaseSRWLockShared | 不存在时,调用NtReleaseKeyedEvent。 |
InitializeConditionVariable | 初始化为 CONDITION_VARIABLE_INIT。 |
SleepConditionVariableCS | 不存在时,调用NtWaitForKeyedEvent。 |
SleepConditionVariableSRW | 不存在时,调用NtWaitForKeyedEvent。 |
WakeConditionVariable | 不存在时,调用NtReleaseKeyedEvent。 |
WakeAllConditionVariable | 不存在时,调用NtReleaseKeyedEvent。 |
InitializeSynchronizationBarrier | 不存在时,调用CreateEvent。 |
EnterSynchronizationBarrier | 不存在时,调用WaitForSingleObject。 |
DeleteSynchronizationBarrier | 不存在时,调用CloseHandle。 |
WaitOnAddress | 不存在时,调用NtWaitForKeyedEvent。警告,此函数请勿跨模块使用!!! |
WakeByAddressSingle | 不存在时,调用NtReleaseKeyedEvent。警告,此函数请勿跨模块使用!!! |
WakeByAddressAll | 不存在时,调用NtReleaseKeyedEvent。警告,此函数请勿跨模块使用!!! |
*GetCurrentProcessorNumber | 不存在时,返回0。 |
*GetCurrentProcessorNumberEx | 不存在时,调用GetCurrentProcessorNumber。 |
*GetNumaNodeProcessorMask | 不存在时,返回FALSE,并设置 LastError = ERROR_INVALID_PARAMETER。 |
*GetNumaNodeProcessorMaskEx | 不存在时,调用GetNumaNodeProcessorMask。 |
*SetThreadGroupAffinity | 不存在时,调用SetThreadAffinityMask。 |
*CancelIoEx | 不存在时,调用CancelIo(会把此句柄的所有IO操作取消掉!)。 |
*QueryWorkingSetEx | 不存在时,返回FALSE,并设置 LastError = ERROR_INVALID_FUNCTION。 |
SHGetKnownFolderPath | 不存在时,调用SHGetFolderPathW。 |
SHSetKnownFolderPath | 不存在时,调用SHSetFolderPathW。 |
SHGetKnownFolderIDList | 不存在时,调用SHGetFolderLocation。 |
SHBindToFolderIDListParent | 不存在时,调用IShellFolder。 |
SHBindToFolderIDListParentEx | 不存在时,调用IShellFolder。 |
SHBindToObject | 不存在时,调用IShellFolder。 |
SHCreateItemFromIDList | 不存在时,调用IShellItem。 |
SHCreateItemWithParent | 不存在时,调用IShellItem。 |
SHCreateItemFromRelativeName | 不存在时,调用IShellItem。 |
SHGetNameFromIDList | 不存在时,调用IShellItem。 |
SHCreateShellItem | 不存在时,调用IShellItem。 |
OpenFileById | 不存在时,调用NtCreateFile。 |
CreateSymbolicLinkW(A) | 不存在时,调用DeviceIoControl。 |
ReOpenFile | 不存在时,调用NtCreateFile。 |
- YY-Thunks第一版
- 添加RegSetKeyValueW(A)
- 添加RegDeleteKeyValueW(A)
- 添加RegDeleteTreeW(A)
- 添加IsWow64Process2
- 添加IsWow64GuestMachineSupported
- 添加GetTickCount64
- 添加GetSystemTimePreciseAsFileTime
- 添加InitializeCriticalSectionEx
- 添加InitOnceExecuteOnce
- 添加GetCurrentProcessorNumber
- 添加GetCurrentProcessorNumberEx
- 添加GetNumaNodeProcessorMask
- 添加GetNumaNodeProcessorMaskEx
- 添加SetThreadGroupAffinity
- 解决Bug,将初始化时机推迟到
.CRT$XID
,避免VC2008下过早初始化导致atexit崩溃(感谢 死田鸡)。
- 添加RegGetValueW(A)
- 添加LocaleNameToLCID
- 添加LCIDToLocaleName
- 添加GetLocaleInfoEx
- 添加GetDateFormatEx
- 添加GetTimeFormatEx
- 添加GetNumberFormatEx
- 添加GetCurrencyFormatEx
- 添加GetUserDefaultLocaleName
- 添加GetSystemDefaultLocaleName
- 重新编译,解决潜在符号修饰问题。
- 添加EnumCalendarInfoExEx
- 添加EnumDateFormatsExEx
- 添加GetFileInformationByHandleEx
- 添加SetFileInformationByHandle
- 添加GetFinalPathNameByHandleW(A)
- 解决Bug,1.0.0.10版意外引入ntdll.lib问题(感谢 小古)。
- 添加GetLogicalProcessorInformation
- 添加GetLogicalProcessorInformationEx
- 添加InetPtonW(inet_pton)
- 添加InetNtopW(inet_ntop)
- 添加GetNumaHighestNodeNumber
- 添加RaiseFailFastException(感谢 过客)
- 添加GetThreadId(感谢 过客)
- 添加GetProcessIdOfThread
- 添加QueryThreadCycleTime(感谢 过客)
- 添加QueryProcessCycleTime
- 添加K32EnumProcessModules
- 添加K32EnumProcessModulesEx
- 添加K32GetModuleBaseNameW(A)
- 添加K32GetModuleFileNameExW(A)
- 添加K32EmptyWorkingSet
- 添加K32QueryWorkingSet
- 添加K32QueryWorkingSetEx
- 添加K32InitializeProcessForWsWatch
- 添加K32GetWsChanges
- 添加K32GetWsChangesEx
- 添加K32GetMappedFileNameW(A)
- 添加K32EnumDeviceDrivers
- 添加K32GetDeviceDriverBaseNameW(A)
- 添加K32GetDeviceDriverFileNameW(A)
- 添加K32GetPerformanceInfo
- 添加K32EnumPageFilesW(A)
- 添加K32GetProcessImageFileNameW(A)
- 添加EnumProcessModulesEx
- 添加GetWsChangesEx
- 添加QueryFullProcessImageNameW(A)
- 添加CreateFile2
- 添加CreateEventExW(A)
- 添加CreateMutexExW(A)
- 添加CreateSemaphoreExW
- 添加CreateWaitableTimerExW
- 添加GetFileVersionInfoExW(A)
- 添加GetFileVersionInfoSizeExW(A)
- 添加QueryFullProcessImageNameW(A)
- 添加InterlockedCompareExchange64
- 添加GetProcessId
- 添加SetThreadErrorMode
- 添加GetThreadErrorMode
- 添加GetErrorMode
- 添加CancelIoEx
- 添加InitializeSRWLock
- 添加AcquireSRWLockExclusive
- 添加ReleaseSRWLockExclusive
- 添加AcquireSRWLockShared
- 添加ReleaseSRWLockShared
- 添加TryAcquireSRWLockExclusive
- 添加TryAcquireSRWLockShared
- 优化代码结构,减少不必要的全局对象引入。
- 内部函数采用__fastcall约定,减少栈操作。
- 增加YY-Thunks版本互斥。
- 改进与联想一键影音的兼容性,由于联想一键影音错会乱Hook,导致LoadLibraryExW行为异常。这样将导致YY-Thunks等在没有安装KB2533623的系统上无法正常使用问题。
- 改进Windows 7 RTM以及以下系统的兼容性,由于这些老版本系统由于在LoadLibraryExW期间不会恢复重定向,因此当目标线程关闭重定向时可能导致YY-Thunks无法正常工作。
- 添加QueryWorkingSetEx
- 解决Bug,VS2010无法使用问题(感谢 柒零)。
- 改进 SRWLock 以及 One-Time Initialization调整为KeyedEvent实现。
- 添加WSAPoll
- 添加InitializeConditionVariable
- 添加SleepConditionVariableCS
- 添加SleepConditionVariableSRW
- 添加WakeConditionVariable
- 添加WakeAllConditionVariable
- 添加InitOnceInitialize
- 添加InitOnceBeginInitialize
- 添加InitOnceComplete
- 添加InitializeSynchronizationBarrier
- 添加EnterSynchronizationBarrier
- 添加DeleteSynchronizationBarrier
- 添加WaitOnAddress
- 添加WakeByAddressSingle
- 添加WakeByAddressAll
- 添加SHGetKnownFolderPath
- 添加SHSetKnownFolderPath
- 添加SHGetKnownFolderIDList
- 解决 Bug,CreateFile2 dwSecurityQosFlags成员可能无法发挥作用问题(感谢 賈可)。
- 解决 Bug,KnownFoldersIdsMap缺少 FOLDERID_ProgramFilesCommonX86问题(感谢 賈可)。
- 解决 Bug,KnownFoldersIdsMap会生static静态对象初始化代码问题(感谢 Joe)。
- 行为调整,EnumCalendarInfoExEx、EnumDateFormatsExEx消除 TLS 依赖。
- 添加 SHBindToFolderIDListParent(感谢 賈可)
- 添加 SHBindToFolderIDListParentEx(感谢 賈可)
- 添加 SHBindToObject(感谢 賈可)
- 添加 SHCreateItemFromIDList(感谢 賈可)
- 添加 SHCreateItemWithParent(感谢 賈可)
- 添加 SHCreateItemFromRelativeName(感谢 賈可)
- 添加 SHGetNameFromIDList(感谢 賈可)
- 添加 SHCreateShellItem
- 添加 OpenFileById
- 添加 RegCopyTreeW(A)(感谢 賈可)
- 添加 CreateSymbolicLinkW(A)
- 添加 ReOpenFile