mirror of
https://github.com/meshtastic/firmware.git
synced 2025-02-10 14:41:26 +00:00
257 lines
6.9 KiB
C++
257 lines
6.9 KiB
C++
/**
|
|
* @file FSCommon.cpp
|
|
* @brief This file contains functions for common filesystem operations such as copying, renaming, listing and deleting files and
|
|
* directories.
|
|
*
|
|
* The functions in this file are used to perform common filesystem operations such as copying, renaming, listing and deleting
|
|
* files and directories. These functions are used in the Meshtastic-device project to manage files and directories on the
|
|
* device's filesystem.
|
|
*
|
|
*/
|
|
#include "FSCommon.h"
|
|
#include "configuration.h"
|
|
|
|
#ifdef HAS_SDCARD
|
|
#include <SD.h>
|
|
#include <SPI.h>
|
|
|
|
#ifdef SDCARD_USE_SPI1
|
|
SPIClass SPI1(HSPI);
|
|
#define SDHandler SPI1
|
|
#else
|
|
#define SDHandler SPI
|
|
#endif
|
|
|
|
#endif // HAS_SDCARD
|
|
|
|
/**
|
|
* @brief Copies a file from one location to another.
|
|
*
|
|
* @param from The path of the source file.
|
|
* @param to The path of the destination file.
|
|
* @return true if the file was successfully copied, false otherwise.
|
|
*/
|
|
bool copyFile(const char *from, const char *to)
|
|
{
|
|
#ifdef FSCom
|
|
unsigned char cbuffer[16];
|
|
|
|
File f1 = FSCom.open(from, FILE_O_READ);
|
|
if (!f1) {
|
|
LOG_ERROR("Failed to open source file %s\n", from);
|
|
return false;
|
|
}
|
|
|
|
File f2 = FSCom.open(to, FILE_O_WRITE);
|
|
if (!f2) {
|
|
LOG_ERROR("Failed to open destination file %s\n", to);
|
|
return false;
|
|
}
|
|
|
|
while (f1.available() > 0) {
|
|
byte i = f1.read(cbuffer, 16);
|
|
f2.write(cbuffer, i);
|
|
}
|
|
|
|
f2.flush();
|
|
f2.close();
|
|
f1.close();
|
|
return true;
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* Renames a file from pathFrom to pathTo.
|
|
*
|
|
* @param pathFrom The original path of the file.
|
|
* @param pathTo The new path of the file.
|
|
*
|
|
* @return True if the file was successfully renamed, false otherwise.
|
|
*/
|
|
bool renameFile(const char *pathFrom, const char *pathTo)
|
|
{
|
|
#ifdef FSCom
|
|
#ifdef ARCH_ESP32
|
|
// rename was fixed for ESP32 IDF LittleFS in April
|
|
return FSCom.rename(pathFrom, pathTo);
|
|
#else
|
|
if (copyFile(pathFrom, pathTo) && FSCom.remove(pathFrom)) {
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* Lists the contents of a directory.
|
|
*
|
|
* @param dirname The name of the directory to list.
|
|
* @param levels The number of levels of subdirectories to list.
|
|
* @param del Whether or not to delete the contents of the directory after listing.
|
|
*/
|
|
void listDir(const char *dirname, uint8_t levels, bool del = false)
|
|
{
|
|
#ifdef FSCom
|
|
#if (defined(ARCH_ESP32) || defined(ARCH_RP2040) || defined(ARCH_PORTDUINO))
|
|
char buffer[255];
|
|
#endif
|
|
File root = FSCom.open(dirname, FILE_O_READ);
|
|
if (!root) {
|
|
return;
|
|
}
|
|
if (!root.isDirectory()) {
|
|
return;
|
|
}
|
|
|
|
File file = root.openNextFile();
|
|
while (file) {
|
|
if (file.isDirectory() && !String(file.name()).endsWith(".")) {
|
|
if (levels) {
|
|
#ifdef ARCH_ESP32
|
|
listDir(file.path(), levels - 1, del);
|
|
if (del) {
|
|
LOG_DEBUG("Removing %s\n", file.path());
|
|
strncpy(buffer, file.path(), sizeof(buffer));
|
|
file.close();
|
|
FSCom.rmdir(buffer);
|
|
} else {
|
|
file.close();
|
|
}
|
|
#elif (defined(ARCH_RP2040) || defined(ARCH_PORTDUINO))
|
|
listDir(file.name(), levels - 1, del);
|
|
if (del) {
|
|
LOG_DEBUG("Removing %s\n", file.name());
|
|
strncpy(buffer, file.name(), sizeof(buffer));
|
|
file.close();
|
|
FSCom.rmdir(buffer);
|
|
} else {
|
|
file.close();
|
|
}
|
|
#else
|
|
listDir(file.name(), levels - 1, del);
|
|
file.close();
|
|
#endif
|
|
}
|
|
} else {
|
|
#ifdef ARCH_ESP32
|
|
if (del) {
|
|
LOG_DEBUG("Deleting %s\n", file.path());
|
|
strncpy(buffer, file.path(), sizeof(buffer));
|
|
file.close();
|
|
FSCom.remove(buffer);
|
|
} else {
|
|
LOG_DEBUG(" %s (%i Bytes)\n", file.path(), file.size());
|
|
file.close();
|
|
}
|
|
#elif (defined(ARCH_RP2040) || defined(ARCH_PORTDUINO))
|
|
if (del) {
|
|
LOG_DEBUG("Deleting %s\n", file.name());
|
|
strncpy(buffer, file.name(), sizeof(buffer));
|
|
file.close();
|
|
FSCom.remove(buffer);
|
|
} else {
|
|
LOG_DEBUG(" %s (%i Bytes)\n", file.name(), file.size());
|
|
file.close();
|
|
}
|
|
#else
|
|
LOG_DEBUG(" %s (%i Bytes)\n", file.name(), file.size());
|
|
file.close();
|
|
#endif
|
|
}
|
|
file = root.openNextFile();
|
|
}
|
|
#ifdef ARCH_ESP32
|
|
if (del) {
|
|
LOG_DEBUG("Removing %s\n", root.path());
|
|
strncpy(buffer, root.path(), sizeof(buffer));
|
|
root.close();
|
|
FSCom.rmdir(buffer);
|
|
} else {
|
|
root.close();
|
|
}
|
|
#elif (defined(ARCH_RP2040) || defined(ARCH_PORTDUINO))
|
|
if (del) {
|
|
LOG_DEBUG("Removing %s\n", root.name());
|
|
strncpy(buffer, root.name(), sizeof(buffer));
|
|
root.close();
|
|
FSCom.rmdir(buffer);
|
|
} else {
|
|
root.close();
|
|
}
|
|
#else
|
|
root.close();
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* @brief Removes a directory and all its contents.
|
|
*
|
|
* This function recursively removes a directory and all its contents, including subdirectories and files.
|
|
*
|
|
* @param dirname The name of the directory to remove.
|
|
*/
|
|
void rmDir(const char *dirname)
|
|
{
|
|
#ifdef FSCom
|
|
#if (defined(ARCH_ESP32) || defined(ARCH_RP2040) || defined(ARCH_PORTDUINO))
|
|
listDir(dirname, 10, true);
|
|
#elif defined(ARCH_NRF52)
|
|
// nRF52 implementation of LittleFS has a recursive delete function
|
|
FSCom.rmdir_r(dirname);
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
void fsInit()
|
|
{
|
|
#ifdef FSCom
|
|
if (!FSBegin()) {
|
|
LOG_ERROR("Filesystem mount Failed.\n");
|
|
// assert(0); This auto-formats the partition, so no need to fail here.
|
|
}
|
|
#ifdef ARCH_ESP32
|
|
LOG_DEBUG("Filesystem files (%d/%d Bytes):\n", FSCom.usedBytes(), FSCom.totalBytes());
|
|
#else
|
|
LOG_DEBUG("Filesystem files:\n");
|
|
#endif
|
|
listDir("/", 10);
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* Initializes the SD card and mounts the file system.
|
|
*/
|
|
void setupSDCard()
|
|
{
|
|
#ifdef HAS_SDCARD
|
|
SDHandler.begin(SPI_SCK, SPI_MISO, SPI_MOSI);
|
|
|
|
if (!SD.begin(SDCARD_CS, SDHandler)) {
|
|
LOG_DEBUG("No SD_MMC card detected\n");
|
|
return;
|
|
}
|
|
uint8_t cardType = SD.cardType();
|
|
if (cardType == CARD_NONE) {
|
|
LOG_DEBUG("No SD_MMC card attached\n");
|
|
return;
|
|
}
|
|
LOG_DEBUG("SD_MMC Card Type: ");
|
|
if (cardType == CARD_MMC) {
|
|
LOG_DEBUG("MMC\n");
|
|
} else if (cardType == CARD_SD) {
|
|
LOG_DEBUG("SDSC\n");
|
|
} else if (cardType == CARD_SDHC) {
|
|
LOG_DEBUG("SDHC\n");
|
|
} else {
|
|
LOG_DEBUG("UNKNOWN\n");
|
|
}
|
|
|
|
uint64_t cardSize = SD.cardSize() / (1024 * 1024);
|
|
LOG_DEBUG("SD Card Size: %lluMB\n", cardSize);
|
|
LOG_DEBUG("Total space: %llu MB\n", SD.totalBytes() / (1024 * 1024));
|
|
LOG_DEBUG("Used space: %llu MB\n", SD.usedBytes() / (1024 * 1024));
|
|
#endif
|
|
} |