CreateRemoteThread和WriteProcessMemory技术

CreateRemoteThread和WriteProcessMemory技术CreateRemoteThread和WriteProcessMemory技术示例程序:WinSpy另一种注入代码到其他进程地址空间的方法是使用WriteProcessMemoryAPI。这次你不用编写一个独立的DLL而是直接复制你的代码到远程进程(WriteProcessMemory)并用CreateRemoteThread执行之。让我们看一下CreateRemoteThread

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全家桶1年46,售后保障稳定

CreateRemoteThread和WriteProcessMemory技术

示例程序:WinSpy

另一种注入代码到其他进程地址空间的方法是使用WriteProcessMemory API。这次你不用编写一个独立的DLL而是直接复制你的代码到远程进程(WriteProcessMemory)并用CreateRemoteThread执行之。

让我们看一下CreateRemoteThread的声明:

HANDLE CreateRemoteThread( 
   HANDLE hProcess,         

//
 handle to process to create thread in 


   LPSECURITY_ATTRIBUTES lpThreadAttributes,   
//
 pointer to security
                                              

//
 attributes 


   DWORD dwStackSize,       
//
 initial thread stack size, in bytes 


   LPTHREAD_START_ROUTINE lpStartAddress,      
//
 pointer to thread 
                                              

//
 function 


   LPVOID lpParameter,      
//
 argument for new thread 


   DWORD dwCreationFlags,   
//
 creation flags 


   LPDWORD lpThreadId       
//
 pointer to returned thread identifier 


);

和CreateThread相比,有一下不同:
●增加了hProcess参数。这是要在其中创建线程的进程的句柄。
●CreateRemoteThread的lpStartAddress参数必须指向远程进程的地址空间中的函数。这个函数必须存在于远程进程中,所以我们不能简单地传递一个本地ThreadFucn的地址,我们必须把代码复制到远程进程。
●同样,lpParameter参数指向的数据也必须存在于远程进程中,我们也必须复制它。

现在,我们总结一下使用该技术的步骤:
1.   得到远程进程的HANDLE(OpenProcess)。
2.   在远程进程中为要注入的数据分配内存(VirtualAllocEx)、
3.   把初始化后的INJDATA结构复制到分配的内存中(WriteProcessMemory)。
4.   在远程进程中为要注入的数据分配内存(VirtualAllocEx)。
5.   把ThreadFunc复制到分配的内存中(WriteProcessMemory)。
6.   用CreateRemoteThread启动远程的ThreadFunc。
7.   等待远程线程的结束(WaitForSingleObject)。
8.   从远程进程取回指执行结果(ReadProcessMemory 或 GetExitCodeThread)。
9.   释放第2、4步分配的内存(VirtualFreeEx)。
10.   关闭第6、1步打开打开的句柄。

另外,编写ThreadFunc时必须遵守以下规则:
1.   ThreadFunc不能调用除kernel32.dll和user32.dll之外动态库中的API函数。只有kernel32.dll和user32.dll(如果被加载)可以保证在本地和目的进程中的加载地址是一样的。(注意:user32并不一定被所有的Win32进程加载!)参考附录A。如果你需要调用其他库中的函数,在注入的代码中使用LoadLibrary和GetProcessAddress强制加载。如果由于某种原因,你需要的动态库已经被映射进了目的进程,你也可以使用GetMoudleHandle代替LoadLibrary。同样,如果你想在ThreadFunc中调用你自己的函数,那么就分别复制这些函数到远程进程并通过INJDATA把地址提供给ThreadFunc。

2.   不要使用static字符串。把所有的字符串提供INJDATA传递。为什么?编译器会把所有的静态字符串放在可执行文件的“.data”段,而仅仅在代码中保留它们的引用(即指针)。这样,远程进程中的ThreadFunc就会执行不存在的内存数据(至少没有在它自己的内存空间中)。

