0ca37387be
Improves safety of use and clarity of users. Termination is not included in the reported size to avoid changing user behavior or misrepresenting the file contents.
268 lines
6.2 KiB
C++
268 lines
6.2 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/>.
|
|
*/
|
|
/*
|
|
ArduPilot filesystem interface for ROMFS
|
|
*/
|
|
|
|
#include "AP_Filesystem_config.h"
|
|
|
|
#if AP_FILESYSTEM_ROMFS_ENABLED
|
|
|
|
#include "AP_Filesystem.h"
|
|
#include "AP_Filesystem_ROMFS.h"
|
|
#include <AP_HAL/AP_HAL.h>
|
|
#include <AP_Math/AP_Math.h>
|
|
#include <AP_ROMFS/AP_ROMFS.h>
|
|
|
|
int AP_Filesystem_ROMFS::open(const char *fname, int flags, bool allow_absolute_paths)
|
|
{
|
|
if ((flags & O_ACCMODE) != O_RDONLY) {
|
|
errno = EROFS;
|
|
return -1;
|
|
}
|
|
uint8_t idx;
|
|
for (idx=0; idx<max_open_file; idx++) {
|
|
if (file[idx].data == nullptr) {
|
|
break;
|
|
}
|
|
}
|
|
if (idx == max_open_file) {
|
|
errno = ENFILE;
|
|
return -1;
|
|
}
|
|
if (file[idx].data != nullptr) {
|
|
errno = EBUSY;
|
|
return -1;
|
|
}
|
|
file[idx].data = AP_ROMFS::find_decompress(fname, file[idx].size);
|
|
if (file[idx].data == nullptr) {
|
|
errno = ENOENT;
|
|
return -1;
|
|
}
|
|
file[idx].ofs = 0;
|
|
return idx;
|
|
}
|
|
|
|
int AP_Filesystem_ROMFS::close(int fd)
|
|
{
|
|
if (fd < 0 || fd >= max_open_file || file[fd].data == nullptr) {
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
AP_ROMFS::free(file[fd].data);
|
|
file[fd].data = nullptr;
|
|
return 0;
|
|
}
|
|
|
|
int32_t AP_Filesystem_ROMFS::read(int fd, void *buf, uint32_t count)
|
|
{
|
|
if (fd < 0 || fd >= max_open_file || file[fd].data == nullptr) {
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
count = MIN(file[fd].size - file[fd].ofs, count);
|
|
if (count == 0) {
|
|
return 0;
|
|
}
|
|
memcpy(buf, &file[fd].data[file[fd].ofs], count);
|
|
file[fd].ofs += count;
|
|
return count;
|
|
}
|
|
|
|
int32_t AP_Filesystem_ROMFS::write(int fd, const void *buf, uint32_t count)
|
|
{
|
|
errno = EROFS;
|
|
return -1;
|
|
}
|
|
|
|
int AP_Filesystem_ROMFS::fsync(int fd)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int32_t AP_Filesystem_ROMFS::lseek(int fd, int32_t offset, int seek_from)
|
|
{
|
|
if (fd < 0 || fd >= max_open_file || file[fd].data == nullptr) {
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
switch (seek_from) {
|
|
case SEEK_SET:
|
|
if (offset < 0) {
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
file[fd].ofs = MIN(file[fd].size, (uint32_t)offset);
|
|
break;
|
|
case SEEK_CUR:
|
|
file[fd].ofs = MIN(file[fd].size, offset+file[fd].ofs);
|
|
break;
|
|
case SEEK_END:
|
|
file[fd].ofs = file[fd].size;
|
|
break;
|
|
}
|
|
return file[fd].ofs;
|
|
}
|
|
|
|
int AP_Filesystem_ROMFS::stat(const char *name, struct stat *stbuf)
|
|
{
|
|
uint32_t size;
|
|
if (!AP_ROMFS::find_size(name, size)) {
|
|
errno = ENOENT;
|
|
return -1;
|
|
}
|
|
memset(stbuf, 0, sizeof(*stbuf));
|
|
stbuf->st_size = size;
|
|
return 0;
|
|
}
|
|
|
|
int AP_Filesystem_ROMFS::unlink(const char *pathname)
|
|
{
|
|
errno = EROFS;
|
|
return -1;
|
|
}
|
|
|
|
int AP_Filesystem_ROMFS::mkdir(const char *pathname)
|
|
{
|
|
errno = EROFS;
|
|
return -1;
|
|
}
|
|
|
|
void *AP_Filesystem_ROMFS::opendir(const char *pathname)
|
|
{
|
|
uint8_t idx;
|
|
for (idx=0; idx<max_open_dir; idx++) {
|
|
if (dir[idx].path == nullptr) {
|
|
break;
|
|
}
|
|
}
|
|
if (idx == max_open_dir) {
|
|
errno = ENFILE;
|
|
return nullptr;
|
|
}
|
|
dir[idx].ofs = 0;
|
|
dir[idx].path = strdup(pathname);
|
|
if (!dir[idx].path) {
|
|
return nullptr;
|
|
}
|
|
|
|
// Take a sneak peek and reset
|
|
const char *name = AP_ROMFS::dir_list(dir[idx].path, dir[idx].ofs);
|
|
dir[idx].ofs = 0;
|
|
if (!name) {
|
|
// Directory does not exist
|
|
return nullptr;
|
|
}
|
|
|
|
return (void*)&dir[idx];
|
|
}
|
|
|
|
struct dirent *AP_Filesystem_ROMFS::readdir(void *dirp)
|
|
{
|
|
uint32_t idx = ((rdir*)dirp) - &dir[0];
|
|
if (idx >= max_open_dir) {
|
|
errno = EBADF;
|
|
return nullptr;
|
|
}
|
|
const char *name = AP_ROMFS::dir_list(dir[idx].path, dir[idx].ofs);
|
|
if (!name) {
|
|
return nullptr;
|
|
}
|
|
const uint32_t plen = strlen(dir[idx].path);
|
|
if (plen > 0) {
|
|
// Offset to get just file/directory name
|
|
name += plen + 1;
|
|
}
|
|
|
|
// Copy full name
|
|
strncpy(dir[idx].de.d_name, name, sizeof(dir[idx].de.d_name));
|
|
|
|
const char* slash = strchr(name, '/');
|
|
if (slash == nullptr) {
|
|
// File
|
|
dir[idx].de.d_type = DT_REG;
|
|
|
|
} else {
|
|
// Directory
|
|
dir[idx].de.d_type = DT_DIR;
|
|
|
|
// Add null termination after directory name
|
|
const size_t index = slash - name;
|
|
dir[idx].de.d_name[index] = 0;
|
|
}
|
|
|
|
return &dir[idx].de;
|
|
}
|
|
|
|
int AP_Filesystem_ROMFS::closedir(void *dirp)
|
|
{
|
|
uint32_t idx = ((rdir *)dirp) - &dir[0];
|
|
if (idx >= max_open_dir) {
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
free(dir[idx].path);
|
|
dir[idx].path = nullptr;
|
|
return 0;
|
|
}
|
|
|
|
// return free disk space in bytes
|
|
int64_t AP_Filesystem_ROMFS::disk_free(const char *path)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
// return total disk space in bytes
|
|
int64_t AP_Filesystem_ROMFS::disk_space(const char *path)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
set mtime on a file
|
|
*/
|
|
bool AP_Filesystem_ROMFS::set_mtime(const char *filename, const uint32_t mtime_sec)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
Load a file's contents into memory. Returned object must be `delete`d to free
|
|
the data. The data is guaranteed to be null-terminated such that it can be
|
|
treated as a string. Overridden in ROMFS to avoid taking twice the memory.
|
|
*/
|
|
FileData *AP_Filesystem_ROMFS::load_file(const char *filename)
|
|
{
|
|
FileData *fd = new FileData(this);
|
|
if (!fd) {
|
|
return nullptr;
|
|
}
|
|
// AP_ROMFS adds the guaranteed termination so we don't have to.
|
|
fd->data = AP_ROMFS::find_decompress(filename, fd->length);
|
|
if (fd->data == nullptr) {
|
|
delete fd;
|
|
return nullptr;
|
|
}
|
|
return fd;
|
|
}
|
|
|
|
// unload data from load_file()
|
|
void AP_Filesystem_ROMFS::unload_file(FileData *fd)
|
|
{
|
|
AP_ROMFS::free(fd->data);
|
|
}
|
|
|
|
#endif // AP_FILESYSTEM_ROMFS_ENABLED
|