mirror of
https://github.com/torvalds/linux.git
synced 2024-11-15 00:21:59 +00:00
d01447b319
This implements a bit of rework for the PMB code, which permits us to kill off the legacy PMB mode completely. Rather than trusting the boot loader to do the right thing, we do a quick verification of the PMB contents to determine whether to have the kernel setup the initial mappings or whether it needs to mangle them later on instead. If we're booting from legacy mappings, the kernel will now take control of them and make them match the kernel's initial mapping configuration. This is accomplished by breaking the initialization phase out in to multiple steps: synchronization, merging, and resizing. With the recent rework, the synchronization code establishes page links for compound mappings already, so we build on top of this for promoting mappings and reclaiming unused slots. At the same time, the changes introduced for the uncached helpers also permit us to dynamically resize the uncached mapping without any particular headaches. The smallest page size is more than sufficient for mapping all of kernel text, and as we're careful not to jump to any far off locations in the setup code the mapping can safely be resized regardless of whether we are executing from it or not. Signed-off-by: Paul Mundt <lethal@linux-sh.org>
135 lines
2.6 KiB
C
135 lines
2.6 KiB
C
/*
|
|
* arch/sh/boot/compressed/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
|
|
*
|
|
* Adapted for SH by Stuart Menefy, Aug 1999
|
|
*
|
|
* Modified to use standard LinuxSH BIOS by Greg Banks 7Jul2000
|
|
*/
|
|
|
|
#include <asm/uaccess.h>
|
|
#include <asm/addrspace.h>
|
|
#include <asm/page.h>
|
|
|
|
/*
|
|
* gzip declarations
|
|
*/
|
|
|
|
#define STATIC static
|
|
|
|
#undef memset
|
|
#undef memcpy
|
|
#define memzero(s, n) memset ((s), 0, (n))
|
|
|
|
/* cache.c */
|
|
#define CACHE_ENABLE 0
|
|
#define CACHE_DISABLE 1
|
|
int cache_control(unsigned int command);
|
|
|
|
extern char input_data[];
|
|
extern int input_len;
|
|
static unsigned char *output;
|
|
|
|
static void error(char *m);
|
|
|
|
int puts(const char *);
|
|
|
|
extern int _text; /* Defined in vmlinux.lds.S */
|
|
extern int _end;
|
|
static unsigned long free_mem_ptr;
|
|
static unsigned long free_mem_end_ptr;
|
|
|
|
#ifdef CONFIG_HAVE_KERNEL_BZIP2
|
|
#define HEAP_SIZE 0x400000
|
|
#else
|
|
#define HEAP_SIZE 0x10000
|
|
#endif
|
|
|
|
#ifdef CONFIG_KERNEL_GZIP
|
|
#include "../../../../lib/decompress_inflate.c"
|
|
#endif
|
|
|
|
#ifdef CONFIG_KERNEL_BZIP2
|
|
#include "../../../../lib/decompress_bunzip2.c"
|
|
#endif
|
|
|
|
#ifdef CONFIG_KERNEL_LZMA
|
|
#include "../../../../lib/decompress_unlzma.c"
|
|
#endif
|
|
|
|
#ifdef CONFIG_KERNEL_LZO
|
|
#include "../../../../lib/decompress_unlzo.c"
|
|
#endif
|
|
|
|
int puts(const char *s)
|
|
{
|
|
/* This should be updated to use the sh-sci routines */
|
|
return 0;
|
|
}
|
|
|
|
void* memset(void* s, int c, size_t n)
|
|
{
|
|
int i;
|
|
char *ss = (char*)s;
|
|
|
|
for (i=0;i<n;i++) ss[i] = c;
|
|
return s;
|
|
}
|
|
|
|
void* memcpy(void* __dest, __const void* __src,
|
|
size_t __n)
|
|
{
|
|
int i;
|
|
char *d = (char *)__dest, *s = (char *)__src;
|
|
|
|
for (i=0;i<__n;i++) d[i] = s[i];
|
|
return __dest;
|
|
}
|
|
|
|
static void error(char *x)
|
|
{
|
|
puts("\n\n");
|
|
puts(x);
|
|
puts("\n\n -- System halted");
|
|
|
|
while(1); /* Halt */
|
|
}
|
|
|
|
#ifdef CONFIG_SUPERH64
|
|
#define stackalign 8
|
|
#else
|
|
#define stackalign 4
|
|
#endif
|
|
|
|
#define STACK_SIZE (4096)
|
|
long __attribute__ ((aligned(stackalign))) user_stack[STACK_SIZE];
|
|
long *stack_start = &user_stack[STACK_SIZE];
|
|
|
|
void decompress_kernel(void)
|
|
{
|
|
unsigned long output_addr;
|
|
|
|
#ifdef CONFIG_SUPERH64
|
|
output_addr = (CONFIG_MEMORY_START + 0x2000);
|
|
#else
|
|
output_addr = __pa((unsigned long)&_text+PAGE_SIZE);
|
|
#if defined(CONFIG_29BIT)
|
|
output_addr |= P2SEG;
|
|
#endif
|
|
#endif
|
|
|
|
output = (unsigned char *)output_addr;
|
|
free_mem_ptr = (unsigned long)&_end;
|
|
free_mem_end_ptr = free_mem_ptr + HEAP_SIZE;
|
|
|
|
puts("Uncompressing Linux... ");
|
|
cache_control(CACHE_ENABLE);
|
|
decompress(input_data, input_len, NULL, NULL, output, NULL, error);
|
|
cache_control(CACHE_DISABLE);
|
|
puts("Ok, booting the kernel.\n");
|
|
}
|