Skip to content

Latest commit

 

History

History
1815 lines (1008 loc) · 40.1 KB

eu-19-Lutas-Bypassing-KPTI-Using-The-Speculative-Behavior-Of-The-SWAPGS-Instruction.pdf.md

File metadata and controls

1815 lines (1008 loc) · 40.1 KB

Bypassing KPTI Using the Speculative Behavior of the SWAPGS Instruction (CVE-2019-1125) Andrei LUA ([email protected]) Dan LUA ([email protected])

AGENDA | Introduction | Explanation of side-channel Attacks | Speculative segmentation | Exploiting SWAPGS | Mitigations and recommendations

2

November 21, 2019

The Problem SWAPGS instruction can be executed speculatively

INSIDE USER SPACE a variant of Rogue System Register Load, allows to bypass KASLR and obtain the addresses of some kernel structures November 21, 2019

INSIDE KERNEL SPACE Speculatively executing code which require SWAPGS, but didn't execute it Speculatively executing code which does not require SWAPGS, but still executes it

· Speculative execution of the SWAPGS instruction leads to new Spectre variant · This attack allows an attacker to leak portions of the kernel memory · This attack bypases existing protective measures implemented for Spectre, Meltdown, L1TF, MDS, etc. · A patch has been issued by Microsoft November 21, 2019

SWAPGS Attack The effect: instructions which reference arbitrary memory can be executed speculatively The result: sensitive kernel memory can be accessed by an attacker

Affected CPUs and OSes · Affected CPUs: 64 bit Intel CPUs supporting SWAPGS and WRGSBASE instructions (Ivy Bridge and newer) · AMD CPUs ­ only kernel-space SWAPGS scenario 1 · Affected OSes: 64 bit Windows (tested on Windows 7 and newer); Linux seems very difficult, if not impossible, to exploit November 21, 2019

SIDE CHANNEL ATTACK

Cache Side-Channel Attack Allows the attacker to infer information by making careful measurements Example: testing if a variable has been previously accessed or not: · Measure how long it takes to access it now · If the access time is small ­ the variable has already been accessed · If the access time is high ­ the variable has not been accessed Such attack already demonstrated · prefetch attacks, TSX attacks, etc. November 21, 2019

Cache Side-Channel Attack

3 2 1

Multiple measurement techniques: · FLUSH + RELOAD · EVICT + TIME · PRIME + PROBE Each technique suitable for different scenarios They allow the attacker to probe for memory accesses made by a victim

November 21, 2019

Speculative Side-Channels Speculative side-channels ­ Spectre, Meltdown, L1TF, MDS, etc. This type of attacks allow information disclosure across arbitrary security boundaries (user-kernel, kernel-VMM, user-enclave, etc.) This type of attacks cannot be detected by modern software Mitigations involve microcode updates in the CPU or software mitigations made by the OS kernel/compilers November 21, 2019

Recap: Spectre · Modern CPUs use branch prediction · Whenever an if is encountered, the CPU will try to guess to correct branch direction (taken or not taken) · If the guess is correct, execution proceeds normally · If the guess is not correct, the CPU has to discard all executed instructions, and begin executing the correct branch November 21, 2019

Recap: Spectre if (a < b)

true

false

foo()

bar()

Simple if statement, with two branches

November 21, 2019

Recap: Spectre if (a < b)

a = 2, b = 1

true

false

foo()

bar()

  1. The true branch is predicted, foo() is executed speculatively

November 21, 2019

Recap: Spectre if (a < b)

a = 2, b = 1

true

false

foo()

bar()

  1. CPU detects mispredicted branch, discards everything executed on true branch...

November 21, 2019

Recap: Spectre if (a < b)

a = 2, b = 1

true

false

foo()

bar()

  1. ... and then executes the false branch, which is the correct path

November 21, 2019

Recap: Spectre if (a < b)

a = 1, b = 2

true

false

foo()

bar()

  1. The true branch is predicted, foo() is executed speculatively

November 21, 2019

Recap: Spectre if (a < b)

a = 1, b = 2

true

false

foo()

bar()

  1. The branch was predicted correctly, execution continues normally

November 21, 2019

Recap: Meltdown · Modern OSes use hardware mechanisms for preventing user-mode code access (R/W/E) to kernelmode code and data · When accessing memory, the CPU translates a Virtual Address into a Physical Address by consulting the Page Tables · Page Tables are under the OS control · OS configures Page Tables to isolate ring-3 (UM) from accessing ring-0 (KM) data · e.g. A virtual address cannot be used in UM to access physical memory, if the Page Table Entries (PTEs) translating that virtual address don't have the User/Supervisor (U/S) flag set (U/S = 1) · Classical Meltdown abuses that ring-3 out-of-order execution of memory-load instructions can temporary access kernel-mode data, if a VA translation exists in the process Page Tables for that kernelmode address. · Even is U/S bit in PTE is 0 November 21, 2019

