DuplicateHandle文档化解释

  The DuplicateHandle function duplicates an object handle. The duplicate handle refers to the same object as the original handle. Therefore, any changes to the object are reflected through both handles. For example, the current file mark for a file handle is always the same for both handles.
使用 DuplicateHandle 函数将源进程 ProcessHandle 的句柄表中的信息如对象句柄 MutexHandle 复制到当前目标进程的句柄表中

BOOL DuplicateHandle(
  HANDLE hSourceProcessHandle, // handle to source process 源进程句柄
  HANDLE hSourceHandle, // handle to duplicate 被拷贝的信息(应与源进程句柄的进程有关)
  HANDLE hTargetProcessHandle, // handle to target process 目标进程句柄
  LPHANDLE lpTargetHandle, // duplicate handle 用来接收被拷贝的信息_Out_
  DWORD dwDesiredAccess, // requested access 传0
  BOOL bInheritHandle, // handle inheritance option 处理继承选项传FALSE或者TURE
  DWORD dwOptions // optional actions 如DUPLICATE_SAME_ACCESS,让目标进程对被拷贝的句柄的内核对象也有访问权限
);

一、参数说明:

  调用DuplicateHandle时,它的第一个参数和第三个参数 (hSourceProcessHandle 和 hTargetProcessHandle) 是进程内核对象句柄。这两个句柄本身必须相对于调用DuplicateHandle函数的那个进程。此外,这两个参数标识的必须是进程内核对象;如果我们传递的句柄指向的是其他类型的内核对象,函数调用就会失败。我们将在第4章详细讨论进程内核对象。就目前来说,我们只需知道一旦启动一个新的进程,系统就会创建一个进程内核对象。

  第二个参数 hSourceHandle 是指向任何类型的内核对象的一个句柄。但是,它的句柄值一定不能与调用 DuplicateHandle 函数的那个进程相关(不能指向目标进程 TargetProcess 的内核对象)。相反,该句柄必须与 hSourceProcessHandle 句柄所标识的源进程相关(例如源进程创建了文件,返回了文件对象句柄)。函数会将源进程中的指定的句柄信息(FileHandle)复制到 hTargetProcessHanle 所标识的目标进程的句柄表中。

  第四个参数是 phTargetHandle ,它是一个 HANDLE 变量的地址,用来接收复制得到的 HANDLE 值。

二、参考使用:

[B进程]

参考使用1:将伪句柄转换成真实句柄
HANDLE PseudoProcessHandle = GetCurrentProcess(); //获得伪句柄 -1
HANDLE RealProcessHandle = NULL; //将伪句柄转换成真实句柄
DuplicateHandle(GetCurrentProcess(), PseudoProcessHandle, GetCurrentProcess(), &RealProcessHandle,
0, FALSE, DUPLICATE_SAME_ACCESS);
//通过上面的函数就可以将一个进程中的一个线程的伪句柄转换成正真的句柄
//DuplicateHandle递增了内核对象的句柄数最好要调用CloseHandle();
参考使用2:拷贝当前进程中的指定对象的句柄到目标进程中
HANDLE v1 = NULL;
//DuplicateHandle拷贝当前进程中的FileHandle句柄到目标进程中
BOOL IsOk = DuplicateHandle(GetCurrentProcess(), FileHandle, ProcessHandle, &v1, 0, FALSE, DUPLICATE_SAME_ACCESS);
//当前进程句柄ProcessHandle
//被拷贝的文件句柄(当前进程里面创建的文件句柄)
//拷贝到目标进程的句柄
//v1也是句柄,用来接收目标进程上拷贝上的句柄的数
//DUPLICATE_SAME_ACCESS,让目标进程对被拷贝的句柄的内核对象也有访问权限
参考使用3:拷贝目标进程句柄表中指定句柄到当前进程句柄表中
    //如何知道进程
_tscanf(_T("%d"), &ProcessIdentify); // 目标进程id
_tscanf(_T("%d"), &MappingHandle); // 目标进程句柄表下的MappingHandle //根据目标进程的ID打开目标进程
ProcessHandle = SunOpenProcess(PROCESS_DUP_HANDLE, FALSE, ProcessIdentify);
//Anti策略
//DuplicateHandle获取进程虚拟空间地址
BOOL IsOk = DuplicateHandle(ProcessHandle, MappingHandle, GetCurrentProcess(), &v1, 0, FALSE, DUPLICATE_SAME_ACCESS);

三、其他说明

  使用DuplicateHandle函数(来复制内核对象句柄)所遇到的问题和继承(内核对象句柄)时同样:目标进程不知道它现在能访问一个新的内核对象。所以,进程C必须以某种方式来通1进程T,告诉它现在可以访问一个内核对象了,而且必须使用某种形式的进程间通信机制,将hObj中的句柄值传给进程T,显然,使用命令行参数或者更改进程T的环境变量是行不通的,因为进程已经启动并开始运行了。我们必须使用窗口消息或者其他进程间通信(IPC)机制。

