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

📄 bootsect.s

📁 Simple Operating Systems (简称SOS)是一个可以运行在X86平台上(包括QEMU
💻 S
字号:
/* * @(#) $Id: bootsect.S,v 1.6 2004/06/18 07:43:51 d2 Exp $ * Description : Bootsecteur en syntaxe AT&T * Auteurs : Thomas Petazzoni & Fabrice Gautier & Emmanuel Marty *	     Jerome Petazzoni & Bernard Cassagne & coffeeman *	     David Decotigny * Bug reports to kos-misc@enix.org *//* * But global de ce bootsecteur : * *		- Initialiser la becane *		- Charger le kernel *		- Passer en mode protege *		- Executer le kernel * * Taille restante : Je vous rappelle qu'un bootsecteur ne peut faire * qu'au maximum 512 octets dont 2 octets obligatoires 0xAA55.  Sur * les 510 octets reellement utilisables, il reste 3 octets dispo (60 * si on decide d'enlever le BPB un jour) !!! * * thomas_petazzoni :  - detection des codes d'erreurs de chargement * David_Decotigny  :  - Passage en GNU as * David_Decotigny  :  - Chargement du noyau au-dela du 1er Mega (taille *                       max = 0x9e000 octets = 632ko), pour avoir le *                       meme noyau sous grub et avec le bootsecteur */ /*  * Sequence d'operations :  * - Le BIOS charge le bootsect en 0x7c00 (BOOT_ADRESS). On choisit  *   la representation 0x7c0:0000 pour que le .org 0 reste valide  * - Le bootsect se deplace de lui-meme en 0x9f000 (COPY_ADRESS). On  *   choisit la representation 0x9f00:0000 pour que le .org 0 reste  *   valide  * - Le bootsect verifie que le processeur est du type 386+  * - Il charge le noyau depuis la disquette en memoire a partir de  *   0x1000 (LOAD_ADRESS). Le noyau peut au max tenir sur  *   SECTORS_TO_LOAD secteurs  * - Il passe en pmode flat (apres ouverture a20)  * - Il recopie le noyau (situe en LOAD_ADRESS) vers son adresse  *   finale (FINAL_ADDRESS = 2Mo). La recopie se fait sur tout l'espace  *   LOAD_ADRESS ---> COPY_ADRESS, c'est a dire sur 0x9e000 octets =  *   632ko. Le noyau peut donc au max faire 632ko. Le nombre max de  *   secteurs de disquette qu'on peut charger est donc 1264  *//* La taille de la pile */#define BOOT_STACK_SIZE	        0x4000	.file	"bootsect.S"	/* Tout est place dans une seule section */	.section ".bootsect"	/* L'essentiel du bootsector (sauf les 1eres instructions)	   sont a un offset 0. On fait en sorte que le compilo soit	   d'accord la-dessus. Quand on a des adresse realm exotiques	   (0x7c00, 0x9f000, ...), on s'arrange toujours pour avoir un	   offset de 0 => on choisira le segment adapte (0x7c0,	   0x9f00, ...). Il ne faut pas oublier le ld -Ttext 0 */	.org 0	/* Pour que gas genere du 16bits, afin que ca marche en realm */	.code16#define SECTORS_TO_LOAD 128 /* 64 ko */ /* MAX=1264 *//* * Parametres de la disquette. Comme c'est chiant de faire une * procedure de detection auto, et que ca prend de la place, on fait * ca "a la main". Par exemple, une DD 720 Ko a 9 secteurs/piste, une * 1.44 Mo a 18 secteurs/pistes */#define CYLS    80#define HEADS   1#define SECTS   18#define BOOT_ADRESS 0x07C00  	       /* Adresse de demarrage (lineaire) */#define BOOT_SEG (BOOT_ADRESS>>4)      /* Segment de Boot */#define BOOT_SIZE 512		       /* Taille bu bootsecteur */#define COPY_ADRESS 0x9F000  	       /* La ou on va copier le					  bootsecteur (lineaire) */#define COPY_SEG (COPY_ADRESS>>4)      /* Segment de la ou on va					  copier le bootsecteur */#define LOAD_ADRESS 0x01000  	       /* 1er chargement du systeme */#define LOAD_SEG (LOAD_ADRESS>>4)      /* Segment du 1er chargement du */#define MAX_KERN_LEN COPY_ADRESS-LOAD_ADRESS /* Taille noyau maxi *//* IMPORTANT : Cette valeur DOIT etre identique a l'adresse presente	       dans sos.lds ! */#define FINAL_ADDRESS 0x200000         /* Adresse finale (physique de 0 a 4G)	                                  ou est charge le noyau */#define OP16 .byte 0x66 ;#define OP32 .byte 0x66 ;/* * Procedure qui vide le buffer clavier. */#define WAITKB     \  1:               ;\    .word 0xeb     ;\    .word 0xeb     ;\    inb $0x64, %al ;\    andb $0x2, %al ;\    jnz 1b	/* Le point d'entree dans le bootsect */.globl _bsect_bsect:	/*	 * La portion qui suit est situee a un offset 0x7c00 en	 * memoire. Attention donc aux references memoire dans cette	 * partie. On choisit de rester en offset 0 (.org 0), mais on         * charge correctement les segments a 0x7c0.         */	movw $BOOT_SEG, %ax /* le bootsecteur est a 0x7C00 en lineaire */	movw %ax, %ds      /* on le copie a l'adresse COPY_ADRESS */	xorw %si, %si      /* comme cette adresse est la plus haute de la mem */	xorw %di, %di      /* on pourra charger un kernel + gros */	movw $(BOOT_SIZE>>1), %cx	movw $COPY_SEG, %ax	movw %ax, %es	cld	rep ; movsw				/* on continue a executer le bootsecteur, mais maintenant a           partir de 0x9F000, qu'on represente sous la forme           0x9f00:offset */	ljmp $COPY_SEG, $here	/*	 * A partir de maintenant, on est a un offset 0 en memoire         * (segment 0x9f00), conformement a ce que veut le compilo.	 */here:	movw %ax, %ds	/* Petite pile temporaire (1k - 3.84k en RAM ; les adresses 0-1k	   correspondent au vecteur d'interruptions). */	movw %ax, %ss    	movw $(LOAD_ADRESS - 0x10), %sp        	/* Efface l'ecran */	movb $0x0, %ah	movb $0x3, %al	int 	$0x10	/* Affiche les messages d'attente */	movw $loadkern, %si	call message	movw $check, %si	call messagecheck386:	/*	 * la attention, plus complexe : on teste si le proc est un	 * 386+ pour cela, on va essayer de modifier les bits 12 ? 14	 * du registre E-flag si la modification reste, alors le proc	 * est un 386+, sinon, c'est =< 286	 *	 * Merci a Emmanuel Marty pour la compatibilite avec les 386	 * "pre-jurassique"	 */	pushf /* on sauvegarde le E-Flag */	movb $0x70, %ah	pushw %ax	popf	pushf	popw %ax	orb %ah, %ah	je no386  /* si la modif n'est pas valable, alors on saute a                     no386 */	popf      /* on les restaure ? la fin ... */	/* Message de confirmation de 386+ et d'attente */        movw $found386, %si	call message	movw $loading, %si	call message/* Copie du noyau disquette => RAM a partir de 0x1000   L'adresse de destination est d閒inie par es:0, o

⌨️ 快捷键说明

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