Recap: Meltdown · Mitigations : in hardware (newer CPUs) or in software (at OS level) · Software mitigations (KPTI, KVAShadow) un-map the kernel address space while the CPU executes in ring-3. · Split Page Tables : one used when executing in Ring-3 (no KM VA translations), another one used when executing in Ring-0 (KM VA translations and UM VA translations) · In UM no VA translation present for a KM address => no leak during out-of-order loads · While the CPU executes in ring-0, on Windows OSes the KM Page Tables map the user-mode address space November 21, 2019

SPECULATIVE SEGMENTATION

Speculative segmentation · A closer look at x86 Virtual Address Translation reveals that segmentation is also involved in translating a Virtual Address (VA) to a Physical Address (PA) N o v e m b e r 2 1 , 2 0 1 9Picture taken from page 90 of "Intel® 64 and IA-32 Architectures Software Developer's Manual Volume 3 (3A, 3B, 3C & 3D): System Programming Guide, Order Number: 325384-069US January 2019"

Speculative segmentation · Segment registers (CS, DS, SS, ES, FS, GS) cache the Segment Base and Access Rights into a hidden portion whenever a new segment descriptor is loaded (e.g. mov ds, SegmentSelector) · On x64, the FS and GS segment register bases are physically mapped to MSRs · FS segment register mapped to IA32_FS_BASE MSR · GS segment register mapped to IA32_GS_BASE · On Windows x64 the GS segment is used to access · per-thread information in ring-3 (Thread Environment Block ­ TEB), address of TEB typically stored in IA32_GS_BASE MSR · per-cpu information in ring-0 (Kernel Processor Control Region ­ KPCR), address of KPCR typically stored in IA32_KERNEL_GS_BASE MSR November 21, 2019

Speculative segmentation · On user to kernel transitions, the kernel uses the swapgs instruction to exchange the contents of IA32_GS_BASE MSR with the IA32_KERNEL_GS_BASE MSR · So further memory accesses via gs segment overrides will point into KPCR · On kernel to user transitions, before returning control in usermode, the kernel again uses swapgs · So further memory accesses via gs segment overrides will point into TEB November 21, 2019

Speculative segmentation · We studied the security implications of speculatively executing segmentation related instructions on x86 CPUs Key Technical Finding 1 During speculative execution, after loading GS or FS segment registers with an invalid segment selector, and then subsequently using that segment in further speculatively executed memoryaccessing instructions, Intel® CPUs use the previously stored segment base address of the segment register to compute the linear address used for memory addressing Key Technical Finding 2 A value written speculatively to the base of a segment register survives instruction retirement and can be later retrieved by loading an invalid segment selector into that segment register Technical white-paper available at bitdefender.com/SWAPGSAttack November 21, 2019

Speculative segmentationsecurity implications

  1. Subverting KASLR
  2. Leaking general-purpose register contents across privilege boundaries (ring3-ring0) · Only in absence of SMEP, SMAP and RSB Stuffing mitigations (they are present on Linux / Windows)
  3. Retrieving FS.Base value explicitly written inside of an Intel® SGX enclave · Admittedly minor issue
  4. Store-to-Load Forwarding on segment descriptor loads · Allows bypass segment base & limit checks · Minor impact, since these segmentation checks are already disabled in X64 mode
  5. Potential KPTI / KVAShadow bypass · Only if we can force speculative execution of segment loading instructions in kernel-mode Technical white-paper available at bitdefender.com/SWAPGSAttack November 21, 2019

Speculative segmentation ­ Subverting KASLR

  1. Retrieving the IA32_KERNEL_GS_BASE MSR value from User-Mode · On Windows, this gives us the pointer to KPCR (Kernel Processor Control Region) Let's assume we execute code, in ring-3, in an unprivileged process on Windows X64. We have : Current GS.base = IA32_GS_BASE, MSR value is the pointer to the TEB Previous GS.base = IA32_KERNEL_GS_BASE, MSR value is the pointer to KPCR (as a result from prior execution in kernel-mode) November 21, 2019

Speculative segmentation ­ Subverting KASLR

  1. Retrieving the IA32_KERNEL_GS_BASE MSR value from User-Mode · On Windows, this gives us the pointer to KPCR (Kernel Processor Control Region)

