update: 调整mobile.scell函数的返回值为table,一次性返回足够多的信息, 并可以容纳更多信息
[LuatOS.git] / components / fal / README.md
blobbdafd8e3bfa7cb3a82814e0ad9ed0fc0d28dc842
1 # FAL:Flash 抽象层
3 ## 1、FAL介绍
5 FAL (Flash Abstraction Layer) Flash 抽象层,是对 Flash 及基于 Flash 的分区进行管理、操作的抽象层,对上层统一了 Flash 及 分区操作的 API (框架图如下所示),并具有以下特性:
7 - 支持静态可配置的分区表,并可关联多个 Flash 设备;
8 - 分区表支持 **自动装载** 。避免在多固件项目,分区表被多次定义的问题;
9 - 代码精简,对操作系统 **无依赖** ,可运行于裸机平台,比如对资源有一定要求的 Bootloader;
10 - 统一的操作接口。保证了文件系统、OTA、NVM(例如:[EasyFlash](https://github.com/armink-rtt-pkgs/EasyFlash)) 等对 Flash 有一定依赖的组件,底层 Flash 驱动的可重用性;
11 - 自带基于 Finsh/MSH 的测试命令,可以通过 Shell 按字节寻址的方式操作(读写擦) Flash 或分区,方便开发者进行调试、测试;
13 ![FAL framework](docs/figures/fal_framework.png)
15 ### 1.1、打开 FAL
17 使用 fal package 需要在 RT-Thread 的包管理器中选择它,具体路径如下:
19 ```
20 RT-Thread online packages
21     system packages --->
22         --- fal: Flash Abstraction Layer implement. Manage flash device and partition.
23         [*]   Enable debug log output
24         [*]   FAL partition table config has defined on 'fal_cfg.h'
25         (onchip) The flash device which saving partition table
26         (65536) The patition table end address relative to flash device offset.
27         [ ]   FAL uses SFUD drivers
28         (norflash0) The name of the device used by FAL (NEW)
29                 version (latest)  --->
30 ```
32 每个功能的配置说明如下:
34 - 开启调试日志输出(默认开启);
35 - 分区表是否在 `fal_cfg.h` 中定义(默认开启)。如果关闭此选项,fal 将会自动去指定 Flash 的指定位置去检索并装载分区表,具体配置详见下面两个选项;
36   - 存放分区表的 Flash 设备;
37   - 分区表的 **结束地址** 位于 Flash 设备上的偏移。fal 将从此地址开始往回进行检索分区表,直接读取到 Flash 顶部。如果不确定分区表具体位置,这里也可以配置为 Flash 的结束地址,fal 将会检索整个 Flash,检索时间可能会增加。
38 - 启用 FAL 针对 SFUD 的移植文件(默认关闭);
39   - 应输入调用 `rt_sfud_flash_probe` 函数时传入的 FLASH 设备名称(也可以通过 list_device 命令查看 Block Device 的名字获取)。该名称与分区表中的 Flash 名称对应,只有正确设置设备名字,才能完成对 FLASH 的读写操作。
41 然后让 RT-Thread 的包管理器自动更新,或者使用 `pkgs --update` 命令更新包到 BSP 中。
43 ### 1.2、FAL 目录
45 | 名称    | 说明       |
46 | ------- | ---------- |
47 | inc     | 头文件目录 |
48 | src     | 源代码目录 |
49 | samples | 例程目录   |
51 ### 1.3、FAL API
53 FAL 相关的 API 如图所示,[点击此处查看 API 参数详解](docs/fal_api.md)。
55 ![FAL API](docs/figures/fal-api.png)
57 ### 1.4、许可证
59 fal package 遵循 LGPLv2.1 许可,详见 `LICENSE` 文件。
61 ### 1.5、依赖
63 对 RT-Thread 无依赖,也可用于裸机。
65 > 测试命令功能需要依赖 RT-Thread Finsh/MSH
67 ## 2、使用 FAL
69 使用 FAL 的基本步骤如下所示:
71 1. 打开 FAL:从 Env 中打开 fal 软件包并下载到工程。
72 2. FAL 移植:定义 flash 设备、定义 flash 设备表、定义 flash 分区表。以下主要对步骤 2 展开讲解。
73 3. 调用 fal_init() 初始化该库:移植完成后,可在应用层调用,如在 main 函数中调用。
75 ![fal 移植](docs/figures/fal-port.png)
77 ### 2.1、定义 flash 设备
79 在定义 Flash 设备表前,需要先定义 Flash 设备。可以是片内 flash,  也可以是片外基于 SFUD 的 spi flash:
81 - 定义片内 flash 设备可以参考 [`fal_flash_sfud_port.c`](https://github.com/RT-Thread-packages/fal/blob/master/samples/porting/fal_flash_sfud_port.c) 。
82 - 定义片外 spi flash 设备可以参考 [`fal_flash_stm32f2_port.c`](https://github.com/RT-Thread-packages/fal/blob/master/samples/porting/fal_flash_stm32f2_port.c) 。
84 定义具体的 Flash 设备对象,用户需要根据自己的 Flash 情况分别实现 `init`、 `read`、 `write`、 `erase` 这些操作函数:
86 - `static int init(void)`:**可选** 的初始化操作。
87 - `static int read(long offset, uint8_t *buf, size_t size)`:读取操作。
89 | 参数   | 描述                      |
90 | ------ | ------------------------- |
91 | offset | 读取数据的 Flash 偏移地址 |
92 | buf    | 存放待读取数据的缓冲区    |
93 | size   | 待读取数据的大小          |
94 | return | 返回实际读取的数据大小    |
96 - `static int write(long offset, const uint8_t *buf, size_t size)` :写入操作。
98 | 参数   | 描述                      |
99 | ------ | ------------------------- |
100 | offset | 写入数据的 Flash 偏移地址 |
101 | buf    | 存放待写入数据的缓冲区    |
102 | size   | 待写入数据的大小          |
103 | return | 返回实际写入的数据大小    |
105 - `static int erase(long offset, size_t size)` :擦除操作。
107 | 参数   | 描述                      |
108 | ------ | ------------------------- |
109 | offset | 擦除区域的 Flash 偏移地址 |
110 | size   | 擦除区域的大小            |
111 | return | 返回实际擦除的区域大小    |
113 用户需要根据自己的 Flash 情况分别实现这些操作函数。在文件最底部定义了具体的 Flash 设备对象 ,如下示例定义了 stm32f2 片上 flash:stm32f2_onchip_flash
115 ```c
116 const struct fal_flash_dev stm32f2_onchip_flash =
118     .name       = "stm32_onchip",
119     .addr       = 0x08000000,
120     .len        = 1024*1024,
121     .blk_size   = 128*1024,
122     .ops        = {init, read, write, erase},
123     .write_gran = 8
127 - `"stm32_onchip"` : Flash 设备的名字。
128 - `0x08000000`: 对 Flash 操作的起始地址。
129 - `1024*1024`:Flash 的总大小(1MB)。
130 - `128*1024`:Flash 块/扇区大小(因为 STM32F2 各块大小不均匀,所以擦除粒度为最大块的大小:128K)。
131 - `{init, read, write, erase}` :Flash 的操作函数。 如果没有 init 初始化过程,第一个操作函数位置可以置空。
132 - `8` : 设置写粒度,单位 bit, 0 表示未生效(默认值为 0 ),该成员是 fal 版本大于 0.4.0 的新增成员。各个 flash 写入粒度不尽相同,可通过该成员进行设置,以下列举几种常见 Flash 写粒度:
133   - nor flash: 1 bit
134   - stm32f4:  8 bit
135   - stm32f1:  32 bit
136   - stm32l4:  64 bit
138 ### 2.2、定义 flash 设备表
140 Flash 设备表定义在 `fal_cfg.h` 头文件中,定义分区表前需 **新建 `fal_cfg.h` 文件** ,请将该文件统一放在对应 BSP 或工程目录的 port 文件夹下,并将该头文件路径加入到工程。fal_cfg.h 可以参考 [示例文件 fal/samples/porting/fal_cfg.h](https://github.com/RT-Thread-packages/fal/blob/master/samples/porting/fal_cfg.h) 完成。
142 设备表示例:
144 ```c
145 /* ===================== Flash device Configuration ========================= */
146 extern const struct fal_flash_dev stm32f2_onchip_flash;
147 extern struct fal_flash_dev nor_flash0;
149 /* flash device table */
150 #define FAL_FLASH_DEV_TABLE                                          \
151 {                                                                    \
152     &stm32f2_onchip_flash,                                           \
153     &nor_flash0,                                                     \
157 Flash 设备表中,有两个 Flash 对象,一个为 STM32F2 的片内 Flash ,一个为片外的 Nor Flash。
159 ### 2.3、定义 flash 分区表
161 分区表也定义在 `fal_cfg.h` 头文件中。Flash 分区基于 Flash 设备,每个 Flash 设备又可以有 N 个分区,这些分区的集合就是分区表。在配置分区表前,务必保证已定义好 **Flash 设备** 及 **设备表**。fal_cfg.h 可以参考 [示例文件 fal/samples/porting/fal_cfg.h](https://github.com/RT-Thread-packages/fal/blob/master/samples/porting/fal_cfg.h) 完成。
163 分区表示例:
165 ```c
166 #define NOR_FLASH_DEV_NAME             "norflash0"
167 /* ====================== Partition Configuration ========================== */
168 #ifdef FAL_PART_HAS_TABLE_CFG
169 /* partition table */
170 #define FAL_PART_TABLE                                                               \
171 {                                                                                    \
172     {FAL_PART_MAGIC_WORD,        "bl",     "stm32_onchip",         0,   64*1024, 0}, \
173     {FAL_PART_MAGIC_WORD,       "app",     "stm32_onchip",   64*1024,  704*1024, 0}, \
174     {FAL_PART_MAGIC_WORD, "easyflash", NOR_FLASH_DEV_NAME,         0, 1024*1024, 0}, \
175     {FAL_PART_MAGIC_WORD,  "download", NOR_FLASH_DEV_NAME, 1024*1024, 1024*1024, 0}, \
177 #endif /* FAL_PART_HAS_TABLE_CFG */
180 上面这个分区表详细描述信息如下:
182 | 分区名      | Flash 设备名   | 偏移地址  | 大小  | 说明               |
183 | ----------- | -------------- | --------- | ----- | ------------------ |
184 | "bl"        | "stm32_onchip" | 0         | 64KB  | 引导程序           |
185 | "app"       | "stm32_onchip" | 64*1024   | 704KB | 应用程序           |
186 | "easyflash" | "norflash0"    | 0         | 1MB   | EasyFlash 参数存储 |
187 | "download"  | "norflash0"    | 1024*1024 | 1MB   | OTA 下载区         |
189 用户需要修改的分区参数包括:分区名称、关联的 Flash 设备名、偏移地址(相对 Flash 设备内部)、大小,需要注意以下几点:
191 - 分区名保证 **不能重复**;
192 - 关联的 Flash 设备 **务必已经在 Flash 设备表中定义好** ,并且 **名称一致** ,否则会出现无法找到 Flash 设备的错误;
193 - 分区的起始地址和大小 **不能超过 Flash 设备的地址范围** ,否则会导致包初始化错误;
195 > 注意:每个分区定义时,除了填写上面介绍的参数属性外,需在前面增加 `FAL_PART_MAGIC_WORD` 属性,末尾增加 `0` (目前用于保留功能)
197 ## 3、Finsh/MSH 测试命令
199 fal 提供了丰富的测试命令,项目只要在 RT-Thread 上开启 Finsh/MSH 功能即可。在做一些基于 Flash 的应用开发、调试时,这些命令会非常实用。它可以准确的写入或者读取指定位置的原始 Flash 数据,快速的验证 Flash 驱动的完整性,甚至可以对 Flash 进行性能测试。
201 具体功能如下:输入 fal 可以看到完整的命令列表
204 msh />fal
205 Usage:
206 fal probe [dev_name|part_name]   - probe flash device or partition by given name
207 fal read addr size               - read 'size' bytes starting at 'addr'
208 fal write addr data1 ... dataN   - write some bytes 'data' starting at 'addr'
209 fal erase addr size              - erase 'size' bytes starting at 'addr'
210 fal bench <blk_size>             - benchmark test with per block size
212 msh />
215 ### 3.1、指定待操作的 Flash 设备或 Flash 分区
217 当第一次使用 fal 命令时,直接输入 `fal probe`  将会显示分区表信息。可以指定待操作的对象为分区表里的某个分区,或者某个 Flash 设备。
219 分区或者 Flash 被成功选中后,还将会显示它的一些属性情况。大致效果如下:
222 msh />fal probe    
223 No flash device or partition was probed.
224 Usage: fal probe [dev_name|part_name]   - probe flash device or partition by given name.
225 [I/FAL] ==================== FAL partition table ====================
226 [I/FAL] | name      | flash_dev    |   offset   |    length  |
227 [I/FAL] -------------------------------------------------------------
228 [I/FAL] | bl        | stm32_onchip | 0x00000000 | 0x00010000 |
229 [I/FAL] | app       | stm32_onchip | 0x00010000 | 0x000b0000 |
230 [I/FAL] | ef        | norflash0    | 0x00000000 | 0x00100000 |
231 [I/FAL] | download  | norflash0    | 0x00100000 | 0x00100000 |
232 [I/FAL] =============================================================
233 msh />
234 msh />fal probe download
235 Probed a flash partition | download | flash_dev: norflash0 | offset: 1048576 | len: 1048576 |.
236 msh />
239 ### 3.2、擦除数据
241 先输入 `fal erase` ,后面跟着待擦除数据的起始地址以及长度。以下命令为:从 0 地址(相对 Flash 或分区)开始擦除 4096 字节数据
243 > 注意:根据 Flash 特性,擦除动作将按扇区对齐进行处理。所以,如果擦除操作地址或长度未按照 Flash 的扇区对齐,将会擦除掉与其关联的整个扇区数据。
246 msh />fal erase 0 4096
247 Erase data success. Start from 0x00000000, size is 4096.
248 msh />
251 ### 3.3、写入数据
253 先输入 `fal write` ,后面跟着 N 个待写入的数据,并以空格隔开。以下命令为:从地址 8 的位置依次开始写入 1、2、3、4 、 5 这 5 个字节数据
256 msh />fal write 8 1 2 3 4 5
257 Write data success. Start from 0x00000008, size is 5.
258 Write data: 1 2 3 4 5 .
259 msh />
262 ### 3.4、读取数据
264 先输入 `fal read` ,后面跟着待读取数据的起始地址以及长度。以下命令为:从 0 地址开始读取 64 字节数据
267 msh />fal read 0 64
268 Read data success. Start from 0x00000000, size is 64. The data is:
269 Offset (h) 00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F
270 [00000000] FF FF FF FF FF FF FF FF 01 02 03 04 05 FF FF FF 
271 [00000010] FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF 
272 [00000020] FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF 
273 [00000030] FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF 
275 msh />
278 ### 3.5、性能测试
280 性能测试将会测试 Flash 的擦除、写入及读取速度,同时将会测试写入及读取数据的准确性,保证整个 Flash 或整个分区的 写入与读取 数据的一致性。
282 先输入 `fal bench` ,后面跟着待测试 Flash 的扇区大小(请查看对应的 Flash 手册,SPI Nor Flash 一般为 4096)。由于性能测试将会让整个 Flash 或者整个分区的数据丢失,所以命令最后必须跟 `yes` 。
285 msh />fal bench 4096 yes
286 Erasing 1048576 bytes data, waiting...
287 Erase benchmark success, total time: 2.674S.
288 Writing 1048576 bytes data, waiting...
289 Write benchmark success, total time: 7.107S.
290 Reading 1048576 bytes data, waiting...
291 Read benchmark success, total time: 2.716S.
292 msh />
295 ## 4、常见应用
297 - [基于 FAL 分区的 fatfs 文件系统例程](https://github.com/RT-Thread/IoT_Board/tree/master/examples/15_component_fs_flash)
298 - [基于 FAL 分区的 littlefs 文件系统应用笔记](https://www.rt-thread.org/document/site/application-note/components/dfs/an0027-littlefs/)
299 - [基于 FAL 分区的 EasyFlash 移植说明](https://github.com/armink-rtt-pkgs/EasyFlash/tree/master/ports)
301 ## 5、常见问题
303 **1、使用 FAL 时,无法找到 `fal_cfg.h` 头文件**
305 `fal_cfg.h` 为 fal 软件包的配置文件,需要用户手动新建,并定义相关的分区表信息。请将该文件统一放在 BSP 的 port 文件夹下或工程目录的 port 文件夹下(若没有则新建 port 文件夹),并将该头文件路径加入到工程,详见 "`2.2、定义 flash 设备表`" 小节。
307 ## 6、联系方式
309 * 维护:[armink](https://github.com/armink)
310 * 主页:https://github.com/RT-Thread-packages/fal