diff options
Diffstat (limited to 'init/do_mounts_rd.c')
-rw-r--r-- | init/do_mounts_rd.c | 182 |
1 files changed, 65 insertions, 117 deletions
diff --git a/init/do_mounts_rd.c b/init/do_mounts_rd.c index a7c748fa977a..dcaeb1f90b32 100644 --- a/init/do_mounts_rd.c +++ b/init/do_mounts_rd.c | |||
@@ -10,6 +10,12 @@ | |||
10 | 10 | ||
11 | #include "do_mounts.h" | 11 | #include "do_mounts.h" |
12 | 12 | ||
13 | #include <linux/decompress/generic.h> | ||
14 | |||
15 | #include <linux/decompress/bunzip2.h> | ||
16 | #include <linux/decompress/unlzma.h> | ||
17 | #include <linux/decompress/inflate.h> | ||
18 | |||
13 | int __initdata rd_prompt = 1;/* 1 = prompt for RAM disk, 0 = don't prompt */ | 19 | int __initdata rd_prompt = 1;/* 1 = prompt for RAM disk, 0 = don't prompt */ |
14 | 20 | ||
15 | static int __init prompt_ramdisk(char *str) | 21 | static int __init prompt_ramdisk(char *str) |
@@ -28,7 +34,7 @@ static int __init ramdisk_start_setup(char *str) | |||
28 | } | 34 | } |
29 | __setup("ramdisk_start=", ramdisk_start_setup); | 35 | __setup("ramdisk_start=", ramdisk_start_setup); |
30 | 36 | ||
31 | static int __init crd_load(int in_fd, int out_fd); | 37 | static int __init crd_load(int in_fd, int out_fd, decompress_fn deco); |
32 | 38 | ||
33 | /* | 39 | /* |
34 | * This routine tries to find a RAM disk image to load, and returns the | 40 | * This routine tries to find a RAM disk image to load, and returns the |
@@ -44,7 +50,7 @@ static int __init crd_load(int in_fd, int out_fd); | |||
44 | * gzip | 50 | * gzip |
45 | */ | 51 | */ |
46 | static int __init | 52 | static int __init |
47 | identify_ramdisk_image(int fd, int start_block) | 53 | identify_ramdisk_image(int fd, int start_block, decompress_fn *decompressor) |
48 | { | 54 | { |
49 | const int size = 512; | 55 | const int size = 512; |
50 | struct minix_super_block *minixsb; | 56 | struct minix_super_block *minixsb; |
@@ -70,6 +76,7 @@ identify_ramdisk_image(int fd, int start_block) | |||
70 | sys_lseek(fd, start_block * BLOCK_SIZE, 0); | 76 | sys_lseek(fd, start_block * BLOCK_SIZE, 0); |
71 | sys_read(fd, buf, size); | 77 | sys_read(fd, buf, size); |
72 | 78 | ||
79 | #ifdef CONFIG_RD_GZIP | ||
73 | /* | 80 | /* |
74 | * If it matches the gzip magic numbers, return 0 | 81 | * If it matches the gzip magic numbers, return 0 |
75 | */ | 82 | */ |
@@ -77,9 +84,39 @@ identify_ramdisk_image(int fd, int start_block) | |||
77 | printk(KERN_NOTICE | 84 | printk(KERN_NOTICE |
78 | "RAMDISK: Compressed image found at block %d\n", | 85 | "RAMDISK: Compressed image found at block %d\n", |
79 | start_block); | 86 | start_block); |
87 | *decompressor = gunzip; | ||
88 | nblocks = 0; | ||
89 | goto done; | ||
90 | } | ||
91 | #endif | ||
92 | |||
93 | #ifdef CONFIG_RD_BZIP2 | ||
94 | /* | ||
95 | * If it matches the bzip2 magic numbers, return -1 | ||
96 | */ | ||
97 | if (buf[0] == 0x42 && (buf[1] == 0x5a)) { | ||
98 | printk(KERN_NOTICE | ||
99 | "RAMDISK: Bzipped image found at block %d\n", | ||
100 | start_block); | ||
101 | *decompressor = bunzip2; | ||
102 | nblocks = 0; | ||
103 | goto done; | ||
104 | } | ||
105 | #endif | ||
106 | |||
107 | #ifdef CONFIG_RD_LZMA | ||
108 | /* | ||
109 | * If it matches the lzma magic numbers, return -1 | ||
110 | */ | ||
111 | if (buf[0] == 0x5d && (buf[1] == 0x00)) { | ||
112 | printk(KERN_NOTICE | ||
113 | "RAMDISK: Lzma image found at block %d\n", | ||
114 | start_block); | ||
115 | *decompressor = unlzma; | ||
80 | nblocks = 0; | 116 | nblocks = 0; |
81 | goto done; | 117 | goto done; |
82 | } | 118 | } |
119 | #endif | ||
83 | 120 | ||
84 | /* romfs is at block zero too */ | 121 | /* romfs is at block zero too */ |
85 | if (romfsb->word0 == ROMSB_WORD0 && | 122 | if (romfsb->word0 == ROMSB_WORD0 && |
@@ -143,6 +180,7 @@ int __init rd_load_image(char *from) | |||
143 | int nblocks, i, disk; | 180 | int nblocks, i, disk; |
144 | char *buf = NULL; | 181 | char *buf = NULL; |
145 | unsigned short rotate = 0; | 182 | unsigned short rotate = 0; |
183 | decompress_fn decompressor = NULL; | ||
146 | #if !defined(CONFIG_S390) && !defined(CONFIG_PPC_ISERIES) | 184 | #if !defined(CONFIG_S390) && !defined(CONFIG_PPC_ISERIES) |
147 | char rotator[4] = { '|' , '/' , '-' , '\\' }; | 185 | char rotator[4] = { '|' , '/' , '-' , '\\' }; |
148 | #endif | 186 | #endif |
@@ -155,12 +193,12 @@ int __init rd_load_image(char *from) | |||
155 | if (in_fd < 0) | 193 | if (in_fd < 0) |
156 | goto noclose_input; | 194 | goto noclose_input; |
157 | 195 | ||
158 | nblocks = identify_ramdisk_image(in_fd, rd_image_start); | 196 | nblocks = identify_ramdisk_image(in_fd, rd_image_start, &decompressor); |
159 | if (nblocks < 0) | 197 | if (nblocks < 0) |
160 | goto done; | 198 | goto done; |
161 | 199 | ||
162 | if (nblocks == 0) { | 200 | if (nblocks == 0) { |
163 | if (crd_load(in_fd, out_fd) == 0) | 201 | if (crd_load(in_fd, out_fd, decompressor) == 0) |
164 | goto successful_load; | 202 | goto successful_load; |
165 | goto done; | 203 | goto done; |
166 | } | 204 | } |
@@ -259,138 +297,48 @@ int __init rd_load_disk(int n) | |||
259 | return rd_load_image("/dev/root"); | 297 | return rd_load_image("/dev/root"); |
260 | } | 298 | } |
261 | 299 | ||
262 | /* | ||
263 | * gzip declarations | ||
264 | */ | ||
265 | |||
266 | #define OF(args) args | ||
267 | |||
268 | #ifndef memzero | ||
269 | #define memzero(s, n) memset ((s), 0, (n)) | ||
270 | #endif | ||
271 | |||
272 | typedef unsigned char uch; | ||
273 | typedef unsigned short ush; | ||
274 | typedef unsigned long ulg; | ||
275 | |||
276 | #define INBUFSIZ 4096 | ||
277 | #define WSIZE 0x8000 /* window size--must be a power of two, and */ | ||
278 | /* at least 32K for zip's deflate method */ | ||
279 | |||
280 | static uch *inbuf; | ||
281 | static uch *window; | ||
282 | |||
283 | static unsigned insize; /* valid bytes in inbuf */ | ||
284 | static unsigned inptr; /* index of next byte to be processed in inbuf */ | ||
285 | static unsigned outcnt; /* bytes in output buffer */ | ||
286 | static int exit_code; | 300 | static int exit_code; |
287 | static int unzip_error; | 301 | static int decompress_error; |
288 | static long bytes_out; | ||
289 | static int crd_infd, crd_outfd; | 302 | static int crd_infd, crd_outfd; |
290 | 303 | ||
291 | #define get_byte() (inptr < insize ? inbuf[inptr++] : fill_inbuf()) | 304 | static int __init compr_fill(void *buf, unsigned int len) |
292 | |||
293 | /* Diagnostic functions (stubbed out) */ | ||
294 | #define Assert(cond,msg) | ||
295 | #define Trace(x) | ||
296 | #define Tracev(x) | ||
297 | #define Tracevv(x) | ||
298 | #define Tracec(c,x) | ||
299 | #define Tracecv(c,x) | ||
300 | |||
301 | #define STATIC static | ||
302 | #define INIT __init | ||
303 | |||
304 | static int __init fill_inbuf(void); | ||
305 | static void __init flush_window(void); | ||
306 | static void __init error(char *m); | ||
307 | |||
308 | #define NO_INFLATE_MALLOC | ||
309 | |||
310 | #include "../lib/inflate.c" | ||
311 | |||
312 | /* =========================================================================== | ||
313 | * Fill the input buffer. This is called only when the buffer is empty | ||
314 | * and at least one byte is really needed. | ||
315 | * Returning -1 does not guarantee that gunzip() will ever return. | ||
316 | */ | ||
317 | static int __init fill_inbuf(void) | ||
318 | { | 305 | { |
319 | if (exit_code) return -1; | 306 | int r = sys_read(crd_infd, buf, len); |
320 | 307 | if (r < 0) | |
321 | insize = sys_read(crd_infd, inbuf, INBUFSIZ); | 308 | printk(KERN_ERR "RAMDISK: error while reading compressed data"); |
322 | if (insize == 0) { | 309 | else if (r == 0) |
323 | error("RAMDISK: ran out of compressed data"); | 310 | printk(KERN_ERR "RAMDISK: EOF while reading compressed data"); |
324 | return -1; | 311 | return r; |
325 | } | ||
326 | |||
327 | inptr = 1; | ||
328 | |||
329 | return inbuf[0]; | ||
330 | } | 312 | } |
331 | 313 | ||
332 | /* =========================================================================== | 314 | static int __init compr_flush(void *window, unsigned int outcnt) |
333 | * Write the output window window[0..outcnt-1] and update crc and bytes_out. | ||
334 | * (Used for the decompressed data only.) | ||
335 | */ | ||
336 | static void __init flush_window(void) | ||
337 | { | 315 | { |
338 | ulg c = crc; /* temporary variable */ | 316 | int written = sys_write(crd_outfd, window, outcnt); |
339 | unsigned n, written; | 317 | if (written != outcnt) { |
340 | uch *in, ch; | 318 | if (decompress_error == 0) |
341 | 319 | printk(KERN_ERR | |
342 | written = sys_write(crd_outfd, window, outcnt); | 320 | "RAMDISK: incomplete write (%d != %d)\n", |
343 | if (written != outcnt && unzip_error == 0) { | 321 | written, outcnt); |
344 | printk(KERN_ERR "RAMDISK: incomplete write (%d != %d) %ld\n", | 322 | decompress_error = 1; |
345 | written, outcnt, bytes_out); | 323 | return -1; |
346 | unzip_error = 1; | 324 | } |
347 | } | 325 | return outcnt; |
348 | in = window; | ||
349 | for (n = 0; n < outcnt; n++) { | ||
350 | ch = *in++; | ||
351 | c = crc_32_tab[((int)c ^ ch) & 0xff] ^ (c >> 8); | ||
352 | } | ||
353 | crc = c; | ||
354 | bytes_out += (ulg)outcnt; | ||
355 | outcnt = 0; | ||
356 | } | 326 | } |
357 | 327 | ||
358 | static void __init error(char *x) | 328 | static void __init error(char *x) |
359 | { | 329 | { |
360 | printk(KERN_ERR "%s\n", x); | 330 | printk(KERN_ERR "%s\n", x); |
361 | exit_code = 1; | 331 | exit_code = 1; |
362 | unzip_error = 1; | 332 | decompress_error = 1; |
363 | } | 333 | } |
364 | 334 | ||
365 | static int __init crd_load(int in_fd, int out_fd) | 335 | static int __init crd_load(int in_fd, int out_fd, decompress_fn deco) |
366 | { | 336 | { |
367 | int result; | 337 | int result; |
368 | |||
369 | insize = 0; /* valid bytes in inbuf */ | ||
370 | inptr = 0; /* index of next byte to be processed in inbuf */ | ||
371 | outcnt = 0; /* bytes in output buffer */ | ||
372 | exit_code = 0; | ||
373 | bytes_out = 0; | ||
374 | crc = (ulg)0xffffffffL; /* shift register contents */ | ||
375 | |||
376 | crd_infd = in_fd; | 338 | crd_infd = in_fd; |
377 | crd_outfd = out_fd; | 339 | crd_outfd = out_fd; |
378 | inbuf = kmalloc(INBUFSIZ, GFP_KERNEL); | 340 | result = deco(NULL, 0, compr_fill, compr_flush, NULL, NULL, error); |
379 | if (!inbuf) { | 341 | if (decompress_error) |
380 | printk(KERN_ERR "RAMDISK: Couldn't allocate gzip buffer\n"); | ||
381 | return -1; | ||
382 | } | ||
383 | window = kmalloc(WSIZE, GFP_KERNEL); | ||
384 | if (!window) { | ||
385 | printk(KERN_ERR "RAMDISK: Couldn't allocate gzip window\n"); | ||
386 | kfree(inbuf); | ||
387 | return -1; | ||
388 | } | ||
389 | makecrc(); | ||
390 | result = gunzip(); | ||
391 | if (unzip_error) | ||
392 | result = 1; | 342 | result = 1; |
393 | kfree(inbuf); | ||
394 | kfree(window); | ||
395 | return result; | 343 | return result; |
396 | } | 344 | } |