aboutsummaryrefslogtreecommitdiffhomepage
path: root/arch/x86/seg/seg.c
blob: b06e0cb7af61e117f4181ef0a343629589c840c9 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
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);
}