| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858 | /** * 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 FsFile_h#define FsFile_h/** * \file * \brief FsBaseFile include file. */#include "FsNew.h"#include "FatLib/FatLib.h"#include "ExFatLib/ExFatLib.h"/** * \class FsBaseFile * \brief FsBaseFile class. */class FsBaseFile { public:  /** Create an instance. */  FsBaseFile() {}  /**  Create a file object and open it in the current working directory.   *   * \param[in] path A path for a file to be opened.   *   * \param[in] oflag Values for \a oflag are constructed by a bitwise-inclusive   * OR of open flags. see FatFile::open(FatFile*, const char*, uint8_t).   */  FsBaseFile(const char* path, oflag_t oflag) {    open(path, oflag);  }  ~FsBaseFile() {close();}  /** Copy constructor.   *   * \param[in] from Object used to initialize this instance.   */  FsBaseFile(const FsBaseFile& from);  /** Copy assignment operator   * \param[in] from Object used to initialize this instance.   * \return assigned object.   */  FsBaseFile& operator=(const FsBaseFile& from);  /** The parenthesis operator.    *    * \return true if a file is open.    */  operator bool() const {return isOpen();}  /**   * \return user settable file attributes for success else -1.   */  int attrib() {     return m_fFile ? m_fFile->attrib() :            m_xFile ? m_xFile->attrib() : -1;  }  /** Set file attributes   *   * \param[in] bits bit-wise or of selected attributes: FS_ATTRIB_READ_ONLY,   *            FS_ATTRIB_HIDDEN, FS_ATTRIB_SYSTEM, FS_ATTRIB_ARCHIVE.   *   * \note attrib() will fail for set read-only if the file is open for write.   * \return true for success or false for failure.   */  bool attrib(uint8_t bits) {    return m_fFile ? m_fFile->attrib(bits) :           m_xFile ? m_xFile->attrib(bits) : false;  }  /** \return number of bytes available from the current position to EOF   *   or INT_MAX if more than INT_MAX bytes are available.   */  int available() const {    return m_fFile ? m_fFile->available() :           m_xFile ? m_xFile->available() : 0;  }  /** \return The number of bytes available from the current position   * to EOF for normal files.  Zero is returned for directory files.   */  uint64_t available64() const {    return m_fFile ? m_fFile->available32() :           m_xFile ? m_xFile->available64() : 0;  }  /** Clear writeError. */  void clearWriteError() {    if (m_fFile) m_fFile->clearWriteError();    if (m_xFile) m_xFile->clearWriteError();  }  /** Close a file and force cached data and directory information   *  to be written to the storage device.   *   * \return true for success or false for failure.   */  bool close();  /** Check for contiguous file and return its raw sector range.   *   * \param[out] bgnSector the first sector address for the file.   * \param[out] endSector the last  sector address for the file.   *   * Set contiguous flag for FAT16/FAT32 files.   * Parameters may be nullptr.   *   * \return true for success or false for failure.   */  bool contiguousRange(uint32_t* bgnSector, uint32_t* endSector) {    return m_fFile ? m_fFile->contiguousRange(bgnSector, endSector) :           m_xFile ? m_xFile->contiguousRange(bgnSector, endSector) : false;  }  /** \return The current cluster number for a file or directory. */  uint32_t curCluster() const {    return m_fFile ? m_fFile->curCluster() :           m_xFile ? m_xFile->curCluster() : 0;  }  /** \return The current position for a file or directory. */  uint64_t curPosition() const {    return m_fFile ? m_fFile->curPosition() :           m_xFile ? m_xFile->curPosition() : 0;  }  /** \return Directory entry index. */  uint32_t dirIndex() const {    return m_fFile ? m_fFile->dirIndex() :           m_xFile ? m_xFile->dirIndex() : 0;  }  /** Test for the existence of a file in a directory   *   * \param[in] path Path of the file to be tested for.   *   * The calling instance must be an open directory file.   *   * dirFile.exists("TOFIND.TXT") searches for "TOFIND.TXT" in  the directory   * dirFile.   *   * \return true if the file exists else false.   */  bool exists(const char* path) {    return m_fFile ? m_fFile->exists(path) :           m_xFile ? m_xFile->exists(path) : false;  }  /** get position for streams   * \param[out] pos struct to receive position   */  void fgetpos(fspos_t* pos) const {    if (m_fFile) m_fFile->fgetpos(pos);    if (m_xFile) m_xFile->fgetpos(pos);  } /**   * Get a string from a file.   *   * fgets() reads bytes from a file into the array pointed to by \a str, until   * \a num - 1 bytes are read, or a delimiter is read and transferred to \a str,   * or end-of-file is encountered. The string is then terminated   * with a null byte.   *   * fgets() deletes CR, '\\r', from the string.  This insures only a '\\n'   * terminates the string for Windows text files which use CRLF for newline.   *   * \param[out] str Pointer to the array where the string is stored.   * \param[in] num Maximum number of characters to be read   * (including the final null byte). Usually the length   * of the array \a str is used.   * \param[in] delim Optional set of delimiters. The default is "\n".   *   * \return For success fgets() returns the length of the string in \a str.   * If no data is read, fgets() returns zero for EOF or -1 if an error occurred.   */  int fgets(char* str, int num, char* delim = nullptr) {    return m_fFile ? m_fFile->fgets(str, num, delim) :           m_xFile ? m_xFile->fgets(str, num, delim) : -1;  }  /** \return The total number of bytes in a file. */  uint64_t fileSize() const {    return m_fFile ? m_fFile->fileSize() :           m_xFile ? m_xFile->fileSize() : 0;  }  /** \return Address of first sector or zero for empty file. */  uint32_t firstSector() const {    return m_fFile ? m_fFile->firstSector() :           m_xFile ? m_xFile->firstSector() : 0;  }  /** Ensure that any bytes written to the file are saved to the SD card. */  void flush() {sync();}  /** set position for streams   * \param[in] pos struct with value for new position   */  void fsetpos(const fspos_t* pos) {    if (m_fFile) m_fFile->fsetpos(pos);    if (m_xFile) m_xFile->fsetpos(pos);  }  /** Get a file's access date and time.   *   * \param[out] pdate Packed date for directory entry.   * \param[out] ptime Packed time for directory entry.   *   * \return true for success or false for failure.   */  bool getAccessDateTime(uint16_t* pdate, uint16_t* ptime) {    return m_fFile ? m_fFile->getAccessDateTime(pdate, ptime) :           m_xFile ? m_xFile->getAccessDateTime(pdate, ptime) : false;  }  /** Get a file's create date and time.   *   * \param[out] pdate Packed date for directory entry.   * \param[out] ptime Packed time for directory entry.   *   * \return true for success or false for failure.   */  bool getCreateDateTime(uint16_t* pdate, uint16_t* ptime) {    return m_fFile ? m_fFile->getCreateDateTime(pdate, ptime) :           m_xFile ? m_xFile->getCreateDateTime(pdate, ptime) : false;  }  /** \return All error bits. */  uint8_t getError() const {    return m_fFile ? m_fFile->getError() :           m_xFile ? m_xFile->getError() : 0XFF;  }  /** Get a file's Modify date and time.   *   * \param[out] pdate Packed date for directory entry.   * \param[out] ptime Packed time for directory entry.   *   * \return true for success or false for failure.   */  bool getModifyDateTime(uint16_t* pdate, uint16_t* ptime) {    return m_fFile ? m_fFile->getModifyDateTime(pdate, ptime) :           m_xFile ? m_xFile->getModifyDateTime(pdate, ptime) : false;  }  /**   * Get a file's name followed by a zero byte.   *   * \param[out] name An array of characters for the file's name.   * \param[in] len The size of the array in bytes. The array   *             must be at least 13 bytes long.  The file's name will be   *             truncated if the file's name is too long.   * \return The length of the returned string.   */  size_t getName(char* name, size_t len) {    *name = 0;    return m_fFile ? m_fFile->getName(name, len) :           m_xFile ? m_xFile->getName(name, len) : 0;  }  /** \return value of writeError */  bool getWriteError() const {    return m_fFile ? m_fFile->getWriteError() :           m_xFile ? m_xFile->getWriteError() : true;  }  /**   * Check for FsBlockDevice busy.   *   * \return true if busy else false.   */  bool isBusy() {    return m_fFile ? m_fFile->isBusy() :           m_xFile ? m_xFile->isBusy() : true;  }  /** \return True if the file is contiguous. */  bool isContiguous() const {#if USE_FAT_FILE_FLAG_CONTIGUOUS    return m_fFile ? m_fFile->isContiguous() :           m_xFile ? m_xFile->isContiguous() : false;#else  // USE_FAT_FILE_FLAG_CONTIGUOUS    return m_xFile ? m_xFile->isContiguous() : false;#endif  // USE_FAT_FILE_FLAG_CONTIGUOUS  }  /** \return True if this is a directory else false. */  bool isDir() const {    return m_fFile ? m_fFile->isDir() :           m_xFile ? m_xFile->isDir() : false;  }  /** This function reports if the current file is a directory or not.   * \return true if the file is a directory.   */  bool isDirectory() const {return isDir();}  /** \return True if this is a normal file. */  bool isFile() const {    return m_fFile ? m_fFile->isFile() :           m_xFile ? m_xFile->isFile() : false;  }  /** \return True if this is a normal file or sub-directory. */  bool isFileOrSubDir() const {    return m_fFile ? m_fFile->isFileOrSubDir() :           m_xFile ? m_xFile->isFileOrSubDir() : false;  }  /** \return True if this is a hidden file else false. */  bool isHidden() const {    return m_fFile ? m_fFile->isHidden() :           m_xFile ? m_xFile->isHidden() : false;  }  /** \return True if this is an open file/directory else false. */  bool isOpen() const {return m_fFile || m_xFile;}  /** \return True file is readable. */  bool isReadable() const {    return m_fFile ? m_fFile->isReadable() :           m_xFile ? m_xFile->isReadable() : false;    }  /** \return True if file is read-only */  bool isReadOnly() const {    return m_fFile ? m_fFile->isReadOnly() :           m_xFile ? m_xFile->isReadOnly() : false;  }  /** \return True if this is a sub-directory file else false. */  bool isSubDir() const {    return m_fFile ? m_fFile->isSubDir() :           m_xFile ? m_xFile->isSubDir() : false;  }  /** \return True file is writable. */  bool isWritable() const {    return m_fFile ? m_fFile->isWritable() :           m_xFile ? m_xFile->isWritable() : false;  }#if ENABLE_ARDUINO_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.   */  bool ls(uint8_t flags) {    return ls(&Serial, flags);  }  /** List directory contents. */  bool ls() {    return ls(&Serial);  }#endif  // ENABLE_ARDUINO_SERIAL  /** List directory contents.   *   * \param[in] pr Print object.   *   * \return true for success or false for failure.   */  bool ls(print_t* pr) {    return m_fFile ? m_fFile->ls(pr) :           m_xFile ? m_xFile->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_fFile ? m_fFile->ls(pr, flags) :           m_xFile ? m_xFile->ls(pr, flags) : false;  }  /** Make a new directory.   *   * \param[in] dir An open FatFile instance for the directory that will   *                   contain the new directory.   *   * \param[in] path A path with a valid 8.3 DOS name for the new directory.   *   * \param[in] pFlag Create missing parent directories if true.   *   * \return true for success or false for failure.   */  bool mkdir(FsBaseFile* dir, const char* path, bool pFlag = true);  /** Open a file or directory by name.   *   * \param[in] dir An open file instance for the directory containing   *                    the file to be opened.   *   * \param[in] path A path with a valid 8.3 DOS name for a file to be opened.   *   * \param[in] oflag Values for \a oflag are constructed by a   *                  bitwise-inclusive OR of flags from the following list   *   * O_RDONLY - Open for reading only..   *   * O_READ - Same as O_RDONLY.   *   * O_WRONLY - Open for writing only.   *   * O_WRITE - Same as O_WRONLY.   *   * O_RDWR - Open for reading and writing.   *   * O_APPEND - If set, the file offset shall be set to the end of the   * file prior to each write.   *   * O_AT_END - Set the initial position at the end of the file.   *   * O_CREAT - If the file exists, this flag has no effect except as noted   * under O_EXCL below. Otherwise, the file shall be created   *   * O_EXCL - If O_CREAT and O_EXCL are set, open() shall fail if the file exists.   *   * O_TRUNC - If the file exists and is a regular file, and the file is   * successfully opened and is not read only, its length shall be truncated to 0.   *   * WARNING: A given file must not be opened by more than one file object   * or file corruption may occur.   *   * \note Directory files must be opened read only.  Write and truncation is   * not allowed for directory files.   *   * \return true for success or false for failure.   */  bool open(FsBaseFile* dir, const char* path, oflag_t oflag = O_RDONLY);  /** Open a file by index.   *   * \param[in] dir An open FsFile instance for the directory.   *   * \param[in] index The \a index of the directory entry for the file to be   * opened.  The value for \a index is (directory file position)/32.   *   * \param[in] oflag bitwise-inclusive OR of open flags.   *            See see FsFile::open(FsFile*, const char*, uint8_t).   *   * See open() by path for definition of flags.   * \return true for success or false for failure.   */  bool open(FsBaseFile* dir, uint32_t index, oflag_t oflag = O_RDONLY);  /** Open a file or directory by name.   *   * \param[in] vol Volume where the file is located.   *   * \param[in] path A path for a file to be opened.   *   * \param[in] oflag Values for \a oflag are constructed by a   *                  bitwise-inclusive OR of open flags.   *   * \return true for success or false for failure.   */  bool open(FsVolume* vol, const char* path, oflag_t oflag = O_RDONLY);  /** Open a file or directory by name.   *   * \param[in] path A path for a file to be opened.   *   * \param[in] oflag Values for \a oflag are constructed by a   *                  bitwise-inclusive OR of open flags.   *   * \return true for success or false for failure.   */  bool open(const char* path, oflag_t oflag = O_RDONLY) {    return FsVolume::m_cwv && open(FsVolume::m_cwv, path, oflag);  }   /** Open a file or directory by index in the current working directory.   *   * \param[in] index The \a index of the directory entry for the file to be   * opened.  The value for \a index is (directory file position)/32.   *   * \param[in] oflag Values for \a oflag are constructed by a   *                  bitwise-inclusive OR of open flags.   *   * \return true for success or false for failure.   */  bool open(uint32_t index, oflag_t oflag = O_RDONLY) {    FsBaseFile cwd;    return cwd.openCwd() && open(&cwd, index, oflag);  }  /** Open the current working directory.   *   * \return true for success or false for failure.   */  bool openCwd();  /** Opens the next file or folder in a directory.   * \param[in] dir directory containing files.   * \param[in] oflag open flags.   * \return a file object.   */  bool openNext(FsBaseFile* dir, oflag_t oflag = O_RDONLY);  /** Open a volume's root directory.   *   * \param[in] vol The SdFs volume containing the root directory to be opened.   *   * \return true for success or false for failure.   */  bool openRoot(FsVolume* vol);  /** \return the current file position. */  uint64_t position() const {return curPosition();}  /** Return the next available byte without consuming it.   *   * \return The byte if no error and not at eof else -1;   */  int peek() {    return m_fFile ? m_fFile->peek() :           m_xFile ? m_xFile->peek() : -1;  }  /** Allocate contiguous clusters to an empty file.   *   * The file must be empty with no clusters allocated.   *   * The file will contain uninitialized data for FAT16/FAT32 files.   * exFAT files will have zero validLength and dataLength will equal   * the requested length.   *   * \param[in] length size of the file in bytes.   * \return true for success or false for failure.   */  bool preAllocate(uint64_t length) {    return m_fFile ? length < (1ULL << 32) && m_fFile->preAllocate(length) :           m_xFile ? m_xFile->preAllocate(length) : false;  }  /** Print a file's access date and time   *   * \param[in] pr Print stream for output.   *   * \return true for success or false for failure.   */  size_t printAccessDateTime(print_t* pr) {    return m_fFile ? m_fFile->printAccessDateTime(pr) :           m_xFile ? m_xFile->printAccessDateTime(pr) : 0;  }  /** Print a file's creation date and time   *   * \param[in] pr Print stream for output.   *   * \return true for success or false for failure.   */  size_t printCreateDateTime(print_t* pr) {    return m_fFile ? m_fFile->printCreateDateTime(pr) :           m_xFile ? m_xFile->printCreateDateTime(pr) : 0;  }  /** Print a number followed by a field terminator.   * \param[in] value The number to be printed.   * \param[in] term The field terminator.  Use '\\n' for CR LF.   * \param[in] prec Number of digits after decimal point.   * \return The number of bytes written or -1 if an error occurs.   */  size_t printField(double value, char term, uint8_t prec = 2) {    return m_fFile ? m_fFile->printField(value, term, prec) :           m_xFile ? m_xFile->printField(value, term, prec) : 0;  }  /** Print a number followed by a field terminator.   * \param[in] value The number to be printed.   * \param[in] term The field terminator.  Use '\\n' for CR LF.   * \param[in] prec Number of digits after decimal point.   * \return The number of bytes written or -1 if an error occurs.   */  size_t printField(float value, char term, uint8_t prec = 2) {     return printField(static_cast<double>(value), term, prec);  }  /** Print a number followed by a field terminator.   * \param[in] value The number to be printed.   * \param[in] term The field terminator.  Use '\\n' for CR LF.   * \return The number of bytes written or -1 if an error occurs.   */  template<typename Type>  size_t printField(Type value, char term) {    return m_fFile ? m_fFile->printField(value, term) :           m_xFile ? m_xFile->printField(value, term) : 0;  }  /** Print a file's size.   *   * \param[in] pr Print stream for output.   *   * \return The number of characters printed is returned   *         for success and zero is returned for failure.   */  size_t printFileSize(print_t* pr) {    return m_fFile ? m_fFile->printFileSize(pr) :           m_xFile ? m_xFile->printFileSize(pr) : 0;  }  /** Print a file's modify date and time   *   * \param[in] pr Print stream for output.   *   * \return true for success or false for failure.   */  size_t printModifyDateTime(print_t* pr) {    return m_fFile ? m_fFile->printModifyDateTime(pr) :           m_xFile ? m_xFile->printModifyDateTime(pr) : 0;  }  /** Print a file's name   *   * \param[in] pr Print stream for output.   *   * \return true for success or false for failure.   */  size_t printName(print_t* pr) {    return m_fFile ? m_fFile->printName(pr) :           m_xFile ? m_xFile->printName(pr) : 0;  }  /** Read the next byte from a file.   *   * \return For success return the next byte in the file as an int.   * If an error occurs or end of file is reached return -1.   */  int read() {    uint8_t b;    return read(&b, 1) == 1 ? b : -1;  }  /** Read data from a file starting at the current position.   *   * \param[out] buf Pointer to the location that will receive the data.   *   * \param[in] count Maximum number of bytes to read.   *   * \return For success read() returns the number of bytes read.   * A value less than \a count, including zero, will be returned   * if end of file is reached.   * If an error occurs, read() returns -1.  Possible errors include   * read() called before a file has been opened, corrupt file system   * or an I/O error occurred.   */  int read(void* buf, size_t count) {    return m_fFile ? m_fFile->read(buf, count) :           m_xFile ? m_xFile->read(buf, count) : -1;  }  /** Remove a file.   *   * The directory entry and all data for the file are deleted.   *   * \note This function should not be used to delete the 8.3 version of a   * file that has a long name. For example if a file has the long name   * "New Text Document.txt" you should not delete the 8.3 name "NEWTEX~1.TXT".   *   * \return true for success or false for failure.   */  bool remove();   /** Remove a file.   *   * The directory entry and all data for the file are deleted.   *   * \param[in] path Path for the file to be removed.   *   * Example use: dirFile.remove(filenameToRemove);   *   * \note This function should not be used to delete the 8.3 version of a   * file that has a long name. For example if a file has the long name   * "New Text Document.txt" you should not delete the 8.3 name "NEWTEX~1.TXT".   *   * \return true for success or false for failure.   */  bool remove(const char* path) {    return m_fFile ? m_fFile->remove(path) :           m_xFile ? m_xFile->remove(path) : false;  }  /** Rename a file or subdirectory.   *   * \param[in] newPath New path name for the file/directory.   *   * \return true for success or false for failure.   */  bool rename(const char* newPath) {    return m_fFile ? m_fFile->rename(newPath) :           m_xFile ? m_xFile->rename(newPath) : false;  }  /** Rename a file or subdirectory.   *   * \param[in] dir Directory for the new path.   * \param[in] newPath New path name for the file/directory.   *   * \return true for success or false for failure.   */  bool rename(FsBaseFile* dir, const char* newPath) {    return m_fFile && dir->m_fFile ? m_fFile->rename(dir->m_fFile, newPath) :           m_xFile && dir->m_xFile ? m_xFile->rename(dir->m_xFile, newPath) :           false;  }  /** Set the file's current position to zero. */  void rewind() {    if (m_fFile) m_fFile->rewind();    if (m_xFile) m_xFile->rewind();  }  /** Rewind a file if it is a directory */  void rewindDirectory() {    if (isDir()) rewind();  }  /** Remove a directory file.   *   * The directory file will be removed only if it is empty and is not the   * root directory.  rmdir() follows DOS and Windows and ignores the   * read-only attribute for the directory.   *   * \note This function should not be used to delete the 8.3 version of a   * directory that has a long name. For example if a directory has the   * long name "New folder" you should not delete the 8.3 name "NEWFOL~1".   *   * \return true for success or false for failure.   */  bool rmdir();  /** Seek to a new position in the file, which must be between   * 0 and the size of the file (inclusive).   *   * \param[in] pos the new file position.   * \return true for success or false for failure.   */  bool seek(uint64_t pos) {return seekSet(pos);}  /** Set the files position to current position + \a pos. See seekSet().   * \param[in] offset The new position in bytes from the current position.   * \return true for success or false for failure.   */  bool seekCur(int64_t offset) {    return seekSet(curPosition() + offset);  }  /** Set the files position to end-of-file + \a offset. See seekSet().   * Can't be used for directory files since file size is not defined.   * \param[in] offset The new position in bytes from end-of-file.   * \return true for success or false for failure.   */  bool seekEnd(int64_t offset = 0) {    return seekSet(fileSize() + offset);  }  /** Sets a file's position.   *   * \param[in] pos The new position in bytes from the beginning of the file.   *   * \return true for success or false for failure.   */  bool seekSet(uint64_t pos) {    return m_fFile ? pos < (1ULL << 32) && m_fFile->seekSet(pos) :           m_xFile ? m_xFile->seekSet(pos) : false;  }  /** \return the file's size. */  uint64_t size() const {return fileSize();}  /** The sync() call causes all modified data and directory fields   * to be written to the storage device.   *   * \return true for success or false for failure.   */  bool sync() {    return m_fFile ? m_fFile->sync() :           m_xFile ? m_xFile->sync() : false;  }  /** Set a file's timestamps in its directory entry.   *   * \param[in] flags Values for \a flags are constructed by a bitwise-inclusive   * OR of flags from the following list   *   * T_ACCESS - Set the file's last access date and time.   *   * T_CREATE - Set the file's creation date and time.   *   * T_WRITE - Set the file's last write/modification date and time.   *   * \param[in] year Valid range 1980 - 2107 inclusive.   *   * \param[in] month Valid range 1 - 12 inclusive.   *   * \param[in] day Valid range 1 - 31 inclusive.   *   * \param[in] hour Valid range 0 - 23 inclusive.   *   * \param[in] minute Valid range 0 - 59 inclusive.   *   * \param[in] second Valid range 0 - 59 inclusive   *   * \note It is possible to set an invalid date since there is no check for   * the number of days in a month.   *   * \note   * Modify and access timestamps may be overwritten if a date time callback   * function has been set by dateTimeCallback().   *   * \return true for success or false for failure.   */  bool timestamp(uint8_t flags, uint16_t year, uint8_t month, uint8_t day,                 uint8_t hour, uint8_t minute, uint8_t second) {    return m_fFile ?           m_fFile->timestamp(flags, year, month, day, hour, minute, second) :           m_xFile ?           m_xFile->timestamp(flags, year, month, day, hour, minute, second) :           false;  }  /** Truncate a file to the current position.   *   * \return true for success or false for failure.   */  bool truncate() {    return m_fFile ? m_fFile->truncate() :           m_xFile ? m_xFile->truncate() : false;  }  /** Truncate a file to a specified length.   * The current file position will be set to end of file.   *   * \param[in] length The desired length for the file.   *   * \return true for success or false for failure.   */  bool truncate(uint64_t length) {    return m_fFile ? length < (1ULL << 32) && m_fFile->truncate(length) :           m_xFile ? m_xFile->truncate(length) : false;  }  /** Write a string to a file. Used by the Arduino Print class.   * \param[in] str Pointer to the string.   * Use getWriteError to check for errors.   * \return count of characters written for success or -1 for failure.   */  size_t write(const char* str) {    return write(str, strlen(str));  }  /** Write a byte to a file. Required by the Arduino Print class.   * \param[in] b the byte to be written.   * Use getWriteError to check for errors.   * \return 1 for success and 0 for failure.   */  size_t write(uint8_t b) {return write(&b, 1);}  /** Write data to an open file.   *   * \note Data is moved to the cache but may not be written to the   * storage device until sync() is called.   *   * \param[in] buf Pointer to the location of the data to be written.   *   * \param[in] count Number of bytes to write.   *   * \return For success write() returns the number of bytes written, always   * \a nbyte.  If an error occurs, write() returns zero and writeError is set.   */  size_t write(const void* buf, size_t count) {    return m_fFile ? m_fFile->write(buf, count) :           m_xFile ? m_xFile->write(buf, count) : 0;  } private:  newalign_t m_fileMem[FS_ALIGN_DIM(ExFatFile, FatFile)];  FatFile*   m_fFile = nullptr;  ExFatFile* m_xFile = nullptr;};/** * \class FsFile * \brief FsBaseFile file with Arduino Stream. */class FsFile : public StreamFile<FsBaseFile, uint64_t> { public:  /** Opens the next file or folder in a directory.   *   * \param[in] oflag open flags.   * \return a FatStream object.   */  FsFile openNextFile(oflag_t oflag = O_RDONLY) {    FsFile tmpFile;    tmpFile.openNext(this, oflag);    return tmpFile;  }};#endif  // FsFile_h
 |