⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 l.s

📁 著名操作系统Plan 9的第三版的部分核心源代码。现在很难找到了。Plan 9是bell实验室开发的Unix后继者。
💻 S
字号:
#include "mem.h"#define NoScreenBlank	1/*#define ResetDiscs	1*/#define OP16	BYTE	$0x66/* * Some horrid macros for writing 16-bit code. */#define rAX		0		/* rX  */#define rCX		1#define rDX		2#define rBX		3#define rSP		4		/* SP */#define rBP		5		/* BP */#define rSI		6		/* SI */#define rDI		7		/* DI */#define rAL		0		/* rL  */#define rCL		1#define rDL		2#define rBL		3#define rAH		4		/* rH */#define rCH		5#define rDH		6#define rBH		7#define rES		0		/* rS */#define rCS		1#define rSS		2#define rDS		3#define rFS		4#define rGS		5#define rCR0		0		/* rC */#define rCR2		2#define rCR3		3#define rCR4		4#define OP(o, m, r, rm)	BYTE $o;		/* op + modr/m byte */	\			BYTE $((m<<6)|(r<<3)|rm)#define OPrr(o, r0, r1)	OP(o, 0x03, r0, r1);	/* general r -> r */#define LWI(i, rX)	BYTE $(0xB8+rX);	/* i -> rX */		\			WORD $i;#define LBI(i, rB)	BYTE $(0xB0+rB);	/* i -> r[HL] */	\			BYTE $i#define MFSR(rS, rX)	OPrr(0x8C, rS, rX)	/* rS -> rX */#define MTSR(rX, rS)	OPrr(0x8E, rS, rX)	/* rX -> rS */#define MFCR(rC, rX)	BYTE $0x0F;		/* rC -> rX */		\			OP(0x20, 0x03, rC, rX)#define MTCR(rX, rC)	BYTE $0x0F;		/* rX -> rC */		\			OP(0x22, 0x03, rC, rX)#define ANDI(i, r)	OP(0x81, 0x03, 0x04, r);/* i & r -> r */	\			WORD $i;#define CLR(r)		OPrr(0x31, r, r)	/* r^r -> r */#define FARJUMP(s, o)	BYTE $0xEA;		/* far jump to s:o */	\			WORD $o; WORD $s#define	DELAY		BYTE $0xEB;		/* jmp .+2 */		\			BYTE $0x00#define OUTb(p, d)	LBI(d, rAL);		/* d -> I/O port p */	\			BYTE $0xE6;					\			BYTE $p; DELAYTEXT origin(SB), $0	/*	 * 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#ifdef NoScreenBlank	/*	 * Get the current video mode. If it isn't mode 3,	 * set text mode 3.	 * Well, no. Windows95 won't co-operate here so we have	 * to explicitly set mode 3.	 */	XORL	AX, AX	MOVB	$0x0F, AH	INT	$0x10			/* get current video mode in AL */	CMPB	AL, $03	JEQ	_BIOSputs#endif /* NoScreenBlank */	XORL	AX, AX	MOVB	$0x03, AL	INT	$0x10			/* set video mode in AL */_BIOSputs:	MOVW	$_hello(SB), SI	XORL	BX, BX_BIOSputsloop:	LODSB	ORB	AL, AL	JEQ	_BIOSputsret	MOVB	$0x0E, AH	INT	$0x10	JMP	_BIOSputsloop_BIOSputsret:#ifdef ResetDiscs	XORL	AX, AX			/* reset disc system */	XORL	DX, DX	MOVB	$0x80, DL	INT	$0x13#endif /* ResetDiscs */#ifdef DOTCOM/* *	relocate everything to a half meg and jump there *	- looks weird because it is being assembled by a 32 bit *	  assembler for a 16 bit world * *	only b.com does this - not 9load */	MOVL	$0,BX	INCL	BX	SHLL	$15,BX	MOVL	BX,CX	MOVW	BX,ES	MOVL	$0,SI	MOVL	SI,DI	CLD	REP	MOVSL	/*	 * Jump to the copied image;	 * fix up the DS for the new location.	 */	FARJUMP(0x8000, _start8000(SB))TEXT _start8000(SB), $0	MFSR(rCS, rAX)			/* fix up DS (0x8000) */	MTSR(rAX, rDS)	/*	 * If we are already in protected mode, have to get back	 * to real mode before trying any privileged operations	 * (like going into protected mode...).	 * Try to reset with a restart vector.	 */	MFCR(rCR0, rAX)			/* are we in protected mode? */	ANDI(0x0001, rAX)	JEQ	_real	CLR(rBX)	MTSR(rBX, rES)	LWI(0x0467, rBX)		/* reset entry point */	LWI(_start8000(SB), rAX)	/* offset within segment */	BYTE	$0x26	BYTE	$0x89	BYTE	$0x07			/* MOVW	AX, ES:[BX] */	LBI(0x69, rBL)	MFSR(rCS, rAX)			/* segment */	BYTE	$0x26	BYTE	$0x89	BYTE	$0x07			/* MOVW	AX, ES:[BX] */	CLR(rDX)	OUTb(0x70, 0x8F)	OUTb(0x71, 0x0A)	FARJUMP(0xFFFF, 0x0000)		/* reset */#endif /* DOTCOM */_real:/* *	turn off interrupts */	CLI/* * 	goto protected mode *//*	MOVL	tgdtptr(SB),GDTR /**/	 BYTE	$0x0f	 BYTE	$0x01	 BYTE	$0x16	 WORD	$tgdtptr(SB)	MOVL	CR0,AX	ORL	$1,AX	MOVL	AX,CR0/* *	clear prefetch queue (weird code to avoid optimizations) */	CLC	JCC	flush	MOVL	AX,AXflush:/* *	set all segs *//*	MOVW	$SELECTOR(1, SELGDT, 0),AX	/**/	 BYTE	$0xc7	 BYTE	$0xc0	 WORD	$SELECTOR(1, SELGDT, 0)	MOVW	AX,DS	MOVW	AX,SS	MOVW	AX,ES	MOVW	AX,FS	MOVW	AX,GS/*	JMPFAR	SELECTOR(2, SELGDT, 0):$mode32bit(SB) /**/	 BYTE	$0x66	 BYTE	$0xEA	 LONG	$mode32bit-KZERO(SB)	 WORD	$SELECTOR(2, SELGDT, 0)/* * When we load 9load from DOS, the bootstrap jumps * to the instruction right after `JUMP', which gets * us into mode32bit. * * The name is so we are not optimized away. */TEXT jumplabel(SB), $0	BYTE $'J'; BYTE $'U'; BYTE $'M'; BYTE $'P'TEXT	mode32bit(SB),$0	/*	 * Clear BSS	 */	LEAL	edata-KZERO(SB),SI	MOVL	SI,DI	ADDL	$4,DI	MOVL	$0,AX	MOVL	AX,(SI)	LEAL	end-KZERO(SB),CX	SUBL	DI,CX	SHRL	$2,CX	CLD	REP	MOVSL	/*	 *  make a bottom level page table page that maps the first	 *  16 meg of physical memory	 */	LEAL	tpt-KZERO(SB),AX	/* get phys addr of temporary page table */	ADDL	$(BY2PG-1),AX		/* must be page aligned */	ANDL	$(~(BY2PG-1)),AX	/* ... */	MOVL	$(4*1024),CX		/* pte's per page */	MOVL	$((((4*1024)-1)<<PGSHIFT)|PTEVALID|PTEKERNEL|PTEWRITE),BXsetpte:	MOVL	BX,-4(AX)(CX*4)	SUBL	$(1<<PGSHIFT),BX	LOOP	setpte	/*	 *  make a top level page table page that maps the first	 *  16 meg of memory to 0 thru 16meg and to KZERO thru KZERO+16meg	 */	MOVL	AX,BX	ADDL	$(4*BY2PG),AX	ADDL	$(PTEVALID|PTEKERNEL|PTEWRITE),BX	MOVL	BX,0(AX)	MOVL	BX,((((KZERO>>1)&0x7FFFFFFF)>>(2*PGSHIFT-1-4))+0)(AX)	ADDL	$BY2PG,BX	MOVL	BX,4(AX)	MOVL	BX,((((KZERO>>1)&0x7FFFFFFF)>>(2*PGSHIFT-1-4))+4)(AX)	ADDL	$BY2PG,BX	MOVL	BX,8(AX)	MOVL	BX,((((KZERO>>1)&0x7FFFFFFF)>>(2*PGSHIFT-1-4))+8)(AX)	ADDL	$BY2PG,BX	MOVL	BX,12(AX)	MOVL	BX,((((KZERO>>1)&0x7FFFFFFF)>>(2*PGSHIFT-1-4))+12)(AX)	/*	 *  point processor to top level page & turn on paging	 */	MOVL	AX,CR3	MOVL	CR0,AX	ORL	$0X80000000,AX	MOVL	AX,CR0	/*	 *  use a jump to an absolute location to get the PC into	 *  KZERO.	 */	LEAL	tokzero(SB),AX	JMP*	AXTEXT	tokzero(SB),$0	/*	 *  stack and mach	 */	MOVL	$mach0(SB),SP	MOVL	SP,m(SB)	MOVL	$0,0(SP)	ADDL	$(MACHSIZE-4),SP	/* start stack above machine struct */	CALL	main(SB)loop:	JMP	loopGLOBL	mach0+0(SB), $MACHSIZEGLOBL	m(SB), $4GLOBL	tpt(SB), $(BY2PG*6)/* *  gdt to get us to 32-bit/segmented/unpaged mode */TEXT	tgdt(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)/* *  pointer to initial gdt */TEXT	tgdtptr(SB),$0	WORD	$(3*8)	LONG	$tgdt-KZERO(SB)/* *  input a byte */TEXT	inb(SB),$0	MOVL	p+0(FP),DX	XORL	AX,AX	INB	RET/* * input a short from a port */TEXT	ins(SB), $0	MOVL	p+0(FP), DX	XORL	AX, AX	OP16; INL	RET/* * input a long from a port */TEXT	inl(SB), $0	MOVL	p+0(FP), DX	XORL	AX, AX	INL	RET/* *  output a byte */TEXT	outb(SB),$0	MOVL	p+0(FP),DX	MOVL	b+4(FP),AX	OUTB	RET/* * output a short to a port */TEXT	outs(SB), $0	MOVL	p+0(FP), DX	MOVL	s+4(FP), AX	OP16; OUTL	RET/* * output a long to a port */TEXT	outl(SB), $0	MOVL	p+0(FP), DX	MOVL	s+4(FP), AX	OUTL	RET/* *  input a string of bytes from a port */TEXT	insb(SB),$0	MOVL	p+0(FP),DX	MOVL	a+4(FP),DI	MOVL	c+8(FP),CX	CLD; REP; INSB	RET/* *  input a string of shorts from a port */TEXT	inss(SB),$0	MOVL	p+0(FP),DX	MOVL	a+4(FP),DI	MOVL	c+8(FP),CX	CLD	REP; OP16; INSL	RET/* *  output a string of bytes to a port */TEXT	outsb(SB),$0	MOVL	p+0(FP),DX	MOVL	a+4(FP),SI	MOVL	c+8(FP),CX	CLD; REP; OUTSB	RET/* *  output a string of shorts to a port */TEXT	outss(SB),$0	MOVL	p+0(FP),DX	MOVL	a+4(FP),SI	MOVL	c+8(FP),CX	CLD	REP; OP16; OUTSL	RET/* *  input a string of longs from a port */TEXT	insl(SB),$0	MOVL	p+0(FP),DX	MOVL	a+4(FP),DI	MOVL	c+8(FP),CX	CLD; REP; INSL	RET/* *  output a string of longs to a port */TEXT	outsl(SB),$0	MOVL	p+0(FP),DX	MOVL	a+4(FP),SI	MOVL	c+8(FP),CX	CLD; REP; OUTSL	RET/* *  routines to load/read various system registers */GLOBL	idtptr(SB),$6TEXT	putidt(SB),$0		/* interrupt descriptor table */	MOVL	t+0(FP),AX	MOVL	AX,idtptr+2(SB)	MOVL	l+4(FP),AX	MOVW	AX,idtptr(SB)	MOVL	idtptr(SB),IDTR	RETTEXT	putcr3(SB),$0		/* top level page table pointer */	MOVL	t+0(FP),AX	MOVL	AX,CR3	RETTEXT	getcr0(SB),$0		/* coprocessor bits */	MOVL	CR0,AX	RETTEXT	getcr2(SB),$0		/* fault address */	MOVL	CR2,AX	RETTEXT	getcr3(SB),$0		/* page directory base */	MOVL	CR3,AX	RET/* *  special traps */TEXT	intr0(SB),$0	PUSHL	$0	PUSHL	$0	JMP	intrcommonTEXT	intr1(SB),$0	PUSHL	$0	PUSHL	$1	JMP	intrcommonTEXT	intr2(SB),$0	PUSHL	$0	PUSHL	$2	JMP	intrcommonTEXT	intr3(SB),$0	PUSHL	$0	PUSHL	$3	JMP	intrcommonTEXT	intr4(SB),$0	PUSHL	$0	PUSHL	$4	JMP	intrcommonTEXT	intr5(SB),$0	PUSHL	$0	PUSHL	$5	JMP	intrcommonTEXT	intr6(SB),$0	PUSHL	$0	PUSHL	$6	JMP	intrcommonTEXT	intr7(SB),$0	PUSHL	$0	PUSHL	$7	JMP	intrcommonTEXT	intr8(SB),$0	PUSHL	$8	JMP	intrcommonTEXT	intr9(SB),$0	PUSHL	$0	PUSHL	$9	JMP	intrcommonTEXT	intr10(SB),$0	PUSHL	$10	JMP	intrcommonTEXT	intr11(SB),$0	PUSHL	$11	JMP	intrcommonTEXT	intr12(SB),$0	PUSHL	$12	JMP	intrcommonTEXT	intr13(SB),$0	PUSHL	$13	JMP	intrcommonTEXT	intr14(SB),$0	PUSHL	$14	JMP	intrcommonTEXT	intr15(SB),$0	PUSHL	$0	PUSHL	$15	JMP	intrcommonTEXT	intr16(SB),$0	PUSHL	$0	PUSHL	$16	JMP	intrcommonTEXT	intr24(SB),$0	PUSHL	$0	PUSHL	$24	JMP	intrcommonTEXT	intr25(SB),$0	PUSHL	$0	PUSHL	$25	JMP	intrcommonTEXT	intr26(SB),$0	PUSHL	$0	PUSHL	$26	JMP	intrcommonTEXT	intr27(SB),$0	PUSHL	$0	PUSHL	$27	JMP	intrcommonTEXT	intr28(SB),$0	PUSHL	$0	PUSHL	$28	JMP	intrcommonTEXT	intr29(SB),$0	PUSHL	$0	PUSHL	$29	JMP	intrcommonTEXT	intr30(SB),$0	PUSHL	$0	PUSHL	$30	JMP	intrcommonTEXT	intr31(SB),$0	PUSHL	$0	PUSHL	$31	JMP	intrcommonTEXT	intr32(SB),$0	PUSHL	$0	PUSHL	$32	JMP	intrcommonTEXT	intr33(SB),$0	PUSHL	$0	PUSHL	$33	JMP	intrcommonTEXT	intr34(SB),$0	PUSHL	$0	PUSHL	$34	JMP	intrcommonTEXT	intr35(SB),$0	PUSHL	$0	PUSHL	$35	JMP	intrcommonTEXT	intr36(SB),$0	PUSHL	$0	PUSHL	$36	JMP	intrcommonTEXT	intr37(SB),$0	PUSHL	$0	PUSHL	$37	JMP	intrcommonTEXT	intr38(SB),$0	PUSHL	$0	PUSHL	$38	JMP	intrcommonTEXT	intr39(SB),$0	PUSHL	$0	PUSHL	$39	JMP	intrcommonTEXT	intr64(SB),$0	PUSHL	$0	PUSHL	$64	JMP	intrcommonTEXT	intrbad(SB),$0	PUSHL	$0	PUSHL	$0x1ff	JMP	intrcommonintrcommon:	PUSHL	DS	PUSHL	ES	PUSHL	FS	PUSHL	GS	PUSHAL	MOVL	$(KDSEL),AX	MOVW	AX,DS	MOVW	AX,ES	LEAL	0(SP),AX	PUSHL	AX	CALL	trap(SB)	POPL	AX	POPAL	POPL	GS	POPL	FS	POPL	ES	POPL	DS	ADDL	$8,SP	/* error code and trap type */	IRETL/* *  interrupt level is interrupts on or off */TEXT	spllo(SB),$0	PUSHFL	POPL	AX	STI	RETTEXT	splhi(SB),$0	PUSHFL	POPL	AX	CLI	RETTEXT	splx(SB),$0	MOVL	s+0(FP),AX	PUSHL	AX	POPFL	RET/* *  do nothing whatsoever till interrupt happens */TEXT	idle(SB),$0	HLT	RET/* * Try to determine the CPU type which requires fiddling with EFLAGS. * If the Id bit can be toggled then the CPUID instruciton can be used * to determine CPU identity and features. First have to check if it's * a 386 (Ac bit can't be set). If it's not a 386 and the Id bit can't be * toggled then it's an older 486 of some kind. * *	cpuid(id[], &ax, &dx); */#define CPUID		BYTE $0x0F; BYTE $0xA2	/* CPUID, argument in AX */TEXT cpuid(SB), $0	MOVL	$0x240000, AX	PUSHL	AX	POPFL					/* set Id|Ac */	PUSHFL	POPL	BX				/* retrieve value */	MOVL	$0, AX	PUSHL	AX	POPFL					/* clear Id|Ac, EFLAGS initialised */	PUSHFL	POPL	AX				/* retrieve value */	XORL	BX, AX	TESTL	$0x040000, AX			/* Ac */	JZ	_cpu386				/* can't set this bit on 386 */	TESTL	$0x200000, AX			/* Id */	JZ	_cpu486				/* can't toggle this bit on some 486 */	MOVL	$0, AX	CPUID	MOVL	id+0(FP), BP	MOVL	BX, 0(BP)			/* "Genu" "Auth" "Cyri" */	MOVL	DX, 4(BP)			/* "ineI" "enti" "xIns" */	MOVL	CX, 8(BP)			/* "ntel" "cAMD" "tead" */	MOVL	$1, AX	CPUID	JMP	_cpuid_cpu486:	MOVL	$0x400, AX	MOVL	$0, DX	JMP	_cpuid_cpu386:	MOVL	$0x300, AX	MOVL	$0, DX_cpuid:	MOVL	ax+4(FP), BP	MOVL	AX, 0(BP)	MOVL	dx+8(FP), BP	MOVL	DX, 0(BP)	RET/* *  basic timing loop to determine CPU frequency */TEXT	aamloop(SB),$0	MOVL	c+0(FP),CXaaml1:	AAM	LOOP	aaml1	RETTEXT _hello(SB), $0	BYTE $'P'; BYTE $'l'; BYTE $'a'; BYTE $'n';	BYTE $' '; BYTE $'9'; BYTE $' '; BYTE $'f';	BYTE $'r'; BYTE $'o'; BYTE $'m'; BYTE $' ';	BYTE $'B'; BYTE $'e'; BYTE $'l'; BYTE $'l';	BYTE $' '; BYTE $'L'; BYTE $'a'; BYTE $'b';	BYTE $'s'; 	BYTE $'\r';	BYTE $'\n';	BYTE $'\z';

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -