Luc 93312ff8b5
Idf 5.1.4/Arduino 3.0.4 porting for esp32 (#1046)
* Update WebSocket library
* Update SSDP library
* Update TFT_eSPI library
* Update EspLuaEngine library
* Update SDFat library
* Change to pioarduino
* Make ESP3DMessageFIFO and ESP3DMessage  more thread safe
* Fix sanity checks for BT
* Add some C6 support
* Refactor ethernet code
* Split Ethernet Sta / WiFi sta ESP Commands  and settings
* Simplify wait and wdtFeed code
* Set C3 with 4MB by default in platformio.ini
* Apply Disable brown out only on ESP32 to avoid crash e.g:ESP32S3
* Add missing entries in platformio.ini
2024-09-05 16:27:47 +08:00

403 lines
13 KiB
C++

/**
* Copyright (c) 2011-2022 Bill Greiman
* This file is part of the SdFat library for SD memory cards.
*
* MIT License
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included
* in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*/
#ifndef FsVolume_h
#define FsVolume_h
/**
* \file
* \brief FsVolume include file.
*/
#include "../ExFatLib/ExFatLib.h"
#include "../FatLib/FatLib.h"
#include "FsNew.h"
class FsFile;
/**
* \class FsVolume
* \brief FsVolume class.
*/
class FsVolume {
public:
FsVolume() = default;
~FsVolume() { end(); }
/** Get file's user settable attributes.
* \param[in] path path to file.
* \return user settable file attributes for success else -1.
*/
int attrib(const char* path) {
return m_fVol ? m_fVol->attrib(path) : m_xVol ? m_xVol->attrib(path) : -1;
}
/** Set file's user settable attributes.
* \param[in] path path to file.
* \param[in] bits bit-wise or of selected attributes: FS_ATTRIB_READ_ONLY,
* FS_ATTRIB_HIDDEN, FS_ATTRIB_SYSTEM, FS_ATTRIB_ARCHIVE.
*
* \return true for success or false for failure.
*/
bool attrib(const char* path, uint8_t bits) {
return m_fVol ? m_fVol->attrib(path, bits)
: m_xVol ? m_xVol->attrib(path, bits)
: false;
}
/**
* Initialize an FatVolume object.
* \param[in] blockDev Device block driver.
* \param[in] setCwv Set current working volume if true.
* \param[in] part partition to initialize.
* \param[in] volStart Start sector of volume if part is zero.
* \return true for success or false for failure.
*/
bool begin(FsBlockDevice* blockDev, bool setCwv = true, uint8_t part = 1,
uint32_t volStart = 0);
#ifndef DOXYGEN_SHOULD_SKIP_THIS
uint32_t __attribute__((error("use sectorsPerCluster()"))) blocksPerCluster();
#endif // DOXYGEN_SHOULD_SKIP_THIS
/** \return the number of bytes in a cluster. */
uint32_t bytesPerCluster() const {
return m_fVol ? m_fVol->bytesPerCluster()
: m_xVol ? m_xVol->bytesPerCluster()
: 0;
}
/**
* Set volume working directory to root.
* \return true for success or false for failure.
*/
bool chdir() {
return m_fVol ? m_fVol->chdir() : m_xVol ? m_xVol->chdir() : false;
}
/**
* Set volume working directory.
* \param[in] path Path for volume working directory.
* \return true for success or false for failure.
*/
bool chdir(const char* path) {
return m_fVol ? m_fVol->chdir(path) : m_xVol ? m_xVol->chdir(path) : false;
}
/** Change global working volume to this volume. */
void chvol() { m_cwv = this; }
/** \return The total number of clusters in the volume. */
uint32_t clusterCount() const {
return m_fVol ? m_fVol->clusterCount()
: m_xVol ? m_xVol->clusterCount()
: 0;
}
/** \return The logical sector number for the start of file data. */
uint32_t dataStartSector() const {
return m_fVol ? m_fVol->dataStartSector()
: m_xVol ? m_xVol->clusterHeapStartSector()
: 0;
}
/** End access to volume
* \return pointer to sector size buffer for format.
*/
uint8_t* end() {
m_fVol = nullptr;
m_xVol = nullptr;
static_assert(sizeof(m_volMem) >= 512, "m_volMem too small");
return reinterpret_cast<uint8_t*>(m_volMem);
}
/** Test for the existence of a file in a directory
*
* \param[in] path Path of the file to be tested for.
*
* \return true if the file exists else false.
*/
bool exists(const char* path) {
return m_fVol ? m_fVol->exists(path)
: m_xVol ? m_xVol->exists(path)
: false;
}
/** \return The logical sector number for the start of the first FAT. */
uint32_t fatStartSector() const {
return m_fVol ? m_fVol->fatStartSector()
: m_xVol ? m_xVol->fatStartSector()
: 0;
}
/** \return Partition type, FAT_TYPE_EXFAT, FAT_TYPE_FAT32,
* FAT_TYPE_FAT16, or zero for error.
*/
uint8_t fatType() const {
return m_fVol ? m_fVol->fatType() : m_xVol ? m_xVol->fatType() : 0;
}
/** \return free cluster count or -1 if an error occurs. */
int32_t freeClusterCount() const {
return m_fVol ? m_fVol->freeClusterCount()
: m_xVol ? m_xVol->freeClusterCount()
: -1;
}
/**
* Check for device busy.
*
* \return true if busy else false.
*/
bool isBusy() {
return m_fVol ? m_fVol->isBusy() : m_xVol ? m_xVol->isBusy() : false;
}
/** List directory contents.
*
* \param[in] pr Print object.
*
* \return true for success or false for failure.
*/
bool ls(print_t* pr) {
return m_fVol ? m_fVol->ls(pr) : m_xVol ? m_xVol->ls(pr) : false;
}
/** List directory contents.
*
* \param[in] pr Print object.
* \param[in] flags The inclusive OR of
*
* LS_DATE - %Print file modification date
*
* LS_SIZE - %Print file size.
*
* LS_R - Recursive list of subdirectories.
*
* \return true for success or false for failure.
*/
bool ls(print_t* pr, uint8_t flags) {
return m_fVol ? m_fVol->ls(pr, flags)
: m_xVol ? m_xVol->ls(pr, flags)
: false;
}
/** List the directory contents of a directory.
*
* \param[in] pr Print stream for list.
*
* \param[in] path directory to list.
*
* \param[in] flags The inclusive OR of
*
* LS_DATE - %Print file modification date
*
* LS_SIZE - %Print file size.
*
* LS_R - Recursive list of subdirectories.
*
* \return true for success or false for failure.
*/
bool ls(print_t* pr, const char* path, uint8_t flags);
/** Make a subdirectory in the volume root directory.
*
* \param[in] path A path with a valid 8.3 DOS name for the subdirectory.
*
* \param[in] pFlag Create missing parent directories if true.
*
* \return true for success or false for failure.
*/
bool mkdir(const char* path, bool pFlag = true) {
return m_fVol ? m_fVol->mkdir(path, pFlag)
: m_xVol ? m_xVol->mkdir(path, pFlag)
: false;
}
/** open a file
*
* \param[in] path location of file to be opened.
* \param[in] oflag open flags.
* \return a FsBaseFile object.
*/
FsFile open(const char* path, oflag_t oflag = O_RDONLY);
/** Remove a file from the volume root directory.
*
* \param[in] path A path with a valid 8.3 DOS name for the file.
*
* \return true for success or false for failure.
*/
bool remove(const char* path) {
return m_fVol ? m_fVol->remove(path)
: m_xVol ? m_xVol->remove(path)
: false;
}
/** Rename a file or subdirectory.
*
* \param[in] oldPath Path name to the file or subdirectory to be renamed.
*
* \param[in] newPath New path name of the file or subdirectory.
*
* The \a newPath object must not exist before the rename call.
*
* The file to be renamed must not be open. The directory entry may be
* moved and file system corruption could occur if the file is accessed by
* a file object that was opened before the rename() call.
*
* \return true for success or false for failure.
*/
bool rename(const char* oldPath, const char* newPath) {
return m_fVol ? m_fVol->rename(oldPath, newPath)
: m_xVol ? m_xVol->rename(oldPath, newPath)
: false;
}
/** Remove a subdirectory from the volume's root directory.
*
* \param[in] path A path with a valid 8.3 DOS name for the subdirectory.
*
* The subdirectory file will be removed only if it is empty.
*
* \return true for success or false for failure.
*/
bool rmdir(const char* path) {
return m_fVol ? m_fVol->rmdir(path) : m_xVol ? m_xVol->rmdir(path) : false;
}
/** \return The volume's cluster size in sectors. */
uint32_t sectorsPerCluster() const {
return m_fVol ? m_fVol->sectorsPerCluster()
: m_xVol ? m_xVol->sectorsPerCluster()
: 0;
}
#if ENABLE_ARDUINO_SERIAL
/** List directory contents.
* \return true for success or false for failure.
*/
bool ls() { return ls(&Serial); }
/** List directory contents.
*
* \param[in] flags The inclusive OR of
*
* LS_DATE - %Print file modification date
*
* LS_SIZE - %Print file size.
*
* LS_R - Recursive list of subdirectories.
*
* \return true for success or false for failure.
*/
bool ls(uint8_t flags) { return ls(&Serial, flags); }
/** List the directory contents of a directory to Serial.
*
* \param[in] path directory to list.
*
* \param[in] flags The inclusive OR of
*
* LS_DATE - %Print file modification date
*
* LS_SIZE - %Print file size.
*
* LS_R - Recursive list of subdirectories.
*
* \return true for success or false for failure.
*
* \return true for success or false for failure.
*/
bool ls(const char* path, uint8_t flags = 0) {
return ls(&Serial, path, flags);
}
#endif // ENABLE_ARDUINO_SERIAL
#if ENABLE_ARDUINO_STRING
/**
* Set volume working directory.
* \param[in] path Path for volume working directory.
* \return true for success or false for failure.
*/
bool chdir(const String& path) { return chdir(path.c_str()); }
/** Test for the existence of a file in a directory
*
* \param[in] path Path of the file to be tested for.
*
* \return true if the file exists else false.
*/
bool exists(const String& path) { return exists(path.c_str()); }
/** Make a subdirectory in the volume root directory.
*
* \param[in] path A path with a valid 8.3 DOS name for the subdirectory.
*
* \param[in] pFlag Create missing parent directories if true.
*
* \return true for success or false for failure.
*/
bool mkdir(const String& path, bool pFlag = true) {
return mkdir(path.c_str(), pFlag);
}
/** open a file
*
* \param[in] path location of file to be opened.
* \param[in] oflag open flags.
* \return a FsBaseFile object.
*/
FsFile open(const String& path, oflag_t oflag = O_RDONLY);
/** Remove a file from the volume root directory.
*
* \param[in] path A path with a valid 8.3 DOS name for the file.
*
* \return true for success or false for failure.
*/
bool remove(const String& path) { return remove(path.c_str()); }
/** Rename a file or subdirectory.
*
* \param[in] oldPath Path name to the file or subdirectory to be renamed.
*
* \param[in] newPath New path name of the file or subdirectory.
*
* The \a newPath object must not exist before the rename call.
*
* The file to be renamed must not be open. The directory entry may be
* moved and file system corruption could occur if the file is accessed by
* a file object that was opened before the rename() call.
*
* \return true for success or false for failure.
*/
bool rename(const String& oldPath, const String& newPath) {
return rename(oldPath.c_str(), newPath.c_str());
}
/** Remove a subdirectory from the volume's root directory.
*
* \param[in] path A path with a valid 8.3 DOS name for the subdirectory.
*
* The subdirectory file will be removed only if it is empty.
*
* \return true for success or false for failure.
*/
bool rmdir(const String& path) { return rmdir(path.c_str()); }
/** Rename a file or subdirectory.
*
* \param[in] oldPath Path name to the file or subdirectory to be renamed.
*
* \param[in] newPath New path name of the file or subdirectory.
*
* The \a newPath object must not exist before the rename call.
*
* The file to be renamed must not be open. The directory entry may be
* moved and file system corruption could occur if the file is accessed by
* a file object that was opened before the rename() call.
*
* \return true for success or false for failure.
*/
#endif // ENABLE_ARDUINO_STRING
protected:
newalign_t m_volMem[FS_ALIGN_DIM(ExFatVolume, FatVolume)];
private:
/** FsBaseFile allowed access to private members. */
friend class FsBaseFile;
static FsVolume* cwv() { return m_cwv; }
FsVolume(const FsVolume& from);
FsVolume& operator=(const FsVolume& from);
static FsVolume* m_cwv;
FatVolume* m_fVol = nullptr;
ExFatVolume* m_xVol = nullptr;
};
#endif // FsVolume_h