November 21, 2019

; RDX = probe buffer for Flush + Reload, ; RCX = offset of the byte we want to retrieve

leak_kernel_gs_base_byte PROC FRAME .endprolog mov rax, [0] mov r9d, 0xFFFF mov gs, r9d rdgsbase rax shr rax, cl and rax, 0xFF shl rax, 0xC mov rax, qword [rdx + rax] ret leak_kernel_gs_base_byte ENDP

;(1) ;(2) ;(3) ;(4) ;(5) ;(6) ;(7) ;(8)

Speculative segmentation ­ Subverting KASLR We force a page fault, thus forcing the CPU to speculatively execute the next instructions

November 21, 2019

; RDX = probe buffer for Flush + Reload, ; RCX = offset of the byte we want to retrieve

leak_kernel_gs_base_byte PROC FRAME .endprolog mov rax, [0] mov r9d, 0xFFFF mov gs, r9d rdgsbase rax shr rax, cl and rax, 0xFF shl rax, 0xC mov rax, qword [rdx + rax] ret leak_kernel_gs_base_byte ENDP

;(1) ;(2) ;(3) ;(4) ;(5) ;(6) ;(7) ;(8)

Speculative segmentation ­ Subverting KASLR Next, we load an invalid segment selector into r9d, way outside the GDT Limit

November 21, 2019

; RDX = probe buffer for Flush + Reload, ; RCX = offset of the byte we want to retrieve

leak_kernel_gs_base_byte PROC FRAME .endprolog mov rax, [0] mov r9d, 0xFFFF mov gs, r9d rdgsbase rax shr rax, cl and rax, 0xFF shl rax, 0xC mov rax, qword [rdx + rax] ret leak_kernel_gs_base_byte ENDP

;(1) ;(2) ;(3) ;(4) ;(5) ;(6) ;(7) ;(8)

Speculative segmentation ­ Subverting KASLR We then load the invalid segment selector into the GS segment register. This causes a fault, since the descriptor points outside the GDT limit

November 21, 2019

; RDX = probe buffer for Flush + Reload, ; RCX = offset of the byte we want to retrieve

leak_kernel_gs_base_byte PROC FRAME .endprolog mov rax, [0] mov r9d, 0xFFFF mov gs, r9d rdgsbase rax shr rax, cl and rax, 0xFF shl rax, 0xC mov rax, qword [rdx + rax] ret leak_kernel_gs_base_byte ENDP

;(1) ;(2) ;(3) ;(4) ;(5) ;(6) ;(7) ;(8)

Speculative segmentation ­ Subverting KASLR We use rdgsbase to read the GS.BASE value, expecting RAX to contain IA32_GS_BASE (since we are in user-mode), but it contains instead IA32_KERNEL_GS_BASE MSR

November 21, 2019

; RDX = probe buffer for Flush + Reload, ; RCX = offset of the byte we want to retrieve

leak_kernel_gs_base_byte PROC FRAME .endprolog mov rax, [0] mov r9d, 0xFFFF mov gs, r9d rdgsbase rax shr rax, cl and rax, 0xFF shl rax, 0xC mov rax, qword [rdx + rax] ret leak_kernel_gs_base_byte ENDP

;(1) ;(2) ;(3) ;(4) ;(5) ;(6) ;(7) ;(8)

Speculative segmentation ­ Subverting KASLR Isolate the byte we want to retrieve from the pointer value, and access the Flush+Reload probe buffer with the respective byte

November 21, 2019

; RDX = probe buffer for Flush + Reload, ; RCX = offset of the byte we want to retrieve

leak_kernel_gs_base_byte PROC FRAME .endprolog mov rax, [0] mov r9d, 0xFFFF mov gs, r9d rdgsbase rax shr rax, cl and rax, 0xFF shl rax, 0xC mov rax, qword [rdx + rax] ret leak_kernel_gs_base_byte ENDP

;(1) ;(2) ;(3) ;(4) ;(5) ;(6) ;(7) ;(8)

Speculative segmentation ­ Subverting KASLR

  1. Retrieving values from KPCR via classical Meltdown · On Windows, KPCR is mapped in user-mode page tables, so it is accessible via Meltdown

;; RDX : offset of the byte inside KPCR we want to access ;; RCX : probe buffer (for Flush + Reload)

leak_byte_from_kpcr PROC FRAME

.endprolog

mov rax, [0] ; (1)

push 018H

; (2) 0x18 is on Win10 x64 RS4 the ; selector for Data Segment,DPL = 0

pop gs

; (3)

movzx rax, byte ptr gs:[rdx] ; (4)