3.   去掉编译器的/GZ编译选项。这个选项是默认的(看附录B)。
4.   要么把ThreadFunc和AfterThreadFunc声明为static,要么关闭编译器的“增量连接(incremental linking)”(看附录C)。
5.   ThreadFunc中的局部变量总大小必须小于4k字节(看附录D)。注意,当degug编译时,这4k中大约有10个字节会被事先占用。
6.   如果有多于3tch分支的case语句,必须像下面这样分割开,或用if-else if代替.

     case constant1: statement1; goto END;
     case constant2: statement2; goto END;
     case constant3: statement2; goto END;
}

     case constant4: statement4; goto END;
     case constant5: statement5; goto END;
     case constant6: statement6; goto END;
}
END:

 

=====简单的CreateRemoteThread例程-初学者必看

 

//
 _remotethreaddemo.cpp : Defines the entry point for the console application.

//
 Author:秋镇菜



#include 


stdafx.h


#include 


windows.h


//
 ========== 定义一个代码结构,本例为一个对话框============


struct
 MyData
{

 

char
 sz[
64
]; 
//
 对话框显示内容


 DWORD dwMessageBox; 
//
 对话框的地址


};


//
 ========== 远程线程的函数 ==============================


DWORD __stdcall RMTFunc(MyData 
*
pData)
{

 typedef 

int
(__stdcall
*
MMessageBox)(HWND,LPCTSTR,LPCTSTR,UINT);
 MMessageBox MsgBox 

=
 (MMessageBox)pData
->
dwMessageBox;
 MsgBox(NULL, pData

->
sz, NULL, MB_OK);
 

return
 
0
;
}

int
 main(
int
 argc, 
char
*
 argv[])
{


//
 ===== 获得需要创建REMOTETHREAD的进程句柄 ===============================


 HWND hWnd 
=
 FindWindow(

notepad

, NULL); 
//
 以NOTEPAD为例


 DWORD dwProcessId;
 ::GetWindowThreadProcessId(hWnd, 

&
dwProcessId);
 HANDLE hProcess 

=
 OpenProcess(
        PROCESS_ALL_ACCESS,
        FALSE,
        dwProcessId);


//
 ========= 代码结构 ================================================


 MyData data;
 ZeroMemory(

&
data, 
sizeof
 (MyData));
 strcat(data.sz, 


对话框的内容.

);
 HINSTANCE hUser 

=
 LoadLibrary(

user32.dll

);
 

if
 (
!
 hUser)
 {

  printf(


Can not load library.

);
  

return
 
0
;
 }
 data.dwMessageBox 

=
 (DWORD)GetProcAddress(hUser, 

MessageBoxA

);
 FreeLibrary(hUser);
 

if
 (
!
 data.dwMessageBox)
  

return
 
0
;


//
 ======= 分配空间 ===================================================


 
void
 
*
pRemoteThread
  

=
 VirtualAllocEx(hProcess, 
0
,
      

1024
*
4
, MEM_COMMIT
|
MEM_RESERVE,
      PAGE_EXECUTE_READWRITE);
 

if
 (
!
 pRemoteThread)
  

return
 
0
;
 

if
 (
!
 WriteProcessMemory(hProcess, pRemoteThread, 
&
RMTFunc, 
1024
*
4

0
))
  

return
 
0
;

 MyData 
*
pData
  

=
 (MyData
*
)VirtualAllocEx(hProcess, 
0
,
      

sizeof
 (MyData), MEM_COMMIT,
      PAGE_READWRITE);
 

if
 (
!
pData)
  

return
 
0
;

 
if
 (
!
 WriteProcessMemory(hProcess, pData, 
&
data, 
sizeof
 (MyData), 
0
))
  

return
 
0
;


//
 =========== 创建远程线程 ===========================================


 HANDLE hThread
  

=
 CreateRemoteThread(hProcess, 
0
,
       

0
, (LPTHREAD_START_ROUTINE)pRemoteThread,
       pData, 

0

0
);
 

if
 (
!
 hThread)
 {

  printf(


远程线程创建失败

);
  

return
 
0
;
 }
 CloseHandle(hThread);
 VirtualFreeEx(hProcess, pRemoteThread, 

1024
*
3
, MEM_RELEASE);
 VirtualFreeEx(hProcess, pData, 

sizeof
 (MyData), MEM_RELEASE);
 CloseHandle(hProcess);
 printf(


Hello World!

);
 

return
 
0
;
}