四、演示程序

  将当前进程EProcess里的FileHandle的句柄拷贝到另一个进程explore.exe身上,并使其拥有对被拷贝句柄的内核对象的访问权限。

  代码测试:

  重新生成文件,将在文件目录下生成一个ReadMe文件,打开文件,再运行exe程序,发现无法删除此文件。explorer进程不关闭,文件句柄关不掉,FIleObject就关不掉,所以无法关闭记事本。
在代码里就是把第二参数句柄粘贴到第三参数身上,就相当于你把自己的句柄拷贝到了别人身上了,所以就存在占坑现象,第二参数句柄指向的内核对象关闭不了,其对应的进程无法关闭。

#include "DuplicateHandle.h"

BOOL __EnableDebugPrivilege = TRUE;

//定义参数
void _tmain(int argc, TCHAR* argv[], TCHAR *envp[])
{
Sub_1();
return;
} void Sub_1()
{ HANDLE ProcessHandle = NULL; //打开一个进程,得到进程句柄
ProcessHandle = SunOpenProcess(PROCESS_DUP_HANDLE, FALSE, (HANDLE)10688); //explore.exe进程
//SunOpenProcess中需要提权函数SunEnableSunDebugPrivilege if (ProcessHandle == NULL)
{
goto Exit;
} HANDLE FileHandle = INVALID_HANDLE_VALUE;
//在当前进程中创建一个文件,返回文件句柄
FileHandle = CreateFile(_T("ReadMe.txt"), GENERIC_READ, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); if (FileHandle == INVALID_HANDLE_VALUE)
{
goto Exit;
} HANDLE v1 = NULL;
//DuplicateHandle拷贝当前进程中的FileHandle句柄到目标进程中
BOOL IsOk = DuplicateHandle(GetCurrentProcess(), FileHandle, ProcessHandle, &v1, 0,
FALSE, DUPLICATE_SAME_ACCESS);
//当前进程句柄ProcessHandle
//被拷贝的文件句柄(当前进程里面创建的文件句柄)
//拷贝到目标进程的句柄
//v1也是句柄,用来接收目标进程上拷贝上的句柄的数
//DUPLICATE_SAME_ACCESS,让目标进程对被拷贝的句柄的内核对象也有访问权限
if (FileHandle != INVALID_HANDLE_VALUE)
{
SunCloseHandle(FileHandle);
FileHandle = INVALID_HANDLE_VALUE;
}
Exit:
if (ProcessHandle != NULL)
{ SunCloseHandle(ProcessHandle);
ProcessHandle = NULL;
}
} /*提权函数*/
DWORD SunEnableSunDebugPrivilege(HANDLE ProcessHandle, BOOL IsEnable)
{
DWORD LastError;
HANDLE TokenHandle = 0; if (!OpenProcessToken(ProcessHandle, TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &TokenHandle))
{
LastError = GetLastError();
if (TokenHandle)
SunCloseHandle(TokenHandle);
return LastError;
}
TOKEN_PRIVILEGES TokenPrivileges;
memset(&TokenPrivileges, 0, sizeof(TOKEN_PRIVILEGES));
LUID v1;
if (!LookupPrivilegeValue(NULL, SE_DEBUG_NAME, &v1))
{
LastError = GetLastError();
SunCloseHandle(TokenHandle);
return LastError;
}
TokenPrivileges.PrivilegeCount = 1;
TokenPrivileges.Privileges[0].Luid = v1;
if (IsEnable)
TokenPrivileges.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
else
TokenPrivileges.Privileges[0].Attributes = 0;
AdjustTokenPrivileges(TokenHandle, FALSE, &TokenPrivileges, sizeof(TOKEN_PRIVILEGES), NULL, NULL);
LastError = GetLastError();
SunCloseHandle(TokenHandle);
return LastError;
} HANDLE SunOpenProcess(DWORD DesiredAccess, BOOL IsInheritHandle, HANDLE ProcessIdentify)
{ //提权
if (__EnableDebugPrivilege) //全局变量__EnableDebugPrivilege
{
SunEnableSunDebugPrivilege(GetCurrentProcess(), TRUE);
} //打开进程,根据目标进程的进程ID得到进程句柄
HANDLE ProcessHandle = OpenProcess(DesiredAccess, IsInheritHandle, (DWORD)ProcessIdentify); DWORD LastError = GetLastError(); //关闭权限
if (__EnableDebugPrivilege)
{
SunEnableSunDebugPrivilege(GetCurrentProcess(), FALSE);
}
SetLastError(LastError);
return ProcessHandle;
} BOOL SunCloseHandle(HANDLE HandleValue)
{
DWORD HandleFlags;
if (GetHandleInformation(HandleValue, &HandleFlags)
&& (HandleFlags & HANDLE_FLAG_PROTECT_FROM_CLOSE) != HANDLE_FLAG_PROTECT_FROM_CLOSE)
return !!CloseHandle(HandleValue);
return FALSE;
}

