Skip to content

Instantly share code, notes, and snippets.

@ADeltaX
Created March 22, 2021 17:54
Show Gist options
  • Save ADeltaX/80d220579400a95c336af0eec372ecb0 to your computer and use it in GitHub Desktop.
Save ADeltaX/80d220579400a95c336af0eec372ecb0 to your computer and use it in GitHub Desktop.
Example for accessing SetWindowBand function via dll injection via explorer.exe
#define WIN32_LEAN_AND_MEAN
#include <windows.h>
#include <mutex>
//INSTALL DETOURS FROM NUGET! (or build from source yourself)
#include <detours.h>
//Definitions
typedef BOOL(WINAPI* SetWindowBand)(IN HWND hWnd, IN HWND hwndInsertAfter, IN DWORD dwBand);
typedef BOOL(WINAPI* NtUserEnableIAMAccess)(IN ULONG64 key, IN BOOL enable);
//Fields
NtUserEnableIAMAccess lNtUserEnableIAMAccess;
SetWindowBand lSetWindowBand;
ULONG64 g_iam_key = 0x0;
bool g_is_detached = false; //To prevent detaching twice.
std::mutex g_mutex;
//Forward functions
BOOL WINAPI NtUserEnableIAMAccessHook(ULONG64 key, BOOL enable);
BOOL SetWindowBandInternal(HWND hWnd, HWND hwndInsertAfter, DWORD dwBand);
//Function for detouring NtUserEnableIAMAccess
VOID AttachHook()
{
DetourTransactionBegin();
DetourUpdateThread(GetCurrentThread());
DetourAttach(&(PVOID&)lNtUserEnableIAMAccess, (PVOID)NtUserEnableIAMAccessHook);
DetourTransactionCommit();
}
//Function for restoring NtUserEnableIAMAccess
VOID DetachHook()
{
g_mutex.lock();
if (!g_is_detached)
{
DetourTransactionBegin();
DetourUpdateThread(GetCurrentThread());
DetourDetach(&(PVOID&)lNtUserEnableIAMAccess, NtUserEnableIAMAccessHook);
DetourTransactionCommit();
g_is_detached = true;
}
g_mutex.unlock();
}
//Our detoured function
BOOL WINAPI NtUserEnableIAMAccessHook(ULONG64 key, BOOL enable)
{
const auto res = lNtUserEnableIAMAccess(key, enable);
if (res == TRUE && !g_iam_key)
{
g_iam_key = key;
DetachHook();
//Example, for testing only. Don't call it here, make an IPC for that.
SetWindowBandInternal((HWND)0x503B4, NULL, 16);
}
return res;
}
//This functions is needed to induce explorer.exe (actually twinui.pcshell.dll) to call NtUserEnableIAMAccess
VOID TryForceIAMAccessCallThread(LPVOID lpParam)
{
//These 7 calls will force a call into EnableIAMAccess.
auto hwndFore = GetForegroundWindow();
auto hwndToFocus = FindWindow(L"Shell_TrayWnd", NULL);
SetForegroundWindow(GetDesktopWindow()); //This in case Shell_TrayWnd is already focused
Sleep(100);
SetForegroundWindow(hwndToFocus); //Focus on the taskbar, should trigger EnableIAMAccess
Sleep(100);
SetForegroundWindow(hwndFore); //Restore focus.
}
//Function helper to call SetWindowBand in the proper way.
BOOL SetWindowBandInternal(HWND hWnd, HWND hwndInsertAfter, DWORD dwBand)
{
if (g_iam_key)
{
lNtUserEnableIAMAccess(g_iam_key, TRUE);
const auto callResult = lSetWindowBand(hWnd, hwndInsertAfter, dwBand);
lNtUserEnableIAMAccess(g_iam_key, FALSE);
return callResult;
}
return FALSE;
}
//DllMain
BOOL APIENTRY DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
{
switch (ul_reason_for_call)
{
case DLL_PROCESS_ATTACH:
{
DisableThreadLibraryCalls(hModule);
const auto path = LoadLibrary(L"user32.dll");
lSetWindowBand = SetWindowBand(GetProcAddress(path, "SetWindowBand"));
lNtUserEnableIAMAccess = NtUserEnableIAMAccess(GetProcAddress(path, MAKEINTRESOURCEA(2510)));
AttachHook();
CreateThread(NULL, NULL, (LPTHREAD_START_ROUTINE)&TryForceIAMAccessCallThread, NULL, NULL, NULL);
}
break;
case DLL_PROCESS_DETACH:
{
DetachHook();
}
break;
}
return TRUE;
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment