用 VC 获取其它运行中程序的命令行参数

用VC获取其它运行中程序的命令行参数

 

  我们都知道,在程序里获取命令行参数很简单,WinMain函数会以参数的形式传递给我们,或者可以调用API GetCommandLine 获取。但是GetCommandLine函数不接受参数,获取的只是自己程序的命令行参数。那么如果我们想获取别的应用程序的命令行参数应该怎么办呢?

  有的同学说,既然GetCommandLine只能获取本程序的命令行参数,我们可以在其它进程里插入一个Dll,在那个进程的地址空间调用GetCommandLine函数,然后传回来就可以了。这样好像有点儿不太友好。让我们想想还有没有别的办法。

  我们想,自己的命令行参数既然随时都可以获取到,那么在该进程里一定有一个地方存放它。那么在哪儿呢?看一下GetCommandLine函数的反汇编代码,我们发现,原来世界是如此的美好!

  以下是WinXP系统的GetCommandLine函数反汇编代码:.

  1. text:7C812C8D GetCommandLineA proc near
  2. .text:7C812C8D mov eax, dword_7C8835F4 //dword_7C8835F4 就是命令行参数字符串的地址 
  3. //该指令机器码为 A1 F4 35 88 7C,从第2个字节开始的4个字节就是我们要的地址
  4. .text:7C812C92 retn
  5. .text:7C812C92 GetCommandLineA endp 

  既然知道了放在哪儿了,我们自己去拿就可以了。因为GetCommandLine函数的地址在各个进程内都是一样的,所以可以直接用我们进程里的地址。 win2000/xp系统很简单,98下稍微麻烦一点儿,需要进行一些简单的计算。以下是GetCommandLine函数在win98下的汇编代码:

  1. .text:BFF8C907 GetCommandLineA proc near 
  2. .text:BFF8C907 mov eax, dword_BFFCADE4 
  3. .text:BFF8C90C mov ecx, [eax] 
  4. .text:BFF8C90E mov eax, [ecx+0C0h] 
  5. .text:BFF8C914 test eax, eax
  6. .text:BFF8C916 jnz short locret_BFF8C91E 
  7. .text:BFF8C918 mov eax, [ecx+40h] 
  8. .text:BFF8C91B mov eax, [eax+8] //算到这儿,才是我们想要的地址
  9. .text:BFF8C91E 
  10. .text:BFF8C91E locret_BFF8C91E: ; CODE XREF: GetCommandLineA+F. 
  11. .text:BFF8C91E retn 

  这样,我们就可以调用OpenProcess函数打开其它进程,然后用ReadProcessMemory读取相应的数据即可。 示例代码:

  1. DWORD g_GetCmdLine(DWORD dwPID,TCHAR* pCmdLine,DWORD dwBufLen) 
  2.  #define BUFFER_LEN 512 //reading buffer for the commandline
  3.  HANDLE hProc = OpenProcess(PROCESS_VM_READ,FALSE,dwPID); 
  4.  if(hProc == NULL) 
  5.  { 
  6.   return GetLastError(); 
  7.  } 
  8.  DWORD dwRet = -1; 
  9.  DWORD dwAddr = *(DWORD*)((DWORD)GetCommandLine + 1);//第2个字节开始才是我们要读的地址
  10.  TCHAR tcBuf[BUFFER_LEN] = ; 
  11.  DWORD dwRead = 0; 
  12.  //判断平台
  13.  DWORD dwVer = GetVersion(); 
  14.  try
  15.  { 
  16.   if(dwVer < 0x80000000) // Windows NT/2000/XP
  17.   { 
  18.    if(ReadProcessMemory(hProc,(LPVOID)dwAddr,&dwAddr,4,&dwRead)) 
  19.    { 
  20.     if(ReadProcessMemory(hProc,(LPVOID)dwAddr,tcBuf,BUFFER_LEN,&dwRead)) 
  21.     { 
  22.      _tcsncpy(pCmdLine,tcBuf,dwBufLen); //最好检查一下dwRead和dwBufLen的大小,使用较小的那个
  23.      dwRet = 0; 
  24.     } 
  25.    } 
  26.   } 
  27.   else // Windows 95/98/Me and Win32s
  28.   { 
  29.    while(true//使用while是为了出错时方便跳出循环
  30.    { 
  31.     if(!ReadProcessMemory(hProc,(LPVOID)dwAddr,&dwAddr,4,&dwRead)) break
  32.     if(!ReadProcessMemory(hProc,(LPVOID)dwAddr,&dwAddr,4,&dwRead)) break
  33.     if(!ReadProcessMemory(hProc,(LPVOID)(dwAddr + 0xC0),tcBuf,BUFFER_LEN,&dwRead)) break
  34.     if(*tcBuf == 0) 
  35.     { 
  36.      if(!ReadProcessMemory(hProc,(LPVOID)(dwAddr + 0x40),&dwAddr,4,&dwRead)) break
  37.      if(!ReadProcessMemory(hProc,(LPVOID)(dwAddr + 0x8),&dwAddr,4,&dwRead)) break
  38.      if(!ReadProcessMemory(hProc,(LPVOID)dwAddr,tcBuf,BUFFER_LEN,&dwRead)) break
  39.     } 
  40.     _tcsncpy(pCmdLine,tcBuf,dwBufLen); //最好检查一下dwRead和dwBufLen的大小,使用较小的那个
  41.     dwRet = 0; 
  42.     break
  43.    } 
  44.   } 
  45.  } 
  46.  catch(...) 
  47.  { 
  48.   dwRet = ERROR_INVALID_ACCESS; //exception
  49.  } 
  50.  CloseHandle(hProc); 
  51.  return dwRet; 

 Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=1833170

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值