blob: 344dc1c0b234eff359bcb8da2c6609e69a87278b [file] [log] [blame]
#include "libcflat.h"
#include "apic.h"
#include "io.h"
#define KBD_CCMD_READ_OUTPORT 0xD0 /* read output port */
#define KBD_CCMD_WRITE_OUTPORT 0xD1 /* write output port */
#define KBD_CCMD_RESET 0xFE /* CPU reset */
static inline void kbd_cmd(u8 val)
{
while (inb(0x64) & 2);
outb(val, 0x64);
}
static inline u8 kbd_in(void)
{
kbd_cmd(KBD_CCMD_READ_OUTPORT);
while (inb(0x64) & 2);
return inb(0x60);
}
static inline void kbd_out(u8 val)
{
kbd_cmd(KBD_CCMD_WRITE_OUTPORT);
while (inb(0x64) & 2);
outb(val, 0x60);
}
static inline void rtc_out(u8 reg, u8 val)
{
outb(reg, 0x70);
outb(val, 0x71);
}
extern char resume_start, resume_end;
#define state (*(volatile int *)0x2000)
#define bad (*(volatile int *)0x2004)
#define resumed (*(volatile int *)0x2008)
int main(int argc, char **argv)
{
volatile u16 *resume_vector_ptr = (u16 *)0x467L;
char *addr, *resume_vec = (void*)0x1000;
/* resume execution by indirect jump via 40h:0067h */
rtc_out(0x0f, 0x0a);
resume_vector_ptr[0] = ((u32)(ulong)resume_vec);
resume_vector_ptr[1] = 0;
for (addr = &resume_start; addr < &resume_end; addr++)
*resume_vec++ = *addr;
if (state != 0) {
/*
* Strictly speaking this is a firmware problem, but let's check
* for it as well...
*/
if (resumed != 1) {
printf("Uh, resume vector visited %d times?\n", resumed);
bad |= 2;
}
/*
* Port 92 bit 0 is cleared on system reset. On a soft reset it
* is left to 1. Use this to distinguish INIT from hard reset.
*/
if (resumed != 0 && (inb(0x92) & 1) == 0) {
printf("Uh, hard reset!\n");
bad |= 1;
}
}
resumed = 0;
switch (state++) {
case 0:
printf("testing port 92 init... ");
outb(inb(0x92) & ~1, 0x92);
outb(inb(0x92) | 1, 0x92);
break;
case 1:
printf("testing kbd controller reset... ");
kbd_cmd(KBD_CCMD_RESET);
break;
case 2:
printf("testing kbd controller init... ");
kbd_out(kbd_in() & ~1);
break;
case 3:
printf("testing 0xcf9h init... ");
outb(0, 0xcf9);
outb(4, 0xcf9);
break;
case 4:
printf("testing init to BSP... ");
apic_icr_write(APIC_DEST_SELF | APIC_DEST_PHYSICAL
| APIC_DM_INIT, 0);
break;
case 5:
exit(bad);
}
/* The resume code will get us back to main. */
asm("cli; hlt");
__builtin_unreachable();
}
asm (
".global resume_start\n"
".global resume_end\n"
".code16\n"
"resume_start:\n"
"incb %cs:0x2008\n" // resumed++;
"mov $0x0f, %al\n" // rtc_out(0x0f, 0x00);
"out %al, $0x70\n"
"mov $0x00, %al\n"
"out %al, $0x71\n"
"jmp $0xffff, $0x0000\n" // BIOS reset
"resume_end:\n"
#ifdef __i386__
".code32\n"
#else
".code64\n"
#endif
);