/*
* Bootstrap loader decompressor. Starts at 0x10000 (where pbs puts it)
* or 0x7c00 (where pxe puts it) and memmoves kernel (immediately following)
* into standard kernel location.
*/
#include "mem.h"
#include "/sys/src/boot/pc/x16.h"
#undef BIOSCALL /* we don't know what evil the bios gets up to */
#define BIOSCALL(b) INT $(b); CLI
#define CMPL(r0, r1) BYTE $0x66; CMP(r0, r1)
#define LLI(i, rX) BYTE $0x66; /* i -> rX */ \
BYTE $(0xB8+rX); \
LONG $i;
#define CPUID BYTE $0x0F; BYTE $0xA2 /* CPUID, argument in AX */
#define WBINVD BYTE $0x0F; BYTE $0x09
TEXT origin(SB), $0
/*
* turn off interrupts
*/
CLI
/*
* This part of l.s is used only in the boot kernel.
* It assumes that we are in real address mode, i.e.,
* that we look like an 8086.
*
* Make sure the segments are reasonable.
* If we were started directly from the BIOS
* (i.e. no MS-DOS) then DS may not be
* right.
*/
MOVW CS, AX
MOVW AX, DS
LWI(0, rAX) /* always put stack in first 64k */
MTSR(rAX, rSS)
LWI(origin(SB), rSP) /* set the stack pointer */
LWI(0x2401, rAX) /* enable a20 line */
BIOSCALL(0x15)
XORL AX, AX
MOVB $0x03, AL
// LWI(3, rAX)
INT $0x10 /* set video mode in AL */
/*
* Check for CGA mode.
*/
_cgastart:
LWI(0x0F00, rAX) /* get current video mode in AL */
BIOSCALL(0x10)
ANDI(0x007F, rAX)
SUBI(0x0003, rAX) /* is it mode 3? */
JEQ _cgamode3
LWI(0x0003, rAX) /* turn on text mode 3 */
BIOSCALL(0x10)
_cgamode3:
LWI(_hello(SB), rSI)
CALL _cgaputs(SB)
LLI(BIOSTABLES, rAX) /* tables in low memory, not after end */
OPSIZE; ANDL $~(BY2PG-1), AX
OPSIZE; SHRL $4, AX
SW(rAX, _ES(SB))
CLR(rDI)
SW(rDI, _DI(SB))
MTSR(rAX, rES)
/*
* Check for APM1.2 BIOS support.
*/
LWI(0x5304, rAX) /* disconnect anyone else */
CLR(rBX)
BIOSCALL(0x15)
JCS _apmfail
LWI(0x5303, rAX) /* connect */
CLR(rBX)
CLC
BIOSCALL(0x15)
JCC _apmpush
_apmfail:
LW(_ES(SB), rAX) /* no support */
MTSR(rAX, rES)
LW(_DI(SB), rDI)
JCS _apmend
_apmpush:
OPSIZE; PUSHR(rSI) /* save returned APM data on stack */
OPSIZE; PUSHR(rBX)
PUSHR(rDI)
PUSHR(rDX)
PUSHR(rCX)
PUSHR(rAX)
LW(_ES(SB), rAX)
MTSR(rAX, rES)
LW(_DI(SB), rDI)
LWI(0x5041, rAX) /* first 4 bytes are APM\0 */
STOSW
LWI(0x004D, rAX)
STOSW
LWI(8, rCX) /* pop the saved APM data */
_apmpop:
POPR(rAX)
STOSW
LOOP _apmpop
_apmend:
/*
* Try to retrieve the 0xE820 memory map.
* This is weird because some BIOS do not seem to preserve
* ES/DI on failure. Consequently they may not be valid
* at _e820end:.
*/
SW(rDI, _DI(SB)) /* save DI */
CLR(rAX) /* write terminator */
STOSW
STOSW
CLR(rBX)
PUSHR(rBX) /* keep loop count on stack */
/* BX is the continuation value */
_e820loop:
POPR(rAX)
INC(rAX)
PUSHR(rAX) /* doesn't affect FLAGS */
CMPI(32, rAX) /* mmap[32+1] in C code */
JGT _e820pop
LLI(20, rCX) /* buffer size */
LLI(0x534D4150, rDX) /* signature - ASCII "SMAP" */
LLI(0x0000E820, rAX) /* function code */
BIOSCALL(0x15) /* writes 20 bytes at (es,di) */
JCS _e820pop /* some kind of error */
LLI(0x534D4150, rDX)
CMPL(rDX, rAX) /* verify correct BIOS version */
JNE _e820pop
LLI(20, rDX)
CMPL(rDX, rCX) /* verify correct count */
JNE _e820pop
SUBI(4, rDI) /* overwrite terminator */
LWI(0x414D, rAX) /* first 4 bytes are "MAP\0" */
STOSW
LWI(0x0050, rAX)
STOSW
ADDI(20, rDI) /* bump to next entry */
SW(rDI, _DI(SB)) /* save DI */
CLR(rAX) /* write terminator */
STOSW
STOSW
OR(rBX, rBX) /* zero if last entry */
JNE _e820loop
_e820pop:
POPR(rAX) /* loop count */
LW(_DI(SB), rDI)
CLR(rAX)
MTSR(rAX, rES)
_e820end:
/*
* goto protected mode
*/
/* MOVL loadgdtptr(SB),GDTR /**/
BYTE $0x0f
BYTE $0x01
BYTE $0x16
WORD $loadgdtptr(SB)
DELAY
LWI(1, rAX)
/* MOV AX,MSW */
BYTE $0x0F; BYTE $0x01; BYTE $0xF0
/*
* clear prefetch queue (weird code to avoid optimizations)
*/
DELAY
/*
* set all segs
*/
/* MOVW $KDSEL,AX /**/
BYTE $0xc7
BYTE $0xc0
WORD $KDSEL
MOVW AX,DS
MOVW AX,SS
MOVW AX,ES
MOVW AX,FS
MOVW AX,GS
MOVW $(20*1024*1024-4), SP /* new stack pointer */
DELAY
/* god only knows what the damned bios has been up to... */
CLI
/* jump to C (main) */
/* JMPFAR KESEL:$main(SB) /**/
BYTE $0x66
BYTE $0xEA
LONG $_main(SB)
WORD $KESEL
/* output a cheery wee message (rSI) */
TEXT _cgaputs(SB), $0
//_cgaputs:
CLR(rBX)
_cgaputsloop:
LODSB
ORB(rAL, rAL)
JEQ _cgaend
LBI(0x0E,rAH)
BIOSCALL(0x10)
JMP _cgaputsloop
_cgaend:
RET
TEXT _hello(SB), $0
BYTE $'\r'; BYTE $'\n'
BYTE $'9'; BYTE $'b'; BYTE $'o'; BYTE $'o'
BYTE $'t'; BYTE $' '
BYTE $'\z'
/* offset into bios tables using segment ES. stos? use (es,di) */
TEXT _DI(SB), $0
LONG $0
/* segment address of bios tables (BIOSTABLES >> 4) */
TEXT _ES(SB), $0
LONG $0
/*
* pointer to initial gdt
*/
TEXT loadgdtptr(SB),$0
WORD $(4*8)
LONG $loadgdt(SB)
/*
* gdt to get us to 32-bit/segmented/unpaged mode
*/
TEXT loadgdt(SB),$0
/* null descriptor */
LONG $0
LONG $0
/* data segment descriptor for 4 gigabytes (PL 0) */
LONG $(0xFFFF)
LONG $(SEGG|SEGB|(0xF<<16)|SEGP|SEGPL(0)|SEGDATA|SEGW)
/* exec segment descriptor for 4 gigabytes (PL 0) */
LONG $(0xFFFF)
LONG $(SEGG|SEGD|(0xF<<16)|SEGP|SEGPL(0)|SEGEXEC|SEGR)
/* exec segment descriptor for 4 gigabytes (PL 0) 16-bit */
LONG $(0xFFFF)
LONG $(SEGG|(0xF<<16)|SEGP|SEGPL(0)|SEGEXEC|SEGR)
/*
* output a byte
*/
TEXT outb(SB),$0
MOVL p+0(FP),DX
MOVL b+4(FP),AX
OUTB
RET
/*
* input a byte
*/
TEXT inb(SB),$0
MOVL p+0(FP),DX
XORL AX,AX
INB
RET
TEXT mb586(SB), $0
XORL AX, AX
CPUID
RET
TEXT wbinvd(SB), $0
WBINVD
RET
TEXT splhi(SB),$0
PUSHFL
POPL AX
CLI
RET
|