// Copyright (c) 2017-2020, Mudita Sp. z.o.o. All rights reserved.
// For licensing, see https://github.com/mudita/MuditaOS/LICENSE.md
#define CATCH_CONFIG_MAIN
#include <catch2/catch.hpp>
#include <purefs/fs/filesystem.hpp>
#include <purefs/blkdev/disk_manager.hpp>
#include <purefs/blkdev/disk_image.hpp>
#include <purefs/fs/drivers/filesystem_vfat.hpp>
#include <purefs/vfs_subsystem.hpp>
#include <purefs/fs/thread_local_cwd.hpp>
#include <sys/statvfs.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <tuple>
namespace
{
constexpr auto disk_image = "PurePhone.img";
}
TEST_CASE("Corefs: Registering and unregistering block device")
{
using namespace purefs;
auto dm = std::make_shared<blkdev::disk_manager>();
auto disk = std::make_shared<blkdev::disk_image>(disk_image);
REQUIRE(disk);
REQUIRE(dm->register_device(disk, "emmc0") == 0);
purefs::fs::filesystem fscore(dm);
/* Requested filesystem is not registered */
REQUIRE(fscore.mount("emmc0", "/sys", "vfat") == -ENODEV);
const auto vfs_vfat = std::make_shared<fs::drivers::filesystem_vfat>();
REQUIRE(vfs_vfat->mount_count() == 0);
auto ret = fscore.register_filesystem("vfat", vfs_vfat);
REQUIRE(ret == 0);
ret = fscore.mount("emmc0part1", "/sys", "vfat");
REQUIRE(ret == 0);
REQUIRE(vfs_vfat->mount_count() == 1);
REQUIRE(fscore.umount("/ala") == -ENOENT);
ret = fscore.mount("emmc0part1", "/sys", "vfat");
REQUIRE(ret == -EBUSY);
ret = fscore.mount("emmc0part1", "/path", "vfat");
REQUIRE(ret == -EBUSY);
ret = fscore.mount("emmc0part2", "/path", "nonexisting_fs");
REQUIRE(ret == -ENODEV);
ret = fscore.umount("/sys");
REQUIRE(ret == 0);
REQUIRE(vfs_vfat->mount_count() == 0);
ret = fscore.mount("emmc0part1", "/path", "vfat");
REQUIRE(ret == 0);
REQUIRE(vfs_vfat->mount_count() == 1);
ret = fscore.umount("/path");
REQUIRE(ret == 0);
}
TEST_CASE("Corefs: Basic API test")
{
using namespace purefs;
auto dm = std::make_shared<blkdev::disk_manager>();
auto disk = std::make_shared<blkdev::disk_image>(disk_image);
REQUIRE(disk);
REQUIRE(dm->register_device(disk, "emmc0") == 0);
purefs::fs::filesystem fscore(dm);
const auto vfs_vfat = std::make_shared<fs::drivers::filesystem_vfat>();
REQUIRE(vfs_vfat->mount_count() == 0);
auto ret = fscore.register_filesystem("vfat", vfs_vfat);
REQUIRE(ret == 0);
// List partitions
for (const auto &part : dm->partitions("emmc0")) {
std::cout << part.name << " " << part.bootable << std::endl;
}
ret = fscore.mount("emmc0part0", "/sys", "vfat");
REQUIRE(ret == 0);
{
struct statvfs ssv;
ret = fscore.stat_vfs("/sys/", ssv);
REQUIRE(ret == 0);
}
{
ret = fscore.open("/sys/ala/ma/kota/", 0, 0);
REQUIRE(ret == -ENOENT);
// Simple file test
int hwnd = fscore.open("/sys/.boot.json", 0, 0);
REQUIRE(hwnd >= 3);
std::cout << "File open handle " << hwnd << std::endl;
struct stat st;
ret = fscore.fstat(hwnd, st);
REQUIRE(ret == 0);
std::cout << "File size " << st.st_size << std::endl;
char buf[4096]{};
ret = fscore.read(hwnd, buf, sizeof buf);
REQUIRE(ret > 0);
ret = fscore.close(hwnd);
REQUIRE(ret == 0);
{
// Simple directory test
auto dirhandle = fscore.diropen("/sys");
REQUIRE(dirhandle);
REQUIRE(dirhandle->error() == 0);
for (std::string fnm;;) {
if (fscore.dirnext(dirhandle, fnm, st) != 0) {
break;
}
else {
std::cout << "name " << fnm << " size " << st.st_size << std::endl;
}
}
fscore.dirclose(dirhandle);
dirhandle = nullptr;
}
}
REQUIRE(fscore.umount("/sys") == 0);
}
TEST_CASE("Corefs: Create new file, write, read from it")
{
using namespace purefs;
auto dm = std::make_shared<blkdev::disk_manager>();
auto disk = std::make_shared<blkdev::disk_image>(disk_image);
REQUIRE(disk);
REQUIRE(dm->register_device(disk, "emmc0") == 0);
purefs::fs::filesystem fscore(dm);
const auto vfs_vfat = std::make_shared<fs::drivers::filesystem_vfat>();
REQUIRE(vfs_vfat->mount_count() == 0);
auto ret = fscore.register_filesystem("vfat", vfs_vfat);
REQUIRE(ret == 0);
REQUIRE(fscore.mount("emmc0part0", "/sys", "vfat") == 0);
int hwnd = fscore.open("/sys/test.txt", O_RDWR | O_CREAT, 0660);
REQUIRE(hwnd >= 3);
const std::string text = "test";
fscore.write(hwnd, text.c_str(), text.size());
REQUIRE(fscore.close(hwnd) == 0);
SECTION("Read from file")
{
int hwnd = fscore.open("/sys/test.txt", O_RDONLY, 0);
REQUIRE(hwnd >= 3);
char buf[sizeof(text.c_str())] = {0};
REQUIRE(fscore.read(hwnd, buf, sizeof(buf)) == 4);
REQUIRE(strcmp(buf, text.c_str()) == 0);
fscore.close(hwnd);
REQUIRE(fscore.umount("/sys") == 0);
}
SECTION("Test seek file")
{
int hwnd = fscore.open("/sys/test.txt", O_RDONLY, 0);
REQUIRE(hwnd >= 3);
char buf[4096]{};
REQUIRE(fscore.read(hwnd, buf, sizeof(buf)) == 4);
REQUIRE(fscore.seek(hwnd, 0, SEEK_END) == 4);
REQUIRE(fscore.read(hwnd, buf, sizeof(buf)) == 0);
REQUIRE(fscore.seek(hwnd, 0, SEEK_SET) == 0);
fscore.close(hwnd);
REQUIRE(fscore.umount("/sys") == 0);
}
}
TEST_CASE("Corefs: Register null filesystem")
{
using namespace purefs;
auto dm = std::make_shared<blkdev::disk_manager>();
auto disk = std::make_shared<blkdev::disk_image>(disk_image);
REQUIRE(disk);
REQUIRE(dm->register_device(disk, "emmc0") == 0);
purefs::fs::filesystem fscore(dm);
REQUIRE(fscore.register_filesystem("vfat", nullptr) == -EINVAL);
}
TEST_CASE("Corefs: Mount empty strings")
{
using namespace purefs;
auto dm = std::make_shared<blkdev::disk_manager>();
auto disk = std::make_shared<blkdev::disk_image>(disk_image);
REQUIRE(disk);
REQUIRE(dm->register_device(disk, "emmc0") == 0);
purefs::fs::filesystem fscore(dm);
const auto vfs_vfat = std::make_shared<fs::drivers::filesystem_vfat>();
REQUIRE(vfs_vfat->mount_count() == 0);
auto ret = fscore.register_filesystem("vfat", vfs_vfat);
REQUIRE(ret == 0);
REQUIRE(fscore.mount("", "", "") == -EINVAL);
REQUIRE(fscore.umount("") == -ENOENT);
}
TEST_CASE("Corefs: Write to not valid file descriptor")
{
using namespace purefs;
auto dm = std::make_shared<blkdev::disk_manager>();
auto disk = std::make_shared<blkdev::disk_image>(disk_image);
REQUIRE(disk);
REQUIRE(dm->register_device(disk, "emmc0") == 0);
purefs::fs::filesystem fscore(dm);
const auto vfs_vfat = std::make_shared<fs::drivers::filesystem_vfat>();
REQUIRE(vfs_vfat->mount_count() == 0);
auto ret = fscore.register_filesystem("vfat", vfs_vfat);
REQUIRE(ret == 0);
REQUIRE(fscore.mount("emmc0part0", "/sys", "vfat") == 0);
const auto fd = fscore.open("/sys/.boot.json", 0, 0);
REQUIRE(fd >= 3);
const auto text = "test";
REQUIRE(fscore.write(0, text, sizeof(text)) == -EBADF);
REQUIRE(fscore.close(fd) == 0);
REQUIRE(fscore.umount("/sys") == 0);
}
TEST_CASE("Corefs: Directory operations")
{
using namespace purefs;
auto dm = std::make_shared<blkdev::disk_manager>();
auto disk = std::make_shared<blkdev::disk_image>(disk_image);
REQUIRE(disk);
REQUIRE(dm->register_device(disk, "emmc0") == 0);
purefs::fs::filesystem fscore(dm);
const auto vfs_vfat = std::make_shared<fs::drivers::filesystem_vfat>();
REQUIRE(vfs_vfat->mount_count() == 0);
auto ret = fscore.register_filesystem("vfat", vfs_vfat);
REQUIRE(ret == 0);
REQUIRE(fscore.mount("emmc0part0", "/sys", "vfat") == 0);
const auto dirhandle = fscore.diropen("/sys/current");
REQUIRE(dirhandle);
REQUIRE(dirhandle->error() == 0);
SECTION("Null pointer handle dirnext")
{
struct stat st;
std::string fnm;
REQUIRE(fscore.dirnext(nullptr, fnm, st) == -ENXIO);
REQUIRE(fscore.dirclose(dirhandle) == 0);
REQUIRE(fscore.umount("/sys") == 0);
}
SECTION("Null pointer handle dirclose")
{
REQUIRE(fscore.dirclose(nullptr) == -ENXIO);
REQUIRE(fscore.dirclose(dirhandle) == 0);
REQUIRE(fscore.umount("/sys") == 0);
}
SECTION("Directory reset")
{
struct stat st;
std::vector<std::tuple<std::string, struct stat>> vec;
for (std::string fnm;;) {
if (fscore.dirnext(dirhandle, fnm, st) != 0) {
break;
}
else {
vec.push_back(std::make_tuple(fnm, st));
std::cout << "name " << fnm << " size " << st.st_size << std::endl;
}
}
fscore.dirreset(dirhandle);
int i = 0;
for (std::string fnm;; i++) {
if (fscore.dirnext(dirhandle, fnm, st) != 0) {
break;
}
else {
const auto [fnm_vec, st_vec] = vec[i];
REQUIRE(fnm == fnm_vec);
REQUIRE(st_vec.st_size == st.st_size);
std::cout << "name " << fnm << " size " << st.st_size << std::endl;
}
}
REQUIRE(fscore.dirclose(dirhandle) == 0);
REQUIRE(fscore.umount("/sys") == 0);
}
}
TEST_CASE("Corefs: Read only filesystem")
{
using namespace purefs;
auto dm = std::make_shared<blkdev::disk_manager>();
auto disk = std::make_shared<blkdev::disk_image>(disk_image);
REQUIRE(disk);
REQUIRE(dm->register_device(disk, "emmc0") == 0);
purefs::fs::filesystem fscore(dm);
const auto vfs_vfat = std::make_shared<fs::drivers::filesystem_vfat>();
REQUIRE(vfs_vfat->mount_count() == 0);
auto ret = fscore.register_filesystem("vfat", vfs_vfat);
REQUIRE(ret == 0);
REQUIRE(fscore.mount("emmc0part0", "/sys", "vfat", fs::mount_flags::read_only) == 0);
SECTION("Open file in O_RDWR")
{
int hwnd = fscore.open("/sys/rotest.txt", O_RDWR | O_CREAT, 0660);
REQUIRE(hwnd == -EACCES);
const std::string text = "test";
fscore.write(hwnd, text.c_str(), text.size());
}
SECTION("Check function which not modify fs")
{
struct statvfs ssv;
ret = fscore.stat_vfs("/sys/", ssv);
REQUIRE(ret == 0);
}
SECTION("Check stat to not set S_IW...")
{
struct stat st;
ret = fscore.stat("/sys/current", st);
REQUIRE(ret == 0);
REQUIRE(st.st_mode & S_IFDIR);
REQUIRE((st.st_mode & (S_IWGRP | S_IWUSR | S_IWOTH)) == 0);
}
REQUIRE(fscore.umount("/sys") == 0);
}
TEST_CASE("Corefs: Remount filesystem from RO to RW and to RO")
{
using namespace purefs;
auto dm = std::make_shared<blkdev::disk_manager>();
auto disk = std::make_shared<blkdev::disk_image>(disk_image);
REQUIRE(disk);
REQUIRE(dm->register_device(disk, "emmc0") == 0);
auto fscore = std::make_shared<purefs::fs::filesystem>(dm);
const auto vfs_vfat = std::make_shared<fs::drivers::filesystem_vfat>();
REQUIRE(vfs_vfat->mount_count() == 0);
auto ret = fscore->register_filesystem("vfat", vfs_vfat);
REQUIRE(ret == 0);
REQUIRE(fscore->mount("emmc0part0", "/sys", "vfat", fs::mount_flags::read_only) == 0);
{
const int hwnd = fscore->open("/sys/remount_test.txt", O_RDWR | O_CREAT, 0660);
REQUIRE(hwnd == -EACCES);
}
REQUIRE(fscore->mount("", "/sys", "", fs::mount_flags::remount) == 0);
{
int hwnd = fscore->open("/sys/remount_test4.txt", O_RDWR | O_CREAT, 0660);
REQUIRE(hwnd > 2);
const std::string text = "test";
fscore->write(hwnd, text.c_str(), text.size());
REQUIRE(fscore->close(hwnd) == 0);
}
{
struct stat st;
ret = fscore->stat("/sys/current", st);
REQUIRE(ret == 0);
REQUIRE(st.st_mode & S_IFDIR);
REQUIRE(st.st_mode & (S_IWGRP | S_IWUSR | S_IWOTH));
}
REQUIRE(fscore->umount("/sys") == 0);
}
TEST_CASE("Corefs: Autodetect filesystems")
{
using namespace purefs;
auto dm = std::make_shared<blkdev::disk_manager>();
auto disk = std::make_shared<blkdev::disk_image>(disk_image);
REQUIRE(disk);
REQUIRE(dm->register_device(disk, "emmc0") == 0);
auto fscore = std::make_shared<purefs::fs::filesystem>(dm);
const auto vfs_vfat = std::make_shared<fs::drivers::filesystem_vfat>();
REQUIRE(vfs_vfat->mount_count() == 0);
auto ret = fscore->register_filesystem("vfat", vfs_vfat);
REQUIRE(ret == 0);
REQUIRE(fscore->mount("emmc0part0", "/sys", "auto") == 0);
REQUIRE(fscore->umount("/sys") == 0);
}
TEST_CASE("Corefs: Unittest integrated subsystem")
{
auto [disk, vfs] = purefs::subsystem::initialize();
REQUIRE(purefs::subsystem::mount_defaults() == 0);
REQUIRE(purefs::fs::internal::get_thread_local_cwd_path() == "/sys/current");
REQUIRE(vfs->getcwd() == "/sys/current");
REQUIRE(purefs::subsystem::unmount_all() == 0);
}