Blame view

init/do_mounts_rd.c 8.1 KB
b24413180   Greg Kroah-Hartman   License cleanup: ...
1
  // SPDX-License-Identifier: GPL-2.0
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2
3
4
5
6
  #include <linux/kernel.h>
  #include <linux/fs.h>
  #include <linux/minix_fs.h>
  #include <linux/ext2_fs.h>
  #include <linux/romfs_fs.h>
f7f4f4dd6   Al Viro   cramfs: take head...
7
  #include <uapi/linux/cramfs_fs.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
8
9
  #include <linux/initrd.h>
  #include <linux/string.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
10
  #include <linux/slab.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
11
12
  
  #include "do_mounts.h"
b8fed87d3   Phillip Lougher   Squashfs: initrd ...
13
  #include "../fs/squashfs/squashfs_fs.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
14

30d65dbfe   Alain Knaff   bzip2/lzma: confi...
15
  #include <linux/decompress/generic.h>
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
16

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
  int __initdata rd_prompt = 1;/* 1 = prompt for RAM disk, 0 = don't prompt */
  
  static int __init prompt_ramdisk(char *str)
  {
  	rd_prompt = simple_strtol(str,NULL,0) & 1;
  	return 1;
  }
  __setup("prompt_ramdisk=", prompt_ramdisk);
  
  int __initdata rd_image_start;		/* starting block # of image */
  
  static int __init ramdisk_start_setup(char *str)
  {
  	rd_image_start = simple_strtol(str,NULL,0);
  	return 1;
  }
  __setup("ramdisk_start=", ramdisk_start_setup);
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
34
  static int __init crd_load(int in_fd, int out_fd, decompress_fn deco);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
35
36
37
38
39
40
41
42
  
  /*
   * This routine tries to find a RAM disk image to load, and returns the
   * number of blocks to read for a non-compressed image, 0 if the image
   * is a compressed image, and -1 if an image with the right magic
   * numbers could not be found.
   *
   * We currently check for the following magic numbers:
b172fd882   H. Peter Anvin   bzip2/lzma: use a...
43
44
   *	minix
   *	ext2
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
45
46
   *	romfs
   *	cramfs
b8fed87d3   Phillip Lougher   Squashfs: initrd ...
47
   *	squashfs
b172fd882   H. Peter Anvin   bzip2/lzma: use a...
48
   *	gzip
1bf49dd4b   P J P   ./Makefile: expor...
49
50
51
52
53
   *	bzip2
   *	lzma
   *	xz
   *	lzo
   *	lz4
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
54
   */
b172fd882   H. Peter Anvin   bzip2/lzma: use a...
55
  static int __init
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
56
  identify_ramdisk_image(int fd, int start_block, decompress_fn *decompressor)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