shl rax, 0CH ; (5)

mov r8, qword ptr [rcx + rax] ; (6)

ret

leak_byte_from_kpcr ENDP

November 21, 2019

Speculative segmentation ­ Subverting KASLR As before, we load an invalid (for ring-3) segment selector into GS, forcing a GP

;; RDX : offset of the byte inside KPCR we want to access ;; RCX : probe buffer (for Flush + Reload)

leak_byte_from_kpcr PROC FRAME

.endprolog

mov rax, [0] ; (1)

push 018H

; (2) 0x18 is on Win10 x64 RS4 the ; selector for Data Segment,DPL = 0

pop gs

; (3)

movzx rax, byte ptr gs:[rdx] ; (4)

shl rax, 0CH ; (5)

mov r8, qword ptr [rcx + rax] ; (6)

ret

leak_byte_from_kpcr ENDP

November 21, 2019

Speculative segmentation ­ Subverting KASLR

When accessing memory through the invalid selector, because these instructions execute speculatively and access checking is delayed until instruction retirement, and because a valid translation exists in the process page-tables for the KPCR, the value of the byte in the KPCR is successfully retri;e;veRdDXin:toofrafxset of the byte inside KPCR we want to access ;; RCX : probe buffer (for Flush + Reload)

leak_byte_from_kpcr PROC FRAME

.endprolog

mov rax, [0] ; (1)

push 018H

; (2) 0x18 is on Win10 x64 RS4 the ; selector for Data Segment,DPL = 0

pop gs

; (3)

movzx rax, byte ptr gs:[rdx] ; (4)

shl rax, 0CH ; (5)

mov r8, qword ptr [rcx + rax] ; (6)

ret

leak_byte_from_kpcr ENDP

November 21, 2019

Speculative segmentation ­ Subverting KASLR 2. Retrieving values from KPCR via classical Meltdown · On Windows, KPCR is mapped in user-mode page tables, so it is accessible via Meltdown Conclusion : Reading arbitrary values from KPCR allows us to retrieve pointers into NTOSKRNL, thus subverting Windows KASLR November 21, 2019

Speculative segmentation ­ Insights Key Insight : since during speculative execution of segmentation instructions in Ring-3 we have access to IA32_KERNEL_GS_BASE and to kernel data, it follows that during speculative execution of segmentation instructions in Ring-0, the kernel would access IA32_GS_BASE and user-mode data Key Insight : if we can force speculative execution of segmentation instructions in Ring-0 after the kernel moved to its Kernel Page Tables, we may be able to force it to access arbitrary kernel memory, now that this memory is mapped Research Question : can we find instances of such instructions in modern OS kernels, and can we identify speculatively-executed gadgets around them that would leak arbitrary kernel memory, thus defeating KPTI ? November 21, 2019

SWAPGS INTRODUCTION

SWAPGS instruction

· System instruction introduced in x86-64 architecture · Segmentation mostly disabled in 64 bit · Segment bases default to 0 for all segments, except FS and GS · The bases of these registers are stored in MSRs: IA32_FS_BASE and IA32_GS_BASE · An additional MSR exists: IA32_KERNEL_GS_BASE · SWAPGS exchanges the values of IA32_GS_BASE and IA32_KERNEL_GS_BASE

November 21, 2019

The SWAPGS Attack allows an attacker to leak portions of the kernel memory even if patched against existing speculative side-channel attacks. November 21, 2019

EXPLOITING SWAPGS

The SWAPGS attack: scenario 1

First scenario: code which would require SWAPGS to be executed does not actually execute it (due to a mispredicted branch)

1

Harder to exploit (especially on Windows) ­ not executing SWAPGS would also not load the kernel

CR3, and thus would not allow for KPTI bypass

November 21, 2019

The SWAPGS attack: scenario 1

nt!KiPageFaultShadow:

f644241001

test

7462

je

0f01f8

swapgs

650fba24251870000001 bt

720c

jb

65488b242500700000 mov

0f22dc

mov

...

fault_from_kernel:

...

byte ptr [rsp+10h],1 fault_from_kernel dword ptr gs:[7018h],1 nt!KiPageFaultShadow+0x22 rsp,qword ptr gs:[7000h] cr3,rsp

November 21, 2019

The SWAPGS attack: scenario 1

nt!KiPageFaultShadow:

f644241001

test

7462

je

0f01f8

swapgs

650fba24251870000001 bt

720c

jb

65488b242500700000 mov

0f22dc

mov

...

fault_from_kernel:

...

