* conf/i386-pc.rmk (kernel_img_SOURCES): Add `kern/i386/misc.S'.
        * conf/i386-ieee1275.rmk: Likewise.
        * conf/i386-coreboot.rmk: Likewise.
        * kern/i386/pc/startup.S (grub_stop): Remove function.
        * kern/i386/ieee1275/startup.S: Likewise.
        * kern/i386/coreboot/startup.S: Likewise.
        * kern/i386/misc.S (grub_stop): New function.
		
	
			
		
			
				
	
	
		
			2115 lines
		
	
	
	
		
			38 KiB
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
			
		
		
	
	
			2115 lines
		
	
	
	
		
			38 KiB
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
| /*
 | |
|  *  GRUB  --  GRand Unified Bootloader
 | |
|  *  Copyright (C) 1999,2000,2001,2002,2003,2005,2006,2007,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/>.
 | |
|  */
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Note: These functions defined in this file may be called from C.
 | |
|  *       Be careful of that you must not modify some registers. Quote
 | |
|  *       from gcc-2.95.2/gcc/config/i386/i386.h:
 | |
| 
 | |
|    1 for registers not available across function calls.
 | |
|    These must include the FIXED_REGISTERS and also any
 | |
|    registers that can be used without being saved.
 | |
|    The latter must include the registers where values are returned
 | |
|    and the register where structure-value addresses are passed.
 | |
|    Aside from that, you can include as many other registers as you like.
 | |
| 
 | |
|   ax,dx,cx,bx,si,di,bp,sp,st,st1,st2,st3,st4,st5,st6,st7,arg
 | |
| {  1, 1, 1, 0, 0, 0, 0, 1, 1,  1,  1,  1,  1,  1,  1,  1,  1 }
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * Note: GRUB is compiled with the options -mrtd and -mregparm=3.
 | |
|  *       So the first three arguments are passed in %eax, %edx, and %ecx,
 | |
|  *       respectively, and if a function has a fixed number of arguments
 | |
|  *       and the number if greater than three, the function must return
 | |
|  *       with "ret $N" where N is ((the number of arguments) - 3) * 4.
 | |
|  */
 | |
| 
 | |
| #include <config.h>
 | |
| #include <grub/symbol.h>
 | |
| #include <grub/boot.h>
 | |
| #include <grub/machine/boot.h>
 | |
| #include <grub/machine/memory.h>
 | |
| #include <grub/machine/console.h>
 | |
| #include <grub/cpu/linux.h>
 | |
| #include <grub/machine/kernel.h>
 | |
| #include <grub/term.h>
 | |
| #include <multiboot.h>
 | |
| #include <multiboot2.h>
 | |
| 
 | |
| #define ABS(x)	((x) - _start + GRUB_BOOT_MACHINE_KERNEL_ADDR + 0x200)
 | |
| 
 | |
| 	.file	"startup.S"
 | |
| 
 | |
| 	.text
 | |
| 
 | |
| 	/* Tell GAS to generate 16-bit instructions so that this code works
 | |
| 	   in real mode. */
 | |
| 	.code16
 | |
| 
 | |
| 	.globl	start, _start
 | |
| start:
 | |
| _start:
 | |
| 	/*
 | |
| 	 *  Guarantee that "main" is loaded at 0x0:0x8200.
 | |
| 	 */
 | |
| #ifdef APPLE_CC
 | |
| 	codestart_abs = ABS(codestart) - 0x10000
 | |
| 	ljmp $0, $(codestart_abs)
 | |
| #else
 | |
| 	ljmp $0, $ABS(codestart)
 | |
| #endif
 | |
| 
 | |
| 	/*
 | |
| 	 *  Compatibility version number
 | |
| 	 *
 | |
| 	 *  These MUST be at byte offset 6 and 7 of the executable
 | |
| 	 *  DO NOT MOVE !!!
 | |
| 	 */
 | |
| 	. = _start + 0x6
 | |
| 	.byte	GRUB_BOOT_VERSION_MAJOR, GRUB_BOOT_VERSION_MINOR
 | |
| 
 | |
| 	/*
 | |
| 	 *  This is a special data area 8 bytes from the beginning.
 | |
| 	 */
 | |
| 
 | |
| 	. = _start + 0x8
 | |
| 
 | |
| VARIABLE(grub_total_module_size)
 | |
| 	.long	0
 | |
| VARIABLE(grub_kernel_image_size)
 | |
| 	.long	0
 | |
| VARIABLE(grub_compressed_size)
 | |
| 	.long	0
 | |
| VARIABLE(grub_install_dos_part)
 | |
| 	.long	0xFFFFFFFF
 | |
| VARIABLE(grub_install_bsd_part)
 | |
| 	.long	0xFFFFFFFF
 | |
| VARIABLE(grub_prefix)
 | |
| 	/* to be filled by grub-mkimage */
 | |
| 
 | |
| 	/*
 | |
| 	 *  Leave some breathing room for the prefix.
 | |
| 	 */
 | |
| 
 | |
| 	. = _start + GRUB_KERNEL_MACHINE_DATA_END
 | |
| 
 | |
| #ifdef APPLE_CC
 | |
| bss_start:
 | |
| 	.long 0
 | |
| bss_end:
 | |
| 	.long 0
 | |
| #endif
 | |
| 
 | |
| /*
 | |
|  * 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
 | |
| 
 | |
| 	/* relocate the code */
 | |
| 	movl	$(GRUB_KERNEL_MACHINE_RAW_SIZE + 0x200), %ecx
 | |
| 	addl	EXT_C(grub_compressed_size) - _start + 0x100000 + 0x200, %ecx
 | |
| 	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, %eax
 | |
| 	shrl	$8, %eax
 | |
| 	xorl	%ebx, %ebx
 | |
| 	cmpb	$0xFF, %ah
 | |
| 	je	1f
 | |
| 	movb	%ah, %bl
 | |
| 	movl	%ebx, EXT_C(grub_install_dos_part)
 | |
| 1:
 | |
| 	cmpb	$0xFF, %al
 | |
| 	je	2f
 | |
| 	movb	%al, %bl
 | |
| 	movl	%ebx, EXT_C(grub_install_bsd_part)
 | |
| 2:
 | |
| 	shrl	$24, %edx
 | |
|         movb    $0xFF, %dh
 | |
| 	/* enter the usual booting */
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| /* the real mode code continues... */
 | |
| 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, EXT_C(grub_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
 | |
| 	call	EXT_C(grub_gate_a20)
 | |
| 
 | |
| #if defined(ENABLE_LZO)
 | |
| 	/* decompress the compressed part and put the result at 1MB */
 | |
| 	movl	$GRUB_MEMORY_MACHINE_DECOMPRESSION_ADDR, %esi
 | |
| 	movl	$(_start + GRUB_KERNEL_MACHINE_RAW_SIZE), %edi
 | |
| 
 | |
| 	pushl	%esi
 | |
| 	pushl	EXT_C(grub_compressed_size)
 | |
| 	pushl	%edi
 | |
| 	call	lzo1x_decompress
 | |
| 	addl	$12, %esp
 | |
| 
 | |
| 	movl	%eax, %ecx
 | |
| 	cld
 | |
| #elif defined(ENABLE_LZMA)
 | |
| 	movl	$GRUB_MEMORY_MACHINE_DECOMPRESSION_ADDR, %edi
 | |
| 	movl	$(_start + GRUB_KERNEL_MACHINE_RAW_SIZE), %esi
 | |
| 	pushl	%edi
 | |
| 	pushl	%esi
 | |
| 	movl	EXT_C(grub_kernel_image_size), %ecx
 | |
| 	addl	EXT_C(grub_total_module_size), %ecx
 | |
| 	subl	$GRUB_KERNEL_MACHINE_RAW_SIZE, %ecx
 | |
| 	pushl	%ecx
 | |
| 	leal	(%edi, %ecx), %ebx
 | |
| 	call	_LzmaDecodeA
 | |
| 	/* _LzmaDecodeA clears DF, so no need to run cld */
 | |
| 	popl	%ecx
 | |
| 	popl	%edi
 | |
| 	popl	%esi
 | |
| #endif
 | |
| 
 | |
| 	/* copy back the decompressed part (except the modules) */
 | |
| 	subl	EXT_C(grub_total_module_size), %ecx
 | |
| 	rep
 | |
| 	movsb
 | |
| 
 | |
| #if 0
 | |
| 	/* copy modules before cleaning out the bss */
 | |
| 	movl	EXT_C(grub_total_module_size), %ecx
 | |
| 	movl	EXT_C(grub_kernel_image_size), %esi
 | |
| 	addl	%ecx, %esi
 | |
| 	addl	$_start, %esi
 | |
| 	decl	%esi
 | |
| 	movl	$END_SYMBOL, %edi
 | |
| 	addl	%ecx, %edi
 | |
| 	decl	%edi
 | |
| 	std
 | |
| 	rep
 | |
| 	movsb
 | |
| #endif
 | |
| 
 | |
| #ifdef APPLE_CC
 | |
| 	/* clean out the bss */
 | |
| 	bss_start_abs = ABS (bss_start)
 | |
| 	bss_end_abs = ABS (bss_end)
 | |
| 
 | |
| 	movl    bss_start_abs, %edi
 | |
| 
 | |
| 	/* compute the bss length */
 | |
| 	movl	bss_end_abs, %ecx
 | |
| 	subl	%edi, %ecx
 | |
| #else
 | |
| 	/* clean out the bss */
 | |
| 	movl	$BSS_START_SYMBOL, %edi
 | |
| 
 | |
| 	/* compute the bss length */
 | |
| 	movl	$END_SYMBOL, %ecx
 | |
| 	subl	%edi, %ecx
 | |
| #endif
 | |
| 
 | |
| 	/* clean out */
 | |
| 	xorl	%eax, %eax
 | |
| 	cld
 | |
| 	rep
 | |
| 	stosb
 | |
| 
 | |
| 	/*
 | |
| 	 *  Call the start of main body of C code.
 | |
| 	 */
 | |
| 	call EXT_C(grub_main)
 | |
| 
 | |
| /*
 | |
|  *  This is the area for all of the special variables.
 | |
|  */
 | |
| 
 | |
| VARIABLE(grub_boot_drive)
 | |
| 	.byte	0
 | |
| 
 | |
| 	.p2align	2	/* force 4-byte alignment */
 | |
| 
 | |
| #include "../realmode.S"
 | |
| 
 | |
| /*
 | |
|  * 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.  :-(
 | |
|  */
 | |
| 
 | |
| FUNCTION(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
 | |
| 
 | |
| #if defined(ENABLE_LZO)
 | |
| #include "lzo1x.S"
 | |
| #elif defined(ENABLE_LZMA)
 | |
| #include "lzma_decode.S"
 | |
| #endif
 | |
| 
 | |
| /*
 | |
|  * The code beyond this point is compressed.  Assert that the uncompressed
 | |
|  * code fits GRUB_KERNEL_MACHINE_RAW_SIZE.
 | |
|  */
 | |
| 	. = _start + GRUB_KERNEL_MACHINE_RAW_SIZE
 | |
| 
 | |
| 	/*
 | |
| 	 * This next part is sort of evil.  It takes advantage of the
 | |
| 	 * byte ordering on the x86 to work in either 16-bit or 32-bit
 | |
| 	 * mode, so think about it before changing it.
 | |
| 	 */
 | |
| 
 | |
| FUNCTION(grub_hard_stop)
 | |
| 	hlt
 | |
| 	jmp EXT_C(grub_hard_stop)
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * grub_stop_floppy()
 | |
|  *
 | |
|  * Stop the floppy drive from spinning, so that other software is
 | |
|  * jumped to with a known state.
 | |
|  */
 | |
| FUNCTION(grub_stop_floppy)
 | |
| 	movw	$0x3F2, %dx
 | |
| 	xorb	%al, %al
 | |
| 	outb	%al, %dx
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_exit()
 | |
|  *
 | |
|  * Exit the system.
 | |
|  */
 | |
| FUNCTION(grub_exit)
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 	/* Tell the BIOS a boot failure. If this does not work, reboot.  */
 | |
| 	int	$0x18
 | |
| 	jmp	cold_reboot
 | |
| 	.code32
 | |
| 
 | |
| /*
 | |
|  * grub_reboot()
 | |
|  *
 | |
|  * Reboot the system. At the moment, rely on BIOS.
 | |
|  */
 | |
| FUNCTION(grub_reboot)
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| cold_reboot:
 | |
| 	/* cold boot */
 | |
| 	movw	$0x0472, %di
 | |
| 	movw	%ax, (%di)
 | |
| 	ljmp	$0xFFFF, $0x0000
 | |
| 	.code32
 | |
| 
 | |
| /*
 | |
|  * grub_halt(int no_apm)
 | |
|  *
 | |
|  * Halt the system, using APM if possible. If NO_APM is true, don't use
 | |
|  * APM even if it is available.
 | |
|  */
 | |
| FUNCTION(grub_halt)
 | |
| 	/* see if zero */
 | |
| 	testl	%eax, %eax
 | |
| 	jnz	EXT_C(grub_stop)
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	/* detect APM */
 | |
| 	movw	$0x5300, %ax
 | |
| 	xorw	%bx, %bx
 | |
| 	int	$0x15
 | |
| 	jc	EXT_C(grub_hard_stop)
 | |
| 	/* don't check %bx for buggy BIOSes... */
 | |
| 
 | |
| 	/* disconnect APM first */
 | |
| 	movw	$0x5304, %ax
 | |
| 	xorw	%bx, %bx
 | |
| 	int	$0x15
 | |
| 
 | |
| 	/* connect APM */
 | |
| 	movw	$0x5301, %ax
 | |
| 	xorw	%bx, %bx
 | |
| 	int	$0x15
 | |
| 	jc	EXT_C(grub_hard_stop)
 | |
| 
 | |
| 	/* set APM protocol level - 1.1 or bust. (this covers APM 1.2 also) */
 | |
| 	movw	$0x530E, %ax
 | |
| 	xorw	%bx, %bx
 | |
| 	movw	$0x0101, %cx
 | |
| 	int	$0x15
 | |
| 	jc	EXT_C(grub_hard_stop)
 | |
| 
 | |
| 	/* set the power state to off */
 | |
| 	movw	$0x5307, %ax
 | |
| 	movw	$1, %bx
 | |
| 	movw	$3, %cx
 | |
| 	int	$0x15
 | |
| 
 | |
| 	/* shouldn't reach here */
 | |
| 	jmp	EXT_C(grub_hard_stop)
 | |
| 	.code32
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *  void grub_chainloader_real_boot (int drive, void *part_addr)
 | |
|  *
 | |
|  *  This starts another boot loader.
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_chainloader_real_boot)
 | |
| 	pushl	%edx
 | |
| 	pushl	%eax
 | |
| 
 | |
| 	call	EXT_C(grub_dl_unload_all)
 | |
| 
 | |
| 	/* Turn off Gate A20 */
 | |
| 	xorl	%eax, %eax
 | |
| 	call	EXT_C(grub_gate_a20)
 | |
| 
 | |
| 	/* set up to pass boot drive */
 | |
| 	popl	%edx
 | |
| 
 | |
| 	/* ESI must point to a partition table entry */
 | |
| 	popl	%esi
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 	ljmp	$0, $GRUB_MEMORY_MACHINE_BOOT_LOADER_ADDR
 | |
| 	.code32
 | |
| 
 | |
| #include "../loader.S"
 | |
| 
 | |
| /*
 | |
|  *   int grub_biosdisk_rw_int13_extensions (int ah, int drive, void *dap)
 | |
|  *
 | |
|  *   Call IBM/MS INT13 Extensions (int 13 %ah=AH) for DRIVE. DAP
 | |
|  *   is passed for disk address packet. If an error occurs, return
 | |
|  *   non-zero, otherwise zero.
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_biosdisk_rw_int13_extensions)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%esi
 | |
| 
 | |
| 	/* compute the address of disk_address_packet */
 | |
| 	movw	%cx, %si
 | |
| 	xorw	%cx, %cx
 | |
| 	shrl	$4, %ecx	/* save the segment to cx */
 | |
| 
 | |
| 	/* ah */
 | |
| 	movb	%al, %dh
 | |
| 	/* enter real mode */
 | |
| 	call	prot_to_real
 | |
| 
 | |
| 	.code16
 | |
| 	movb	%dh, %ah
 | |
| 	movw	%cx, %ds
 | |
| 	int	$0x13		/* do the operation */
 | |
| 	movb	%ah, %dl	/* save return value */
 | |
| 	/* back to protected mode */
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movb	%dl, %al	/* return value in %eax */
 | |
| 
 | |
| 	popl	%esi
 | |
| 	popl	%ebp
 | |
| 
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  *   int grub_biosdisk_rw_standard (int ah, int drive, int coff, int hoff,
 | |
|  *                                  int soff, int nsec, int segment)
 | |
|  *
 | |
|  *   Call standard and old INT13 (int 13 %ah=AH) for DRIVE. Read/write
 | |
|  *   NSEC sectors from COFF/HOFF/SOFF into SEGMENT. If an error occurs,
 | |
|  *   return non-zero, otherwise zero.
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_biosdisk_rw_standard)
 | |
| 	pushl	%ebp
 | |
| 	movl	%esp, %ebp
 | |
| 
 | |
| 	pushl	%ebx
 | |
| 	pushl	%edi
 | |
| 	pushl	%esi
 | |
| 
 | |
| 	/* set up CHS information */
 | |
| 
 | |
| 	/* set %ch to low eight bits of cylinder */
 | |
| 	xchgb	%cl, %ch
 | |
| 	/* set bits 6-7 of %cl to high two bits of cylinder */
 | |
| 	shlb	$6, %cl
 | |
| 	/* set bits 0-5 of %cl to sector */
 | |
| 	addb	0xc(%ebp), %cl
 | |
| 	/* set %dh to head */
 | |
| 	movb	0x8(%ebp), %dh
 | |
| 	/* set %ah to AH */
 | |
| 	movb	%al, %ah
 | |
| 	/* set %al to NSEC */
 | |
| 	movb	0x10(%ebp), %al
 | |
| 	/* save %ax in %di */
 | |
| 	movw	%ax, %di
 | |
| 	/* save SEGMENT in %bx */
 | |
| 	movw	0x14(%ebp), %bx
 | |
| 
 | |
| 	/* enter real mode */
 | |
| 	call	prot_to_real
 | |
| 
 | |
| 	.code16
 | |
| 	movw	%bx, %es
 | |
| 	xorw	%bx, %bx
 | |
| 	movw	$3, %si		/* attempt at least three times */
 | |
| 
 | |
| 1:
 | |
| 	movw	%di, %ax
 | |
| 	int	$0x13		/* do the operation */
 | |
| 	jnc	2f		/* check if successful */
 | |
| 
 | |
| 	movb	%ah, %bl	/* save return value */
 | |
| 	/* if fail, reset the disk system */
 | |
| 	xorw	%ax, %ax
 | |
| 	int	$0x13
 | |
| 
 | |
| 	decw	%si
 | |
| 	cmpw	$0, %si
 | |
| 	je	2f
 | |
| 	xorb	%bl, %bl
 | |
| 	jmp	1b		/* retry */
 | |
| 2:
 | |
| 	/* back to protected mode */
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movb	%bl, %al	/* return value in %eax */
 | |
| 
 | |
| 	popl	%esi
 | |
| 	popl	%edi
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 
 | |
| 	ret	$(4 * 4)
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *   int grub_biosdisk_check_int13_extensions (int drive)
 | |
|  *
 | |
|  *   Check if LBA is supported for DRIVE. If it is supported, then return
 | |
|  *   the major version of extensions, otherwise zero.
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_biosdisk_check_int13_extensions)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 
 | |
| 	/* drive */
 | |
| 	movb	%al, %dl
 | |
| 	/* enter real mode */
 | |
| 	call	prot_to_real
 | |
| 
 | |
| 	.code16
 | |
| 	movb	$0x41, %ah
 | |
| 	movw	$0x55aa, %bx
 | |
| 	int	$0x13		/* do the operation */
 | |
| 
 | |
| 	/* check the result */
 | |
| 	jc	1f
 | |
| 	cmpw	$0xaa55, %bx
 | |
| 	jne	1f
 | |
| 
 | |
| 	movb	%ah, %bl	/* save the major version into %bl */
 | |
| 
 | |
| 	/* check if AH=0x42 is supported */
 | |
| 	andw	$1, %cx
 | |
| 	jnz	2f
 | |
| 
 | |
| 1:
 | |
| 	xorb	%bl, %bl
 | |
| 2:
 | |
| 	/* back to protected mode */
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movb	%bl, %al	/* return value in %eax */
 | |
| 
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *   int grub_biosdisk_get_cdinfo_int13_extensions (int drive, void *cdrp)
 | |
|  *
 | |
|  *   Return the cdrom information of DRIVE in CDRP. If an error occurs,
 | |
|  *   then return non-zero, otherwise zero.
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_biosdisk_get_cdinfo_int13_extensions)
 | |
| 	movw	$0x4B01, %cx
 | |
| 	jmp	1f
 | |
| 
 | |
| /*
 | |
|  *   int grub_biosdisk_get_diskinfo_int13_extensions (int drive, void *drp)
 | |
|  *
 | |
|  *   Return the geometry of DRIVE in a drive parameters, DRP. If an error
 | |
|  *   occurs, then return non-zero, otherwise zero.
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_biosdisk_get_diskinfo_int13_extensions)
 | |
| 	movb	$0x48, %ch
 | |
| 1:
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 	pushl	%esi
 | |
| 
 | |
| 	/* compute the address of drive parameters */
 | |
| 	movw	%dx, %si
 | |
| 	andl	$0xf, %esi
 | |
| 	shrl	$4, %edx
 | |
| 	movw	%dx, %bx	/* save the segment into %bx */
 | |
| 	/* drive */
 | |
| 	movb	%al, %dl
 | |
| 	/* enter real mode */
 | |
| 	call	prot_to_real
 | |
| 
 | |
| 	.code16
 | |
| 	movw	%cx, %ax
 | |
| 	movw	%bx, %ds
 | |
| 	int	$0x13		/* do the operation */
 | |
| 	movb	%ah, %bl	/* save return value in %bl */
 | |
| 	/* back to protected mode */
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movb	%bl, %al	/* return value in %eax */
 | |
| 
 | |
| 	popl	%esi
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *   int grub_biosdisk_get_diskinfo_standard (int drive,
 | |
|  *                                            unsigned long *cylinders,
 | |
|  *                                            unsigned long *heads,
 | |
|  *                                            unsigned long *sectors)
 | |
|  *
 | |
|  *   Return the geometry of DRIVE in CYLINDERS, HEADS and SECTORS. If an
 | |
|  *   error occurs, then return non-zero, otherwise zero.
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_biosdisk_get_diskinfo_standard)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 	pushl	%edi
 | |
| 
 | |
| 	/* push CYLINDERS */
 | |
| 	pushl	%edx
 | |
| 	/* push HEADS */
 | |
| 	pushl	%ecx
 | |
| 	/* SECTORS is on the stack */
 | |
| 
 | |
| 	/* drive */
 | |
| 	movb	%al, %dl
 | |
| 	/* enter real mode */
 | |
| 	call	prot_to_real
 | |
| 
 | |
| 	.code16
 | |
| 	movb	$0x8, %ah
 | |
| 	int	$0x13		/* do the operation */
 | |
| 	/* check if successful */
 | |
| 	testb	%ah, %ah
 | |
| 	jnz	1f
 | |
| 	/* bogus BIOSes may not return an error number */
 | |
| 	testb	$0x3f, %cl	/* 0 sectors means no disk */
 | |
| 	jnz	1f		/* if non-zero, then succeed */
 | |
| 	/* XXX 0x60 is one of the unused error numbers */
 | |
| 	movb	$0x60, %ah
 | |
| 1:
 | |
| 	movb	%ah, %bl	/* save return value in %bl */
 | |
| 	/* back to protected mode */
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	/* pop HEADS */
 | |
| 	popl	%edi
 | |
| 	movb	%dh, %al
 | |
| 	incl	%eax	/* the number of heads is counted from zero */
 | |
| 	movl	%eax, (%edi)
 | |
| 
 | |
| 	/* pop CYLINDERS */
 | |
| 	popl	%edi
 | |
| 	movb	%ch, %al
 | |
| 	movb	%cl, %ah
 | |
| 	shrb	$6, %ah	/* the number of cylinders is counted from zero */
 | |
| 	incl	%eax
 | |
| 	movl	%eax, (%edi)
 | |
| 
 | |
| 	/* SECTORS */
 | |
| 	movl	0x10(%esp), %edi
 | |
| 	andb	$0x3f, %cl
 | |
| 	movzbl	%cl, %eax
 | |
| 	movl	%eax, (%edi)
 | |
| 
 | |
| 	xorl	%eax, %eax
 | |
| 	movb	%bl, %al	/* return value in %eax */
 | |
| 
 | |
| 	popl	%edi
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 
 | |
| 	ret	$4
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * int grub_biosdisk_get_num_floppies (void)
 | |
|  */
 | |
| FUNCTION(grub_biosdisk_get_num_floppies)
 | |
| 	pushl	%ebp
 | |
| 
 | |
| 	xorl	%edx, %edx
 | |
| 	call	prot_to_real
 | |
| 
 | |
| 	.code16
 | |
| 	/* reset the disk system first */
 | |
| 	int	$0x13
 | |
| 1:
 | |
| 	stc
 | |
| 
 | |
| 	/* call GET DISK TYPE */
 | |
| 	movb	$0x15, %ah
 | |
| 	int	$0x13
 | |
| 
 | |
| 	jc	2f
 | |
| 
 | |
| 	/* check if this drive exists */
 | |
| 	testb	$0x3, %ah
 | |
| 	jz	2f
 | |
| 
 | |
| 	incb	%dl
 | |
| 	cmpb	$2, %dl
 | |
| 	jne	1b
 | |
| 2:
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movl	%edx, %eax
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *
 | |
|  * grub_get_memsize(i) :  return the memory size in KB. i == 0 for conventional
 | |
|  *		memory, i == 1 for extended memory
 | |
|  *	BIOS call "INT 12H" to get conventional memory size
 | |
|  *	BIOS call "INT 15H, AH=88H" to get extended memory size
 | |
|  *		Both have the return value in AX.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_get_memsize)
 | |
| 	pushl	%ebp
 | |
| 
 | |
| 	movl	%eax, %edx
 | |
| 
 | |
| 	call	prot_to_real	/* enter real mode */
 | |
| 	.code16
 | |
| 
 | |
| 	testl	%edx, %edx
 | |
| 	jnz	xext
 | |
| 
 | |
| 	int	$0x12
 | |
| 	jmp	xdone
 | |
| 
 | |
| xext:
 | |
| 	movb	$0x88, %ah
 | |
| 	int	$0x15
 | |
| 
 | |
| xdone:
 | |
| 	movw	%ax, %dx
 | |
| 
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movw	%dx, %ax
 | |
| 
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *
 | |
|  * grub_get_eisa_mmap() :  return packed EISA memory map, lower 16 bits is
 | |
|  *		memory between 1M and 16M in 1K parts, upper 16 bits is
 | |
|  *		memory above 16M in 64K parts.  If error, return zero.
 | |
|  *	BIOS call "INT 15H, AH=E801H" to get EISA memory map,
 | |
|  *		AX = memory between 1M and 16M in 1K parts.
 | |
|  *		BX = memory above 16M in 64K parts.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_get_eisa_mmap)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 
 | |
| 	call	prot_to_real	/* enter real mode */
 | |
| 	.code16
 | |
| 
 | |
| 	movw	$0xe801, %ax
 | |
| 	int	$0x15
 | |
| 
 | |
| 	shll	$16, %ebx
 | |
| 	movw	%ax, %bx
 | |
| 
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	cmpb	$0x86, %bh
 | |
| 	je	xnoteisa
 | |
| 
 | |
| 	movl	%ebx, %eax
 | |
| 
 | |
| xnoteisa:
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  *
 | |
|  * grub_get_mmap_entry(addr, cont) : address and old continuation value (zero to
 | |
|  *		start), for the Query System Address Map BIOS call.
 | |
|  *
 | |
|  *  Sets the first 4-byte int value of "addr" to the size returned by
 | |
|  *  the call.  If the call fails, sets it to zero.
 | |
|  *
 | |
|  *	Returns:  new (non-zero) continuation value, 0 if done.
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_get_mmap_entry)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 	pushl	%edi
 | |
| 	pushl	%esi
 | |
| 
 | |
| 	/* push ADDR */
 | |
| 	pushl	%eax
 | |
| 
 | |
| 	/* place address (+4) in ES:DI */
 | |
| 	addl	$4, %eax
 | |
| 	movl	%eax, %edi
 | |
| 	andl	$0xf, %edi
 | |
| 	shrl	$4, %eax
 | |
| 	movl	%eax, %esi
 | |
| 
 | |
| 	/* set continuation value */
 | |
| 	movl	%edx, %ebx
 | |
| 
 | |
| 	/* set default maximum buffer size */
 | |
| 	movl	$0x14, %ecx
 | |
| 
 | |
| 	/* set EDX to 'SMAP' */
 | |
| 	movl	$0x534d4150, %edx
 | |
| 
 | |
| 	call	prot_to_real	/* enter real mode */
 | |
| 	.code16
 | |
| 
 | |
| 	movw	%si, %es
 | |
| 	movl	$0xe820, %eax
 | |
| 	int	$0x15
 | |
| 
 | |
| 	DATA32	jc	xnosmap
 | |
| 
 | |
| 	cmpl	$0x534d4150, %eax
 | |
| 	jne	xnosmap
 | |
| 
 | |
| 	cmpl	$0x14, %ecx
 | |
| 	jl	xnosmap
 | |
| 
 | |
| 	cmpl	$0x400, %ecx
 | |
| 	jg	xnosmap
 | |
| 
 | |
| 	jmp	xsmap
 | |
| 
 | |
| xnosmap:
 | |
| 	xorl	%ecx, %ecx
 | |
| 
 | |
| /* 	Apple's cc jumps few bytes before the correct
 | |
| 	label in this context. Hence nops. */
 | |
| #ifdef APPLE_CC
 | |
| 	nop
 | |
| 	nop
 | |
| 	nop
 | |
| 	nop
 | |
| 	nop
 | |
| 	nop
 | |
| #endif
 | |
| 
 | |
| xsmap:
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	/* write length of buffer (zero if error) into ADDR */
 | |
| 	popl	%eax
 | |
| 	movl	%ecx, (%eax)
 | |
| 
 | |
| 	/* set return value to continuation */
 | |
| 	movl	%ebx, %eax
 | |
| 
 | |
| 	popl	%esi
 | |
| 	popl	%edi
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * void grub_console_real_putchar (int c)
 | |
|  *
 | |
|  * Put the character C on the console. Because GRUB wants to write a
 | |
|  * character with an attribute, this implementation is a bit tricky.
 | |
|  * If C is a control character (CR, LF, BEL, BS), use INT 10, AH = 0Eh
 | |
|  * (TELETYPE OUTPUT). Otherwise, save the original position, put a space,
 | |
|  * save the current position, restore the original position, write the
 | |
|  * character and the attribute, and restore the current position.
 | |
|  *
 | |
|  * The reason why this is so complicated is that there is no easy way to
 | |
|  * get the height of the screen, and the TELETYPE OUTPUT BIOS call doesn't
 | |
|  * support setting a background attribute.
 | |
|  */
 | |
| FUNCTION(grub_console_real_putchar)
 | |
| 	movl	%eax, %edx
 | |
| 	pusha
 | |
| 	movb	EXT_C(grub_console_cur_color), %bl
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 	movb	%dl, %al
 | |
| 	xorb	%bh, %bh
 | |
| 
 | |
| 	/* use teletype output if control character */
 | |
| 	cmpb	$0x7, %al
 | |
| 	je	1f
 | |
| 	cmpb	$0x8, %al
 | |
| 	je	1f
 | |
| 	cmpb	$0xa, %al
 | |
| 	je	1f
 | |
| 	cmpb	$0xd, %al
 | |
| 	je	1f
 | |
| 
 | |
| 	/* save the character and the attribute on the stack */
 | |
| 	pushw	%ax
 | |
| 	pushw	%bx
 | |
| 
 | |
| 	/* get the current position */
 | |
| 	movb	$0x3, %ah
 | |
| 	int	$0x10
 | |
| 
 | |
| 	/* check the column with the width */
 | |
| 	cmpb	$79, %dl
 | |
| 	jl	2f
 | |
| 
 | |
| 	/* print CR and LF, if next write will exceed the width */
 | |
| 	movw	$0x0e0d, %ax
 | |
| 	int	$0x10
 | |
| 	movb	$0x0a, %al
 | |
| 	int	$0x10
 | |
| 
 | |
| 	/* get the current position */
 | |
| 	movb	$0x3, %ah
 | |
| 	int	$0x10
 | |
| 
 | |
| 2:
 | |
| 	/* restore the character and the attribute */
 | |
| 	popw	%bx
 | |
| 	popw	%ax
 | |
| 
 | |
| 	/* write the character with the attribute */
 | |
| 	movb	$0x9, %ah
 | |
| 	movw	$1, %cx
 | |
| 	int	$0x10
 | |
| 
 | |
| 	/* move the cursor forward */
 | |
| 	incb	%dl
 | |
| 	movb	$0x2, %ah
 | |
| 	int	$0x10
 | |
| 
 | |
| 	jmp	3f
 | |
| 
 | |
| 1:	movw	$1, %bx
 | |
| 	movb	$0xe, %ah
 | |
| 	int	$0x10
 | |
| 
 | |
| 3:	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	popa
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * int grub_console_getkey (void)
 | |
|  * BIOS call "INT 16H Function 00H" to read character from keyboard
 | |
|  *	Call with	%ah = 0x0
 | |
|  *	Return:		%ah = keyboard scan code
 | |
|  *			%al = ASCII character
 | |
|  */
 | |
| 
 | |
| /* this table is used in translate_keycode below */
 | |
| translation_table:
 | |
| 	.word	GRUB_CONSOLE_KEY_LEFT, GRUB_TERM_LEFT
 | |
| 	.word	GRUB_CONSOLE_KEY_RIGHT, GRUB_TERM_RIGHT
 | |
| 	.word	GRUB_CONSOLE_KEY_UP, GRUB_TERM_UP
 | |
| 	.word	GRUB_CONSOLE_KEY_DOWN, GRUB_TERM_DOWN
 | |
| 	.word	GRUB_CONSOLE_KEY_HOME, GRUB_TERM_HOME
 | |
| 	.word	GRUB_CONSOLE_KEY_END, GRUB_TERM_END
 | |
| 	.word	GRUB_CONSOLE_KEY_DC, GRUB_TERM_DC
 | |
| 	.word	GRUB_CONSOLE_KEY_BACKSPACE, GRUB_TERM_BACKSPACE
 | |
| 	.word	GRUB_CONSOLE_KEY_PPAGE, GRUB_TERM_PPAGE
 | |
| 	.word	GRUB_CONSOLE_KEY_NPAGE, GRUB_TERM_NPAGE
 | |
| 	.word	0
 | |
| 
 | |
| /*
 | |
|  * translate_keycode translates the key code %dx to an ascii code.
 | |
|  */
 | |
| 	.code16
 | |
| 
 | |
| translate_keycode:
 | |
| 	pushw	%bx
 | |
| 	pushw	%si
 | |
| 
 | |
| #ifdef APPLE_CC
 | |
| 	translation_table_abs = ABS (translation_table) - 0x10000
 | |
| 	movw	$(translation_table_abs), %si
 | |
| #else
 | |
| 	movw	$ABS(translation_table), %si
 | |
| #endif
 | |
| 
 | |
| 1:	lodsw
 | |
| 	/* check if this is the end */
 | |
| 	testw	%ax, %ax
 | |
| 	jz	2f
 | |
| 	/* load the ascii code into %ax */
 | |
| 	movw	%ax, %bx
 | |
| 	lodsw
 | |
| 	/* check if this matches the key code */
 | |
| 	cmpw	%bx, %dx
 | |
| 	jne	1b
 | |
| 	/* translate %dx, if successful */
 | |
| 	movw	%ax, %dx
 | |
| 
 | |
| 2:	popw	%si
 | |
| 	popw	%bx
 | |
| 	ret
 | |
| 
 | |
| 	.code32
 | |
| 
 | |
| FUNCTION(grub_console_getkey)
 | |
| 	pushl	%ebp
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	/*
 | |
| 	 * Due to a bug in apple's bootcamp implementation, INT 16/AH = 0 would
 | |
| 	 * cause the machine to hang at the second keystroke. However, we can
 | |
| 	 * work around this problem by ensuring the presence of keystroke with
 | |
| 	 * INT 16/AH = 1 before calling INT 16/AH = 0.
 | |
| 	 */
 | |
| 
 | |
| 1:
 | |
| 	movb	$1, %ah
 | |
| 	int	$0x16
 | |
| 	jnz	2f
 | |
| 	hlt
 | |
| 	jmp	1b
 | |
| 
 | |
| 2:
 | |
| 
 | |
| 	movb	$0, %ah
 | |
| 	int	$0x16
 | |
| 
 | |
| 	movw	%ax, %dx		/* real_to_prot uses %eax */
 | |
| 	call	translate_keycode
 | |
| 
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movw	%dx, %ax
 | |
| 
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * int grub_console_checkkey (void)
 | |
|  *	if there is a character pending, return it; otherwise return -1
 | |
|  * BIOS call "INT 16H Function 01H" to check whether a character is pending
 | |
|  *	Call with	%ah = 0x1
 | |
|  *	Return:
 | |
|  *		If key waiting to be input:
 | |
|  *			%ah = keyboard scan code
 | |
|  *			%al = ASCII character
 | |
|  *			Zero flag = clear
 | |
|  *		else
 | |
|  *			Zero flag = set
 | |
|  */
 | |
| FUNCTION(grub_console_checkkey)
 | |
| 	pushl	%ebp
 | |
| 	xorl	%edx, %edx
 | |
| 
 | |
| 	call	prot_to_real	/* enter real mode */
 | |
| 	.code16
 | |
| 
 | |
| 	movb	$0x1, %ah
 | |
| 	int	$0x16
 | |
| 
 | |
| 	jz	notpending
 | |
| 
 | |
| 	movw	%ax, %dx
 | |
| 	DATA32	jmp	pending
 | |
| 
 | |
| notpending:
 | |
| 	decl	%edx
 | |
| 
 | |
| pending:
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movl	%edx, %eax
 | |
| 
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * grub_uint16_t grub_console_getxy (void)
 | |
|  * BIOS call "INT 10H Function 03h" to get cursor position
 | |
|  *	Call with	%ah = 0x03
 | |
|  *			%bh = page
 | |
|  *      Returns         %ch = starting scan line
 | |
|  *                      %cl = ending scan line
 | |
|  *                      %dh = row (0 is top)
 | |
|  *                      %dl = column (0 is left)
 | |
|  */
 | |
| 
 | |
| 
 | |
| FUNCTION(grub_console_getxy)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx                    /* save EBX */
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
|         xorb	%bh, %bh                /* set page to 0 */
 | |
| 	movb	$0x3, %ah
 | |
| 	int	$0x10			/* get cursor position */
 | |
| 
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movb	%dl, %ah
 | |
| 	movb	%dh, %al
 | |
| 
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * void grub_console_gotoxy(grub_uint8_t x, grub_uint8_t y)
 | |
|  * BIOS call "INT 10H Function 02h" to set cursor position
 | |
|  *	Call with	%ah = 0x02
 | |
|  *			%bh = page
 | |
|  *                      %dh = row (0 is top)
 | |
|  *                      %dl = column (0 is left)
 | |
|  */
 | |
| 
 | |
| 
 | |
| FUNCTION(grub_console_gotoxy)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx                    /* save EBX */
 | |
| 
 | |
| 	movb	%dl, %dh	/* %dh = y */
 | |
| 	movb	%al, %dl	/* %dl = x */
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
|         xorb	%bh, %bh                /* set page to 0 */
 | |
| 	movb	$0x2, %ah
 | |
| 	int	$0x10			/* set cursor position */
 | |
| 
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * void grub_console_cls (void)
 | |
|  * BIOS call "INT 10H Function 09h" to write character and attribute
 | |
|  *	Call with	%ah = 0x09
 | |
|  *                      %al = (character)
 | |
|  *                      %bh = (page number)
 | |
|  *                      %bl = (attribute)
 | |
|  *                      %cx = (number of times)
 | |
|  */
 | |
| 
 | |
| FUNCTION(grub_console_cls)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx                    /* save EBX */
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	/* move the cursor to the beginning */
 | |
| 	movb	$0x02, %ah
 | |
| 	xorb	%bh, %bh
 | |
| 	xorw	%dx, %dx
 | |
| 	int	$0x10
 | |
| 
 | |
| 	/* write spaces to the entire screen */
 | |
| 	movw	$0x0920, %ax
 | |
| 	movw	$0x07, %bx
 | |
| 	movw	$(80 * 25), %cx
 | |
|         int	$0x10
 | |
| 
 | |
| 	/* move back the cursor */
 | |
| 	movb	$0x02, %ah
 | |
| 	int	$0x10
 | |
| 
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * void grub_console_setcursor (int on)
 | |
|  * BIOS call "INT 10H Function 01h" to set cursor type
 | |
|  *      Call with       %ah = 0x01
 | |
|  *                      %ch = cursor starting scanline
 | |
|  *                      %cl = cursor ending scanline
 | |
|  */
 | |
| 
 | |
| console_cursor_state:
 | |
| 	.byte	1
 | |
| console_cursor_shape:
 | |
| 	.word	0
 | |
| 
 | |
| FUNCTION(grub_console_setcursor)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 
 | |
| 	/* push ON */
 | |
| 	pushl	%eax
 | |
| 
 | |
| 	/* check if the standard cursor shape has already been saved */
 | |
| 	movw	console_cursor_shape, %ax
 | |
| 	testw	%ax, %ax
 | |
| 	jne	1f
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	movb	$0x03, %ah
 | |
| 	xorb	%bh, %bh
 | |
| 	int	$0x10
 | |
| 
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movw	%cx, console_cursor_shape
 | |
| 1:
 | |
| 	/* set %cx to the designated cursor shape */
 | |
| 	movw	$0x2000, %cx
 | |
| 	popl	%eax
 | |
| 	testl	%eax, %eax
 | |
| 	jz	2f
 | |
| 	movw	console_cursor_shape, %cx
 | |
| 2:
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	movb    $0x1, %ah
 | |
| 	int     $0x10
 | |
| 
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_getrtsecs()
 | |
|  *	if a seconds value can be read, read it and return it (BCD),
 | |
|  *      otherwise return 0xFF
 | |
|  * BIOS call "INT 1AH Function 02H" to check whether a character is pending
 | |
|  *	Call with	%ah = 0x2
 | |
|  *	Return:
 | |
|  *		If RT Clock can give correct values
 | |
|  *			%ch = hour (BCD)
 | |
|  *			%cl = minutes (BCD)
 | |
|  *                      %dh = seconds (BCD)
 | |
|  *                      %dl = daylight savings time (00h std, 01h daylight)
 | |
|  *			Carry flag = clear
 | |
|  *		else
 | |
|  *			Carry flag = set
 | |
|  *                         (this indicates that the clock is updating, or
 | |
|  *                          that it isn't running)
 | |
|  */
 | |
| FUNCTION(grub_getrtsecs)
 | |
| 	pushl	%ebp
 | |
| 
 | |
| 	call	prot_to_real	/* enter real mode */
 | |
| 	.code16
 | |
| 
 | |
| 	clc
 | |
| 	movb	$0x2, %ah
 | |
| 	int	$0x1a
 | |
| 
 | |
| 	DATA32	jnc	gottime
 | |
| 	movb	$0xff, %dh
 | |
| 
 | |
| gottime:
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movb	%dh, %al
 | |
| 
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * grub_get_rtc()
 | |
|  *	return the real time in ticks, of which there are about
 | |
|  *	18-20 per second
 | |
|  */
 | |
| FUNCTION(grub_get_rtc)
 | |
| 	pushl	%ebp
 | |
| 
 | |
| 	call	prot_to_real	/* enter real mode */
 | |
| 	.code16
 | |
| 
 | |
| 	/* %ax is already zero */
 | |
|         int	$0x1a
 | |
| 
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movl	%ecx, %eax
 | |
| 	shll	$16, %eax
 | |
| 	movw	%dx, %ax
 | |
| 
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * unsigned char grub_vga_set_mode (unsigned char mode)
 | |
|  */
 | |
| FUNCTION(grub_vga_set_mode)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 	movl	%eax, %ecx
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 	/* get current mode */
 | |
| 	xorw	%bx, %bx
 | |
| 	movb	$0x0f, %ah
 | |
| 	int	$0x10
 | |
| 	movb	%al, %dl
 | |
| 
 | |
| 	/* set the new mode */
 | |
| 	movb	%cl, %al
 | |
| 	xorb	%ah, %ah
 | |
| 	int	$0x10
 | |
| 
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movb	%dl, %al
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * unsigned char *grub_vga_get_font (void)
 | |
|  */
 | |
| FUNCTION(grub_vga_get_font)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 	movw	$0x1130, %ax
 | |
| 	movb	$0x06, %bh
 | |
| 	int	$0x10
 | |
| 	movw	%es, %bx
 | |
| 	movw	%bp, %dx
 | |
| 	DATA32	call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movzwl	%bx, %ecx
 | |
| 	shll	$4, %ecx
 | |
| 	movw	%dx, %ax
 | |
| 	addl	%ecx, %eax
 | |
| 
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_bios_status_t grub_vbe_get_controller_info (struct grub_vbe_info_block *controller_info)
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		*controller_info
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_get_controller_info)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%edi
 | |
| 	pushl	%edx
 | |
| 
 | |
| 	movw	%ax, %di	/* Store *controller_info to %edx:%di.  */
 | |
| 	xorw	%ax, %ax
 | |
| 	shrl	$4, %eax
 | |
| 	mov	%eax, %edx	/* prot_to_real destroys %eax.  */
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	pushw	%es
 | |
| 
 | |
| 	movw	%dx, %es	/* *controller_info is now on %es:%di.  */
 | |
| 	movw	$0x4f00, %ax
 | |
| 	int	$0x10
 | |
| 
 | |
| 	movw	%ax, %dx	/* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	popw	%es
 | |
| 
 | |
| 	DATA32 call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movl	%edx, %eax
 | |
| 	andl	$0x0FFFF, %eax	/* Return value in %eax.  */
 | |
| 
 | |
| 	pop	%edx
 | |
| 	popl	%edi
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_status_t grub_vbe_bios_get_mode_info (grub_uint32_t mode,
 | |
|  *						  struct grub_vbe_mode_info_block *mode_info)
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		mode
 | |
|  * %edx		*mode_info
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_get_mode_info)
 | |
| 	pushl   %ebp
 | |
| 	pushl   %edi
 | |
| 
 | |
| 	movl	%eax, %ecx	/* Store mode number to %ecx.  */
 | |
| 
 | |
| 	movw    %dx, %di	/* Store *mode_info to %edx:%di.  */
 | |
| 	xorw    %dx, %dx
 | |
| 	shrl    $4, %edx
 | |
| 
 | |
| 	call    prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	pushw   %es
 | |
| 
 | |
| 	movw    %dx, %es	/* *mode_info is now on %es:%di.  */
 | |
| 	movw    $0x4f01, %ax
 | |
| 	int     $0x10
 | |
| 
 | |
| 	movw    %ax, %dx        /* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	popw    %es
 | |
| 
 | |
| 	DATA32 call     real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movl    %edx, %eax
 | |
| 	andl    $0x0FFFF, %eax  /* Return value in %eax.  */
 | |
| 
 | |
| 	popl    %edi
 | |
| 	popl    %ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_status_t grub_vbe_bios_set_mode (grub_uint32_t mode,
 | |
|  *					     struct grub_vbe_crtc_info_block *crtc_info)
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		mode
 | |
|  * %edx		*crtc_info
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_set_mode)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 	pushl	%edi
 | |
| 
 | |
| 	movl	%eax, %ebx	/* Store mode in %ebx.  */
 | |
| 
 | |
| 	movw    %dx, %di	/* Store *crtc_info to %edx:%di.  */
 | |
| 	xorw    %dx, %dx
 | |
| 	shrl    $4, %edx
 | |
| 
 | |
| 	call    prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	pushw   %es
 | |
| 
 | |
| 	movw    %dx, %es	/* *crtc_info is now on %es:%di.  */
 | |
| 
 | |
| 	movw	$0x4f02, %ax
 | |
| 	int	$0x10
 | |
| 
 | |
| 	movw	%ax, %dx	/* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	popw	%es
 | |
| 
 | |
| 	DATA32 call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movw	%dx, %ax
 | |
| 	andl	$0xFFFF, %eax	/* Return value in %eax.  */
 | |
| 
 | |
| 	popl	%edi
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_status_t grub_vbe_bios_get_mode (grub_uint32_t *mode)
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		*mode
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_get_mode)
 | |
| 	pushl   %ebp
 | |
| 	pushl   %ebx
 | |
| 	pushl	%edi
 | |
| 	pushl	%edx
 | |
| 	pushl	%eax		/* Push *mode to stack.  */
 | |
| 
 | |
| 	call    prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	movw    $0x4f03, %ax
 | |
| 	int     $0x10
 | |
| 
 | |
| 	movw	%ax, %dx	/* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	DATA32 call     real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	popl	%edi		/* Pops *mode from stack to %edi.  */
 | |
| 	andl	$0xFFFF, %ebx
 | |
| 	movl	%ebx, (%edi)
 | |
| 
 | |
| 	movw	%dx, %ax
 | |
| 	andl	$0xFFFF, %eax	/* Return value in %eax.  */
 | |
| 
 | |
| 	popl	%edx
 | |
| 	popl	%edi
 | |
| 	popl    %ebx
 | |
| 	popl    %ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_status_t grub_vbe_bios_set_memory_window (grub_uint32_t window,
 | |
|  *						      grub_uint32_t position);
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		window
 | |
|  * %edx		position
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_set_memory_window)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 
 | |
| 	movl	%eax, %ebx
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	movw	$0x4f05, %ax
 | |
| 	andw	$0x00ff, %bx	/* BL = window, BH = 0, Set memory window.  */
 | |
| 	int	$0x10
 | |
| 
 | |
| 	movw	%ax, %dx	/* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	DATA32 call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movw	%dx, %ax
 | |
| 	andl	$0xFFFF, %eax	/* Return value in %eax.  */
 | |
| 
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_status_t grub_vbe_bios_get_memory_window (grub_uint32_t window,
 | |
|  * 						      grub_uint32_t *position);
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		window
 | |
|  * %edx		*position
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_get_memory_window)
 | |
| 	pushl   %ebp
 | |
| 	pushl   %ebx
 | |
| 	pushl	%edi
 | |
| 	pushl	%edx		/* Push *position to stack.  */
 | |
| 
 | |
| 	movl	%eax, %ebx	/* Store window in %ebx.  */
 | |
| 
 | |
| 	call    prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	movw    $0x4f05, %ax
 | |
| 	andw	$0x00ff, %bx	/* BL = window.  */
 | |
| 	orw	$0x0100, %bx	/* BH = 1, Get memory window.  */
 | |
| 	int     $0x10
 | |
| 
 | |
| 	movw	%ax, %bx	/* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	DATA32 call     real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	popl	%edi		/* pops *position from stack to %edi.  */
 | |
| 	andl	$0xFFFF, %edx
 | |
| 	movl	%edx, (%edi)	/* Return position to caller.  */
 | |
| 
 | |
| 	movw	%bx, %ax
 | |
| 	andl	$0xFFFF, %eax	/* Return value in %eax.  */
 | |
| 
 | |
| 	popl	%edi
 | |
| 	popl    %ebx
 | |
| 	popl    %ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_status_t grub_vbe_bios_set_scanline_length (grub_uint32_t length)
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		length
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_set_scanline_length)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 	pushl	%edx
 | |
| 
 | |
| 	movl	%eax, %ecx	/* Store length in %ecx.  */
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	movw	$0x4f06, %ax
 | |
| 	movw	$0x0002, %bx	/* BL = 2, Set Scan Line in Bytes.  */
 | |
| 	int	$0x10
 | |
| 
 | |
| 	movw	%ax, %dx	/* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	DATA32 call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movw	%dx, %ax
 | |
| 	andl	$0xFFFF, %eax	/* Return value in %eax.  */
 | |
| 
 | |
| 	popl	%edx
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_status_t grub_vbe_bios_get_scanline_length (grub_uint32_t *length)
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		*length
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_get_scanline_length)
 | |
| 	pushl   %ebp
 | |
| 	pushl   %ebx
 | |
| 	pushl	%edi
 | |
| 	pushl	%edx		/* Push *length to stack.  */
 | |
| 
 | |
| 	call    prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	movw    $0x4f06, %ax
 | |
| 	movw	$0x0001, %bx	/* BL = 1, Get Scan Line Length (in bytes).  */
 | |
| 	int     $0x10
 | |
| 
 | |
| 	movw	%ax, %dx	/* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	DATA32 call     real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	popl	%edi		/* Pops *length from stack to %edi.  */
 | |
| 	andl	$0xFFFF, %ebx
 | |
| 	movl	%ebx, (%edi)	/* Return length to caller.  */
 | |
| 
 | |
| 	movw	%dx, %ax
 | |
| 	andl	$0xFFFF, %eax	/* Return value in %eax.  */
 | |
| 
 | |
| 	popl	%edi
 | |
| 	popl    %ebx
 | |
| 	popl    %ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_status_t grub_vbe_bios_set_display_start (grub_uint32_t x,
 | |
|  *						      grub_uint32_t y)
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		x
 | |
|  * %edx		y
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_set_display_start)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 
 | |
| 	movl	%eax, %ecx	/* Store x in %ecx.  */
 | |
| 
 | |
| 	call	prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	movw	$0x4f07, %ax
 | |
| 	movw	$0x0080, %bx	/* BL = 80h, Set Display Start
 | |
| 				   during Vertical Retrace.  */
 | |
| 	int	$0x10
 | |
| 
 | |
| 	movw	%ax, %dx	/* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	DATA32 call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movw	%dx, %ax
 | |
| 	andl	$0xFFFF, %eax	/* Return value in %eax.  */
 | |
| 
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_status_t grub_vbe_bios_get_display_start (grub_uint32_t *x,
 | |
|  *						      grub_uint32_t *y)
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		*x
 | |
|  * %edx		*y
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_get_display_start)
 | |
| 	pushl   %ebp
 | |
| 	pushl   %ebx
 | |
| 	pushl	%edi
 | |
| 	pushl	%eax		/* Push *x to stack.  */
 | |
| 	pushl	%edx		/* Push *y to stack.  */
 | |
| 
 | |
| 	call    prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	movw    $0x4f07, %ax
 | |
| 	movw	$0x0001, %bx	/* BL = 1, Get Display Start.  */
 | |
| 	int     $0x10
 | |
| 
 | |
| 	movw	%ax, %bx	/* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	DATA32 call     real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	popl	%edi		/* Pops *y from stack to %edi.  */
 | |
| 	andl	$0xFFFF, %edx
 | |
| 	movl	%edx, (%edi)	/* Return y-position to caller.  */
 | |
| 
 | |
| 	popl	%edi		/* Pops *x from stack to %edi.  */
 | |
| 	andl	$0xFFFF, %ecx
 | |
| 	movl	%ecx, (%edi)	/* Return x-position to caller.  */
 | |
| 
 | |
| 	movw	%bx, %ax
 | |
| 	andl	$0xFFFF, %eax	/* Return value in %eax.  */
 | |
| 
 | |
| 	popl	%edi
 | |
| 	popl    %ebx
 | |
| 	popl    %ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * grub_vbe_status_t grub_vbe_bios_set_palette_data (grub_uint32_t color_count,
 | |
|  *						     grub_uint32_t start_index,
 | |
|  *						     struct grub_vbe_palette_data *palette_data)
 | |
|  *
 | |
|  * Register allocations for parameters:
 | |
|  * %eax		color_count
 | |
|  * %edx		start_index
 | |
|  * %ecx		*palette_data
 | |
|  */
 | |
| FUNCTION(grub_vbe_bios_set_palette_data)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 	pushl	%edi
 | |
| 
 | |
| 	movl	%eax, %ebx	/* Store color_count in %ebx.  */
 | |
| 
 | |
| 	movw    %cx, %di	/* Store *palette_data to %ecx:%di.  */
 | |
| 	xorw    %cx, %cx
 | |
| 	shrl    $4, %ecx
 | |
| 
 | |
| 	call    prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	pushw   %es
 | |
| 
 | |
| 	movw    %cx, %es	/* *palette_data is now on %es:%di.  */
 | |
| 	movw	%bx, %cx	/* color_count is now on %cx.  */
 | |
| 
 | |
| 	movw	$0x4f09, %ax
 | |
| 	xorw	%bx, %bx	/* BL = 0, Set Palette Data.  */
 | |
| 	int	$0x10
 | |
| 
 | |
| 	movw	%ax, %dx	/* real_to_prot destroys %eax.  */
 | |
| 
 | |
| 	popw	%es
 | |
| 
 | |
| 	DATA32 call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movw	%dx, %ax
 | |
| 	andl	$0xFFFF, %eax	/* Return value in %eax.  */
 | |
| 
 | |
| 	popl	%edi
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| 
 | |
| pxe_rm_entry:
 | |
| 	.long	0
 | |
| 
 | |
| /*
 | |
|  * struct grub_pxenv *grub_pxe_scan (void);
 | |
|  */
 | |
| FUNCTION(grub_pxe_scan)
 | |
| 	pushl	%ebp
 | |
| 	pushl	%ebx
 | |
| 
 | |
| 	xorl	%ebx, %ebx
 | |
| 	xorl	%ecx, %ecx
 | |
| 
 | |
| 	call    prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	pushw	%es
 | |
| 
 | |
| 	movw	$0x5650, %ax
 | |
| 	int	$0x1A
 | |
| 	cmpw	$0x564E, %ax
 | |
| 	jnz	1f
 | |
| 	cmpl	$0x4E455850, %es:(%bx)		/* PXEN(V+)  */
 | |
| 	jnz	1f
 | |
| 	cmpw	$0x201, %es:6(%bx)		/* API version  */
 | |
| 	jb	1f
 | |
| 	lesw	%es:0x28(%bx), %bx		/* !PXE structure  */
 | |
| 	cmpl	$0x45585021, %es:(%bx)		/* !PXE  */
 | |
| 	jnz	1f
 | |
| 	movw	%es, %cx
 | |
| 	jmp	2f
 | |
| 1:
 | |
| 	xorw	%bx, %bx
 | |
| 	xorw	%cx, %cx
 | |
| 2:
 | |
| 
 | |
| 	popw	%es
 | |
| 
 | |
| 	DATA32 call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	xorl	%eax, %eax
 | |
| 	leal	(%eax, %ecx, 4), %ecx
 | |
| 	leal	(%ebx, %ecx, 4), %eax		/* eax = ecx * 16 + ebx  */
 | |
| 
 | |
| 	orl	%eax, %eax
 | |
| 	jz	1f
 | |
| 
 | |
| 	movl	0x10(%eax), %ecx
 | |
| 	movl	%ecx, pxe_rm_entry
 | |
| 
 | |
| 1:
 | |
| 
 | |
| 	popl	%ebx
 | |
| 	popl	%ebp
 | |
| 	ret
 | |
| 
 | |
| /*
 | |
|  * int grub_pxe_call (int func, void* data);
 | |
|  */
 | |
| FUNCTION(grub_pxe_call)
 | |
| 	pushl	%ebp
 | |
| 	movl	%esp, %ebp
 | |
| 	pushl	%esi
 | |
| 	pushl	%edi
 | |
| 	pushl	%ebx
 | |
| 
 | |
| 	movl	%eax, %ecx
 | |
| 	movl	%edx, %eax
 | |
| 	andl	$0xF, %eax
 | |
| 	shrl	$4, %edx
 | |
| 	shll	$16, %edx
 | |
| 	addl	%eax, %edx
 | |
| 	movl	pxe_rm_entry, %ebx
 | |
| 
 | |
| 	call    prot_to_real
 | |
| 	.code16
 | |
| 
 | |
| 	pushl	%ebx
 | |
| 	pushl	%edx
 | |
| 	pushw	%cx
 | |
| 	movw	%sp, %bx
 | |
| 	lcall	*%ss:6(%bx)
 | |
| 	cld
 | |
| 	addw	$10, %sp
 | |
| 	movw	%ax, %cx
 | |
| 
 | |
| 	DATA32  call	real_to_prot
 | |
| 	.code32
 | |
| 
 | |
| 	movzwl	%cx, %eax
 | |
| 
 | |
| 	popl	%ebx
 | |
| 	popl	%edi
 | |
| 	popl	%esi
 | |
| 	popl	%ebp
 | |
| 	ret
 |