/*
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 .
*/
/*
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 AP_FILESYSTEM_FATFS_ENABLED || AP_FILESYSTEM_POSIX_ENABLED || AP_FILESYSTEM_ESP32_ENABLED
#include "posix_compat.h"
#include
#include
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 flag;
}
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));
if (f->fd == -1) {
delete f;
return nullptr;
}
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;
}
#undef fgets
char *apfs_fgets(char *s, int size, APFS_FILE *stream)
{
CHECK_STREAM(stream, NULL);
auto &fs = AP::FS();
if (!fs.fgets(s, size, stream->fd)) {
return NULL;
}
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 // AP_FILESYSTEM_POSIX_ENABLED