head_32.S 4.2 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_types.h>
29
#include <asm/boot.h>
R
Rusty Russell 已提交
30
#include <asm/asm-offsets.h>
L
Linus Torvalds 已提交
31

32
.section ".text.head","ax",@progbits
33
ENTRY(startup_32)
34
	cld
R
Rusty Russell 已提交
35 36 37
	/* test KEEP_SEGMENTS flag to see if the bootloader is asking
	 * us to not reload segments */
	testb $(1<<6), BP_loadflags(%esi)
38
	jnz 1f
R
Rusty Russell 已提交
39

40
	cli
L
Linus Torvalds 已提交
41 42 43 44 45
	movl $(__BOOT_DS),%eax
	movl %eax,%ds
	movl %eax,%es
	movl %eax,%fs
	movl %eax,%gs
46
	movl %eax,%ss
47
1:
R
Rusty Russell 已提交
48

49 50 51 52
/* 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
53 54
 * data at 0x1e4 (defined as a scratch field) are used as the stack
 * for this calculation. Only 4 bytes are needed.
55
 */
56
	leal (BP_scratch+4)(%esi), %esp
57 58 59 60
	call 1f
1:	popl %ebp
	subl $1b, %ebp

61 62 63
/* %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.
64
 */
65

66
#ifdef CONFIG_RELOCATABLE
67 68 69
	movl 	%ebp, %ebx
	addl    $(CONFIG_PHYSICAL_ALIGN - 1), %ebx
	andl    $(~(CONFIG_PHYSICAL_ALIGN - 1)), %ebx
70
#else
71
	movl $LOAD_PHYSICAL_ADDR, %ebx
72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90
#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
91 92 93
	leal _ebss(%ebp), %esi
	leal _ebss(%ebx), %edi
	movl $(_ebss - startup_32), %ecx
94 95 96 97 98 99 100 101 102
	std
	rep
	movsb
	cld
	popl %esi

/* Compute the kernel start address.
 */
#ifdef CONFIG_RELOCATABLE
103 104
	addl    $(CONFIG_PHYSICAL_ALIGN - 1), %ebp
	andl    $(~(CONFIG_PHYSICAL_ALIGN - 1)), %ebp
105
#else
106
	movl	$LOAD_PHYSICAL_ADDR, %ebp
107
#endif
L
Linus Torvalds 已提交
108 109

/*
110
 * Jump to the relocated address.
L
Linus Torvalds 已提交
111
 */
112 113
	leal relocated(%ebx), %eax
	jmp *%eax
114 115
ENDPROC(startup_32)

116 117 118
.section ".text"
relocated:

L
Linus Torvalds 已提交
119 120 121 122
/*
 * Clear BSS
 */
	xorl %eax,%eax
123
	leal _edata(%ebx),%edi
124
	leal _ebss(%ebx), %ecx
L
Linus Torvalds 已提交
125 126 127 128
	subl %edi,%ecx
	cld
	rep
	stosb
129 130 131 132

/*
 * Setup the stack for the decompressor
 */
133
	leal boot_stack_end(%ebx), %esp
134

L
Linus Torvalds 已提交
135 136 137
/*
 * Do the decompression, and jump to the new kernel..
 */
138 139
	movl output_len(%ebx), %eax
	pushl %eax
140
			# push arguments for decompress_kernel:
141 142 143 144 145
	pushl %ebp	# output address
	movl input_len(%ebx), %eax
	pushl %eax	# input_len
	leal input_data(%ebx), %eax
	pushl %eax	# input_data
146
	leal boot_heap(%ebx), %eax
147 148
	pushl %eax	# heap area
	pushl %esi	# real mode pointer
L
Linus Torvalds 已提交
149
	call decompress_kernel
150 151 152 153 154 155 156 157 158 159 160 161 162
	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
163 164
	subl $LOAD_PHYSICAL_ADDR, %ebx
	jz   2f		/* Nothing to be done if loaded at compiled addr. */
L
Linus Torvalds 已提交
165
/*
166
 * Process relocations.
L
Linus Torvalds 已提交
167
 */
168 169 170 171 172 173 174 175 176

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 已提交
177 178

/*
179
 * Jump to the decompressed kernel.
L
Linus Torvalds 已提交
180 181
 */
	xorl %ebx,%ebx
182 183 184
	jmp *%ebp

.bss
185
/* Stack and heap for uncompression */
186
.balign 4
187 188 189 190 191
boot_heap:
	.fill BOOT_HEAP_SIZE, 1, 0
boot_stack:
	.fill BOOT_STACK_SIZE, 1, 0
boot_stack_end: