openwrt spi flash 分区适配过程

这里基于 openwrt mt7620a 平台来跟踪,主要是想理清 dts 里的分区描述是如何一步步转化成内核分区行为。

先来看看 dts 中关于分区的描述:

	palmbus@10000000 {
spi@b00 {
status = "okay"; m25p80@0 {
#address-cells = <1>;
#size-cells = <1>;
compatible = "w25q128";
reg = <0 0>;
linux,modalias = "m25p80", "w25q128";
spi-max-frequency = <10000000>; partition@0 {
label = "u-boot";
reg = <0x0 0x30000>;
read-only;
}; partition@30000 {
label = "u-boot-env";
reg = <0x30000 0x10000>;
read-only;
}; factory: partition@40000 {
label = "factory";
reg = <0x40000 0x10000>;
read-only;
}; partition@50000 {
label = "firmware";
reg = <0x50000 0xfb0000>;
};
};
};

dts 描述的是一个树状结构。spi 控制器挂在 platform 总线上,spi flash (w25q128) 挂在 spi 总线上。 探测到 spi flash 的流程如下:

  1. plat_of_setup() 遍历 palmbus 上的设备,并为每一个动态创建 platform_device,添加到系统总线上 device_add()。对于 spi 这里会创建一个名为 "ralink,rt2880-spi" 的 platfrom_device 并添加到系统中。
  2. drivers/spi/spi-rt2880.c 中会注册 spi 的 platform_driver,与上一步的 platfrom_device match 上了之后,触发调用 rt2880_spi_probe() 。
  3. spi_register_master() 向系统注册 spi 主控制器,并最后调用 of_register_spi_devices(master) 看看 dts 中在 spi 总线上有哪些设备。
  4. 对 dts 中描述的每一个 spi 总线下的设备,为其创建相应的 spi_device,同时根据 dts 中描述的 reg, spi-cpha, spi-cpol, spi-cs-high, spi-3wire, spi-max-frequency 等属性来配置该 spi 设备。对于这里,创建了一个名为 “m25p80” 的 spi_device。
  5. drivers/mtd/device/m25p80.c 中有名为 “m25p80" 的 spi_driver,于是 match 上了。触发执行 m25p_probe()。
  6. m25p_probe() 中读到了这颗 spi flash 的 id 后,确认了一些基本信息(如页大小、块大小), 最后调用 mtd_device_parse_register() 开始真正的分区。

分区解析器

part_parser 用来按照某种规则将分区信息解析出来。这些规则可以有很多,内核里调用 register_mtd_parser() 即可注册一个新的解析器。

drivers/mtd/mtdpart.c 中维护了一个链表 part_parsers,解析器按注册顺序添加到这个链表里。

parse_mtd_partitions() 中,如果未指定解析器的话,则默认只允许用 cmdlinepart, ofpart 两种解析器。对于我们这里,实际上起作用的是 ofpart。

static struct mtd_part_parser ofpart_parser = {
.owner = THIS_MODULE,
.parse_fn = parse_ofpart_partitions,
.name = "ofpart",
};

parse_ofpart_partitions() 遍历 dts 中 spi flash 设备下的分区描述信息,取出其中的 reg, label, name, read-only, lock 等信息以填充一个 struct mtd_partition 结构体。上面 dts 里描述了 4 个分区, 就有一个大小为 4 的 struct mtd_partition 数组,最后由 add_mtd_partitions() 添加为各 mtd 分区。

分区的情况可以待系统启动后在 /proc/mtd 文件中查看到。

# cat /proc/mtd
dev: size erasesize name
mtd0: 00030000 00010000 "u-boot"
mtd1: 00010000 00010000 "u-boot-env"
mtd2: 00010000 00010000 "factory"
mtd3: 00fb0000 00010000 "firmware"
mtd4: 00ea9283 00010000 "rootfs"
mtd5: 00b30000 00010000 "rootfs_data"

根文件系统的解析

上面 /proc/mtd 的内容中相比 dts 中的描述多了两个分区 rootfs, rootfs_data。这两个分区是何时添加的呢?

看看添加 mtd 分区的函数:

int add_mtd_partitions(struct mtd_info *master,
const struct mtd_partition *parts,
int nbparts)
{
struct mtd_part *slave;
uint64_t cur_offset = 0;
int i; printk(KERN_NOTICE "Creating %d MTD partitions on \"%s\":\n", nbparts, master->name); for (i = 0; i < nbparts; i++) {
slave = allocate_partition(master, parts + i, i, cur_offset);
if (IS_ERR(slave))
return PTR_ERR(slave); mutex_lock(&mtd_partitions_mutex);
list_add(&slave->list, &mtd_partitions);
mutex_unlock(&mtd_partitions_mutex); add_mtd_device(&slave->mtd);
mtd_partition_split(master, slave); cur_offset = slave->offset + slave->mtd.size;
} return 0;
}

最后调用了 mtd_partition_split()。

static void mtd_partition_split(struct mtd_info *master, struct mtd_part *part)
{
static int rootfs_found = 0; if (rootfs_found)
return; if (!strcmp(part->mtd.name, "rootfs")) {
rootfs_found = 1; if (config_enabled(CONFIG_MTD_ROOTFS_SPLIT))
split_rootfs_data(master, part);
} if (!strcmp(part->mtd.name, SPLIT_FIRMWARE_NAME) &&
config_enabled(CONFIG_MTD_SPLIT_FIRMWARE))
split_firmware(master, part); arch_split_mtd_part(master, part->mtd.name, part->offset,
part->mtd.size);
}

如果:

  1. rootfs 还没有被找到
  2. 当前分区名是 "firmware"
  3. 内核配置时开启了 CONFIG_MTD_SPLIT_FIRMWARE

则调用 split_firmware() 来解析。在该函数中做了以下几件事:

  1. 找 type 为 MTD_PARSER_TYPE_FIRMWARE 的分区解析器来分析。
  2. "uimage-fw" 解析器读出 firmware 分区的头部,成功找到一个 uImage。
  3. 跃过 uImage,紧接着成功找到 squashfs 的头信息,于是找到了格式为 squashfs 的 rootfs。
  4. 解析器在找到一个分区后,会调用 __mtd_add_partition() 将此分区添加到系统中。
  5. __mtd_add_partition() 最后又调用 mtd_partition_split(),因为此时 rootfs 已经找到,所以会调用 split_rootfs_data() 找 rootfs_data 分区。
  6. rootfs 为 squashfs 分区,该格式的文件系统只读,且头信息里有标记分区大小。所以很容易就可以找到 rootfs_data 的起始位置。

openwrt spi flash 分区适配过程的更多相关文章

  1. (三)修改内核大小,适配目标板Nand flash分区配置

    一. 修改内核大小 1. 在你的配置文件下uboot/include/config/xxx.h 里面有一个宏定义 #define MTDPARTS_DEFAULT "mtdparts=jz2 ...

  2. nand flash,nor flash,spi flash,片上RAM,片外RAM

    Flash有掉电数据保存的特点,RAM掉电则数据丢失,但是RAM的速度更高,擦写次数理论上没有限制,而Flash则不行. Nand Flash相比其余的几种flash优势在于可擦写次数多,擦写速度快, ...

  3. OpenRisc-32-ORPSoC烧写外部spi flash

    引言 经过前面的分析和介绍,我们对ORPSoC的启动过程(http://blog.csdn.net/rill_zhen/article/details/8855743)和 ORpSoC的debug子系 ...

  4. RTT下spi flash+elm fat文件系统移植小记

    背景: MCU:STM32F207 SPI flash: Winbond W25Q16BV OS: RTT V1.1.1 bsp: STM32F20x 1 将spi_core.c,spi_dev.c及 ...

  5. RTThread DFS文件系统使用: 基于使用SFUD驱动的SPI FLASH之上的ELM FATFS文件系统

    参考博文: 博文很长,但是实际要操作的步骤没几下. http://m.elecfans.com/article/730878.html  为了防止几年后文章链接找不到,我把文章复制过来了 /***** ...

  6. 16经典的SPI Flash的扇区擦除flash_se功能

    一设计功能 对SPI_flash进行扇区擦除,分为写指令和扇区擦除两个时序部分. 二设计知识点 我简单理解flash,第一它是掉电不丢失数据的存储器,第二它每次写入新数据前首先得擦除数据,分为扇区擦除 ...

  7. 【原创】All in One i.MXRT1050/RT1020 SPI Flash Algorithm for J-Flash

    2020年,这个给大家一种很漫长的恍惚感的一年,终于是过去了.这一年我们很多新的人生第一次就这么被发生了,第一次居家办公这么长时间(很多人肥膘都长了不少,我却瘦了2斤,不知是工作太积极了还是被家里小怪 ...

  8. 双系统Ubuntu分区扩容过程记录

    本人电脑上安装了Win10 + Ubuntu 12.04双系统.前段时间因为在Ubuntu上做项目要安装一个比较大的软件,导致Ubuntu根分区的空间不够了.于是,从硬盘又分出来一部分空间,分给Ubu ...

  9. Nand Flash,Nor Flash,CFI Flash,SPI Flash 之间的关系

    前言:    在嵌入式开发中,如uboot的移植,kernel的移植都需要对Flash 有基本的了解.下面细说一下标题中的中Flash中的关系 一,Flash的内存存储结构    flash按照内部存 ...

随机推荐

  1. 《常见问题集》Maven

    1.Maven Eclipse插件要不要安装? [解决方法] 打开你的Eclipse,如果已经有Maven了就不用装插件了. 方法一:没有的话或者下载最新的Eclipse(maven插件,eclips ...

  2. iOS-文件断点续传

    * 移动客户端在和服务器交互的时候,上传和下载使用十分广泛. * 在我们下载文件的时候,我们在点击暂停的时候可以暂停下载,点击下载的时候可以继续下载,这个功能如何实现? * 下载进度条如何显示? 先大 ...

  3. [转] Makefile 基础 (1) —— Makefile 介绍

    该篇文章为转载,是对原作者系列文章的总汇加上标注. 支持原创,请移步陈浩大神博客:(最原始版本) http://blog.csdn.net/haoel/article/details/2886 我转自 ...

  4. socket编程-微软小兵

    socket两端建立连接,不断开的连接的情况下做数据交互,客户端发送数据和服务端返回数据.直到客户端要求断开,则关闭连接. 代码目录结构:

  5. 剑指Offer 二进制中一的个数

    链接:https://www.nowcoder.com/questionTerminal/8ee967e43c2c4ec193b040ea7fbb10b8 来源:牛客网 public class So ...

  6. java中过滤查询文件

    需求,过滤出C盘demo目录下 所有以.java的文件不区分大小写 通过实现FileFilter接口 定义过滤规则,然后将这个实现类对象传给ListFiles方法作为参数即可. 使用递归方法实现 pa ...

  7. js中加“var”和不加“var”的区别,看完觉得这么多年js白学了

    Javascript声明变量的时候,虽然用var关键字声明和不用关键字声明,很多时候运行并没有问题,但是这两种方式还是有区别的.可以正常运行的代码并不代表是合适的代码. var num = 1: 是在 ...

  8. 天梯赛 - L2-002 链表去重

    GG思密达,第二个测试点的三分怎么也拿不上,我还是比较熟悉指针,用指针来写~,写完去上概率论 题目链接:https://www.patest.cn/contests/gplt/L2-002 #incl ...

  9. LibieOJ 6170 字母树 (Trie)

    题目链接 字母树 (以每个点为根遍历,插入到trie中,统计答案即可)——SamZhang #include <bits/stdc++.h> using namespace std; #d ...

  10. 快速构造FFT/NTT

    @(学习笔记)[FFT, NTT] 问题概述 给出两个次数为\(n\)的多项式\(A\)和\(B\), 要求在\(O(n \log n)\)内求出它们的卷积, 即对于结果\(C\)的每一项, 都有\[ ...