2011-11-12 18:53:25 +00:00
|
|
|
/*
|
|
|
|
* GRUB -- GRand Unified Bootloader
|
|
|
|
* Copyright (C) 1999,2000,2001,2002,2003,2005,2006,2007,2008,2009,2011 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 <config.h>
|
|
|
|
#include <grub/symbol.h>
|
|
|
|
#include <grub/offsets.h>
|
|
|
|
#include <grub/machine/boot.h>
|
|
|
|
#include <grub/machine/memory.h>
|
|
|
|
#include <grub/machine/kernel.h>
|
|
|
|
|
|
|
|
#define ABS(x) ((x) - LOCAL (base) + GRUB_BOOT_MACHINE_KERNEL_ADDR + 0x200)
|
|
|
|
|
|
|
|
.file "startup_raw.S"
|
|
|
|
|
|
|
|
.text
|
|
|
|
|
|
|
|
/* Tell GAS to generate 16-bit instructions so that this code works
|
|
|
|
in real mode. */
|
|
|
|
.code16
|
|
|
|
|
|
|
|
.globl start, _start
|
|
|
|
start:
|
|
|
|
_start:
|
|
|
|
LOCAL (base):
|
|
|
|
/*
|
|
|
|
* Guarantee that "main" is loaded at 0x0:0x8200.
|
|
|
|
*/
|
|
|
|
#ifdef __APPLE__
|
|
|
|
ljmp $0, $(ABS(LOCAL (codestart)) - 0x10000)
|
|
|
|
#else
|
|
|
|
ljmp $0, $ABS(LOCAL (codestart))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is a special data area.
|
|
|
|
*/
|
|
|
|
|
|
|
|
. = _start + GRUB_DECOMPRESSOR_MACHINE_COMPRESSED_SIZE
|
|
|
|
LOCAL(compressed_size):
|
|
|
|
.long 0
|
|
|
|
. = _start + GRUB_DECOMPRESSOR_MACHINE_UNCOMPRESSED_SIZE
|
|
|
|
LOCAL(uncompressed_size):
|
|
|
|
.long 0
|
|
|
|
|
|
|
|
. = _start + GRUB_KERNEL_I386_PC_REED_SOLOMON_REDUNDANCY
|
|
|
|
reed_solomon_redundancy:
|
|
|
|
.long 0
|
2012-01-24 13:39:29 +00:00
|
|
|
. = _start + GRUB_KERNEL_I386_PC_NO_REED_SOLOMON_LENGTH
|
|
|
|
.short (LOCAL(reed_solomon_part) - _start)
|
2011-11-12 18:53:25 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* This is the area for all of the special variables.
|
|
|
|
*/
|
2012-01-29 22:20:02 +00:00
|
|
|
. = _start + GRUB_DECOMPRESSOR_I386_PC_BOOT_DEVICE
|
2011-11-12 20:12:52 +00:00
|
|
|
LOCAL(boot_dev):
|
|
|
|
.byte 0xFF, 0xFF, 0xFF
|
2011-11-12 18:53:25 +00:00
|
|
|
LOCAL(boot_drive):
|
2011-11-12 20:12:52 +00:00
|
|
|
.byte 0x00
|
2011-11-12 18:53:25 +00:00
|
|
|
|
|
|
|
/* the real mode code continues... */
|
|
|
|
LOCAL (codestart):
|
|
|
|
cli /* we're not safe here! */
|
|
|
|
|
|
|
|
/* set up %ds, %ss, and %es */
|
|
|
|
xorw %ax, %ax
|
|
|
|
movw %ax, %ds
|
|
|
|
movw %ax, %ss
|
|
|
|
movw %ax, %es
|
|
|
|
|
|
|
|
/* set up the real mode/BIOS stack */
|
|
|
|
movl $GRUB_MEMORY_MACHINE_REAL_STACK, %ebp
|
|
|
|
movl %ebp, %esp
|
|
|
|
|
|
|
|
sti /* we're safe again */
|
|
|
|
|
|
|
|
/* save the boot drive */
|
|
|
|
ADDR32 movb %dl, LOCAL(boot_drive)
|
|
|
|
|
|
|
|
/* reset disk system (%ah = 0) */
|
|
|
|
int $0x13
|
|
|
|
|
|
|
|
/* transition to protected mode */
|
|
|
|
DATA32 call real_to_prot
|
|
|
|
|
|
|
|
/* The ".code32" directive takes GAS out of 16-bit mode. */
|
|
|
|
.code32
|
|
|
|
|
|
|
|
incl %eax
|
2011-12-15 18:22:36 +00:00
|
|
|
cld
|
2011-11-12 18:53:25 +00:00
|
|
|
call grub_gate_a20
|
|
|
|
|
|
|
|
movl LOCAL(compressed_size), %edx
|
2012-05-28 15:51:57 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
addl $decompressor_end, %edx
|
|
|
|
subl $(LOCAL(reed_solomon_part)), %edx
|
|
|
|
#else
|
2012-03-02 10:24:24 +00:00
|
|
|
addl $(LOCAL(decompressor_end) - LOCAL(reed_solomon_part)), %edx
|
2012-05-28 15:51:57 +00:00
|
|
|
#endif
|
2011-11-12 18:53:25 +00:00
|
|
|
movl reed_solomon_redundancy, %ecx
|
2012-01-24 13:39:29 +00:00
|
|
|
leal LOCAL(reed_solomon_part), %eax
|
2011-12-15 18:22:36 +00:00
|
|
|
cld
|
2011-11-12 18:53:25 +00:00
|
|
|
call EXT_C (grub_reed_solomon_recover)
|
|
|
|
jmp post_reed_solomon
|
|
|
|
|
2011-12-15 18:20:41 +00:00
|
|
|
#include "../../../kern/i386/realmode.S"
|
|
|
|
|
2011-11-12 18:53:25 +00:00
|
|
|
#include <rs_decoder.S>
|
|
|
|
|
|
|
|
.text
|
|
|
|
|
|
|
|
/*
|
|
|
|
* grub_gate_a20(int on)
|
|
|
|
*
|
|
|
|
* Gate address-line 20 for high memory.
|
|
|
|
*
|
|
|
|
* This routine is probably overconservative in what it does, but so what?
|
|
|
|
*
|
|
|
|
* It also eats any keystrokes in the keyboard buffer. :-(
|
|
|
|
*/
|
|
|
|
|
|
|
|
grub_gate_a20:
|
|
|
|
movl %eax, %edx
|
|
|
|
|
|
|
|
gate_a20_test_current_state:
|
|
|
|
/* first of all, test if already in a good state */
|
|
|
|
call gate_a20_check_state
|
|
|
|
cmpb %al, %dl
|
|
|
|
jnz gate_a20_try_bios
|
|
|
|
ret
|
|
|
|
|
|
|
|
gate_a20_try_bios:
|
|
|
|
/* second, try a BIOS call */
|
|
|
|
pushl %ebp
|
|
|
|
call prot_to_real
|
|
|
|
|
|
|
|
.code16
|
|
|
|
movw $0x2400, %ax
|
|
|
|
testb %dl, %dl
|
|
|
|
jz 1f
|
|
|
|
incw %ax
|
|
|
|
1: int $0x15
|
|
|
|
|
|
|
|
DATA32 call real_to_prot
|
|
|
|
.code32
|
|
|
|
|
|
|
|
popl %ebp
|
|
|
|
call gate_a20_check_state
|
|
|
|
cmpb %al, %dl
|
|
|
|
jnz gate_a20_try_system_control_port_a
|
|
|
|
ret
|
|
|
|
|
|
|
|
gate_a20_try_system_control_port_a:
|
|
|
|
/*
|
|
|
|
* In macbook, the keyboard test would hang the machine, so we move
|
|
|
|
* this forward.
|
|
|
|
*/
|
|
|
|
/* fourth, try the system control port A */
|
|
|
|
inb $0x92
|
|
|
|
andb $(~0x03), %al
|
|
|
|
testb %dl, %dl
|
|
|
|
jz 6f
|
|
|
|
orb $0x02, %al
|
|
|
|
6: outb $0x92
|
|
|
|
|
|
|
|
/* When turning off Gate A20, do not check the state strictly,
|
|
|
|
because a failure is not fatal usually, and Gate A20 is always
|
|
|
|
on some modern machines. */
|
|
|
|
testb %dl, %dl
|
|
|
|
jz 7f
|
|
|
|
call gate_a20_check_state
|
|
|
|
cmpb %al, %dl
|
|
|
|
jnz gate_a20_try_keyboard_controller
|
|
|
|
7: ret
|
|
|
|
|
|
|
|
gate_a20_flush_keyboard_buffer:
|
|
|
|
inb $0x64
|
|
|
|
andb $0x02, %al
|
|
|
|
jnz gate_a20_flush_keyboard_buffer
|
|
|
|
2:
|
|
|
|
inb $0x64
|
|
|
|
andb $0x01, %al
|
|
|
|
jz 3f
|
|
|
|
inb $0x60
|
|
|
|
jmp 2b
|
|
|
|
3:
|
|
|
|
ret
|
|
|
|
|
|
|
|
gate_a20_try_keyboard_controller:
|
|
|
|
/* third, try the keyboard controller */
|
|
|
|
call gate_a20_flush_keyboard_buffer
|
|
|
|
|
|
|
|
movb $0xd1, %al
|
|
|
|
outb $0x64
|
|
|
|
4:
|
|
|
|
inb $0x64
|
|
|
|
andb $0x02, %al
|
|
|
|
jnz 4b
|
|
|
|
|
|
|
|
movb $0xdd, %al
|
|
|
|
testb %dl, %dl
|
|
|
|
jz 5f
|
|
|
|
orb $0x02, %al
|
|
|
|
5: outb $0x60
|
|
|
|
call gate_a20_flush_keyboard_buffer
|
|
|
|
|
|
|
|
/* output a dummy command (USB keyboard hack) */
|
|
|
|
movb $0xff, %al
|
|
|
|
outb $0x64
|
|
|
|
call gate_a20_flush_keyboard_buffer
|
|
|
|
|
|
|
|
call gate_a20_check_state
|
|
|
|
cmpb %al, %dl
|
|
|
|
/* everything failed, so restart from the beginning */
|
|
|
|
jnz gate_a20_try_bios
|
|
|
|
ret
|
|
|
|
|
|
|
|
gate_a20_check_state:
|
|
|
|
/* iterate the checking for a while */
|
|
|
|
movl $100, %ecx
|
|
|
|
1:
|
|
|
|
call 3f
|
|
|
|
cmpb %al, %dl
|
|
|
|
jz 2f
|
|
|
|
loop 1b
|
|
|
|
2:
|
|
|
|
ret
|
|
|
|
3:
|
|
|
|
pushl %ebx
|
|
|
|
pushl %ecx
|
|
|
|
xorl %eax, %eax
|
|
|
|
/* compare the byte at 0x8000 with that at 0x108000 */
|
|
|
|
movl $GRUB_BOOT_MACHINE_KERNEL_ADDR, %ebx
|
|
|
|
pushl %ebx
|
|
|
|
/* save the original byte in CL */
|
|
|
|
movb (%ebx), %cl
|
|
|
|
/* store the value at 0x108000 in AL */
|
|
|
|
addl $0x100000, %ebx
|
|
|
|
movb (%ebx), %al
|
|
|
|
/* try to set one less value at 0x8000 */
|
|
|
|
popl %ebx
|
|
|
|
movb %al, %ch
|
|
|
|
decb %ch
|
|
|
|
movb %ch, (%ebx)
|
|
|
|
/* serialize */
|
|
|
|
outb %al, $0x80
|
|
|
|
outb %al, $0x80
|
|
|
|
/* obtain the value at 0x108000 in CH */
|
|
|
|
pushl %ebx
|
|
|
|
addl $0x100000, %ebx
|
|
|
|
movb (%ebx), %ch
|
|
|
|
/* this result is 1 if A20 is on or 0 if it is off */
|
|
|
|
subb %ch, %al
|
|
|
|
xorb $1, %al
|
|
|
|
/* restore the original */
|
|
|
|
popl %ebx
|
|
|
|
movb %cl, (%ebx)
|
|
|
|
popl %ecx
|
|
|
|
popl %ebx
|
|
|
|
ret
|
|
|
|
|
2012-01-24 13:39:29 +00:00
|
|
|
LOCAL(reed_solomon_part):
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Support for booting GRUB from a Multiboot boot loader (e.g. GRUB itself).
|
|
|
|
* This uses the a.out kludge to load raw binary to the area starting at 1MB,
|
|
|
|
* and relocates itself after loaded.
|
|
|
|
*/
|
|
|
|
.p2align 2 /* force 4-byte alignment */
|
|
|
|
multiboot_header:
|
|
|
|
/* magic */
|
|
|
|
.long 0x1BADB002
|
|
|
|
/* flags */
|
|
|
|
.long (1 << 16)
|
|
|
|
/* checksum */
|
|
|
|
.long -0x1BADB002 - (1 << 16)
|
|
|
|
/* header addr */
|
|
|
|
.long multiboot_header - _start + 0x100000 + 0x200
|
|
|
|
/* load addr */
|
|
|
|
.long 0x100000
|
|
|
|
/* load end addr */
|
|
|
|
.long 0
|
|
|
|
/* bss end addr */
|
|
|
|
.long 0
|
|
|
|
/* entry addr */
|
|
|
|
.long multiboot_entry - _start + 0x100000 + 0x200
|
|
|
|
|
|
|
|
multiboot_entry:
|
|
|
|
.code32
|
|
|
|
/* obtain the boot device */
|
|
|
|
movl 12(%ebx), %edx
|
|
|
|
|
|
|
|
movl $GRUB_MEMORY_MACHINE_PROT_STACK, %ebp
|
|
|
|
movl %ebp, %esp
|
|
|
|
|
|
|
|
/* relocate the code */
|
2012-05-28 15:51:57 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
LOCAL(compressed_size_offset) = LOCAL(compressed_size) - LOCAL(base)
|
|
|
|
movl $0x200, %ecx
|
|
|
|
addl $decompressor_end, %ecx
|
|
|
|
subl $LOCAL(base), %ecx
|
|
|
|
addl LOCAL(compressed_size_offset) + 0x100000 + 0x200, %ecx
|
|
|
|
#else
|
2012-03-08 17:54:25 +00:00
|
|
|
movl $(LOCAL(decompressor_end) - _start + 0x200), %ecx
|
2012-01-24 13:39:29 +00:00
|
|
|
addl LOCAL(compressed_size) - _start + 0x100000 + 0x200, %ecx
|
2012-05-28 15:51:57 +00:00
|
|
|
#endif
|
2012-01-24 13:39:29 +00:00
|
|
|
movl $0x100000, %esi
|
|
|
|
movl $GRUB_BOOT_MACHINE_KERNEL_ADDR, %edi
|
|
|
|
cld
|
|
|
|
rep
|
|
|
|
movsb
|
|
|
|
/* jump to the real address */
|
|
|
|
movl $multiboot_trampoline, %eax
|
|
|
|
jmp *%eax
|
|
|
|
|
|
|
|
multiboot_trampoline:
|
|
|
|
/* fill the boot information */
|
|
|
|
movl %edx, LOCAL(boot_dev)
|
|
|
|
shrl $24, %edx
|
|
|
|
/* enter the usual booting */
|
|
|
|
call prot_to_real
|
|
|
|
.code16
|
|
|
|
jmp LOCAL (codestart)
|
|
|
|
.code32
|
|
|
|
|
|
|
|
post_reed_solomon:
|
|
|
|
|
|
|
|
#ifdef ENABLE_LZMA
|
|
|
|
movl $GRUB_MEMORY_MACHINE_DECOMPRESSION_ADDR, %edi
|
2012-05-28 15:51:57 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
movl $decompressor_end, %esi
|
|
|
|
#else
|
2012-01-24 13:39:29 +00:00
|
|
|
movl $LOCAL(decompressor_end), %esi
|
2012-05-28 15:51:57 +00:00
|
|
|
#endif
|
2012-01-24 13:39:29 +00:00
|
|
|
pushl %edi
|
|
|
|
movl LOCAL (uncompressed_size), %ecx
|
|
|
|
leal (%edi, %ecx), %ebx
|
|
|
|
/* Don't remove this push: it's an argument. */
|
|
|
|
push %ecx
|
|
|
|
call _LzmaDecodeA
|
|
|
|
pop %ecx
|
|
|
|
/* _LzmaDecodeA clears DF, so no need to run cld */
|
|
|
|
popl %esi
|
|
|
|
#endif
|
|
|
|
|
|
|
|
movl LOCAL(boot_dev), %edx
|
|
|
|
movl $prot_to_real, %edi
|
|
|
|
movl $real_to_prot, %ecx
|
2012-02-26 21:10:13 +00:00
|
|
|
movl $LOCAL(realidt), %eax
|
2012-01-24 13:39:29 +00:00
|
|
|
jmp *%esi
|
|
|
|
|
2011-11-12 18:53:25 +00:00
|
|
|
#ifdef ENABLE_LZMA
|
|
|
|
#include "lzma_decode.S"
|
|
|
|
#endif
|
|
|
|
|
2011-12-15 18:17:36 +00:00
|
|
|
.p2align 4
|
2011-11-12 18:53:25 +00:00
|
|
|
|
2012-05-28 15:51:57 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
.zerofill __DATA, __aa_before_bss, decompressor_end, 10, 0
|
|
|
|
#else
|
2012-03-02 10:24:24 +00:00
|
|
|
.bss
|
2011-11-12 18:53:25 +00:00
|
|
|
LOCAL(decompressor_end):
|
2012-05-28 15:51:57 +00:00
|
|
|
#endif
|