UBI文件系统是一种用于裸flash的文件系统管理层,它是专为管理原始闪存设备而设计的,特别适用于嵌入式系统。
-
UBI与MTD、UBIFS的关系:
- MTD(Memory Technology Device):是Linux内核中的一个子系统,用于支持不同类型的闪存设备,如NOR Flash和NAND Flash。MTD提供了一个抽象层,使得文件系统和用户空间程序可以方便地访问底层的闪存硬件。
- UBI:在MTD设备之上增加了一层管理,专门为NAND Flash设计,处理了NAND Flash固有的一些复杂性,如坏块管理和磨损均衡(wear leveling)。UBI将闪存划分为逻辑擦除块,并对它们进行管理。
- UBIFS(UBI File System):是在UBI卷上运行的文件系统,充分利用UBI的特性,提供了高效可靠的文件存储解决方案。
-
UBI的主要功能:
- 坏块管理:UBI能够检测和管理坏块,确保数据写入时不会使用坏块。
- 磨损均衡:UBI通过均匀分布擦写操作,延长闪存的使用寿命。
- 逻辑卷管理:UBI支持在MTD设备上创建多个逻辑卷,每个卷可以独立使用。
-
UBI的应用场景:
- UBI文件系统特别适用于大容量的NAND Flash,并可以提供整个flash空间的磨损平衡和良好的扩展性。它常被用于嵌入式系统中,以提高闪存设备的可靠性和使用寿命。
查看/dev/mtd8相关信息:
# mtdinfo /dev/mtd8
mtd8
Name: database
Type: mlc-nand
Eraseblock size: 2097152 bytes, 2.0 MiB
Amount of eraseblocks: 3584 (7516192768 bytes, 7.0 GiB)
Minimum input/output unit size: 8192 bytes
Sub-page size: 8192 bytes
OOB size: 448 bytes
Character device major/minor: 90:16
Bad blocks are allowed: true
Device is writable: true
/dev/mtd8在flash中的1c0000000 00200000:
# cat /proc/mtd
dev: size erasesize name
mtd0: 00010000 00001000 "at91bootstrap"
mtd1: 00018000 00001000 "u-boot env"
mtd2: 00020000 00001000 "u-boot env redundant"
mtd3: 000e0000 00001000 "u-boot"
mtd4: 000e0000 00001000 "u-boot logo"
mtd5: 00020000 00001000 "device tree"
mtd6: 00600000 00001000 "kernel"
mtd7: 3e000000 00200000 "rootfs"
mtd8: 1c0000000 00200000 "database"
mtd9: 32d98000 001fc000 "rootfs"
mtd10: 190198000 001fc000 "database"
要将ubi系统挂载到pc上,必须要生成镜像文件进行操作:
方法一:使用系统自带的dd命令创建ubi镜像:
dd if=/dev/mtd8 of=/mnt/ubi.img // 用mtd8创建虚拟镜像ubi.img
mkdosfs ubi.img // 使用mkdosfs工具把ubi.img 文件格式化成FAT32文件系统
modprobe g_mass_storage file=/mnt/ubi.img removable=1 // 加载驱动,挂载到pc
modprobe g_mass_storage file=/mnt/ubi.img -r(modprobe g_mass_storage -r) // 卸载驱动
ubi本地挂载:
mount -t vfat -o sync /mnt/ubi.img /mnt/mmcblk0p1 // 挂载到本地
缺陷:从相关信息看出有7G左右,使用dd创建虚拟镜像时间过长,不利于批量生产;
优点:不需要担心nand flash的每个块擦写次数限制;
方法二:使用相关工具(mkfs.ubifs、ubinize)生产镜像文件;
1.下载源码mtd-utils-2.0.0.tar.bz2,版本2.1.1-2.1.2需要添加zstd相关:
a.解压缩(tar -xvf mtd-utils-0.0.0.tar.bz2)
b.配置环境变量export PATH=$PATH:/home/arm/arm-linux-gnueabihf-7.5.0/bin使交叉编译工具链到环境变量中
2.下载源码 lzo-2.10.tar.gz:
a.解压缩(tar -xvf lzo-2.10.tar.gz)
b.配置变量(./configure --host=arm --prefix=/home/arm/armUbi/armlzo CC=arm-linux-gnueabihf-gcc --enable-shared)
c.make
d.make install
3.下载源码e2fsprogs-1.41.14.tar.gz:
@ubuntu:~/arm$ export PATH=$PATH:/home/arm/arm-linux-gnueabihf-7.5.0/bin
@ubuntu:~/arm/armUbi$ tar -xvf lzo-2.10.tar.gz
@ubuntu:~/arm/armUbi/lzo-2.10$ ./configure --host=arm --prefix=/home/arm/armUbi/armlzo CC=arm-linux-gnueabihf-gcc --enable-shared
@ubuntu:~/arm/armUbi/lzo-2.10$ make
@ubuntu:~/arm/armUbi/lzo-2.10$ make install
@ubuntu:~/arm/armUbi$ tar -xvf zlib-1.2.11.tar.gz
@ubuntu:~/arm/armUbi/zlib-1.2.11$ ./configure --host=arm --prefix=/home/arm/armUbi/armZlib CC=arm-linux-gnueabihf-gcc
@ubuntu:~/arm/armUbi/zlib-1.2.11$ make
@ubuntu:~/arm/armUbi/zlib-1.2.11$ make install
@ubuntu:~/arm/armUbi$ tar -zvf e2fsprogs-1.45.6.tar.gz
@ubuntu:~/arm/armUbi/e2fsprogs-1.45.6$ ./configure --host=arm --prefix=/home/arm/armUbi/armUuid CC=arm-linux-gnueabihf-gcc
@ubuntu:~/arm/armUbi/e2fsprogs-1.45.6$ make
@ubuntu:~/arm/armUbi/e2fsprogs-1.45.6$ make install
@ubuntu:~/arm$ tar -xvf mtd-utils-0.0.0.tar.bz2
@ubuntu:~/arm/armUbi/mtd-utils-2.0.0$ ./configure --host=arm --prefix=/home/arm/armUbi/armMtaUtils CC=arm-linux-gnueabihf-gcc --enable-shared --without-crypto
@ubuntu:~/arm/armUbi/mtd-utils-2.0.0$ make
@ubuntu:~/arm/armUbi/mtd-utils-2.0.0$ make install
a.错误,缺少一个或多个依赖项:
configure: WARNING: cannot find uuid library required for mkfs.ubifs
configure: mtd-utils can optionally be built without mkfs.ubifs
configure: WARNING: cannot find LZO library required for mkfs programs
configure: mtd-utils can optionally be built without mkfs.ubifs
configure: mtd-utils can optionally be built without mkfs.jffs2
configure: mtd-utils can optionally be built without LZO support
configure: WARNING: cannot find ZSTD library required for mkfs program
configure: mtd-utils can optionally be built without mkfs.ubifs
configure: mtd-utils can optionally be built without ZSTD support
configure: WARNING: cannot find headers for extended attributes
configure: WARNING: disabling XATTR support
configure: error: missing one or more dependencies
解决(无法解决):
sudo apt-get install libuuid1:i386
sudo apt-get install uuid-dev
下载lzo-2.10.tar.gz e2fsprogs-1.41.14.tar.gz zlib-1.2.11.tar.gz交叉编译,配置环境变量解决:
export ZLIB_CFLAGS=-I/home/arm/armUbi/armZlib/include
export ZLIB_LIBS=-L/home/arm/armUbi/armZlib/lib
export LZO_CFLAGS=-I/home/arm/armUbi/armlzo/include
export LZO_LIBS=-L/home/arm/armUbi/armlzo/lib
export UUID_CFLAGS=-I/home/arm/armUbi/armUuid/include
export UUID_LIBS=-L/home/arm/armUbi/armUuid/lib
错误,找不到库:
ubifs-utils/mkfs.ubifs/mkfs_ubifs-mkfs.ubifs.o: In function `write_super':
/home/arm/armUbi/mtd-utils-2.0.0/ubifs-utils/mkfs.ubifs/mkfs.ubifs.c:2144: undefined reference to `uuid_generate_random'
/home/arm/armUbi/mtd-utils-2.0.0/ubifs-utils/mkfs.ubifs/mkfs.ubifs.c:2148: undefined reference to `uuid_unparse_upper'
collect2: error: ld returned 1 exit status
Makefile:1983: recipe for target 'mkfs.ubifs' failed
make: *** [mkfs.ubifs] Error 1
配置环境变量:
export LDFLAGS="$ZLIB_LIBS $LZO_LIBS $UUID_LIBS -luuid -lz -luuid"
export CFLAGS="-O2 -g $ZLIB_CFLAGS $LZO_CFLAGS $UUID_CFLAGS"
使用相关工具(mkfs.ubifs、ubinize)生产镜像文件:
1.查看相关的命令ls /home/libs/armMtdUtils/sbin
doc_loadbios ftl_check nandwrite ubidetach
docfdisk ftl_format nftl_format ubiformat
flash_erase jffs2dump nftldump ubimkvol
flash_eraseall jffs2reader recv_image ubinfo
flash_lock mkfs.jffs2 rfddump ubinize
flash_otp_dump mkfs.ubifs rfdformat ubirename
flash_otp_info mtd_debug serve_image ubirmvol
flash_otp_lock mtdinfo sumtool ubirsvol
flash_otp_write mtdpart ubiattach ubiupdatevol
flash_unlock nanddump ubiblock
flashcp nandtest ubicrc32
2../mkfs.ubifs -m 8192 -e 2097152 -c 3584 -r /mnt/mmcblk0p1 /home/ubifs.img
-r 指定镜像目录
-m 最小单元(Minimum input/output unit size: 8192 bytes)
-e 2097152 逻辑擦除大小,通过ubi族命令可以看到该值,比如用format,attach先看看该值(Eraseblock size:2097152 bytes, 2.0 MiB)
-c 3584 (Amount of eraseblocks:3584 (7516192768 bytes, 7.0 GiB))
# mtdinfo /dev/mtd8
mtd8
Name: database
Type: mlc-nand
Eraseblock size: 2097152 bytes, 2.0 MiB
Amount of eraseblocks: 3584 (7516192768 bytes, 7.0 GiB)
Minimum input/output unit size: 8192 bytes
Sub-page size: 8192 bytes
OOB size: 448 bytes
Character device major/minor: 90:16
Bad blocks are allowed: true
Device is writable: true
# cd home/libs/armMtdUtils/
# cd sbin/
# ./mkfs.ubifs -m 8192 -e 2097152 -c 3584 -r ^C
# ubiattach /dev/ubi_ctrl -m 8
ubi1: default fastmap pool size: 175
ubi1: default fastmap WL pool size: 87
ubi1: attaching mtd8
ubi1: scanning is finished
ubi1: attached mtd8 (name "database", size 7168 MiB)
ubi1: PEB size: 2097152 bytes (2048 KiB), LEB size: 2080768 bytes
ubi1: min./max. I/O unit sizes: 8192/8192, sub-page size 8192
ubi1: VID header offset: 8192 (aligned 8192), data offset: 16384
ubi1: good PEBs: 3580, bad PEBs: 4, corrupted PEBs: 0
ubi1: user volume: 1, internal volumes: 1, max. volumes count: 128
ubi1: max/mean erase counter: 1/0, WL threshold: 4096, image sequence number: 1409005220
ubi1: available PEBs: 121, total reserved PEBs: 3459, PEBs reserved for bad PEB handling: 76
ubi1: background thread "ubi_bgt1d" started, PID 183
UBI device number 1, total 3580 LEBs (7449149440 bytes, 6.9 GiB), available 121 LEBs (251772928 bytes, 240.1 MiB), LEB size 2080768 bytes (1.9 MiB)
# mount -t ubifs ubi1_0 /mnt/mmcblk0p1
UBIFS (ubi1:0): background thread "ubifs_bgt1_0" started, PID 189
UBIFS (ubi1:0): UBIFS: mounted UBI device 1, volume 0, name "database"
UBIFS (ubi1:0): LEB size: 2080768 bytes (2032 KiB), min./max. I/O unit sizes: 8192 bytes/8192 bytes
UBIFS (ubi1:0): FS size: 7005945856 bytes (6681 MiB, 3367 LEBs), journal size 33292288 bytes (31 MiB, 16 LEBs)
UBIFS (ubi1:0): reserved for root: 4952683 bytes (4836 KiB)
UBIFS (ubi1:0): media format: w5/r0 (latest is w5/r0), UUID EA8F755E-8AB4-4942-A4B0-9D04A44D51E2, small LPT model
# ./mkfs.ubifs -m 8192 -e 2097152 -c 3584 -r /mnt/mmcblk0p1 /home.ubifs.img^C
# ls /home/
shared ubifs.img
# rm /home/ubifs.img
# ./mkfs.ubifs -m 8192 -e 2097152 -c 3584 -r /mnt/mmcblk0p1 /home.ubifs.img
# ls /home -l
total 0
drwxr-xr-x 2 root root 160 Jan 1 2012 shared
# ls
doc_loadbios ftl_check nandwrite ubidetach
docfdisk ftl_format nftl_format ubiformat
flash_erase jffs2dump nftldump ubimkvol
flash_eraseall jffs2reader recv_image ubinfo
flash_lock mkfs.jffs2 rfddump ubinize
flash_otp_dump mkfs.ubifs rfdformat ubirename
flash_otp_info mtd_debug serve_image ubirmvol
flash_otp_lock mtdinfo sumtool ubirsvol
flash_otp_write mtdpart ubiattach ubiupdatevol
flash_unlock nanddump ubiblock
flashcp nandtest ubicrc32
# ls /
Settings lib opt tmp
bin lib32 proc usr
dev linuxrc root var
etc lost+found run
home media sbin
home.ubifs.img mnt sys
# rm /home.ubifs.img
# ./mkfs.ubifs -m 8192 -e 2097152 -c 3584 -r /mnt/mmcblk0p1 /home/ubifs.img
# ls /home/ -l
total 26624
drwxr-xr-x 2 root root 160 Jan 1 2012 shared
-rw-r--r-- 1 root root 27262976 Jan 1 00:11 ubifs.img
# modprobe g_mass_storage file=/home/ubifs.img removable=1 ^C
# mkdosfs /home/ubifs.img
mkfs.fat 4.1 (2017-01-24)
# modprobe g_mass_storage file=/home/ubifs.img removable=1
Mass Storage Function, version: 2009/09/11
LUN: removable file: (no medium)
LUN: removable file: /home/ubifs.img
Number of LUNs=1
g_mass_storage gadget: Mass Storage Gadget, version: 2009/09/11
g_mass_storage gadget: userspace failed to provide iSerialNumber
g_mass_storage gadget: g_mass_storage ready
# g_mass_storage gadget: high-speed config #1: Linux File-Backed Storage
# modprobe g_mass_storage -r
# mount
ubi0:rootfs on / type ubifs (rw,relatime,assert=read-only,ubi=0,vol=0)
devtmpfs on /dev type devtmpfs (rw,relatime,size=91024k,nr_inodes=22756,mode=755)
proc on /proc type proc (rw,relatime)
devpts on /dev/pts type devpts (rw,relatime,gid=5,mode=620,ptmxmode=666)
tmpfs on /dev/shm type tmpfs (rw,relatime,mode=777)
tmpfs on /tmp type tmpfs (rw,relatime)
tmpfs on /run type tmpfs (rw,nosuid,nodev,relatime,mode=755)
sysfs on /sys type sysfs (rw,relatime)
ubi1_0 on /mnt/mmcblk0p1 type ubifs (rw,relatime,assert=read-only,ubi=1,vol=0)
# ls /mnt/
mmcblk0p1 ubi.img
# umount /mnt/mmcblk0p1/
UBIFS (ubi1:0): un-mount UBI device 1
UBIFS (ubi1:0): background thread "ubifs_bgt1_0" stops
# mount -t vfat -o sync /home/ubifs.img /mnt/mmcblk0p1/
# ls /mnt/mmcblk0p1/
System Volume Information sc.txt
方法三:使用自带/dev/mtdblock8挂载到pc:
mkdosfs -I /dev/mtdblock8 // 强制初始化 /dev/mtdblock8
modprobe g_mass_storage file=/dev/mtdblock8 removable=1 // 加载驱动,挂载到pc
modprobe g_mass_storage -r // 卸载驱动
mount -t vfat -o sync /dev/mtdblock8 /mnt/mmcblk0p1/ // 挂载到本地与pc间进行同步
当pc写入数据时, umount /mnt/mmcblk0p1/ 然后重新mount 数据就同步了
当linux写入数据时,PC端插拔下USB线 或者rmmod g_mass_storage 然后重新g_mass_storage.ko来同步数据
#mount -t vfat -o sync /dev/mtdblock8 /mnt/mmcblk0p1/
mount: mounting /dev/mtdblock8 on /mnt/mmcblk0p1/ failed: Device or resource busy
#mount
/dev/mtdblock8 on /mnt/mmcblk0p1 type vfat (rw,sync,relatime,fmask=0022,dmask=0022,codepage=437,iocharset=iso8859-1,shortname=mixed,errors=remount-ro)
mount 时会打印mount: mounting /dev/mtdblock8 on /mnt/mmcblk0p1/ failed: Device or resource busy
装载/dev/mtdblock8/失败:设备或资源忙,已经mount上了
缺陷:nand flash的每个块擦写次数大约十万次,意思就是你一直怼一个nand块 那就很快没用了。然后如果FAT没有对flash写入进行负载均衡,很容易出现坏块。
写入数据太慢,造成板子非常卡顿,无法正常操作;
优点:简单方便,不需要复杂过程;