2007-07-04 23:21:01 +00:00
|
|
|
#define BOOT_SEG 0x07c0
|
|
|
|
#define EXEC_SEG 0x0100
|
|
|
|
#define STACK_SEG 0x0200
|
|
|
|
#define STACK_SIZE 0x2000
|
|
|
|
|
|
|
|
.text
|
|
|
|
.arch i386
|
|
|
|
.section ".prefix", "awx", @progbits
|
|
|
|
.code16
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find active partition
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* %dl : BIOS drive number
|
|
|
|
* %bp : Active partition handler routine
|
|
|
|
*/
|
|
|
|
find_active_partition:
|
|
|
|
/* Set up stack at STACK_SEG:STACK_SIZE */
|
|
|
|
movw $STACK_SEG, %ax
|
|
|
|
movw %ax, %ss
|
|
|
|
movw $STACK_SIZE, %sp
|
2007-07-05 11:36:39 +00:00
|
|
|
|
2007-07-04 23:21:01 +00:00
|
|
|
/* Relocate self to EXEC_SEG */
|
|
|
|
pushw $BOOT_SEG
|
|
|
|
popw %ds
|
|
|
|
pushw $EXEC_SEG
|
|
|
|
popw %es
|
|
|
|
xorw %si, %si
|
|
|
|
xorw %di, %di
|
|
|
|
movw $0x200, %cx
|
|
|
|
rep movsb
|
|
|
|
ljmp $EXEC_SEG, $1f
|
|
|
|
1: pushw %ds
|
|
|
|
popw %es
|
|
|
|
pushw %cs
|
|
|
|
popw %ds
|
2007-07-05 11:36:39 +00:00
|
|
|
|
|
|
|
/* Check for LBA extensions */
|
|
|
|
movb $0x41, %ah
|
|
|
|
movw $0x55aa, %bx
|
|
|
|
stc
|
|
|
|
int $0x13
|
|
|
|
jc 1f
|
|
|
|
cmpw $0xaa55, %bx
|
|
|
|
jne 1f
|
|
|
|
movw $read_lba, read_sector
|
|
|
|
1:
|
2007-07-04 23:21:01 +00:00
|
|
|
/* Read and process root partition table */
|
|
|
|
xorb %dh, %dh
|
|
|
|
movw $0x0001, %cx
|
|
|
|
xorl %esi, %esi
|
|
|
|
xorl %edi, %edi
|
|
|
|
call process_table
|
2007-07-05 11:36:39 +00:00
|
|
|
|
2007-07-04 23:21:01 +00:00
|
|
|
/* Print failure message */
|
|
|
|
movw $10f, %si
|
|
|
|
movw $(20f-10f), %cx
|
|
|
|
1: movw $0x0007, %bx
|
|
|
|
movb $0x0e, %ah
|
|
|
|
lodsb
|
|
|
|
int $0x10
|
|
|
|
loop 1b
|
2007-07-05 11:36:39 +00:00
|
|
|
|
2007-07-04 23:21:01 +00:00
|
|
|
/* Boot next device */
|
|
|
|
int $0x18
|
|
|
|
10: .ascii "Could not locate active partition\r\n"
|
|
|
|
20:
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Process partition table
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* %dl : BIOS drive number
|
|
|
|
* %dh : Head
|
|
|
|
* %cl : Sector (bits 0-5), high two bits of cylinder (bits 6-7)
|
|
|
|
* %ch : Low eight bits of cylinder
|
|
|
|
* %esi:%edi : LBA address
|
|
|
|
* %bp : Active partition handler routine
|
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* CF set on error
|
|
|
|
*/
|
|
|
|
process_table:
|
|
|
|
pushal
|
2007-07-05 11:36:39 +00:00
|
|
|
call read_boot_sector
|
2007-07-04 23:21:01 +00:00
|
|
|
jc 99f
|
2007-07-05 11:36:39 +00:00
|
|
|
movw $446, %bx
|
|
|
|
1: call process_partition
|
2007-07-04 23:21:01 +00:00
|
|
|
addw $16, %bx
|
|
|
|
cmpw $510, %bx
|
|
|
|
jne 1b
|
|
|
|
99: popal
|
|
|
|
ret
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Process partition
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* %dl : BIOS drive number
|
|
|
|
* %dh : Head
|
|
|
|
* %cl : Sector (bits 0-5), high two bits of cylinder (bits 6-7)
|
|
|
|
* %ch : Low eight bits of cylinder
|
|
|
|
* %esi:%edi : LBA address
|
|
|
|
* %bx : Offset within partition table
|
|
|
|
* %bp : Active partition handler routine
|
|
|
|
*/
|
|
|
|
process_partition:
|
|
|
|
pushal
|
|
|
|
/* Load C/H/S values from partition entry */
|
|
|
|
movb %es:1(%bx), %dh
|
|
|
|
movw %es:2(%bx), %cx
|
|
|
|
/* Update LBA address from partition entry */
|
|
|
|
addl %es:8(%bx), %edi
|
|
|
|
adcl $0, %esi
|
|
|
|
/* Check active flag */
|
|
|
|
testb $0x80, %es:(%bx)
|
|
|
|
jz 1f
|
2007-07-05 11:36:39 +00:00
|
|
|
call read_boot_sector
|
2007-07-04 23:21:01 +00:00
|
|
|
jc 99f
|
|
|
|
jmp *%bp
|
|
|
|
1: /* Check for extended partition */
|
|
|
|
movb %es:4(%bx), %al
|
|
|
|
cmpb $0x05, %al
|
|
|
|
je 2f
|
|
|
|
cmpb $0x0f, %al
|
|
|
|
je 2f
|
|
|
|
cmpb $0x85, %al
|
|
|
|
jne 99f
|
|
|
|
2: call process_table
|
|
|
|
99: popal
|
2007-07-05 11:36:39 +00:00
|
|
|
/* Reload original partition table */
|
|
|
|
call read_boot_sector
|
2007-07-04 23:21:01 +00:00
|
|
|
ret
|
|
|
|
|
|
|
|
/*
|
2007-07-05 11:36:39 +00:00
|
|
|
* Read single sector to %es:0000 and verify 0x55aa signature
|
2007-07-04 23:21:01 +00:00
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* %dl : BIOS drive number
|
|
|
|
* %dh : Head
|
|
|
|
* %cl : Sector (bits 0-5), high two bits of cylinder (bits 6-7)
|
|
|
|
* %ch : Low eight bits of cylinder
|
|
|
|
* %esi:%edi : LBA address
|
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* CF set on error
|
|
|
|
*/
|
2007-07-05 11:36:39 +00:00
|
|
|
read_boot_sector:
|
|
|
|
pushw %ax
|
|
|
|
movw $1, %ax
|
|
|
|
call *read_sector
|
|
|
|
jc 99f
|
|
|
|
cmpw $0xaa55, %es:(510)
|
|
|
|
je 99f
|
|
|
|
stc
|
|
|
|
99: popw %ax
|
|
|
|
ret
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read single sector to %es:0000 and verify 0x55aa signature
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* %dl : BIOS drive number
|
|
|
|
* %dh : Head
|
|
|
|
* %cl : Sector (bits 0-5), high two bits of cylinder (bits 6-7)
|
|
|
|
* %ch : Low eight bits of cylinder
|
|
|
|
* %esi:%edi : LBA address
|
|
|
|
* %ax : Number of sectors (max 127)
|
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* CF set on error
|
|
|
|
*/
|
|
|
|
read_sector: .word read_chs
|
|
|
|
|
2007-07-04 23:21:01 +00:00
|
|
|
read_chs:
|
2007-07-05 11:36:39 +00:00
|
|
|
/* Read sectors using C/H/S address */
|
|
|
|
pushal
|
2007-07-04 23:21:01 +00:00
|
|
|
xorw %bx, %bx
|
2007-07-05 11:36:39 +00:00
|
|
|
movb $0x02, %ah
|
2007-07-04 23:21:01 +00:00
|
|
|
stc
|
|
|
|
int $0x13
|
|
|
|
sti
|
2007-07-05 11:36:39 +00:00
|
|
|
popal
|
|
|
|
ret
|
|
|
|
|
2007-07-04 23:21:01 +00:00
|
|
|
read_lba:
|
2007-07-05 11:36:39 +00:00
|
|
|
/* Read sectors using LBA address */
|
|
|
|
pushal
|
|
|
|
movw %ax, (lba_desc + 2)
|
|
|
|
pushw %es
|
|
|
|
popw (lba_desc + 6)
|
2007-07-04 23:21:01 +00:00
|
|
|
movl %edi, (lba_desc + 8)
|
2007-07-05 11:36:39 +00:00
|
|
|
movl %esi, (lba_desc + 12)
|
2007-07-04 23:21:01 +00:00
|
|
|
movw $lba_desc, %si
|
2007-07-05 11:36:39 +00:00
|
|
|
movb $0x42, %ah
|
2007-07-04 23:21:01 +00:00
|
|
|
int $0x13
|
2007-07-05 11:36:39 +00:00
|
|
|
popal
|
2007-07-04 23:21:01 +00:00
|
|
|
ret
|
|
|
|
|
|
|
|
lba_desc:
|
|
|
|
.byte 0x10
|
|
|
|
.byte 0
|
|
|
|
.word 1
|
|
|
|
.word 0x0000
|
2007-07-05 11:36:39 +00:00
|
|
|
.word 0x0000
|
2007-07-04 23:21:01 +00:00
|
|
|
.long 0, 0
|