2018-02-14 15:07:35 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 OTA keys S.A.
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup examples
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief File system usage example application
|
|
|
|
*
|
|
|
|
* @author Vincent Dupont <vincent@otakeys.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
|
|
|
#include "shell.h"
|
|
|
|
#include "board.h" /* MTD_0 is defined in board.h */
|
|
|
|
|
2020-05-20 11:58:46 +02:00
|
|
|
/* Configure MTD device for SD card if none is provided */
|
2020-05-20 11:58:11 +02:00
|
|
|
#if !defined(MTD_0) && MODULE_MTD_SDCARD
|
|
|
|
#include "mtd_sdcard.h"
|
|
|
|
#include "sdcard_spi.h"
|
|
|
|
#include "sdcard_spi_params.h"
|
|
|
|
|
|
|
|
#define SDCARD_SPI_NUM ARRAY_SIZE(sdcard_spi_params)
|
|
|
|
|
2020-07-09 08:49:20 +02:00
|
|
|
/* SD card devices are provided by drivers/sdcard_spi/sdcard_spi.c */
|
2020-05-20 11:58:46 +02:00
|
|
|
extern sdcard_spi_t sdcard_spi_devs[SDCARD_SPI_NUM];
|
2020-05-20 11:58:11 +02:00
|
|
|
|
2020-05-20 11:58:46 +02:00
|
|
|
/* Configure MTD device for the first SD card */
|
|
|
|
static mtd_sdcard_t mtd_sdcard_dev = {
|
|
|
|
.base = {
|
|
|
|
.driver = &mtd_sdcard_driver
|
|
|
|
},
|
|
|
|
.sd_card = &sdcard_spi_devs[0],
|
|
|
|
.params = &sdcard_spi_params[0],
|
|
|
|
};
|
|
|
|
static mtd_dev_t *mtd0 = (mtd_dev_t*)&mtd_sdcard_dev;
|
2020-05-20 11:58:11 +02:00
|
|
|
#define MTD_0 mtd0
|
|
|
|
#endif
|
|
|
|
|
2018-02-14 15:07:35 +01:00
|
|
|
/* Flash mount point */
|
|
|
|
#define FLASH_MOUNT_POINT "/sda"
|
|
|
|
|
|
|
|
/* In this example, MTD_0 is used as mtd interface for littlefs or spiffs */
|
|
|
|
/* littlefs and spiffs basic usage are shown */
|
|
|
|
#ifdef MTD_0
|
|
|
|
/* File system descriptor initialization */
|
|
|
|
#if defined(MODULE_LITTLEFS)
|
|
|
|
/* include file system header for driver */
|
|
|
|
#include "fs/littlefs_fs.h"
|
|
|
|
|
|
|
|
/* file system specific descriptor
|
|
|
|
* for littlefs, some fields can be tweaked to define the size
|
|
|
|
* of the partition, see header documentation.
|
|
|
|
* In this example, default behavior will be used, i.e. the entire
|
|
|
|
* memory will be used (parameters come from mtd) */
|
|
|
|
static littlefs_desc_t fs_desc = {
|
|
|
|
.lock = MUTEX_INIT,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* littlefs file system driver will be used */
|
|
|
|
#define FS_DRIVER littlefs_file_system
|
|
|
|
|
2020-06-30 16:23:20 +02:00
|
|
|
#elif defined(MODULE_LITTLEFS2)
|
|
|
|
/* include file system header for driver */
|
|
|
|
#include "fs/littlefs2_fs.h"
|
|
|
|
|
|
|
|
/* file system specific descriptor
|
|
|
|
* for littlefs2, some fields can be tweaked to define the size
|
|
|
|
* of the partition, see header documentation.
|
|
|
|
* In this example, default behavior will be used, i.e. the entire
|
|
|
|
* memory will be used (parameters come from mtd) */
|
|
|
|
static littlefs2_desc_t fs_desc = {
|
|
|
|
.lock = MUTEX_INIT,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* littlefs file system driver will be used */
|
|
|
|
#define FS_DRIVER littlefs2_file_system
|
|
|
|
|
2018-02-14 15:07:35 +01:00
|
|
|
#elif defined(MODULE_SPIFFS)
|
|
|
|
/* include file system header */
|
|
|
|
#include "fs/spiffs_fs.h"
|
|
|
|
|
|
|
|
/* file system specific descriptor
|
|
|
|
* as for littlefs, some fields can be changed if needed,
|
|
|
|
* this example focus on basic usage, i.e. entire memory used */
|
|
|
|
static spiffs_desc_t fs_desc = {
|
|
|
|
.lock = MUTEX_INIT,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* spiffs driver will be used */
|
|
|
|
#define FS_DRIVER spiffs_file_system
|
2020-05-20 11:58:46 +02:00
|
|
|
|
|
|
|
#elif defined(MODULE_FATFS_VFS)
|
|
|
|
/* include file system header */
|
|
|
|
#include "fs/fatfs.h"
|
|
|
|
|
|
|
|
/* file system specific descriptor
|
|
|
|
* as for littlefs, some fields can be changed if needed,
|
|
|
|
* this example focus on basic usage, i.e. entire memory used */
|
|
|
|
static fatfs_desc_t fs_desc;
|
|
|
|
|
|
|
|
/* provide mtd devices for use within diskio layer of fatfs */
|
|
|
|
mtd_dev_t *fatfs_mtd_devs[FF_VOLUMES];
|
|
|
|
|
|
|
|
/* fatfs driver will be used */
|
|
|
|
#define FS_DRIVER fatfs_file_system
|
2018-02-14 15:07:35 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* this structure defines the vfs mount point:
|
|
|
|
* - fs field is set to the file system driver
|
|
|
|
* - mount_point field is the mount point name
|
|
|
|
* - private_data depends on the underlying file system. For both spiffs and
|
|
|
|
* littlefs, it needs to be a pointer to the file system descriptor */
|
|
|
|
static vfs_mount_t flash_mount = {
|
|
|
|
.fs = &FS_DRIVER,
|
|
|
|
.mount_point = FLASH_MOUNT_POINT,
|
|
|
|
.private_data = &fs_desc,
|
|
|
|
};
|
|
|
|
#endif /* MTD_0 */
|
|
|
|
|
2020-07-17 15:38:27 +02:00
|
|
|
/* Add simple macro to check if an MTD device together with a filesystem is
|
|
|
|
* compiled in */
|
|
|
|
#if defined(MTD_0) && \
|
|
|
|
(defined(MODULE_SPIFFS) || \
|
|
|
|
defined(MODULE_LITTLEFS) || \
|
|
|
|
defined(MODULE_LITTLEFS2) || \
|
|
|
|
defined(MODULE_FATFS_VFS))
|
|
|
|
#define FLASH_AND_FILESYSTEM_PRESENT 1
|
|
|
|
#else
|
|
|
|
#define FLASH_AND_FILESYSTEM_PRESENT 0
|
|
|
|
#endif
|
|
|
|
|
2018-02-14 15:07:35 +01:00
|
|
|
/* constfs example */
|
|
|
|
#include "fs/constfs.h"
|
|
|
|
|
|
|
|
#define HELLO_WORLD_CONTENT "Hello World!\n"
|
|
|
|
#define HELLO_RIOT_CONTENT "Hello RIOT!\n"
|
|
|
|
|
|
|
|
/* this defines two const files in the constfs */
|
|
|
|
static constfs_file_t constfs_files[] = {
|
|
|
|
{
|
|
|
|
.path = "/hello-world",
|
|
|
|
.size = sizeof(HELLO_WORLD_CONTENT),
|
|
|
|
.data = (const uint8_t *)HELLO_WORLD_CONTENT,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.path = "/hello-riot",
|
|
|
|
.size = sizeof(HELLO_RIOT_CONTENT),
|
|
|
|
.data = (const uint8_t *)HELLO_RIOT_CONTENT,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/* this is the constfs specific descriptor */
|
|
|
|
static constfs_t constfs_desc = {
|
2019-07-18 15:18:32 +02:00
|
|
|
.nfiles = ARRAY_SIZE(constfs_files),
|
2018-02-14 15:07:35 +01:00
|
|
|
.files = constfs_files,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* constfs mount point, as for previous example, it needs a file system driver,
|
2018-04-13 11:18:37 +02:00
|
|
|
* a mount point and private_data as a pointer to the constfs descriptor */
|
2018-02-14 15:07:35 +01:00
|
|
|
static vfs_mount_t const_mount = {
|
|
|
|
.fs = &constfs_file_system,
|
|
|
|
.mount_point = "/const",
|
|
|
|
.private_data = &constfs_desc,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Command handlers */
|
|
|
|
static int _mount(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
2020-07-17 15:38:27 +02:00
|
|
|
#if FLASH_AND_FILESYSTEM_PRESENT
|
2018-02-14 15:07:35 +01:00
|
|
|
int res = vfs_mount(&flash_mount);
|
|
|
|
if (res < 0) {
|
|
|
|
printf("Error while mounting %s...try format\n", FLASH_MOUNT_POINT);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("%s successfully mounted\n", FLASH_MOUNT_POINT);
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
puts("No external flash file system selected");
|
|
|
|
return 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _format(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
2020-07-17 15:38:27 +02:00
|
|
|
#if FLASH_AND_FILESYSTEM_PRESENT
|
2018-02-14 15:07:35 +01:00
|
|
|
int res = vfs_format(&flash_mount);
|
|
|
|
if (res < 0) {
|
|
|
|
printf("Error while formatting %s\n", FLASH_MOUNT_POINT);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("%s successfully formatted\n", FLASH_MOUNT_POINT);
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
puts("No external flash file system selected");
|
|
|
|
return 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _umount(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
2020-07-17 15:38:27 +02:00
|
|
|
#if FLASH_AND_FILESYSTEM_PRESENT
|
2018-02-14 15:07:35 +01:00
|
|
|
int res = vfs_umount(&flash_mount);
|
|
|
|
if (res < 0) {
|
|
|
|
printf("Error while unmounting %s\n", FLASH_MOUNT_POINT);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("%s successfully unmounted\n", FLASH_MOUNT_POINT);
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
puts("No external flash file system selected");
|
|
|
|
return 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _cat(int argc, char **argv)
|
|
|
|
{
|
|
|
|
if (argc < 2) {
|
|
|
|
printf("Usage: %s <file>\n", argv[0]);
|
|
|
|
return 1;
|
|
|
|
}
|
2020-07-08 19:38:28 +02:00
|
|
|
/* With newlib or picolibc, low-level syscalls are plugged to RIOT vfs
|
2018-02-14 15:07:35 +01:00
|
|
|
* on native, open/read/write/close/... are plugged to RIOT vfs */
|
2020-07-08 19:38:28 +02:00
|
|
|
#if defined(MODULE_NEWLIB) || defined(MODULE_PICOLIBC)
|
2018-02-14 15:07:35 +01:00
|
|
|
FILE *f = fopen(argv[1], "r");
|
|
|
|
if (f == NULL) {
|
|
|
|
printf("file %s does not exist\n", argv[1]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
char c;
|
|
|
|
while (fread(&c, 1, 1, f) != 0) {
|
|
|
|
putchar(c);
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
#else
|
|
|
|
int fd = open(argv[1], O_RDONLY);
|
|
|
|
if (fd < 0) {
|
|
|
|
printf("file %s does not exist\n", argv[1]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
char c;
|
|
|
|
while (read(fd, &c, 1) != 0) {
|
|
|
|
putchar(c);
|
|
|
|
}
|
|
|
|
close(fd);
|
|
|
|
#endif
|
2020-01-17 20:33:32 +01:00
|
|
|
fflush(stdout);
|
2018-02-14 15:07:35 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _tee(int argc, char **argv)
|
|
|
|
{
|
|
|
|
if (argc != 3) {
|
|
|
|
printf("Usage: %s <file> <str>\n", argv[0]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-07-08 19:38:28 +02:00
|
|
|
#if defined(MODULE_NEWLIB) || defined(MODULE_PICOLIBC)
|
2018-02-14 15:07:35 +01:00
|
|
|
FILE *f = fopen(argv[1], "w+");
|
|
|
|
if (f == NULL) {
|
|
|
|
printf("error while trying to create %s\n", argv[1]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (fwrite(argv[2], 1, strlen(argv[2]), f) != strlen(argv[2])) {
|
|
|
|
puts("Error while writing");
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
#else
|
|
|
|
int fd = open(argv[1], O_RDWR | O_CREAT);
|
|
|
|
if (fd < 0) {
|
|
|
|
printf("error while trying to create %s\n", argv[1]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (write(fd, argv[2], strlen(argv[2])) != (ssize_t)strlen(argv[2])) {
|
|
|
|
puts("Error while writing");
|
|
|
|
}
|
|
|
|
close(fd);
|
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const shell_command_t shell_commands[] = {
|
|
|
|
{ "mount", "mount flash filesystem", _mount },
|
|
|
|
{ "format", "format flash file system", _format },
|
|
|
|
{ "umount", "unmount flash filesystem", _umount },
|
|
|
|
{ "cat", "print the content of a file", _cat },
|
|
|
|
{ "tee", "write a string in a file", _tee },
|
|
|
|
{ NULL, NULL, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
2020-06-30 16:23:20 +02:00
|
|
|
#if defined(MTD_0) && (defined(MODULE_SPIFFS) || defined(MODULE_LITTLEFS) || defined(MODULE_LITTLEFS2))
|
2018-02-14 15:07:35 +01:00
|
|
|
/* spiffs and littlefs need a mtd pointer
|
|
|
|
* by default the whole memory is used */
|
|
|
|
fs_desc.dev = MTD_0;
|
2020-05-20 11:58:46 +02:00
|
|
|
#elif defined(MTD_0) && defined(MODULE_FATFS_VFS)
|
|
|
|
fatfs_mtd_devs[fs_desc.vol_idx] = MTD_0;
|
2018-02-14 15:07:35 +01:00
|
|
|
#endif
|
|
|
|
int res = vfs_mount(&const_mount);
|
|
|
|
if (res < 0) {
|
|
|
|
puts("Error while mounting constfs");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("constfs mounted successfully");
|
|
|
|
}
|
|
|
|
|
|
|
|
char line_buf[SHELL_DEFAULT_BUFSIZE];
|
|
|
|
shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|