分区只是内核的概念,就是说A~B地址放内核,C~D地址放文件系统,(也就是规定哪一个地址区间放内核或者文件系统)等等。linux
1:在内核MTD中能够定义分区A~B,C~D。。。。。。并予以绝对的地址赋值给每一个分区。咱们能够来看看在内核中是怎样来对MTD进行分区的:arch\arm\plat-s3c24xx\common-smdk.c
app
static struct mtd_partition smdk_default_nand_part[] = {
[0] = {
.name = "Boot",
.size = SZ_16K,
.offset = 0,
},
[1] = {
.name = "S3C2410 flash partition 1",
.offset = 0,
.size = SZ_2M,
},
[2] = {
.name = "S3C2410 flash partition 2",
.offset = SZ_4M,
.size = SZ_4M,
},
[3] = {
.name = "S3C2410 flash partition 3",
.offset = SZ_8M,
.size = SZ_2M,
},
[4] = {
.name = "S3C2410 flash partition 4",
.offset = SZ_1M * 10,
.size = SZ_4M,
},ide
......
};测试
通常咱们只须要分3-4个区,第一个为boot区,一个为boot参数区(传递给内核的参数),一个为内核区,一个为文件系统区。spa
而对于bootloader中只要能将内核下载到A~B区的A地址开始处就能够,C~D区的C起始地址下载文件系统。。。这些起始地址在MTD的分区信息中能找到。因此bootloader对分区的概念不重要,只要它能把内核烧到A位置,把文件系统烧到C位置。
因此,在bootloader对Flash进行操做时,哪块区域放什么是之内核为主。命令行
而为了方便操做,bootloader相似也引入分区的概念,如,可使用“nand write 0x3000000 kernel 200000”命令将uImage烧到kernel分区,而没必要写那么长:nand write 3000000 A 200000,也就是用分区名来代替具体的地址。server
这要对bootloader对内核从新分区:这须要从新设置一下bootloader环境参数,就能够同步更新内核分区信息htm
如:ip
setenv bootargs 'noinitrd console=ttySAC0 root=/dev/mtdblock3 rootfstype=jffs2内存
mtdparts=nand_flash:128k(u-boot)ro,64k(u-boot envs),3m(kernel),30m(root.jffs2),30m(root.yaffs)'
内核配置时选上Device Drivers ---> Memory Technology Device (MTD) support ---> Command line partition table parsing
在设置了mtdparts变量以后,就能够在nand read/write/erase命令中直接使用分区的名字而没必要指定分区的偏移位置.而这须要内核MTD最好没有规划分区。
若是你是经过uboot的内核命令行给MTD层传递MTD分区信息,这种状况下,内核读取到的分区信息始终和u-boot中的保持一致(推荐的作法)
若是你是把分区信息写在内核源代码MTD里定义好的方法,那最好保证它和u-boot中的保持一致,即同步修改uboot及内核的相关部分。
内核经过bootargs找到文件系统,bootargs中的mtdblockx即表明分区,block1,2,3表明哪一个分区。
事实上,bootargs中的"root=/dev/mtdblockx"只是告诉内核,root fs从第x个(x=0,1,2...)MTD分区挂载,mtdblock0对应第一个分区,mtdblock1对应第二个分区,以此类推.
3:分区方法
1) MTD层的分区
2) 经过U-boot传递给内核的命令行中的mtdparts=...
3) 其余可让内核知道分区信息的任何办法,(内核默认的命令参数)
下面说到mtdparts,及它的用法:
mtdparts
mtdparts=fc000000.nor_flash:1920k(linux),128k(fdt),20M(ramdisk),4M(jffs2),38272k(user),256k(env),384k(uboot)
要想这个参数起做用,内核中的mtd驱动必需要支持,即内核配置时须要选上Device Drivers ---> Memory Technology Device (MTD) support ---> Command line partition table parsing
mtdparts的格式以下:
mtdparts=<mtddef>[;<mtddef]
<mtddef> := <mtd-id>:<partdef>[,<partdef>]
<partdef> := <size>[@offset][<name>][ro]
<mtd-id> := unique id used in mapping driver/device
<size> := standard linux memsize OR "-" to denote all remaining space
<name> := (NAME)
所以你在使用的时候须要按照下面的格式来设置:
mtdparts=mtd-id:<size1>@<offset1>(<name1>),<size2>@<offset2>(<name2>)
这里面有几个必需要注意的:
a. mtd-id 必需要跟你当前平台的flash的mtd-id一致,否则整个mtdparts会失效 怎样获取到当前平台的flash的mtd-id?
在bootargs参数列表中能够指定当前flash的mtd-id,如指定 mtdids:nand0=gen_nand.1,前面的nand0则表示第一个flash
b. size在设置的时候能够为实际的size(xxM,xxk,xx),也能够为'-'这表示剩余的全部空间。
相关信息能够查看drivers/mtd/cmdlinepart.c中的注释找到相关描述。
U-boot的环境变量值得注意的有两个: bootcmd 和bootargs。
引用:
u bootcmd
前面有说过bootcmd是自动启动时默认执行的一些命令,所以你能够在当前环境中定义各类不一样配置,不一样环境的参数设置,而后设置bootcmd为你常用的那种参数。
u bootargs
bootargs是环境变量中的重中之重,甚至能够说整个环境变量都是围绕着bootargs来设置的。bootargs的种类很是很是的多,咱们日常只是使用了几种而已,感兴趣的能够看看这篇文章说的很全:http://www.linuxidc.com/Linux/2011-03/33599p4.htm。bootargs很是的灵活,内核和文件系统的不一样搭配就会有不一样的设置方法,甚至你也能够不设置bootargs,而直接将其写到内核中去(在配置内核的选项中能够进行这样的设置),正是这些缘由致使了bootargs使用上的困难。
下面介绍一下bootargs经常使用参数,bootargs的种类很是的多,并且随着kernel的发展会出现一些新的参数,使得设置会更加灵活多样。
A. root
用来指定rootfs的位置, 常见的状况有:
root=/dev/ram rw
root=/dev/ram0 rw
请注意上面的这两种设置状况是通用的,我作过测试甚至root=/dev/ram1 rw和root=/dev/ram2 rw也是能够的,网上有人说在某些状况下是不通用的,即必须设置成ram或者ram0,可是目前尚未遇到,还须要进一步确认,遇到不行的时候能够逐一尝试。
root=/dev/mtdx rw
root=/dev/mtdblockx rw
root=/dev/mtdblock/x rw
root=31:0x
上面的这几个在必定状况下是通用的,固然这要看你当前的系统是否支持,不过mtd是字符设备,而mtdblock是块设备,有时候你的挨个的试到底当前的系统支持上面那种状况下,不过root=/dev/mtdblockx rw比较通用。此外,若是直接指定设备名能够的话,那么使用此设备的设备号也是能够的。
root=/dev/nfs
在文件系统为基于nfs的文件系统的时候使用。固然指定root=/dev/nfs以后,还须要指定nfsroot=serverip:nfs_dir,即指明文件系统存在那个主机的那个目录下面。
B. rootfstype
这个选项须要跟root一块儿配合使用,通常若是根文件系统是ext2的话,有没有这个选项是无所谓的,可是若是是jffs2,squashfs等文件系统的话,就须要rootfstype指明文件系统的类型,否则会没法挂载根分区.
C. console
console=tty<n> 使用虚拟串口终端设备 <n>.
console=ttyS<n>[,options] 使用特定的串口<n>,options能够是这样的形式bbbbpnx,这里bbbb是指串口的波特率,p是奇偶校验位,n是指的bits。
console=ttySAC<n>[,options] 同上面。
看你当前的环境,有时用ttyS<n>,有时用ttySAC<n>,网上有人说,这是跟内核的版本有关,2.4用ttyS<n>,2.6用ttySAC<n>,但实际状况是官方文档中也是使用ttyS<n>,因此应该是跟内核版本没有关联的。能够查看Documentation/serial-console.txt找到相关描述。
D. mem
mem=xxM 指定内存的大小,不是必须的
E. ramdisk_size
ramdisk=xxxxx 不推荐
ramdisk_size=xxxxx 推荐
上面这两个均可以告诉ramdisk 驱动,建立的ramdisk的size,默认状况下是4m(s390默认8M),你能够查看Documentation/ramdisk.txt找到相关的描述,不过ramdisk=xxxxx在新版的内核都已经没有提了,不推荐使用。
F. initrd, noinitrd
当你没有使用ramdisk启动系统的时候,你须要使用noinitrd这个参数,可是若是使用了的话,就须要指定initrd=r_addr,size, r_addr表示initrd在内存中的位置,size表示initrd的大小。
G. init
init指定的是内核启起来后,进入系统中运行的第一个脚本,通常init=/linuxrc, 或者init=/etc/preinit,preinit的内容通常是建立console,null设备节点,运行init程序,挂载一些文件系统等等操做。请注意,不少初学者觉得init=/linuxrc是固定写法,其实否则,/linuxrc指的是/目录下面的linuxrc脚本,通常是一个链接罢了。
H. ip
指定系统启动以后网卡的ip地址,若是你使用基于nfs的文件系统,那么必需要有这个参数,其余的状况下就看你本身的喜爱了。设置ip有两种方法:
ip = ip addr
ip=ip addr:server ip addr:gateway:netmask::which netcard:off
这两种方法能够用,不过很明显第二种要详细不少,请注意第二种中which netcard 是指开发板上的网卡,而不是主机上的网卡。
说完常见的几种bootargs,那么咱们来讨论日常我常用的几种组合:
1). 假设文件系统是ramdisk,且直接就在内存中,bootargs的设置应该以下:
setenv bootargs ‘initrd=0x32000000,0xa00000 root=/dev/ram0 console=ttySAC0 mem=64M init=/linuxrc’
2). 假设文件系统是ramdisk,且在flash中,bootargs的设置应该以下:
setenv bootargs ‘mem=32M console=ttyS0,115200 root=/dev/ram rw init=/linuxrc’
注意这种状况下你应该要在bootm命令中指定ramdisk在flash中的地址,如bootm kernel_addr ramdisk_addr (fdt_addr)
3). 假设文件系统是jffs2类型的,且在flash中,bootargs的设置应该以下
setenv bootargs ‘mem=32M console=ttyS0,115200 noinitrd root=/dev/mtdblock2 rw rootfstype=jffs2 init=/linuxrc’
4). 假设文件系统是基于nfs的,bootargs的设置应该以下
setenv bootargs ‘noinitrd mem=64M console=ttySAC0 root=/dev/nfs nfsroot=192.168.0.3:/nfs ip=192.168.0.5:192.168.0.3:192.168.0.3:255.255.255.0::eth0:off’
或者
setenv bootargs ‘noinitrd mem=64M console=ttySAC0 root=/dev/nfs nfsroot=192.168.0.3:/nfs ip=192.168.0.5’