mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-21 16:18:29 -04:00
0db8745320
this ensures the directory entry in the FAT is updated on fflush() Fixes #14617
269 lines
6.3 KiB
C++
269 lines
6.3 KiB
C++
/*
|
|
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 3 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
/*
|
|
compatibility with posix APIs using AP_Filesystem
|
|
|
|
This implements the FILE* API from posix sufficiently well for Lua
|
|
scripting to function. It has no buffering so is inefficient for
|
|
single character operations. We deliberately use this implementation
|
|
in HAL_SITL and HAL_Linux where it is not needed in order to have a
|
|
uniform implementation across all platforms
|
|
*/
|
|
|
|
#include "AP_Filesystem.h"
|
|
|
|
#if HAVE_FILESYSTEM_SUPPORT
|
|
|
|
#include "posix_compat.h"
|
|
#include <stdarg.h>
|
|
#include <AP_Math/AP_Math.h>
|
|
|
|
struct apfs_file {
|
|
int fd;
|
|
bool error;
|
|
bool eof;
|
|
int16_t unget;
|
|
char *tmpfile_name;
|
|
};
|
|
|
|
#define CHECK_STREAM(stream, ret) while (stream == NULL || stream->fd < 0) { errno = EBADF; return ret; }
|
|
|
|
#define modecmp(str, pat) (strcmp(str, pat) == 0 ? 1: 0)
|
|
|
|
/*
|
|
map a fopen() file mode to a open() mode
|
|
*/
|
|
static int posix_fopen_modes_to_open(const char *mode)
|
|
{
|
|
int flag = 0;
|
|
|
|
if (modecmp(mode,"r") || modecmp(mode,"rb")) {
|
|
flag = O_RDONLY;
|
|
return flag;
|
|
}
|
|
if (modecmp(mode,"r+") || modecmp(mode, "r+b" ) || modecmp(mode, "rb+" )) {
|
|
flag = O_RDWR | O_TRUNC;
|
|
return flag;
|
|
}
|
|
if (modecmp(mode,"w") || modecmp(mode,"wb")) {
|
|
flag = O_WRONLY | O_CREAT | O_TRUNC;
|
|
return flag;
|
|
}
|
|
if (modecmp(mode,"w+") || modecmp(mode, "w+b" ) || modecmp(mode, "wb+" )) {
|
|
flag = O_RDWR | O_CREAT | O_TRUNC;
|
|
return flag;
|
|
}
|
|
if (modecmp(mode,"a") || modecmp(mode,"ab")) {
|
|
flag = O_WRONLY | O_CREAT | O_APPEND;
|
|
return flag;
|
|
}
|
|
if (modecmp(mode,"a+") || modecmp(mode, "a+b" ) || modecmp(mode, "ab+" )) {
|
|
flag = O_RDWR | O_CREAT | O_APPEND;
|
|
return -1;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
APFS_FILE *apfs_fopen(const char *pathname, const char *mode)
|
|
{
|
|
APFS_FILE *f = new APFS_FILE;
|
|
if (!f) {
|
|
return nullptr;
|
|
}
|
|
f->fd = AP::FS().open(pathname, posix_fopen_modes_to_open(mode));
|
|
f->unget = -1;
|
|
return f;
|
|
}
|
|
|
|
int apfs_fprintf(APFS_FILE *stream, const char *fmt, ...)
|
|
{
|
|
CHECK_STREAM(stream, -1);
|
|
va_list va;
|
|
char* buf = NULL;
|
|
int16_t len;
|
|
va_start(va, fmt);
|
|
len = vasprintf(&buf, fmt, va);
|
|
va_end(va);
|
|
if (len > 0) {
|
|
len = AP::FS().write(stream->fd, buf, len);
|
|
free(buf);
|
|
}
|
|
|
|
return len;
|
|
}
|
|
|
|
int apfs_fflush(APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, EOF);
|
|
if (AP::FS().fsync(stream->fd) == 0) {
|
|
return 0;
|
|
}
|
|
return EOF;
|
|
}
|
|
|
|
size_t apfs_fread(void *ptr, size_t size, size_t nmemb, APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, 0);
|
|
ssize_t ret = AP::FS().read(stream->fd, ptr, size*nmemb);
|
|
if (ret <= 0) {
|
|
stream->eof = true;
|
|
return 0;
|
|
}
|
|
return ret / size;
|
|
}
|
|
|
|
size_t apfs_fwrite(const void *ptr, size_t size, size_t nmemb, APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, 0);
|
|
ssize_t ret = AP::FS().write(stream->fd, ptr, size*nmemb);
|
|
if (ret <= 0) {
|
|
stream->error = true;
|
|
return 0;
|
|
}
|
|
return ret / size;
|
|
}
|
|
|
|
int apfs_fputs(const char *s, APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, EOF);
|
|
ssize_t ret = AP::FS().write(stream->fd, s, strlen(s));
|
|
if (ret < 0) {
|
|
stream->error = true;
|
|
return EOF;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
char *apfs_fgets(char *s, int size, APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, NULL);
|
|
ssize_t ret = AP::FS().read(stream->fd, s, size-1);
|
|
if (ret < 0) {
|
|
stream->error = true;
|
|
return NULL;
|
|
}
|
|
s[ret] = 0;
|
|
return s;
|
|
}
|
|
|
|
void apfs_clearerr(APFS_FILE *stream)
|
|
{
|
|
stream->error = false;
|
|
}
|
|
|
|
int apfs_fseek(APFS_FILE *stream, long offset, int whence)
|
|
{
|
|
CHECK_STREAM(stream, EOF);
|
|
stream->eof = false;
|
|
return AP::FS().lseek(stream->fd, offset, whence);
|
|
}
|
|
|
|
int apfs_ferror(APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, EOF);
|
|
return stream->error;
|
|
}
|
|
|
|
int apfs_fclose(APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, EOF);
|
|
int ret = AP::FS().close(stream->fd);
|
|
stream->fd = -1;
|
|
if (stream->tmpfile_name) {
|
|
AP::FS().unlink(stream->tmpfile_name);
|
|
free(stream->tmpfile_name);
|
|
stream->tmpfile_name = NULL;
|
|
}
|
|
delete stream;
|
|
return ret;
|
|
}
|
|
|
|
APFS_FILE *apfs_tmpfile(void)
|
|
{
|
|
char *fname = NULL;
|
|
if (asprintf(&fname, "tmp.%03u", unsigned(get_random16()) % 1000) <= 0) {
|
|
return NULL;
|
|
}
|
|
APFS_FILE *ret = apfs_fopen(fname, "w");
|
|
if (!ret) {
|
|
free(fname);
|
|
return NULL;
|
|
}
|
|
ret->tmpfile_name = fname;
|
|
return ret;
|
|
}
|
|
|
|
int apfs_getc(APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, EOF);
|
|
uint8_t c;
|
|
if (stream->unget != -1) {
|
|
c = stream->unget;
|
|
stream->unget = -1;
|
|
return c;
|
|
}
|
|
ssize_t ret = AP::FS().read(stream->fd, &c, 1);
|
|
if (ret <= 0) {
|
|
stream->eof = true;
|
|
return EOF;
|
|
}
|
|
return c;
|
|
}
|
|
|
|
int apfs_ungetc(int c, APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, EOF);
|
|
stream->unget = c;
|
|
stream->eof = false;
|
|
return c;
|
|
}
|
|
|
|
int apfs_feof(APFS_FILE *stream)
|
|
{
|
|
return stream->eof;
|
|
}
|
|
|
|
APFS_FILE *apfs_freopen(const char *pathname, const char *mode, APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, NULL);
|
|
int ret = AP::FS().close(stream->fd);
|
|
if (ret < 0) {
|
|
return NULL;
|
|
}
|
|
if (stream->tmpfile_name) {
|
|
AP::FS().unlink(stream->tmpfile_name);
|
|
free(stream->tmpfile_name);
|
|
stream->tmpfile_name = NULL;
|
|
}
|
|
stream->fd = AP::FS().open(pathname, posix_fopen_modes_to_open(mode));
|
|
stream->error = false;
|
|
stream->eof = false;
|
|
stream->unget = -1;
|
|
return stream;
|
|
}
|
|
|
|
long apfs_ftell(APFS_FILE *stream)
|
|
{
|
|
CHECK_STREAM(stream, EOF);
|
|
return AP::FS().lseek(stream->fd, 0, SEEK_CUR);
|
|
}
|
|
|
|
int apfs_remove(const char *pathname)
|
|
{
|
|
return AP::FS().unlink(pathname);
|
|
}
|
|
|
|
#endif // HAVE_FILESYSTEM_SUPPORT
|