Skip to content

Latest commit

 

History

History

grsec-code-analysis

PaX/Grsecurity 代码分析

简介

这个目录主要是针对 PaX/Grsecurity 的代码分析的文档, 这个文档是针对这部分文档的简介。

PaX/Grsecurity是一组针对Linux Kernel的加固补丁,跟“传统”的基于LSM( SELinux/AppArmor/etc)不同的是PaX/GRsecurity不仅是具备RBAC对权限以及信息流的控制,而是通过一系列的安全改进让Linux内核在为用户空间提供防御能力( ASLR/PAGEEXEC/SEGEXEC/etc)的同时内核本身的防护能力也大大加强,PaX/GRsecurity作为整个系统安全防御领域最重要的起源,不仅仅影响了Linux内核,也大大的影响了Windows以及BSD内核的安全特性,甚至影响了包含Intel和ARM系列处理器在内的硬件厂商,在可预见的未来必然会影响RISC-V的安全特性支持,由于Linux内核社区多年以来都奉行"A bug is bug"和"security through obscurity"的哲学导致了Linux内核自第一枚核弹null-ptr ref利用方式曝光后进入了“隐蔽战争纪元”,大规模利用安全事件持续性危害着数据中心的GNU/Linux用户,后来随着Android手机的崛起因为其使用的是Linux内核而必须承受极大风险,下一个受害者或许是嵌入式设备( Internet of Shxt),虽然在2015年华盛顿邮报向公众曝光了Linux安全真相后Linux内核社区迫于Linux基金会的压力成立了KSPP(内核自防护项目),KSPP的参与者大多为Linux基金会的客户(Google/RedHat/Intel/etc),参与者基在移植PaX/GRsecurity的某些特性或者代码级加固的过程中多次没有理解代码造成的抄袭错误引入了新的漏洞,加上一些其他社区政治因素导致PaX/GRsecurity停止公开下载,HardenedLinux社区认为这一切的罪魁祸首是Linux基金会,在这篇“方舟之役”的申明中已经有详细的阐述这里不再描述。过去的17年中PaX/GRsecurity对自由软件社区和内核安全领域的巨大贡献但并不是被很多技术人员所了解,这是HardenedLinux社区公开一些PaX/GRsecurity特性的代码分析的主要原因,也算是对社区的馈赠。

PaX 部分

PaX 部分主要是针对内核 Memory corruption 的漏洞利用的防御补丁。PaX 的特性有大的完整特性,比如整个 ASLR 实现,也有零散的针对加固特性。下面的文档会把特性进行归类,并做相应简介,尽可能勾画出某个特性在整体加固中发挥的作用,方便后续文档阅读。

ASLR

简介

PaX 针对进程地址空间实现的 ASLR,是目前最强的 ASLR 实现,也是 PaX 的一个非常重要的特性。能够有效的抵挡很多基于地址的攻击,比如一些代码注入,执行流重定向,代码复用等问题。上述基于内存的利用往往要知道目标代码的位置,进程内存布局的随机化,使得每一次执行程序都有不同的虚拟地址空间。无论是注入代码的硬编码地址、使用相对偏移或者共享库代码的复用等都会因为地址的随机化,使得攻击难度上升。尝试克服随机化的常见方法有暴力破解、漏洞的内存信息泄漏。前者需要克服的是随机化的程度高低(熵),后者依赖于具体漏洞。

设计实现

ASLR 设计的目的是:在进程地址空间的执行代码主程序,堆,共享库,映射区域,栈的地址都引入随机化,使得每一次运行都有不同的虚拟地址。引入随机化主要由三个子特性来实现,包括: PAX_RANDMMAP、PAX_RANDEXEC、PAX_RANDUSTACK。他们承担的任务如下:

Option 特性
PAX_RANDMMAP 映射基址(包括共享库映射),堆基址, ET_DYN 类型 ELF 主程序基址的随机化
PAX_RANDEXEC ET_EXE 类型 ELF 主程序基址的随机化
PAX_RANDUSTACK 栈基址的随机化

这三个的特性的代码盘踞在进程地址空间变化的各个阶段,大部分位于初始化的时候。比如针对 ELF 类型的可执行代码,进程代码的加载,共享库的映射,内存资源的分配,都是在 load_elf_binary 函数里面完成的,随机化在这里面也是最多的,一些随机化只需要随机化基址,有些需要后续操作的配合。这些子特性的文档介绍如下:

  1. PAX_RANDMMAP 按照随机化的对象分节分析。
  2. PAX_RANDUSTACK 按照栈初始化流程中引入随机化的点逐点分析。

虽然 ASLR 是由几个子特性组成的,但是子特性之间有时会有交叉或者相关的代码,应该以一个整体的特性去看待。比如栈的随机化会影响到映射区域的随机化。

相关漏洞分析

这篇文档从二进制可执行文件加载函数 load_elf_binary 开始,引出以 offset2lib 为例,分析 offset2lib 原理和加固的实现,具体可以结合 ASLR 其他三个文档一起阅读。 PaX 的 ASLR gap 实现 这篇文章介绍 PaX 的在虚拟地址空间实现 gap 的加固。这个特性成功地防御了 stack_clash 这个内核的实现问题带来的利用平面。

PAX_NOEXEC

简介

