4  * This is a collection of several routines from gzip-1.0.3 
 
   7  * malloc by Hannu Savolainen 1993 and Matthias Urlichs 1994
 
   9  * Modified for ARM Linux by Russell King
 
  11  * Nicolas Pitre <nico@visuaide.com>  1999/04/14 :
 
  12  *  For this code to run directly from Flash, all constant variables must
 
  13  *  be marked with 'const' and all other variables initialized at run-time 
 
  14  *  only.  This way all non constant variables will end up in the bss segment,
 
  15  *  which should point to addresses in RAM and cleared to 0 on start.
 
  16  *  This allows for a much quicker boot time.
 
  19 unsigned int __machine_arch_type;
 
  21 #include <linux/string.h>
 
  23 #ifdef STANDALONE_DEBUG
 
  27 static void putstr(const char *ptr);
 
  29 #include <linux/compiler.h>
 
  30 #include <mach/uncompress.h>
 
  32 #ifdef CONFIG_DEBUG_ICEDCC
 
  36 static void icedcc_putc(int ch)
 
  38         int status, i = 0x4000000;
 
  44                 asm volatile ("mrc p14, 0, %0, c0, c1, 0" : "=r" (status));
 
  45         } while (status & (1 << 29));
 
  47         asm("mcr p14, 0, %0, c0, c5, 0" : : "r" (ch));
 
  52 static void icedcc_putc(int ch)
 
  54         int status, i = 0x4000000;
 
  60                 asm volatile ("mrc p14, 0, %0, c0, c0, 0" : "=r" (status));
 
  63         asm("mcr p14, 0, %0, c1, c0, 0" : : "r" (ch));
 
  68 #define putc(ch)        icedcc_putc(ch)
 
  69 #define flush() do { } while (0)
 
  72 static void putstr(const char *ptr)
 
  76         while ((c = *ptr++) != '\0') {
 
  87 #define __ptr_t void *
 
  90  * Optimised C version of memzero for the ARM.
 
  92 void __memzero (__ptr_t s, size_t n)
 
  94         union { void *vp; unsigned long *ulp; unsigned char *ucp; } u;
 
  99         for (i = n >> 5; i > 0; i--) {
 
 134 static inline __ptr_t memcpy(__ptr_t __dest, __const __ptr_t __src,
 
 138         unsigned char *d = (unsigned char *)__dest, *s = (unsigned char *)__src;
 
 140         for (i = __n >> 3; i > 0; i--) {
 
 172 #define OF(args)  args
 
 173 #define STATIC static
 
 175 typedef unsigned char  uch;
 
 176 typedef unsigned short ush;
 
 177 typedef unsigned long  ulg;
 
 179 #define WSIZE 0x8000            /* Window size must be at least 32k, */
 
 180                                 /* and a power of two */
 
 182 static uch *inbuf;              /* input buffer */
 
 183 static uch window[WSIZE];       /* Sliding window buffer */
 
 185 static unsigned insize;         /* valid bytes in inbuf */
 
 186 static unsigned inptr;          /* index of next byte to be processed in inbuf */
 
 187 static unsigned outcnt;         /* bytes in output buffer */
 
 190 #define ASCII_FLAG   0x01 /* bit 0 set: file probably ascii text */
 
 191 #define CONTINUATION 0x02 /* bit 1 set: continuation of multi-part gzip file */
 
 192 #define EXTRA_FIELD  0x04 /* bit 2 set: extra field present */
 
 193 #define ORIG_NAME    0x08 /* bit 3 set: original file name present */
 
 194 #define COMMENT      0x10 /* bit 4 set: file comment present */
 
 195 #define ENCRYPTED    0x20 /* bit 5 set: file is encrypted */
 
 196 #define RESERVED     0xC0 /* bit 6,7:   reserved */
 
 198 #define get_byte()  (inptr < insize ? inbuf[inptr++] : fill_inbuf())
 
 200 /* Diagnostic functions */
 
 202 #  define Assert(cond,msg) {if(!(cond)) error(msg);}
 
 203 #  define Trace(x) fprintf x
 
 204 #  define Tracev(x) {if (verbose) fprintf x ;}
 
 205 #  define Tracevv(x) {if (verbose>1) fprintf x ;}
 
 206 #  define Tracec(c,x) {if (verbose && (c)) fprintf x ;}
 
 207 #  define Tracecv(c,x) {if (verbose>1 && (c)) fprintf x ;}
 
 209 #  define Assert(cond,msg)
 
 214 #  define Tracecv(c,x)
 
 217 static int  fill_inbuf(void);
 
 218 static void flush_window(void);
 
 219 static void error(char *m);
 
 221 extern char input_data[];
 
 222 extern char input_data_end[];
 
 224 static uch *output_data;
 
 225 static ulg output_ptr;
 
 226 static ulg bytes_out;
 
 228 static void error(char *m);
 
 230 static void putstr(const char *);
 
 233 static ulg free_mem_ptr;
 
 234 static ulg free_mem_end_ptr;
 
 236 #ifdef STANDALONE_DEBUG
 
 237 #define NO_INFLATE_MALLOC
 
 240 #define ARCH_HAS_DECOMP_WDOG
 
 242 #include "../../../../lib/inflate.c"
 
 244 /* ===========================================================================
 
 245  * Fill the input buffer. This is called only when the buffer is empty
 
 246  * and at least one byte is really needed.
 
 251                 error("ran out of input data");
 
 254         insize = &input_data_end[0] - &input_data[0];
 
 260 /* ===========================================================================
 
 261  * Write the output window window[0..outcnt-1] and update crc and bytes_out.
 
 262  * (Used for the decompressed data only.)
 
 264 void flush_window(void)
 
 271         out = &output_data[output_ptr];
 
 272         for (n = 0; n < outcnt; n++) {
 
 274                 c = crc_32_tab[((int)c ^ ch) & 0xff] ^ (c >> 8);
 
 277         bytes_out += (ulg)outcnt;
 
 278         output_ptr += (ulg)outcnt;
 
 284 #define arch_error(x)
 
 287 static void error(char *x)
 
 293         putstr("\n\n -- System halted");
 
 298 #ifndef STANDALONE_DEBUG
 
 301 decompress_kernel(ulg output_start, ulg free_mem_ptr_p, ulg free_mem_ptr_end_p,
 
 304         output_data             = (uch *)output_start;  /* Points to kernel start */
 
 305         free_mem_ptr            = free_mem_ptr_p;
 
 306         free_mem_end_ptr        = free_mem_ptr_end_p;
 
 307         __machine_arch_type     = arch_id;
 
 312         putstr("Uncompressing Linux...");
 
 314         putstr(" done, booting the kernel.\n");
 
 319 char output_buffer[1500*1024];
 
 323         output_data = output_buffer;
 
 326         putstr("Uncompressing Linux...");