1虚拟文件系统组件
2============================
3
4:link_to_translation:`en:[English]`
5
6概述
7--------
8
9虚拟文件系统 (VFS) 组件可为一些驱动提供一个统一接口。有了该接口,用户可像操作普通文件一样操作虚拟文件。这类驱动程序可以是 FAT、SPIFFS 等真实文件系统,也可以是有文件类接口的设备驱动程序。
10
11VFS 组件支持 C 库函数(如 fopen 和 fprintf 等)与文件系统 (FS) 驱动程序协同工作。在高层级,每个 FS 驱动程序均与某些路径前缀相关联。当一个 C 库函数需要打开文件时,VFS 组件将搜索与该文件所在文件路径相关联的 FS 驱动程序,并将调用传递给该驱动程序。针对该文件的读取、写入等其他操作的调用也将传递给这个驱动程序。
12
13例如,您可以使用 ``/fat`` 前缀注册 FAT 文件系统驱动,之后即可调用 ``fopen("/fat/file.txt", "w")``。之后,VFS 将调用 FAT 驱动的 ``open`` 函数,并将参数 ``/file.txt`` 和合适的打开模式传递给 ``open`` 函数;后续对返回的 ``FILE*`` 数据流调用 C 库函数也同样会传递给 FAT 驱动。
14
15注册 FS 驱动程序
16---------------------
17
18如需注册 FS 驱动程序,首先要定义一个 :cpp:type:`esp_vfs_t` 结构体实例,并用指向 FS API 的函数指针填充它。
19
20.. highlight:: c
21
22::
23
24    esp_vfs_t myfs = {
25        .flags = ESP_VFS_FLAG_DEFAULT,
26        .write = &myfs_write,
27        .open = &myfs_open,
28        .fstat = &myfs_fstat,
29        .close = &myfs_close,
30        .read = &myfs_read,
31    };
32
33    ESP_ERROR_CHECK(esp_vfs_register("/data", &myfs, NULL));
34
35在上述代码中需要用到 ``read``、 ``write`` 或 ``read_p``、 ``write_p``,具体使用哪组函数由 FS 驱动程序 API 的声明方式决定。
36
37示例 1:声明 API 函数时不带额外的上下文指针参数,即 FS 驱动程序为单例模式,此时使用 ``write`` ::
38
39    ssize_t myfs_write(int fd, const void * data, size_t size);
40
41    // In definition of esp_vfs_t:
42        .flags = ESP_VFS_FLAG_DEFAULT,
43        .write = &myfs_write,
44    // ... other members initialized
45
46    // When registering FS, context pointer (third argument) is NULL:
47    ESP_ERROR_CHECK(esp_vfs_register("/data", &myfs, NULL));
48
49示例 2:声明 API 函数时需要一个额外的上下文指针作为参数,即可支持多个 FS 驱动程序实例,此时使用 ``write_p`` ::
50
51    ssize_t myfs_write(myfs_t* fs, int fd, const void * data, size_t size);
52
53    // In definition of esp_vfs_t:
54        .flags = ESP_VFS_FLAG_CONTEXT_PTR,
55        .write_p = &myfs_write,
56    // ... other members initialized
57
58    // When registering FS, pass the FS context pointer into the third argument
59    // (hypothetical myfs_mount function is used for illustrative purposes)
60    myfs_t* myfs_inst1 = myfs_mount(partition1->offset, partition1->size);
61    ESP_ERROR_CHECK(esp_vfs_register("/data1", &myfs, myfs_inst1));
62
63    // Can register another instance:
64    myfs_t* myfs_inst2 = myfs_mount(partition2->offset, partition2->size);
65    ESP_ERROR_CHECK(esp_vfs_register("/data2", &myfs, myfs_inst2));
66
67同步输入/输出多路复用
68^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
69
70如需通过 :cpp:func:`select` 使用同步输入/输出多路复用,首先需要把 :cpp:func:`start_select` 和 :cpp:func:`end_select` 注册到 VFS,如下所示:
71
72.. highlight:: c
73
74::
75
76    // In definition of esp_vfs_t:
77        .start_select = &uart_start_select,
78        .end_select = &uart_end_select,
79    // ... other members initialized
80
81调用 :cpp:func:`start_select` 设置环境,用以检测某一 VFS 文件描述符的读取/写入/错误条件。调用 :cpp:func:`end_select` 终止、析构或释放 :cpp:func:`start_select` 设置的资源。请在 :component_file:`vfs/vfs_uart.c` 中查看 UART 外设参考实现、:cpp:func:`esp_vfs_dev_uart_register`、:cpp:func:`uart_start_select` 和 :cpp:func:`uart_end_select` 函数。
82
83请参考以下示例,查看如何使用 VFS 文件描述符调用 :cpp:func:`select`:
84
85- :example:`peripherals/uart/uart_select`
86- :example:`system/select`
87
88如果 :cpp:func:`select` 用于套接字文件描述符,您可以启用 :envvar:`CONFIG_LWIP_USE_ONLY_LWIP_SELECT` 选项来减少代码量,提高性能。
89
90路径
91-----
92
93已注册的 FS 驱动程序均有一个路径前缀与之关联,此路径前缀即为分区的挂载点。
94
95如果挂载点中嵌套了其他挂载点,则在打开文件时使用具有最长匹配路径前缀的挂载点。例如,假设以下文件系统已在 VFS 中注册:
96
97- 在 /data 下注册 FS 驱动程序 1
98- 在 /data/static 下注册 FS 驱动程序 2
99
100那么:
101
102- 打开 ``/data/log.txt`` 会调用驱动程序 FS 1;
103- 打开 ``/data/static/index.html`` 需调用 FS 驱动程序 2;
104- 即便 FS 驱动程序 2 中没有 ``/index.html``,也不会在 FS 驱动程序 1 中查找 ``/static/index.html``。
105
106挂载点名称必须以路径分隔符 (``/``) 开头,且分隔符后至少包含一个字符。但在以下情况中,VFS 同样支持空的挂载点名称:1. 应用程序需要提供一个”最后方案“下使用的文件系统;2. 应用程序需要同时覆盖 VFS 功能。如果没有与路径匹配的前缀,就会使用到这种文件系统。
107
108VFS 不会对路径中的点 (``.``) 进行特殊处理,也不会将 ``..`` 视为对父目录的引用。在上述示例中,使用 ``/data/static/../log.txt`` 路径不会调用 FS 驱动程序 1 打开 ``/log.txt``。特定的 FS 驱动程序(如 FATFS)可能以不同的方式处理文件名中的点。
109
110执行打开文件操作时,FS 驱动程序仅得到文件的相对路径(挂载点前缀已经被去除):
111
1121. 以 ``/data`` 为路径前缀注册 ``myfs`` 驱动;
1132. 应用程序调用 ``fopen("/data/config.json", ...)``;
1143. VFS 调用 ``myfs_open("/config.json", ...)``;
1154. ``myfs`` 驱动打开 ``/config.json`` 文件。
116
117VFS 对文件路径长度没有限制,但文件系统路径前缀受 ``ESP_VFS_PATH_MAX`` 限制,即路径前缀上限为 ``ESP_VFS_PATH_MAX``。各个文件系统驱动则可能会对自己的文件名长度设置一些限制。
118
119
120文件描述符
121----------------
122
123文件描述符是一组很小的正整数,从 ``0`` 到 ``FD_SETSIZE - 1``,``FD_SETSIZE`` 在 newlib ``sys/types.h`` 中定义。最大文件描述符由 ``CONFIG_LWIP_MAX_SOCKETS`` 定义,且为套接字保留。VFS 中包含一个名为 ``s_fd_table`` 的查找表,用于将全局文件描述符映射至 ``s_vfs`` 数组中注册的 VFS 驱动索引。
124
125
126标准 IO 流 (stdin, stdout, stderr)
127-------------------------------------------
128
129如果 menuconfig 中 ``UART for console output`` 选项没有设置为 ``None``,则 ``stdin``、 ``stdout`` 和 ``stderr`` 将默认从 UART 读取或写入。UART0 或 UART1 可用作标准 IO。默认情况下,UART0 使用 115200 波特率,TX 管脚为 GPIO1,RX 管脚为 GPIO3。您可以在 menuconfig 中更改上述参数。
130
131对 ``stdout`` 或 ``stderr`` 执行写入操作将会向 UART 发送 FIFO 发送字符,对 ``stdin`` 执行读取操作则会从 UART 接收 FIFO 中取出字符。
132
133默认情况下,VFS 使用简单的函数对 UART 进行读写操作。在所有数据放进 UART FIFO 之前,写操作将处于 busy-wait 状态,读操处于非阻塞状态,仅返回 FIFO 中已有数据。由于读操作为非阻塞,高层级 C 库函数调用(如 ``fscanf("%d\n", &var);``)可能获取不到所需结果。
134
135如果应用程序使用 UART 驱动,则可以调用 ``esp_vfs_dev_uart_use_driver`` 函数来指导 VFS 使用驱动中断、读写阻塞功能等。您也可以调用 ``esp_vfs_dev_uart_use_nonblocking`` 来恢复非阻塞函数。
136
137VFS 还为输入和输出提供换行符转换功能(可选)。多数应用程序在程序内部发送或接收以 LF (''\n'') 结尾的行,但不同的终端程序可能需要不同的换行符,比如 CR 或 CRLF。应用程序可以通过 menuconfig 或者调用 ``esp_vfs_dev_uart_port_set_rx_line_endings`` 和 ``esp_vfs_dev_uart_port_set_tx_line_endings`` 为输入输出配置换行符。
138
139
140标准流和 FreeRTOS 任务
141^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
142
143``stdin``、``stdout`` 和 ``stderr`` 的 ``FILE`` 对象在所有 FreeRTOS 任务之间共享,指向这些对象的指针分别存储在每个任务的 ``struct _reent`` 中。
144
145预处理器把如下代码:
146
147.. highlight:: c
148
149::
150
151    fprintf(stderr, "42\n");
152
153解释为:
154
155.. highlight:: c
156
157::
158
159    fprintf(__getreent()->_stderr, "42\n");
160
161其中 ``__getreent()`` 函数将为每个任务返回一个指向 ``struct _reent`` 的指针。每个任务的 TCB 均拥有一个 ``struct _reent`` 结构体,任务初始化后,``struct _reent`` 结构体中的 ``_stdin``、``_stdout`` 和 ``_stderr`` 将会被赋予 ``_GLOBAL_REENT`` 中 ``_stdin``、 ``_stdout`` 和 ``_stderr`` 的值,``_GLOBAL_REENT`` 即为 FreeRTOS 启动之前所用结构体。
162
163这样设计带来的结果是:
164
165- 允许重定向给定任务的 ``stdin``、 ``stdout`` 和 ``stderr``,而不影响其他任务,例如通过 ``stdin = fopen("/dev/uart/1", "r")``;
166- 但使用 ``fclose`` 关闭默认 ``stdin``、 ``stdout`` 或 ``stderr`` 将同时关闭相应的 ``FILE`` 流对象,因此会影响其他任务;
167- 如需更改新任务的默认 ``stdin``、 ``stdout`` 和 ``stderr`` 流,请在创建新任务之前修改 ``_GLOBAL_REENT->_stdin`` (``_stdout``、``_stderr``)。
168
169
170应用示例
171-------------------
172
173`指南`_ (未完成)
174
175.. _指南: ../../template.html
176
177API 参考
178-------------
179
180.. include-build-file:: inc/esp_vfs.inc
181
182.. include-build-file:: inc/esp_vfs_dev.inc
183