misc.c 3.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
/*
 * misc.c
 * 
 * This is a collection of several routines from gzip-1.0.3 
 * adapted for Linux.
 *
 * malloc by Hannu Savolainen 1993 and Matthias Urlichs 1994
 *
 * Modified for ARM Linux by Russell King
 *
 * Nicolas Pitre <nico@visuaide.com>  1999/04/14 :
 *  For this code to run directly from Flash, all constant variables must
 *  be marked with 'const' and all other variables initialized at run-time 
 *  only.  This way all non constant variables will end up in the bss segment,
 *  which should point to addresses in RAM and cleared to 0 on start.
 *  This allows for a much quicker boot time.
 */

unsigned int __machine_arch_type;

21
#include <linux/compiler.h>	/* for inline */
22
#include <linux/types.h>
23
#include <linux/linkage.h>
L
Linus Torvalds 已提交
24

25
static void putstr(const char *ptr);
26
extern void error(char *x);
27

28
#include CONFIG_UNCOMPRESS_INCLUDE
L
Linus Torvalds 已提交
29

30
#ifdef CONFIG_DEBUG_ICEDCC
31

32
#if defined(CONFIG_CPU_V6) || defined(CONFIG_CPU_V6K) || defined(CONFIG_CPU_V7)
33 34 35 36 37 38 39 40 41 42 43 44 45 46

static void icedcc_putc(int ch)
{
	int status, i = 0x4000000;

	do {
		if (--i < 0)
			return;

		asm volatile ("mrc p14, 0, %0, c0, c1, 0" : "=r" (status));
	} while (status & (1 << 29));

	asm("mcr p14, 0, %0, c0, c5, 0" : : "r" (ch));
}
47 48


49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
#elif defined(CONFIG_CPU_XSCALE)

static void icedcc_putc(int ch)
{
	int status, i = 0x4000000;

	do {
		if (--i < 0)
			return;

		asm volatile ("mrc p14, 0, %0, c14, c0, 0" : "=r" (status));
	} while (status & (1 << 28));

	asm("mcr p14, 0, %0, c8, c0, 0" : : "r" (ch));
}
64 65 66

#else

67 68 69 70 71 72 73 74
static void icedcc_putc(int ch)
{
	int status, i = 0x4000000;

	do {
		if (--i < 0)
			return;

75
		asm volatile ("mrc p14, 0, %0, c0, c0, 0" : "=r" (status));
76 77
	} while (status & 2);

78
	asm("mcr p14, 0, %0, c1, c0, 0" : : "r" (ch));
79 80
}

81 82
#endif

83 84
#define putc(ch)	icedcc_putc(ch)
#endif
L
Linus Torvalds 已提交
85

86
static void putstr(const char *ptr)
L
Linus Torvalds 已提交
87
{
88 89 90 91 92 93
	char c;

	while ((c = *ptr++) != '\0') {
		if (c == '\n')
			putc('\r');
		putc(c);
L
Linus Torvalds 已提交
94
	}
95 96

	flush();
L
Linus Torvalds 已提交
97 98 99
}

/*
100
 * gzip declarations
L
Linus Torvalds 已提交
101 102 103 104
 */
extern char input_data[];
extern char input_data_end[];

105
unsigned char *output_data;
L
Linus Torvalds 已提交
106

107 108
unsigned long free_mem_ptr;
unsigned long free_mem_end_ptr;
L
Linus Torvalds 已提交
109

110 111 112 113
#ifndef arch_error
#define arch_error(x)
#endif

114
void error(char *x)
L
Linus Torvalds 已提交
115
{
116 117
	arch_error(x);

L
Linus Torvalds 已提交
118 119 120 121 122 123 124
	putstr("\n\n");
	putstr(x);
	putstr("\n\n -- System halted");

	while(1);	/* Halt */
}

125 126 127 128 129
asmlinkage void __div0(void)
{
	error("Attempting division by 0!");
}

130 131 132 133 134 135 136 137 138 139 140 141
unsigned long __stack_chk_guard;

void __stack_chk_guard_setup(void)
{
	__stack_chk_guard = 0x000a0dff;
}

void __stack_chk_fail(void)
{
	error("stack-protector: Kernel stack is corrupted\n");
}

142
extern int do_decompress(u8 *input, int len, u8 *output, void (*error)(char *x));
143

L
Linus Torvalds 已提交
144

145
void
146 147 148
decompress_kernel(unsigned long output_start, unsigned long free_mem_ptr_p,
		unsigned long free_mem_ptr_end_p,
		int arch_id)
L
Linus Torvalds 已提交
149
{
150
	int ret;
151

152 153
	__stack_chk_guard_setup();

154
	output_data		= (unsigned char *)output_start;
L
Linus Torvalds 已提交
155
	free_mem_ptr		= free_mem_ptr_p;
156
	free_mem_end_ptr	= free_mem_ptr_end_p;
L
Linus Torvalds 已提交
157 158 159 160 161
	__machine_arch_type	= arch_id;

	arch_decomp_setup();

	putstr("Uncompressing Linux...");
162 163 164 165 166 167
	ret = do_decompress(input_data, input_data_end - input_data,
			    output_data, error);
	if (ret)
		error("decompressor returned an error");
	else
		putstr(" done, booting the kernel.\n");
L
Linus Torvalds 已提交
168
}