head_32.S 4.1 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 (0x1e4+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 91 92 93 94 95 96 97 98 99 100 101 102
#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
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 124
	leal _edata(%ebx),%edi
	leal _end(%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: