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