复制对象句柄DuplicateHandle(文件占坑)的更多相关文章

  1. 速战速决 (5) - PHP: 动态地创建属性和方法, 对象的复制, 对象的比较, 加载指定的文件, 自动加载类文件, 命名空间

    [源码下载] 速战速决 (5) - PHP: 动态地创建属性和方法, 对象的复制, 对象的比较, 加载指定的文件, 自动加载类文件, 命名空间 作者:webabcd 介绍速战速决 之 PHP 动态地创 ...

  2. 插件开发之360 DroidPlugin源码分析(四)Activity预注册占坑

    请尊重分享成果,转载请注明出处: http://blog.csdn.net/hejjunlin/article/details/52258434 在了解系统的activity,service,broa ...

  3. C#对象与XMl文件之间的相互转换

    C#提供三种序列化方式,分别为: 1.是使用BinaryFormatter进行串行化: 2.使用SoapFormatter进行串行化: 3.使用XmlSerializer进行串行化.其中对于Binar ...

  4. windows内核对象句柄

    内核对象用于管理进程.线程和文件等诸多种类的大量资源,每一个内核对象都只是一个句内存快,它由操作系统内核分配,并只能右操作系统内核访问.这个内存块是一个数据结构,其维护着与对象相关的信息,其中少数成员 ...

  5. C#压缩文件夹坑~

    dotNet疯狂之路No.29  今天很残酷,明天更残酷,后天很美好,但是绝大部分人是死在明天晚上,只有那些真正的英雄才能见到后天的太阳.  We're here to put a dent in t ...

  6. 插件开发之360 DroidPlugin源码分析(五)Service预注册占坑

    请尊重分享成果,转载请注明出处: http://blog.csdn.net/hejjunlin/article/details/52264977 在了解系统的activity,service,broa ...

  7. 插件占坑,四大组件动态注册前奏(三) 系统BroadCast的注册发送流程

    转载请注明出处:http://blog.csdn.net/hejjunlin/article/details/52204143 前言:为什么要了解系统Activity,Service,BroadCas ...

  8. 插件占坑,四大组件动态注册前奏(二) 系统Service的启动流程

    转载请注明出处:http://blog.csdn.net/hejjunlin/article/details/52203903 前言:为什么要了解系统Activity,Service,BroadCas ...

  9. 插件占坑,四大组件动态注册前奏(一) 系统Activity的启动流程

    转载请注明出处:http://blog.csdn.net/hejjunlin/article/details/52190050 前言:为什么要了解系统Activity,Service,,BroadCa ...

  10. java:利用java的输入/输出流将一个文件的每一行+行号复制到一个新文件中去

    import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.File; import java.io.Fi ...

随机推荐

  1. S3C2440移植linux3.4.2内核之内核裁剪

    上一节S3C2440移植linux3.4.2内核之支持YAFFS文件系统我们修改了内核支持了yaffs2文件系统,这节我们裁剪内核. 目录 为什么要裁剪内核? 首先裁剪内核里无关的CPU/单板文件 裁 ...

  2. vue+elementUI+WebSocket接收后台实时消息推送

    vue+elementUI+WebSocket接收后台实时消息推送 https://blog.csdn.net/weixin_40888956/article/details/105971432?ut ...

  3. vue prop 会接收不同的数据类型

    refAge: { type: Number, default: 0 }, refName: { type: String, default: '' }, hotDataLoading: { type ...

  4. Oceanbase部分参数学习与验证

    Oceanbase部分参数学习与验证 字符集等参数查看 yum install obclient -y 然后使用客户端连接: obclient -h172.24.110.175 -P2881 -uro ...

  5. K8S Only IPV6的创建过程之二 完整版

    K8S Only IPV6的创建过程之二 完整版 整体过程 1. 阿里云新增一台虚拟机, 开通IPV6.与数据库,redis实现物理隔离. 并且进行环境基本处理,安装kubeadm等组件. 2. ku ...

  6. [转帖]HAProxy 在 TiDB 中的最佳实践

    https://docs.pingcap.com/zh/tidb/stable/haproxy-best-practices 本文介绍 HAProxy 在 TiDB 中的最佳配置和使用方法.HAPro ...

  7. [转帖]tiup cluster reload

    https://docs.pingcap.com/zh/tidb/stable/tiup-component-cluster-reload 4 Contributors 在修改集群配置之后,需要通过  ...

  8. [转帖]我们为什么放弃 MongoDB 和 MySQL,选择 TiDB

    https://zhuanlan.zhihu.com/p/164706527 写在前面的话 技术选型是由技术方向和业务场景 trade-off 决定的,脱离业务场景来说技术选型是没有任何意义的,所以本 ...

  9. 虚拟化平台IO劣化分析

    虚拟化平台IO劣化分析 背景 最近同事让帮忙做几个虚拟机进行性能测试. 本来应该搭建CentOS/Winodws平台进行相关的测试工作. 但是为了环境一致性, 使用了ESXi6.7 进行虚拟化 然后这 ...

  10. kafka的学习之一_带SASL鉴权的集群安装与启动

    kafka的学习之一_带SASL鉴权的集群安装与启动 背景 想开始一段新的里程. 可能会比现在累, 可能会需要更多的学习和努力. kafka可能就是其中之一. 自己之前总是畏缩不前. 不想面对很多压力 ...