问答
发起
提问
文章
攻防
活动
Toggle navigation
首页
(current)
问答
商城
实战攻防技术
漏洞分析与复现
NEW
活动
摸鱼办
搜索
登录
注册
GPU 驱动漏洞:窥探驱动漏洞利用的技术奥秘
本文尝试以 GPU 漏洞为引介绍围绕 GPU 驱动这一攻击面,安全研究人员对内核漏洞利用技术做的一些探索。 背景介绍 目前移动 SOC 平台上由多个硬件模块组成,常见的硬件模块有:CPU、GPU、Modem基...
本文尝试以 GPU 漏洞为引介绍围绕 GPU 驱动这一攻击面,安全研究人员对内核漏洞利用技术做的一些探索。 背景介绍 ---- 目前移动 SOC 平台上由多个硬件模块组成,常见的硬件模块有:CPU、GPU、Modem基带处理器、ISP(图像处理器)等,这些硬件模块通过硬件总线互联,协同完成任务。 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-bfda087d8892abb2cae61dc1319f0bc9e12016b6.png) 对于 GPU 驱动漏洞研究来说,我们需要关注的一个关键特性是 **GPU 和 CPU 共用同一块 RAM。** 在 CPU 侧操作系统通过管理 CPU MMU 的页表来实现虚拟地址到物理地址的映射 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-9a0769933cbbd55176d6a54809ca2e3d45eb86ab.png) GPU 也有自己的 MMU,不过 GPU 的页表由 CPU 内核中的 GPU 驱动管理,从而限制 GPU 能够访问的物理地址范围。 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-6addbae99cdda81d1a80899d82fd186188398ad2.png) 在实际的业务使用中,一般是 CPU 侧分配一段物理内存,然后映射给 GPU , GPU 从共享内存中取出数据完成计算、渲染后再将结果写回共享内存,从而完成 GPU 与 GPU 之间的交互。对于 GPU 驱动安全研究来说,特殊的攻击面在于由于其需要维护 GPU 页表,这个过程比较复杂,涉及到内核中的各个模块的配合,非常容易出现问题,历史上也出现了多个由于 GPU 页表管理失误导致的安全漏洞 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-9994ce32e05e442aaab046d466d3fd4b7226d372.png) 以 ARM Mali 驱动为例,这几年出现的几个比较有代表性的漏洞如下: | 漏洞 | 类型 | 漏洞原语 | |---|---|---| | CVE-2021-39793 | 页权限错误 | 篡改 只读映射到用户进程的物理页 | | CVE-2021-28664 | 页权限错误 | 篡改 只读映射到用户进程的物理页 | | CVE-2021-28663 | GPU MMU 操作异常 | 物理页 UAF | | CVE-2023-4211 | 条件竞争 UAF | SLUB 对象 UAF | | CVE-2023-48409 | 整数溢出 | 堆溢出 | | CVE-2023-26083 | 内核地址泄露 | 内核地址泄露 | | CVE-2022-46395 | 条件竞争 UAF | 物理页 UAF | > 其中前 3 个漏洞是管理 GPU 页表映射时的漏洞,后面几个就是常规驱动漏洞类型 CVE-2021-28664 -------------- 分析代码下载:<https://armkeil.blob.core.windows.net/developer/Files/downloads/mali-drivers/kernel/mali-bifrost-gpu/BX304L01B-SW-99002-r19p0-01rel0.tar> 先以最简单的漏洞开始讲起,这个漏洞算是 Mali 第一个出名的漏洞了,同期出道的还有 CVE-2021-28664,这个漏洞是由 [Project Zero](https://googleprojectzero.github.io/0days-in-the-wild/0day-RCAs/2021/CVE-2021-39793.html) 捕获的在野利用,该漏洞的 Patch 如下 ```c static struct kbase_va_region *kbase_mem_from_user_buffer( struct kbase_context *kctx, unsigned long address, unsigned long size, u64 *va_pages, u64 *flags) { [...] + int write; [...] + write = reg->flags & (KBASE_REG_CPU_WR | KBASE_REG_GPU_WR); + #if KERNEL_VERSION(4, 6, 0) > LINUX_VERSION_CODE faulted_pages = get_user_pages(current, current->mm, address, *va_pages, #if KERNEL_VERSION(4, 4, 168) <= LINUX_VERSION_CODE && \ KERNEL_VERSION(4, 5, 0) > LINUX_VERSION_CODE - reg->flags & KBASE_REG_CPU_WR ? FOLL_WRITE : 0, - pages, NULL); + write ? FOLL_WRITE : 0, pages, NULL); #else - reg->flags & KBASE_REG_CPU_WR, 0, pages, NULL); + write, 0, pages, NULL); #endif #elif KERNEL_VERSION(4, 9, 0) > LINUX_VERSION_CODE faulted_pages = get_user_pages(address, *va_pages, - reg->flags & KBASE_REG_CPU_WR, 0, pages, NULL); + write, 0, pages, NULL); #else faulted_pages = get_user_pages(address, *va_pages, - reg->flags & KBASE_REG_CPU_WR ? FOLL_WRITE : 0, - pages, NULL); + write ? FOLL_WRITE : 0, pages, NULL); #endif ``` Patch 的关键点在于将 get\_user\_pages 参数中的 reg->flags & KBASE\_REG\_CPU\_WR 换成了 reg->flags & (KBASE\_REG\_CPU\_WR | KBASE\_REG\_GPU\_WR) ,这两个标记的作用如下: - KBASE\_REG\_CPU\_WR:表示 reg 能够已写权限映射到用户态进程 - KBASE\_REG\_GPU\_WR: 表示 reg 能够已写权限映射到 GPU reg 的类型为 struct kbase\_va\_region , MALI 驱动中使用 kbase\_va\_region 管理物理内存,包括物理内存的申请/释放、GPU/CPU 页表映射管理等。 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-8adc1bf70c15d1a962a7adc1e91dc5c0f1a920af.png) 图中的关键要素如下: - kbase\_va\_region 中 cpu\_alloc 和 gpu\_alloc 指向 kbase\_mem\_phy\_alloc ,表示该 reg 拥有的物理页,且大部分场景下 cpu\_alloc = gpu\_alloc - kbase\_va\_region 的 flags 字段控制驱动映射 reg 时的权限,假如 flags 为 KBASE\_REG\_CPU\_WR 表示该 reg 能够被 CPU 映射为可写权限,如果没有该 flag 则不允许将 reg 以可写模式映射到 CPU 进程,确保无法进程修改这些物理页 核心观点:驱动利用 kbase\_va\_region 表示一组物理内存,这组物理内存可以被 CPU 上的用户进程和 GPU 分别映射,映射的权限由 reg->flags 字段控制. 回到漏洞本身,其调用路径中的关键代码如下: ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-104adf09acabfe04d6eab2cc9da9dbbeded5fd45.png) 漏洞在于传递 get\_user\_pages 参数是只考虑了 KBASE\_REG\_GPU\_WR 情况,没有考虑 KBASE\_REG\_CPU\_WR,当 reg->flags 为 KBASE\_REG\_CPU\_WR 时 get\_user\_pages 的第三个参数为 0 ```c /* * This is the same as get_user_pages_remote(), just with a * less-flexible calling convention where we assume that the task * and mm being operated on are the current task's and don't allow * passing of a locked parameter. We also obviously don't pass * FOLL_REMOTE in here. */ long get_user_pages(unsigned long start, unsigned long nr_pages, unsigned int gup_flags, struct page **pages, struct vm_area_struct **vmas) { return __get_user_pages_locked(current, current->mm, start, nr_pages, pages, vmas, NULL, false, gup_flags | FOLL_TOUCH); } ``` get\_user\_pages 的作用的是根据用户进程提供的 va (start)遍历进程页表,返回的是 va 对应物理地址对应的 page 结构体指针,结果保存到 pages 数组中。 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-6a2b3f6998f993d947a3dad3c26ecaad57106fa8.png) > 即根据 task\_struct->mm 找到进程页表,遍历页表获取物理地址 其中如果 gup\_flags 为 1,表示获取 va 对应 page 后会写入 page 对应的物理页,然后在 get\_user\_pages 内部需要对只读页面和 COW 页面做额外处理,避免这些特殊 va 对应的物理页被修改导致非预期行为。 - 如果 vma 为只读,API 返回错误码 - VA 的映射为 COW 页,在 API 内会完成写时拷贝,并返回新分配的 page ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-d07d9fe84254d5692880a9f2f50772600a93e659.png) 当 gup\_flags 为 0 时则直接返回页表遍历的结果(P0) 对于这个漏洞而言,我们可以创建一个 reg->flags 为 KBASE\_REG\_CPU\_WR 的 kbase\_va\_region,再导入页面时就可以获取进程中任意 va 对应 page 到 kbase\_va\_region,最后再将其以可写权限映射到用户态进程,这样就可以实现篡改进程中任意只读映射对应的物理页。 这一原语要进一步利用需要依赖操作系统的机制,首先介绍最简单的一种利用方式,Linux 内核在处理磁盘中的文件系统时,会对从磁盘中读取的物理页做缓存来加速文件访问的性能,同时减少重复文件物理页,减少开销 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-390f2d00715d69ef22cdddff0356a5ecd6fec9b0.png) 如果所示: - 当进程尝试读取物理页时,比如只读权限 mmap ,内核会搜索 page cache 如果找到就直接返回,否则就从磁盘中加载物理页到 Page Cache 中,然后返回 - 如果是写则会有对应的 flush cache 机制 具体来说,当两个进程同时以只读权限 mmap libc.so 文件时,这两个进程的 VA 会指向同一个物理页 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-f168a05e9a0fb9d86b0ff9f9856cf97564f2ea58.png) 这样当我们利用漏洞修改 Page Cache 中的物理页后,其他进程也会受到影响,因为都是映射的同一块物理地址,因此攻击思路就来了: - 只读映射 libc.so 利用漏洞篡改其在 Page Cache 中物理页,在其中注入 shellcode,等高权限进程调用时就能提权 - 类似的手法修改 /etc/passwd 完成提权 除了修改文件系统的 Page Cache 外,在 Android 平台上还有一个非常好的目标,binder 驱动会往用户态进程映射只读 page,里面的数据结构为 flat\_binder\_object,binder\_transaction\_buffer\_release 里面会使用 flat\_binder\_object->handle,相关代码如下: ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-411c9155475c9887106096de79118dfc4a74f785.png) 首先通过 binder\_get\_node 查找 node,然后会调用 binder\_put\_node 减少 node 的引用计数,当 node 引用为0时会释放 node。 由于 flat\_binder\_object 所在物理页用户态无法修改,所以可以保证这个流程的正确性,当我们只读物理页写漏洞篡改 flat\_binder\_object->handle 指向另一个 node 时,触发 binder\_transaction\_buffer\_release 就能导致 node 引用计数不平衡 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-c3a02c2c7122e70a15de2c5d546f1eb99b442768.png) 最后可以将漏洞转换为 binder\_node 的UAF,然后采用 [CVE-2019-2205](https://labs.bluefrostsecurity.de/files/OffensiveCon2020_bug_collision_tale.pdf) 的利用方式进行漏洞利用即可。 此外类似的漏洞在 2016 年就已经出现在高通 GPU 驱动中,[CVE-2016-2067](https://www.blackhat.com/docs/eu-16/materials/eu-16-Taft-GPU-Security-Exposed.pdf): ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-7f2d35eaa3ddadf9da84f9c95b9345ffd77e83f6.png) > 同样的业务场景,也意味着同类型的漏洞也可能会产生 CVE-2021-28663 -------------- 该漏洞是 Mali 驱动在管理 GPU 物理页映射时导致的物理页 UAF 漏洞,为了能够理解该漏洞,首先需要对 Mali 驱动的相关代码有所了解,上节提到 Mali 使用 kbase\_va\_region 对象表示物理内存资源,然后 CPU 用户进程 和 GPU 可以按需映射,对物理内存进行访问。 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-c53a6fe84016199c275e3c08e90f4f9840993418.png) kbase\_va\_region 的创建位于 kbase\_api\_mem\_alloc 接口,其关键代码如下: ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-4ac7b3c98d5317b94b5f1830b647f63214f46374.png) 对于 BASE\_MEM\_SAME\_VA 情况驱动会做特殊处理,SAME\_VA 的意思是在映射 reg 时,GPU 和 CPU 的虚拟地址是一样的,这样可能是为了便于数据传递时,之间进行指针传递。 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-4ec2c39bc6223f82654b5bd42c7d5e2345a99bdb.png) kbase\_va\_region 的释放位于 kbase\_api\_mem\_free 接口,其关键代码如下: ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-14a956dce4fc31686e6af2129781c28ceca09dc4.png) 这个的大体逻辑是先根据 gpu\_addr 找到 reg,然后释放 reg 和 reg->xx\_alloc 的引用,对于这种复杂的对象管理,可以先按照正常流程分析下对象之间的关系, kbase\_va\_region 中与生命周期相关的字段如下: ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-db0590af786f38c206dc1099383ff3dde06c8062.png) 上图表示的是 kbase\_api\_mem\_alloc 创建非 SAME\_VA 内存的场景,kbase\_gpu\_mmap 执行后会对 gpu\_mappings 加一,然后通过 kbase\_api\_mem\_free 释放时,会将 kbase\_va\_region 和 kbase\_mem\_phy\_alloc 的引用计数减成0,从而释放两个对象 如果是 SAME\_VA 的情况如下,区别在于 SAME\_VA 内存在 kbase\_api\_mem\_alloc 中会调用 vm\_mmap 把 reg 同时映射到 CPU 和 GPU 侧,这就需要增加对应的引用计数(va\_refcnt、kref、gpu\_mappings),然后在 munmap 进程 VA 时,减少对应的引用计数 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-16c36b5ed2f7666b5202bf30159358ad805c81db.png) 对驱动的对象管理有大概的认知后,具体看下漏洞相关的两个接口 kbase\_api\_mem\_alias 和 kbase\_api\_mem\_flags\_change,分别利用的功能: - kbase\_api\_mem\_alias:创建别名映射,即新分配一个 reg 与其他已有的 reg 共享 kbase\_mem\_phy\_alloc - kbase\_api\_mem\_flags\_change:释放 kbase\_mem\_phy\_alloc 中的物理页 kbase\_api\_mem\_alias 的关键代码如下: ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-8ced5b9e1cda9372b4a6a61cc661da03b849947d.png) 主要是增加了 alloc 的引用计数 (kref),然后将其放入 kctx->pending\_regions,之后进程再通过 mmap 完成 CPU 和 GPU 映射 (kbase\_context\_mmap) ```c if (reg->gpu_alloc->type == KBASE_MEM_TYPE_ALIAS) { u64 const stride = alloc->imported.alias.stride; for (i = 0; i < alloc->imported.alias.nents; i++) { // 映射 aliased 中的各个 alloc 并增加 gpu_mappings if (alloc->imported.alias.aliased[i].alloc) { err = kbase_mmu_insert_pages(kctx->kbdev, &kctx->mmu, reg->start_pfn + (i * stride), alloc->imported.alias.aliased[i].alloc->pages + alloc->imported.alias.aliased[i].offset, alloc->imported.alias.aliased[i].length, reg->flags & gwt_mask, kctx->as_nr, group_id); kbase_mem_phy_alloc_gpu_mapped(alloc->imported.alias.aliased[i].alloc); } } ``` 创建别名映射进程调用 mmap 前后, reg 对象相关引用情况如下: ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-d21edf59880d3f69dcbe4a2a66cf051ca2cb3284.png) 在 kbase\_api\_mem\_alias 里面增加 aliased\[i\]->kref 确保其在使用过程中不会被释放,然后 kbase\_mmap 映射内存时增加 aliased\[i\]->gpu\_mappings 记录其被 GPU 映射的次数,同时增加 reg->va\_refcnt 记录其被 CPU 映射的次数,这个流程是没有问题的,通过引用计数确保 aliased 中的对象不会释放。 问题就出在 kbase\_api\_mem\_flags\_change 能在不释放 alloc 时释放其中的物理页: ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-5b4c8029f1370cf3cc765d2b416e7f6d27110189.png) kbase\_api\_mem\_flags\_change 可以利用 kbase\_mem\_evictable\_make 将 gpu\_alloc 放到驱动自己管理的链表中(kctx->evict\_list)当内核指向 shrink 操作时驱动会释放该链表上挂的所有 gpu\_alloc。 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-4c95d21164272cfbe07c6f1f75067c1d06813c80.png) shrink 时释放 kbase\_mem\_phy\_alloc 物理页的代码: ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-3a5e552301da301493604835381176c17a8a88fb.png) kbase\_mem\_flags\_change 在调用 kbase\_mem\_evictable\_make 前会校验 gpu\_mappings ,大概意思是如果这个 reg 被 GPU 多次映射了就不能执行物理内存释放操作,但是回到 alias 的流程,在 kbase\_api\_mem\_alias 结束后,aliased 数组中的 gpu\_mappings 还是 1 此时调用 kbase\_mem\_flags\_change 将 aliased\[i\] 放到 kctx->evict\_list,此时 alloc->pages 里面的值没有变化 然后再调用 mmap 映射 kbase\_mem\_alias 创建的 reg 将 aliased\[i\] 中的物理页(alloc->pages)映射到 GPU 侧,假设为映射的 VA 为 ALIAS\_VA 最后触发 shrink 机制,释放 aliased\[i\] 中的物理页,之后 ALIAS\_VA 还指向已经释放的物理页,导致物理页 UAF. ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-3d68efe2075f67179a93403c9c8a4cbf47b60f9a.png) 再次回顾漏洞根因,漏洞是驱动在建立 别名映射时对 gpu\_mappings 的管理不当,结合 kbase\_api\_mem\_flags\_change 释放物理页的逻辑,达成物理页 UAF,这种漏洞的挖掘个人理解需要先分析内存对象(堆、物理内存)的生命周期,然后组合各个 API 的时序看是否会产生非预期行为,重点还是对象的释放、申请、复制等逻辑。 物理页 UAF 的漏洞利用技术目前已经比较成熟,这里列几个常用的方式: - 篡改进程页表:通过 fork + mmap 大量分配进程页表占位释放的物理页,然后通过 GPU 修改页表实现任意物理内存读写 - 篡改 GPU 页表:通过 GPU 驱动接口分配 GPU 页表占位释放的物理页,然后通过 GPU 修改页表实现任意物理内存读写 - 篡改内核对象:通过喷射内核对象(比如 task\_struct、cred)占位,然后 GPU 修改对象实现利用 CVE-2022-46395 -------------- 前面两个漏洞的利用路径大概是:发现一个新漏洞,挖掘一种新漏洞利用方式完成利用,本节这个漏洞则是将漏洞转换为 CVE-2021-28663 ,因为 28663 的能力确实太强大了,物理页 UAF 的利用简单、直接,目前堆上的漏洞利用也逐步往物理页UAF 转换([Dirty Pagetable](https://yanglingxi1993.github.io/dirty_pagetable/dirty_pagetable.html)) 漏洞是一个条件竞争漏洞,kbase\_vmap\_prot 后其他线程可以释放 mapped\_evt 对应的物理页 ```c static int kbasep_write_soft_event_status( struct kbase_context *kctx, u64 evt, unsigned char new_status) { ... mapped_evt = kbase_vmap_prot(kctx, evt, sizeof(*mapped_evt), KBASE_REG_CPU_WR, &map); //Race window start if (!mapped_evt) return -EFAULT; *mapped_evt = new_status; //Race window end kbase_vunmap(kctx, &map); return 0; } ``` 为了扩大 race 的时间窗,作者利用 [timerfd 时钟中断技术](https://googleprojectzero.blogspot.com/2022/03/racing-against-clock-hitting-tiny.html) ```c migrate_to_cpu(0); //<------- pin this task to a cpu int tfd = timerfd_create(CLOCK_MONOTONIC, 0); //<----- creates timerfd //Adds epoll watchers int epfds[NR_EPFDS]; for (int i=0; i<NR_EPFDS; i++) epfds[i] = epoll_create1(0); for (int i=0; i<NR_EPFDS; i++) { struct epoll_event ev = { .events = EPOLLIN }; epoll_ctl(epfd[i], EPOLL_CTL_ADD, fd, &ev); } timerfd_settime(tfd, TFD_TIMER_ABSTIME, ...); //<----- schedule tfd to be available at a later time ioctl(mali_fd, KBASE_IOCTL_SOFT_EVENT_UPDATE,...); //<---- tfd becomes available and interrupts this ioctl ``` 大致思路就是在 kbase\_vmap\_prot 和 \*mapped\_evt 之间出发时钟中断,从而扩大时间窗,在两步之间释放 mapped\_evt 对应的物理页,就能够达到物理页 UAF 的能力。 ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-3518bc1e8c26aead6b8e35c03c08623aea34e88f.png) mapped\_evt 在页内的偏移可控,写的内容为 0 或者 1,总结下来漏洞的原语是物理内存 UAF 写,写的值只能 0 或者 1 ```c static inline struct kbase_mem_phy_alloc *kbase_alloc_create( struct kbase_context *kctx, size_t nr_pages, enum kbase_memory_type type, int group_id) { ... size_t alloc_size = sizeof(*alloc) + sizeof(*alloc->pages) * nr_pages; ... /* Allocate based on the size to reduce internal fragmentation of vmem */ if (alloc_size > KBASE_MEM_PHY_ALLOC_LARGE_THRESHOLD) alloc = vzalloc(alloc_size); else alloc = kzalloc(alloc_size, GFP_KERNEL); ... } ``` kbase\_alloc\_create 分配 kbase\_mem\_phy\_alloc 时会调用 vzalloc 分配内存,vzalloc 的逻辑是根据大小计算分配的物理页数目,然后逐次调用 alloc\_page 分配物理页,利用这个逻辑可以比较快速的占位刚刚释放的物理页(slab cross cache 时间相对较长) 根据之前的漏洞分析,我们知道 gpu\_mappings 控制的物理页的释放,如果通过 UAF 将其修改为 0 或者 1,就能提前释放一个被别名映射的 kbase\_mem\_phy\_alloc 中的物理页,导致物理页UAF ```c struct kbase_mem_phy_alloc { struct kref kref; atomic_t gpu_mappings; size_t nents; struct tagged_addr *pages; struct list_head mappings; ``` ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-688f2ebaffb518cc3a1d5ea3b7736b0c5c85ccc0.png) [利用非GPU漏洞攻击 GPU](https://www.youtube.com/watch?v=Mw6iCqjOV9Q&list=PLiIDIO1Gp6V_I4mSvz8WDfLVt6xy85RvP&index=6) ------------------------------------------------------------------------------------------------------------- 前面的案例都是利用 GPU 自身漏洞,这个案例则是将其他驱动、模块漏洞(摄像头驱动的 堆溢出漏洞) 的漏洞 转换为 GPU 漏洞,进而实现物理页 UAF 漏洞,核心思路与 CVE-2022-46395 一致,就是篡改 kbase\_mem\_phy\_alloc 的 gpu\_mappings 为 0,构造物理页 UAF ```c static inline struct kbase_mem_phy_alloc *kbase_alloc_create( struct kbase_context *kctx, size_t nr_pages, enum kbase_memory_type type, int group_id) { ... size_t alloc_size = sizeof(*alloc) + sizeof(*alloc->pages) * nr_pages; ... alloc = kzalloc(alloc_size, GFP_KERNEL); ... } ``` 一个比较有意思的点是研究员发现及时安卓内核启用了 MTE,仍然有 50% 的概率能够完成溢出而不被检测,且及时 MTE 检测到溢出,也不会导致内核 Panic,只是杀掉用户进程,这样就给了攻击者无限尝试的能力,相当于 Bypass 了 MTE. ![image.png](https://shs3.b.qianxin.com/attack_forum/2024/11/attach-40047e43dfc4f1b8d4447bae9dfe70f6140e6553.png) 总结 -- 从 CVE-2021-28663/CVE-2021-28664 开始研究人员逐渐重视并投入到 GPU 驱动安全领域,从一开始的挖掘 GPU 特有漏洞,到后面逐步将各种通用漏洞往 GPU 漏洞上转换,核心原因在于 GPU 驱动本身的能力太强大了,只要能够控制 GPU硬件的页表,就能实现任意物理页的读写,而且由于是独立的硬件,可以直接 Bypass 掉 CPU 侧的安全特性(比如 KNOX、PAC、MTE),Patch 内核代码。 GPU 安全研究还带来了另一个漏洞利用方向,GPU 驱动由于要管理物理内存,所以容易出现物理内存 UAF,物理 UAF 的利用手段被发掘后,大家发现这个原语实在太强大了,后面涌现了很多将不同漏洞转换为物理页UAF的技术,比如 [Dirty Pagetable](https://yanglingxi1993.github.io/dirty_pagetable/dirty_pagetable.html)、USMA、 pipe\_buffer->page 指针劫持等。 从 GPU 攻击的路径来看,也可以了解到一点,即漏洞的修复并不代表漏洞生命的结束,如果一个漏洞的原语过于强大、好用,就可以考虑将其他漏洞往这上面转换,从而复用历史的漏洞利用经验。 参考链接 ---- - [Internal of the Android kernel backdoor vulnerability](https://vul.360.net/archives/263) - [Two bugs with one PoC: Roo2ng Pixel 6 from Android 12 to Android 13](https://i.blackhat.com/Asia-23/AS-23-WANG-Two-bugs-with-one-PoC-Rooting-Pixel-6-from-Android-12-to-Android-13.pdf) - [The inside story of our CVE-2019-2025 exploit](https://labs.bluefrostsecurity.de/files/OffensiveCon2020_bug_collision_tale.pdf) - <https://www.blackhat.com/docs/eu-16/materials/eu-16-Taft-GPU-Security-Exposed.pdf> - [Rooting with root cause: finding a variant of a Project Zero bug](https://github.blog/2023-05-25-rooting-with-root-cause-finding-a-variant-of-a-project-zero-bug/) - [Off-By-One 2024 Day 1 - GPUAF Using a general GPU exploit tech to attack Pixel8](https://www.youtube.com/watch?v=Mw6iCqjOV9Q&list=PLiIDIO1Gp6V_I4mSvz8WDfLVt6xy85RvP&index=6)
发表于 2024-12-06 09:00:00
阅读 ( 894 )
分类:
二进制
0 推荐
收藏
0 条评论
请先
登录
后评论
hac425
19 篇文章
×
发送私信
请先
登录
后发送私信
×
举报此文章
垃圾广告信息:
广告、推广、测试等内容
违规内容:
色情、暴力、血腥、敏感信息等内容
不友善内容:
人身攻击、挑衅辱骂、恶意行为
其他原因:
请补充说明
举报原因:
×
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!