linux ptrace函数

linux ptrace函数【ptrace系统调用】功能描述:提供父进程观察和控制另一个进程执行的机制,同时提供查询和修改另一进程的核心影像与寄存器的能力。主要用于执行断点调试和系统调用跟踪。父进程可通过调用fork,接着指定所产生的子进程的PTRACE_TRACEME行为,最后使用exec等操作来初始化一个进程跟踪。可替代的做法是,父进程通过PTRACE_ATTACH请求跟踪一个现存进程的执行。当子进程

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全家桶1年46,售后保障稳定

【ptrace系统调用】

ptrace是 process trace 的缩写。
功能描述:
提供父进程观察和控制另一个进程执行的机制,同时提供查询和修改另一进程的核心影像与寄存器的能力。主要用于执行断点调试和系统调用跟踪。

父进程可通过调用fork,接着指定所产生的子进程的PTRACE_TRACEME行为,最后使用exec等操作来初始化一个进程跟踪。可替代的做法是,父进

程通过PTRACE_ATTACH请求跟踪一个现存进程的执行。

当子进程被跟踪时,每次接收到信号都会停止执行,即使不对信号进行处理(SIGKILL信号除外)。父进程下次执行wait调用时,会接收到核心的通告,

并可能检查和修改已停止的子进程。父进程使子进程继续执行,并有可能忽略接收到的信号。

用法:
#include <sys/ptrace.h>

long ptrace(enum __ptrace_request request, pid_t pid, void *addr, void *data);

参数:
request:请求执行的行为,可能选择有

PTRACE_TRACEME //指示父进程跟踪某个子进程的执行。任何传给子进程的信号将导致其停止执行,同时父进程调用wait()时会得到通告。之后,子进程

调用exec()时,核心会给它传送SIGTRAP信号,在新程序开始执行前,给予父进程控制的机会。pid, addr, 和 data参数被忽略。

以上是唯一由子进程使用的请求,剩下部分将由父进程使用的请求。

PTRACE_PEEKTEXT, PTRACE_PEEKDATA //从子进程内存空间addr指向的位置读取一个字,并作为调用的结果返回。Linux内部对文本段和数据段不加区分,

所以目前这两个请求相等。data参数被忽略。
PTRACE_PEEKUSR //从子进程的用户区addr指向的位置读取一个字,并作为调用的结果返回。
PTRACE_POKETEXT, PTRACE_POKEDATA //将data指向的字拷贝到子进程内存空间由addr指向的位置。
PTRACE_POKEUSR //将data指向的字拷贝到子进程用户区由addr指向的位置。
PTRACE_GETREGS, PTRACE_GETFPREGS //将子进程通用和浮点寄存器的值拷贝到父进程内由data指向的位置。addr参数被忽略。
PTRACE_GETSIGINFO //获取导致子进程停止执行的信号信息,并将其存放在父进程内由data指向的位置。addr参数被忽略。
PTRACE_SETREGS, PTRACE_SETFPREGS //从父进程内将data指向的数据拷贝到子进程的通用和浮点寄存器。addr参数被忽略。
PTRACE_SETSIGINFO //将父进程内由data指向的数据作为siginfo_t结构体拷贝到子进程。addr参数被忽略。
PTRACE_SETOPTIONS //将父进程内由data指向的值设定为ptrace选项,data作为位掩码来解释,由下面的标志指定
PTRACE_O_TRACESYSGOOD //当转发syscall陷阱(traps)时,在信号编码中设置位7,即第一个字节的最高位。例如:SIGTRAP | 0x80。这有利于追踪者

识别一般的陷阱和那些由syscall引起的陷阱。

PTRACE_O_TRACEFORK //通过 (SIGTRAP | PTRACE_EVENT_FORK << 8) 使子进程下次调用fork()时停止其执行,并自动跟踪开始执行时就已设置SIGSTOP

信号的新进程。新进程的PID可以通过PTRACE_GETEVENTMSG获取。
PTRACE_O_TRACEVFORK //通过 (SIGTRAP | PTRACE_EVENT_VFORK << 8) 使子进程下次调用vfork()时停止其执行,并自动跟踪开始执行时就已设置SIGSTOP

信号的新进程。新进程的PID可以通过PTRACE_GETEVENTMSG获取。
PTRACE_O_TRACECLONE //通过 (SIGTRAP | PTRACE_EVENT_CLONE << 8) 使子进程下次调用clone()时停止其执行,并自动跟踪开始执行时就已设置SIGSTOP

信号的新进程。新进程的PID可以通过PTRACE_GETEVENTMSG获取。

PTRACE_O_TRACEEXEC //通过 (IGTRAP | PTRACE_EVENT_EXEC << 8) 使子进程下次调用exec()时停止其执行。