zhuangyu1982@hotmail.com: 我用你的程序在windows xp下运行, 弹出messagebox之后, 只要点击确定宿主进程就会崩溃。 
而且不止是messagebox,调用其它的api函数也是一样,请问这是什么原因?有什么办法可以解决吗? 

——————————————————–
 

编译成RELEASE版本就不会出错了,主要是DEBUG版本加了一个__chkesp的函数导致调用了非法地址 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/226584.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Java实现并查集

    Java实现并查集喝杯 82 年的 Java 压压惊这次需要介绍的就是并查集并查集的简单应用就是连通图 网络通信连接等等总之很重要那么先说一下这次的算法是 1 union find 简单并查集 2 quick union 优化的并查集 3 加权值 quick union 处理了 2 的最坏情况 4 路径压缩加权值 quick union 如果只是想要一下算法 你可以直接跳到最后看第 4 个算法接下来 我

    2025年6月1日
    5
  • 编码器计数原理与电机测速原理——多图解析[通俗易懂]

    编码器计数原理与电机测速原理——多图解析[通俗易懂]编码器,是一种用来测量机械旋转或位移的传感器。它能够测量机械部件在旋转或直线运动时的位移位置或速度等信息,并将其转换成一系列电信号。编码器分类按监测原理分类光电编码器光电编码器,是一种通过光电转换将输出轴上的机械几何位移量转换成脉冲或数字量的传感器。这是目前应用最多的传感器,光电编码器是由光源、光码盘和光敏元件组成。光栅盘是在一定直径的圆板上等分地开通若干个长方形孔。由于光电码盘与电动机同轴,电动机旋转时,光栅盘与电动机同速旋转,经发光二极管等电子元件组成的检测装置检测输出若干脉冲信号,通过计算每

    2022年10月1日
    2
  • python中导入numpy为什么错误_pycharm安装配置教程

    python中导入numpy为什么错误_pycharm安装配置教程今天网上复制了一个代码,其中有个importnumpyasnp,运行时提示需要安装numpy库,然后我按照网上的方法,按顺序点击File–>Settings–>Project:pythonProject–>PythonInterpreter,然后找到+那里准备添加库,如下:然后就报erroroccurredwheninstallingpackage”numpy”的错误,搞了半天都没搞定,遂找了一个经……

    2022年8月29日
    2
  • clover默认引导mac(clover win10引导)

    搞定Clover引导的Win&Mac双系统系统迁移至SSD作者:毛毛卷日期:2018-07-20字体大小:小中大从发完贴到现在总算搞定了,具体操作记录如下:由于当年折腾双系统的时候就经历了很多波折而且一般是默认启动MAC而我却是WIN10所以本身的要求和实现方法就有点特殊因此最终并没有偷懒用分区克隆的方法还是按部就班的进行首先把自己提的几个问题回答一下吧首先大概试了A…

    2022年4月11日
    38
  • pytest-allure_pytest数据驱动

    pytest-allure_pytest数据驱动前言allure是一个report框架,支持java的Junit/testng等框架,当然也可以支持python的pytest框架,也可以集成到Jenkins上展示高大上的报告界面。mac环境:

    2022年7月28日
    11
  • windows10查看端口被占用_win10端口被占用怎么解决

    windows10查看端口被占用_win10端口被占用怎么解决1、查看windows所有端口进程同时按下Win+R,接着在对话框中输入“cmd”,按下回车键打开命令提示符窗口,然后在窗口中输入netstat-ano按下回车键,之后就会显示所有的端口占用情况。2、如果是要查询指定的端口占用情况的话,可以在命令提示符窗口中继续输入netstat-aon|findstr”提示的端口”这里的提示的端口假设为8080,那么就输入命令netstat-aon|findstr”8080″按下回车键之后就可以看见列表中的PID然后.

    2022年7月27日
    19

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号