2014-04-07 16:15:40 +02:00
|
|
|
/*
|
|
|
|
* \brief Directory file system
|
|
|
|
* \author Norman Feske
|
2017-02-01 11:28:15 +01:00
|
|
|
* \author Emery Hemingway
|
|
|
|
* \author Christian Helmuth
|
2014-04-07 16:15:40 +02:00
|
|
|
* \date 2012-04-23
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2017-02-01 11:28:15 +01:00
|
|
|
* Copyright (C) 2011-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__DIR_FILE_SYSTEM_H_
|
|
|
|
#define _INCLUDE__VFS__DIR_FILE_SYSTEM_H_
|
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
#include <base/registry.h>
|
2014-04-07 16:15:40 +02:00
|
|
|
#include <vfs/file_system_factory.h>
|
|
|
|
#include <vfs/vfs_handle.h>
|
|
|
|
|
|
|
|
|
|
|
|
namespace Vfs { class Dir_file_system; }
|
|
|
|
|
|
|
|
|
|
|
|
class Vfs::Dir_file_system : public File_system
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
enum { MAX_NAME_LEN = 128 };
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
struct Dir_vfs_handle : Vfs_handle
|
|
|
|
{
|
|
|
|
struct Sync_dir_handle_element;
|
|
|
|
|
|
|
|
typedef Genode::Registry<Sync_dir_handle_element> Sync_dir_handle_registry;
|
|
|
|
|
|
|
|
struct Sync_dir_handle_element : Sync_dir_handle_registry::Element
|
|
|
|
{
|
|
|
|
Vfs_handle &vfs_handle;
|
|
|
|
Sync_dir_handle_element(Sync_dir_handle_registry ®istry,
|
|
|
|
Vfs_handle &vfs_handle)
|
|
|
|
: Sync_dir_handle_registry::Element(registry, *this),
|
|
|
|
vfs_handle(vfs_handle) { }
|
|
|
|
};
|
|
|
|
|
|
|
|
Absolute_path path;
|
|
|
|
File_system *fs_for_complete_read { nullptr };
|
|
|
|
Vfs_handle *fs_dir_handle { nullptr };
|
|
|
|
Sync_dir_handle_registry sync_dir_handle_registry;
|
|
|
|
|
|
|
|
Dir_vfs_handle(Directory_service &ds,
|
|
|
|
File_io_service &fs,
|
|
|
|
Genode::Allocator &alloc,
|
|
|
|
char const *path)
|
|
|
|
: Vfs_handle(ds, fs, alloc, 0),
|
|
|
|
path(path) { }
|
|
|
|
};
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
/* pointer to first child file system */
|
|
|
|
File_system *_first_file_system;
|
|
|
|
|
|
|
|
/* add new file system to the list of children */
|
|
|
|
void _append_file_system(File_system *fs)
|
|
|
|
{
|
|
|
|
if (!_first_file_system) {
|
|
|
|
_first_file_system = fs;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
File_system *curr = _first_file_system;
|
|
|
|
while (curr->next)
|
|
|
|
curr = curr->next;
|
|
|
|
|
|
|
|
curr->next = fs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Directory name
|
|
|
|
*/
|
|
|
|
char _name[MAX_NAME_LEN];
|
|
|
|
|
2016-05-11 18:21:47 +02:00
|
|
|
bool _root() const { return _name[0] == 0; }
|
2014-04-07 16:15:40 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Perform operation on a file system
|
|
|
|
*
|
|
|
|
* \param fn functor that takes a file-system reference and
|
|
|
|
* the path as arguments
|
|
|
|
*/
|
|
|
|
template <typename RES, typename FN>
|
|
|
|
RES _dir_op(RES const no_entry, RES const no_perm, RES const ok,
|
|
|
|
char const *path, FN const &fn)
|
|
|
|
{
|
|
|
|
path = _sub_path(path);
|
|
|
|
|
|
|
|
/* path does not match directory name */
|
|
|
|
if (!path)
|
|
|
|
return no_entry;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Prevent operation if path equals directory name defined
|
|
|
|
* via the static VFS configuration.
|
|
|
|
*/
|
|
|
|
if (strlen(path) == 0)
|
|
|
|
return no_perm;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If any of the sub file systems returns a permission error and
|
|
|
|
* there exists no sub file system that takes the request, we
|
|
|
|
* return the permission error.
|
|
|
|
*/
|
|
|
|
bool permission_denied = false;
|
|
|
|
|
2015-02-04 19:23:01 +01:00
|
|
|
/*
|
|
|
|
* Keep the most meaningful error code. When using stacked file
|
|
|
|
* systems, most child file systems will eventually return no
|
|
|
|
* entry (or leave the error code unchanged). If any of those
|
|
|
|
* file systems has anything more interesting to tell, return
|
|
|
|
* this information after all file systems have been tried and
|
|
|
|
* none could handle the request.
|
|
|
|
*/
|
|
|
|
RES error = ok;
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
/*
|
|
|
|
* The given path refers to at least one of our sub directories.
|
|
|
|
* Propagate the request into all of our file systems. If at least
|
|
|
|
* one operation succeeds, we return success.
|
|
|
|
*/
|
|
|
|
for (File_system *fs = _first_file_system; fs; fs = fs->next) {
|
|
|
|
|
|
|
|
RES const err = fn(*fs, path);
|
|
|
|
|
|
|
|
if (err == ok)
|
|
|
|
return err;
|
|
|
|
|
2015-02-04 19:23:01 +01:00
|
|
|
if (err != no_entry && err != no_perm) {
|
|
|
|
error = err;
|
|
|
|
}
|
2014-04-07 16:15:40 +02:00
|
|
|
|
|
|
|
if (err == no_perm)
|
|
|
|
permission_denied = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* none of our file systems could successfully operate on the path */
|
2015-02-04 19:23:01 +01:00
|
|
|
return error != ok ? error : permission_denied ? no_perm : no_entry;
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return portion of the path after the element corresponding to
|
|
|
|
* the current directory.
|
|
|
|
*/
|
|
|
|
char const *_sub_path(char const *path) const
|
|
|
|
{
|
|
|
|
/* do not strip anything from the path when we are root */
|
2016-05-11 18:21:47 +02:00
|
|
|
if (_root())
|
2014-04-07 16:15:40 +02:00
|
|
|
return path;
|
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
if (strcmp(path, "/") == 0)
|
|
|
|
return path;
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
/* skip heading slash in path if present */
|
|
|
|
if (path[0] == '/')
|
|
|
|
path++;
|
|
|
|
|
2014-09-09 14:32:31 +02:00
|
|
|
Genode::size_t const name_len = strlen(_name);
|
2014-04-07 16:15:40 +02:00
|
|
|
if (strcmp(path, _name, name_len) != 0)
|
|
|
|
return 0;
|
|
|
|
path += name_len;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The first characters of the first path element are equal to
|
|
|
|
* the current directory name. Let's check if the length of the
|
|
|
|
* first path element matches the name length.
|
|
|
|
*/
|
|
|
|
if (*path != 0 && *path != '/')
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
/*
|
|
|
|
* Accumulate number of directory entries that match in any of
|
|
|
|
* our sub file systems.
|
2014-04-07 16:15:40 +02:00
|
|
|
*/
|
2017-08-15 20:51:53 +02:00
|
|
|
file_size _sum_dirents_of_file_systems(char const *path)
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
2017-08-15 20:51:53 +02:00
|
|
|
file_size cnt = 0;
|
|
|
|
for (File_system *fs = _first_file_system; fs; fs = fs->next) {
|
|
|
|
cnt += fs->num_dirent(path);
|
|
|
|
}
|
|
|
|
return cnt;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool _queue_read_of_file_systems(Dir_vfs_handle *dir_vfs_handle)
|
|
|
|
{
|
|
|
|
file_offset index = dir_vfs_handle->seek() / sizeof(Dirent);
|
|
|
|
|
|
|
|
char const *sub_path = _sub_path(dir_vfs_handle->path.base());
|
|
|
|
|
|
|
|
if (strlen(sub_path) == 0)
|
|
|
|
sub_path = "/";
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
int base = 0;
|
|
|
|
for (File_system *fs = _first_file_system; fs; fs = fs->next) {
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Determine number of matching directory entries within
|
|
|
|
* the current file system.
|
|
|
|
*/
|
2017-08-15 20:51:53 +02:00
|
|
|
int const fs_num_dirent = fs->num_dirent(sub_path);
|
2014-04-07 16:15:40 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Query directory entry if index lies with the file
|
|
|
|
* system.
|
|
|
|
*/
|
|
|
|
if (index - base < fs_num_dirent) {
|
2017-08-15 20:51:53 +02:00
|
|
|
|
|
|
|
dir_vfs_handle->fs_for_complete_read = fs;
|
|
|
|
|
|
|
|
Opendir_result opendir_result =
|
|
|
|
fs->opendir(sub_path, false,
|
|
|
|
&dir_vfs_handle->fs_dir_handle,
|
|
|
|
dir_vfs_handle->alloc());
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Errors of this kind can only be communicated by
|
|
|
|
* 'complete_read()'
|
|
|
|
*/
|
|
|
|
if (opendir_result != OPENDIR_OK)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
dir_vfs_handle->fs_dir_handle->context =
|
|
|
|
dir_vfs_handle->context;
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
index = index - base;
|
2017-08-15 20:51:53 +02:00
|
|
|
dir_vfs_handle->fs_dir_handle->seek(index * sizeof(Dirent));
|
|
|
|
|
|
|
|
bool result = fs->queue_read(dir_vfs_handle->fs_dir_handle,
|
|
|
|
sizeof(Dirent));
|
|
|
|
|
|
|
|
return result;
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* adjust base index for next file system */
|
|
|
|
base += fs_num_dirent;
|
|
|
|
}
|
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
return true;
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
Read_result _complete_read_of_file_systems(Dir_vfs_handle *dir_vfs_handle,
|
|
|
|
char *dst, file_size count,
|
|
|
|
file_size &out_count)
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
2017-08-15 20:51:53 +02:00
|
|
|
if (!dir_vfs_handle->fs_for_complete_read ||
|
|
|
|
!dir_vfs_handle->fs_dir_handle) {
|
2014-04-07 16:15:40 +02:00
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
/*
|
|
|
|
* no fs was found for the given index or
|
|
|
|
* fs->opendir() failed
|
|
|
|
*/
|
2014-04-07 16:15:40 +02:00
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
if (count < sizeof(Dirent))
|
|
|
|
return READ_ERR_INVALID;
|
|
|
|
|
|
|
|
Dirent *dirent = (Dirent*)dst;
|
|
|
|
*dirent = Dirent();
|
|
|
|
|
|
|
|
out_count = sizeof(Dirent);
|
|
|
|
|
|
|
|
return READ_OK;
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
2017-08-15 20:51:53 +02:00
|
|
|
|
|
|
|
Read_result result = dir_vfs_handle->fs_for_complete_read->
|
|
|
|
complete_read(dir_vfs_handle->fs_dir_handle,
|
|
|
|
dst, count, out_count);
|
|
|
|
|
|
|
|
if (result != READ_OK)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
dir_vfs_handle->fs_for_complete_read->close(dir_vfs_handle->fs_dir_handle);
|
|
|
|
dir_vfs_handle->fs_dir_handle = nullptr;
|
|
|
|
dir_vfs_handle->fs_for_complete_read = nullptr;
|
|
|
|
|
|
|
|
return result;
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
2016-05-25 15:47:22 +02:00
|
|
|
Dir_file_system(Genode::Env &env,
|
|
|
|
Genode::Allocator &alloc,
|
|
|
|
Genode::Xml_node node,
|
2017-01-31 16:38:23 +01:00
|
|
|
Io_response_handler &io_handler,
|
2016-05-25 15:47:22 +02:00
|
|
|
File_system_factory &fs_factory)
|
2014-04-07 16:15:40 +02:00
|
|
|
:
|
|
|
|
_first_file_system(0)
|
|
|
|
{
|
2015-11-23 13:39:29 +01:00
|
|
|
using namespace Genode;
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
/* remember directory name */
|
|
|
|
if (node.has_type("fstab") || node.has_type("vfs"))
|
|
|
|
_name[0] = 0;
|
|
|
|
else
|
|
|
|
node.attribute("name").value(_name, sizeof(_name));
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < node.num_sub_nodes(); i++) {
|
|
|
|
|
|
|
|
Xml_node sub_node = node.sub_node(i);
|
|
|
|
|
|
|
|
/* traverse into <dir> nodes */
|
|
|
|
if (sub_node.has_type("dir")) {
|
2016-05-25 15:47:22 +02:00
|
|
|
_append_file_system(new (alloc)
|
2017-01-31 16:38:23 +01:00
|
|
|
Dir_file_system(env, alloc, sub_node, io_handler, fs_factory));
|
2014-04-07 16:15:40 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-01-31 16:38:23 +01:00
|
|
|
File_system *fs = fs_factory.create(env, alloc, sub_node, io_handler);
|
2014-04-07 16:15:40 +02:00
|
|
|
if (fs) {
|
|
|
|
_append_file_system(fs);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
base: avoid use of deprecated base/printf.h
Besides adapting the components to the use of base/log.h, the patch
cleans up a few base headers, i.e., it removes unused includes from
root/component.h, specifically base/heap.h and
ram_session/ram_session.h. Hence, components that relied on the implicit
inclusion of those headers have to manually include those headers now.
While adjusting the log messages, I repeatedly stumbled over the problem
that printing char * arguments is ambiguous. It is unclear whether to
print the argument as pointer or null-terminated string. To overcome
this problem, the patch introduces a new type 'Cstring' that allows the
caller to express that the argument should be handled as null-terminated
string. As a nice side effect, with this type in place, the optional len
argument of the 'String' class could be removed. Instead of supplying a
pair of (char const *, size_t), the constructor accepts a 'Cstring'.
This, in turn, clears the way let the 'String' constructor use the new
output mechanism to assemble a string from multiple arguments (and
thereby getting rid of snprintf within Genode in the near future).
To enforce the explicit resolution of the char * ambiguity, the 'char *'
overload of the 'print' function is marked as deleted.
Issue #1987
2016-07-13 19:07:09 +02:00
|
|
|
Genode::error("failed to create <", sub_node.type(), "> VFS node");
|
2015-11-23 13:39:29 +01:00
|
|
|
try {
|
|
|
|
String<64> value;
|
|
|
|
for (unsigned i = 0; i < 16; ++i) {
|
|
|
|
Xml_attribute attr = sub_node.attribute(i);
|
|
|
|
attr.value(&value);
|
|
|
|
|
base: avoid use of deprecated base/printf.h
Besides adapting the components to the use of base/log.h, the patch
cleans up a few base headers, i.e., it removes unused includes from
root/component.h, specifically base/heap.h and
ram_session/ram_session.h. Hence, components that relied on the implicit
inclusion of those headers have to manually include those headers now.
While adjusting the log messages, I repeatedly stumbled over the problem
that printing char * arguments is ambiguous. It is unclear whether to
print the argument as pointer or null-terminated string. To overcome
this problem, the patch introduces a new type 'Cstring' that allows the
caller to express that the argument should be handled as null-terminated
string. As a nice side effect, with this type in place, the optional len
argument of the 'String' class could be removed. Instead of supplying a
pair of (char const *, size_t), the constructor accepts a 'Cstring'.
This, in turn, clears the way let the 'String' constructor use the new
output mechanism to assemble a string from multiple arguments (and
thereby getting rid of snprintf within Genode in the near future).
To enforce the explicit resolution of the char * ambiguity, the 'char *'
overload of the 'print' function is marked as deleted.
Issue #1987
2016-07-13 19:07:09 +02:00
|
|
|
Genode::error("\t", attr.name(), "=\"", value, "\"");
|
2015-11-23 13:39:29 +01:00
|
|
|
}
|
|
|
|
} catch (Xml_node::Nonexistent_attribute) { }
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************
|
|
|
|
** Directory-service interface **
|
|
|
|
*********************************/
|
|
|
|
|
|
|
|
Dataspace_capability dataspace(char const *path) override
|
|
|
|
{
|
|
|
|
path = _sub_path(path);
|
|
|
|
if (!path)
|
|
|
|
return Dataspace_capability();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Query sub file systems for dataspace using the path local to
|
|
|
|
* the respective file system
|
|
|
|
*/
|
|
|
|
File_system *fs = _first_file_system;
|
|
|
|
for (; fs; fs = fs->next) {
|
|
|
|
Dataspace_capability ds = fs->dataspace(path);
|
|
|
|
if (ds.valid())
|
|
|
|
return ds;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Dataspace_capability();
|
|
|
|
}
|
|
|
|
|
|
|
|
void release(char const *path, Dataspace_capability ds_cap) override
|
|
|
|
{
|
|
|
|
path = _sub_path(path);
|
|
|
|
if (!path)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (File_system *fs = _first_file_system; fs; fs = fs->next)
|
|
|
|
fs->release(path, ds_cap);
|
|
|
|
}
|
|
|
|
|
|
|
|
Stat_result stat(char const *path, Stat &out) override
|
|
|
|
{
|
|
|
|
path = _sub_path(path);
|
|
|
|
|
|
|
|
/* path does not match directory name */
|
|
|
|
if (!path)
|
|
|
|
return STAT_ERR_NO_ENTRY;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If path equals directory name, return information about the
|
|
|
|
* current directory.
|
|
|
|
*/
|
|
|
|
if (strlen(path) == 0 || (strcmp(path, "/") == 0)) {
|
2016-04-01 16:10:40 +02:00
|
|
|
out.size = 0;
|
|
|
|
out.mode = STAT_MODE_DIRECTORY | 0755;
|
|
|
|
out.uid = 0;
|
|
|
|
out.gid = 0;
|
|
|
|
out.inode = 1;
|
|
|
|
out.device = (Genode::addr_t)this;
|
2014-04-07 16:15:40 +02:00
|
|
|
return STAT_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The given path refers to one of our sub directories.
|
|
|
|
* Propagate the request into our file systems.
|
|
|
|
*/
|
|
|
|
for (File_system *fs = _first_file_system; fs; fs = fs->next) {
|
|
|
|
|
|
|
|
Stat_result const err = fs->stat(path, out);
|
|
|
|
|
|
|
|
if (err == STAT_OK)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
if (err != STAT_ERR_NO_ENTRY)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* none of our file systems felt responsible for the path */
|
|
|
|
return STAT_ERR_NO_ENTRY;
|
|
|
|
}
|
|
|
|
|
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()) {
|
2014-04-07 16:15:40 +02:00
|
|
|
return _sum_dirents_of_file_systems(path);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
if (strcmp(path, "/") == 0)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The path contains at least one element. Remove current
|
|
|
|
* element from path.
|
|
|
|
*/
|
|
|
|
path = _sub_path(path);
|
|
|
|
|
|
|
|
/*
|
2015-03-20 17:50:41 +01:00
|
|
|
* If the resulting 'path' is non-null, the path lies
|
2014-04-07 16:15:40 +02:00
|
|
|
* within our tree. In this case, determine the sum of
|
|
|
|
* matching dirents of all our file systems. Otherwise,
|
|
|
|
* the specified path lies outside our directory node.
|
|
|
|
*/
|
2016-03-31 16:42:28 +02:00
|
|
|
return path ? _sum_dirents_of_file_systems(*path ? path : "/") : 0;
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-11 18:21:47 +02:00
|
|
|
/**
|
|
|
|
* Return true if specified path is a directory
|
|
|
|
*/
|
|
|
|
bool directory(char const *path) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
2017-08-15 20:51:53 +02:00
|
|
|
if (strcmp(path, "/") == 0)
|
|
|
|
return true;
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
path = _sub_path(path);
|
2017-08-15 20:51:53 +02:00
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
if (!path)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (strlen(path) == 0)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
for (File_system *fs = _first_file_system; fs; fs = fs->next)
|
2016-05-11 18:21:47 +02:00
|
|
|
if (fs->directory(path))
|
2014-04-07 16:15:40 +02:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-05-11 18:21:47 +02:00
|
|
|
/**
|
|
|
|
* Return true if specified path is a directory
|
|
|
|
*
|
|
|
|
* \noapi
|
|
|
|
* \deprecated use 'directory instead
|
|
|
|
*/
|
|
|
|
bool is_directory(char const *path) { return directory(path); }
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
char const *leaf_path(char const *path) override
|
|
|
|
{
|
|
|
|
path = _sub_path(path);
|
|
|
|
if (!path)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (strlen(path) == 0)
|
|
|
|
return path;
|
|
|
|
|
|
|
|
for (File_system *fs = _first_file_system; fs; fs = fs->next) {
|
|
|
|
char const *leaf_path = fs->leaf_path(path);
|
|
|
|
if (leaf_path)
|
|
|
|
return leaf_path;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-03-30 15:24:19 +02:00
|
|
|
Open_result open(char const *path,
|
2017-02-01 21:07:14 +01:00
|
|
|
unsigned mode,
|
|
|
|
Vfs_handle **out_handle,
|
|
|
|
Allocator &alloc) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* If 'path' is a directory, we create a 'Vfs_handle'
|
|
|
|
* for the root directory so that subsequent 'dirent' calls
|
|
|
|
* are subjected to the stacked file-system layout.
|
|
|
|
*/
|
2016-05-11 18:21:47 +02:00
|
|
|
if (directory(path)) {
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If 'path' refers to a non-directory node, create a
|
|
|
|
* 'Vfs_handle' local to the file system that provides the
|
|
|
|
* file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
path = _sub_path(path);
|
|
|
|
|
|
|
|
/* check if path does not match directory name */
|
|
|
|
if (!path)
|
|
|
|
return OPEN_ERR_UNACCESSIBLE;
|
|
|
|
|
|
|
|
/* path equals directory name */
|
|
|
|
if (strlen(path) == 0) {
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* path refers to any of our sub file systems */
|
|
|
|
for (File_system *fs = _first_file_system; fs; fs = fs->next) {
|
|
|
|
|
2016-03-30 15:24:19 +02:00
|
|
|
Open_result const err = fs->open(path, mode, out_handle, alloc);
|
|
|
|
switch (err) {
|
|
|
|
case OPEN_ERR_UNACCESSIBLE:
|
|
|
|
continue;
|
|
|
|
default:
|
2014-04-07 16:15:40 +02:00
|
|
|
return err;
|
2016-03-30 15:24:19 +02:00
|
|
|
}
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* path does not match any existing file or directory */
|
|
|
|
return OPEN_ERR_UNACCESSIBLE;
|
|
|
|
}
|
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
Opendir_result opendir(char const *path, bool create,
|
|
|
|
Vfs_handle **out_handle, Allocator &alloc) override
|
|
|
|
{
|
|
|
|
/* path equals "/" (for reading the name of this directory) */
|
|
|
|
if (strcmp(path, "/") == 0) {
|
|
|
|
if (create)
|
|
|
|
return OPENDIR_ERR_PERMISSION_DENIED;
|
|
|
|
*out_handle = new (alloc) Dir_vfs_handle(*this, *this, alloc,
|
|
|
|
path);
|
|
|
|
return OPENDIR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
char const *sub_path = _sub_path(path);
|
|
|
|
|
|
|
|
if (!sub_path)
|
|
|
|
return OPENDIR_ERR_LOOKUP_FAILED;
|
|
|
|
|
|
|
|
if (create) {
|
|
|
|
auto opendir_fn = [&] (File_system &fs, char const *path)
|
|
|
|
{
|
|
|
|
Vfs_handle *tmp_handle;
|
|
|
|
Opendir_result opendir_result =
|
|
|
|
fs.opendir(path, true, &tmp_handle, alloc);
|
|
|
|
if (opendir_result == OPENDIR_OK)
|
|
|
|
fs.close(tmp_handle);
|
|
|
|
return opendir_result;
|
|
|
|
};
|
|
|
|
|
|
|
|
Opendir_result opendir_result =
|
|
|
|
_dir_op(OPENDIR_ERR_LOOKUP_FAILED,
|
|
|
|
OPENDIR_ERR_PERMISSION_DENIED,
|
|
|
|
OPENDIR_OK,
|
|
|
|
path, opendir_fn);
|
|
|
|
|
|
|
|
if (opendir_result != OPENDIR_OK)
|
|
|
|
return opendir_result;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_handle = new (alloc) Dir_vfs_handle(*this, *this, alloc,
|
|
|
|
path);
|
|
|
|
|
|
|
|
return OPENDIR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
Openlink_result openlink(char const *path, bool create,
|
|
|
|
Vfs_handle **out_handle,
|
|
|
|
Allocator &alloc) override
|
|
|
|
{
|
|
|
|
auto openlink_fn = [&] (File_system &fs, char const *path)
|
|
|
|
{
|
|
|
|
return fs.openlink(path, create, out_handle, alloc);
|
|
|
|
};
|
|
|
|
|
|
|
|
return _dir_op(OPENLINK_ERR_LOOKUP_FAILED,
|
|
|
|
OPENLINK_ERR_PERMISSION_DENIED,
|
|
|
|
OPENLINK_OK,
|
|
|
|
path, openlink_fn);
|
|
|
|
}
|
|
|
|
|
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 *path) override
|
|
|
|
{
|
|
|
|
auto unlink_fn = [] (File_system &fs, char const *path)
|
|
|
|
{
|
|
|
|
return fs.unlink(path);
|
|
|
|
};
|
|
|
|
|
|
|
|
return _dir_op(UNLINK_ERR_NO_ENTRY, UNLINK_ERR_NO_PERM, UNLINK_OK,
|
|
|
|
path, unlink_fn);
|
|
|
|
}
|
|
|
|
|
|
|
|
Rename_result rename(char const *from_path, char const *to_path) override
|
|
|
|
{
|
2016-03-09 08:56:49 +01:00
|
|
|
from_path = _sub_path(from_path);
|
|
|
|
to_path = _sub_path(to_path);
|
|
|
|
|
|
|
|
/* path does not match directory name */
|
|
|
|
if (!from_path)
|
|
|
|
return RENAME_ERR_NO_ENTRY;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Cannot rename a path in the static VFS configuration.
|
|
|
|
*/
|
|
|
|
if (strlen(from_path) == 0)
|
|
|
|
return RENAME_ERR_NO_PERM;
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
/*
|
|
|
|
* Check if destination path resides within the same file
|
|
|
|
* system instance as the source path.
|
|
|
|
*/
|
|
|
|
if (!to_path)
|
|
|
|
return RENAME_ERR_CROSS_FS;
|
|
|
|
|
2016-03-09 08:56:49 +01:00
|
|
|
Rename_result final = RENAME_ERR_NO_ENTRY;
|
|
|
|
for (File_system *fs = _first_file_system; fs; fs = fs->next) {
|
|
|
|
switch (fs->rename(from_path, to_path)) {
|
|
|
|
case RENAME_OK: return RENAME_OK;
|
|
|
|
case RENAME_ERR_NO_ENTRY: continue;
|
|
|
|
case RENAME_ERR_NO_PERM: return RENAME_ERR_NO_PERM;
|
|
|
|
case RENAME_ERR_CROSS_FS: final = RENAME_ERR_CROSS_FS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return final;
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/***************************
|
|
|
|
** File_system interface **
|
|
|
|
***************************/
|
|
|
|
|
2017-02-07 19:03:23 +01:00
|
|
|
char const *name() const { return "dir"; }
|
|
|
|
char const *type() override { return "dir"; }
|
2014-04-07 16:15:40 +02:00
|
|
|
|
2017-02-07 19:03:23 +01:00
|
|
|
void apply_config(Genode::Xml_node const &node) override
|
|
|
|
{
|
|
|
|
using namespace Genode;
|
|
|
|
|
|
|
|
File_system *curr = _first_file_system;
|
|
|
|
for (unsigned i = 0; i < node.num_sub_nodes(); i++, curr = curr->next) {
|
|
|
|
Xml_node const &sub_node = node.sub_node(i);
|
|
|
|
|
|
|
|
/* check if type of XML node matches current file-system type */
|
|
|
|
if (sub_node.has_type(curr->type()) == false) {
|
|
|
|
Genode::error("VFS config update failed (node type '",
|
|
|
|
sub_node.type(), "' != fs type '", curr->type(),"')");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
curr->apply_config(node.sub_node(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-07 16:15:40 +02:00
|
|
|
|
|
|
|
/********************************
|
|
|
|
** File I/O service interface **
|
|
|
|
********************************/
|
|
|
|
|
2014-09-09 14:32:31 +02:00
|
|
|
Write_result write(Vfs_handle *handle, char const *, file_size,
|
|
|
|
file_size &) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
|
|
|
return WRITE_ERR_INVALID;
|
|
|
|
}
|
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
bool queue_read(Vfs_handle *vfs_handle, file_size count) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
2017-08-15 20:51:53 +02:00
|
|
|
Dir_vfs_handle *dir_vfs_handle =
|
|
|
|
static_cast<Dir_vfs_handle*>(vfs_handle);
|
|
|
|
|
|
|
|
if (_root())
|
|
|
|
return _queue_read_of_file_systems(dir_vfs_handle);
|
|
|
|
|
|
|
|
if (strcmp(dir_vfs_handle->path.base(), "/") == 0)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return _queue_read_of_file_systems(dir_vfs_handle);
|
2014-04-07 16:15:40 +02:00
|
|
|
}
|
|
|
|
|
2017-08-15 20:51:53 +02:00
|
|
|
Read_result complete_read(Vfs_handle *vfs_handle,
|
|
|
|
char *dst, file_size count,
|
|
|
|
file_size &out_count) override
|
|
|
|
{
|
|
|
|
out_count = 0;
|
|
|
|
|
|
|
|
if (count < sizeof(Dirent))
|
|
|
|
return READ_ERR_INVALID;
|
|
|
|
|
|
|
|
Dir_vfs_handle *dir_vfs_handle =
|
|
|
|
static_cast<Dir_vfs_handle*>(vfs_handle);
|
|
|
|
|
|
|
|
if (_root())
|
|
|
|
return _complete_read_of_file_systems(dir_vfs_handle, dst, count, out_count);
|
|
|
|
|
|
|
|
if (strcmp(dir_vfs_handle->path.base(), "/") == 0) {
|
|
|
|
Dirent *dirent = (Dirent*)dst;
|
|
|
|
file_offset index = vfs_handle->seek() / sizeof(Dirent);
|
|
|
|
|
|
|
|
if (index == 0) {
|
|
|
|
strncpy(dirent->name, _name, sizeof(dirent->name));
|
|
|
|
|
|
|
|
dirent->type = DIRENT_TYPE_DIRECTORY;
|
|
|
|
dirent->fileno = 1;
|
|
|
|
} else {
|
|
|
|
dirent->type = DIRENT_TYPE_END;
|
|
|
|
}
|
|
|
|
|
|
|
|
out_count = sizeof(Dirent);
|
|
|
|
|
|
|
|
return READ_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return _complete_read_of_file_systems(dir_vfs_handle, dst, count, out_count);
|
|
|
|
}
|
|
|
|
|
2014-09-09 14:32:31 +02:00
|
|
|
Ftruncate_result ftruncate(Vfs_handle *, file_size) override
|
2014-04-07 16:15:40 +02:00
|
|
|
{
|
|
|
|
return FTRUNCATE_ERR_NO_PERM;
|
|
|
|
}
|
2017-02-01 11:28:15 +01:00
|
|
|
|
|
|
|
bool read_ready(Vfs_handle *handle) override
|
|
|
|
{
|
|
|
|
if (&handle->fs() == this)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return handle->fs().read_ready(handle);
|
|
|
|
}
|
2017-02-01 21:07:14 +01:00
|
|
|
|
|
|
|
bool notify_read_ready(Vfs_handle *handle) override
|
|
|
|
{
|
|
|
|
if (&handle->fs() == this)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return handle->fs().notify_read_ready(handle);
|
|
|
|
}
|
2017-08-15 20:51:53 +02:00
|
|
|
|
|
|
|
bool queue_sync(Vfs_handle *vfs_handle) override
|
|
|
|
{
|
|
|
|
Dir_vfs_handle *dir_vfs_handle =
|
|
|
|
static_cast<Dir_vfs_handle*>(vfs_handle);
|
|
|
|
|
|
|
|
char const *sub_path = _sub_path(dir_vfs_handle->path.base());
|
|
|
|
|
|
|
|
if (strlen(sub_path) == 0)
|
|
|
|
sub_path = "/";
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Call 'opendir()' on each file system and, if successful,
|
|
|
|
* 'queue_sync()'. If one file system's 'queue_sync()' returns
|
|
|
|
* false, this function returns false. Any VFS handles returned by
|
|
|
|
* 'opendir()' are kept in a registry.
|
|
|
|
*/
|
|
|
|
for (File_system *fs = _first_file_system; fs; fs = fs->next) {
|
|
|
|
|
|
|
|
bool fs_dir_already_open = false;
|
|
|
|
|
|
|
|
dir_vfs_handle->sync_dir_handle_registry.for_each(
|
|
|
|
[&] (Dir_vfs_handle::Sync_dir_handle_element &sync_dir_handle_element) {
|
|
|
|
if (&sync_dir_handle_element.vfs_handle.fs() == fs) {
|
|
|
|
fs_dir_already_open = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
if (fs_dir_already_open)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Vfs_handle *sync_dir_handle;
|
|
|
|
|
|
|
|
if (fs->opendir(sub_path, false, &sync_dir_handle,
|
|
|
|
dir_vfs_handle->alloc()) != OPENDIR_OK)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
sync_dir_handle->context = dir_vfs_handle->context;
|
|
|
|
|
|
|
|
new (dir_vfs_handle->alloc())
|
|
|
|
Dir_vfs_handle::Sync_dir_handle_element(
|
|
|
|
dir_vfs_handle->sync_dir_handle_registry,
|
|
|
|
*sync_dir_handle);
|
|
|
|
|
|
|
|
if (!sync_dir_handle->fs().queue_sync(sync_dir_handle))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Sync_result complete_sync(Vfs_handle *vfs_handle) override
|
|
|
|
{
|
|
|
|
Sync_result result = SYNC_OK;
|
|
|
|
|
|
|
|
Dir_vfs_handle *dir_vfs_handle =
|
|
|
|
static_cast<Dir_vfs_handle*>(vfs_handle);
|
|
|
|
|
|
|
|
dir_vfs_handle->sync_dir_handle_registry.for_each(
|
|
|
|
[&] (Dir_vfs_handle::Sync_dir_handle_element &sync_dir_handle_element) {
|
|
|
|
|
|
|
|
Vfs_handle *vfs_handle = &sync_dir_handle_element.vfs_handle;
|
|
|
|
|
|
|
|
result =
|
|
|
|
vfs_handle->fs().complete_sync(vfs_handle);
|
|
|
|
|
|
|
|
if (result != SYNC_OK)
|
|
|
|
return;
|
|
|
|
|
|
|
|
vfs_handle->ds().close(vfs_handle);
|
|
|
|
destroy(vfs_handle->alloc(), &sync_dir_handle_element);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2014-04-07 16:15:40 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif /* _INCLUDE__VFS__DIR_FILE_SYSTEM_H_ */
|