2014-04-07 16:15:40 +02:00
|
|
|
/*
|
|
|
|
* \brief File system that hosts a single node
|
|
|
|
* \author Norman Feske
|
|
|
|
* \date 2014-04-07
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2017-02-20 13:23:52 +01:00
|
|
|
* Copyright (C) 2014-2017 Genode Labs GmbH
|
2014-04-07 16:15:40 +02:00
|
|
|
*
|
|
|
|
* This file is part of the Genode OS framework, which is distributed
|
2017-02-20 13:23:52 +01:00
|
|
|
* under the terms of the GNU Affero General Public License version 3.
|
2014-04-07 16:15:40 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _INCLUDE__VFS__SINGLE_FILE_SYSTEM_H_
|
|
|
|
#define _INCLUDE__VFS__SINGLE_FILE_SYSTEM_H_
|
|
|
|
|
|
|
|
#include <vfs/file_system.h>
|
2014-08-19 15:38:28 +02:00
|
|
|
#include <vfs/vfs_handle.h>
|
2014-04-07 16:15:40 +02:00
|
|
|
|
|
|
|
namespace Vfs { class Single_file_system; }
|
|
|
|
|
|
|
|
|
|
|
|
class Vfs::Single_file_system : public File_system
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
2016-06-27 10:47:23 +02:00
|
|
|
enum Node_type {
|
|
|
|
NODE_TYPE_FILE, NODE_TYPE_SYMLINK,
|
|
|
|
NODE_TYPE_CHAR_DEVICE, NODE_TYPE_BLOCK_DEVICE
|
|
|
|
};
|
2014-04-07 16:15:40 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
Node_type const _node_type;
|
|
|
|
|
|
|
|
enum { FILENAME_MAX_LEN = 64 };
|
|
|
|
char _filename[FILENAME_MAX_LEN];
|
|
|
|
|
2016-06-27 10:47:23 +02:00
|
|
|
protected:
|
|
|
|
|
2016-05-11 18:21:47 +02:00
|
|
|
bool _root(const char *path)
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
|
|
|
return (strcmp(path, "") == 0) || (strcmp(path, "/") == 0);
|
|
|
|
}
|
|
|
|
|
2016-05-11 18:21:47 +02:00
|
|
|
bool _single_file(const char *path)
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
|
|
|
return (strlen(path) == (strlen(_filename) + 1)) &&
|
|
|
|
(strcmp(&path[1], _filename) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
Single_file_system(Node_type node_type, char const *type_name, Xml_node config)
|
|
|
|
:
|
|
|
|
_node_type(node_type)
|
|
|
|
{
|
|
|
|
strncpy(_filename, type_name, sizeof(_filename));
|
|
|
|
|
|
|
|
try { config.attribute("name").value(_filename, sizeof(_filename)); }
|
|
|
|
catch (...) { }
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************
|
|
|
|
** Directory-service interface **
|
|
|
|
*********************************/
|
|
|
|
|
|
|
|
Dataspace_capability dataspace(char const *path) override
|
|
|
|
{
|
|
|
|
return Dataspace_capability();
|
|
|
|
}
|
|
|
|
|
|
|
|
void release(char const *path, Dataspace_capability ds_cap) override { }
|
|
|
|
|
|
|
|
Stat_result stat(char const *path, Stat &out) override
|
|
|
|
{
|
2016-06-22 15:38:42 +02:00
|
|
|
out = Stat();
|
2016-04-01 16:10:40 +02:00
|
|
|
out.device = (Genode::addr_t)this;
|
2014-04-07 16:15:40 +02:00
|
|
|
|
2016-05-11 18:21:47 +02:00
|
|
|
if (_root(path)) {
|
2014-04-07 16:15:40 +02:00
|
|
|
out.mode = STAT_MODE_DIRECTORY;
|
|
|
|
|
2016-05-11 18:21:47 +02:00
|
|
|
} else if (_single_file(path)) {
|
2014-04-07 16:15:40 +02:00
|
|
|
switch (_node_type) {
|
|
|
|
case NODE_TYPE_FILE: out.mode = STAT_MODE_FILE; break;
|
2016-06-27 10:47:23 +02:00
|
|
|
case NODE_TYPE_SYMLINK: out.mode = STAT_MODE_SYMLINK; break;
|
2014-04-07 16:15:40 +02:00
|
|
|
case NODE_TYPE_CHAR_DEVICE: out.mode = STAT_MODE_CHARDEV; break;
|
|
|
|
case NODE_TYPE_BLOCK_DEVICE: out.mode = STAT_MODE_BLOCKDEV; break;
|
|
|
|
}
|
2016-04-01 16:10:40 +02:00
|
|
|
out.inode = 1;
|
2014-04-07 16:15:40 +02:00
|
|
|
} else {
|
|
|
|
return STAT_ERR_NO_ENTRY;
|
|
|
|
}
|
|
|
|
return STAT_OK;
|
|
|
|
}
|
|
|
|
|
2014-09-09 14:32:31 +02:00
|
|
|
Dirent_result dirent(char const *path, file_offset index, Dirent &out) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
2016-05-11 18:21:47 +02:00
|
|
|
if (!_root(path))
|
2014-04-07 16:15:40 +02:00
|
|
|
return DIRENT_ERR_INVALID_PATH;
|
|
|
|
|
|
|
|
if (index == 0) {
|
2016-04-01 16:10:40 +02:00
|
|
|
out.fileno = (Genode::addr_t)this;
|
2014-04-07 16:15:40 +02:00
|
|
|
switch (_node_type) {
|
|
|
|
case NODE_TYPE_FILE: out.type = DIRENT_TYPE_FILE; break;
|
2016-06-27 10:47:23 +02:00
|
|
|
case NODE_TYPE_SYMLINK: out.type = DIRENT_TYPE_SYMLINK; break;
|
2014-04-07 16:15:40 +02:00
|
|
|
case NODE_TYPE_CHAR_DEVICE: out.type = DIRENT_TYPE_CHARDEV; break;
|
|
|
|
case NODE_TYPE_BLOCK_DEVICE: out.type = DIRENT_TYPE_BLOCKDEV; break;
|
|
|
|
}
|
|
|
|
strncpy(out.name, _filename, sizeof(out.name));
|
|
|
|
} else {
|
|
|
|
out.type = DIRENT_TYPE_END;
|
|
|
|
}
|
|
|
|
|
|
|
|
return DIRENT_OK;
|
|
|
|
}
|
|
|
|
|
2014-09-09 14:32:31 +02:00
|
|
|
file_size num_dirent(char const *path) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
2016-05-11 18:21:47 +02:00
|
|
|
if (_root(path))
|
2014-04-07 16:15:40 +02:00
|
|
|
return 1;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-05-11 18:21:47 +02:00
|
|
|
bool directory(char const *path) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
2016-05-11 18:21:47 +02:00
|
|
|
if (_root(path))
|
2014-04-07 16:15:40 +02:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
char const *leaf_path(char const *path) override
|
|
|
|
{
|
2016-05-11 18:21:47 +02:00
|
|
|
return _single_file(path) ? path : 0;
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
|
2016-03-30 15:24:19 +02:00
|
|
|
Open_result open(char const *path, unsigned,
|
|
|
|
Vfs_handle **out_handle,
|
|
|
|
Allocator &alloc) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
2016-05-11 18:21:47 +02:00
|
|
|
if (!_single_file(path))
|
2014-04-07 16:15:40 +02:00
|
|
|
return OPEN_ERR_UNACCESSIBLE;
|
|
|
|
|
2016-03-30 15:24:19 +02:00
|
|
|
*out_handle = new (alloc) Vfs_handle(*this, *this, alloc, 0);
|
2014-04-07 16:15:40 +02:00
|
|
|
return OPEN_OK;
|
|
|
|
}
|
|
|
|
|
2016-03-30 15:24:19 +02:00
|
|
|
void close(Vfs_handle *handle) override
|
|
|
|
{
|
|
|
|
if (handle && (&handle->ds() == this))
|
|
|
|
destroy(handle->alloc(), handle);
|
|
|
|
}
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
Unlink_result unlink(char const *) override
|
|
|
|
{
|
|
|
|
return UNLINK_ERR_NO_PERM;
|
|
|
|
}
|
|
|
|
|
2014-09-09 14:32:31 +02:00
|
|
|
Readlink_result readlink(char const *, char *, file_size,
|
|
|
|
file_size &) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
|
|
|
return READLINK_ERR_NO_ENTRY;
|
|
|
|
}
|
|
|
|
|
2016-03-02 14:32:43 +01:00
|
|
|
Rename_result rename(char const *from, char const *to) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
2016-05-11 18:21:47 +02:00
|
|
|
if (_single_file(from) || _single_file(to))
|
2016-03-02 14:32:43 +01:00
|
|
|
return RENAME_ERR_NO_PERM;
|
|
|
|
return RENAME_ERR_NO_ENTRY;
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Mkdir_result mkdir(char const *, unsigned) override
|
|
|
|
{
|
|
|
|
return MKDIR_ERR_NO_PERM;
|
|
|
|
}
|
|
|
|
|
|
|
|
Symlink_result symlink(char const *, char const *) override
|
|
|
|
{
|
|
|
|
return SYMLINK_ERR_NO_ENTRY;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/********************************
|
|
|
|
** File I/O service interface **
|
|
|
|
********************************/
|
|
|
|
|
2014-09-09 14:32:31 +02:00
|
|
|
Ftruncate_result ftruncate(Vfs_handle *vfs_handle, file_size) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
|
|
|
return FTRUNCATE_ERR_NO_PERM;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif /* _INCLUDE__VFS__SINGLE_FILE_SYSTEM_H_ */
|