2

我想在微控制器(STM32L4xx)上显示一些数据,看起来好像它是通过 USB 安装到主机 PC 的文件系统上的单个文件。我有可以在(模拟)文件中任何请求的偏移量处生成数据的函数。该文件可以只读,没有理由对其进行写入;它可以有一个固定的名称和大小。

这几乎是Chan 的 FatFS和类似库的相反用例:我没有真正的文件系统,我不想访问文件系统,我只想做任何必要的事情来呈现一个模拟的(假的)文件系统(带有单个文件)到 USB 主机。

我认为显示为大容量存储设备(例如 Mbed)的引导加载程序必须执行类似的操作,因为当您将 hex 文件拖放到看起来像 Mbed 的存储设备上时,程序数据(可能)不会写入真实的文件系统; 我想它存储在某个地方,但不是在实际文件系统中,因为它是可见的。

任何人都可以向我指出这些方面的一些示例代码吗?

4

1 回答 1

3

因为只有一个固定长度的文件,所以你不必费心文件系统结构,只需在 PC 上创建一次,并将其视为固定头。该设备根本不需要 FatFS 代码。其他空 FAT 文件系统上的文件将从第一个空闲扇区开始连续存储。

首先,让我们创建一个最小的 FAT 文件系统映像,例如使用 linux 上的mtools包。

mformat -v "EMBEDDED FS" -t 1 -h 1 -s 5 -S 2 -C -i fs.img -c 1 -r 1 -L 1

这将创建一个具有一侧、一个磁道和五个 512 字节长扇区的磁盘映像。前四个扇区是引导扇区、两个 FAT 和根目录,第五个扇区可以容纳一个 512 字节长的文件。

$ mdir -i fs.img 
 Volume in drive : is EMBEDDED FS
 Volume Serial Number is 061F-DA50
Directory for ::/

No files
                                512 bytes free

如果您需要更多空间,请增加扇区数。


更新

我发现我mformat无法创建更大的文件系统(或者我无法找出正确的参数)。还有另一个名为 的实用程序mkdosfs,它对我来说效果更好,例如

/sbin/mkdosfs -f 1 -n "EMBEDDED FS" -r 16 -s 64 -S 512 -v -C big.img 524256

创建一个 512 MB 的文件系统。

$ mdir -i big.img
 Volume in drive : is EMBEDDED FS
 Volume Serial Number is 5457-0DF4
Directory for ::/

No files
                        536 739 840 bytes free

更大的文件系统当然会包含更多的开销,但您会发现其中大部分只是一个运行数字(FAT 中的簇链),也可以动态生成,只要映像包含单个仅限大文件。


现在创建一个具有所需长度的文件并将其复制到图像中

echo 'hello world' > testfile.txt
mcopy -i fs.img testfile.txt ::/

hd并用(*表示重复前一行)转储结果

$ hd fs.img
00000000  eb 3c 90 4d 54 4f 4f 34  30 31 38 00 02 01 01 00  |.<.MTOO4018.....|
00000010  02 10 00 05 00 f0 01 00  05 00 01 00 00 00 00 00  |................|
00000020  00 00 00 00 00 00 29 65  83 db 06 45 4d 42 45 44  |......)e...EMBED|
00000030  44 45 44 20 46 53 46 41  54 31 32 20 20 20 fa 31  |DED FSFAT12   .1|
00000040  c0 8e d8 8e c0 fc b9 00  01 be 00 7c bf 00 80 f3  |...........|....|
00000050  a5 ea 56 00 00 08 b8 01  02 bb 00 7c ba 80 00 b9  |..V........|....|
00000060  01 00 cd 13 72 05 ea 00  7c 00 00 cd 19 00 00 00  |....r...|.......|
00000070  00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00  |................|
*
000001b0  00 00 00 00 00 00 00 00  00 00 00 00 00 00 80 00  |................|
000001c0  01 00 01 00 05 00 00 00  00 00 05 00 00 00 00 00  |................|
000001d0  00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00  |................|
*
000001f0  00 00 00 00 00 00 00 00  00 00 00 00 00 00 55 aa  |..............U.|
00000200  f0 ff ff ff 0f 00 00 00  00 00 00 00 00 00 00 00  |................|
00000210  00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00  |................|
*
00000400  f0 ff ff ff 0f 00 00 00  00 00 00 00 00 00 00 00  |................|
00000410  00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00  |................|
*
00000600  45 4d 42 45 44 44 45 44  20 46 53 08 00 00 33 80  |EMBEDDED FS...3.|
00000610  3e 4f 3e 4f 00 00 33 80  3e 4f 00 00 00 00 00 00  |>O>O..3.>O......|
00000620  54 45 53 54 46 49 4c 45  54 58 54 20 18 00 a7 80  |TESTFILETXT ....|
00000630  3e 4f 3e 4f 00 00 a7 80  3e 4f 02 00 0c 00 00 00  |>O>O....>O......|
00000640  00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00  |................|
*
00000800  68 65 6c 6c 6f 20 77 6f  72 6c 64 0a 00 00 00 00  |hello world.....|
00000810  00 00 00 00 00 00 00 00  00 00 00 00 00 00 00 00  |................|
*
00000a00

正如预期的那样,文件内容从 offset 开始存储,紧随0x0800四个保留扇区之​​后。您现在可以获取映像的前 2048 个字节,并将其放入您的程序中以初始化虚拟磁盘映像。

USB设备代码可以从Applications/USB_Device/MSC_StandaloneSTM32CubeL4中的demo开始,将STM32变成SD读卡器。扔掉 SD 卡相关代码,只提供前四个扇区的固定图像数据,其余部分是您的文件数据。

我自己没有尝试过,因此某些操作系统可能会在这个小磁盘映像上阻塞,在这种情况下,请尝试使用mformat.

于 2019-09-30T14:49:58.660 回答