2008-02-03 08:27:15 +00:00
|
|
|
/* -*-Asm-*- */
|
|
|
|
/*
|
|
|
|
* GRUB -- GRand Unified Bootloader
|
|
|
|
* Copyright (C) 2008 Free Software Foundation, Inc.
|
|
|
|
*
|
|
|
|
* GRUB is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* GRUB is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with GRUB. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <grub/boot.h>
|
|
|
|
#include <grub/machine/boot.h>
|
|
|
|
#include <grub/machine/kernel.h>
|
|
|
|
#include <multiboot.h>
|
|
|
|
|
|
|
|
.file "cdboot.S"
|
|
|
|
|
|
|
|
#define CODE_ADDR 0x6000
|
|
|
|
#define DATA_ADDR ((GRUB_BOOT_MACHINE_KERNEL_ADDR) + 0x200)
|
|
|
|
|
|
|
|
#define CDSEC_SHIFT 11
|
|
|
|
#define CDBLK_LENG 16
|
|
|
|
|
|
|
|
.text
|
|
|
|
|
|
|
|
.code16
|
|
|
|
|
|
|
|
.globl start, _start
|
|
|
|
|
|
|
|
start:
|
2008-07-03 00:52:28 +00:00
|
|
|
_start:
|
2008-02-03 08:27:15 +00:00
|
|
|
call next
|
|
|
|
|
|
|
|
next:
|
|
|
|
jmp 1f
|
|
|
|
|
|
|
|
. = start + 8
|
|
|
|
|
|
|
|
bi_pvd:
|
2009-05-04 20:06:05 +00:00
|
|
|
.long 0 /* LBA of primary volume descriptor. */
|
2008-02-03 08:27:15 +00:00
|
|
|
bi_file:
|
|
|
|
.long 0 /* LBA of boot file. */
|
|
|
|
bi_length:
|
|
|
|
.long 0 /* Length of boot file. */
|
|
|
|
bi_csum:
|
|
|
|
.long 0 /* Checksum of boot file */
|
|
|
|
bi_reserved:
|
|
|
|
.space (10*4) /* Reserved */
|
|
|
|
|
|
|
|
1:
|
|
|
|
popw %bx
|
|
|
|
|
|
|
|
/* Boot from CDROM. */
|
|
|
|
|
|
|
|
xorw %ax, %ax
|
|
|
|
movw %ax, %ss
|
|
|
|
movw $(CODE_ADDR), %sp
|
|
|
|
movw %ax, %ds
|
|
|
|
movw %ax, %es
|
|
|
|
|
2009-06-04 20:01:19 +00:00
|
|
|
#ifdef APPLE_CC
|
|
|
|
err_noboot_msg_abs = 0x7C00 + err_noboot_msg - start
|
|
|
|
movw $err_noboot_msg_abs, %si
|
|
|
|
bi_length_dif = bi_length - next
|
|
|
|
movl %cs:bi_length_dif(%bx), %ecx
|
|
|
|
#else
|
2008-02-03 08:27:15 +00:00
|
|
|
movw $(0x7C00 + err_noboot_msg - start), %si
|
|
|
|
movl %cs: bi_length - next(%bx), %ecx
|
2009-06-04 20:01:19 +00:00
|
|
|
#endif
|
2008-02-03 08:27:15 +00:00
|
|
|
orl %ecx, %ecx
|
|
|
|
jz fail
|
|
|
|
|
|
|
|
addl $((1 << CDSEC_SHIFT) - 1), %ecx
|
|
|
|
shrl $CDSEC_SHIFT, %ecx
|
|
|
|
|
|
|
|
movl %cs: bi_file - next(%bx), %esi
|
|
|
|
|
|
|
|
call read_cdrom
|
|
|
|
|
2008-02-03 18:56:08 +00:00
|
|
|
/* Root drive will default to boot drive */
|
|
|
|
movb $0xFF, %dh
|
2009-06-10 21:04:23 +00:00
|
|
|
|
2008-02-03 08:27:15 +00:00
|
|
|
ljmp $(DATA_ADDR >> 4), $0
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Parameters:
|
|
|
|
* esi: start sector
|
|
|
|
* ecx: number of sectors
|
|
|
|
*/
|
|
|
|
read_cdrom:
|
|
|
|
xorl %eax, %eax
|
|
|
|
|
|
|
|
/* Number of blocks to read. */
|
|
|
|
pushw $CDBLK_LENG
|
|
|
|
|
|
|
|
/* Block number. */
|
|
|
|
pushl %eax
|
|
|
|
pushl %esi
|
|
|
|
|
|
|
|
/* Buffer address. */
|
|
|
|
pushw $((DATA_ADDR - 0x400)>> 4)
|
|
|
|
pushl %eax
|
|
|
|
pushw $0x10
|
|
|
|
|
|
|
|
xorl %edi, %edi
|
|
|
|
movw %sp, %si
|
|
|
|
|
|
|
|
1:
|
|
|
|
movw 0x10(%si), %di
|
|
|
|
cmpl %ecx, %edi
|
|
|
|
jbe 2f
|
|
|
|
movl %ecx, %edi
|
|
|
|
|
|
|
|
2:
|
|
|
|
mov %di, 2(%si)
|
|
|
|
|
|
|
|
pushl %ecx
|
|
|
|
|
|
|
|
movb $0x42, %ah
|
|
|
|
int $0x13
|
|
|
|
|
|
|
|
jnc 3f
|
|
|
|
|
|
|
|
movb $0x42, %ah /* Try again. */
|
|
|
|
int $0x13
|
|
|
|
|
|
|
|
jnc 3f
|
|
|
|
|
|
|
|
2:
|
|
|
|
shrw $1, %di /* Reduce transfer size. */
|
|
|
|
jz cdrom_fail
|
|
|
|
movw %di, 0x10(%si)
|
|
|
|
movw %di, 2(%si)
|
|
|
|
movb $0x42, %ah
|
|
|
|
int $0x13
|
|
|
|
jc 2b
|
|
|
|
|
|
|
|
3:
|
|
|
|
|
|
|
|
movw %di, %ax
|
|
|
|
shlw $(CDSEC_SHIFT - 4), %ax
|
|
|
|
addw %ax, 6(%si)
|
|
|
|
addl %edi, 8(%si)
|
|
|
|
|
|
|
|
popl %ecx
|
|
|
|
subl %edi, %ecx
|
|
|
|
jnz 1b
|
|
|
|
|
|
|
|
addw $0x12, %sp
|
|
|
|
ret
|
|
|
|
|
|
|
|
cdrom_fail:
|
2009-06-04 20:01:19 +00:00
|
|
|
#ifdef APPLE_CC
|
|
|
|
err_cdfail_msg_abs = 0x7C00 + err_cdfail_msg - start
|
|
|
|
movw $(err_cdfail_msg_abs), %si
|
|
|
|
#else
|
2008-02-03 08:27:15 +00:00
|
|
|
movw $(0x7C00 + err_cdfail_msg - start), %si
|
2009-06-04 20:01:19 +00:00
|
|
|
#endif
|
2008-02-03 08:27:15 +00:00
|
|
|
|
|
|
|
fail:
|
|
|
|
movb $0x0e, %ah
|
|
|
|
xorw %bx, %bx
|
|
|
|
1:
|
|
|
|
lodsb (%si), %al
|
|
|
|
int $0x10
|
|
|
|
cmpb $0, %al
|
|
|
|
jne 1b
|
|
|
|
1: jmp 1b
|
|
|
|
|
|
|
|
err_noboot_msg:
|
|
|
|
.ascii "no boot info\0"
|
|
|
|
|
|
|
|
err_cdfail_msg:
|
|
|
|
.ascii "cdrom read fails\0"
|
|
|
|
|
|
|
|
. = start + 0x1FF
|
|
|
|
|
|
|
|
.byte 0
|