PHP HOOK的若干方法

2024-06-16 03:32
文章标签 方法 php 若干 hook

本文主要是介绍PHP HOOK的若干方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

0x00 针对OP进行hook

在PHP内核中,每一个OP操作都是由一个固定的Handler函数去负责的,看_zend_op的结构体属性第一个就是opcode_handler_t,表示该OP对应的handler函数具体是哪个。

主要是调用zend_set_user_opcode_handler,将对应的ZEND OP的handler函数替换成自己定义的函数实现HOOK机制。

比如我们对ZEND_ECHO这条OP指令进行HOOK,即调用如下语句:

zend_set_user_opcode_handler(ZEND_ECHO, hook_echo_handler);

这时候当PHP执行ZEND_ECHO指令(即调用echo输出)的时候就会先调用我们的handler函数,这样就实现了HOOK。

HOOK之后,需要从opline里获取到参数,大致思路是,根据HOOK的OP指令的不同,获取op1或者op2,然后根据op1_type或者op2_type分情况抽取参数值:

(1)    IS_TMP_VAR

如果op的类型为临时变量,则调用get_zval_ptr_tmp获取参数值。

(2)    IS_VAR

如果是变量类型,则直接从opline->var.ptr里获取

(3)    IS_CV

如果是编译变量参考ZEND_ECHO_SPEC_CV_HANDLER中的处理方式,是直接从EG(active_symbol_table)中寻找。

(4)IS_CONST

如果op类型是常量,则直接获取opline->op1.zv即可。

上述方法都是从PHP源码中选取的,比如一个ZEND_ECHO指令的Handler会有多个,分别处理不同类型的op,这里有:

ZEND_ECHO_SPEC_VAR_HANDLER

ZEND_ECHO_SPEC_TMP_HANDLER

ZEND_ECHO_SPEC_CV_HANDLER

ZEND_ECHO_SPEC_CONST_HANDLER

参考其中的源码实现即可。

 

0x01 针对FCALL进行hook

很多PHP内置函数的调用底层都是ZEND_FCALL进行实现的。


因此,我们需要HOOK两个OP,ZEND_DO_FCALL和ZEND_DO_FCALL_BY_NAME。

我们自己的handler中,通过获取函数名称来对不同的函数进行hook操作。

HOOK思路如下:

(1)    从EG(current_execute_data)->function_state.function中获取原有的函数。

(2)    然后分直接方法调用和类方法调用来执行HOOK操作

(3)    HOOK完成后,恢复原有函数,将function_state.function重新赋值

在函数调用时,函数参数是保存在一个zend虚拟机的栈结构中,因此我们需要移动指针来获取函数中的所有参数。

在FCALL的自定义handler中,函数参数分布图如下:


参数是放入了EG(argument_stack),因此首先获取一个栈顶指针:

void** p = EG(argument_stack)->top ;

函数的参数个数保存在int arg_count = opline->extended_value;中,拿到参数个数之后,我们就可以移动指针了。

比如获取该函数的各个参数,表达式为:

zval *arg1 =*((zval**)(p - arg_count)) ;   //参数1

zval *arg2 =*((zval**)(p - (arg_count - 1))) ;  //参数2

zval *arg3 =*((zval**)(p - (arg_count - 2))) ;  //参数3

栈顶指针的类型为void **,是一个指向指针的指针,即在vm_stack中保存的实际上保存的是参数的指针,因此栈顶指针是一个指针的指针(void** p)。

0x02 针对PHP_FUNCTION进行HOOK

这种形式的HOOK最为简单,在寻找危险函数的时候只需要寻找PHP_FUNCTION(funcName)即看到。

主要思路是在PHP扩展中定义函数别名,然后初始化时从function_table中删除原有的函数定义,并且注册我们自己编写的扩展函数,在执行HOOK操作之后,再恢复原有的内置函数即可。

internal_function是保存内置函数的属性,在恢复原有函数的时候手动调用一次即可。


这篇关于PHP HOOK的若干方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1065362

相关文章

