1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/pkg/lwext4/fs/lwext4_fs.c
2023-02-25 14:01:20 +01:00

520 lines
12 KiB
C

/*
* Copyright (C) 2022 ML!PA Consulting GmbH
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup sys_lwext4
* @{
*
* @file
* @brief lwEXT4 integration with vfs
*
* @author Benjamin Valentin <benjamin.valentin@ml-pa.com>
*
* @}
*/
#include <assert.h>
#include <errno.h>
#include <fcntl.h>
#include <inttypes.h>
#include <stdlib.h>
#include <string.h>
#include "fs/lwext4_fs.h"
#include <ext4_super.h>
#include <ext4_mkfs.h>
#define ENABLE_DEBUG 0
#include <debug.h>
/**
* @brief file system block size to use when formatting a new file system
*/
#ifndef CONFIG_EXT_BLOCKSIZE_DEFAULT
#define CONFIG_EXT_BLOCKSIZE_DEFAULT (1024)
#endif
/**
* @brief Automatic mountpoints
*/
XFA_USE(vfs_mount_t, vfs_mountpoints_xfa);
/**
* @brief Number of automatic mountpoints
*/
#define MOUNTPOINTS_NUMOF XFA_LEN(vfs_mount_t, vfs_mountpoints_xfa)
struct ext4_mountpoint *ext4_get_mount(const char *path)
{
size_t strlen_path = strlen(path);
for (unsigned i = 0; i < MOUNTPOINTS_NUMOF; ++i) {
/* lwext4 wants terminating '/' for mountpoint, but VFS does not */
size_t strlen_mp = strlen(vfs_mountpoints_xfa[i].mount_point) - 1;
if (strlen_mp < strlen_path &&
strncmp(path, vfs_mountpoints_xfa[i].mount_point, strlen_mp) == 0) {
lwext4_desc_t *fs = vfs_mountpoints_xfa[i].private_data;
return &fs->mp;
}
}
DEBUG("lwext4: no mountpoint found for '%s'\n", path);
return NULL;
}
static int _noop(struct ext4_blockdev *bdev)
{
(void)bdev;
return 0;
}
static int blockdev_bread(struct ext4_blockdev *bdev, void *buf, uint64_t blk_id,
uint32_t blk_cnt)
{
mtd_dev_t *dev = bdev->bdif->p_user;
uint32_t page = blk_id * dev->pages_per_sector;
uint32_t size = blk_cnt * dev->pages_per_sector * dev->page_size;
assert(blk_id <= UINT32_MAX);
DEBUG("lwext4: read %"PRIu32" bytes from page %"PRIu32"\n", size, page);
return -mtd_read_page(dev, buf, page, 0, size);
}
static int blockdev_bwrite(struct ext4_blockdev *bdev, const void *buf,
uint64_t blk_id, uint32_t blk_cnt)
{
mtd_dev_t *dev = bdev->bdif->p_user;
uint32_t page = blk_id * dev->pages_per_sector;
uint32_t size = blk_cnt * dev->pages_per_sector * dev->page_size;
assert(blk_id <= UINT32_MAX);
DEBUG("lwext4: erase %"PRIu32" sectors starting with %"PRIu64"\n", blk_cnt, blk_id);
int res = mtd_erase_sector(dev, blk_id, blk_cnt);
if (res) {
return -res;
}
DEBUG("lwext4: write %"PRIu32" bytes to page %"PRIu32"\n", size, page);
return -mtd_write_page_raw(dev, buf, page, 0, size);
}
static int prepare(lwext4_desc_t *fs, const char *mount_point)
{
mtd_dev_t *dev = fs->dev;
struct ext4_blockdev_iface *iface = &fs->iface;
memset(&fs->mp, 0, sizeof(fs->mp));
memset(&fs->bdev, 0, sizeof(fs->bdev));
memset(&fs->iface, 0, sizeof(fs->iface));
strncpy(fs->mp.name, mount_point, CONFIG_EXT4_MAX_MP_NAME);
strcat(fs->mp.name, "/");
int res = mtd_init(dev);
if (res) {
return res;
}
iface->open = _noop;
iface->bread = blockdev_bread;
iface->bwrite = blockdev_bwrite;
iface->close = _noop;
iface->lock = _noop;
iface->unlock = _noop;
iface->p_user = dev;
iface->ph_bcnt = dev->sector_count;
iface->ph_bsize = dev->pages_per_sector * dev->page_size;
iface->ph_bbuf = malloc(iface->ph_bsize);
fs->bdev.bdif = iface;
fs->bdev.part_size = iface->ph_bcnt * iface->ph_bsize;
return -ext4_block_init(&fs->bdev);
}
static mutex_t _lwext4_mutex;
static void _lock(void)
{
mutex_lock(&_lwext4_mutex);
}
static void _unlock(void)
{
mutex_unlock(&_lwext4_mutex);
}
static const struct ext4_lock _lwext4_os_lock = {
.lock = _lock,
.unlock = _unlock,
};
static int _mount(vfs_mount_t *mountp)
{
/* if one of the lines below fail to compile you probably need to adjust
vfs buffer sizes ;) */
static_assert(VFS_DIR_BUFFER_SIZE >= sizeof(ext4_dir),
"ext4_dir must fit in VFS_DIR_BUFFER_SIZE");
static_assert(VFS_FILE_BUFFER_SIZE >= sizeof(ext4_file),
"ext4_file must fit in VFS_FILE_BUFFER_SIZE");
lwext4_desc_t *fs = mountp->private_data;
struct ext4_mountpoint *mp = &fs->mp;
struct ext4_bcache *bc = &mp->bc;
if (mp->mounted) {
return 0;
}
int res = prepare(fs, mountp->mount_point);
if (res) {
return res;
}
res = ext4_fs_init(&mp->fs, &fs->bdev, false);
DEBUG("lwext4 mount: %s\n", strerror(res));
if (res != EOK) {
return -res;
}
size_t bsize = ext4_sb_get_block_size(&mp->fs.sb);
ext4_block_set_lb_size(&fs->bdev, bsize);
res = ext4_bcache_init_dynamic(bc, CONFIG_BLOCK_DEV_CACHE_SIZE, bsize);
if (res != EOK) {
return -res;
}
assert(bsize == bc->itemsize);
bc->bdev = &fs->bdev;
fs->bdev.bc = bc;
fs->bdev.fs = &mp->fs;
/*Bind block cache to block device*/
res = ext4_block_bind_bcache(bc->bdev, bc);
if (res != EOK) {
ext4_bcache_cleanup(bc);
ext4_block_fini(bc->bdev);
ext4_bcache_fini_dynamic(bc);
return -res;
}
mp->os_locks = &_lwext4_os_lock;
mp->mounted = true;
res = ext4_recover(fs->mp.name);
if (res != EOK && res != ENOTSUP) {
DEBUG("ext4_recover: rc = %d\n", res);
return -res;
}
res = ext4_journal_start(fs->mp.name);
if (res != EOK) {
DEBUG("ext4_journal_start: rc = %d\n", res);
return -res;
}
ext4_cache_write_back(fs->mp.name, 1);
return -res;
}
static int _umount(vfs_mount_t *mountp)
{
lwext4_desc_t *fs = mountp->private_data;
struct ext4_mountpoint *mp = &fs->mp;
int res;
ext4_cache_write_back(fs->mp.name, 0);
res = ext4_journal_stop(fs->mp.name);
if (res != EOK) {
DEBUG("ext4_journal_stop: fail %d", res);
return false;
}
res = ext4_fs_fini(&mp->fs);
if (res != EOK) {
goto out;
}
ext4_bcache_cleanup(mp->fs.bdev->bc);
ext4_bcache_fini_dynamic(mp->fs.bdev->bc);
res = ext4_block_fini(mp->fs.bdev);
if (fs->iface.ph_bbuf) {
free(fs->iface.ph_bbuf);
fs->iface.ph_bbuf = NULL;
}
out:
mp->mounted = false;
mp->fs.bdev->fs = NULL;
return -res;
}
#ifdef MODULE_LWEXT4_VFS_FORMAT
static int _format(vfs_mount_t *mountp)
{
lwext4_desc_t *fs = mountp->private_data;
struct ext4_mountpoint *mp = &fs->mp;
if (mp->mounted) {
return -EBUSY;
}
int res = prepare(fs, mountp->mount_point);
if (res) {
return res;
}
struct ext4_mkfs_info info = {
.block_size = CONFIG_EXT_BLOCKSIZE_DEFAULT,
.journal = CONFIG_JOURNALING_ENABLE,
};
assert(fs->dev->pages_per_sector * fs->dev->page_size <= info.block_size);
res = ext4_mkfs(&mp->fs, &fs->bdev, &info, CONFIG_EXT_FEATURE_SET_LVL);
free(fs->iface.ph_bbuf);
fs->iface.ph_bbuf = NULL;
return -res;
}
#endif /* MODULE_LWEXT4_VFS_FORMAT */
static int _mkdir(vfs_mount_t *mountp, const char *name, mode_t mode)
{
(void)mountp;
int res = ext4_dir_mk(name);
if (res != EOK) {
return -res;
}
if (mode) {
return -ext4_mode_set(name, mode);
}
return 0;
}
static int _rmdir(vfs_mount_t *mountp, const char *name)
{
(void)mountp;
return -ext4_dir_rm(name);
}
static int _statvfs(vfs_mount_t *mountp, const char *restrict path, struct statvfs *restrict buf)
{
(void)path;
struct ext4_mount_stats stats;
lwext4_desc_t *fs = mountp->private_data;
int res = ext4_mount_point_stats(fs->mp.name, &stats);
if (res) {
return -res;
}
buf->f_blocks = stats.blocks_count;
buf->f_bfree = stats.free_blocks_count;
buf->f_bavail = stats.free_blocks_count;
buf->f_bsize = stats.block_size;
return 0;
}
static inline ext4_file * _get_ext4_file(vfs_file_t *f)
{
/* The buffer in `private_data` is part of a union that also contains a
* pointer, so the alignment is fine. Adding an intermediate cast to
* uintptr_t to silence -Wcast-align
*/
return (ext4_file *)(uintptr_t)f->private_data.buffer;
}
static int _open(vfs_file_t *filp, const char *name, int flags, mode_t mode)
{
int res;
ext4_file *file = _get_ext4_file(filp);
res = ext4_fopen2(file, name, flags);
if (res != EOK) {
return -res;
}
file->mp = ext4_get_mount(name);
if (mode && (flags & O_CREAT)) {
return -ext4_mode_set(name, mode);
}
return 0;
}
static ssize_t _read(vfs_file_t *filp, void *dest, size_t nbytes)
{
ext4_file *file = _get_ext4_file(filp);
int res = ext4_fread(file, dest, nbytes, &nbytes);
if (res != EOK) {
return -res;
} else {
return nbytes;
}
}
static ssize_t _write(vfs_file_t *filp, const void *src, size_t nbytes)
{
ext4_file *file = _get_ext4_file(filp);
int res = ext4_fwrite(file, src, nbytes, &nbytes);
if (res != EOK) {
return -res;
} else {
return nbytes;
}
}
static off_t _lseek(vfs_file_t *filp, off_t off, int whence)
{
ext4_file *file = _get_ext4_file(filp);
int res = ext4_fseek(file, off, whence);
if (res) {
return -res;
}
return ext4_ftell(file);
}
static int _fstat(vfs_file_t *filp, struct stat *buf)
{
ext4_file *file = _get_ext4_file(filp);
buf->st_ino = file->inode;
buf->st_size = file->fsize;
buf->st_mode = S_IFREG;
return 0;
}
static int _fsync(vfs_file_t *filp)
{
return -ext4_cache_flush(filp->mp->mount_point);
}
static int _close(vfs_file_t *filp)
{
ext4_file *file = _get_ext4_file(filp);
return ext4_fclose(file);
}
static int _unlink(vfs_mount_t *mountp, const char *name)
{
(void)mountp;
return -ext4_fremove(name);
}
static inline ext4_dir *_get_ext4_dir(vfs_DIR *dirp)
{
return (ext4_dir *)(uintptr_t)dirp->private_data.buffer;
}
static int _rename(vfs_mount_t *mountp, const char *from_path, const char *to_path)
{
(void)mountp;
return -ext4_frename(from_path, to_path);
}
static int _opendir(vfs_DIR *dirp, const char *dirname)
{
ext4_dir *dir = _get_ext4_dir(dirp);
/* lwext4 doesn't like an empty path (relative to mount point) */
if (strcmp(dirp->mp->mount_point, dirname) == 0) {
lwext4_desc_t *fs = dirp->mp->private_data;
dirname = fs->mp.name;
}
return -ext4_dir_open(dir, dirname);
}
static int _readdir(vfs_DIR *dirp, vfs_dirent_t *entry)
{
ext4_dir *dir = _get_ext4_dir(dirp);
const ext4_direntry *dirent = ext4_dir_entry_next(dir);
if (dirent == NULL) {
return 0;
}
strncpy(entry->d_name, (char *)dirent->name, sizeof(entry->d_name));
return 1;
}
static int _closedir(vfs_DIR *dirp)
{
ext4_dir *dir = _get_ext4_dir(dirp);
return -ext4_dir_close(dir);
}
static const vfs_file_system_ops_t lwext4_fs_ops = {
#ifdef MODULE_LWEXT4_VFS_FORMAT
.format = _format,
#endif
.mount = _mount,
.umount = _umount,
.unlink = _unlink,
.mkdir = _mkdir,
.rmdir = _rmdir,
.rename = _rename,
.stat = vfs_sysop_stat_from_fstat,
.statvfs = _statvfs,
};
static const vfs_file_ops_t lwext4_file_ops = {
.open = _open,
.close = _close,
.read = _read,
.write = _write,
.lseek = _lseek,
.fstat = _fstat,
.fsync = _fsync,
};
static const vfs_dir_ops_t lwext4_dir_ops = {
.opendir = _opendir,
.readdir = _readdir,
.closedir = _closedir,
};
const vfs_file_system_t lwext4_file_system = {
.fs_op = &lwext4_fs_ops,
.f_op = &lwext4_file_ops,
.d_op = &lwext4_dir_ops,
.flags = VFS_FS_FLAG_WANT_ABS_PATH,
};