PTRACE_O_TRACEVFORKDONE //通过 (SIGTRAP | PTRACE_EVENT_VFORK_DONE << 8) 使子进程下次调用exec()并完成时停止其执行。

PTRACE_O_TRACEEXIT //通过 (SIGTRAP | PTRACE_EVENT_EXIT << 8) 使子进程退出时停止其执行。子进程的退出状态可通过PTRACE_GETEVENTMSG

PTRACE_GETEVENTMSG //获取刚发生的ptrace事件消息,并存放在父进程内由data指向的位置。addr参数被忽略。

PTRACE_CONT //重启动已停止的进程。如果data指向的数据并非0,同时也不是SIGSTOP信号,将会作为传递给子进程的信号来解释。那样,父进程可以控制是否将

一个信号发送给子进程。 addr参数被忽略。

PTRACE_SYSCALL, PTRACE_SINGLESTEP //如同PTRACE_CONT一样重启子进程的执行,但指定子进程在下个入口或从系统调用退出时,或者执行单个指令后停止

执行,这可用于实现单步调试。addr参数被忽略。

PTRACE_SYSEMU, PTRACE_SYSEMU_SINGLESTEP //用于用户模式的程序仿真子进程的所有系统调用。

PTRACE_KILL //给子进程发送SIGKILL信号,从而终止其执行。data,addr参数被忽略。

PTRACE_ATTACH //衔接到pid指定的进程,从而使其成为当前进程的追踪目标。

PTRACE_DETACH //PTRACE_ATTACH的反向操作。

pid:目标进程标识。

addr:执行peek和poke操作的目标地址。

data:对于poke操作,存放数据的地方。对于peek操作,获取数据的地方。

返回说明:

成功执行时,PTRACE_PEEK*请求返回所请求的数据,其它返回0。失败返回-1,errno被设为以下的某个值。由于一个成功的PTRACE_PEEK*请求可能返回-1,

决定错误是否发生前,调用者应检查errno。

EBUSY:分配和释放调试寄存器时出错

EFAULT:读写不可访问的内存空间

EINVAL:尝试设置无效选项

EIO:请求无效,或者尝试读写父子进程不可访问的空间

EPERM:没有权限追踪指定的进程

ESRCH:指定的子进程不存在,或者当前正由调用者追踪

 

 

一个简单的 ptrace 例子

// test.cpp

#include <stdio.h>

int main()
{
    printf("---------- test 1 ----------\n");
    printf("---------- test 2 ----------\n");
    printf("---------- test 3 ----------\n");
    return 0;
}

Jetbrains全家桶1年46,售后保障稳定

编译 g++ test.cpp -o test –static

// ptrace.cpp

#include <stdio.h>
#include <stdlib.h>
#include <sys/ptrace.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <sys/reg.h>
#include <unistd.h>

int main()
{
    pid_t pid;
    int orig_eax, eax, ebx, ecx, edx;
   
    pid = fork();
    if(pid == 0)
    {
            ptrace(PTRACE_TRACEME, 0, NULL, NULL);
       
            printf("execve = %d\n", execve("./test", NULL, NULL));//执行待监控的进程
        
            exit(0);
        }
       
    while(1) // if (WIFSIGNALED(status)) //检查是否因为信号挂起
    {
        int status;
        wait(&status);
        if(WIFEXITED(status)) break;

        orig_eax = ptrace(PTRACE_PEEKUSER, pid, ORIG_EAX<<2, NULL);
        eax = ptrace(PTRACE_PEEKUSER, pid, EAX<<2, NULL);
        ebx = ptrace(PTRACE_PEEKUSER, pid, EBX<<2, NULL);
       
        printf("ORIG_EAX = %d,        EAX = %d,        EBX = %d\n", orig_eax, eax, ebx);

        ptrace(PTRACE_SYSCALL, pid, NULL, NULL);
    }
    return 0;
}

编译 g++ ptrace.cpp -o ptrace –static

测试 ./ptrace

输出

ORIG_EAX = 11,        EAX = 0,        EBX = 0

ORIG_EAX = 122,        EAX = -38,        EBX = -1074643290

ORIG_EAX = 122,        EAX = 0,        EBX = -1074643290

ORIG_EAX = 45,        EAX = -38,        EBX = 0

ORIG_EAX = 45,        EAX = 161513472,        EBX = 0

ORIG_EAX = 45,        EAX = -38,        EBX = 161516752

ORIG_EAX = 45,        EAX = 161516752,        EBX = 161516752

ORIG_EAX = 243,        EAX = -38,        EBX = -1074642896

ORIG_EAX = 243,        EAX = 0,        EBX = -1074642896

ORIG_EAX = 45,        EAX = -38,        EBX = 161651920

ORIG_EAX = 45,        EAX = 161651920,        EBX = 161651920

