1.1 什么是romfs
romfs是一个只读文件系统,主要用在 mainly for initial RAM disks of installation disks.使用romfs文件系统可以构造出一个最小的内核,并且很节省内存。相比而言,早期的minix和xiafs(现在已经过时)文件系统如果编译为模块的形式则大小超过20000字节(在x86机器上大小为38502字节),而romfs却小于一页(在linux系统中,一页大小为PAGE_OFFSET,一般为4K),大约4000字节(在x86机器上大小为10479字节)。在相同的条件下,msdos文件系统模块大约30K(并且不支持设备节点和符号链接,在x86机器上大小为12K)。ntfs和nfsroot文件系统模块大约57K(在x86机器上大小为102K)。
注:上面叙述中的数值都是针对i586机器,括号中叙述的数值是在现在的x86机器上的大小,针对2.6.28内核。
1.2 romfs的用途
romfs本设计的主要目标是构造一个最小内核,在内核中只链接romfs文件系统,这样就可以使用romfs在稍后加载其他模块。romfs也可以用来运行一些程序,从而决定你是否需要SCSI设备,或者IDE设备,或者如果你使用的是"initrd"结构的内核,romfs也可以用来在之后加载软驱驱动。romfs的另一个用途是在你使用romfs文件系统的时候,你可以关闭ext2或者minix甚至affs文件系统直到你确信需要的时候再开启。
1.3 romfs的性能
romfs的操作是基于块设备的,它的底层结构非常简单。为了快速访问,每个单元被设计为起始于16字节边界。一个最小的文件为32字节(文件内容为空,并且文件名长度小于16字节)。对于一个非空文件的最大的开销是位于文件内容前面的文件头和其后的16字节的文件名(因为大多数的文件名长度大于3字节并且小于15字节,所以预置文件名长度为16字节)。
1.4 如何使用romfs映像
要使用一个制作好的romfs格式的映像,是将其挂载在其他文件系统的某个节点上。并且还有一个很重要的前提,就是内核要支持romfs文件系统。这一点可以通过配置内核实现,有两个方法:
1.将romfs配置成直接编译进内核,方法为使用make menuconfig命令进入内核配置界面,选择"File systems"并进入,选择“Miscellaneous filesystems”并进入,选择“ROM file system support(ROMFS)”,将其配置成"*"(直接编译进内核)。这样生成的内核就直接包含对romfs文件系统的支持。
2.将romfs配置成模块的形式,步骤和前面一样,只是在最后选择"ROM file system support(ROMFS)"的时候将其配置成"M"(编译为内核模块)。这样编译好的内核并不包含对romfs文件系统的支持,只是生成了romfs.ko模块(fs/romfs/romfs.ko),需要在启动系统后将其加载进内核才能使内核支持romfs文件系统。
有了内核对romfs文件系统的支持,就可以直接挂载romfs格式的映像了,挂载方法为:
niutao@niutao:~/kernel/romfs$ ls
hello.img
niutao@niutao:~/kernel/romfs$ file hello.img
hello.img: romfs filesystem, version 1 208 bytes, named rom 49e05ac0.
niutao@niutao:~/kernel/romfs$ sudo mount -o loop hello.img /mnt
niutao@niutao:~/kernel/romfs$ cd /mnt/
niutao@niutao:/mnt$ ls
hello.c
niutao@niutao:/mnt$
可以看到使用mount命令将hello.img挂载到了/mnt目录下,其内只有一个文件。
卸载一个已经被挂载的romfs格式映像使用umount命令。
1.5 如何制作romfs映像
如果要创建一个romfs文件系统,需要使用genromfs工具。具体用法为:
-f IMAGE 指定输出romfs映像的名字
-d DIRECTORY 指定源目录(将该目录制作成romfs文件系统)
-v 显示详细的创建过程
-V VOLUME 指定卷标
-a ALIGN 指定普通文件的对齐边界(默认为16字节)
-A ALIGN,PATTERN 匹配参数PATTERN的对象对齐在ALIGN边界上
-x PATTERN 不包括匹配PATTERN的对象。
-h 显示帮助文档。
下面一一解释每个参数的含义。
1.5.1 -f参数
该参数指定要生成的romfs映像的名字,是一个必须参数,如果缺少,则无法生成映像。例如:
niutao@niutao:~/kernel/romfs$ ls
hello.c
niutao@niutao:~/kernel/romfs$ genromfs -f hello.img
niutao@niutao:~/kernel/romfs$ ls
hello.c hello.img
可以看到生成了名为hello.img的romfs格式的映像。那么genromfs是将那个目录制作成了hello.img?将hello.img挂载之后,可以看到其内的文件为当前路径下的文件(hello.c和hello.img)。所以在没有指定源目录(要将那个目录制作成romfs映像)的时候,默认为当前目录。这一点也可以从genromfs的源代码中看到:
701 int main(int argc, char *argv[])
702 {
703 int c;
704 char *dir = ".";
705 char *outf = NULL;
706 char *volname = NULL;
注:代码摘自genromfs-0.5.2/genromfs.c。
可以看到其中的dir就是指定的源目录,默认为当前目录。那么如何指定源目录?通过使用-d参数指定。
1.5.2 -d参数
该参数指定源目录,意思是要将那个目录及其下面的文件制作成romfs格式的映像。如果不使用-d参数指定,则默认为当前目录。例如:
niutao@niutao:~/kernel/romfs$ ls
hello
niutao@niutao:~/kernel/romfs$ ls hello/
hello.c
niutao@niutao:~/kernel/romfs$ genromfs -f hello.img -d hello/
niutao@niutao:~/kernel/romfs$ ls
hello hello.img
niutao@niutao:~/kernel/romfs$
1.5.3 -v参数
-v表示显示romfs映像创建的详细过程。例如:
niutao@niutao:~/kernel/romfs$ ls
hello
niutao@niutao:~/kernel/romfs$ genromfs -f hello.img -d hello/ -v
0 rom 49e06097 [0xffffffff, 0xffffffff] 37777777777, sz 0, at 0x0
1 . [0x80e , 0x1e4204 ] 0040755, sz 0, at 0x20
1 .. [0x80e , 0x1e41ea ] 0040755, sz 0, at 0x40 [link to 0x20 ]
1 hello.c [0x80e , 0x1e4266 ] 0100644, sz 71, at 0x60
niutao@niutao:~/kernel/romfs$ ls
hello hello.img
niutao@niutao:~/kernel/romfs$
1.第一列为目录深度(0为顶极目录),各项的含义具体见genromfs.c的shownode函数。
2.第二列为文件或者目录名。
3.第三列为前一个为文件或者目录所在设备的设备号,后一个为文件或者目录的节点号(这两个对应struct stat结构体的st_dev项和st_ino项)。
4.第四列为文件或者目录的属性(对应struct stat结构体的st_mode项)。
5.第五列为文件或者目录的大小,对于目录为0,文件则为文件以字节为单位的大小。
6.第六列为文件或者目录在生成的romfs格式映像文件中的偏移。
7.第七列表示该文件或者目录是一个硬链接,其指向的位置。这里需要注意一个细节:对于顶极目录下的目录,..目录是一个硬链接,除此之外,所有的目录都是目录(也就是说此第七列对于顶极目录下的目录就不存在)。而对于顶极目录下的子目录内的目录,则都是硬链接。这个在往后还会继续从代码的角度分析为什么会是这样(见对genromfs工具的分析)。
1.5.4 -V参数
指定当前要生成的romfs格式映像的卷标,如果没有指定,则默认为字符串"rom "加当前的时间(16进制格式)。例如:
niutao@niutao:~/kernel/romfs$ genromfs -f hello.img -d hello/
niutao@niutao:~/kernel/romfs$ ls
hello hello.img
niutao@niutao:~/kernel/romfs$ file hello.img
hello.img: romfs filesystem, version 1 256 bytes, named rom 49e069d8.
niutao@niutao:~/kernel/romfs$ genromfs -f hello.img -d hello/ -V niutao
niutao@niutao:~/kernel/romfs$ file hello.img
hello.img: romfs filesystem, version 1 256 bytes, named niutao.
niutao@niutao:~/kernel/romfs$
1.5.5 -a参数
指定普通文件的对齐边界(默认为16字节),目的是为了快速访问文件内容。给出的对齐边界值必须大于16并且为16的倍数。这里所说的对齐指的是普通文件的文件内容的对齐,而并非文件头(见内核include/linux/romfs_fs.h中结构体struct romfs_inode )的对齐。假如指定普通文件x的对齐边界为align,其在生成的romfs格式映像中的偏移为start,文件名长度为len,则文件内容在romfs映像中的偏移offset为:
offset = start + 16 + (len / 16) * 16 + len % 16 ? 16 : 0
其计算过程为文件头偏移加上文件头大小(struct romfs_inode),其文件头大小为16字节,再加上文件名以16字节对齐的长度。
例如:
niutao@niutao:~/kernel/romfs$ genromfs -f hello.img -d hello/ -V niutao -a 64 -v
0 niutao [0xffffffff, 0xffffffff] 37777777777, sz 0, at 0x0
1 . [0x80e , 0x1e4204 ] 0040755, sz 0, at 0x20
1 .. [0x80e , 0x1e41ea ] 0040755, sz 0, at 0x40 [link to 0x20 ]
1 hello.c [0x80e , 0x1e4266 ] 0100644, sz 71, at 0x60
niutao@niutao:~/kernel/romfs$
我们使用-a参数指定普通文件的对齐algin = 64。可以看到普通文件hello.c文件头(也叫节点)开始偏移为start = 0x60,文件名长度len = 7,则文件内容起始位置offset为:
offset = start + 16 + (len / 16) * 16 + len % 16 ? 16 : 0
= 0x60 + 16 + (7 / 16) * 16 + 7 % 16 ? 16 : 0
= 0x80
其正好对齐在64边界上。
注意:-a参数只对普通文件起作用,如果要对所有或者部分或者指定的文件起作用,则要使用-A或者-x参数。
1.5.6 -A参数
genromfs -A ALIGN,PATTERN
匹配PATTERN的对象对齐在ALIGN边界上。给出的对齐边界值必须大于16并且为16的倍数。和-a不同的是,-A参数可以指定某个文件或者某些文件对其在ALIGN边界上。例如:
niutao@niutao:~/kernel/romfs$ genromfs -f hello.img -d hello/ -v -A 64,.
0 rom 49e08490 [0xffffffff, 0xffffffff] 37777777777, sz 0, at 0x0
1 . [0x80e , 0x1e4204 ] 0040755, sz 0, at 0x40
1 .. [0x80e , 0x1e41ea ] 0040755, sz 0, at 0x60 [link to 0x40 ]
1 hello.c [0x80e , 0x1e4266 ] 0100644, sz 71, at 0x80
niutao@niutao:~/kernel/romfs$
指定目录"."对齐在64字节边界上。
niutao@niutao:~/kernel/romfs$ genromfs -f hello.img -d hello/ -v -A 64,*.c
0 rom 49e08553 [0xffffffff, 0xffffffff] 37777777777, sz 0, at 0x0
1 . [0x80e , 0x1e4204 ] 0040755, sz 0, at 0x20
1 .. [0x80e , 0x1e41ea ] 0040755, sz 0, at 0x40 [link to 0x20 ]
1 hello.c [0x80e , 0x1e4266 ] 0100644, sz 71, at 0x60
niutao@niutao:~/kernel/romfs$
指定所有的.c文件对其在64字节边界上。
1.5.7 -x参数
不包括匹配PATTERN的对象。也就是说在生成romfs格式映像的时候,不包括指定的源目录下的匹配PATTERN的文件或者目录。例如:
niutao@niutao:~/kernel/romfs$ ls -al hello
total 12
drwxr-xr-x 2 niutao niutao 4096 2009-04-11 20:02 .
drwxr-xr-x 3 niutao niutao 4096 2009-04-11 19:49 ..
-rw-r--r-- 1 niutao niutao 0 2009-04-11 20:02 aa.c
-rw-r--r-- 1 niutao niutao 71 2009-04-11 16:54 hello.c
niutao@niutao:~/kernel/romfs$ genromfs -f hello.img -d hello/ -v -x hello.c
0 rom 49e086ed [0xffffffff, 0xffffffff] 37777777777, sz 0, at 0x0
1 . [0x80e , 0x1e4204 ] 0040755, sz 0, at 0x20
1 .. [0x80e , 0x1e41ea ] 0040755, sz 0, at 0x40 [link to 0x20 ]
1 aa.c [0x80e , 0x1e4276 ] 0100644, sz 0, at 0x60
可以看出,我们使用-x参数将hello.c文件过滤掉了,在生成的romfs格式的映像中,没有hello.c。