7

我正在尝试编写一个CreateRemoteThread用于注入 dll 的程序。

问题是 CreateRemoteThread 拒绝工作。GetLastError() 返回 5,即 ERROR_ACCESS_DENIED。我想不通为什么!

我正在观看此视频http://www.youtube.com/watch?v=H3O3hmXkt1I

#include <iostream>
#include <direct.h>
#include <Windows.h>
#include <TlHelp32.h>

using namespace std;


char* GetCurrentDir()
{
    char*   szRet = (char*)malloc(MAX_PATH);
    _getcwd(szRet, MAX_PATH);
    return szRet;
}

LPCTSTR SzToLPCTSTR(char* szString)
{
    LPTSTR  lpszRet;
    size_t  size = strlen(szString)+1;

    lpszRet = (LPTSTR)malloc(MAX_PATH);
    mbstowcs_s(NULL, lpszRet, size, szString, _TRUNCATE);

    return lpszRet;
}

void WaitForProcessToAppear(LPCTSTR lpcszProc, DWORD dwDelay)
{
    HANDLE          hSnap;
    PROCESSENTRY32  peProc;
    BOOL            bAppeared = FALSE;

    while(!bAppeared)
    {
        if((hSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0)) != INVALID_HANDLE_VALUE)
        {
            peProc.dwSize = sizeof(PROCESSENTRY32);
            if(Process32First(hSnap, &peProc))
                while(Process32Next(hSnap, &peProc) && !bAppeared)
                    if(!lstrcmp(lpcszProc, peProc.szExeFile))
                        bAppeared = TRUE;
        }
        CloseHandle(hSnap);
        Sleep(dwDelay);
    }
}

DWORD GetProcessIdByName(LPCTSTR lpcszProc)
{
    HANDLE          hSnap;
    PROCESSENTRY32  peProc;
    DWORD           dwRet = -1;

    if((hSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0)) != INVALID_HANDLE_VALUE)
    {
        peProc.dwSize = sizeof(PROCESSENTRY32);
        if(Process32First(hSnap, &peProc))
            while(Process32Next(hSnap, &peProc))
                if(!lstrcmp(lpcszProc, peProc.szExeFile))
                    dwRet = peProc.th32ProcessID;
    }
    CloseHandle(hSnap);

    return dwRet;
}

BOOL InjectDll(DWORD dwPid, char* szDllPath)
{
    DWORD   dwMemSize;
    HANDLE  hProc;
    LPVOID  lpRemoteMem, lpLoadLibrary;
    BOOL    bRet = FALSE;

    if((hProc = OpenProcess(PROCESS_ALL_ACCESS, FALSE, dwPid)) != NULL)
    {
        dwMemSize = strlen(szDllPath);
        if((lpRemoteMem = VirtualAllocEx(hProc, NULL, dwMemSize, MEM_COMMIT, PAGE_READWRITE)) != NULL)
            if(WriteProcessMemory(hProc, lpRemoteMem, szDllPath, dwMemSize, NULL))
            {
                lpLoadLibrary = GetProcAddress(GetModuleHandleA("kernel32.dll"), "LoadLibraryA");
                if(CreateRemoteThread(hProc, NULL, 0, (LPTHREAD_START_ROUTINE)lpLoadLibrary, lpRemoteMem, 0, NULL) != NULL)
                {
                    bRet = TRUE;
                }
                cout << GetLastError();
            }
    }
    CloseHandle(hProc);

    return bRet;
}