ORIG_EAX = 45,        EAX = -38,        EBX = 161652736

ORIG_EAX = 45,        EAX = 161652736,        EBX = 161652736

ORIG_EAX = 197,        EAX = -38,        EBX = 1

ORIG_EAX = 197,        EAX = 0,        EBX = 1

ORIG_EAX = 192,        EAX = -38,        EBX = 0

ORIG_EAX = 192,        EAX = -1217093632,        EBX = 0

ORIG_EAX = 4,        EAX = -38,        EBX = 1

———- test 1 ———-

ORIG_EAX = 4,        EAX = 29,        EBX = 1

ORIG_EAX = 4,        EAX = -38,        EBX = 1

———- test 2 ———-

ORIG_EAX = 4,        EAX = 29,        EBX = 1

ORIG_EAX = 4,        EAX = -38,        EBX = 1

———- test 3 ———-

ORIG_EAX = 4,        EAX = 29,        EBX = 1

ORIG_EAX = 252,        EAX = -38,        EBX = 0

内核 Linux 2.6.32-35-generic

 参考:

http://en.wikipedia.org/wiki/Ptrace

http://blog.csdn.net/daniel_ustc/article/details/9672455

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/230967.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • mongodb 集合_mongodb原理

    mongodb 集合_mongodb原理最近公司用MongoDB,整合一下网上的优缺点,学习下MongoDB没有找到原作者一:MongoDB的优点和缺点优点面向文档存储(类JSON数据模式简单而强大) 动态查询 全索引支持,扩展到内部对象和内嵌数组 查询记录分析 快速,就地更新 高效存储二进制大对象(比如照片和视频) 复制和故障切换支持 Auto-Sh…

    2025年6月26日
    5
  • UML一一 类图关系 (泛化、实现、依赖、关联、聚合、组合)[通俗易懂]

    UML一一 类图关系 (泛化、实现、依赖、关联、聚合、组合)[通俗易懂]目录类图关系概述面向对象是符合人们对现实世界的思维模式,利用面向对象设计,特别是采用各种设计模式来解决问题时,会设计多个类,然后创建多个对象,一个设计良好的类,应该是兼顾信息和行为并且高内聚。而不同的类之间,应该做到松耦合。当面对应用系统或者需要解决的问题经常是复杂的、高度抽象的,我们创建的多个对象往往是有联系的,通常对象之间的关系可以分为以下几类:泛化关系实现关系依赖关…

    2022年7月12日
    21
  • UE4/UE5 使用Unreal Datasmith完全教程

    UE4/UE5 使用Unreal Datasmith完全教程(2019.12.9)Datasmith功能已内置到虚幻4.24版本Datasmith支持导入Twinmotion:Twinmotion教程汇总(2021.4.2)Datasmith安装到4.20-4.23版本:UE4在4.20-23版本安装Datasmith插件(2021.3.30更新)Datasmith导出插件:Datasmith导出插件(2021.4.12更新)导入sketchup草图大师:草图大师SU导入Unity和虚幻UE4对比导入Revit2018.3+:UE4使用Da.

    2022年10月5日
    6
  • java默认上传文件大小限制_文件大小超过云附件总限制

    java默认上传文件大小限制_文件大小超过云附件总限制win系统下解决ASP上传文件大小限制200K_电脑基础知识_IT/计算机_专业资料。win系统下解决ASP上传文件大小限制200K解决ASP上传文件大小限制200Kwin2003系统下的……介绍上传//新建一个SmartUpload对象SmartUploadsu=newSmartUpload();//上传初始化su.initialize(pageCo…

    2025年9月13日
    7
  • dubbo负载均衡策略(XML、注解、SpringBoot配置)「建议收藏」

    dubbo负载均衡策略(XML、注解、SpringBoot配置)「建议收藏」本示例是在上一篇文章中搭建的实例来讲解,详情先查看:SpringBoot集成dubbo最新实战教程:dubbo-spring-boot-starter一.简介在集群负载均衡时,Dubbo提供了多种均衡策略,默认为random随机调用。二.负载均衡策略1.RandomLoadBalance随机,按权重设置随机概率。 在一个截面上碰撞的概率高,但调用量越大分布越均…

    2022年7月27日
    3
  • axure 发布后隐藏顶部菜单 或展开顶部菜单[通俗易懂]

    axure 发布后隐藏顶部菜单 或展开顶部菜单[通俗易懂]这样可以实现菜单栏最小化,而且在你鼠标不移动到左上角时,小箭头会隐藏,效果就可以了。axure9.0版本在发布后HTML页面打开时总是在顶部弹出菜单既不美观也影响效果。本人axure小白,摸索半天后发现也不能完全关闭或者不显示(除非代码修改);解决方案就是在请求地址后面拼接。…

    2022年8月19日
    17

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号