byte ptr [rsp+10h],1 fault_from_kernel dword ptr gs:[7018h],1 nt!KiPageFaultShadow+0x22 rsp,qword ptr gs:[7000h] cr3,rsp

November 21, 2019

The SWAPGS attack: scenario 1

nt!KiPageFaultShadow:

f644241001

test

7462

je

0f01f8

swapgs

650fba24251870000001 bt

720c

jb

65488b242500700000 mov

0f22dc

mov

...

fault_from_kernel:

...

byte ptr [rsp+10h],1 fault_from_kernel dword ptr gs:[7018h],1 nt!KiPageFaultShadow+0x22 rsp,qword ptr gs:[7000h] cr3,rsp

November 21, 2019

The SWAPGS attack: scenario 1

nt!KiPageFaultShadow:

f644241001

test

7462

je

0f01f8

swapgs

650fba24251870000001 bt

720c

jb

65488b242500700000 mov

0f22dc

mov

...

fault_from_kernel:

...

byte ptr [rsp+10h],1 fault_from_kernel dword ptr gs:[7018h],1 nt!KiPageFaultShadow+0x22 rsp,qword ptr gs:[7000h] cr3,rsp

November 21, 2019

The SWAPGS attack: scenario 1

nt!KiPageFaultShadow:

f644241001

test byte ptr [rsp+10h],1

7462

je

fault_from_kernel

0f01f8

swapgs

650fba24251870000001 bt

dword ptr gs:[7018h],1

720c

jb

nt!KiPageFaultShadow+0x22

65488b242500700000 mov

rsp,qword ptr gs:[7000h]

0f22dc

mov

cr3,rsp

...

fault_from_kernel:

...

November 21, 2019

The SWAPGS attack: scenario 2

Second scenario: the SWAPGS instruction gets executed speculatively, even if it shouldn't

This allows code to be speculatively executed with

2

the user GS active

Easily exploitable

November 21, 2019

The SWAPGS attack: scenario 2

f60596a1390001

test

7503

jne

0f01f8

swapgs

654c8b142588010000 mov

65488b0c2588010000 mov

488b8920020000

mov

488b8930080000

mov

6548890c2570020000 mov

byte ptr [nt!KiKvaShadow],1 skip_swapgs r10,qword ptr gs:[188h] rcx,qword ptr gs:[188h] rcx,qword ptr [rcx+220h] rcx,qword ptr [rcx+830h] qword ptr gs:[270h],rcx

November 21, 2019

The SWAPGS attack: scenario 2

f60596a1390001

test

7503

jne

0f01f8

swapgs

654c8b142588010000 mov

65488b0c2588010000 mov

488b8920020000

mov

488b8930080000

mov

6548890c2570020000 mov

byte ptr [nt!KiKvaShadow],1 skip_swapgs r10,qword ptr gs:[188h] rcx,qword ptr gs:[188h] rcx,qword ptr [rcx+220h] rcx,qword ptr [rcx+830h] qword ptr gs:[270h],rcx

November 21, 2019

The SWAPGS attack: scenario 2

f60596a1390001

test

7503

jne

0f01f8

swapgs

654c8b142588010000 mov

65488b0c2588010000 mov

488b8920020000

mov

488b8930080000

mov

6548890c2570020000 mov

byte ptr [nt!KiKvaShadow],1 skip_swapgs r10,qword ptr gs:[188h] rcx,qword ptr gs:[188h] rcx,qword ptr [rcx+220h] rcx,qword ptr [rcx+830h] qword ptr gs:[270h],rcx

November 21, 2019

The SWAPGS attack: scenario 2

f60596a1390001

test

7503

jne

0f01f8

swapgs

654c8b142588010000 mov

65488b0c2588010000 mov

488b8920020000

mov

488b8930080000

mov

6548890c2570020000 mov

byte ptr [nt!KiKvaShadow],1 skip_swapgs r10,qword ptr gs:[188h] rcx,qword ptr gs:[188h] rcx,qword ptr [rcx+220h] rcx,qword ptr [rcx+830h] qword ptr gs:[270h],rcx

November 21, 2019

The SWAPGS attack: scenario 2

f60596a1390001

test

7503

jne

0f01f8

swapgs

654c8b142588010000 mov

65488b0c2588010000 mov

488b8920020000

mov

488b8930080000

mov

6548890c2570020000 mov

byte ptr [nt!KiKvaShadow],1 skip_swapgs r10,qword ptr gs:[188h] rcx,qword ptr gs:[188h] rcx,qword ptr [rcx+220h] rcx,qword ptr [rcx+830h] qword ptr gs:[270h],rcx

November 21, 2019

