ObCallback回调钩子检测
ObCallback回调钩子检测
2013-12-20 Nie.Meining Ring0
在 PatchGuard 的摧残下,通过 ObRegisterCallbacks 函数注册回调钩子已经成了 RK/ARK 中的主流技术之一。注册回调钩子的具体做法可以参考MSDN上的示例代码:http://code.msdn.microsoft.com/windowshardware/ObCallback-Sample-67a47841。
今天研究了一下检测这类回调钩子的方法,发出来跟大家分享。本人水平有限,错漏之处还请大家指正。
首先关注的还是注册回调钩子时调用的 ObRegisterCallbacks 函数,MSDN上的函数声明如下:
- NTSTATUS ObRegisterCallbacks(
- _In_ POB_CALLBACK_REGISTRATION CallBackRegistration,
- _Out_ PVOID *RegistrationHandle
- );
也就是说调用该函数后,我们会得到一个神奇的 RegistrationHandle 指针,而该指针指向的数据结构必然用于描述本次注册的回调钩子(否则 ObUnRegisterCallbacks 就摘不掉钩子了)。
经过一番逆向和调试,发现这个数据结构由三个部分顺序组织在内存中:
-----------------------------------------------------------------------------------------------------
Part1:
钩子数据结构的头部信息,如版本号,CallbackOperator 个数等
Part2:(这个部分的长度取决于该回调钩子包含多少个CallbackOperator)
钩子数据结构的CallbackOperator1 相关信息
钩子数据结构的CallbackOperator2 相关信息
……
Part3:
钩子数据结构的额外信息,如 Altitude 等
-----------------------------------------------------------------------------------------------------
具体来讲,用 C 语言可以描述如下:
- struct CALLBACK_NODE {
- // 版本号,目前是 0x100, 可通过 ObGetFilterVersion 获取该值
- USHORT usVersion;
- // 本节点上有多少个 CallbackOperator,即多少个 CallbackBodies
- USHORT usCallbackBodyCount;
- // 注册回调时设定的 OB_CALLBACK_REGISTRATION.RegistrationContext
- // 即 PreCallbackRoutine 和 PostCallbackRoutine 的参数
- PVOID pContext;
- // 这里似乎是 0x000C000C/0x00080008 一类的值
- ULONG ulXXX_000X000X;
- // 指向 Altitude 字符串
- // 注意这个字符串因为最大长度是恒定的,所以不一定以 NULL 结尾
- PWCHAR wcsAltitude;
- // CALLBACK_BODY 数组,其元素个数为 ulCallbackCount
- struct CALLBACK_BODY CallbackBodies[1];
- //
- // CALLBACK_BODY数组后,紧接着 Altitude 字符串和一些其它数据
- //
- };
其中的 struct CALLBACK_BODY CallbackBodies[] 数组就是描述该 CallbackNode 上的各个 CallbackOperator 的数据结构:
- struct CALLBACK_BODY {
- // 所有的 CALLBACK_NODE 通过这个链表串在一起
- LIST_ENTRY ListEntry;
- // 注册回调时设定的 OB_OPERATION_REGISTRATION.Operations,
- // 即 OB_OPERATION_HANDLE_CREATE 这些东西
- ULONG ulOperations;
- // 这里似乎始终是 1
- ULONG ulXXX_1;
- // 指向该 CallbackBody 对应的 CallbackNode
- struct CALLBACK_NODE* pCallbackNode;
- // 注册回调时设定的 OB_OPERATION_REGISTRATION.pObjectType
- // 即 PsProcessType,PsThreadType 这些东西
- PVOID pObjectType;
- // 这就是注册回调时设定的 PreCallbackRoutine 函数地址
- PVOID pPreCallbackRoutine;
- // 这就是注册回调时设定的 PostCallbackRoutine 函数地址
- PVOID pPostCallbackRoutine;
- // 这里似乎是个引用计数
- ULONG ulRefCount;
- };
注意这里面的 CALLBACK_BODY.ListEntry 链表很有意思,它将系统中的其它 CALLBACK_NODE.CallbackBodies 都链在了一起。因此我们就可以通过遍历这个链表检测系统中的其它回调钩子。
检测的大致思路是,先自己调用ObRegisterCallbacks注册一个假钩子,这样就得到了一个节点,然后用这个节点开始遍历。
写了一个测试代码,在64位Win8系统上测试截图如下:
图中 CallbackDetect.sys 中钩子是我的检测模块注册的假钩子;bd0001.sys 中的钩子是百度卫士注册的钩子;其它是360注册的钩子。
完整测试代码下载(64位/32位系统通用):CallbackDetect.zip
jpg改rar
ObCallback回调钩子检测的更多相关文章
- 电信NBIOT平台的CA证书上传-消息订阅回调地址检测503错误
在NBIOT北向开发过程中,遇到消息订阅回调地址检测503错误,经过论坛查询与文档查阅一直都没有解决问题,大多人都说是RESTful地址格式问题,但其实不是.最终发现是我们在电信平台创建应用时,上传C ...
- js回调函数,检测这个值是否重复
//校验提交的数据是否重复 /** * url:后端的查询地址 * filedVal: 要传到后台的值 * ele:要绑定显示的元素,一般就是当前的input就可以,直接在其后边追加显示 * fn:回 ...
- Spring-IOC bean 生命周期之 Lifecycle 钩子
Lifecycle callbacks Initialization callbacks.Destruction callbacks 要与容器的bean生命周期管理交互,即容器在启动后和容器在销毁前对 ...
- 常见注入手法第四讲,SetWindowsHookEx全局钩子注入.以及注入QQ32位实战.
常见注入手法第四讲,SetWindowsHookEx全局钩子注入.以及注入QQ32位实战. PS:上面是操作.最后是原理 一丶需要了解的API 使用全局钩子注入.我们需要了解几个WindowsAPI. ...
- 通过TLS回调函数的反调试
下面是TLS数据结构的定义 typedef struct _IMAGE_TLS_DIRECTORY { DWORD StartAddressOfRawData; DWORD EndAddressOfR ...
- Java并发编程-扩展可回调的Future
前提 最近在看JUC线程池java.util.concurrent.ThreadPoolExecutor的源码实现,其中了解到java.util.concurrent.Future的实现原理.从目前j ...
- Deformity PHP Webshell、Webshell Hidden Learning
目录 . 引言 . webshell原理介绍 . webshell的常见类型以及变种方法 . webshell的检测原理以及检测工具 . webshell隐藏反检测对抗手段 0. 引言 本文旨在研究W ...
- PowerTool(杀毒辅助工具) V4.6 中文免费绿色版
软件名称: PowerTool(杀毒辅助工具)软件语言: 简体中文授权方式: 免费软件运行环境: Win7 / Vista / Win2003 / WinXP 软件大小: 968KB图片预览: 软件简 ...
- Spring事务源码阅读笔记
1. 背景 本文主要介绍Spring声明式事务的实现原理及源码.对一些工作中的案例与事务源码中的参数进行总结. 2. 基本概念 2.1 基本名词解释 名词 概念 PlatformTransaction ...
随机推荐
- mysql备份与还原
一.直接拷贝数据库文件 直接拷贝数据库文件一般是使用文件系统备份工具cp,适合小型数据库,是最可靠的. 当你拷贝数据库文件时,必须保证表没有正在使用.如果服务器在你拷贝一个表的时候改变这个表,拷贝就失 ...
- Linux下安装gcc和g++
以CentOS为例,安装后是没有C语言和C++编译环境的,需要手动安装,最简单的是用yum的方式安装,过程如下: 1.安装gcc yum install gcc 询问是否,按y键回车即可,或者 yum ...
- uva 489.Hangman Judge 解题报告
题目链接:https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem ...
- springmvc 文件下传、上载、预览。以二进制形式存放到数据库(转载)
springmvc 文件上传.下载.预览.以二进制形式存放到数据库.数据库中的关于传入附件的字段我写了2个:一个存放内容accessory,一个存放文件的后缀filetype 上传:首先需要2个必须的 ...
- 最牛逼android上的图表库MpChart(一) 介绍篇
最牛逼android上的图表库MpChart一 介绍篇 MpChart优点 MpChart是什么 MpChart支持哪些图表 MpChart效果如何 最牛逼android上的图表库MpChart(一) ...
- 【leetcode】Surrounded Regions(middle)☆
Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A region is captured ...
- POJ 1753 Flip game ( 高斯消元枚举自由变量)
题目链接 题意:给定一个4*4的矩阵,有两种颜色,每次反转一个颜色会反转他自身以及上下左右的颜色,问把他们全变成一种颜色的最少步数. 题解:4*4的矩阵打表可知一共有四个自由变元,枚举变元求最小解即可 ...
- IOS- Run Loops
Run Loops Run loops是线程相关的的基础框架的一部分.一个run loop就是一个事件处理的循环,用来不停的调度工作以及处理输入事件.使用run loop的目的是让你的线程在有工作的时 ...
- HTML标记之a标签
一.a标签的语法 <a href=”链接目标地址” title=”注释” target=”打开链接窗口的形式”>链接显示内容</a> target值: _blank在新窗口中打 ...
- Javascript 封装方法
基本封装方法 请看下面的例子: var Person = function(name,age){ this.name = name; this.age = age || "未填写" ...