C#高效实现Word文档内容查找与替换的6种方法

《C#高效实现Word文档内容查找与替换的6种方法》在日常文档处理工作中,尤其是面对大型Word文档时,手动查找、替换文本往往既耗时又容易出错,本文整理了C#查找与替换Word内容的6种方法,大家可以... 目录环境准备方法一:查找文本并替换为新文本方法二:使用正则表达式查找并替换文本方法三:将文本替换为图

SQL Server 查询数据库及数据文件大小的方法

《SQLServer查询数据库及数据文件大小的方法》文章介绍了查询数据库大小的SQL方法及存储过程实现,涵盖当前数据库、所有数据库的总大小及文件明细,本文结合实例代码给大家介绍的非常详细,感兴趣的... 目录1. 直接使用SQL1.1 查询当前数据库大小1.2 查询所有数据库的大小1.3 查询每个数据库的详

Java实现本地缓存的四种方法实现与对比

《Java实现本地缓存的四种方法实现与对比》本地缓存的优点就是速度非常快,没有网络消耗,本地缓存比如caffine,guavacache这些都是比较常用的,下面我们来看看这四种缓存的具体实现吧... 目录1、HashMap2、Guava Cache3、Caffeine4、Encache本地缓存比如 caff

Java 中编码与解码的具体实现方法

《Java中编码与解码的具体实现方法》在Java中,字符编码与解码是处理数据的重要组成部分,正确的编码和解码可以确保字符数据在存储、传输、读取时不会出现乱码,本文将详细介绍Java中字符编码与解码的... 目录Java 中编码与解码的实现详解1. 什么是字符编码与解码?1.1 字符编码(Encoding)1

Python Flask实现定时任务的不同方法详解

《PythonFlask实现定时任务的不同方法详解》在Flask中实现定时任务,最常用的方法是使用APScheduler库,本文将提供一个完整的解决方案,有需要的小伙伴可以跟随小编一起学习一下... 目录完js整实现方案代码解释1. 依赖安装2. 核心组件3. 任务类型4. 任务管理5. 持久化存储生产环境

Python批量替换多个Word文档的多个关键字的方法

《Python批量替换多个Word文档的多个关键字的方法》有时,我们手头上有多个Excel或者Word文件,但是领导突然要求对某几个术语进行批量的修改,你是不是有要崩溃的感觉,所以本文给大家介绍了Py... 目录工具准备先梳理一下思路神奇代码来啦!代码详解激动人心的测试结语嘿,各位小伙伴们,大家好!有没有想

Python如何调用另一个类的方法和属性

《Python如何调用另一个类的方法和属性》在Python面向对象编程中,类与类之间的交互是非常常见的场景,本文将详细介绍在Python中一个类如何调用另一个类的方法和属性,大家可以根据需要进行选择... 目录一、前言二、基本调用方式通过实例化调用通过类继承调用三、高级调用方式通过组合方式调用通过类方法/静

java -jar example.jar 产生的日志输出到指定文件的方法

《java-jarexample.jar产生的日志输出到指定文件的方法》这篇文章给大家介绍java-jarexample.jar产生的日志输出到指定文件的方法,本文给大家介绍的非常详细,对大家的... 目录怎么让 Java -jar example.jar 产生的日志输出到指定文件一、方法1:使用重定向1、

Java报错:org.springframework.beans.factory.BeanCreationException的五种解决方法

《Java报错:org.springframework.beans.factory.BeanCreationException的五种解决方法》本文解析Spring框架中BeanCreationExce... 目录引言一、问题描述1.1 报错示例假设我们有一个简单的Java类,代表一个用户信息的实体类:然后,

Linux查询服务器系统版本号的多种方法

《Linux查询服务器系统版本号的多种方法》在Linux系统管理和维护工作中,了解当前操作系统的版本信息是最基础也是最重要的操作之一,系统版本不仅关系到软件兼容性、安全更新策略,还直接影响到故障排查和... 目录一、引言:系统版本查询的重要性二、基础命令解析:cat /etc/Centos-release详