diff options
Diffstat (limited to 'init/do_mounts_rd.c')
| -rw-r--r-- | init/do_mounts_rd.c | 178 |
1 files changed, 47 insertions, 131 deletions
diff --git a/init/do_mounts_rd.c b/init/do_mounts_rd.c index 0f0f0cf3ba9..027a402708d 100644 --- a/init/do_mounts_rd.c +++ b/init/do_mounts_rd.c | |||
| @@ -11,6 +11,9 @@ | |||
| 11 | #include "do_mounts.h" | 11 | #include "do_mounts.h" |
| 12 | #include "../fs/squashfs/squashfs_fs.h" | 12 | #include "../fs/squashfs/squashfs_fs.h" |
| 13 | 13 | ||
| 14 | #include <linux/decompress/generic.h> | ||
| 15 | |||
| 16 | |||
| 14 | int __initdata rd_prompt = 1;/* 1 = prompt for RAM disk, 0 = don't prompt */ | 17 | int __initdata rd_prompt = 1;/* 1 = prompt for RAM disk, 0 = don't prompt */ |
| 15 | 18 | ||
| 16 | static int __init prompt_ramdisk(char *str) | 19 | static int __init prompt_ramdisk(char *str) |
| @@ -29,7 +32,7 @@ static int __init ramdisk_start_setup(char *str) | |||
| 29 | } | 32 | } |
| 30 | __setup("ramdisk_start=", ramdisk_start_setup); | 33 | __setup("ramdisk_start=", ramdisk_start_setup); |
| 31 | 34 | ||
| 32 | static int __init crd_load(int in_fd, int out_fd); | 35 | static int __init crd_load(int in_fd, int out_fd, decompress_fn deco); |
| 33 | 36 | ||
| 34 | /* | 37 | /* |
| 35 | * This routine tries to find a RAM disk image to load, and returns the | 38 | * This routine tries to find a RAM disk image to load, and returns the |
| @@ -38,15 +41,15 @@ static int __init crd_load(int in_fd, int out_fd); | |||
| 38 | * numbers could not be found. | 41 | * numbers could not be found. |
| 39 | * | 42 | * |
| 40 | * We currently check for the following magic numbers: | 43 | * We currently check for the following magic numbers: |
| 41 | * minix | 44 | * minix |
| 42 | * ext2 | 45 | * ext2 |
| 43 | * romfs | 46 | * romfs |
| 44 | * cramfs | 47 | * cramfs |
| 45 | * squashfs | 48 | * squashfs |
| 46 | * gzip | 49 | * gzip |
| 47 | */ | 50 | */ |
| 48 | static int __init | 51 | static int __init |
| 49 | identify_ramdisk_image(int fd, int start_block) | 52 | identify_ramdisk_image(int fd, int start_block, decompress_fn *decompressor) |
| 50 | { | 53 | { |
| 51 | const int size = 512; | 54 | const int size = 512; |
| 52 | struct minix_super_block *minixsb; | 55 | struct minix_super_block *minixsb; |
| @@ -56,6 +59,7 @@ identify_ramdisk_image(int fd, int start_block) | |||
| 56 | struct squashfs_super_block *squashfsb; | 59 | struct squashfs_super_block *squashfsb; |
| 57 | int nblocks = -1; | 60 | int nblocks = -1; |
| 58 | unsigned char *buf; | 61 | unsigned char *buf; |
| 62 | const char *compress_name; | ||
| 59 | 63 | ||
| 60 | buf = kmalloc(size, GFP_KERNEL); | 64 | buf = kmalloc(size, GFP_KERNEL); |
| 61 | if (!buf) | 65 | if (!buf) |
| @@ -69,18 +73,19 @@ identify_ramdisk_image(int fd, int start_block) | |||
| 69 | memset(buf, 0xe5, size); | 73 | memset(buf, 0xe5, size); |
| 70 | 74 | ||
| 71 | /* | 75 | /* |
| 72 | * Read block 0 to test for gzipped kernel | 76 | * Read block 0 to test for compressed kernel |
| 73 | */ | 77 | */ |
| 74 | sys_lseek(fd, start_block * BLOCK_SIZE, 0); | 78 | sys_lseek(fd, start_block * BLOCK_SIZE, 0); |
| 75 | sys_read(fd, buf, size); | 79 | sys_read(fd, buf, size); |
| 76 | 80 | ||
| 77 | /* | 81 | *decompressor = decompress_method(buf, size, &compress_name); |
| 78 | * If it matches the gzip magic numbers, return 0 | 82 | if (compress_name) { |
| 79 | */ | 83 | printk(KERN_NOTICE "RAMDISK: %s image found at block %d\n", |
| 80 | if (buf[0] == 037 && ((buf[1] == 0213) || (buf[1] == 0236))) { | 84 | compress_name, start_block); |
| 81 | printk(KERN_NOTICE | 85 | if (!*decompressor) |
| 82 | "RAMDISK: Compressed image found at block %d\n", | 86 | printk(KERN_EMERG |
| 83 | start_block); | 87 | "RAMDISK: %s decompressor not configured!\n", |
| 88 | compress_name); | ||
| 84 | nblocks = 0; | 89 | nblocks = 0; |
| 85 | goto done; | 90 | goto done; |
| 86 | } | 91 | } |
| @@ -142,7 +147,7 @@ identify_ramdisk_image(int fd, int start_block) | |||
| 142 | printk(KERN_NOTICE | 147 | printk(KERN_NOTICE |
| 143 | "RAMDISK: Couldn't find valid RAM disk image starting at %d.\n", | 148 | "RAMDISK: Couldn't find valid RAM disk image starting at %d.\n", |
| 144 | start_block); | 149 | start_block); |
| 145 | 150 | ||
| 146 | done: | 151 | done: |
| 147 | sys_lseek(fd, start_block * BLOCK_SIZE, 0); | 152 | sys_lseek(fd, start_block * BLOCK_SIZE, 0); |
| 148 | kfree(buf); | 153 | kfree(buf); |
| @@ -157,6 +162,7 @@ int __init rd_load_image(char *from) | |||
| 157 | int nblocks, i, disk; | 162 | int nblocks, i, disk; |
| 158 | char *buf = NULL; | 163 | char *buf = NULL; |
| 159 | unsigned short rotate = 0; | 164 | unsigned short rotate = 0; |
| 165 | decompress_fn decompressor = NULL; | ||
| 160 | #if !defined(CONFIG_S390) && !defined(CONFIG_PPC_ISERIES) | 166 | #if !defined(CONFIG_S390) && !defined(CONFIG_PPC_ISERIES) |
| 161 | char rotator[4] = { '|' , '/' , '-' , '\\' }; | 167 | char rotator[4] = { '|' , '/' , '-' , '\\' }; |
| 162 | #endif | 168 | #endif |
| @@ -169,12 +175,12 @@ int __init rd_load_image(char *from) | |||
| 169 | if (in_fd < 0) | 175 | if (in_fd < 0) |
| 170 | goto noclose_input; | 176 | goto noclose_input; |
| 171 | 177 | ||
| 172 | nblocks = identify_ramdisk_image(in_fd, rd_image_start); | 178 | nblocks = identify_ramdisk_image(in_fd, rd_image_start, &decompressor); |
| 173 | if (nblocks < 0) | 179 | if (nblocks < 0) |
| 174 | goto done; | 180 | goto done; |
| 175 | 181 | ||
| 176 | if (nblocks == 0) { | 182 | if (nblocks == 0) { |
| 177 | if (crd_load(in_fd, out_fd) == 0) | 183 | if (crd_load(in_fd, out_fd, decompressor) == 0) |
| 178 | goto successful_load; | 184 | goto successful_load; |
| 179 | goto done; | 185 | goto done; |
| 180 | } | 186 | } |
| @@ -200,7 +206,7 @@ int __init rd_load_image(char *from) | |||
| 200 | nblocks, rd_blocks); | 206 | nblocks, rd_blocks); |
| 201 | goto done; | 207 | goto done; |
| 202 | } | 208 | } |
| 203 | 209 | ||
| 204 | /* | 210 | /* |
| 205 | * OK, time to copy in the data | 211 | * OK, time to copy in the data |
| 206 | */ | 212 | */ |
| @@ -273,138 +279,48 @@ int __init rd_load_disk(int n) | |||
| 273 | return rd_load_image("/dev/root"); | 279 | return rd_load_image("/dev/root"); |
| 274 | } | 280 | } |
| 275 | 281 | ||
| 276 | /* | ||
| 277 | * gzip declarations | ||
| 278 | */ | ||
| 279 | |||
| 280 | #define OF(args) args | ||
| 281 | |||
| 282 | #ifndef memzero | ||
| 283 | #define memzero(s, n) memset ((s), 0, (n)) | ||
| 284 | #endif | ||
| 285 | |||
| 286 | typedef unsigned char uch; | ||
| 287 | typedef unsigned short ush; | ||
| 288 | typedef unsigned long ulg; | ||
| 289 | |||
| 290 | #define INBUFSIZ 4096 | ||
| 291 | #define WSIZE 0x8000 /* window size--must be a power of two, and */ | ||
| 292 | /* at least 32K for zip's deflate method */ | ||
| 293 | |||
| 294 | static uch *inbuf; | ||
| 295 | static uch *window; | ||
| 296 | |||
| 297 | static unsigned insize; /* valid bytes in inbuf */ | ||
| 298 | static unsigned inptr; /* index of next byte to be processed in inbuf */ | ||
| 299 | static unsigned outcnt; /* bytes in output buffer */ | ||
| 300 | static int exit_code; | 282 | static int exit_code; |
| 301 | static int unzip_error; | 283 | static int decompress_error; |
| 302 | static long bytes_out; | ||
| 303 | static int crd_infd, crd_outfd; | 284 | static int crd_infd, crd_outfd; |
| 304 | 285 | ||
| 305 | #define get_byte() (inptr < insize ? inbuf[inptr++] : fill_inbuf()) | 286 | static int __init compr_fill(void *buf, unsigned int len) |
| 306 | |||
| 307 | /* Diagnostic functions (stubbed out) */ | ||
| 308 | #define Assert(cond,msg) | ||
| 309 | #define Trace(x) | ||
| 310 | #define Tracev(x) | ||
| 311 | #define Tracevv(x) | ||
| 312 | #define Tracec(c,x) | ||
| 313 | #define Tracecv(c,x) | ||
| 314 | |||
| 315 | #define STATIC static | ||
| 316 | #define INIT __init | ||
| 317 | |||
| 318 | static int __init fill_inbuf(void); | ||
| 319 | static void __init flush_window(void); | ||
| 320 | static void __init error(char *m); | ||
| 321 | |||
| 322 | #define NO_INFLATE_MALLOC | ||
| 323 | |||
| 324 | #include "../lib/inflate.c" | ||
| 325 | |||
| 326 | /* =========================================================================== | ||
| 327 | * Fill the input buffer. This is called only when the buffer is empty | ||
| 328 | * and at least one byte is really needed. | ||
| 329 | * Returning -1 does not guarantee that gunzip() will ever return. | ||
| 330 | */ | ||
| 331 | static int __init fill_inbuf(void) | ||
| 332 | { | 287 | { |
| 333 | if (exit_code) return -1; | 288 | int r = sys_read(crd_infd, buf, len); |
| 334 | 289 | if (r < 0) | |
| 335 | insize = sys_read(crd_infd, inbuf, INBUFSIZ); | 290 | printk(KERN_ERR "RAMDISK: error while reading compressed data"); |
| 336 | if (insize == 0) { | 291 | else if (r == 0) |
| 337 | error("RAMDISK: ran out of compressed data"); | 292 | printk(KERN_ERR "RAMDISK: EOF while reading compressed data"); |
| 338 | return -1; | 293 | return r; |
| 339 | } | ||
| 340 | |||
| 341 | inptr = 1; | ||
| 342 | |||
| 343 | return inbuf[0]; | ||
| 344 | } | 294 | } |
| 345 | 295 | ||
| 346 | /* =========================================================================== | 296 | static int __init compr_flush(void *window, unsigned int outcnt) |
| 347 | * Write the output window window[0..outcnt-1] and update crc and bytes_out. | ||
| 348 | * (Used for the decompressed data only.) | ||
| 349 | */ | ||
| 350 | static void __init flush_window(void) | ||
| 351 | { | 297 | { |
| 352 | ulg c = crc; /* temporary variable */ | 298 | int written = sys_write(crd_outfd, window, outcnt); |
| 353 | unsigned n, written; | 299 | if (written != outcnt) { |
| 354 | uch *in, ch; | 300 | if (decompress_error == 0) |
| 355 | 301 | printk(KERN_ERR | |
| 356 | written = sys_write(crd_outfd, window, outcnt); | 302 | "RAMDISK: incomplete write (%d != %d)\n", |
| 357 | if (written != outcnt && unzip_error == 0) { | 303 | written, outcnt); |
| 358 | printk(KERN_ERR "RAMDISK: incomplete write (%d != %d) %ld\n", | 304 | decompress_error = 1; |
| 359 | written, outcnt, bytes_out); | 305 | return -1; |
| 360 | unzip_error = 1; | 306 | } |
| 361 | } | 307 | return outcnt; |
| 362 | in = window; | ||
| 363 | for (n = 0; n < outcnt; n++) { | ||
| 364 | ch = *in++; | ||
| 365 | c = crc_32_tab[((int)c ^ ch) & 0xff] ^ (c >> 8); | ||
| 366 | } | ||
| 367 | crc = c; | ||
| 368 | bytes_out += (ulg)outcnt; | ||
| 369 | outcnt = 0; | ||
| 370 | } | 308 | } |
| 371 | 309 | ||
| 372 | static void __init error(char *x) | 310 | static void __init error(char *x) |
| 373 | { | 311 | { |
| 374 | printk(KERN_ERR "%s\n", x); | 312 | printk(KERN_ERR "%s\n", x); |
| 375 | exit_code = 1; | 313 | exit_code = 1; |
| 376 | unzip_error = 1; | 314 | decompress_error = 1; |
| 377 | } | 315 | } |
| 378 | 316 | ||
| 379 | static int __init crd_load(int in_fd, int out_fd) | 317 | static int __init crd_load(int in_fd, int out_fd, decompress_fn deco) |
| 380 | { | 318 | { |
| 381 | int result; | 319 | int result; |
| 382 | |||
| 383 | insize = 0; /* valid bytes in inbuf */ | ||
| 384 | inptr = 0; /* index of next byte to be processed in inbuf */ | ||
| 385 | outcnt = 0; /* bytes in output buffer */ | ||
| 386 | exit_code = 0; | ||
| 387 | bytes_out = 0; | ||
| 388 | crc = (ulg)0xffffffffL; /* shift register contents */ | ||
| 389 | |||
| 390 | crd_infd = in_fd; | 320 | crd_infd = in_fd; |
| 391 | crd_outfd = out_fd; | 321 | crd_outfd = out_fd; |
| 392 | inbuf = kmalloc(INBUFSIZ, GFP_KERNEL); | 322 | result = deco(NULL, 0, compr_fill, compr_flush, NULL, NULL, error); |
| 393 | if (!inbuf) { | 323 | if (decompress_error) |
| 394 | printk(KERN_ERR "RAMDISK: Couldn't allocate gzip buffer\n"); | ||
| 395 | return -1; | ||
| 396 | } | ||
| 397 | window = kmalloc(WSIZE, GFP_KERNEL); | ||
| 398 | if (!window) { | ||
| 399 | printk(KERN_ERR "RAMDISK: Couldn't allocate gzip window\n"); | ||
| 400 | kfree(inbuf); | ||
| 401 | return -1; | ||
| 402 | } | ||
| 403 | makecrc(); | ||
| 404 | result = gunzip(); | ||
| 405 | if (unzip_error) | ||
| 406 | result = 1; | 324 | result = 1; |
| 407 | kfree(inbuf); | ||
| 408 | kfree(window); | ||
| 409 | return result; | 325 | return result; |
| 410 | } | 326 | } |