57
58
59
  {
  	const int size = 512;
  	struct minix_super_block *minixsb;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
60
61
  	struct romfs_super_block *romfsb;
  	struct cramfs_super *cramfsb;
b8fed87d3   Phillip Lougher   Squashfs: initrd ...
62
  	struct squashfs_super_block *squashfsb;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
63
64
  	int nblocks = -1;
  	unsigned char *buf;
889c92d21   H. Peter Anvin   bzip2/lzma: centr...
65
  	const char *compress_name;
39429c5e4   Al Viro   new helper: ext2_...
66
  	unsigned long n;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
67
68
  
  	buf = kmalloc(size, GFP_KERNEL);
c80544dc0   Stephen Hemminger   sparse pointer us...
69
  	if (!buf)
ea611b269   Davidlohr Bueso   init: return prop...
70
  		return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
71
72
  
  	minixsb = (struct minix_super_block *) buf;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
73
74
  	romfsb = (struct romfs_super_block *) buf;
  	cramfsb = (struct cramfs_super *) buf;
b8fed87d3   Phillip Lougher   Squashfs: initrd ...
75
  	squashfsb = (struct squashfs_super_block *) buf;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
76
77
78
  	memset(buf, 0xe5, size);
  
  	/*
b172fd882   H. Peter Anvin   bzip2/lzma: use a...
79
  	 * Read block 0 to test for compressed kernel
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
80
  	 */
76847e434   Dominik Brodowski   fs: add ksys_lsee...
81
  	ksys_lseek(fd, start_block * BLOCK_SIZE, 0);
3ce4a7bf6   Dominik Brodowski   fs: add ksys_read...
82
  	ksys_read(fd, buf, size);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
83

889c92d21   H. Peter Anvin   bzip2/lzma: centr...
84
  	*decompressor = decompress_method(buf, size, &compress_name);
23a22d57a   H. Peter Anvin   bzip2/lzma: compr...
85
  	if (compress_name) {
889c92d21   H. Peter Anvin   bzip2/lzma: centr...
86
87
88
  		printk(KERN_NOTICE "RAMDISK: %s image found at block %d
  ",
  		       compress_name, start_block);
23a22d57a   H. Peter Anvin   bzip2/lzma: compr...
89
  		if (!*decompressor)
73310a169   H. Peter Anvin   init: make initrd...
90
91
92
  			printk(KERN_EMERG
  			       "RAMDISK: %s decompressor not configured!
  ",
23a22d57a   H. Peter Anvin   bzip2/lzma: compr...
93
  			       compress_name);
889c92d21   H. Peter Anvin   bzip2/lzma: centr...
94
95
  		nblocks = 0;
  		goto done;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
  	}
  
  	/* romfs is at block zero too */
  	if (romfsb->word0 == ROMSB_WORD0 &&
  	    romfsb->word1 == ROMSB_WORD1) {
  		printk(KERN_NOTICE
  		       "RAMDISK: romfs filesystem found at block %d
  ",
  		       start_block);
  		nblocks = (ntohl(romfsb->size)+BLOCK_SIZE-1)>>BLOCK_SIZE_BITS;
  		goto done;
  	}
  
  	if (cramfsb->magic == CRAMFS_MAGIC) {
  		printk(KERN_NOTICE
  		       "RAMDISK: cramfs filesystem found at block %d
  ",
  		       start_block);
  		nblocks = (cramfsb->size + BLOCK_SIZE - 1) >> BLOCK_SIZE_BITS;
  		goto done;
  	}
b8fed87d3   Phillip Lougher   Squashfs: initrd ...
117
118
119
120
121
122
123
124
125
126
  	/* squashfs is at block zero too */
  	if (le32_to_cpu(squashfsb->s_magic) == SQUASHFS_MAGIC) {
  		printk(KERN_NOTICE
  		       "RAMDISK: squashfs filesystem found at block %d
  ",
  		       start_block);
  		nblocks = (le64_to_cpu(squashfsb->bytes_used) + BLOCK_SIZE - 1)
  			 >> BLOCK_SIZE_BITS;
  		goto done;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
127
  	/*
f919b9235   Neil Armstrong   init/do_mounts_rd...
128
129
  	 * Read 512 bytes further to check if cramfs is padded
  	 */
76847e434   Dominik Brodowski   fs: add ksys_lsee...
130
  	ksys_lseek(fd, start_block * BLOCK_SIZE + 0x200, 0);
3ce4a7bf6   Dominik Brodowski   fs: add ksys_read...
131
  	ksys_read(fd, buf, size);
f919b9235   Neil Armstrong   init/do_mounts_rd...
132
133
134
135
136
137
138
139
140
141
142
  
  	if (cramfsb->magic == CRAMFS_MAGIC) {
  		printk(KERN_NOTICE
  		       "RAMDISK: cramfs filesystem found at block %d
  ",
  		       start_block);
  		nblocks = (cramfsb->size + BLOCK_SIZE - 1) >> BLOCK_SIZE_BITS;
  		goto done;
  	}
  
  	/*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
143
144
  	 * Read block 1 to test for minix and ext2 superblock
  	 */
76847e434   Dominik Brodowski   fs: add ksys_lsee...
145
  	ksys_lseek(fd, (start_block+1) * BLOCK_SIZE, 0);
3ce4a7bf6   Dominik Brodowski   fs: add ksys_read...
146
  	ksys_read(fd, buf, size);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
147
148
149
150
151
152
153
154
155
156
157
158
159
  
  	/* Try minix */
  	if (minixsb->s_magic == MINIX_SUPER_MAGIC ||
  	    minixsb->s_magic == MINIX_SUPER_MAGIC2) {
  		printk(KERN_NOTICE
  		       "RAMDISK: Minix filesystem found at block %d
  ",
  		       start_block);
  		nblocks = minixsb->s_nzones << minixsb->s_log_zone_size;
  		goto done;
  	}
  
  	/* Try ext2 */
39429c5e4   Al Viro   new helper: ext2_...
160
161
  	n = ext2_image_size(buf);
  	if (n) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
162
163
164
165
  		printk(KERN_NOTICE
  		       "RAMDISK: ext2 filesystem found at block %d
  ",
  		       start_block);
39429c5e4   Al Viro   new helper: ext2_...
166
  		nblocks = n;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
167
168
169
170
171
172
173
  		goto done;
  	}
  
  	printk(KERN_NOTICE
  	       "RAMDISK: Couldn't find valid RAM disk image starting at %d.
  ",
  	       start_block);
b172fd882   H. Peter Anvin   bzip2/lzma: use a...
174

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
175
  done:
76847e434   Dominik Brodowski   fs: add ksys_lsee...
176
  	ksys_lseek(fd, start_block * BLOCK_SIZE, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
177
178
179
180
181
182
183
184
185
186
187
188
  	kfree(buf);
  	return nblocks;
  }
  
  int __init rd_load_image(char *from)
  {
  	int res = 0;
  	int in_fd, out_fd;
  	unsigned long rd_blocks, devblocks;
  	int nblocks, i, disk;
  	char *buf = NULL;
  	unsigned short rotate = 0;
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
189
  	decompress_fn decompressor = NULL;
bc58450b0   Stephen Rothwell   init: Remove CONF...
190
  #if !defined(CONFIG_S390)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
191
192
  	char rotator[4] = { '|' , '/' , '-' , '\\' };
  #endif
bae217ea8   Dominik Brodowski   fs: add ksys_open...
193
  	out_fd = ksys_open("/dev/ram", O_RDWR, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
194
195
  	if (out_fd < 0)
  		goto out;
bae217ea8   Dominik Brodowski   fs: add ksys_open...
196
  	in_fd = ksys_open(from, O_RDONLY, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
197
198
  	if (in_fd < 0)
  		goto noclose_input;
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
199
  	nblocks = identify_ramdisk_image(in_fd, rd_image_start, &decompressor);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
200
201
202
203
  	if (nblocks < 0)
  		goto done;
  
  	if (nblocks == 0) {
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
204
  		if (crd_load(in_fd, out_fd, decompressor) == 0)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
205
  			goto successful_load;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
206
207
208
209
210
211
  		goto done;
  	}
  
  	/*
  	 * NOTE NOTE: nblocks is not actually blocks but
  	 * the number of kibibytes of data to load into a ramdisk.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
212
  	 */
cbb60b924   Dominik Brodowski   fs: add ksys_ioct...
213
  	if (ksys_ioctl(out_fd, BLKGETSIZE, (unsigned long)&rd_blocks) < 0)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
214
215
216
217
218
219
220
221
222
223
  		rd_blocks = 0;
  	else
  		rd_blocks >>= 1;
  
  	if (nblocks > rd_blocks) {
  		printk("RAMDISK: image too big! (%dKiB/%ldKiB)
  ",
  		       nblocks, rd_blocks);
  		goto done;
  	}
b172fd882   H. Peter Anvin   bzip2/lzma: use a...
224

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
225
226
227
  	/*
  	 * OK, time to copy in the data
  	 */
cbb60b924   Dominik Brodowski   fs: add ksys_ioct...
228
  	if (ksys_ioctl(in_fd, BLKGETSIZE, (unsigned long)&devblocks) < 0)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
229
230
231
232
233
234
235
236
237
238
239
240
241
242
  		devblocks = 0;
  	else
  		devblocks >>= 1;
  
  	if (strcmp(from, "/initrd.image") == 0)
  		devblocks = nblocks;
  
  	if (devblocks == 0) {
  		printk(KERN_ERR "RAMDISK: could not determine device size
  ");
  		goto done;
  	}
  
  	buf = kmalloc(BLOCK_SIZE, GFP_KERNEL);
d613c3e2d   Harvey Harrison   init: fix integer...
243
  	if (!buf) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
244
245
246
247
248
249
250
251
252
  		printk(KERN_ERR "RAMDISK: could not allocate buffer
  ");
  		goto done;
  	}
  
  	printk(KERN_NOTICE "RAMDISK: Loading %dKiB [%ld disk%s] into ram disk... ",
  		nblocks, ((nblocks-1)/devblocks)+1, nblocks>devblocks ? "s" : "");
  	for (i = 0, disk = 1; i < nblocks; i++) {
  		if (i && (i % devblocks == 0)) {
1a6a05a4f   Aaro Koskinen   init/ramdisk: use...
253
254
  			pr_cont("done disk #%d.
  ", disk++);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
255
  			rotate = 0;
2ca2a09d6   Dominik Brodowski   fs: add ksys_clos...
256
  			if (ksys_close(in_fd)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
257
258
259
260
261
  				printk("Error closing the disk.
  ");
  				goto noclose_input;
  			}
  			change_floppy("disk #%d", disk);
bae217ea8   Dominik Brodowski   fs: add ksys_open...
262
  			in_fd = ksys_open(from, O_RDONLY, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
263
264
265
266
267
268
269
  			if (in_fd < 0)  {
  				printk("Error opening disk.
  ");
  				goto noclose_input;
  			}
  			printk("Loading disk #%d... ", disk);
  		}
3ce4a7bf6   Dominik Brodowski   fs: add ksys_read...
270
  		ksys_read(in_fd, buf, BLOCK_SIZE);
e7a3e8b2e   Dominik Brodowski   fs: add ksys_writ...
271
  		ksys_write(out_fd, buf, BLOCK_SIZE);
bc58450b0   Stephen Rothwell   init: Remove CONF...
272
  #if !defined(CONFIG_S390)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
273
  		if (!(i % 16)) {
18594e9bc   Nicolas Schichan   init: use pr_cont...
274
  			pr_cont("%c\b", rotator[rotate & 0x3]);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
275
276
277
278
  			rotate++;
  		}
  #endif
  	}
1a6a05a4f   Aaro Koskinen   init/ramdisk: use...
279
280
  	pr_cont("done.
  ");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
281
282
283
284
  
  successful_load:
  	res = 1;
  done:
2ca2a09d6   Dominik Brodowski   fs: add ksys_clos...
285
  	ksys_close(in_fd);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
286
  noclose_input:
2ca2a09d6   Dominik Brodowski   fs: add ksys_clos...
287
  	ksys_close(out_fd);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
288
289
  out:
  	kfree(buf);
0f32ab8cf   Dominik Brodowski   fs: add ksys_unli...
290
  	ksys_unlink("/dev/ram");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
291
292
293
294
295
296
297
  	return res;
  }
  
  int __init rd_load_disk(int n)
  {
  	if (rd_prompt)
  		change_floppy("root floppy disk to be loaded into RAM disk");
bdaf85293   Greg Kroah-Hartman   [PATCH] devfs: Re...
298
299
  	create_dev("/dev/root", ROOT_DEV);
  	create_dev("/dev/ram", MKDEV(RAMDISK_MAJOR, n));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
300
301
  	return rd_load_image("/dev/root");
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
302
  static int exit_code;
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
303
  static int decompress_error;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
304
  static int crd_infd, crd_outfd;
d97b07c54   Yinghai Lu   initramfs: suppor...
305
  static long __init compr_fill(void *buf, unsigned long len)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
306
  {
3ce4a7bf6   Dominik Brodowski   fs: add ksys_read...
307
  	long r = ksys_read(crd_infd, buf, len);
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
308
309
310
311
312
  	if (r < 0)
  		printk(KERN_ERR "RAMDISK: error while reading compressed data");
  	else if (r == 0)
  		printk(KERN_ERR "RAMDISK: EOF while reading compressed data");
  	return r;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
313
  }
d97b07c54   Yinghai Lu   initramfs: suppor...
314
  static long __init compr_flush(void *window, unsigned long outcnt)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
315
  {
e7a3e8b2e   Dominik Brodowski   fs: add ksys_writ...
316
  	long written = ksys_write(crd_outfd, window, outcnt);
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
317
318
319
  	if (written != outcnt) {
  		if (decompress_error == 0)
  			printk(KERN_ERR
d97b07c54   Yinghai Lu   initramfs: suppor...
320
321
  			       "RAMDISK: incomplete write (%ld != %ld)
  ",
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
322
323
324
325
326
  			       written, outcnt);
  		decompress_error = 1;
  		return -1;
  	}
  	return outcnt;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
327
328
329
330
331
332
333
  }
  
  static void __init error(char *x)
  {
  	printk(KERN_ERR "%s
  ", x);
  	exit_code = 1;
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
334
  	decompress_error = 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
335
  }
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
336
  static int __init crd_load(int in_fd, int out_fd, decompress_fn deco)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
337
338
  {
  	int result;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
339
340
  	crd_infd = in_fd;
  	crd_outfd = out_fd;
df3ef3af5   P J P   init/do_mounts_rd...
341
342
343
344
345
346
347
  
  	if (!deco) {
  		pr_emerg("Invalid ramdisk decompression routine.  "
  			 "Select appropriate config option.
  ");
  		panic("Could not decompress initial ramdisk image.");
  	}
30d65dbfe   Alain Knaff   bzip2/lzma: confi...
348
349
  	result = deco(NULL, 0, compr_fill, compr_flush, NULL, NULL, error);
  	if (decompress_error)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
350
  		result = 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
351
352
  	return result;
  }