int main()
{
    char    szProc[MAX_PATH], szDll[MAX_PATH];
    char*   szDllPath = (char*)malloc(MAX_PATH);
    LPTSTR  lpszProc = NULL;

    for(;;)
    {
        cout << "Process: ";
        cin >> szProc;
        cout << "DLL: ";
        cin >> szDll;

        szDllPath = GetCurrentDir();
        strcat_s(szDllPath, MAX_PATH, "\\");
        strcat_s(szDllPath, MAX_PATH, szDll);

        cout << "Waiting for process.. ." << szDllPath << " " << szDll << endl;
        WaitForProcessToAppear(SzToLPCTSTR(szProc), 100);
        if(InjectDll(GetProcessIdByName(SzToLPCTSTR(szProc)), szDllPath))
            cout << "Injection Succeeded!" << endl;
        else
            cout << "Injection Failed!" << endl;
        cout << "\n";

    }

    return 0;

经过大量的谷歌搜索后,我找不到这不应该工作的原因。

CreateRemoteThread 在 Windows 7 下不起作用吗?如果是这样,我是否犯了任何明显的错误?

4

6 回答 6

16

它失败的原因是因为您的代码是 32 位的,而您的目标进程是 64 位的。

您拥有多少特权并不重要。Windows 不会让这种情况发生。

我有同样的问题。要么生成一个系统 32 位 exe 并将其注入,要么将代码移植到 64 位(这意味着它不能在 32 位系统上运行)。

编辑

很久以前,我发现了一种将代码注入和从任何处理器模式目标中注入的好方法。它涉及将处理器模式动态切换到(任何)目标的模式。被称为“天堂之门”。为此,您必须使用内联汇编。所以基本上你可以在 32 位 exe 中同时拥有 64 位和 32 位代码,检测机器是否为 64 位,然后跳转到 64 位模式并运行 64 位代码。然后,您将遍历导入以找到 ntdll 并加载 64 位 kernel.dll 和其他库。这是任何有兴趣的人的示例链接:http: //bit.ly/19P0Lh3

于 2012-10-28T03:29:19.780 回答
3

我看到的直接问题是您没有获得应该这样做的访问令牌:

HANDLE hToken; 
TOKEN_PRIVILEGES tp; 
HANDLE hProcess = OpenProcess( PROCESS_ALL_ACCESS, FALSE, GetCurrentProcessId() ); 

tp.PrivilegeCount = 1; 
LookupPrivilegeValue( NULL, _T("SeDebugPrivilege"), &tp.Privileges[0].Luid ); 
tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED; 
OpenProcessToken( hProcess, TOKEN_ADJUST_PRIVILEGES, &hToken ); 

AdjustTokenPrivileges( hToken, FALSE, &tp, NULL, NULL, NULL ); 
CloseHandle( hToken );

我现在没有时间查看您的所有代码,但这是我从以前的一个项目中摘录的内容:

// returns open process handle
HANDLE InjectDLL( DWORD dwPID, LPCWSTR szDLLPath, HMODULE* lphInjected ) {
  int     cszDLL;
  LPVOID  lpAddress;
  HMODULE hMod;
  HANDLE  hThread;
  HANDLE  hProcess = OpenProcess( PROCESS_CREATE_THREAD | 
      PROCESS_QUERY_INFORMATION | PROCESS_VM_OPERATION |
      PROCESS_VM_WRITE | PROCESS_VM_READ, FALSE, dwPID );

  if( hProcess == NULL ) {
    return NULL;
  }

  cszDLL = ( wcslen( szDLLPath ) + 1 ) * sizeof( WCHAR );

  // Injection
  lpAddress = VirtualAllocEx( hProcess, NULL, cszDLL, MEM_COMMIT, PAGE_EXECUTE_READWRITE );
  if( lpAddress == NULL ) {
    return NULL;
  }

  WriteProcessMemory( hProcess, lpAddress, szDLLPath, cszDLL, NULL );

  hMod = GetModuleHandle( L"kernel32.dll" );
  if( hMod == NULL ) {
    return NULL;
  }

  hThread = CreateRemoteThread( hProcess, NULL, 0,
        (LPTHREAD_START_ROUTINE)( GetProcAddress( hMod,
        "LoadLibraryW" ) ), lpAddress, 0, NULL );

  // Locate address our payload was loaded
  if( hThread != 0 ) {
    WaitForSingleObject( hThread, INFINITE );
    GetExitCodeThread( hThread, ( LPDWORD )lphInjected );
    VirtualFreeEx( hProcess, lpAddress, 0, MEM_RELEASE );
    CloseHandle( hThread );
  }

  return hThread != 0 ? hProcess : NULL;
}

看看有没有帮助。以后再看。

于 2012-02-26T23:04:15.580 回答
2

好的,由于“受保护的进程”,您的代码可能会在 Windows 7 和 Vista 中失败,即只能由其他受保护进程(如 explorer.exe 等)操作的进程......在 Windows 7 x32 中有一个方式:因为您能够加载未签名的驱动程序,...好吧,您完成了(在 google 中搜索 Alex Ionescu)。但是,在 Windows 7 x64 中,您不能(呃!)

“CreateRemoteThread() 的第四个参数是地址。在您的情况下,它是 LoadLibraryA 地址。但是,在 Windows 7 中,Kernel32.dll/LoadLibraryA 基地址在不同的进程中会有所不同;”

好吧,这远非如此,因为尽管有 ASLR,但 DLL 在每个进程中都在相同的地址上共享。不过,DLL 可以重新定位,但您可以在调用 CreateRemoteThread 之前调用 GetProcAddress,因此 DLL 不太可能同时重新定位。

于 2012-05-07T16:36:27.410 回答
0

我认为 CreateRemoteThread() dll 注入方法不能在 Windows 7 中工作。

CreateRemoteThread() 的第四个参数是地址。在您的情况下,它是 LoadLibraryA 地址。但是,在windows 7中,Kernel32.dll/LoadLibrary的基地址在不同的进程中会有所不同;因此, CreateRemoteThread() 将不起作用,因为地址不是您所期望的。这是我个人的看法,希望对你有所帮助。:)

于 2012-04-19T15:39:05.757 回答
0

CreateRemoteThread 函数在 Win Vista/7 中不起作用。为此,您必须使用未记录的 NTCreateThread 函数。

于 2013-05-26T03:01:03.690 回答
0

TLDR:您的代码还可以,将 Visual Studio 调试/编译目标更改为 x64。

我以前也遇到过同样的问题,你的代码很好,问题是,Visual Studio(或任何理智的人)默认以 x86 模式(32 位)执行它的程序,因为在在 x86 或 x64 架构中运行但不在进程注入场景中的方式!因为它的系统调用。

在代码注入的情况下,您应该在项目资源管理器的项目属性中更改 VS 的构建和调试设置为 x64 处理器编译/调试,或者如果您交叉编译程序,则应该使用 x64 编译器。

如果您正在寻找全局进程注入方法,有一种称为Heaven's GateThe 0x33 Segment Selector的方法,用于Vawtrak银行恶意软件。

你可以看到这个关于天堂之门方法的链接,上面写着:

换句话说,它使人们能够创建“裸”的 64 位代码,这些代码将能够秘密运行,包括发出系统调用,而大多数产品都无法拦截和/或自省其执行

于 2020-02-03T13:45:11.423 回答