The SWAPGS attack: scenario 2

f60596a1390001

test

7503

jne

0f01f8

swapgs

654c8b142588010000 mov

65488b0c2588010000 mov

488b8920020000

mov

488b8930080000

mov

6548890c2570020000 mov

byte ptr [nt!KiKvaShadow],1 skip_swapgs r10,qword ptr gs:[188h] rcx,qword ptr gs:[188h] rcx,qword ptr [rcx+220h] rcx,qword ptr [rcx+830h] qword ptr gs:[270h],rcx

November 21, 2019

The SWAPGS attack: scenario 2

f60596a1390001

test

7503

jne

0f01f8

swapgs

654c8b142588010000 mov

65488b0c2588010000 mov

488b8920020000

mov

488b8930080000

mov

6548890c2570020000 mov

byte ptr [nt!KiKvaShadow],1 skip_swapgs r10,qword ptr gs:[188h] rcx,qword ptr gs:[188h] rcx,qword ptr [rcx+220h] rcx,qword ptr [rcx+830h] qword ptr gs:[270h],rcx

November 21, 2019

The SWAPGS attack: scenario 2

f60596a1390001

test

7503

jne

0f01f8

swapgs

654c8b142588010000 mov

65488b0c2588010000 mov

488b8920020000

mov

488b8930080000

mov

6548890c2570020000 mov

byte ptr [nt!KiKvaShadow],1 skip_swapgs r10,qword ptr gs:[188h] rcx,qword ptr gs:[188h] rcx,qword ptr [rcx+220h] rcx,qword ptr [rcx+830h] qword ptr gs:[270h],rcx

November 21, 2019

The SWAPGS attack: scenario 2 · On a Windows RS5 x64 kernel, there are exactly 38 such gadgets! · Most of them (and the most dangerous) are inside exception/interrupt handlers · Exception handlers are directly executable by the attacker (by generating a fault) November 21, 2019

SWAPGS variant 1: search for kernel values First variant: search for values in kernel memory The attacker can "brute force" or do a linear search of target addresses for a designated value November 21, 2019

We are inside user space, inside the attacker process

Attacker process

Kernel Space

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

We wish to search for a value in kernel, for example, 0x0000000300905A4D

Attacker process

Kernel Space

TIB November 21, 2019

0xFFFF800012340000 4D 5A 90 00 03 00 00 00

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Allocate memory in user-mode at the address 0x0000000300905000 ­ probe buffer

Attacker process

Kernel Space

0x300905000 TIB November 21, 2019

0xFFFF800012340000 4D 5A 90 00 03 00 00 00

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

We wish to see if offset 0xA4D+0x220 in this memory area gets cached

Attacker process

Kernel Space

0x300905000 0x300905C6D

0xBDBDBDBD

0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Make the IA32_GS_BASE point to the target address, using WRGSBASE (- 0x188)

Attacker process

Kernel Space

0x300905000 0x300905C6D

0xBDBD0xBAD4BDD

0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Issue a user-kernel transition (for example, by executing UD2)

Attacker process

Kernel Space

0x300905000 0x300905C6D

0xBDBD0xBAD4BDD

0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attacker process

0x300905000 0x300905C6D

0xBDBD0xBAD4BDD

One of the first things done in kernel - SWAPGS

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

Kernel Space

0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attacker process

0x300905000 0x300905C6D

0xBDBD0xBAD4BDD

Vulnerable gadget is hit swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

Kernel Space 0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

0x300905000 0x300905C6D

If the branch is mispredicted, the gadget starts executing speculatively

Attacker process 0xBDBD0xBAD4BDD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

Kernel Space 0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attacker process

0x300905000 0x300905C6D

0xBDBD0xBAD4BDD

GS bases are swapped swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

Kernel Space 0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

0x300905000 0x300905C6D

R10 is loaded with gs:[0x188], which contains the secret

Attacker process 0xBDBD0xBAD4BDD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx R10 = 0x0000000300905A4D

Kernel Space 0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

0x300905000 0x300905C6D

RCX is loaded with gs:[0x188], which contains the secret

Attacker process 0xBDBD0xBAD4BDD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx R10 = 0x0000000300905A4D RCX = 0x0000000300905A4D

Kernel Space 0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

0x300905000 0x300905C6D

RCX is loaded with the value located at the address represented by the secret (+ 0x220)

Attacker process 0xBDB0DxBAD4BDD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx R10 = 0x0000000300905A4D RCX = 0x00000000BDBDBDBD

Kernel Space 0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

The branch misprediction is eventually detected, and everything is reverted (including GS bases)

Attacker process

