mirror of https://github.com/arendst/Tasmota.git
LITTLEFS is a lib in Tasmota core32
This commit is contained in:
parent
d117b495ab
commit
ff5e9aa15b
|
@ -1,339 +0,0 @@
|
||||||
GNU GENERAL PUBLIC LICENSE
|
|
||||||
Version 2, June 1991
|
|
||||||
|
|
||||||
Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
|
|
||||||
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
||||||
Everyone is permitted to copy and distribute verbatim copies
|
|
||||||
of this license document, but changing it is not allowed.
|
|
||||||
|
|
||||||
Preamble
|
|
||||||
|
|
||||||
The licenses for most software are designed to take away your
|
|
||||||
freedom to share and change it. By contrast, the GNU General Public
|
|
||||||
License is intended to guarantee your freedom to share and change free
|
|
||||||
software--to make sure the software is free for all its users. This
|
|
||||||
General Public License applies to most of the Free Software
|
|
||||||
Foundation's software and to any other program whose authors commit to
|
|
||||||
using it. (Some other Free Software Foundation software is covered by
|
|
||||||
the GNU Lesser General Public License instead.) You can apply it to
|
|
||||||
your programs, too.
|
|
||||||
|
|
||||||
When we speak of free software, we are referring to freedom, not
|
|
||||||
price. Our General Public Licenses are designed to make sure that you
|
|
||||||
have the freedom to distribute copies of free software (and charge for
|
|
||||||
this service if you wish), that you receive source code or can get it
|
|
||||||
if you want it, that you can change the software or use pieces of it
|
|
||||||
in new free programs; and that you know you can do these things.
|
|
||||||
|
|
||||||
To protect your rights, we need to make restrictions that forbid
|
|
||||||
anyone to deny you these rights or to ask you to surrender the rights.
|
|
||||||
These restrictions translate to certain responsibilities for you if you
|
|
||||||
distribute copies of the software, or if you modify it.
|
|
||||||
|
|
||||||
For example, if you distribute copies of such a program, whether
|
|
||||||
gratis or for a fee, you must give the recipients all the rights that
|
|
||||||
you have. You must make sure that they, too, receive or can get the
|
|
||||||
source code. And you must show them these terms so they know their
|
|
||||||
rights.
|
|
||||||
|
|
||||||
We protect your rights with two steps: (1) copyright the software, and
|
|
||||||
(2) offer you this license which gives you legal permission to copy,
|
|
||||||
distribute and/or modify the software.
|
|
||||||
|
|
||||||
Also, for each author's protection and ours, we want to make certain
|
|
||||||
that everyone understands that there is no warranty for this free
|
|
||||||
software. If the software is modified by someone else and passed on, we
|
|
||||||
want its recipients to know that what they have is not the original, so
|
|
||||||
that any problems introduced by others will not reflect on the original
|
|
||||||
authors' reputations.
|
|
||||||
|
|
||||||
Finally, any free program is threatened constantly by software
|
|
||||||
patents. We wish to avoid the danger that redistributors of a free
|
|
||||||
program will individually obtain patent licenses, in effect making the
|
|
||||||
program proprietary. To prevent this, we have made it clear that any
|
|
||||||
patent must be licensed for everyone's free use or not licensed at all.
|
|
||||||
|
|
||||||
The precise terms and conditions for copying, distribution and
|
|
||||||
modification follow.
|
|
||||||
|
|
||||||
GNU GENERAL PUBLIC LICENSE
|
|
||||||
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
|
||||||
|
|
||||||
0. This License applies to any program or other work which contains
|
|
||||||
a notice placed by the copyright holder saying it may be distributed
|
|
||||||
under the terms of this General Public License. The "Program", below,
|
|
||||||
refers to any such program or work, and a "work based on the Program"
|
|
||||||
means either the Program or any derivative work under copyright law:
|
|
||||||
that is to say, a work containing the Program or a portion of it,
|
|
||||||
either verbatim or with modifications and/or translated into another
|
|
||||||
language. (Hereinafter, translation is included without limitation in
|
|
||||||
the term "modification".) Each licensee is addressed as "you".
|
|
||||||
|
|
||||||
Activities other than copying, distribution and modification are not
|
|
||||||
covered by this License; they are outside its scope. The act of
|
|
||||||
running the Program is not restricted, and the output from the Program
|
|
||||||
is covered only if its contents constitute a work based on the
|
|
||||||
Program (independent of having been made by running the Program).
|
|
||||||
Whether that is true depends on what the Program does.
|
|
||||||
|
|
||||||
1. You may copy and distribute verbatim copies of the Program's
|
|
||||||
source code as you receive it, in any medium, provided that you
|
|
||||||
conspicuously and appropriately publish on each copy an appropriate
|
|
||||||
copyright notice and disclaimer of warranty; keep intact all the
|
|
||||||
notices that refer to this License and to the absence of any warranty;
|
|
||||||
and give any other recipients of the Program a copy of this License
|
|
||||||
along with the Program.
|
|
||||||
|
|
||||||
You may charge a fee for the physical act of transferring a copy, and
|
|
||||||
you may at your option offer warranty protection in exchange for a fee.
|
|
||||||
|
|
||||||
2. You may modify your copy or copies of the Program or any portion
|
|
||||||
of it, thus forming a work based on the Program, and copy and
|
|
||||||
distribute such modifications or work under the terms of Section 1
|
|
||||||
above, provided that you also meet all of these conditions:
|
|
||||||
|
|
||||||
a) You must cause the modified files to carry prominent notices
|
|
||||||
stating that you changed the files and the date of any change.
|
|
||||||
|
|
||||||
b) You must cause any work that you distribute or publish, that in
|
|
||||||
whole or in part contains or is derived from the Program or any
|
|
||||||
part thereof, to be licensed as a whole at no charge to all third
|
|
||||||
parties under the terms of this License.
|
|
||||||
|
|
||||||
c) If the modified program normally reads commands interactively
|
|
||||||
when run, you must cause it, when started running for such
|
|
||||||
interactive use in the most ordinary way, to print or display an
|
|
||||||
announcement including an appropriate copyright notice and a
|
|
||||||
notice that there is no warranty (or else, saying that you provide
|
|
||||||
a warranty) and that users may redistribute the program under
|
|
||||||
these conditions, and telling the user how to view a copy of this
|
|
||||||
License. (Exception: if the Program itself is interactive but
|
|
||||||
does not normally print such an announcement, your work based on
|
|
||||||
the Program is not required to print an announcement.)
|
|
||||||
|
|
||||||
These requirements apply to the modified work as a whole. If
|
|
||||||
identifiable sections of that work are not derived from the Program,
|
|
||||||
and can be reasonably considered independent and separate works in
|
|
||||||
themselves, then this License, and its terms, do not apply to those
|
|
||||||
sections when you distribute them as separate works. But when you
|
|
||||||
distribute the same sections as part of a whole which is a work based
|
|
||||||
on the Program, the distribution of the whole must be on the terms of
|
|
||||||
this License, whose permissions for other licensees extend to the
|
|
||||||
entire whole, and thus to each and every part regardless of who wrote it.
|
|
||||||
|
|
||||||
Thus, it is not the intent of this section to claim rights or contest
|
|
||||||
your rights to work written entirely by you; rather, the intent is to
|
|
||||||
exercise the right to control the distribution of derivative or
|
|
||||||
collective works based on the Program.
|
|
||||||
|
|
||||||
In addition, mere aggregation of another work not based on the Program
|
|
||||||
with the Program (or with a work based on the Program) on a volume of
|
|
||||||
a storage or distribution medium does not bring the other work under
|
|
||||||
the scope of this License.
|
|
||||||
|
|
||||||
3. You may copy and distribute the Program (or a work based on it,
|
|
||||||
under Section 2) in object code or executable form under the terms of
|
|
||||||
Sections 1 and 2 above provided that you also do one of the following:
|
|
||||||
|
|
||||||
a) Accompany it with the complete corresponding machine-readable
|
|
||||||
source code, which must be distributed under the terms of Sections
|
|
||||||
1 and 2 above on a medium customarily used for software interchange; or,
|
|
||||||
|
|
||||||
b) Accompany it with a written offer, valid for at least three
|
|
||||||
years, to give any third party, for a charge no more than your
|
|
||||||
cost of physically performing source distribution, a complete
|
|
||||||
machine-readable copy of the corresponding source code, to be
|
|
||||||
distributed under the terms of Sections 1 and 2 above on a medium
|
|
||||||
customarily used for software interchange; or,
|
|
||||||
|
|
||||||
c) Accompany it with the information you received as to the offer
|
|
||||||
to distribute corresponding source code. (This alternative is
|
|
||||||
allowed only for noncommercial distribution and only if you
|
|
||||||
received the program in object code or executable form with such
|
|
||||||
an offer, in accord with Subsection b above.)
|
|
||||||
|
|
||||||
The source code for a work means the preferred form of the work for
|
|
||||||
making modifications to it. For an executable work, complete source
|
|
||||||
code means all the source code for all modules it contains, plus any
|
|
||||||
associated interface definition files, plus the scripts used to
|
|
||||||
control compilation and installation of the executable. However, as a
|
|
||||||
special exception, the source code distributed need not include
|
|
||||||
anything that is normally distributed (in either source or binary
|
|
||||||
form) with the major components (compiler, kernel, and so on) of the
|
|
||||||
operating system on which the executable runs, unless that component
|
|
||||||
itself accompanies the executable.
|
|
||||||
|
|
||||||
If distribution of executable or object code is made by offering
|
|
||||||
access to copy from a designated place, then offering equivalent
|
|
||||||
access to copy the source code from the same place counts as
|
|
||||||
distribution of the source code, even though third parties are not
|
|
||||||
compelled to copy the source along with the object code.
|
|
||||||
|
|
||||||
4. You may not copy, modify, sublicense, or distribute the Program
|
|
||||||
except as expressly provided under this License. Any attempt
|
|
||||||
otherwise to copy, modify, sublicense or distribute the Program is
|
|
||||||
void, and will automatically terminate your rights under this License.
|
|
||||||
However, parties who have received copies, or rights, from you under
|
|
||||||
this License will not have their licenses terminated so long as such
|
|
||||||
parties remain in full compliance.
|
|
||||||
|
|
||||||
5. You are not required to accept this License, since you have not
|
|
||||||
signed it. However, nothing else grants you permission to modify or
|
|
||||||
distribute the Program or its derivative works. These actions are
|
|
||||||
prohibited by law if you do not accept this License. Therefore, by
|
|
||||||
modifying or distributing the Program (or any work based on the
|
|
||||||
Program), you indicate your acceptance of this License to do so, and
|
|
||||||
all its terms and conditions for copying, distributing or modifying
|
|
||||||
the Program or works based on it.
|
|
||||||
|
|
||||||
6. Each time you redistribute the Program (or any work based on the
|
|
||||||
Program), the recipient automatically receives a license from the
|
|
||||||
original licensor to copy, distribute or modify the Program subject to
|
|
||||||
these terms and conditions. You may not impose any further
|
|
||||||
restrictions on the recipients' exercise of the rights granted herein.
|
|
||||||
You are not responsible for enforcing compliance by third parties to
|
|
||||||
this License.
|
|
||||||
|
|
||||||
7. If, as a consequence of a court judgment or allegation of patent
|
|
||||||
infringement or for any other reason (not limited to patent issues),
|
|
||||||
conditions are imposed on you (whether by court order, agreement or
|
|
||||||
otherwise) that contradict the conditions of this License, they do not
|
|
||||||
excuse you from the conditions of this License. If you cannot
|
|
||||||
distribute so as to satisfy simultaneously your obligations under this
|
|
||||||
License and any other pertinent obligations, then as a consequence you
|
|
||||||
may not distribute the Program at all. For example, if a patent
|
|
||||||
license would not permit royalty-free redistribution of the Program by
|
|
||||||
all those who receive copies directly or indirectly through you, then
|
|
||||||
the only way you could satisfy both it and this License would be to
|
|
||||||
refrain entirely from distribution of the Program.
|
|
||||||
|
|
||||||
If any portion of this section is held invalid or unenforceable under
|
|
||||||
any particular circumstance, the balance of the section is intended to
|
|
||||||
apply and the section as a whole is intended to apply in other
|
|
||||||
circumstances.
|
|
||||||
|
|
||||||
It is not the purpose of this section to induce you to infringe any
|
|
||||||
patents or other property right claims or to contest validity of any
|
|
||||||
such claims; this section has the sole purpose of protecting the
|
|
||||||
integrity of the free software distribution system, which is
|
|
||||||
implemented by public license practices. Many people have made
|
|
||||||
generous contributions to the wide range of software distributed
|
|
||||||
through that system in reliance on consistent application of that
|
|
||||||
system; it is up to the author/donor to decide if he or she is willing
|
|
||||||
to distribute software through any other system and a licensee cannot
|
|
||||||
impose that choice.
|
|
||||||
|
|
||||||
This section is intended to make thoroughly clear what is believed to
|
|
||||||
be a consequence of the rest of this License.
|
|
||||||
|
|
||||||
8. If the distribution and/or use of the Program is restricted in
|
|
||||||
certain countries either by patents or by copyrighted interfaces, the
|
|
||||||
original copyright holder who places the Program under this License
|
|
||||||
may add an explicit geographical distribution limitation excluding
|
|
||||||
those countries, so that distribution is permitted only in or among
|
|
||||||
countries not thus excluded. In such case, this License incorporates
|
|
||||||
the limitation as if written in the body of this License.
|
|
||||||
|
|
||||||
9. The Free Software Foundation may publish revised and/or new versions
|
|
||||||
of the General Public License from time to time. Such new versions will
|
|
||||||
be similar in spirit to the present version, but may differ in detail to
|
|
||||||
address new problems or concerns.
|
|
||||||
|
|
||||||
Each version is given a distinguishing version number. If the Program
|
|
||||||
specifies a version number of this License which applies to it and "any
|
|
||||||
later version", you have the option of following the terms and conditions
|
|
||||||
either of that version or of any later version published by the Free
|
|
||||||
Software Foundation. If the Program does not specify a version number of
|
|
||||||
this License, you may choose any version ever published by the Free Software
|
|
||||||
Foundation.
|
|
||||||
|
|
||||||
10. If you wish to incorporate parts of the Program into other free
|
|
||||||
programs whose distribution conditions are different, write to the author
|
|
||||||
to ask for permission. For software which is copyrighted by the Free
|
|
||||||
Software Foundation, write to the Free Software Foundation; we sometimes
|
|
||||||
make exceptions for this. Our decision will be guided by the two goals
|
|
||||||
of preserving the free status of all derivatives of our free software and
|
|
||||||
of promoting the sharing and reuse of software generally.
|
|
||||||
|
|
||||||
NO WARRANTY
|
|
||||||
|
|
||||||
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
|
||||||
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
|
||||||
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
|
||||||
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
|
||||||
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
||||||
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
|
||||||
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
|
||||||
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
|
||||||
REPAIR OR CORRECTION.
|
|
||||||
|
|
||||||
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
|
||||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
|
||||||
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
|
||||||
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
|
||||||
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
|
||||||
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
|
||||||
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
|
||||||
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
|
||||||
POSSIBILITY OF SUCH DAMAGES.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
How to Apply These Terms to Your New Programs
|
|
||||||
|
|
||||||
If you develop a new program, and you want it to be of the greatest
|
|
||||||
possible use to the public, the best way to achieve this is to make it
|
|
||||||
free software which everyone can redistribute and change under these terms.
|
|
||||||
|
|
||||||
To do so, attach the following notices to the program. It is safest
|
|
||||||
to attach them to the start of each source file to most effectively
|
|
||||||
convey the exclusion of warranty; and each file should have at least
|
|
||||||
the "copyright" line and a pointer to where the full notice is found.
|
|
||||||
|
|
||||||
<one line to give the program's name and a brief idea of what it does.>
|
|
||||||
Copyright (C) <year> <name of author>
|
|
||||||
|
|
||||||
This program is free software; you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation; either version 2 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License along
|
|
||||||
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
||||||
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
||||||
|
|
||||||
Also add information on how to contact you by electronic and paper mail.
|
|
||||||
|
|
||||||
If the program is interactive, make it output a short notice like this
|
|
||||||
when it starts in an interactive mode:
|
|
||||||
|
|
||||||
Gnomovision version 69, Copyright (C) year name of author
|
|
||||||
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
|
||||||
This is free software, and you are welcome to redistribute it
|
|
||||||
under certain conditions; type `show c' for details.
|
|
||||||
|
|
||||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
|
||||||
parts of the General Public License. Of course, the commands you use may
|
|
||||||
be called something other than `show w' and `show c'; they could even be
|
|
||||||
mouse-clicks or menu items--whatever suits your program.
|
|
||||||
|
|
||||||
You should also get your employer (if you work as a programmer) or your
|
|
||||||
school, if any, to sign a "copyright disclaimer" for the program, if
|
|
||||||
necessary. Here is a sample; alter the names:
|
|
||||||
|
|
||||||
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
|
||||||
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
|
||||||
|
|
||||||
<signature of Ty Coon>, 1 April 1989
|
|
||||||
Ty Coon, President of Vice
|
|
||||||
|
|
||||||
This General Public License does not permit incorporating your program into
|
|
||||||
proprietary programs. If your program is a subroutine library, you may
|
|
||||||
consider it more useful to permit linking proprietary applications with the
|
|
||||||
library. If this is what you want to do, use the GNU Lesser General
|
|
||||||
Public License instead of this License.
|
|
|
@ -1,104 +0,0 @@
|
||||||
# LittleFS_esp32
|
|
||||||
|
|
||||||
#### ***Notice: The Library is been integrated to [Arduino esp32 core idf-release/v4.2 branch](https://github.com/espressif/arduino-esp32/tree/idf-release/v4.2 ) for future major core release. On built-in library, #define tweaks below will be unavailable.***
|
|
||||||
|
|
||||||
|
|
||||||
## LittleFS library for arduino-esp32
|
|
||||||
|
|
||||||
- A LittleFS wrapper for Arduino ESP32 of [littlefs-project](https://github.com/littlefs-project/littlefs)
|
|
||||||
- Based on [ESP-IDF port of joltwallet/esp_littlefs](https://github.com/joltwallet/esp_littlefs) , thank you Brian!
|
|
||||||
- As a reference, see [LillteFS library for ESP8266 core](https://github.com/esp8266/Arduino/tree/master/libraries/LittleFS)
|
|
||||||
- [PR](https://github.com/espressif/arduino-esp32/pull/4096) and [merge](https://github.com/espressif/arduino-esp32/pull/4483) at esp32 core development
|
|
||||||
- [PR](https://github.com/espressif/esp-idf/pull/5469) at esp-idf development
|
|
||||||
- The functionality is similar to SPIFFS
|
|
||||||
|
|
||||||
### Installation
|
|
||||||
|
|
||||||
- Use **Arduino Library Manager**
|
|
||||||
- Or download / use **git** to have latest repository of **LITTLEFS** added to Arduino IDE **/libraries** folder
|
|
||||||
(File > Preferences > Sketchbook location).
|
|
||||||
- See ``` #define CONFIG_LITTLEFS_FOR_IDF_3_2 ``` in **esp_littlefs.c**.
|
|
||||||
Now it is defined / undefined automatically by detecting the IDF version and core version.
|
|
||||||
When defined, the library works with old and new IDFs 3.2 - 4.x but file timestamp feature is removed.
|
|
||||||
See LITTLEFS_time example.
|
|
||||||
- See ``` #define CONFIG_LITTLEFS_SPIFFS_COMPAT ``` in **esp_littlefs.c**.
|
|
||||||
When set to 1, folders are recursively created or deleted if empty on creating/deleting a new file like SPIFFS. Default is 0.
|
|
||||||
- The other ``` #define CONFIG_LITTLEFS_xxxxx ``` are set to optimal default values.
|
|
||||||
Read [here](https://github.com/joltwallet/esp_littlefs/blob/master/Kconfig) and [here](https://github.com/littlefs-project//littlefs/blob/master/README.md) if you want to modify.
|
|
||||||
- For low-level default error reporting modifications, see the defines at beginning of the **lfs.c** here.
|
|
||||||
|
|
||||||
### Usage
|
|
||||||
|
|
||||||
- Use LITTLEFS [with identical methods as SPIFFS](https://diyprojects.io/esp32-get-started-spiff-library-read-write-modify-files/) plus mkdir() and rmdir()
|
|
||||||
- A quick startup based on your existing code you can re-define SPIFFS
|
|
||||||
```
|
|
||||||
#define USE_LittleFS
|
|
||||||
|
|
||||||
#include <FS.h>
|
|
||||||
#ifdef USE_LittleFS
|
|
||||||
#define SPIFFS LITTLEFS
|
|
||||||
#include <LITTLEFS.h>
|
|
||||||
#else
|
|
||||||
#include <SPIFFS.h>
|
|
||||||
#endif
|
|
||||||
```
|
|
||||||
- Note, this may not work if your sketch uses other libraries that use SPIFFS themselves.
|
|
||||||
- See also [esp_partition.h](https://github.com/espressif/esp-idf/blob/master/components/spi_flash/include/esp_partition.h) . LITTLEFS re-uses same type and subtype as SPIFFS
|
|
||||||
|
|
||||||
### Differences with SPIFFS
|
|
||||||
|
|
||||||
- LittleFS has folders, you need to iterate files in folders unless you set ``` #define CONFIG_LITTLEFS_SPIFFS_COMPAT 1 ```
|
|
||||||
- At root a "/folder" = "folder"
|
|
||||||
- Requires a label for mount point, NULL will not work. Recommended is to use default LITTLEFS.begin()
|
|
||||||
- maxOpenFiles parameter is unused, kept for compatibility
|
|
||||||
- LITTLEFS.mkdir(path) and LITTLEFS.rmdir(path) are available
|
|
||||||
- file.seek() behaves like on FFat see [more details](https://github.com/lorol/LITTLEFS/issues/11)
|
|
||||||
- file.write() and file.print() when partition space is ending may return different than really written bytes (on other FS is also inconsistent).
|
|
||||||
- Speed comparison based on **LittleFS_test.ino** sketch (for a file 1048576 bytes):
|
|
||||||
|
|
||||||
|Filesystem|Read time [ms]|Write time [ms]|
|
|
||||||
|----|----|----|
|
|
||||||
|FAT|276|14493|
|
|
||||||
|LITTLEFS|446*|16387|
|
|
||||||
|SPIFFS|767|65622|
|
|
||||||
|
|
||||||
*The read speed improved by changing ```#define CONFIG_LITTLEFS_CACHE_SIZE``` from 128 to 512
|
|
||||||
|
|
||||||
|
|
||||||
### Arduino ESP32 LittleFS filesystem upload tool
|
|
||||||
|
|
||||||
- Use (replace if exists) [arduino-esp32fs-plugin](https://github.com/me-no-dev/arduino-esp32fs-plugin/pull/23 ) with [this variant](https://github.com/lorol/arduino-esp32fs-plugin), which supports SPIFFS, LittleFS and FatFS
|
|
||||||
- Requires [mklittlefs executable](https://github.com/earlephilhower/mklittlefs) which is available [in releases section here](https://github.com/lorol/arduino-esp32fs-plugin ) or download the zipped binary [here](https://github.com/earlephilhower/mklittlefs/releases) or from **esp-quick-toolchain** releases [here](https://github.com/earlephilhower/esp-quick-toolchain/releases)
|
|
||||||
- Copy it to **/tools** folder of esp32 platform where **espota** and **esptool** (.py or.exe) tools are located
|
|
||||||
- Restart Arduino IDE.
|
|
||||||
|
|
||||||
### PlatformIO
|
|
||||||
|
|
||||||
- See [LITTLEFS_PlatformIO example here](https://github.com/lorol/LITTLEFS/tree/master/examples/LITTLEFS_PlatformIO)
|
|
||||||
( based on notes below from [BlueAndi](https://github.com/BlueAndi) )
|
|
||||||
- Add to _platformio.ini_:
|
|
||||||
`extra_scripts = replace_fs.py`
|
|
||||||
|
|
||||||
- Add _replace_fs.py_ to project root directory (where platformio.ini is located):
|
|
||||||
|
|
||||||
```python
|
|
||||||
Import("env")
|
|
||||||
print("Replace MKSPIFFSTOOL with mklittlefs.exe")
|
|
||||||
env.Replace (MKSPIFFSTOOL = "mklittlefs.exe")
|
|
||||||
```
|
|
||||||
|
|
||||||
- Add _mklittlefs.exe_ to project root directory as well.
|
|
||||||
|
|
||||||
## Credits and license
|
|
||||||
|
|
||||||
- This work is based on [littlefs-project](https://github.com/littlefs-project/littlefs) , [ESP-IDF port of joltwallet/esp_littlefs](https://github.com/joltwallet/esp_littlefs) , [Espressif Arduino core for the ESP32, the ESP-IDF - SPIFFS Library](https://github.com/espressif/arduino-esp32/tree/master/libraries/SPIFFS)
|
|
||||||
- Licensed under GPL v2 ([text](LICENSE))
|
|
||||||
|
|
||||||
## To Do
|
|
||||||
- [x] Submit to be added to Arduino Library Manager
|
|
||||||
- [x] Optional drop-in compatibility with SPIFFS' lack of folders, similar to esp8266' way - implemented as a choice by a #define
|
|
||||||
- [recursive folders auto creation](https://github.com/esp8266/Arduino/blob/master/libraries/LittleFS/src/LittleFS.cpp#L60) when a new file is created at non-existing path
|
|
||||||
- [recursive folders auto deletion](https://github.com/esp8266/Arduino/blob/master/libraries/LittleFS/src/LittleFS.h#L149) on "last file" deletion (SPIFFS cannot have "folder w/o file")
|
|
||||||
- review other differences: opendir(), rmdir(), unlink()
|
|
||||||
- [ ] Retire this library when released by esp32 Arduino core
|
|
||||||
- [ ] Cleanup examples
|
|
|
@ -1,4 +0,0 @@
|
||||||
.pio
|
|
||||||
.vscode
|
|
||||||
mklittlefs.exe
|
|
||||||
mklittlefs
|
|
|
@ -1,68 +0,0 @@
|
||||||
# How to run on PlatformIO IDE
|
|
||||||
|
|
||||||
- Download and extract to this project root a **mklittlefs** executable for your OS [from a zipped binary here](https://github.com/earlephilhower/mklittlefs/releases)
|
|
||||||
- Open **LITTLEFS_PlatformIO** folder
|
|
||||||
- Run PlatformIO project task: **Upload Filesystem Image**
|
|
||||||
- Run PlatformIO project task: **Upload and Monitor**
|
|
||||||
- You will see a Serial output like:
|
|
||||||
```
|
|
||||||
--- Miniterm on COM5 115200,8,N,1 ---
|
|
||||||
--- Quit: Ctrl+C | Menu: Ctrl+T | Help: Ctrl+T followed by Ctrl+H ---
|
|
||||||
ets Jun 8 2016 00:22:57
|
|
||||||
|
|
||||||
rst:0x1 (POWERON_RESET),boot:0x13 (Snfigsip: 0, SPIWP:0xee
|
|
||||||
clk_drv:0x00,q_drv:0x00,d_drv:0x00,cs0_drv:0x00,hd_drv:0x00,wp_drv:0x00
|
|
||||||
mode:DIO, clock div:2
|
|
||||||
load:0x3fff0018,len:4
|
|
||||||
load:0x3fff001c,len:1044
|
|
||||||
load:0x40078000,len:10044
|
|
||||||
load:0x40080400,len:5872
|
|
||||||
entry 0x400806ac
|
|
||||||
Listing directory: /
|
|
||||||
FILE: /file1.txt SIZE: 3 LAST WRITE: 2020-10-06 15:10:33
|
|
||||||
DIR : /testfolder LAST WRITE: 2020-10-06 15:10:33
|
|
||||||
Creating Dir: /mydir
|
|
||||||
Dir created
|
|
||||||
Writing file: /mydir/hello2.txt
|
|
||||||
- file written
|
|
||||||
Listing directory: /
|
|
||||||
FILE: /file1.txt SIZE: 3 LAST WRITE: 2020-10-06 15:10:33
|
|
||||||
DIR : /mydir LAST WRITE: 1970-01-01 00:00:00
|
|
||||||
Listing directory: /mydir
|
|
||||||
FILE: /mydir/hello2.txt SIZE: 6 LAST WRITE: 1970-01-01 00:00:00
|
|
||||||
DIR : /testfolder LAST WRITE: 2020-10-06 15:10:33
|
|
||||||
Listing directory: /testfolder
|
|
||||||
FILE: /testfolder/test2.txt SIZE: 3 LAST WRITE: 2020-10-06 15:10:33
|
|
||||||
Deleting file: /mydir/hello2.txt
|
|
||||||
- file deleted
|
|
||||||
Removing Dir: /mydir
|
|
||||||
Dir removed
|
|
||||||
Listing directory: /
|
|
||||||
FILE: /file1.txt SIZE: 3 LAST WRITE: 2020-10-06 15:10:33
|
|
||||||
DIR : /testfolder LAST WRITE: 2020-10-06 15:10:33
|
|
||||||
Listing directory: /testfolder
|
|
||||||
FILE: /testfolder/test2.txt SIZE: 3 LAST WRITE: 2020-10-06 15:10:33
|
|
||||||
Writing file: /hello.txt
|
|
||||||
- file written
|
|
||||||
Appending to file: /hello.txt
|
|
||||||
- message appended
|
|
||||||
Reading file: /hello.txt
|
|
||||||
- read from file:
|
|
||||||
Hello World!
|
|
||||||
Renaming file /hello.txt to /foo.txt
|
|
||||||
- file renamed
|
|
||||||
Reading file: /foo.txt
|
|
||||||
- read from file:
|
|
||||||
Hello World!
|
|
||||||
Deleting file: /foo.txt
|
|
||||||
- file deleted
|
|
||||||
Testing file I/O with /test.txt
|
|
||||||
- writing................................................................
|
|
||||||
- 1048576 bytes written in 12006 ms
|
|
||||||
- reading................................................................
|
|
||||||
- 1048576 bytes read in 547 ms
|
|
||||||
Deleting file: /test.txt
|
|
||||||
- file deleted
|
|
||||||
Test complete
|
|
||||||
```
|
|
||||||
- If you have a module with more than 4MB flash, you can uncomment **partitions_custom.csv** in **platformio.ini** and modify the csv file accordingly
|
|
|
@ -1 +0,0 @@
|
||||||
aaa
|
|
|
@ -1 +0,0 @@
|
||||||
bbb
|
|
|
@ -1,2 +0,0 @@
|
||||||
Import("env")
|
|
||||||
env.Replace( MKSPIFFSTOOL=env.get("PROJECT_DIR") + '/mklittlefs' )
|
|
|
@ -1,6 +0,0 @@
|
||||||
# Name, Type, SubType, Offset, Size, Flags
|
|
||||||
ota_0, app, ota_0, 0x10000, 0x1A0000,
|
|
||||||
ota_1, app, ota_1, , 0x1A0000,
|
|
||||||
otadata, data, ota, 0x350000, 0x2000,
|
|
||||||
nvs, data, nvs, , 0x6000,
|
|
||||||
data, data, spiffs, , 0xA8000,
|
|
|
|
@ -1,35 +0,0 @@
|
||||||
; PlatformIO Project Configuration File
|
|
||||||
;
|
|
||||||
; Build options: build flags, source filter
|
|
||||||
; Upload options: custom upload port, speed and extra flags
|
|
||||||
; Library options: dependencies, extra library storages
|
|
||||||
; Advanced options: extra scripting
|
|
||||||
;
|
|
||||||
; Please visit documentation for the other options and examples
|
|
||||||
; https://docs.platformio.org/page/projectconf.html
|
|
||||||
|
|
||||||
[platformio]
|
|
||||||
default_envs = esp32
|
|
||||||
|
|
||||||
[env]
|
|
||||||
framework = arduino
|
|
||||||
|
|
||||||
[env:esp32]
|
|
||||||
platform = espressif32
|
|
||||||
;platform = https://github.com/platformio/platform-espressif32.git
|
|
||||||
;board_build.mcu = esp32
|
|
||||||
platform_packages = framework-arduinoespressif32 @ https://github.com/espressif/arduino-esp32.git
|
|
||||||
|
|
||||||
build_flags =
|
|
||||||
${env.build_flags}
|
|
||||||
-D=${PIOENV}
|
|
||||||
;-D CONFIG_LITTLEFS_FOR_IDF_3_2
|
|
||||||
|
|
||||||
lib_deps = https://github.com/lorol/LITTLEFS.git
|
|
||||||
|
|
||||||
board = esp32dev
|
|
||||||
;board_build.partitions = partitions_custom.csv
|
|
||||||
monitor_filters = esp32_exception_decoder
|
|
||||||
monitor_speed = 115200
|
|
||||||
|
|
||||||
extra_scripts = ./littlefsbuilder.py
|
|
|
@ -1,291 +0,0 @@
|
||||||
#include <Arduino.h>
|
|
||||||
#include "FS.h"
|
|
||||||
#include <LITTLEFS.h>
|
|
||||||
|
|
||||||
#ifndef CONFIG_LITTLEFS_FOR_IDF_3_2
|
|
||||||
#include <time.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* You only need to format LITTLEFS the first time you run a
|
|
||||||
test or else use the LITTLEFS plugin to create a partition
|
|
||||||
https://github.com/lorol/arduino-esp32littlefs-plugin */
|
|
||||||
|
|
||||||
#define FORMAT_LITTLEFS_IF_FAILED true
|
|
||||||
|
|
||||||
void listDir(fs::FS &fs, const char * dirname, uint8_t levels){
|
|
||||||
Serial.printf("Listing directory: %s\r\n", dirname);
|
|
||||||
|
|
||||||
File root = fs.open(dirname);
|
|
||||||
if(!root){
|
|
||||||
Serial.println("- failed to open directory");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(!root.isDirectory()){
|
|
||||||
Serial.println(" - not a directory");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
File file = root.openNextFile();
|
|
||||||
while(file){
|
|
||||||
if(file.isDirectory()){
|
|
||||||
Serial.print(" DIR : ");
|
|
||||||
|
|
||||||
#ifdef CONFIG_LITTLEFS_FOR_IDF_3_2
|
|
||||||
Serial.println(file.name());
|
|
||||||
#else
|
|
||||||
Serial.print(file.name());
|
|
||||||
time_t t= file.getLastWrite();
|
|
||||||
struct tm * tmstruct = localtime(&t);
|
|
||||||
Serial.printf(" LAST WRITE: %d-%02d-%02d %02d:%02d:%02d\n",(tmstruct->tm_year)+1900,( tmstruct->tm_mon)+1, tmstruct->tm_mday,tmstruct->tm_hour , tmstruct->tm_min, tmstruct->tm_sec);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if(levels){
|
|
||||||
listDir(fs, file.name(), levels -1);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Serial.print(" FILE: ");
|
|
||||||
Serial.print(file.name());
|
|
||||||
Serial.print(" SIZE: ");
|
|
||||||
|
|
||||||
#ifdef CONFIG_LITTLEFS_FOR_IDF_3_2
|
|
||||||
Serial.println(file.size());
|
|
||||||
#else
|
|
||||||
Serial.print(file.size());
|
|
||||||
time_t t= file.getLastWrite();
|
|
||||||
struct tm * tmstruct = localtime(&t);
|
|
||||||
Serial.printf(" LAST WRITE: %d-%02d-%02d %02d:%02d:%02d\n",(tmstruct->tm_year)+1900,( tmstruct->tm_mon)+1, tmstruct->tm_mday,tmstruct->tm_hour , tmstruct->tm_min, tmstruct->tm_sec);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
file = root.openNextFile();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void createDir(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Creating Dir: %s\n", path);
|
|
||||||
if(fs.mkdir(path)){
|
|
||||||
Serial.println("Dir created");
|
|
||||||
} else {
|
|
||||||
Serial.println("mkdir failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void removeDir(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Removing Dir: %s\n", path);
|
|
||||||
if(fs.rmdir(path)){
|
|
||||||
Serial.println("Dir removed");
|
|
||||||
} else {
|
|
||||||
Serial.println("rmdir failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void readFile(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Reading file: %s\r\n", path);
|
|
||||||
|
|
||||||
File file = fs.open(path);
|
|
||||||
if(!file || file.isDirectory()){
|
|
||||||
Serial.println("- failed to open file for reading");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Serial.println("- read from file:");
|
|
||||||
while(file.available()){
|
|
||||||
Serial.write(file.read());
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void writeFile(fs::FS &fs, const char * path, const char * message){
|
|
||||||
Serial.printf("Writing file: %s\r\n", path);
|
|
||||||
|
|
||||||
File file = fs.open(path, FILE_WRITE);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("- failed to open file for writing");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(file.print(message)){
|
|
||||||
Serial.println("- file written");
|
|
||||||
} else {
|
|
||||||
Serial.println("- write failed");
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void appendFile(fs::FS &fs, const char * path, const char * message){
|
|
||||||
Serial.printf("Appending to file: %s\r\n", path);
|
|
||||||
|
|
||||||
File file = fs.open(path, FILE_APPEND);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("- failed to open file for appending");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(file.print(message)){
|
|
||||||
Serial.println("- message appended");
|
|
||||||
} else {
|
|
||||||
Serial.println("- append failed");
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void renameFile(fs::FS &fs, const char * path1, const char * path2){
|
|
||||||
Serial.printf("Renaming file %s to %s\r\n", path1, path2);
|
|
||||||
if (fs.rename(path1, path2)) {
|
|
||||||
Serial.println("- file renamed");
|
|
||||||
} else {
|
|
||||||
Serial.println("- rename failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void deleteFile(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Deleting file: %s\r\n", path);
|
|
||||||
if(fs.remove(path)){
|
|
||||||
Serial.println("- file deleted");
|
|
||||||
} else {
|
|
||||||
Serial.println("- delete failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SPIFFS-like write and delete file, better use #define CONFIG_LITTLEFS_SPIFFS_COMPAT 1
|
|
||||||
|
|
||||||
void writeFile2(fs::FS &fs, const char * path, const char * message){
|
|
||||||
if(!fs.exists(path)){
|
|
||||||
if (strchr(path, '/')) {
|
|
||||||
Serial.printf("Create missing folders of: %s\r\n", path);
|
|
||||||
char *pathStr = strdup(path);
|
|
||||||
if (pathStr) {
|
|
||||||
char *ptr = strchr(pathStr, '/');
|
|
||||||
while (ptr) {
|
|
||||||
*ptr = 0;
|
|
||||||
fs.mkdir(pathStr);
|
|
||||||
*ptr = '/';
|
|
||||||
ptr = strchr(ptr+1, '/');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
free(pathStr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Serial.printf("Writing file to: %s\r\n", path);
|
|
||||||
File file = fs.open(path, FILE_WRITE);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("- failed to open file for writing");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(file.print(message)){
|
|
||||||
Serial.println("- file written");
|
|
||||||
} else {
|
|
||||||
Serial.println("- write failed");
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void deleteFile2(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Deleting file and empty folders on path: %s\r\n", path);
|
|
||||||
|
|
||||||
if(fs.remove(path)){
|
|
||||||
Serial.println("- file deleted");
|
|
||||||
} else {
|
|
||||||
Serial.println("- delete failed");
|
|
||||||
}
|
|
||||||
|
|
||||||
char *pathStr = strdup(path);
|
|
||||||
if (pathStr) {
|
|
||||||
char *ptr = strrchr(pathStr, '/');
|
|
||||||
if (ptr) {
|
|
||||||
Serial.printf("Removing all empty folders on path: %s\r\n", path);
|
|
||||||
}
|
|
||||||
while (ptr) {
|
|
||||||
*ptr = 0;
|
|
||||||
fs.rmdir(pathStr);
|
|
||||||
ptr = strrchr(pathStr, '/');
|
|
||||||
}
|
|
||||||
free(pathStr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void testFileIO(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Testing file I/O with %s\r\n", path);
|
|
||||||
|
|
||||||
static uint8_t buf[512];
|
|
||||||
size_t len = 0;
|
|
||||||
File file = fs.open(path, FILE_WRITE);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("- failed to open file for writing");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t i;
|
|
||||||
Serial.print("- writing" );
|
|
||||||
uint32_t start = millis();
|
|
||||||
for(i=0; i<2048; i++){
|
|
||||||
if ((i & 0x001F) == 0x001F){
|
|
||||||
Serial.print(".");
|
|
||||||
}
|
|
||||||
file.write(buf, 512);
|
|
||||||
}
|
|
||||||
Serial.println("");
|
|
||||||
uint32_t end = millis() - start;
|
|
||||||
Serial.printf(" - %u bytes written in %u ms\r\n", 2048 * 512, end);
|
|
||||||
file.close();
|
|
||||||
|
|
||||||
file = fs.open(path);
|
|
||||||
start = millis();
|
|
||||||
end = start;
|
|
||||||
i = 0;
|
|
||||||
if(file && !file.isDirectory()){
|
|
||||||
len = file.size();
|
|
||||||
size_t flen = len;
|
|
||||||
start = millis();
|
|
||||||
Serial.print("- reading" );
|
|
||||||
while(len){
|
|
||||||
size_t toRead = len;
|
|
||||||
if(toRead > 512){
|
|
||||||
toRead = 512;
|
|
||||||
}
|
|
||||||
file.read(buf, toRead);
|
|
||||||
if ((i++ & 0x001F) == 0x001F){
|
|
||||||
Serial.print(".");
|
|
||||||
}
|
|
||||||
len -= toRead;
|
|
||||||
}
|
|
||||||
Serial.println("");
|
|
||||||
end = millis() - start;
|
|
||||||
Serial.printf("- %u bytes read in %u ms\r\n", flen, end);
|
|
||||||
file.close();
|
|
||||||
} else {
|
|
||||||
Serial.println("- failed to open file for reading");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setup(){
|
|
||||||
Serial.begin(115200);
|
|
||||||
if(!LITTLEFS.begin(FORMAT_LITTLEFS_IF_FAILED)){
|
|
||||||
Serial.println("LITTLEFS Mount Failed");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
listDir(LITTLEFS, "/", 0);
|
|
||||||
createDir(LITTLEFS, "/mydir");
|
|
||||||
writeFile(LITTLEFS, "/mydir/hello2.txt", "Hello2");
|
|
||||||
//writeFile(LITTLEFS, "/mydir/newdir2/newdir3/hello3.txt", "Hello3");
|
|
||||||
writeFile2(LITTLEFS, "/mydir/newdir2/newdir3/hello3.txt", "Hello3");
|
|
||||||
listDir(LITTLEFS, "/", 3);
|
|
||||||
deleteFile(LITTLEFS, "/mydir/hello2.txt");
|
|
||||||
//deleteFile(LITTLEFS, "/mydir/newdir2/newdir3/hello3.txt");
|
|
||||||
deleteFile2(LITTLEFS, "/mydir/newdir2/newdir3/hello3.txt");
|
|
||||||
removeDir(LITTLEFS, "/mydir");
|
|
||||||
listDir(LITTLEFS, "/", 3);
|
|
||||||
writeFile(LITTLEFS, "/hello.txt", "Hello ");
|
|
||||||
appendFile(LITTLEFS, "/hello.txt", "World!\r\n");
|
|
||||||
readFile(LITTLEFS, "/hello.txt");
|
|
||||||
renameFile(LITTLEFS, "/hello.txt", "/foo.txt");
|
|
||||||
readFile(LITTLEFS, "/foo.txt");
|
|
||||||
deleteFile(LITTLEFS, "/foo.txt");
|
|
||||||
testFileIO(LITTLEFS, "/test.txt");
|
|
||||||
deleteFile(LITTLEFS, "/test.txt");
|
|
||||||
|
|
||||||
Serial.println( "Test complete" );
|
|
||||||
}
|
|
||||||
|
|
||||||
void loop(){
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,219 +0,0 @@
|
||||||
#include "FS.h"
|
|
||||||
//#include "SPIFFS.h"
|
|
||||||
#include "LITTLEFS.h"
|
|
||||||
#include <time.h>
|
|
||||||
#include <WiFi.h>
|
|
||||||
|
|
||||||
#define SPIFFS LITTLEFS
|
|
||||||
|
|
||||||
/* This examples uses "quick re-define" of SPIFFS to run
|
|
||||||
an existing sketch with LITTLEFS instead of SPIFFS
|
|
||||||
|
|
||||||
To get time/date stamps by file.getLastWrite(), you need an
|
|
||||||
esp32 core on IDF 3.3 and comment a line in file esp_littlefs.c:
|
|
||||||
|
|
||||||
//#define CONFIG_LITTLEFS_FOR_IDF_3_2
|
|
||||||
|
|
||||||
You only need to format LITTLEFS the first time you run a
|
|
||||||
test or else use the LITTLEFS plugin to create a partition
|
|
||||||
https://github.com/lorol/arduino-esp32littlefs-plugin */
|
|
||||||
|
|
||||||
#define FORMAT_LITTLEFS_IF_FAILED true
|
|
||||||
|
|
||||||
const char* ssid = "yourssid";
|
|
||||||
const char* password = "yourpass";
|
|
||||||
|
|
||||||
long timezone = 1;
|
|
||||||
byte daysavetime = 1;
|
|
||||||
|
|
||||||
void listDir(fs::FS &fs, const char * dirname, uint8_t levels){
|
|
||||||
Serial.printf("Listing directory: %s\n", dirname);
|
|
||||||
|
|
||||||
File root = fs.open(dirname);
|
|
||||||
if(!root){
|
|
||||||
Serial.println("Failed to open directory");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(!root.isDirectory()){
|
|
||||||
Serial.println("Not a directory");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
File file = root.openNextFile();
|
|
||||||
while(file){
|
|
||||||
if(file.isDirectory()){
|
|
||||||
Serial.print(" DIR : ");
|
|
||||||
Serial.print (file.name());
|
|
||||||
time_t t= file.getLastWrite();
|
|
||||||
struct tm * tmstruct = localtime(&t);
|
|
||||||
Serial.printf(" LAST WRITE: %d-%02d-%02d %02d:%02d:%02d\n",(tmstruct->tm_year)+1900,( tmstruct->tm_mon)+1, tmstruct->tm_mday,tmstruct->tm_hour , tmstruct->tm_min, tmstruct->tm_sec);
|
|
||||||
if(levels){
|
|
||||||
listDir(fs, file.name(), levels -1);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Serial.print(" FILE: ");
|
|
||||||
Serial.print(file.name());
|
|
||||||
Serial.print(" SIZE: ");
|
|
||||||
Serial.print(file.size());
|
|
||||||
time_t t= file.getLastWrite();
|
|
||||||
struct tm * tmstruct = localtime(&t);
|
|
||||||
Serial.printf(" LAST WRITE: %d-%02d-%02d %02d:%02d:%02d\n",(tmstruct->tm_year)+1900,( tmstruct->tm_mon)+1, tmstruct->tm_mday,tmstruct->tm_hour , tmstruct->tm_min, tmstruct->tm_sec);
|
|
||||||
}
|
|
||||||
file = root.openNextFile();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void createDir(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Creating Dir: %s\n", path);
|
|
||||||
if(fs.mkdir(path)){
|
|
||||||
Serial.println("Dir created");
|
|
||||||
} else {
|
|
||||||
Serial.println("mkdir failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void removeDir(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Removing Dir: %s\n", path);
|
|
||||||
if(fs.rmdir(path)){
|
|
||||||
Serial.println("Dir removed");
|
|
||||||
} else {
|
|
||||||
Serial.println("rmdir failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void readFile(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Reading file: %s\n", path);
|
|
||||||
|
|
||||||
File file = fs.open(path);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("Failed to open file for reading");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Serial.print("Read from file: ");
|
|
||||||
while(file.available()){
|
|
||||||
Serial.write(file.read());
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void writeFile(fs::FS &fs, const char * path, const char * message){
|
|
||||||
Serial.printf("Writing file: %s\n", path);
|
|
||||||
|
|
||||||
File file = fs.open(path, FILE_WRITE);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("Failed to open file for writing");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(file.print(message)){
|
|
||||||
Serial.println("File written");
|
|
||||||
} else {
|
|
||||||
Serial.println("Write failed");
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void appendFile(fs::FS &fs, const char * path, const char * message){
|
|
||||||
Serial.printf("Appending to file: %s\n", path);
|
|
||||||
|
|
||||||
File file = fs.open(path, FILE_APPEND);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("Failed to open file for appending");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(file.print(message)){
|
|
||||||
Serial.println("Message appended");
|
|
||||||
} else {
|
|
||||||
Serial.println("Append failed");
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void renameFile(fs::FS &fs, const char * path1, const char * path2){
|
|
||||||
Serial.printf("Renaming file %s to %s\n", path1, path2);
|
|
||||||
if (fs.rename(path1, path2)) {
|
|
||||||
Serial.println("File renamed");
|
|
||||||
} else {
|
|
||||||
Serial.println("Rename failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void deleteFile(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Deleting file: %s\n", path);
|
|
||||||
if(fs.remove(path)){
|
|
||||||
Serial.println("File deleted");
|
|
||||||
} else {
|
|
||||||
Serial.println("Delete failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setup(){
|
|
||||||
Serial.begin(115200);
|
|
||||||
// We start by connecting to a WiFi network
|
|
||||||
Serial.println();
|
|
||||||
Serial.println();
|
|
||||||
Serial.print("Connecting to ");
|
|
||||||
Serial.println(ssid);
|
|
||||||
|
|
||||||
WiFi.begin(ssid, password);
|
|
||||||
|
|
||||||
while (WiFi.status() != WL_CONNECTED) {
|
|
||||||
delay(500);
|
|
||||||
Serial.print(".");
|
|
||||||
}
|
|
||||||
Serial.println("WiFi connected");
|
|
||||||
Serial.println("IP address: ");
|
|
||||||
Serial.println(WiFi.localIP());
|
|
||||||
Serial.println("Contacting Time Server");
|
|
||||||
configTime(3600*timezone, daysavetime*3600, "time.nist.gov", "0.pool.ntp.org", "1.pool.ntp.org");
|
|
||||||
struct tm tmstruct ;
|
|
||||||
delay(2000);
|
|
||||||
tmstruct.tm_year = 0;
|
|
||||||
getLocalTime(&tmstruct, 5000);
|
|
||||||
Serial.printf("\nNow is : %d-%02d-%02d %02d:%02d:%02d\n",(tmstruct.tm_year)+1900,( tmstruct.tm_mon)+1, tmstruct.tm_mday,tmstruct.tm_hour , tmstruct.tm_min, tmstruct.tm_sec);
|
|
||||||
Serial.println("");
|
|
||||||
|
|
||||||
if(!SPIFFS.begin(FORMAT_LITTLEFS_IF_FAILED)){
|
|
||||||
Serial.println("LITTLEFS Mount Failed");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Serial.println("----list 1----");
|
|
||||||
listDir(SPIFFS, "/", 1);
|
|
||||||
|
|
||||||
Serial.println("----remove old dir----");
|
|
||||||
removeDir(SPIFFS, "/mydir");
|
|
||||||
|
|
||||||
Serial.println("----create a new dir----");
|
|
||||||
createDir(SPIFFS, "/mydir");
|
|
||||||
|
|
||||||
Serial.println("----remove the new dir----");
|
|
||||||
removeDir(SPIFFS, "/mydir");
|
|
||||||
|
|
||||||
Serial.println("----create the new again----");
|
|
||||||
createDir(SPIFFS, "/mydir");
|
|
||||||
|
|
||||||
Serial.println("----create and work with file----");
|
|
||||||
writeFile(SPIFFS, "/mydir/hello.txt", "Hello ");
|
|
||||||
appendFile(SPIFFS, "/mydir/hello.txt", "World!\n");
|
|
||||||
|
|
||||||
Serial.println("----list 2----");
|
|
||||||
listDir(SPIFFS, "/", 1);
|
|
||||||
|
|
||||||
Serial.println("----attempt to remove dir w/ file----");
|
|
||||||
removeDir(SPIFFS, "/mydir");
|
|
||||||
|
|
||||||
Serial.println("----remove dir after deleting file----");
|
|
||||||
deleteFile(SPIFFS, "/mydir/hello.txt");
|
|
||||||
removeDir(SPIFFS, "/mydir");
|
|
||||||
|
|
||||||
Serial.println("----list 3----");
|
|
||||||
listDir(SPIFFS, "/", 1);
|
|
||||||
|
|
||||||
Serial.println( "Test complete" );
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
void loop(){
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,270 +0,0 @@
|
||||||
#include <Arduino.h>
|
|
||||||
#include "FS.h"
|
|
||||||
#include <LITTLEFS.h>
|
|
||||||
|
|
||||||
/* You only need to format LITTLEFS the first time you run a
|
|
||||||
test or else use the LITTLEFS plugin to create a partition
|
|
||||||
https://github.com/lorol/arduino-esp32littlefs-plugin */
|
|
||||||
|
|
||||||
#define FORMAT_LITTLEFS_IF_FAILED true
|
|
||||||
|
|
||||||
void listDir(fs::FS &fs, const char * dirname, uint8_t levels){
|
|
||||||
Serial.printf("Listing directory: %s\r\n", dirname);
|
|
||||||
|
|
||||||
File root = fs.open(dirname);
|
|
||||||
if(!root){
|
|
||||||
Serial.println("- failed to open directory");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(!root.isDirectory()){
|
|
||||||
Serial.println(" - not a directory");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
File file = root.openNextFile();
|
|
||||||
while(file){
|
|
||||||
if(file.isDirectory()){
|
|
||||||
Serial.print(" DIR : ");
|
|
||||||
Serial.println(file.name());
|
|
||||||
if(levels){
|
|
||||||
listDir(fs, file.name(), levels -1);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Serial.print(" FILE: ");
|
|
||||||
Serial.print(file.name());
|
|
||||||
Serial.print("\tSIZE: ");
|
|
||||||
Serial.println(file.size());
|
|
||||||
}
|
|
||||||
file = root.openNextFile();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void createDir(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Creating Dir: %s\n", path);
|
|
||||||
if(fs.mkdir(path)){
|
|
||||||
Serial.println("Dir created");
|
|
||||||
} else {
|
|
||||||
Serial.println("mkdir failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void removeDir(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Removing Dir: %s\n", path);
|
|
||||||
if(fs.rmdir(path)){
|
|
||||||
Serial.println("Dir removed");
|
|
||||||
} else {
|
|
||||||
Serial.println("rmdir failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void readFile(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Reading file: %s\r\n", path);
|
|
||||||
|
|
||||||
File file = fs.open(path);
|
|
||||||
if(!file || file.isDirectory()){
|
|
||||||
Serial.println("- failed to open file for reading");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Serial.println("- read from file:");
|
|
||||||
while(file.available()){
|
|
||||||
Serial.write(file.read());
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void writeFile(fs::FS &fs, const char * path, const char * message){
|
|
||||||
Serial.printf("Writing file: %s\r\n", path);
|
|
||||||
|
|
||||||
File file = fs.open(path, FILE_WRITE);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("- failed to open file for writing");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(file.print(message)){
|
|
||||||
Serial.println("- file written");
|
|
||||||
} else {
|
|
||||||
Serial.println("- write failed");
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void appendFile(fs::FS &fs, const char * path, const char * message){
|
|
||||||
Serial.printf("Appending to file: %s\r\n", path);
|
|
||||||
|
|
||||||
File file = fs.open(path, FILE_APPEND);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("- failed to open file for appending");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(file.print(message)){
|
|
||||||
Serial.println("- message appended");
|
|
||||||
} else {
|
|
||||||
Serial.println("- append failed");
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void renameFile(fs::FS &fs, const char * path1, const char * path2){
|
|
||||||
Serial.printf("Renaming file %s to %s\r\n", path1, path2);
|
|
||||||
if (fs.rename(path1, path2)) {
|
|
||||||
Serial.println("- file renamed");
|
|
||||||
} else {
|
|
||||||
Serial.println("- rename failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void deleteFile(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Deleting file: %s\r\n", path);
|
|
||||||
if(fs.remove(path)){
|
|
||||||
Serial.println("- file deleted");
|
|
||||||
} else {
|
|
||||||
Serial.println("- delete failed");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SPIFFS-like write and delete file, better use #define CONFIG_LITTLEFS_SPIFFS_COMPAT 1
|
|
||||||
|
|
||||||
void writeFile2(fs::FS &fs, const char * path, const char * message){
|
|
||||||
if(!fs.exists(path)){
|
|
||||||
if (strchr(path, '/')) {
|
|
||||||
Serial.printf("Create missing folders of: %s\r\n", path);
|
|
||||||
char *pathStr = strdup(path);
|
|
||||||
if (pathStr) {
|
|
||||||
char *ptr = strchr(pathStr, '/');
|
|
||||||
while (ptr) {
|
|
||||||
*ptr = 0;
|
|
||||||
fs.mkdir(pathStr);
|
|
||||||
*ptr = '/';
|
|
||||||
ptr = strchr(ptr+1, '/');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
free(pathStr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Serial.printf("Writing file to: %s\r\n", path);
|
|
||||||
File file = fs.open(path, FILE_WRITE);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("- failed to open file for writing");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(file.print(message)){
|
|
||||||
Serial.println("- file written");
|
|
||||||
} else {
|
|
||||||
Serial.println("- write failed");
|
|
||||||
}
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void deleteFile2(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Deleting file and empty folders on path: %s\r\n", path);
|
|
||||||
|
|
||||||
if(fs.remove(path)){
|
|
||||||
Serial.println("- file deleted");
|
|
||||||
} else {
|
|
||||||
Serial.println("- delete failed");
|
|
||||||
}
|
|
||||||
|
|
||||||
char *pathStr = strdup(path);
|
|
||||||
if (pathStr) {
|
|
||||||
char *ptr = strrchr(pathStr, '/');
|
|
||||||
if (ptr) {
|
|
||||||
Serial.printf("Removing all empty folders on path: %s\r\n", path);
|
|
||||||
}
|
|
||||||
while (ptr) {
|
|
||||||
*ptr = 0;
|
|
||||||
fs.rmdir(pathStr);
|
|
||||||
ptr = strrchr(pathStr, '/');
|
|
||||||
}
|
|
||||||
free(pathStr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void testFileIO(fs::FS &fs, const char * path){
|
|
||||||
Serial.printf("Testing file I/O with %s\r\n", path);
|
|
||||||
|
|
||||||
static uint8_t buf[512];
|
|
||||||
size_t len = 0;
|
|
||||||
File file = fs.open(path, FILE_WRITE);
|
|
||||||
if(!file){
|
|
||||||
Serial.println("- failed to open file for writing");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t i;
|
|
||||||
Serial.print("- writing" );
|
|
||||||
uint32_t start = millis();
|
|
||||||
for(i=0; i<2048; i++){
|
|
||||||
if ((i & 0x001F) == 0x001F){
|
|
||||||
Serial.print(".");
|
|
||||||
}
|
|
||||||
file.write(buf, 512);
|
|
||||||
}
|
|
||||||
Serial.println("");
|
|
||||||
uint32_t end = millis() - start;
|
|
||||||
Serial.printf(" - %u bytes written in %u ms\r\n", 2048 * 512, end);
|
|
||||||
file.close();
|
|
||||||
|
|
||||||
file = fs.open(path);
|
|
||||||
start = millis();
|
|
||||||
end = start;
|
|
||||||
i = 0;
|
|
||||||
if(file && !file.isDirectory()){
|
|
||||||
len = file.size();
|
|
||||||
size_t flen = len;
|
|
||||||
start = millis();
|
|
||||||
Serial.print("- reading" );
|
|
||||||
while(len){
|
|
||||||
size_t toRead = len;
|
|
||||||
if(toRead > 512){
|
|
||||||
toRead = 512;
|
|
||||||
}
|
|
||||||
file.read(buf, toRead);
|
|
||||||
if ((i++ & 0x001F) == 0x001F){
|
|
||||||
Serial.print(".");
|
|
||||||
}
|
|
||||||
len -= toRead;
|
|
||||||
}
|
|
||||||
Serial.println("");
|
|
||||||
end = millis() - start;
|
|
||||||
Serial.printf("- %u bytes read in %u ms\r\n", flen, end);
|
|
||||||
file.close();
|
|
||||||
} else {
|
|
||||||
Serial.println("- failed to open file for reading");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void setup(){
|
|
||||||
Serial.begin(115200);
|
|
||||||
if(!LITTLEFS.begin(FORMAT_LITTLEFS_IF_FAILED)){
|
|
||||||
Serial.println("LITTLEFS Mount Failed");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
Serial.println( "SPIFFS-like write file to new path and delete it w/folders" );
|
|
||||||
writeFile2(LITTLEFS, "/new1/new2/new3/hello3.txt", "Hello3");
|
|
||||||
listDir(LITTLEFS, "/", 3);
|
|
||||||
deleteFile2(LITTLEFS, "/new1/new2/new3/hello3.txt");
|
|
||||||
|
|
||||||
listDir(LITTLEFS, "/", 3);
|
|
||||||
createDir(LITTLEFS, "/mydir");
|
|
||||||
writeFile(LITTLEFS, "/mydir/hello2.txt", "Hello2");
|
|
||||||
listDir(LITTLEFS, "/", 1);
|
|
||||||
deleteFile(LITTLEFS, "/mydir/hello2.txt");
|
|
||||||
removeDir(LITTLEFS, "/mydir");
|
|
||||||
listDir(LITTLEFS, "/", 1);
|
|
||||||
writeFile(LITTLEFS, "/hello.txt", "Hello ");
|
|
||||||
appendFile(LITTLEFS, "/hello.txt", "World!\r\n");
|
|
||||||
readFile(LITTLEFS, "/hello.txt");
|
|
||||||
renameFile(LITTLEFS, "/hello.txt", "/foo.txt");
|
|
||||||
readFile(LITTLEFS, "/foo.txt");
|
|
||||||
deleteFile(LITTLEFS, "/foo.txt");
|
|
||||||
testFileIO(LITTLEFS, "/test.txt");
|
|
||||||
deleteFile(LITTLEFS, "/test.txt");
|
|
||||||
|
|
||||||
Serial.println( "Test complete" );
|
|
||||||
}
|
|
||||||
|
|
||||||
void loop(){
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,19 +0,0 @@
|
||||||
{
|
|
||||||
"name":"LittleFS_esp32",
|
|
||||||
"description":"LittleFS for esp32",
|
|
||||||
"keywords":"littlefs, spiffs",
|
|
||||||
"authors":
|
|
||||||
{
|
|
||||||
"name": "lorol",
|
|
||||||
"maintainer": true
|
|
||||||
},
|
|
||||||
"repository":
|
|
||||||
{
|
|
||||||
"type": "git",
|
|
||||||
"url": "https://github.com/lorol/LITTLEFS.git"
|
|
||||||
},
|
|
||||||
"version": "1.0.5",
|
|
||||||
"license": "LGPL-2.0",
|
|
||||||
"frameworks": "arduino",
|
|
||||||
"platforms": "espressif32"
|
|
||||||
}
|
|
|
@ -1,9 +0,0 @@
|
||||||
name=LittleFS_esp32
|
|
||||||
version=1.0.5
|
|
||||||
author=lorol
|
|
||||||
maintainer=lorol
|
|
||||||
sentence=LittleFS for esp32 based on esp_littlefs IDF component. Use esp32 core-provided LITTLEFS library instead of this one when available in future core releases.
|
|
||||||
paragraph= For esp32 core 1.0.4 release, use #define CONFIG_LITTLEFS_FOR_IDF_3_2 and for more SPIFFS compatibility, set #define CONFIG_LITTLEFS_SPIFFS_COMPAT 1
|
|
||||||
category=Data Storage
|
|
||||||
url=https://github.com/lorol/LITTLEFS
|
|
||||||
architectures=esp32
|
|
|
@ -1,7 +0,0 @@
|
||||||
Copyright 2020 Brian Pugh
|
|
||||||
|
|
||||||
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.
|
|
|
@ -1,24 +0,0 @@
|
||||||
Copyright (c) 2017, Arm Limited. All rights reserved.
|
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without modification,
|
|
||||||
are permitted provided that the following conditions are met:
|
|
||||||
|
|
||||||
- Redistributions of source code must retain the above copyright notice, this
|
|
||||||
list of conditions and the following disclaimer.
|
|
||||||
- Redistributions in binary form must reproduce the above copyright notice, this
|
|
||||||
list of conditions and the following disclaimer in the documentation and/or
|
|
||||||
other materials provided with the distribution.
|
|
||||||
- Neither the name of ARM nor the names of its contributors may be used to
|
|
||||||
endorse or promote products derived from this software without specific prior
|
|
||||||
written permission.
|
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
||||||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
|
|
||||||
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
||||||
ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
@ -1,105 +0,0 @@
|
||||||
// Copyright 2015-2020 Espressif Systems (Shanghai) PTE LTD
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
|
|
||||||
static constexpr const char LFS_NAME[] = "spiffs";
|
|
||||||
|
|
||||||
#include "vfs_api.h"
|
|
||||||
|
|
||||||
extern "C" {
|
|
||||||
#include <sys/unistd.h>
|
|
||||||
#include <sys/stat.h>
|
|
||||||
#include <dirent.h>
|
|
||||||
#include "esp_littlefs.h"
|
|
||||||
}
|
|
||||||
|
|
||||||
#include "LITTLEFS.h"
|
|
||||||
|
|
||||||
using namespace fs;
|
|
||||||
|
|
||||||
LITTLEFSFS::LITTLEFSFS() : FS(FSImplPtr(new VFSImpl()))
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
bool LITTLEFSFS::begin(bool formatOnFail, const char * basePath, uint8_t maxOpenFiles)
|
|
||||||
{
|
|
||||||
if(esp_littlefs_mounted(LFS_NAME)){
|
|
||||||
log_w("LITTLEFS Already Mounted!");
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
esp_vfs_littlefs_conf_t conf = {
|
|
||||||
.base_path = basePath,
|
|
||||||
.partition_label = LFS_NAME,
|
|
||||||
.format_if_mount_failed = false
|
|
||||||
};
|
|
||||||
|
|
||||||
esp_err_t err = esp_vfs_littlefs_register(&conf);
|
|
||||||
if(err == ESP_FAIL && formatOnFail){
|
|
||||||
if(format()){
|
|
||||||
err = esp_vfs_littlefs_register(&conf);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if(err != ESP_OK){
|
|
||||||
log_e("Mounting LITTLEFS failed! Error: %d", err);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
_impl->mountpoint(basePath);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void LITTLEFSFS::end()
|
|
||||||
{
|
|
||||||
if(esp_littlefs_mounted(LFS_NAME)){
|
|
||||||
esp_err_t err = esp_vfs_littlefs_unregister(LFS_NAME);
|
|
||||||
if(err){
|
|
||||||
log_e("Unmounting LITTLEFS failed! Error: %d", err);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
_impl->mountpoint(NULL);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool LITTLEFSFS::format()
|
|
||||||
{
|
|
||||||
disableCore0WDT();
|
|
||||||
esp_err_t err = esp_littlefs_format(LFS_NAME);
|
|
||||||
enableCore0WDT();
|
|
||||||
if(err){
|
|
||||||
log_e("Formatting LITTLEFS failed! Error: %d", err);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t LITTLEFSFS::totalBytes()
|
|
||||||
{
|
|
||||||
size_t total,used;
|
|
||||||
if(esp_littlefs_info(LFS_NAME, &total, &used)){
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
return total;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t LITTLEFSFS::usedBytes()
|
|
||||||
{
|
|
||||||
size_t total,used;
|
|
||||||
if(esp_littlefs_info(LFS_NAME, &total, &used)){
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
return used;
|
|
||||||
}
|
|
||||||
|
|
||||||
LITTLEFSFS LITTLEFS;
|
|
||||||
|
|
|
@ -1,38 +0,0 @@
|
||||||
// Copyright 2015-2020 Espressif Systems (Shanghai) PTE LTD
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
#ifndef _LITTLEFS_H_
|
|
||||||
#define _LITTLEFS_H_
|
|
||||||
|
|
||||||
#include "FS.h"
|
|
||||||
|
|
||||||
namespace fs
|
|
||||||
{
|
|
||||||
|
|
||||||
class LITTLEFSFS : public FS
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
LITTLEFSFS();
|
|
||||||
bool begin(bool formatOnFail=false, const char * basePath="/littlefs", uint8_t maxOpenFiles=5);
|
|
||||||
bool format();
|
|
||||||
size_t totalBytes();
|
|
||||||
size_t usedBytes();
|
|
||||||
void end();
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
extern fs::LITTLEFSFS LITTLEFS;
|
|
||||||
|
|
||||||
|
|
||||||
#endif
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,118 +0,0 @@
|
||||||
#ifndef ESP_LITTLEFS_H__
|
|
||||||
#define ESP_LITTLEFS_H__
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <stddef.h>
|
|
||||||
#include <stdarg.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
#include <utime.h>
|
|
||||||
#include "freertos/FreeRTOS.h"
|
|
||||||
#include "freertos/semphr.h"
|
|
||||||
#include "esp_err.h"
|
|
||||||
#include <sys/types.h>
|
|
||||||
#include <sys/reent.h>
|
|
||||||
#include <sys/stat.h>
|
|
||||||
#include <sys/time.h>
|
|
||||||
#include <sys/poll.h>
|
|
||||||
#include <dirent.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include "sdkconfig.h"
|
|
||||||
|
|
||||||
#include "lfs.h" //#include "littlefs/lfs.h"
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Last Modified Time
|
|
||||||
*
|
|
||||||
* Use 't' for LITTLEFS_ATTR_MTIME to match example:
|
|
||||||
* https://github.com/ARMmbed/littlefs/issues/23#issuecomment-482293539
|
|
||||||
* And to match other external tools such as:
|
|
||||||
* https://github.com/earlephilhower/mklittlefs
|
|
||||||
*/
|
|
||||||
#define LITTLEFS_ATTR_MTIME ((uint8_t) 't')
|
|
||||||
|
|
||||||
/**
|
|
||||||
*Configuration structure for esp_vfs_littlefs_register.
|
|
||||||
*/
|
|
||||||
typedef struct {
|
|
||||||
const char *base_path; /**< Mounting point. */
|
|
||||||
const char *partition_label; /**< Label of partition to use. */
|
|
||||||
uint8_t format_if_mount_failed:1; /**< Format the file system if it fails to mount. */
|
|
||||||
uint8_t dont_mount:1; /**< Don't attempt to mount or format. Overrides format_if_mount_failed */
|
|
||||||
} esp_vfs_littlefs_conf_t;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Register and mount littlefs to VFS with given path prefix.
|
|
||||||
*
|
|
||||||
* @param conf Pointer to esp_vfs_littlefs_conf_t configuration structure
|
|
||||||
*
|
|
||||||
* @return
|
|
||||||
* - ESP_OK if success
|
|
||||||
* - ESP_ERR_NO_MEM if objects could not be allocated
|
|
||||||
* - ESP_ERR_INVALID_STATE if already mounted or partition is encrypted
|
|
||||||
* - ESP_ERR_NOT_FOUND if partition for littlefs was not found
|
|
||||||
* - ESP_FAIL if mount or format fails
|
|
||||||
*/
|
|
||||||
esp_err_t esp_vfs_littlefs_register(const esp_vfs_littlefs_conf_t * conf);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Unregister and unmount littlefs from VFS
|
|
||||||
*
|
|
||||||
* @param partition_label Label of the partition to unregister.
|
|
||||||
*
|
|
||||||
* @return
|
|
||||||
* - ESP_OK if successful
|
|
||||||
* - ESP_ERR_INVALID_STATE already unregistered
|
|
||||||
*/
|
|
||||||
esp_err_t esp_vfs_littlefs_unregister(const char* partition_label);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if littlefs is mounted
|
|
||||||
*
|
|
||||||
* @param partition_label Label of the partition to check.
|
|
||||||
*
|
|
||||||
* @return
|
|
||||||
* - true if mounted
|
|
||||||
* - false if not mounted
|
|
||||||
*/
|
|
||||||
bool esp_littlefs_mounted(const char* partition_label);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Format the littlefs partition
|
|
||||||
*
|
|
||||||
* @param partition_label Label of the partition to format.
|
|
||||||
* @return
|
|
||||||
* - ESP_OK if successful
|
|
||||||
* - ESP_FAIL on error
|
|
||||||
*/
|
|
||||||
esp_err_t esp_littlefs_format(const char* partition_label);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get information for littlefs
|
|
||||||
*
|
|
||||||
* @param partition_label Optional, label of the partition to get info for.
|
|
||||||
* @param[out] total_bytes Size of the file system
|
|
||||||
* @param[out] used_bytes Current used bytes in the file system
|
|
||||||
*
|
|
||||||
* @return
|
|
||||||
* - ESP_OK if success
|
|
||||||
* - ESP_ERR_INVALID_STATE if not mounted
|
|
||||||
*/
|
|
||||||
esp_err_t esp_littlefs_info(const char* partition_label, size_t *total_bytes, size_t *used_bytes);
|
|
||||||
|
|
||||||
#if CONFIG_LITTLEFS_HUMAN_READABLE
|
|
||||||
/**
|
|
||||||
* @brief converts an enumerated lfs error into a string.
|
|
||||||
* @param lfs_errno The enumerated littlefs error.
|
|
||||||
*/
|
|
||||||
const char * esp_littlefs_errno(enum lfs_error lfs_errno);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
} // extern "C"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,655 +0,0 @@
|
||||||
/*
|
|
||||||
* The little filesystem
|
|
||||||
*
|
|
||||||
* Copyright (c) 2017, Arm Limited. All rights reserved.
|
|
||||||
* SPDX-License-Identifier: BSD-3-Clause
|
|
||||||
*/
|
|
||||||
#ifndef LFS_H
|
|
||||||
#define LFS_H
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <stdbool.h>
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
/// Version info ///
|
|
||||||
|
|
||||||
// Software library version
|
|
||||||
// Major (top-nibble), incremented on backwards incompatible changes
|
|
||||||
// Minor (bottom-nibble), incremented on feature additions
|
|
||||||
#define LFS_VERSION 0x00020002
|
|
||||||
#define LFS_VERSION_MAJOR (0xffff & (LFS_VERSION >> 16))
|
|
||||||
#define LFS_VERSION_MINOR (0xffff & (LFS_VERSION >> 0))
|
|
||||||
|
|
||||||
// Version of On-disk data structures
|
|
||||||
// Major (top-nibble), incremented on backwards incompatible changes
|
|
||||||
// Minor (bottom-nibble), incremented on feature additions
|
|
||||||
#define LFS_DISK_VERSION 0x00020000
|
|
||||||
#define LFS_DISK_VERSION_MAJOR (0xffff & (LFS_DISK_VERSION >> 16))
|
|
||||||
#define LFS_DISK_VERSION_MINOR (0xffff & (LFS_DISK_VERSION >> 0))
|
|
||||||
|
|
||||||
|
|
||||||
/// Definitions ///
|
|
||||||
|
|
||||||
// Type definitions
|
|
||||||
typedef uint32_t lfs_size_t;
|
|
||||||
typedef uint32_t lfs_off_t;
|
|
||||||
|
|
||||||
typedef int32_t lfs_ssize_t;
|
|
||||||
typedef int32_t lfs_soff_t;
|
|
||||||
|
|
||||||
typedef uint32_t lfs_block_t;
|
|
||||||
|
|
||||||
// Maximum name size in bytes, may be redefined to reduce the size of the
|
|
||||||
// info struct. Limited to <= 1022. Stored in superblock and must be
|
|
||||||
// respected by other littlefs drivers.
|
|
||||||
#ifndef LFS_NAME_MAX
|
|
||||||
#define LFS_NAME_MAX 255
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Maximum size of a file in bytes, may be redefined to limit to support other
|
|
||||||
// drivers. Limited on disk to <= 4294967296. However, above 2147483647 the
|
|
||||||
// functions lfs_file_seek, lfs_file_size, and lfs_file_tell will return
|
|
||||||
// incorrect values due to using signed integers. Stored in superblock and
|
|
||||||
// must be respected by other littlefs drivers.
|
|
||||||
#ifndef LFS_FILE_MAX
|
|
||||||
#define LFS_FILE_MAX 2147483647
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Maximum size of custom attributes in bytes, may be redefined, but there is
|
|
||||||
// no real benefit to using a smaller LFS_ATTR_MAX. Limited to <= 1022.
|
|
||||||
#ifndef LFS_ATTR_MAX
|
|
||||||
#define LFS_ATTR_MAX 1022
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Possible error codes, these are negative to allow
|
|
||||||
// valid positive return values
|
|
||||||
enum lfs_error {
|
|
||||||
LFS_ERR_OK = 0, // No error
|
|
||||||
LFS_ERR_IO = -5, // Error during device operation
|
|
||||||
LFS_ERR_CORRUPT = -84, // Corrupted
|
|
||||||
LFS_ERR_NOENT = -2, // No directory entry
|
|
||||||
LFS_ERR_EXIST = -17, // Entry already exists
|
|
||||||
LFS_ERR_NOTDIR = -20, // Entry is not a dir
|
|
||||||
LFS_ERR_ISDIR = -21, // Entry is a dir
|
|
||||||
LFS_ERR_NOTEMPTY = -39, // Dir is not empty
|
|
||||||
LFS_ERR_BADF = -9, // Bad file number
|
|
||||||
LFS_ERR_FBIG = -27, // File too large
|
|
||||||
LFS_ERR_INVAL = -22, // Invalid parameter
|
|
||||||
LFS_ERR_NOSPC = -28, // No space left on device
|
|
||||||
LFS_ERR_NOMEM = -12, // No more memory available
|
|
||||||
LFS_ERR_NOATTR = -61, // No data/attr available
|
|
||||||
LFS_ERR_NAMETOOLONG = -36, // File name too long
|
|
||||||
};
|
|
||||||
|
|
||||||
// File types
|
|
||||||
enum lfs_type {
|
|
||||||
// file types
|
|
||||||
LFS_TYPE_REG = 0x001,
|
|
||||||
LFS_TYPE_DIR = 0x002,
|
|
||||||
|
|
||||||
// internally used types
|
|
||||||
LFS_TYPE_SPLICE = 0x400,
|
|
||||||
LFS_TYPE_NAME = 0x000,
|
|
||||||
LFS_TYPE_STRUCT = 0x200,
|
|
||||||
LFS_TYPE_USERATTR = 0x300,
|
|
||||||
LFS_TYPE_FROM = 0x100,
|
|
||||||
LFS_TYPE_TAIL = 0x600,
|
|
||||||
LFS_TYPE_GLOBALS = 0x700,
|
|
||||||
LFS_TYPE_CRC = 0x500,
|
|
||||||
|
|
||||||
// internally used type specializations
|
|
||||||
LFS_TYPE_CREATE = 0x401,
|
|
||||||
LFS_TYPE_DELETE = 0x4ff,
|
|
||||||
LFS_TYPE_SUPERBLOCK = 0x0ff,
|
|
||||||
LFS_TYPE_DIRSTRUCT = 0x200,
|
|
||||||
LFS_TYPE_CTZSTRUCT = 0x202,
|
|
||||||
LFS_TYPE_INLINESTRUCT = 0x201,
|
|
||||||
LFS_TYPE_SOFTTAIL = 0x600,
|
|
||||||
LFS_TYPE_HARDTAIL = 0x601,
|
|
||||||
LFS_TYPE_MOVESTATE = 0x7ff,
|
|
||||||
|
|
||||||
// internal chip sources
|
|
||||||
LFS_FROM_NOOP = 0x000,
|
|
||||||
LFS_FROM_MOVE = 0x101,
|
|
||||||
LFS_FROM_USERATTRS = 0x102,
|
|
||||||
};
|
|
||||||
|
|
||||||
// File open flags
|
|
||||||
enum lfs_open_flags {
|
|
||||||
// open flags
|
|
||||||
LFS_O_RDONLY = 1, // Open a file as read only
|
|
||||||
LFS_O_WRONLY = 2, // Open a file as write only
|
|
||||||
LFS_O_RDWR = 3, // Open a file as read and write
|
|
||||||
LFS_O_CREAT = 0x0100, // Create a file if it does not exist
|
|
||||||
LFS_O_EXCL = 0x0200, // Fail if a file already exists
|
|
||||||
LFS_O_TRUNC = 0x0400, // Truncate the existing file to zero size
|
|
||||||
LFS_O_APPEND = 0x0800, // Move to end of file on every write
|
|
||||||
|
|
||||||
// internally used flags
|
|
||||||
LFS_F_DIRTY = 0x010000, // File does not match storage
|
|
||||||
LFS_F_WRITING = 0x020000, // File has been written since last flush
|
|
||||||
LFS_F_READING = 0x040000, // File has been read since last flush
|
|
||||||
LFS_F_ERRED = 0x080000, // An error occured during write
|
|
||||||
LFS_F_INLINE = 0x100000, // Currently inlined in directory entry
|
|
||||||
LFS_F_OPENED = 0x200000, // File has been opened
|
|
||||||
};
|
|
||||||
|
|
||||||
// File seek flags
|
|
||||||
enum lfs_whence_flags {
|
|
||||||
LFS_SEEK_SET = 0, // Seek relative to an absolute position
|
|
||||||
LFS_SEEK_CUR = 1, // Seek relative to the current file position
|
|
||||||
LFS_SEEK_END = 2, // Seek relative to the end of the file
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
// Configuration provided during initialization of the littlefs
|
|
||||||
struct lfs_config {
|
|
||||||
// Opaque user provided context that can be used to pass
|
|
||||||
// information to the block device operations
|
|
||||||
void *context;
|
|
||||||
|
|
||||||
// Read a region in a block. Negative error codes are propogated
|
|
||||||
// to the user.
|
|
||||||
int (*read)(const struct lfs_config *c, lfs_block_t block,
|
|
||||||
lfs_off_t off, void *buffer, lfs_size_t size);
|
|
||||||
|
|
||||||
// Program a region in a block. The block must have previously
|
|
||||||
// been erased. Negative error codes are propogated to the user.
|
|
||||||
// May return LFS_ERR_CORRUPT if the block should be considered bad.
|
|
||||||
int (*prog)(const struct lfs_config *c, lfs_block_t block,
|
|
||||||
lfs_off_t off, const void *buffer, lfs_size_t size);
|
|
||||||
|
|
||||||
// Erase a block. A block must be erased before being programmed.
|
|
||||||
// The state of an erased block is undefined. Negative error codes
|
|
||||||
// are propogated to the user.
|
|
||||||
// May return LFS_ERR_CORRUPT if the block should be considered bad.
|
|
||||||
int (*erase)(const struct lfs_config *c, lfs_block_t block);
|
|
||||||
|
|
||||||
// Sync the state of the underlying block device. Negative error codes
|
|
||||||
// are propogated to the user.
|
|
||||||
int (*sync)(const struct lfs_config *c);
|
|
||||||
|
|
||||||
// Minimum size of a block read. All read operations will be a
|
|
||||||
// multiple of this value.
|
|
||||||
lfs_size_t read_size;
|
|
||||||
|
|
||||||
// Minimum size of a block program. All program operations will be a
|
|
||||||
// multiple of this value.
|
|
||||||
lfs_size_t prog_size;
|
|
||||||
|
|
||||||
// Size of an erasable block. This does not impact ram consumption and
|
|
||||||
// may be larger than the physical erase size. However, non-inlined files
|
|
||||||
// take up at minimum one block. Must be a multiple of the read
|
|
||||||
// and program sizes.
|
|
||||||
lfs_size_t block_size;
|
|
||||||
|
|
||||||
// Number of erasable blocks on the device.
|
|
||||||
lfs_size_t block_count;
|
|
||||||
|
|
||||||
// Number of erase cycles before littlefs evicts metadata logs and moves
|
|
||||||
// the metadata to another block. Suggested values are in the
|
|
||||||
// range 100-1000, with large values having better performance at the cost
|
|
||||||
// of less consistent wear distribution.
|
|
||||||
//
|
|
||||||
// Set to -1 to disable block-level wear-leveling.
|
|
||||||
int32_t block_cycles;
|
|
||||||
|
|
||||||
// Size of block caches. Each cache buffers a portion of a block in RAM.
|
|
||||||
// The littlefs needs a read cache, a program cache, and one additional
|
|
||||||
// cache per file. Larger caches can improve performance by storing more
|
|
||||||
// data and reducing the number of disk accesses. Must be a multiple of
|
|
||||||
// the read and program sizes, and a factor of the block size.
|
|
||||||
lfs_size_t cache_size;
|
|
||||||
|
|
||||||
// Size of the lookahead buffer in bytes. A larger lookahead buffer
|
|
||||||
// increases the number of blocks found during an allocation pass. The
|
|
||||||
// lookahead buffer is stored as a compact bitmap, so each byte of RAM
|
|
||||||
// can track 8 blocks. Must be a multiple of 8.
|
|
||||||
lfs_size_t lookahead_size;
|
|
||||||
|
|
||||||
// Optional statically allocated read buffer. Must be cache_size.
|
|
||||||
// By default lfs_malloc is used to allocate this buffer.
|
|
||||||
void *read_buffer;
|
|
||||||
|
|
||||||
// Optional statically allocated program buffer. Must be cache_size.
|
|
||||||
// By default lfs_malloc is used to allocate this buffer.
|
|
||||||
void *prog_buffer;
|
|
||||||
|
|
||||||
// Optional statically allocated lookahead buffer. Must be lookahead_size
|
|
||||||
// and aligned to a 32-bit boundary. By default lfs_malloc is used to
|
|
||||||
// allocate this buffer.
|
|
||||||
void *lookahead_buffer;
|
|
||||||
|
|
||||||
// Optional upper limit on length of file names in bytes. No downside for
|
|
||||||
// larger names except the size of the info struct which is controlled by
|
|
||||||
// the LFS_NAME_MAX define. Defaults to LFS_NAME_MAX when zero. Stored in
|
|
||||||
// superblock and must be respected by other littlefs drivers.
|
|
||||||
lfs_size_t name_max;
|
|
||||||
|
|
||||||
// Optional upper limit on files in bytes. No downside for larger files
|
|
||||||
// but must be <= LFS_FILE_MAX. Defaults to LFS_FILE_MAX when zero. Stored
|
|
||||||
// in superblock and must be respected by other littlefs drivers.
|
|
||||||
lfs_size_t file_max;
|
|
||||||
|
|
||||||
// Optional upper limit on custom attributes in bytes. No downside for
|
|
||||||
// larger attributes size but must be <= LFS_ATTR_MAX. Defaults to
|
|
||||||
// LFS_ATTR_MAX when zero.
|
|
||||||
lfs_size_t attr_max;
|
|
||||||
};
|
|
||||||
|
|
||||||
// File info structure
|
|
||||||
struct lfs_info {
|
|
||||||
// Type of the file, either LFS_TYPE_REG or LFS_TYPE_DIR
|
|
||||||
uint8_t type;
|
|
||||||
|
|
||||||
// Size of the file, only valid for REG files. Limited to 32-bits.
|
|
||||||
lfs_size_t size;
|
|
||||||
|
|
||||||
// Name of the file stored as a null-terminated string. Limited to
|
|
||||||
// LFS_NAME_MAX+1, which can be changed by redefining LFS_NAME_MAX to
|
|
||||||
// reduce RAM. LFS_NAME_MAX is stored in superblock and must be
|
|
||||||
// respected by other littlefs drivers.
|
|
||||||
char name[LFS_NAME_MAX+1];
|
|
||||||
};
|
|
||||||
|
|
||||||
// Custom attribute structure, used to describe custom attributes
|
|
||||||
// committed atomically during file writes.
|
|
||||||
struct lfs_attr {
|
|
||||||
// 8-bit type of attribute, provided by user and used to
|
|
||||||
// identify the attribute
|
|
||||||
uint8_t type;
|
|
||||||
|
|
||||||
// Pointer to buffer containing the attribute
|
|
||||||
void *buffer;
|
|
||||||
|
|
||||||
// Size of attribute in bytes, limited to LFS_ATTR_MAX
|
|
||||||
lfs_size_t size;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Optional configuration provided during lfs_file_opencfg
|
|
||||||
struct lfs_file_config {
|
|
||||||
// Optional statically allocated file buffer. Must be cache_size.
|
|
||||||
// By default lfs_malloc is used to allocate this buffer.
|
|
||||||
void *buffer;
|
|
||||||
|
|
||||||
// Optional list of custom attributes related to the file. If the file
|
|
||||||
// is opened with read access, these attributes will be read from disk
|
|
||||||
// during the open call. If the file is opened with write access, the
|
|
||||||
// attributes will be written to disk every file sync or close. This
|
|
||||||
// write occurs atomically with update to the file's contents.
|
|
||||||
//
|
|
||||||
// Custom attributes are uniquely identified by an 8-bit type and limited
|
|
||||||
// to LFS_ATTR_MAX bytes. When read, if the stored attribute is smaller
|
|
||||||
// than the buffer, it will be padded with zeros. If the stored attribute
|
|
||||||
// is larger, then it will be silently truncated. If the attribute is not
|
|
||||||
// found, it will be created implicitly.
|
|
||||||
struct lfs_attr *attrs;
|
|
||||||
|
|
||||||
// Number of custom attributes in the list
|
|
||||||
lfs_size_t attr_count;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
/// internal littlefs data structures ///
|
|
||||||
typedef struct lfs_cache {
|
|
||||||
lfs_block_t block;
|
|
||||||
lfs_off_t off;
|
|
||||||
lfs_size_t size;
|
|
||||||
uint8_t *buffer;
|
|
||||||
} lfs_cache_t;
|
|
||||||
|
|
||||||
typedef struct lfs_mdir {
|
|
||||||
lfs_block_t pair[2];
|
|
||||||
uint32_t rev;
|
|
||||||
lfs_off_t off;
|
|
||||||
uint32_t etag;
|
|
||||||
uint16_t count;
|
|
||||||
bool erased;
|
|
||||||
bool split;
|
|
||||||
lfs_block_t tail[2];
|
|
||||||
} lfs_mdir_t;
|
|
||||||
|
|
||||||
// littlefs directory type
|
|
||||||
typedef struct lfs_dir {
|
|
||||||
struct lfs_dir *next;
|
|
||||||
uint16_t id;
|
|
||||||
uint8_t type;
|
|
||||||
lfs_mdir_t m;
|
|
||||||
|
|
||||||
lfs_off_t pos;
|
|
||||||
lfs_block_t head[2];
|
|
||||||
} lfs_dir_t;
|
|
||||||
|
|
||||||
// littlefs file type
|
|
||||||
typedef struct lfs_file {
|
|
||||||
struct lfs_file *next;
|
|
||||||
uint16_t id;
|
|
||||||
uint8_t type;
|
|
||||||
lfs_mdir_t m;
|
|
||||||
|
|
||||||
struct lfs_ctz {
|
|
||||||
lfs_block_t head;
|
|
||||||
lfs_size_t size;
|
|
||||||
} ctz;
|
|
||||||
|
|
||||||
uint32_t flags;
|
|
||||||
lfs_off_t pos;
|
|
||||||
lfs_block_t block;
|
|
||||||
lfs_off_t off;
|
|
||||||
lfs_cache_t cache;
|
|
||||||
|
|
||||||
const struct lfs_file_config *cfg;
|
|
||||||
} lfs_file_t;
|
|
||||||
|
|
||||||
typedef struct lfs_superblock {
|
|
||||||
uint32_t version;
|
|
||||||
lfs_size_t block_size;
|
|
||||||
lfs_size_t block_count;
|
|
||||||
lfs_size_t name_max;
|
|
||||||
lfs_size_t file_max;
|
|
||||||
lfs_size_t attr_max;
|
|
||||||
} lfs_superblock_t;
|
|
||||||
|
|
||||||
typedef struct lfs_gstate {
|
|
||||||
uint32_t tag;
|
|
||||||
lfs_block_t pair[2];
|
|
||||||
} lfs_gstate_t;
|
|
||||||
|
|
||||||
// The littlefs filesystem type
|
|
||||||
typedef struct lfs {
|
|
||||||
lfs_cache_t rcache;
|
|
||||||
lfs_cache_t pcache;
|
|
||||||
|
|
||||||
lfs_block_t root[2];
|
|
||||||
struct lfs_mlist {
|
|
||||||
struct lfs_mlist *next;
|
|
||||||
uint16_t id;
|
|
||||||
uint8_t type;
|
|
||||||
lfs_mdir_t m;
|
|
||||||
} *mlist;
|
|
||||||
uint32_t seed;
|
|
||||||
|
|
||||||
lfs_gstate_t gstate;
|
|
||||||
lfs_gstate_t gdisk;
|
|
||||||
lfs_gstate_t gdelta;
|
|
||||||
|
|
||||||
struct lfs_free {
|
|
||||||
lfs_block_t off;
|
|
||||||
lfs_block_t size;
|
|
||||||
lfs_block_t i;
|
|
||||||
lfs_block_t ack;
|
|
||||||
uint32_t *buffer;
|
|
||||||
} free;
|
|
||||||
|
|
||||||
const struct lfs_config *cfg;
|
|
||||||
lfs_size_t name_max;
|
|
||||||
lfs_size_t file_max;
|
|
||||||
lfs_size_t attr_max;
|
|
||||||
|
|
||||||
#ifdef LFS_MIGRATE
|
|
||||||
struct lfs1 *lfs1;
|
|
||||||
#endif
|
|
||||||
} lfs_t;
|
|
||||||
|
|
||||||
|
|
||||||
/// Filesystem functions ///
|
|
||||||
|
|
||||||
// Format a block device with the littlefs
|
|
||||||
//
|
|
||||||
// Requires a littlefs object and config struct. This clobbers the littlefs
|
|
||||||
// object, and does not leave the filesystem mounted. The config struct must
|
|
||||||
// be zeroed for defaults and backwards compatibility.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_format(lfs_t *lfs, const struct lfs_config *config);
|
|
||||||
|
|
||||||
// Mounts a littlefs
|
|
||||||
//
|
|
||||||
// Requires a littlefs object and config struct. Multiple filesystems
|
|
||||||
// may be mounted simultaneously with multiple littlefs objects. Both
|
|
||||||
// lfs and config must be allocated while mounted. The config struct must
|
|
||||||
// be zeroed for defaults and backwards compatibility.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_mount(lfs_t *lfs, const struct lfs_config *config);
|
|
||||||
|
|
||||||
// Unmounts a littlefs
|
|
||||||
//
|
|
||||||
// Does nothing besides releasing any allocated resources.
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_unmount(lfs_t *lfs);
|
|
||||||
|
|
||||||
/// General operations ///
|
|
||||||
|
|
||||||
// Removes a file or directory
|
|
||||||
//
|
|
||||||
// If removing a directory, the directory must be empty.
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_remove(lfs_t *lfs, const char *path);
|
|
||||||
|
|
||||||
// Rename or move a file or directory
|
|
||||||
//
|
|
||||||
// If the destination exists, it must match the source in type.
|
|
||||||
// If the destination is a directory, the directory must be empty.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_rename(lfs_t *lfs, const char *oldpath, const char *newpath);
|
|
||||||
|
|
||||||
// Find info about a file or directory
|
|
||||||
//
|
|
||||||
// Fills out the info structure, based on the specified file or directory.
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_stat(lfs_t *lfs, const char *path, struct lfs_info *info);
|
|
||||||
|
|
||||||
// Get a custom attribute
|
|
||||||
//
|
|
||||||
// Custom attributes are uniquely identified by an 8-bit type and limited
|
|
||||||
// to LFS_ATTR_MAX bytes. When read, if the stored attribute is smaller than
|
|
||||||
// the buffer, it will be padded with zeros. If the stored attribute is larger,
|
|
||||||
// then it will be silently truncated. If no attribute is found, the error
|
|
||||||
// LFS_ERR_NOATTR is returned and the buffer is filled with zeros.
|
|
||||||
//
|
|
||||||
// Returns the size of the attribute, or a negative error code on failure.
|
|
||||||
// Note, the returned size is the size of the attribute on disk, irrespective
|
|
||||||
// of the size of the buffer. This can be used to dynamically allocate a buffer
|
|
||||||
// or check for existance.
|
|
||||||
lfs_ssize_t lfs_getattr(lfs_t *lfs, const char *path,
|
|
||||||
uint8_t type, void *buffer, lfs_size_t size);
|
|
||||||
|
|
||||||
// Set custom attributes
|
|
||||||
//
|
|
||||||
// Custom attributes are uniquely identified by an 8-bit type and limited
|
|
||||||
// to LFS_ATTR_MAX bytes. If an attribute is not found, it will be
|
|
||||||
// implicitly created.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_setattr(lfs_t *lfs, const char *path,
|
|
||||||
uint8_t type, const void *buffer, lfs_size_t size);
|
|
||||||
|
|
||||||
// Removes a custom attribute
|
|
||||||
//
|
|
||||||
// If an attribute is not found, nothing happens.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_removeattr(lfs_t *lfs, const char *path, uint8_t type);
|
|
||||||
|
|
||||||
|
|
||||||
/// File operations ///
|
|
||||||
|
|
||||||
// Open a file
|
|
||||||
//
|
|
||||||
// The mode that the file is opened in is determined by the flags, which
|
|
||||||
// are values from the enum lfs_open_flags that are bitwise-ored together.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_file_open(lfs_t *lfs, lfs_file_t *file,
|
|
||||||
const char *path, int flags);
|
|
||||||
|
|
||||||
// Open a file with extra configuration
|
|
||||||
//
|
|
||||||
// The mode that the file is opened in is determined by the flags, which
|
|
||||||
// are values from the enum lfs_open_flags that are bitwise-ored together.
|
|
||||||
//
|
|
||||||
// The config struct provides additional config options per file as described
|
|
||||||
// above. The config struct must be allocated while the file is open, and the
|
|
||||||
// config struct must be zeroed for defaults and backwards compatibility.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_file_opencfg(lfs_t *lfs, lfs_file_t *file,
|
|
||||||
const char *path, int flags,
|
|
||||||
const struct lfs_file_config *config);
|
|
||||||
|
|
||||||
// Close a file
|
|
||||||
//
|
|
||||||
// Any pending writes are written out to storage as though
|
|
||||||
// sync had been called and releases any allocated resources.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_file_close(lfs_t *lfs, lfs_file_t *file);
|
|
||||||
|
|
||||||
// Synchronize a file on storage
|
|
||||||
//
|
|
||||||
// Any pending writes are written out to storage.
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_file_sync(lfs_t *lfs, lfs_file_t *file);
|
|
||||||
|
|
||||||
// Read data from file
|
|
||||||
//
|
|
||||||
// Takes a buffer and size indicating where to store the read data.
|
|
||||||
// Returns the number of bytes read, or a negative error code on failure.
|
|
||||||
lfs_ssize_t lfs_file_read(lfs_t *lfs, lfs_file_t *file,
|
|
||||||
void *buffer, lfs_size_t size);
|
|
||||||
|
|
||||||
// Write data to file
|
|
||||||
//
|
|
||||||
// Takes a buffer and size indicating the data to write. The file will not
|
|
||||||
// actually be updated on the storage until either sync or close is called.
|
|
||||||
//
|
|
||||||
// Returns the number of bytes written, or a negative error code on failure.
|
|
||||||
lfs_ssize_t lfs_file_write(lfs_t *lfs, lfs_file_t *file,
|
|
||||||
const void *buffer, lfs_size_t size);
|
|
||||||
|
|
||||||
// Change the position of the file
|
|
||||||
//
|
|
||||||
// The change in position is determined by the offset and whence flag.
|
|
||||||
// Returns the new position of the file, or a negative error code on failure.
|
|
||||||
lfs_soff_t lfs_file_seek(lfs_t *lfs, lfs_file_t *file,
|
|
||||||
lfs_soff_t off, int whence);
|
|
||||||
|
|
||||||
// Truncates the size of the file to the specified size
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_file_truncate(lfs_t *lfs, lfs_file_t *file, lfs_off_t size);
|
|
||||||
|
|
||||||
// Return the position of the file
|
|
||||||
//
|
|
||||||
// Equivalent to lfs_file_seek(lfs, file, 0, LFS_SEEK_CUR)
|
|
||||||
// Returns the position of the file, or a negative error code on failure.
|
|
||||||
lfs_soff_t lfs_file_tell(lfs_t *lfs, lfs_file_t *file);
|
|
||||||
|
|
||||||
// Change the position of the file to the beginning of the file
|
|
||||||
//
|
|
||||||
// Equivalent to lfs_file_seek(lfs, file, 0, LFS_SEEK_SET)
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_file_rewind(lfs_t *lfs, lfs_file_t *file);
|
|
||||||
|
|
||||||
// Return the size of the file
|
|
||||||
//
|
|
||||||
// Similar to lfs_file_seek(lfs, file, 0, LFS_SEEK_END)
|
|
||||||
// Returns the size of the file, or a negative error code on failure.
|
|
||||||
lfs_soff_t lfs_file_size(lfs_t *lfs, lfs_file_t *file);
|
|
||||||
|
|
||||||
|
|
||||||
/// Directory operations ///
|
|
||||||
|
|
||||||
// Create a directory
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_mkdir(lfs_t *lfs, const char *path);
|
|
||||||
|
|
||||||
// Open a directory
|
|
||||||
//
|
|
||||||
// Once open a directory can be used with read to iterate over files.
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_dir_open(lfs_t *lfs, lfs_dir_t *dir, const char *path);
|
|
||||||
|
|
||||||
// Close a directory
|
|
||||||
//
|
|
||||||
// Releases any allocated resources.
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_dir_close(lfs_t *lfs, lfs_dir_t *dir);
|
|
||||||
|
|
||||||
// Read an entry in the directory
|
|
||||||
//
|
|
||||||
// Fills out the info structure, based on the specified file or directory.
|
|
||||||
// Returns a positive value on success, 0 at the end of directory,
|
|
||||||
// or a negative error code on failure.
|
|
||||||
int lfs_dir_read(lfs_t *lfs, lfs_dir_t *dir, struct lfs_info *info);
|
|
||||||
|
|
||||||
// Change the position of the directory
|
|
||||||
//
|
|
||||||
// The new off must be a value previous returned from tell and specifies
|
|
||||||
// an absolute offset in the directory seek.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_dir_seek(lfs_t *lfs, lfs_dir_t *dir, lfs_off_t off);
|
|
||||||
|
|
||||||
// Return the position of the directory
|
|
||||||
//
|
|
||||||
// The returned offset is only meant to be consumed by seek and may not make
|
|
||||||
// sense, but does indicate the current position in the directory iteration.
|
|
||||||
//
|
|
||||||
// Returns the position of the directory, or a negative error code on failure.
|
|
||||||
lfs_soff_t lfs_dir_tell(lfs_t *lfs, lfs_dir_t *dir);
|
|
||||||
|
|
||||||
// Change the position of the directory to the beginning of the directory
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_dir_rewind(lfs_t *lfs, lfs_dir_t *dir);
|
|
||||||
|
|
||||||
|
|
||||||
/// Filesystem-level filesystem operations
|
|
||||||
|
|
||||||
// Finds the current size of the filesystem
|
|
||||||
//
|
|
||||||
// Note: Result is best effort. If files share COW structures, the returned
|
|
||||||
// size may be larger than the filesystem actually is.
|
|
||||||
//
|
|
||||||
// Returns the number of allocated blocks, or a negative error code on failure.
|
|
||||||
lfs_ssize_t lfs_fs_size(lfs_t *lfs);
|
|
||||||
|
|
||||||
// Traverse through all blocks in use by the filesystem
|
|
||||||
//
|
|
||||||
// The provided callback will be called with each block address that is
|
|
||||||
// currently in use by the filesystem. This can be used to determine which
|
|
||||||
// blocks are in use or how much of the storage is available.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_fs_traverse(lfs_t *lfs, int (*cb)(void*, lfs_block_t), void *data);
|
|
||||||
|
|
||||||
#ifdef LFS_MIGRATE
|
|
||||||
// Attempts to migrate a previous version of littlefs
|
|
||||||
//
|
|
||||||
// Behaves similarly to the lfs_format function. Attempts to mount
|
|
||||||
// the previous version of littlefs and update the filesystem so it can be
|
|
||||||
// mounted with the current version of littlefs.
|
|
||||||
//
|
|
||||||
// Requires a littlefs object and config struct. This clobbers the littlefs
|
|
||||||
// object, and does not leave the filesystem mounted. The config struct must
|
|
||||||
// be zeroed for defaults and backwards compatibility.
|
|
||||||
//
|
|
||||||
// Returns a negative error code on failure.
|
|
||||||
int lfs_migrate(lfs_t *lfs, const struct lfs_config *cfg);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
} /* extern "C" */
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -1,33 +0,0 @@
|
||||||
/*
|
|
||||||
* lfs util functions
|
|
||||||
*
|
|
||||||
* Copyright (c) 2017, Arm Limited. All rights reserved.
|
|
||||||
* SPDX-License-Identifier: BSD-3-Clause
|
|
||||||
*/
|
|
||||||
#include "lfs_util.h"
|
|
||||||
|
|
||||||
// Only compile if user does not provide custom config
|
|
||||||
#ifndef LFS_CONFIG
|
|
||||||
|
|
||||||
|
|
||||||
// Software CRC implementation with small lookup table
|
|
||||||
uint32_t lfs_crc(uint32_t crc, const void *buffer, size_t size) {
|
|
||||||
static const uint32_t rtable[16] = {
|
|
||||||
0x00000000, 0x1db71064, 0x3b6e20c8, 0x26d930ac,
|
|
||||||
0x76dc4190, 0x6b6b51f4, 0x4db26158, 0x5005713c,
|
|
||||||
0xedb88320, 0xf00f9344, 0xd6d6a3e8, 0xcb61b38c,
|
|
||||||
0x9b64c2b0, 0x86d3d2d4, 0xa00ae278, 0xbdbdf21c,
|
|
||||||
};
|
|
||||||
|
|
||||||
const uint8_t *data = buffer;
|
|
||||||
|
|
||||||
for (size_t i = 0; i < size; i++) {
|
|
||||||
crc = (crc >> 4) ^ rtable[(crc ^ (data[i] >> 0)) & 0xf];
|
|
||||||
crc = (crc >> 4) ^ rtable[(crc ^ (data[i] >> 4)) & 0xf];
|
|
||||||
}
|
|
||||||
|
|
||||||
return crc;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -1,234 +0,0 @@
|
||||||
/*
|
|
||||||
* lfs utility functions
|
|
||||||
*
|
|
||||||
* Copyright (c) 2017, Arm Limited. All rights reserved.
|
|
||||||
* SPDX-License-Identifier: BSD-3-Clause
|
|
||||||
*/
|
|
||||||
#ifndef LFS_UTIL_H
|
|
||||||
#define LFS_UTIL_H
|
|
||||||
|
|
||||||
// Users can override lfs_util.h with their own configuration by defining
|
|
||||||
// LFS_CONFIG as a header file to include (-DLFS_CONFIG=lfs_config.h).
|
|
||||||
//
|
|
||||||
// If LFS_CONFIG is used, none of the default utils will be emitted and must be
|
|
||||||
// provided by the config file. To start, I would suggest copying lfs_util.h
|
|
||||||
// and modifying as needed.
|
|
||||||
#ifdef LFS_CONFIG
|
|
||||||
#define LFS_STRINGIZE(x) LFS_STRINGIZE2(x)
|
|
||||||
#define LFS_STRINGIZE2(x) #x
|
|
||||||
#include LFS_STRINGIZE(LFS_CONFIG)
|
|
||||||
#else
|
|
||||||
|
|
||||||
// System includes
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <inttypes.h>
|
|
||||||
|
|
||||||
#ifndef LFS_NO_MALLOC
|
|
||||||
#include <stdlib.h>
|
|
||||||
#endif
|
|
||||||
#ifndef LFS_NO_ASSERT
|
|
||||||
#include <assert.h>
|
|
||||||
#endif
|
|
||||||
#if !defined(LFS_NO_DEBUG) || \
|
|
||||||
!defined(LFS_NO_WARN) || \
|
|
||||||
!defined(LFS_NO_ERROR) || \
|
|
||||||
defined(LFS_YES_TRACE)
|
|
||||||
#include <stdio.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C"
|
|
||||||
{
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
// Macros, may be replaced by system specific wrappers. Arguments to these
|
|
||||||
// macros must not have side-effects as the macros can be removed for a smaller
|
|
||||||
// code footprint
|
|
||||||
|
|
||||||
// Logging functions
|
|
||||||
#ifdef LFS_YES_TRACE
|
|
||||||
#define LFS_TRACE_(fmt, ...) \
|
|
||||||
printf("%s:%d:trace: " fmt "%s\n", __FILE__, __LINE__, __VA_ARGS__)
|
|
||||||
#define LFS_TRACE(...) LFS_TRACE_(__VA_ARGS__, "")
|
|
||||||
#else
|
|
||||||
#define LFS_TRACE(...)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef LFS_NO_DEBUG
|
|
||||||
#define LFS_DEBUG_(fmt, ...) \
|
|
||||||
printf("%s:%d:debug: " fmt "%s\n", __FILE__, __LINE__, __VA_ARGS__)
|
|
||||||
#define LFS_DEBUG(...) LFS_DEBUG_(__VA_ARGS__, "")
|
|
||||||
#else
|
|
||||||
#define LFS_DEBUG(...)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef LFS_NO_WARN
|
|
||||||
#define LFS_WARN_(fmt, ...) \
|
|
||||||
printf("%s:%d:warn: " fmt "%s\n", __FILE__, __LINE__, __VA_ARGS__)
|
|
||||||
#define LFS_WARN(...) LFS_WARN_(__VA_ARGS__, "")
|
|
||||||
#else
|
|
||||||
#define LFS_WARN(...)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef LFS_NO_ERROR
|
|
||||||
#define LFS_ERROR_(fmt, ...) \
|
|
||||||
printf("%s:%d:error: " fmt "%s\n", __FILE__, __LINE__, __VA_ARGS__)
|
|
||||||
#define LFS_ERROR(...) LFS_ERROR_(__VA_ARGS__, "")
|
|
||||||
#else
|
|
||||||
#define LFS_ERROR(...)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Runtime assertions
|
|
||||||
#ifndef LFS_NO_ASSERT
|
|
||||||
#define LFS_ASSERT(test) assert(test)
|
|
||||||
#else
|
|
||||||
#define LFS_ASSERT(test)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
// Builtin functions, these may be replaced by more efficient
|
|
||||||
// toolchain-specific implementations. LFS_NO_INTRINSICS falls back to a more
|
|
||||||
// expensive basic C implementation for debugging purposes
|
|
||||||
|
|
||||||
// Min/max functions for unsigned 32-bit numbers
|
|
||||||
static inline uint32_t lfs_max(uint32_t a, uint32_t b) {
|
|
||||||
return (a > b) ? a : b;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline uint32_t lfs_min(uint32_t a, uint32_t b) {
|
|
||||||
return (a < b) ? a : b;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Align to nearest multiple of a size
|
|
||||||
static inline uint32_t lfs_aligndown(uint32_t a, uint32_t alignment) {
|
|
||||||
return a - (a % alignment);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline uint32_t lfs_alignup(uint32_t a, uint32_t alignment) {
|
|
||||||
return lfs_aligndown(a + alignment-1, alignment);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Find the smallest power of 2 greater than or equal to a
|
|
||||||
static inline uint32_t lfs_npw2(uint32_t a) {
|
|
||||||
#if !defined(LFS_NO_INTRINSICS) && (defined(__GNUC__) || defined(__CC_ARM))
|
|
||||||
return 32 - __builtin_clz(a-1);
|
|
||||||
#else
|
|
||||||
uint32_t r = 0;
|
|
||||||
uint32_t s;
|
|
||||||
a -= 1;
|
|
||||||
s = (a > 0xffff) << 4; a >>= s; r |= s;
|
|
||||||
s = (a > 0xff ) << 3; a >>= s; r |= s;
|
|
||||||
s = (a > 0xf ) << 2; a >>= s; r |= s;
|
|
||||||
s = (a > 0x3 ) << 1; a >>= s; r |= s;
|
|
||||||
return (r | (a >> 1)) + 1;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Count the number of trailing binary zeros in a
|
|
||||||
// lfs_ctz(0) may be undefined
|
|
||||||
static inline uint32_t lfs_ctz(uint32_t a) {
|
|
||||||
#if !defined(LFS_NO_INTRINSICS) && defined(__GNUC__)
|
|
||||||
return __builtin_ctz(a);
|
|
||||||
#else
|
|
||||||
return lfs_npw2((a & -a) + 1) - 1;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Count the number of binary ones in a
|
|
||||||
static inline uint32_t lfs_popc(uint32_t a) {
|
|
||||||
#if !defined(LFS_NO_INTRINSICS) && (defined(__GNUC__) || defined(__CC_ARM))
|
|
||||||
return __builtin_popcount(a);
|
|
||||||
#else
|
|
||||||
a = a - ((a >> 1) & 0x55555555);
|
|
||||||
a = (a & 0x33333333) + ((a >> 2) & 0x33333333);
|
|
||||||
return (((a + (a >> 4)) & 0xf0f0f0f) * 0x1010101) >> 24;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Find the sequence comparison of a and b, this is the distance
|
|
||||||
// between a and b ignoring overflow
|
|
||||||
static inline int lfs_scmp(uint32_t a, uint32_t b) {
|
|
||||||
return (int)(unsigned)(a - b);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Convert between 32-bit little-endian and native order
|
|
||||||
static inline uint32_t lfs_fromle32(uint32_t a) {
|
|
||||||
#if !defined(LFS_NO_INTRINSICS) && ( \
|
|
||||||
(defined( BYTE_ORDER ) && defined( ORDER_LITTLE_ENDIAN ) && BYTE_ORDER == ORDER_LITTLE_ENDIAN ) || \
|
|
||||||
(defined(__BYTE_ORDER ) && defined(__ORDER_LITTLE_ENDIAN ) && __BYTE_ORDER == __ORDER_LITTLE_ENDIAN ) || \
|
|
||||||
(defined(__BYTE_ORDER__) && defined(__ORDER_LITTLE_ENDIAN__) && __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__))
|
|
||||||
return a;
|
|
||||||
#elif !defined(LFS_NO_INTRINSICS) && ( \
|
|
||||||
(defined( BYTE_ORDER ) && defined( ORDER_BIG_ENDIAN ) && BYTE_ORDER == ORDER_BIG_ENDIAN ) || \
|
|
||||||
(defined(__BYTE_ORDER ) && defined(__ORDER_BIG_ENDIAN ) && __BYTE_ORDER == __ORDER_BIG_ENDIAN ) || \
|
|
||||||
(defined(__BYTE_ORDER__) && defined(__ORDER_BIG_ENDIAN__) && __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__))
|
|
||||||
return __builtin_bswap32(a);
|
|
||||||
#else
|
|
||||||
return (((uint8_t*)&a)[0] << 0) |
|
|
||||||
(((uint8_t*)&a)[1] << 8) |
|
|
||||||
(((uint8_t*)&a)[2] << 16) |
|
|
||||||
(((uint8_t*)&a)[3] << 24);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline uint32_t lfs_tole32(uint32_t a) {
|
|
||||||
return lfs_fromle32(a);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Convert between 32-bit big-endian and native order
|
|
||||||
static inline uint32_t lfs_frombe32(uint32_t a) {
|
|
||||||
#if !defined(LFS_NO_INTRINSICS) && ( \
|
|
||||||
(defined( BYTE_ORDER ) && defined( ORDER_LITTLE_ENDIAN ) && BYTE_ORDER == ORDER_LITTLE_ENDIAN ) || \
|
|
||||||
(defined(__BYTE_ORDER ) && defined(__ORDER_LITTLE_ENDIAN ) && __BYTE_ORDER == __ORDER_LITTLE_ENDIAN ) || \
|
|
||||||
(defined(__BYTE_ORDER__) && defined(__ORDER_LITTLE_ENDIAN__) && __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__))
|
|
||||||
return __builtin_bswap32(a);
|
|
||||||
#elif !defined(LFS_NO_INTRINSICS) && ( \
|
|
||||||
(defined( BYTE_ORDER ) && defined( ORDER_BIG_ENDIAN ) && BYTE_ORDER == ORDER_BIG_ENDIAN ) || \
|
|
||||||
(defined(__BYTE_ORDER ) && defined(__ORDER_BIG_ENDIAN ) && __BYTE_ORDER == __ORDER_BIG_ENDIAN ) || \
|
|
||||||
(defined(__BYTE_ORDER__) && defined(__ORDER_BIG_ENDIAN__) && __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__))
|
|
||||||
return a;
|
|
||||||
#else
|
|
||||||
return (((uint8_t*)&a)[0] << 24) |
|
|
||||||
(((uint8_t*)&a)[1] << 16) |
|
|
||||||
(((uint8_t*)&a)[2] << 8) |
|
|
||||||
(((uint8_t*)&a)[3] << 0);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline uint32_t lfs_tobe32(uint32_t a) {
|
|
||||||
return lfs_frombe32(a);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Calculate CRC-32 with polynomial = 0x04c11db7
|
|
||||||
uint32_t lfs_crc(uint32_t crc, const void *buffer, size_t size);
|
|
||||||
|
|
||||||
// Allocate memory, only used if buffers are not provided to littlefs
|
|
||||||
// Note, memory must be 64-bit aligned
|
|
||||||
static inline void *lfs_malloc(size_t size) {
|
|
||||||
#ifndef LFS_NO_MALLOC
|
|
||||||
return malloc(size);
|
|
||||||
#else
|
|
||||||
(void)size;
|
|
||||||
return NULL;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Deallocate memory, only used if buffers are not provided to littlefs
|
|
||||||
static inline void lfs_free(void *p) {
|
|
||||||
#ifndef LFS_NO_MALLOC
|
|
||||||
free(p);
|
|
||||||
#else
|
|
||||||
(void)p;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
} /* extern "C" */
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif
|
|
||||||
#endif
|
|
|
@ -1,58 +0,0 @@
|
||||||
/**
|
|
||||||
* @file littlefs_api.c
|
|
||||||
* @brief Maps the HAL of esp_partition <-> littlefs
|
|
||||||
* @author Brian Pugh
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define ESP_LOCAL_LOG_LEVEL ESP_LOG_INFO
|
|
||||||
|
|
||||||
#include "esp_log.h"
|
|
||||||
#include "esp_partition.h"
|
|
||||||
#include "esp_vfs.h"
|
|
||||||
#include "lfs.h"
|
|
||||||
#include "esp_littlefs.h"
|
|
||||||
#include "littlefs_api.h"
|
|
||||||
|
|
||||||
static const char TAG[] = "esp_littlefs_api";
|
|
||||||
|
|
||||||
int littlefs_api_read(const struct lfs_config *c, lfs_block_t block,
|
|
||||||
lfs_off_t off, void *buffer, lfs_size_t size) {
|
|
||||||
esp_littlefs_t * efs = c->context;
|
|
||||||
size_t part_off = (block * c->block_size) + off;
|
|
||||||
esp_err_t err = esp_partition_read(efs->partition, part_off, buffer, size);
|
|
||||||
if (err) {
|
|
||||||
ESP_LOGE(TAG, "failed to read addr %08x, size %08x, err %d", part_off, size, err);
|
|
||||||
return LFS_ERR_IO;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int littlefs_api_prog(const struct lfs_config *c, lfs_block_t block,
|
|
||||||
lfs_off_t off, const void *buffer, lfs_size_t size) {
|
|
||||||
esp_littlefs_t * efs = c->context;
|
|
||||||
size_t part_off = (block * c->block_size) + off;
|
|
||||||
esp_err_t err = esp_partition_write(efs->partition, part_off, buffer, size);
|
|
||||||
if (err) {
|
|
||||||
ESP_LOGE(TAG, "failed to write addr %08x, size %08x, err %d", part_off, size, err);
|
|
||||||
return LFS_ERR_IO;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int littlefs_api_erase(const struct lfs_config *c, lfs_block_t block) {
|
|
||||||
esp_littlefs_t * efs = c->context;
|
|
||||||
size_t part_off = block * c->block_size;
|
|
||||||
esp_err_t err = esp_partition_erase_range(efs->partition, part_off, c->block_size);
|
|
||||||
if (err) {
|
|
||||||
ESP_LOGE(TAG, "failed to erase addr %08x, size %08x, err %d", part_off, c->block_size, err);
|
|
||||||
return LFS_ERR_IO;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
int littlefs_api_sync(const struct lfs_config *c) {
|
|
||||||
/* Unnecessary for esp-idf */
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,106 +0,0 @@
|
||||||
#ifndef ESP_LITTLEFS_API_H__
|
|
||||||
#define ESP_LITTLEFS_API_H__
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <stddef.h>
|
|
||||||
#include "freertos/FreeRTOS.h"
|
|
||||||
#include "freertos/task.h"
|
|
||||||
#include "freertos/semphr.h"
|
|
||||||
#include "esp_vfs.h"
|
|
||||||
#include "esp_partition.h"
|
|
||||||
#include "lfs.h"
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief a file descriptor
|
|
||||||
* That's also a singly linked list used for keeping tracks of all opened file descriptor
|
|
||||||
*
|
|
||||||
* Shortcomings/potential issues of 32-bit hash (when CONFIG_LITTLEFS_USE_ONLY_HASH) listed here:
|
|
||||||
* * unlink - If a different file is open that generates a hash collision, it will report an
|
|
||||||
* error that it cannot unlink an open file.
|
|
||||||
* * rename - If a different file is open that generates a hash collision with
|
|
||||||
* src or dst, it will report an error that it cannot rename an open file.
|
|
||||||
* Potential consequences:
|
|
||||||
* 1. A file cannot be deleted while a collision-geneating file is open.
|
|
||||||
* Worst-case, if the other file is always open during the lifecycle
|
|
||||||
* of your app, it's collision file cannot be deleted, which in the
|
|
||||||
* worst-case could cause storage-capacity issues.
|
|
||||||
* 2. Same as (1), but for renames
|
|
||||||
*/
|
|
||||||
typedef struct _vfs_littlefs_file_t {
|
|
||||||
lfs_file_t file;
|
|
||||||
uint32_t hash;
|
|
||||||
struct _vfs_littlefs_file_t * next; /*!< Pointer to next file in Singly Linked List */
|
|
||||||
#ifndef CONFIG_LITTLEFS_USE_ONLY_HASH
|
|
||||||
char * path;
|
|
||||||
#endif
|
|
||||||
} vfs_littlefs_file_t;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief littlefs definition structure
|
|
||||||
*/
|
|
||||||
typedef struct {
|
|
||||||
lfs_t *fs; /*!< Handle to the underlying littlefs */
|
|
||||||
SemaphoreHandle_t lock; /*!< FS lock */
|
|
||||||
const esp_partition_t* partition; /*!< The partition on which littlefs is located */
|
|
||||||
char base_path[ESP_VFS_PATH_MAX+1]; /*!< Mount point */
|
|
||||||
|
|
||||||
struct lfs_config cfg; /*!< littlefs Mount configuration */
|
|
||||||
|
|
||||||
vfs_littlefs_file_t *file; /*!< Singly Linked List of files */
|
|
||||||
|
|
||||||
vfs_littlefs_file_t **cache; /*!< A cache of pointers to the opened files */
|
|
||||||
uint16_t cache_size; /*!< The cache allocated size (in pointers) */
|
|
||||||
uint16_t fd_count; /*!< The count of opened file descriptor used to speed up computation */
|
|
||||||
} esp_littlefs_t;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Read a region in a block.
|
|
||||||
*
|
|
||||||
* Negative error codes are propogated to the user.
|
|
||||||
*
|
|
||||||
* @return errorcode. 0 on success.
|
|
||||||
*/
|
|
||||||
int littlefs_api_read(const struct lfs_config *c, lfs_block_t block,
|
|
||||||
lfs_off_t off, void *buffer, lfs_size_t size);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Program a region in a block.
|
|
||||||
*
|
|
||||||
* The block must have previously been erased.
|
|
||||||
* Negative error codes are propogated to the user.
|
|
||||||
* May return LFS_ERR_CORRUPT if the block should be considered bad.
|
|
||||||
*
|
|
||||||
* @return errorcode. 0 on success.
|
|
||||||
*/
|
|
||||||
int littlefs_api_prog(const struct lfs_config *c, lfs_block_t block,
|
|
||||||
lfs_off_t off, const void *buffer, lfs_size_t size);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Erase a block.
|
|
||||||
*
|
|
||||||
* A block must be erased before being programmed.
|
|
||||||
* The state of an erased block is undefined.
|
|
||||||
* Negative error codes are propogated to the user.
|
|
||||||
* May return LFS_ERR_CORRUPT if the block should be considered bad.
|
|
||||||
* @return errorcode. 0 on success.
|
|
||||||
*/
|
|
||||||
int littlefs_api_erase(const struct lfs_config *c, lfs_block_t block);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Sync the state of the underlying block device.
|
|
||||||
*
|
|
||||||
* Negative error codes are propogated to the user.
|
|
||||||
*
|
|
||||||
* @return errorcode. 0 on success.
|
|
||||||
*/
|
|
||||||
int littlefs_api_sync(const struct lfs_config *c);
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif
|
|
Loading…
Reference in New Issue