head_32.S 3.9 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
/*
 *  linux/boot/head.S
 *
 *  Copyright (C) 1991, 1992, 1993  Linus Torvalds
 */

/*
 *  head.S contains the 32-bit startup code.
 *
 * NOTE!!! Startup happens at absolute address 0x00001000, which is also where
 * the page directory will exist. The startup code will be overwritten by
 * the page directory. [According to comments etc elsewhere on a compressed
 * kernel it will end up at 0x1000 + 1Mb I hope so as I assume this. - AC]
 *
 * Page 0 is deliberately kept safe, since System Management Mode code in 
 * laptops may need to access the BIOS data stored there.  This is also
 * useful for future device drivers that either access the BIOS via VM86 
 * mode.
 */

/*
 * High loaded stuff by Hans Lermen & Werner Almesberger, Feb. 1996
 */
.text

#include <linux/linkage.h>
#include <asm/segment.h>
28
#include <asm/page.h>
29
#include <asm/boot.h>
L
Linus Torvalds 已提交
30

31
.section ".text.head","ax",@progbits
L
Linus Torvalds 已提交
32
	.globl startup_32
33

L
Linus Torvalds 已提交
34 35 36 37 38 39 40 41
startup_32:
	cld
	cli
	movl $(__BOOT_DS),%eax
	movl %eax,%ds
	movl %eax,%es
	movl %eax,%fs
	movl %eax,%gs
42
	movl %eax,%ss
L
Linus Torvalds 已提交
43

44 45 46 47
/* Calculate the delta between where we were compiled to run
 * at and where we were actually loaded at.  This can only be done
 * with a short local call on x86.  Nothing  else will tell us what
 * address we are running at.  The reserved chunk of the real-mode
48 49
 * data at 0x1e4 (defined as a scratch field) are used as the stack
 * for this calculation. Only 4 bytes are needed.
50
 */
51
	leal (0x1e4+4)(%esi), %esp
52 53 54 55
	call 1f
1:	popl %ebp
	subl $1b, %ebp

56 57 58
/* %ebp contains the address we are loaded at by the boot loader and %ebx
 * contains the address where we should move the kernel image temporarily
 * for safe in-place decompression.
59
 */
60

61
#ifdef CONFIG_RELOCATABLE
62 63 64
	movl 	%ebp, %ebx
	addl    $(CONFIG_PHYSICAL_ALIGN - 1), %ebx
	andl    $(~(CONFIG_PHYSICAL_ALIGN - 1)), %ebx
65
#else
66
	movl $LOAD_PHYSICAL_ADDR, %ebx
67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
#endif

	/* Replace the compressed data size with the uncompressed size */
	subl input_len(%ebp), %ebx
	movl output_len(%ebp), %eax
	addl %eax, %ebx
	/* Add 8 bytes for every 32K input block */
	shrl $12, %eax
	addl %eax, %ebx
	/* Add 32K + 18 bytes of extra slack */
	addl $(32768 + 18), %ebx
	/* Align on a 4K boundary */
	addl $4095, %ebx
	andl $~4095, %ebx

/* Copy the compressed kernel to the end of our buffer
 * where decompression in place becomes safe.
 */
	pushl %esi
	leal _end(%ebp), %esi
	leal _end(%ebx), %edi
	movl $(_end - startup_32), %ecx
	std
	rep
	movsb
	cld
	popl %esi

/* Compute the kernel start address.
 */
#ifdef CONFIG_RELOCATABLE
98 99
	addl    $(CONFIG_PHYSICAL_ALIGN - 1), %ebp
	andl    $(~(CONFIG_PHYSICAL_ALIGN - 1)), %ebp
100
#else
101
	movl	$LOAD_PHYSICAL_ADDR, %ebp
102
#endif
L
Linus Torvalds 已提交
103 104

/*
105
 * Jump to the relocated address.
L
Linus Torvalds 已提交
106
 */
107 108 109 110 111
	leal relocated(%ebx), %eax
	jmp *%eax
.section ".text"
relocated:

L
Linus Torvalds 已提交
112 113 114 115
/*
 * Clear BSS
 */
	xorl %eax,%eax
116 117
	leal _edata(%ebx),%edi
	leal _end(%ebx), %ecx
L
Linus Torvalds 已提交
118 119 120 121
	subl %edi,%ecx
	cld
	rep
	stosb
122 123 124 125 126 127

/*
 * Setup the stack for the decompressor
 */
	leal stack_end(%ebx), %esp

L
Linus Torvalds 已提交
128 129 130
/*
 * Do the decompression, and jump to the new kernel..
 */
131 132 133 134 135 136 137 138 139
	movl output_len(%ebx), %eax
	pushl %eax
	pushl %ebp	# output address
	movl input_len(%ebx), %eax
	pushl %eax	# input_len
	leal input_data(%ebx), %eax
	pushl %eax	# input_data
	leal _end(%ebx), %eax
	pushl %eax	# end of the image as third argument
L
Linus Torvalds 已提交
140 141
	pushl %esi	# real mode pointer as second arg
	call decompress_kernel
142 143 144 145 146 147 148 149 150 151 152 153 154
	addl $20, %esp
	popl %ecx

#if CONFIG_RELOCATABLE
/* Find the address of the relocations.
 */
	movl %ebp, %edi
	addl %ecx, %edi

/* Calculate the delta between where vmlinux was compiled to run
 * and where it was actually loaded.
 */
	movl %ebp, %ebx
155 156
	subl $LOAD_PHYSICAL_ADDR, %ebx
	jz   2f		/* Nothing to be done if loaded at compiled addr. */
L
Linus Torvalds 已提交
157
/*
158
 * Process relocations.
L
Linus Torvalds 已提交
159
 */
160 161 162 163 164 165 166 167 168

1:	subl $4, %edi
	movl 0(%edi), %ecx
	testl %ecx, %ecx
	jz 2f
	addl %ebx, -__PAGE_OFFSET(%ebx, %ecx)
	jmp 1b
2:
#endif
L
Linus Torvalds 已提交
169 170

/*
171
 * Jump to the decompressed kernel.
L
Linus Torvalds 已提交
172 173
 */
	xorl %ebx,%ebx
174 175 176 177 178 179 180
	jmp *%ebp

.bss
.balign 4
stack:
	.fill 4096, 1, 0
stack_end: