diff options
author | Johannes Stoelp <johannes.stoelp@gmail.com> | 2025-03-14 01:07:06 +0100 |
---|---|---|
committer | Johannes Stoelp <johannes.stoelp@gmail.com> | 2025-03-14 01:07:06 +0100 |
commit | d45fbd7b57db7aaad55bdac0a28b47cec33d3c86 (patch) | |
tree | 7c47bb6ba68402c2278d5b51472d1935a84edc6c /src | |
parent | b534253a32b89136b486fbe1b15f645f549b89df (diff) | |
download | notes-d45fbd7b57db7aaad55bdac0a28b47cec33d3c86.tar.gz notes-d45fbd7b57db7aaad55bdac0a28b47cec33d3c86.zip |
x86: gs user-space example
Diffstat (limited to 'src')
-rw-r--r-- | src/arch/x86/seg/Makefile | 5 | ||||
-rw-r--r-- | src/arch/x86/seg/seg.c | 75 | ||||
-rw-r--r-- | src/arch/x86_64.md | 9 |
3 files changed, 89 insertions, 0 deletions
diff --git a/src/arch/x86/seg/Makefile b/src/arch/x86/seg/Makefile new file mode 100644 index 0000000..290025e --- /dev/null +++ b/src/arch/x86/seg/Makefile @@ -0,0 +1,5 @@ +a.out: seg.c + gcc -g seg.c + +clean: + $(RM) a.out diff --git a/src/arch/x86/seg/seg.c b/src/arch/x86/seg/seg.c new file mode 100644 index 0000000..b06e0cb --- /dev/null +++ b/src/arch/x86/seg/seg.c @@ -0,0 +1,75 @@ +#include <stdint.h> +#include <stdio.h> + +// Reads GS segment base address (backed by IA32_GS_BASE msr). +// +// Instruction must be supported. Can be checked as +// cpuid(0x7).ebx & 1 == 1 +// +// In case the instruction is not available, the base address can be +// read with the following syscall. +// uint64_t addr; +// arch_prctl(ARCH_GET_GS, &addr) +static inline uint64_t rdgsbase() { + uint64_t gs; + asm volatile("rdgsbase %0" : "=r"(gs)); + return gs; +} + +// Write GS segment base address (backed by IA32_GS_BASE msr). +// +// Instruction must be supported. Can be checked as +// cpuid(0x7).ebx & 1 == 1 +// +// In case the instruction is not available, the base address can be +// set with the following syscall. +// arch_prctl(ARCH_GET_GS, addr) +static inline void wrgsbase(uint64_t* gs) { + asm volatile("wrgsbase %0" ::"r"(gs)); +} + +// Read u32 from off relative to GS segment. +static inline uint32_t rdgs32(int64_t off) { + uint32_t ret; + asm volatile("mov %%gs:%1, %0" : "=r"(ret) : "m"(*(uint64_t*)off)); + return ret; +} + +// Read u64 from off relative to GS segment. +static inline uint64_t rdgs64(int64_t off) { + uint64_t ret; + asm volatile("mov %%gs:%1, %0" : "=r"(ret) : "m"(*(uint64_t*)off)); + return ret; +} + +// Write u32 to off relative to GS segment. +static inline void wrgs32(int64_t off, uint32_t val) { + asm volatile("mov %1, %%gs:%0" ::"m"(*(uint64_t*)off), "r"(val)); +} + +int main() { + printf("gsbase=%016lx\n", rdgsbase()); + + uint64_t VALS[] = { + 0xa7a6a5a4a3a2a1a0, + 0xb7b6b5b4b3b2b1b0, + }; + + // Setup GS segment base address. + wrgsbase(VALS); + printf("gsbase=%016lx | &VALS=%p\n", rdgsbase(), VALS); + + // Read values relative to GS segment. + int64_t off[] = {0, 8, 4, 1}; + for (int64_t i = 0; i < 4; ++i) { + printf("64: gs:%ld=%016lx\n", off[i], rdgs64(off[i])); + printf("32: gs:%ld=%016lx\n", off[i], rdgs32(off[i])); + } + + // Write value relative to GS segment. + wrgs32(4, 0xddccbbaa); + printf("64: gs:%ld=%016lx\n", 0, rdgs64(0)); + + // Reset GS segment base address. + wrgsbase(0); +} diff --git a/src/arch/x86_64.md b/src/arch/x86_64.md index 0ca18cc..575870f 100644 --- a/src/arch/x86_64.md +++ b/src/arch/x86_64.md @@ -422,6 +422,15 @@ itself. [15:0] Length of GDT table. ``` +> In 64-bit mode the `{cs, ds, es, ss}` segment register have no +> effect, segmentation is effectively disabled. The `{gs, fs}` segment +> register however can still be used for segmented memory access in +> 64-bit with paging enabled. Segmentation takes place before VA -> PA +> address translation. +> +> The example in [seg.c](x86/seg/seg.c) shows how to set the `gs` base +> address and to relative accesses. + ## References - [SystemV AMD64 ABI][sysvabi] - [AMD64 Vol1: Application Programming][amd64_vol1] |