HelenOS sources
This source file includes following definitions.
- cpu_interruptible_sleep
- cpu_halt
- cpu_spin_hint
- pio_read_8
- pio_read_16
- pio_read_32
- pio_write_8
- pio_write_16
- pio_write_32
- read_rflags
- write_rflags
- interrupts_read
- interrupts_enable
- interrupts_disable
- interrupts_restore
- interrupts_disabled
- write_msr
- read_msr
- invlpg
- gdtr_load
- gdtr_store
- idtr_load
- tr_load
#ifndef KERN_amd64_ASM_H_
#define KERN_amd64_ASM_H_
#include <config.h>
#include <typedefs.h>
#include <arch/cpu.h>
#include <trace.h>
#define IO_SPACE_BOUNDARY ((void *) (64 * 1024))
_NO_TRACE static inline void cpu_interruptible_sleep()
{
asm volatile (
"sti\n"
"hlt\n"
"cli\n"
);
}
_NO_TRACE static inline void __attribute__((noreturn)) cpu_halt(void)
{
while (true) {
asm volatile (
"hlt\n"
);
}
}
_NO_TRACE static inline void cpu_spin_hint(void)
{
asm volatile (
"pause\n"
);
}
_NO_TRACE static inline uint8_t pio_read_8(ioport8_t *port)
{
if (port < (ioport8_t *) IO_SPACE_BOUNDARY) {
uint8_t val;
asm volatile (
"inb %w[port], %b[val]\n"
: [val] "=a" (val)
: [port] "d" (port)
);
return val;
} else
return (uint8_t) *port;
}
_NO_TRACE static inline uint16_t pio_read_16(ioport16_t *port)
{
if (port < (ioport16_t *) IO_SPACE_BOUNDARY) {
uint16_t val;
asm volatile (
"inw %w[port], %w[val]\n"
: [val] "=a" (val)
: [port] "d" (port)
);
return val;
} else
return (uint16_t) *port;
}
_NO_TRACE static inline uint32_t pio_read_32(ioport32_t *port)
{
if (port < (ioport32_t *) IO_SPACE_BOUNDARY) {
uint32_t val;
asm volatile (
"inl %w[port], %[val]\n"
: [val] "=a" (val)
: [port] "d" (port)
);
return val;
} else
return (uint32_t) *port;
}
_NO_TRACE static inline void pio_write_8(ioport8_t *port, uint8_t val)
{
if (port < (ioport8_t *) IO_SPACE_BOUNDARY) {
asm volatile (
"outb %b[val], %w[port]\n"
:: [val] "a" (val), [port] "d" (port)
);
} else
*port = val;
}
_NO_TRACE static inline void pio_write_16(ioport16_t *port, uint16_t val)
{
if (port < (ioport16_t *) IO_SPACE_BOUNDARY) {
asm volatile (
"outw %w[val], %w[port]\n"
:: [val] "a" (val), [port] "d" (port)
);
} else
*port = val;
}
_NO_TRACE static inline void pio_write_32(ioport32_t *port, uint32_t val)
{
if (port < (ioport32_t *) IO_SPACE_BOUNDARY) {
asm volatile (
"outl %[val], %w[port]\n"
:: [val] "a" (val), [port] "d" (port)
);
} else
*port = val;
}
_NO_TRACE static inline uint64_t read_rflags(void)
{
uint64_t rflags;
asm volatile (
"pushfq\n"
"popq %[v]\n"
: [v] "=r" (rflags)
);
return rflags;
}
_NO_TRACE static inline void write_rflags(uint64_t rflags)
{
asm volatile (
"pushq %[v]\n"
"popfq\n"
:: [v] "r" (rflags)
);
}
_NO_TRACE static inline ipl_t interrupts_read(void)
{
return (ipl_t) read_rflags();
}
_NO_TRACE static inline ipl_t interrupts_enable(void)
{
ipl_t ipl = interrupts_read();
asm volatile ("sti\n");
return ipl;
}
_NO_TRACE static inline ipl_t interrupts_disable(void)
{
ipl_t ipl = interrupts_read();
asm volatile ("cli\n");
return ipl;
}
_NO_TRACE static inline void interrupts_restore(ipl_t ipl)
{
write_rflags((uint64_t) ipl);
}
_NO_TRACE static inline bool interrupts_disabled(void)
{
return ((read_rflags() & RFLAGS_IF) == 0);
}
_NO_TRACE static inline void write_msr(uint32_t msr, uint64_t value)
{
asm volatile (
"wrmsr\n"
:: "c" (msr),
"a" ((uint32_t) (value)),
"d" ((uint32_t) (value >> 32))
);
}
_NO_TRACE static inline sysarg_t read_msr(uint32_t msr)
{
uint32_t ax, dx;
asm volatile (
"rdmsr\n"
: "=a" (ax), "=d" (dx)
: "c" (msr)
);
return ((uint64_t) dx << 32) | ax;
}
_NO_TRACE static inline void invlpg(uintptr_t addr)
{
asm volatile (
"invlpg %[addr]\n"
:: [addr] "m" (*((sysarg_t *) addr))
);
}
_NO_TRACE static inline void gdtr_load(ptr_16_64_t *gdtr_reg)
{
asm volatile (
"lgdtq %[gdtr_reg]\n"
:: [gdtr_reg] "m" (*gdtr_reg)
);
}
_NO_TRACE static inline void gdtr_store(ptr_16_64_t *gdtr_reg)
{
asm volatile (
"sgdtq %[gdtr_reg]\n"
:: [gdtr_reg] "m" (*gdtr_reg)
);
}
_NO_TRACE static inline void idtr_load(ptr_16_64_t *idtr_reg)
{
asm volatile (
"lidtq %[idtr_reg]\n"
:: [idtr_reg] "m" (*idtr_reg));
}
_NO_TRACE static inline void tr_load(uint16_t sel)
{
asm volatile (
"ltr %[sel]"
:: [sel] "r" (sel)
);
}
#define GEN_READ_REG(reg) _NO_TRACE static inline sysarg_t read_ ##reg (void) \
{ \
sysarg_t res; \
asm volatile ( \
"movq %%" #reg ", %[res]" \
: [res] "=r" (res) \
); \
return res; \
}
#define GEN_WRITE_REG(reg) _NO_TRACE static inline void write_ ##reg (sysarg_t regn) \
{ \
asm volatile ( \
"movq %[regn], %%" #reg \
:: [regn] "r" (regn) \
); \
}
GEN_READ_REG(cr0);
GEN_WRITE_REG(cr0);
GEN_READ_REG(cr2);
GEN_READ_REG(cr3);
GEN_WRITE_REG(cr3);
GEN_READ_REG(cr4);
GEN_WRITE_REG(cr4);
GEN_READ_REG(dr0);
GEN_READ_REG(dr1);
GEN_READ_REG(dr2);
GEN_READ_REG(dr3);
GEN_READ_REG(dr6);
GEN_READ_REG(dr7);
GEN_WRITE_REG(dr0);
GEN_WRITE_REG(dr1);
GEN_WRITE_REG(dr2);
GEN_WRITE_REG(dr3);
GEN_WRITE_REG(dr6);
GEN_WRITE_REG(dr7);
extern void asm_delay_loop(uint32_t);
extern void asm_fake_loop(uint32_t);
extern uintptr_t int_0;
extern uintptr_t int_1;
extern uintptr_t int_2;
extern uintptr_t int_3;
extern uintptr_t int_4;
extern uintptr_t int_5;
extern uintptr_t int_6;
extern uintptr_t int_7;
extern uintptr_t int_8;
extern uintptr_t int_9;
extern uintptr_t int_10;
extern uintptr_t int_11;
extern uintptr_t int_12;
extern uintptr_t int_13;
extern uintptr_t int_14;
extern uintptr_t int_15;
extern uintptr_t int_16;
extern uintptr_t int_17;
extern uintptr_t int_18;
extern uintptr_t int_19;
extern uintptr_t int_20;
extern uintptr_t int_21;
extern uintptr_t int_22;
extern uintptr_t int_23;
extern uintptr_t int_24;
extern uintptr_t int_25;
extern uintptr_t int_26;
extern uintptr_t int_27;
extern uintptr_t int_28;
extern uintptr_t int_29;
extern uintptr_t int_30;
extern uintptr_t int_31;
extern uintptr_t int_32;
extern uintptr_t int_33;
extern uintptr_t int_34;
extern uintptr_t int_35;
extern uintptr_t int_36;
extern uintptr_t int_37;
extern uintptr_t int_38;
extern uintptr_t int_39;
extern uintptr_t int_40;
extern uintptr_t int_41;
extern uintptr_t int_42;
extern uintptr_t int_43;
extern uintptr_t int_44;
extern uintptr_t int_45;
extern uintptr_t int_46;
extern uintptr_t int_47;
extern uintptr_t int_48;
extern uintptr_t int_49;
extern uintptr_t int_50;
extern uintptr_t int_51;
extern uintptr_t int_52;
extern uintptr_t int_53;
extern uintptr_t int_54;
extern uintptr_t int_55;
extern uintptr_t int_56;
extern uintptr_t int_57;
extern uintptr_t int_58;
extern uintptr_t int_59;
extern uintptr_t int_60;
extern uintptr_t int_61;
extern uintptr_t int_62;
extern uintptr_t int_63;
extern void enable_l_apic_in_msr(void);
#endif
HelenOS homepage, sources at GitHub