| Jeff Dike | 9b67a3c | 2005-05-20 13:59:10 -0700 | [diff] [blame] | 1 | /* | 
| Jeff Dike | 0983a88 | 2008-02-04 22:31:08 -0800 | [diff] [blame] | 2 | * Copyright (C) 2000 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com) | 
| Jeff Dike | 9b67a3c | 2005-05-20 13:59:10 -0700 | [diff] [blame] | 3 | * Licensed under the GPL | 
|  | 4 | */ | 
|  | 5 |  | 
|  | 6 | #include "linux/init.h" | 
|  | 7 | #include "linux/bootmem.h" | 
|  | 8 | #include "linux/initrd.h" | 
|  | 9 | #include "asm/types.h" | 
| Jeff Dike | 9b67a3c | 2005-05-20 13:59:10 -0700 | [diff] [blame] | 10 | #include "initrd.h" | 
|  | 11 | #include "init.h" | 
|  | 12 | #include "os.h" | 
|  | 13 |  | 
|  | 14 | /* Changed by uml_initrd_setup, which is a setup */ | 
|  | 15 | static char *initrd __initdata = NULL; | 
|  | 16 |  | 
|  | 17 | static int __init read_initrd(void) | 
|  | 18 | { | 
|  | 19 | void *area; | 
|  | 20 | long long size; | 
|  | 21 | int err; | 
|  | 22 |  | 
| Jeff Dike | 0983a88 | 2008-02-04 22:31:08 -0800 | [diff] [blame] | 23 | if (initrd == NULL) | 
| Jeff Dike | dc764e5 | 2007-05-06 14:51:41 -0700 | [diff] [blame] | 24 | return 0; | 
|  | 25 |  | 
| Jeff Dike | 9b67a3c | 2005-05-20 13:59:10 -0700 | [diff] [blame] | 26 | err = os_file_size(initrd, &size); | 
| Jeff Dike | 0983a88 | 2008-02-04 22:31:08 -0800 | [diff] [blame] | 27 | if (err) | 
| Jeff Dike | dc764e5 | 2007-05-06 14:51:41 -0700 | [diff] [blame] | 28 | return 0; | 
|  | 29 |  | 
| Jeff Dike | 0983a88 | 2008-02-04 22:31:08 -0800 | [diff] [blame] | 30 | /* | 
|  | 31 | * This is necessary because alloc_bootmem craps out if you | 
|  | 32 | * ask for no memory. | 
|  | 33 | */ | 
|  | 34 | if (size == 0) { | 
| Johann Felix Soden | 4631a9a | 2008-02-23 15:23:23 -0800 | [diff] [blame] | 35 | printk(KERN_ERR "\"%s\" is a zero-size initrd\n", initrd); | 
| Jeff Dike | 0983a88 | 2008-02-04 22:31:08 -0800 | [diff] [blame] | 36 | return 0; | 
|  | 37 | } | 
|  | 38 |  | 
| Jeff Dike | 9b67a3c | 2005-05-20 13:59:10 -0700 | [diff] [blame] | 39 | area = alloc_bootmem(size); | 
| Jeff Dike | 0983a88 | 2008-02-04 22:31:08 -0800 | [diff] [blame] | 40 | if (area == NULL) | 
| Jeff Dike | dc764e5 | 2007-05-06 14:51:41 -0700 | [diff] [blame] | 41 | return 0; | 
|  | 42 |  | 
| Jeff Dike | 0983a88 | 2008-02-04 22:31:08 -0800 | [diff] [blame] | 43 | if (load_initrd(initrd, area, size) == -1) | 
| Jeff Dike | dc764e5 | 2007-05-06 14:51:41 -0700 | [diff] [blame] | 44 | return 0; | 
|  | 45 |  | 
| Jeff Dike | 9b67a3c | 2005-05-20 13:59:10 -0700 | [diff] [blame] | 46 | initrd_start = (unsigned long) area; | 
|  | 47 | initrd_end = initrd_start + size; | 
|  | 48 | return 0; | 
|  | 49 | } | 
|  | 50 |  | 
|  | 51 | __uml_postsetup(read_initrd); | 
|  | 52 |  | 
|  | 53 | static int __init uml_initrd_setup(char *line, int *add) | 
|  | 54 | { | 
|  | 55 | initrd = line; | 
|  | 56 | return 0; | 
|  | 57 | } | 
|  | 58 |  | 
|  | 59 | __uml_setup("initrd=", uml_initrd_setup, | 
|  | 60 | "initrd=<initrd image>\n" | 
|  | 61 | "    This is used to boot UML from an initrd image.  The argument is the\n" | 
|  | 62 | "    name of the file containing the image.\n\n" | 
|  | 63 | ); | 
|  | 64 |  | 
|  | 65 | int load_initrd(char *filename, void *buf, int size) | 
|  | 66 | { | 
|  | 67 | int fd, n; | 
|  | 68 |  | 
|  | 69 | fd = os_open_file(filename, of_read(OPENFLAGS()), 0); | 
| Jeff Dike | 0983a88 | 2008-02-04 22:31:08 -0800 | [diff] [blame] | 70 | if (fd < 0) { | 
|  | 71 | printk(KERN_ERR "Opening '%s' failed - err = %d\n", filename, | 
|  | 72 | -fd); | 
| Jeff Dike | dc764e5 | 2007-05-06 14:51:41 -0700 | [diff] [blame] | 73 | return -1; | 
| Jeff Dike | 9b67a3c | 2005-05-20 13:59:10 -0700 | [diff] [blame] | 74 | } | 
| Jeff Dike | a6ea4cc | 2007-05-06 14:51:43 -0700 | [diff] [blame] | 75 | n = os_read_file(fd, buf, size); | 
| Jeff Dike | 0983a88 | 2008-02-04 22:31:08 -0800 | [diff] [blame] | 76 | if (n != size) { | 
|  | 77 | printk(KERN_ERR "Read of %d bytes from '%s' failed, " | 
|  | 78 | "err = %d\n", size, | 
| Jeff Dike | 9b67a3c | 2005-05-20 13:59:10 -0700 | [diff] [blame] | 79 | filename, -n); | 
| Jeff Dike | dc764e5 | 2007-05-06 14:51:41 -0700 | [diff] [blame] | 80 | return -1; | 
| Jeff Dike | 9b67a3c | 2005-05-20 13:59:10 -0700 | [diff] [blame] | 81 | } | 
|  | 82 |  | 
|  | 83 | os_close_file(fd); | 
| Jeff Dike | dc764e5 | 2007-05-06 14:51:41 -0700 | [diff] [blame] | 84 | return 0; | 
| Jeff Dike | 9b67a3c | 2005-05-20 13:59:10 -0700 | [diff] [blame] | 85 | } |