1 // Basic x86 asm functions and function defs.
3 // Copyright (C) 2008 Kevin O'Connor <kevin@koconnor.net>
5 // This file may be distributed under the terms of the GNU GPLv3 license.
9 #include "ioport.h" // outb
10 #include "biosvar.h" // struct bregs
12 static inline void irq_disable(void)
14 asm volatile("cli": : :"memory");
17 static inline void irq_enable(void)
19 asm volatile("sti": : :"memory");
22 static inline unsigned long irq_save(void)
25 asm volatile("pushfl ; popl %0" : "=g" (flags));
30 static inline void irq_restore(unsigned long flags)
32 asm volatile("pushl %0 ; popfl" : : "g" (flags) : "memory", "cc");
35 static inline void cpu_relax(void)
37 asm volatile("rep ; nop": : :"memory");
40 static inline void nop(void)
45 static inline void hlt(void)
50 static inline void wbinvd(void)
52 asm volatile("wbinvd");
55 static inline void cpuid(u32 index, u32 *eax, u32 *ebx, u32 *ecx, u32 *edx)
58 : "=a" (*eax), "=b" (*ebx), "=c" (*ecx), "=d" (*edx)
62 void *memset(void *s, int c, size_t n);
63 void *memcpy(void *d1, const void *s1, size_t len);
64 void *memmove(void *d, const void *s, size_t len);
66 // Call a function with a specified register state. Note that on
67 // return, the interrupt enable/disable flag may be altered.
69 void call16(struct bregs *callregs)
75 "calll __call16_from32\n"
77 : "+a" (callregs), "+m" (*callregs)
79 : "ebx", "ecx", "edx", "esi", "edi", "ebp", "cc");
83 void __call16_int(struct bregs *callregs, u16 offset)
85 callregs->cs = SEG_BIOS;
86 callregs->ip = offset;
91 #define call16_int(nr, callregs) do { \
92 extern void irq_trampoline_ ##nr (); \
93 __call16_int((callregs), (u32)&irq_trampoline_ ##nr ); \
96 #include "../out/rom16.offset.auto.h"
97 #define call16_int(nr, callregs) \
98 __call16_int((callregs), OFFSET_irq_trampoline_ ##nr )
102 void debug_serial_setup();
103 void BX_PANIC(const char *fmt, ...)
104 __attribute__ ((format (printf, 1, 2)))
105 __attribute__ ((noreturn));
106 void printf(const char *fmt, ...)
107 __attribute__ ((format (printf, 1, 2)));
108 void __dprintf(const char *fmt, ...)
109 __attribute__ ((format (printf, 1, 2)));
110 #define dprintf(lvl, fmt, args...) do { \
111 if (CONFIG_DEBUG_LEVEL && (lvl) <= CONFIG_DEBUG_LEVEL) \
112 __dprintf((fmt) , ##args ); \
114 void __debug_enter(const char *fname, struct bregs *regs);
115 void __debug_fail(const char *fname, struct bregs *regs);
116 void __debug_stub(const char *fname, struct bregs *regs);
117 void __debug_isr(const char *fname);
118 #define debug_enter(regs, lvl) do { \
119 if ((lvl) && (lvl) <= CONFIG_DEBUG_LEVEL) \
120 __debug_enter(__func__, regs); \
122 #define debug_isr(lvl) do { \
123 if ((lvl) && (lvl) <= CONFIG_DEBUG_LEVEL) \
124 __debug_isr(__func__); \
126 #define debug_stub(regs) \
127 __debug_stub(__func__, regs)
129 // Frequently used return codes
130 #define RET_EUNSUPPORTED 0x86
132 set_success(struct bregs *regs)
138 set_code_success(struct bregs *regs)
145 set_fail_silent(struct bregs *regs)
151 set_code_fail_silent(struct bregs *regs, u8 code)
157 void __set_fail(const char *fname, struct bregs *regs);
158 void __set_code_fail(const char *fname, struct bregs *regs, u8 code);
160 #define set_fail(regs) \
161 __set_fail(__func__, (regs))
162 #define set_code_fail(regs, code) \
163 __set_code_fail(__func__, (regs), (code))
166 void handle_15c2(struct bregs *regs);
180 int usleep(u32 count);
181 void handle_1583(struct bregs *regs);
182 void handle_1586(struct bregs *regs);
185 void VISIBLE16 handle_1553(struct bregs *regs);
188 void handle_1ab1(struct bregs *regs);
191 u8 checksum(u8 *far_data, u32 len);
194 void make_bios_writable();
195 void make_bios_readonly();
198 void pci_bios_setup(void);
207 void mptable_init(void);
210 void smbios_init(void);
213 void printf_bootdev(u16 bootdev);
216 void interactive_bootmenu();
219 void coreboot_fill_map();