0x300905000 0x300905C6D

0xBDB0DxBAD4BDD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

Kernel Space 0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

0x300905000 0x300905C6D

... but the address represented by the secret (with a cache-line bias) remains cached

Attacker process 0xBDB0DxBAD4BDD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

Kernel Space 0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

When resuming back to user-mode, the attacker measures the access time to the 0x300905C6D, and sees it's cached

Attacker process

Kernel Space

0x300905000 0x300905C6D

0xBDB0DxBAD4BDD

0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

When resuming back to user-mode, the attacker measures the access time to the 0x300905C6D, and sees it's cached

Attacker process

Kernel Space

0x300905000 0x300905C6D

0xBDB0DxBAD4BDD

We now know the that address 0xFFFF800012340000 contains the value 0x0000000300905A4D, with a cache line bias ­ we see that probe address cached!

0xFFFF800012340000 4D 5A 90 00 03 00 00 00

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

SWAPGS variant 2: leak values at arbitrary addresses · Extended version of the variant 1 · Allows to attacker to leak portions of the kernel memory (we'll soon see what values, exactly) · Can be used to leak unknown values at target addresses November 21, 2019

We are inside user space, inside the attacker process

Attacker process

Kernel Space

TIB November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attacker process

A secret exists in kernel

Kernel Space

TIB November 21, 2019

0xFFFF800012345678 00 6C 65 61 6B 00 00 00

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attacker process

Allocate a Test Var & FLUSH it from the caches Kernel Space

0x12345678A39D

Test Var TIB

November 21, 2019

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Allocate 8MB probe buffer (note that the search will begin at address 0, and sequentially try several MB chunks; Attacker process for simplicity, we skip directly to the target address) Kernel Space

0x6B61000000 0x6B61800000

0x12345678A39D

Test Var TIB

November 21, 2019

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attacker process

Fill decoy with the address of Test Var (- 0x830) Kernel Space

0x6B61000000 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

0x12345678A39D

Test Var TIB

November 21, 2019

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Make the IA32_GS_BASE point to the target address, using WRGSBASE (- 0x188)

Attacker process

Kernel Space

0x6B61000000 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

0x12345678A39D

Test Var TIB

November 21, 2019

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Issue a user-kernel transition (for example, by executing UD2)

Attacker process

Kernel Space

0x6B61000000 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

0x12345678A39D

Test Var TIB

November 21, 2019

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attacker process

0x6B61000000 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

One of the first things done in kernel - SWAPGS Kernel Space swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx 00 6C 65 61 6B 00 00 00 0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attacker process

0x6B61000000 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

Vulnerable gadget is hit swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

Kernel Space 00 6C 65 61 6B 00 00 00 0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

0x6B61000000 0x6B61800000

If the branch is mispredicted, the gadget starts executing speculatively

Attacker process

Kernel Space

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

00 6C 65 61 6B 00 00 00

0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attacker process

0x6B61000000 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

GS bases are swapped swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

Kernel Space 00 6C 65 61 6B 00 00 00 0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

0x6B61000000 0x6B61800000

R10 is loaded with gs:[0x188], which contains the secret

Attacker process

Kernel Space

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx R10 = 0x0000006B61656C00

00 6C 65 61 6B 00 00 00

0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

0x6B61000000 0x6B61800000

RCX is loaded with gs:[0x188], which contains the secret

Attacker process

Kernel Space

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx R10 = 0x0000006B61656C00 RCX = 0x0000006B61656C00

00 6C 65 61 6B 00 00 00

0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

0x6B61000000 0x6B61656E20 0x6B61800000

RCX is loaded with the value located at the address represented by the secret (+ 0x220)

Attacker process

Kernel Space

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx R10 = 0x0000006B61656C00 RCX = 0x000012345678ABCD

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

RCX is loaded with the value located at the address represented by the test variable (+ 0x830)

Attacker process

Kernel Space

0x6B61000000 0x6B61656E20 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx R10 = 0x0000006B61656C00 RCX = Test Var

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

The branch misprediction is eventually detected, and everything is reverted (including GS bases)

Attacker process

Kernel Space

0x6B61000000 0x6B61656E20 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attacker process

0x6B61000000 0x6B61656E20 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

... but the Test Var remains cached swapgs ... test byte ptr [nt!KiKvaShadow],1 jne skip_swapgs swapgs mov r10,qword ptr gs:[188h] mov rcx,qword ptr gs:[188h] mov rcx,qword ptr [rcx+220h] mov rcx,qword ptr [rcx+830h] mov qword ptr gs:[270h],rcx

Kernel Space 00 6C 65 61 6B 00 00 00 0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

When resuming back to user-mode, the attacker measures the access time to the Test Var, and sees it's cached

Attacker process

Kernel Space

0x6B61000000 0x6B61656E20 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

0x12345678A39D

Test Var TIB

November 21, 2019

Test access time

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

When resuming back to user-mode, the attacker measures the access time to the Test Var, and sees it's cached

Attacker process

Kernel Space

0x6B61000000 0x6B61656E20 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

We now know the secret is between 0x6B61000000 and 0x6B61800000 Test access time

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

0x12345678A39D

Test Var TIB

November 21, 2019

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Now that the value of the secret is reduced to an 8MB range, we can "zoom in"

Attacker process

Kernel Space

0x6B61000000 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

0x12345678A39D

Test Var TIB

November 21, 2019

Test access time

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Repeat the attack with the test interval reduced in half

Attacker process

Kernel Space

0x6B61000000 0x6B61400000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

0x12345678A39D

Test Var TIB

November 21, 2019

Test access time

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Repeat the attack with the test interval reduced in half

Attacker process

Kernel Space

0x6B61400000 0x6B61656E20 0x6B61800000

0x12345678ABCD 0x12345678ABCD 0x12345678ABCD

0x12345678A39D

Test Var TIB

November 21, 2019

Test access time

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Continue to "zoom in" by halving the interval until the precision is enough (this is usually going to be a cache line)

Attacker process

Kernel Space

0x6B61656E20

0x12345678ABCD

0x12345678A39D

Test Var TIB

November 21, 2019

Test access time

00 6C 65 61 6B 00 00 00 0xFFFF800012345678

IA32_GS_BASE IA32_KERNEL_GS_BASE

KPCR

Attack details: Speculatively executing the gadget The most important part of the exploit is to force our gadget to execute speculatively WE MUST ACHIEVE TWO THINGS

  1. The branch before SWAPGS must be mispredicted 2. The KvaShadow variable must not be cached November 21, 2019

Attack details: Speculatively executing the gadget

  1. Force the branch misprediction Already demonstrated by others as well Simply spray a large enough memory area with similar branches The branches must go in the direction we wish to mispredict
  2. Evict KvaShadow variable from cache Cache thrashing can be used Simply access variables in user-mode which are located at addresses that conflict with the address of the KvaShadow (same page offset, same index) November 21, 2019

Attack details: Alignment & Cache Line Bias For a better attack efficiency and resolution

  1. The alignment of the secret must be accounted for 2. The leaked value will be within 64 bytes range November 21, 2019

Attack details: Alignment & Cache Line Bias

  1. Test Variable Alignment We spray the address of a test variable inside the probed memory area This address will be speculatively accessed inside the kernel, thus spoiling the value range However, we don't know the kernel value ­we don't know the accessed offset We can overcome this by trying the attack 8 times (one time for each possible alignment)
  2. Cache Line Bias The leaked values are biased towards a cache line They can be anywhere in a range of 64 bytes November 21, 2019

Attack details: Speed Most of the times, variant 1 takes less than 10 tries to trigger speculative execution inside kernel (< 1ms) November 21, 2019

Attack details: Speed Variant 2 is much slower, as it has to try many possible ranges for the secret value November 21, 2019

Attack details: Leakable domain This attack cannot leak the entire kernel memory space It can leak any value that resembles a valid user-mode address Normal values in [0, 0x00007FFFFFFFFFFF] If 57 bit addressing is used: values in [0, 0x007FFFFFFFFFFFFF]

Leaked kernel

33%

contents

51%

November 21, 2019

Mitigations · Clobber the user-mode GS base on context switches · Make use of SMAP (on CPUs which have meltdown patches) · Serialize the code which lies on SWAPGS path ­ both branches · The one that will not execute SWAPGS · The one that will execute SWAPGS · Use Hypervisor to dynamically instrument the kernel code & rewrite the vulnerable code sequences in order to mitigate the problem · Basically, serialize them, if not already done by the OS November 21, 2019

Conclusions · A new variant of Spectre has been disclosed: speculative execution (or the lack) of SWAPGS instruction can lead to KPTI bypass · We presented a novel exploitation technique relying on treating leaked data values as virtual addresses · Microsoft published patches for this issue in July 2019

bitdefender.com/SWAPGSAttack

Backup slides

Recap: Spectre v1 · Spectre v1: Bounds Check Bypass · if (index < size) x = array1[array2[index] * 4096]; · If the attacker controls index and array1, it could leak arbitrary data beyond the array2 November 21, 2019