:github_url: https://github.com/nyx-org/gaia .. _program_listing_file_platform_amd64_asm.hpp: Program Listing for File asm.hpp ================================ |exhale_lsh| :ref:`Return to documentation for file ` (``platform/amd64/asm.hpp``) .. |exhale_lsh| unicode:: U+021B0 .. UPWARDS ARROW WITH TIP LEFTWARDS .. code-block:: cpp /* SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include namespace Gaia::Amd64 { static inline void outb(uint16_t port, uint8_t data) { asm volatile("outb %0, %1" ::"a"(data), "Nd"(port)); } static inline uint8_t inb(uint16_t port) { uint8_t data; asm volatile("inb %1, %0" : "=a"(data) : "Nd"(port)); return data; } static inline void outw(uint16_t port, uint16_t data) { asm volatile("outw %0, %1" ::"a"(data), "Nd"(port)); } static inline uint16_t inw(uint16_t port) { uint16_t data; asm volatile("inw %1, %0" : "=a"(data) : "Nd"(port)); return data; } static inline void outl(uint16_t port, uint32_t data) { asm volatile("outl %0, %1" ::"a"(data), "Nd"(port)); } static inline uint32_t inl(uint16_t port) { uint32_t data; asm volatile("inl %1, %0" : "=a"(data) : "Nd"(port)); return data; } static inline uint64_t rdtsc(void) { uint32_t lo, hi; asm volatile("rdtsc" : "=a"(lo), "=d"(hi)); return ((uint64_t)hi << 32) | (uint64_t)lo; } static inline void cli(void) { asm volatile("cli"); } static inline void hlt(void) { asm volatile("hlt"); } static inline void sti(void) { asm volatile("sti"); } #define ASM_MAKE_CRN(N) \ static inline uint64_t read_cr##N(void) { \ uint64_t value = 0; \ asm volatile("mov %%cr" #N ", %0" : "=r"(value)); \ return value; \ } \ \ static inline void write_cr##N(uint64_t value) { \ asm volatile("mov %0, %%cr" #N ::"a"(value)); \ } ASM_MAKE_CRN(0) ASM_MAKE_CRN(1) ASM_MAKE_CRN(2) ASM_MAKE_CRN(3) ASM_MAKE_CRN(4) ASM_MAKE_CRN(8) static inline uint64_t read_xcr(uint32_t i) { uint32_t eax, edx; asm volatile("xgetbv" : "=a"(eax), "=d"(edx) : "c"(i) : "memory"); return eax | ((uint64_t)edx << 32); } static inline void write_xcr(uint32_t i, uint64_t value) { uint32_t edx = value >> 32; uint32_t eax = (uint32_t)value; asm volatile("xsetbv" : : "a"(eax), "d"(edx), "c"(i) : "memory"); } static inline uint64_t rdmsr(uint32_t msr) { uint32_t edx = 0, eax = 0; asm volatile("rdmsr\n\t" : "=a"(eax), "=d"(edx) : "c"(msr) : "memory"); return ((uint64_t)edx << 32) | eax; } static inline uint64_t wrmsr(uint32_t msr, uint64_t val) { uint32_t eax = (uint32_t)val; uint32_t edx = (uint32_t)(val >> 32); asm volatile("wrmsr\n\t" : : "a"(eax), "d"(edx), "c"(msr) : "memory"); return ((uint64_t)edx << 32) | eax; } static inline void xsave(uint8_t *region) { asm volatile("xsave %0" ::"m"(*region), "a"(~(uintptr_t)0), "d"(~(uintptr_t)0) : "memory"); } static inline void xrstor(uint8_t *region) { asm volatile("xrstor %0" ::"m"(*region), "a"(~(uintptr_t)0), "d"(~(uintptr_t)0) : "memory"); } static inline void fxsave(void *region) { asm volatile("fxsave (%0)" ::"a"(region)); } static inline void fxrstor(void *region) { asm volatile("fxrstor (%0)" ::"a"(region)); } static inline void fninit(void) { asm volatile("fninit"); } static inline void set_kernel_gs_base(void *addr) { wrmsr(0xc0000102, (uint64_t)addr); } static inline void *get_kernel_gs_base() { return (void *)rdmsr(0xc0000102); } static inline void *get_gs_base() { return (void *)rdmsr(0xc0000101); } static inline void *get_fs_base() { return (void *)rdmsr(0xc0000100); } static inline void set_gs_base(void *addr) { wrmsr(0xc0000101, (uint64_t)addr); } static inline void set_fs_base(void *addr) { wrmsr(0xc0000100, (uint64_t)addr); } enum Cr4Bit { CR4_VIRTUAL_8086_MODE_EXT = (1 << 0), CR4_PROTECTED_MODE_VIRTUAL_INT = (1 << 1), CR4_TIME_STAMP_DISABLE = (1 << 2), CR4_DEBUGGING_EXT = (1 << 3), CR4_PAGE_SIZE_EXT = (1 << 4), CR4_PHYSICAL_ADDRESS_EXT = (1 << 5), CR4_MACHINE_CHECK_EXCEPTION = (1 << 6), CR4_PAGE_GLOBAL_ENABLE = (1 << 7), CR4_PERFORMANCE_COUNTER_ENABLE = (1 << 8), CR4_FXSR_ENABLE = (1 << 9), CR4_SIMD_EXCEPTION_SUPPORT = (1 << 10), CR4_USER_MODE_INSTRUCTION_PREVENTION = (1 << 11), CR4_5_LEVEL_PAGING_ENABLE = (1 << 12), CR4_VIRTUAL_MACHINE_EXT_ENABLE = (1 << 13), CR4_SAFER_MODE_EXT_ENABLE = (1 << 14), CR4_FS_GS_BASE_ENABLE = (1 << 16), CR4_PCID_ENABLE = (1 << 17), CR4_XSAVE_ENABLE = (1 << 18), CR4_SUPERVISOR_EXE_PROTECTION_ENABLE = (1 << 20), CR4_SUPERVISOR_ACCESS_PROTECTION_ENABLE = (1 << 21), CR4_KEY_PROTECTION_ENABLE = (1 << 22), CR4_CONTROL_FLOW_ENABLE = (1 << 23), CR4_SUPERVISOR_KEY_PROTECTION_ENABLE = (1 << 24), }; enum XCr0Bit { XCR0_XSAVE_SAVE_X87 = (1 << 0), XCR0_XSAVE_SAVE_SSE = (1 << 1), XCR0_AVX_ENABLE = (1 << 2), XCR0_BNDREG_ENABLE = (1 << 3), XCR0_BNDCSR_ENABLE = (1 << 4), XCR0_AVX512_ENABLE = (1 << 5), XCR0_ZMM0_15_ENABLE = (1 << 6), XCR0_ZMM16_32_ENABLE = (1 << 7), XCR0_PKRU_ENABLE = (1 << 9), }; enum Cr0Bit { CR0_PROTECTED_MODE_ENABLE = (1 << 0), CR0_MONITOR_CO_PROCESSOR = (1 << 1), CR0_EMULATION = (1 << 2), CR0_TASK_SWITCHED = (1 << 3), CR0_EXTENSION_TYPE = (1 << 4), CR0_NUMERIC_ERROR_ENABLE = (1 << 5), CR0_WRITE_PROTECT_ENABLE = (1 << 16), CR0_ALIGNMENT_MASK = (1 << 18), CR0_NOT_WRITE_THROUGH_ENABLE = (1 << 29), CR0_CACHE_DISABLE = (1 << 30), CR0_PAGING_ENABLE = (1 << 31), }; } // namespace Gaia::Amd64