PAX_NOEXEC 是 PaX 针对代码注入的利器,斩断了执行流重定向到数据所在区域的可能。由于代码注入往往是透过数据操作写入到内存的数据段中,若将这部分内存区域保持为不可执行,一旦 ip 指针执行到这种区域,就会被拦截,防止恶意代码的执行。

设计实现

PAX_NOEXEC 设计的目的是:保持内存的健康状态,防止非代码段的内存被执行。他的设计方案有两个环节:

Option 特性
PAX_MPROTECT 在内存初始映射时和修改映射性质时确保内存的健康状态(例如写/执行不可同时)
PAX_PAGEEXEC/SEGMEXEC 透过软件或者硬件的方式,保持不可执行的内存不被执行。

这个特性的关键点在于内存性质的维持和执行时的配合,有不合法内存映射的行为都是不允许的,而不合法的访问执行则会被中断处理拦截。文档索引如下:

  1. PAX_PAGEEXEC 这个文档翻译自 PaX 的关于 pageexec 的官方文档。
  2. PAX_SEGMEXEC 这个文档翻译自 PaX 的关于 segexec 的官方文档。
  3. PAX_MPROTECT 按照内存映射初始化和映射修改两个方面分析 PaX 对内存的限制

PAX_USERCOPY

简介

PAX_USERCOPY 是 PaX 针对用户/内核空间之间的内存相互拷贝的检查,解决了拷贝目的缓冲区(包括栈和堆)溢出的问题。

设计实现

这个特性的实现主要是针对 copy_to/from_user 函数操作的目的缓冲区内存的长度和性质进行检查。下面是一些文档:

  1. PAX_USERCOPY.md 这个文档分析比较了 PaX/Grsecurity 以及 KSPP 的实现,PaX 的实现包括目的缓冲区的内存性质和长度的检查, KSPP 最开始的实现有所缺漏,仅仅是针对长度进行检查。

关于内核栈

简介

PaX/Grsecurity 针对内核栈的做了多方位的加固。内核栈是进程使用系统调用陷入内核空间时开辟的栈,PaX/Grsecurity 针对栈帧的溢出破坏,栈的反复进出的信息泄漏,栈上信息的破坏覆写等内核栈实现的薄弱性,进行了全面的加固,使得内核的安全性大大提高。

设计实现

  1. Kernel stack 的演变 按照内核版本,分析 PaX 实现内核栈内存的虚拟连续(物理不连续)和 thread_info 分离,相应比较了 KSPP 极其相似的实现。
  2. GRSECURITY_KSTACKOVERFLOW 按照 Grsecurity 的 KSTACKOVERFLOW 特性加固的地方逐个分析,和上一个特性有交叉的地方。
  3. PAX_RANDKSTACK 这个文档介绍同一个进程多次系统调用时进入内核栈基址的随机化实现。

PAX_REFCOUNT

简介

内核引用计数的溢出可能会导致 use-after-free, PAX_REFCOUNT 针对的就是内核引用计数的溢出的问题进行的加固。

设计实现

PAX_REDCOUNT 的实现非常简单,是在原子操作相关的 atomic_* 函数中添加溢出的检测,从而防止进一步利用。

  1. PAX_REFCOUNT 这个文档以 ARM 为例介绍了这个特性的实现。

PAX_KERNEXEC/PAX_UDEREF

简介

PAX_KERNEXEC/PAX_UDEREF 是 PaX 率先实现的,针对 ret2usr 这类漏洞利用的防御的特性。这组防护特性有效的防止内核执行流重定向到位于用户空间的内存。在 PaX 实现多年以后,受到启发的硬件厂商也提供了相应的硬件支持来防御 ret2usr,但是 PAX_KERNEXEC/PAX_UDEREF 依然是最为强悍的实现。

设计实现

  1. PAX_KERNEXEC 内核版的 PAGEEXEC/MPROTECT,针对内核内存的性质做限制,偏重代码
  2. PAX_MEMORY_UDEREF 分离内核/用户空间,防止内核/用户空间越界操作,偏中数据
  3. smep/smap 是 x86_64 上的硬件支持
  4. PXN/PAN 是 ARM v7 以后的硬件支持 附:recent_arm_security_improvements这是 Grsecurity 针对 KERNEXEC/UDEREF 的一篇介绍,也有非常详细的是实现介绍。

关于内存信息泄漏的问题

简介

内核的内存泄漏对安全来说是一个巨大的威胁。敏感信息的泄漏,可以协助 bypass 各种安全特性,造成很大威胁。关于内存信息泄漏的问题主要是内存泄漏和敏感信息资源的访问,前者发生在内存拷贝向用户空间过多,当内存含有敏感信息时造成 infoleak,后者则是和具体漏洞有关,相对比较零散,加固方式也比较多样。

设计实现

  1. MEMORY_LEAK 这个文档介绍了 PAX_MEMORY_SANITIZE 和 PAX_MEMORY_STACKLEAK, 前者针对已释放内存的擦除,后者则是栈上残留信息的擦除。
  2. infoleak 相关信息汇集。

附录

  1. PaX Team 自己维护的官方文档 这些文档是 PaX Team 官方撰写的,最为直接准确。但是部分文档是最初的设计文档,没有跟进更新。
  2. HardenedLinux 社区贡献了多个针对上述官方文档的翻译版,共计有 PAGEEXEC的最早设计文档,SEGMEXEC设计文档,MPROTECT早期设计文档等。