windows-nt/Source/XPSP1/NT/windows/appcompat/shims/specific/ultrawincleaner2002.cpp
2020-09-26 16:20:57 +08:00

166 lines
5.2 KiB
C++

/*++
Copyright (c) 2002 Microsoft Corporation
Module Name:
UltraWinCleaner2002.cpp
Abstract:
Ultra WinCleaner 2002 - WinJunk Cleaner removes files under
%WINDIR%\Resource\Themes. This causes themes to fail to load. Fixed by
hiding everything under %WINDIR%\Resource from FindFirstFileA.
Notes:
This is a application specific shim.
History:
5/13/2002 mikrause Created
--*/
#include "precomp.h"
#include "StrSafe.h"
#include <nt.h>
IMPLEMENT_SHIM_BEGIN(UltraWinCleaner2002)
#include "ShimHookMacro.h"
APIHOOK_ENUM_BEGIN
APIHOOK_ENUM_ENTRY(NtQueryDirectoryFile)
APIHOOK_ENUM_END
UNICODE_STRING g_strWinResourceDir;
typedef NTSTATUS (WINAPI *_pfn_NtQueryDirectoryFile)(HANDLE, HANDLE,
PIO_APC_ROUTINE, PVOID, PIO_STATUS_BLOCK, PVOID, ULONG,
FILE_INFORMATION_CLASS, BOOLEAN, PUNICODE_STRING, BOOLEAN);
NTSTATUS
APIHOOK(NtQueryDirectoryFile)(
IN HANDLE FileHandle,
IN HANDLE Event OPTIONAL,
IN PIO_APC_ROUTINE ApcRoutine OPTIONAL,
IN PVOID ApcContext OPTIONAL,
OUT PIO_STATUS_BLOCK IoStatusBlock,
OUT PVOID FileInformation,
IN ULONG FileInformationLength,
IN FILE_INFORMATION_CLASS FileInformationClass,
IN BOOLEAN ReturnSingleEntry,
IN PUNICODE_STRING FileName OPTIONAL,
IN BOOLEAN RestartScan
)
{
//
// If caller wants any async behavior, skip.
// FileName must be valid.
//
if (Event == NULL && ApcRoutine == NULL && ApcContext == NULL && FileName != NULL) {
// Only trap complete wildcard searches.
if (lstrcmpW(FileName->Buffer, L"*.*") == 0 ||
lstrcmpW(FileName->Buffer, L"*") == 0) {
DWORD dwSize = MAX_PATH * sizeof(WCHAR) + sizeof(OBJECT_NAME_INFORMATION);
PBYTE pbBuffer = new BYTE[dwSize];
if (pbBuffer) {
ULONG RetLen;
ZeroMemory(pbBuffer, dwSize);
POBJECT_NAME_INFORMATION poni = (POBJECT_NAME_INFORMATION)pbBuffer;
// Get the name of the directory
NTSTATUS status = NtQueryObject(FileHandle, ObjectNameInformation,
poni, dwSize, &RetLen);
// Retry if not enough buffer
if (status == STATUS_BUFFER_TOO_SMALL) {
delete [] pbBuffer;
pbBuffer = new BYTE[RetLen];
if (pbBuffer) {
poni = (POBJECT_NAME_INFORMATION)pbBuffer;
status = NtQueryObject(FileHandle, ObjectNameInformation,
poni, RetLen, &RetLen);
}
}
// Check if it is the Windows Resource directory.
if (NT_SUCCESS(status)) {
if (CompareString(LOCALE_INVARIANT, NORM_IGNORECASE,
poni->Name.Buffer, poni->Name.Length / sizeof(WCHAR),
g_strWinResourceDir.Buffer, g_strWinResourceDir.Length / sizeof(WCHAR)) == CSTR_EQUAL) {
// Pretend this directory doesn't exist.
DPFN(eDbgLevelInfo, "[NtQueryDirectoryFile] Ignoring all file searches in %S",
poni->Name.Buffer);
delete [] pbBuffer;
return STATUS_NO_SUCH_FILE;
}
}
if (pbBuffer) {
delete [] pbBuffer;
}
}
}
}
return ORIGINAL_API(NtQueryDirectoryFile)(FileHandle, Event, ApcRoutine,
ApcContext, IoStatusBlock, FileInformation, FileInformationLength,
FileInformationClass, ReturnSingleEntry, FileName, RestartScan);
}
BOOL
NOTIFY_FUNCTION(
DWORD fdwReason
)
{
if (fdwReason == DLL_PROCESS_ATTACH)
{
CSTRING_TRY
{
// Get the Windows Resource directory (which holds themes)
CString csWinResourceDir;
csWinResourceDir.GetWindowsDirectoryW();
csWinResourceDir.AppendPath(L"Resources");
//
// Convert the DOS name like C:\Windows\Resources to
// \Device\HarddiskVolume0\Windows\Resources which NT likes.
//
CString csWinDrive;
csWinResourceDir.GetDrivePortion(csWinDrive);
WCHAR wszBuffer[1024];
if (QueryDosDeviceW(csWinDrive, wszBuffer, 1024)) {
csWinResourceDir.Replace(csWinDrive, wszBuffer);
ZeroMemory(&g_strWinResourceDir, sizeof(g_strWinResourceDir));
PWSTR wsz = new WCHAR[csWinResourceDir.GetLength()+1];
if (wsz) {
StringCchCopyW(wsz, csWinResourceDir.GetLength() + 1, csWinResourceDir);
DPFN(eDbgLevelInfo, "Ignoring all file searches in %S", wsz);
RtlInitUnicodeString(&g_strWinResourceDir, wsz);
} else {
return FALSE;
}
} else {
return FALSE;
}
}
CSTRING_CATCH
{
DPFN(eDbgLevelError, "CString exception in NotifyFunc!\n");
return FALSE;
}
}
return TRUE;
}
HOOK_BEGIN
CALL_NOTIFY_FUNCTION
APIHOOK_ENTRY(NTDLL.DLL, NtQueryDirectoryFile)
HOOK_END
IMPLEMENT_SHIM_END