mirror of
https://github.com/genodelabs/genode.git
synced 2025-01-18 10:46:25 +00:00
Notification support for the VFS library
Add a new 'Vfs_watch_handle' type to the VFS interface. This handle type will pass a handle context up through the I/O handler to the application when a notification event occurs. Watch support implemented for RAM and File_system plugins, all other file-systems return WATCH_ERR_STATIC by default. Test at run/fs_rom_update_ram and run/fs_rom_update_fs. Fix #1934
This commit is contained in:
parent
344d46ce78
commit
9c6b720ec1
@ -94,6 +94,47 @@ class Vfs::Dir_file_system : public File_system
|
|||||||
Dir_vfs_handle &operator = (Dir_vfs_handle const &);
|
Dir_vfs_handle &operator = (Dir_vfs_handle const &);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct Dir_watch_handle : Vfs_watch_handle
|
||||||
|
{
|
||||||
|
struct Watch_handle_element;
|
||||||
|
|
||||||
|
typedef Genode::Registry<Watch_handle_element> Watch_handle_registry;
|
||||||
|
|
||||||
|
struct Watch_handle_element : Watch_handle_registry::Element
|
||||||
|
{
|
||||||
|
Vfs_watch_handle &watch_handle;
|
||||||
|
Watch_handle_element(Watch_handle_registry ®istry,
|
||||||
|
Vfs_watch_handle &handle)
|
||||||
|
: Watch_handle_registry::Element(registry, *this),
|
||||||
|
watch_handle(handle) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
Watch_handle_registry handle_registry { };
|
||||||
|
|
||||||
|
Dir_watch_handle(File_system &fs, Genode::Allocator &alloc)
|
||||||
|
: Vfs_watch_handle(fs, alloc) { }
|
||||||
|
|
||||||
|
~Dir_watch_handle()
|
||||||
|
{
|
||||||
|
/* close all sub-handles */
|
||||||
|
auto f = [&] (Watch_handle_element &e) {
|
||||||
|
e.watch_handle.fs().close(&e.watch_handle);
|
||||||
|
destroy(alloc(), &e);
|
||||||
|
};
|
||||||
|
handle_registry.for_each(f);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Propagate the handle context to each sub-handle
|
||||||
|
*/
|
||||||
|
void context(Context *ctx) override
|
||||||
|
{
|
||||||
|
handle_registry.for_each( [&] (Watch_handle_element &elem) {
|
||||||
|
elem.watch_handle.context(ctx); } );
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
/* pointer to first child file system */
|
/* pointer to first child file system */
|
||||||
File_system *_first_file_system = nullptr;
|
File_system *_first_file_system = nullptr;
|
||||||
|
|
||||||
@ -733,6 +774,41 @@ class Vfs::Dir_file_system : public File_system
|
|||||||
destroy(handle->alloc(), handle);
|
destroy(handle->alloc(), handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Watch_result watch(char const *path,
|
||||||
|
Vfs_watch_handle **handle,
|
||||||
|
Allocator &alloc) override
|
||||||
|
{
|
||||||
|
/* static by default, no allocations */
|
||||||
|
Watch_result r = WATCH_ERR_STATIC;
|
||||||
|
Dir_watch_handle *meta_handle = nullptr;
|
||||||
|
|
||||||
|
for (File_system *fs = _first_file_system; fs; fs = fs->next) {
|
||||||
|
Vfs_watch_handle *sub_handle;
|
||||||
|
|
||||||
|
if (fs->watch(path, &sub_handle, alloc) == WATCH_OK) {
|
||||||
|
if (meta_handle == nullptr) {
|
||||||
|
/* at least one non-static FS, allocate handle */
|
||||||
|
meta_handle = new (alloc) Dir_watch_handle(*this, alloc);
|
||||||
|
*handle = meta_handle;
|
||||||
|
r = WATCH_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* attach child FS handle to returned handle */
|
||||||
|
new (alloc)
|
||||||
|
Dir_watch_handle::Watch_handle_element(
|
||||||
|
meta_handle->handle_registry, *sub_handle);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
void close(Vfs_watch_handle *handle) override
|
||||||
|
{
|
||||||
|
if (handle && (&handle->fs() == this))
|
||||||
|
destroy(handle->alloc(), handle);
|
||||||
|
}
|
||||||
|
|
||||||
Unlink_result unlink(char const *path) override
|
Unlink_result unlink(char const *path) override
|
||||||
{
|
{
|
||||||
auto unlink_fn = [] (File_system &fs, char const *path)
|
auto unlink_fn = [] (File_system &fs, char const *path)
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
|
|
||||||
namespace Vfs {
|
namespace Vfs {
|
||||||
class Vfs_handle;
|
class Vfs_handle;
|
||||||
|
class Vfs_watch_handle;
|
||||||
struct Directory_service;
|
struct Directory_service;
|
||||||
|
|
||||||
using Genode::Allocator;
|
using Genode::Allocator;
|
||||||
@ -110,6 +111,32 @@ struct Vfs::Directory_service : Interface
|
|||||||
*/
|
*/
|
||||||
virtual void close(Vfs_handle *handle) = 0;
|
virtual void close(Vfs_handle *handle) = 0;
|
||||||
|
|
||||||
|
enum Watch_result
|
||||||
|
{
|
||||||
|
WATCH_ERR_UNACCESSIBLE,
|
||||||
|
WATCH_ERR_STATIC,
|
||||||
|
WATCH_ERR_OUT_OF_RAM,
|
||||||
|
WATCH_ERR_OUT_OF_CAPS,
|
||||||
|
WATCH_OK
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Watch a file-system node for changes.
|
||||||
|
*/
|
||||||
|
virtual Watch_result watch(char const *path,
|
||||||
|
Vfs_watch_handle**,
|
||||||
|
Allocator&)
|
||||||
|
{
|
||||||
|
/* default implementation for static file-systems */
|
||||||
|
Genode::warning("'", path, "' is static and cannot be watched");
|
||||||
|
return WATCH_ERR_STATIC;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void close(Vfs_watch_handle *)
|
||||||
|
{
|
||||||
|
Genode::error("watch handle closed at invalid file-system");
|
||||||
|
throw ~0;
|
||||||
|
};
|
||||||
|
|
||||||
/**********
|
/**********
|
||||||
** Stat **
|
** Stat **
|
||||||
|
@ -27,7 +27,9 @@ namespace Vfs {
|
|||||||
|
|
||||||
struct Vfs::Io_response_handler : Interface
|
struct Vfs::Io_response_handler : Interface
|
||||||
{
|
{
|
||||||
virtual void handle_io_response(Vfs::Vfs_handle::Context *context) = 0;
|
virtual void handle_io_response(Vfs_handle::Context *context) = 0;
|
||||||
|
virtual void handle_watch_response(Vfs_watch_handle::Context*) {
|
||||||
|
Genode::warning(__func__, " discarding event"); };
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -19,6 +19,8 @@
|
|||||||
namespace Vfs{
|
namespace Vfs{
|
||||||
class Vfs_handle;
|
class Vfs_handle;
|
||||||
class File_io_service;
|
class File_io_service;
|
||||||
|
class File_system;
|
||||||
|
class Vfs_watch_handle;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -109,4 +111,41 @@ class Vfs::Vfs_handle
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
class Vfs::Vfs_watch_handle
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Opaque handle context
|
||||||
|
*/
|
||||||
|
struct Context : List<Context>::Element { };
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
File_system &_fs;
|
||||||
|
Genode::Allocator &_alloc;
|
||||||
|
Context *_context = nullptr;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Noncopyable
|
||||||
|
*/
|
||||||
|
Vfs_watch_handle(Vfs_watch_handle const &);
|
||||||
|
Vfs_watch_handle &operator = (Vfs_watch_handle const &);
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
Vfs_watch_handle(File_system &fs,
|
||||||
|
Genode::Allocator &alloc)
|
||||||
|
:
|
||||||
|
_fs(fs), _alloc(alloc)
|
||||||
|
{ }
|
||||||
|
|
||||||
|
virtual ~Vfs_watch_handle() { }
|
||||||
|
|
||||||
|
File_system &fs() { return _fs; }
|
||||||
|
Allocator &alloc() { return _alloc; }
|
||||||
|
virtual void context(Context *context) { _context = context; }
|
||||||
|
Context *context() const { return _context; }
|
||||||
|
};
|
||||||
|
|
||||||
#endif /* _INCLUDE__VFS__VFS_HANDLE_H_ */
|
#endif /* _INCLUDE__VFS__VFS_HANDLE_H_ */
|
||||||
|
138
repos/os/run/fs_rom_update_fs.run
Normal file
138
repos/os/run/fs_rom_update_fs.run
Normal file
@ -0,0 +1,138 @@
|
|||||||
|
#
|
||||||
|
# Build
|
||||||
|
#
|
||||||
|
set build_components {
|
||||||
|
core init
|
||||||
|
app/rom_logger
|
||||||
|
app/rom_to_file
|
||||||
|
drivers/timer
|
||||||
|
server/dynamic_rom
|
||||||
|
server/fs_rom
|
||||||
|
server/vfs
|
||||||
|
}
|
||||||
|
|
||||||
|
build $build_components
|
||||||
|
|
||||||
|
create_boot_directory
|
||||||
|
|
||||||
|
#
|
||||||
|
# Generate config
|
||||||
|
#
|
||||||
|
append config {
|
||||||
|
<config>
|
||||||
|
<parent-provides>
|
||||||
|
<service name="CPU"/>
|
||||||
|
<service name="IO_MEM"/>
|
||||||
|
<service name="IO_PORT"/>
|
||||||
|
<service name="IRQ"/>
|
||||||
|
<service name="LOG"/>
|
||||||
|
<service name="PD"/>
|
||||||
|
<service name="RM"/>
|
||||||
|
<service name="ROM"/>
|
||||||
|
</parent-provides>
|
||||||
|
<default-route>
|
||||||
|
<any-service> <parent/> <any-child/> </any-service>
|
||||||
|
</default-route>
|
||||||
|
<default caps="100"/>
|
||||||
|
<start name="timer">
|
||||||
|
<resource name="RAM" quantum="1M"/>
|
||||||
|
<provides><service name="Timer"/></provides>
|
||||||
|
</start>
|
||||||
|
<start name="vfs_1">
|
||||||
|
<binary name="vfs"/>
|
||||||
|
<resource name="RAM" quantum="10M"/>
|
||||||
|
<provides><service name="File_system"/></provides>
|
||||||
|
<config>
|
||||||
|
<vfs> <ram/> </vfs>
|
||||||
|
<policy label_prefix="rom_to_file" root="/" writeable="yes"/>
|
||||||
|
<policy label_prefix="vfs_2" root="/" writeable="no"/>
|
||||||
|
</config>
|
||||||
|
</start>
|
||||||
|
<start name="vfs_2">
|
||||||
|
<binary name="vfs"/>
|
||||||
|
<resource name="RAM" quantum="10M"/>
|
||||||
|
<provides><service name="File_system"/></provides>
|
||||||
|
<config>
|
||||||
|
<vfs> <fs/> </vfs>
|
||||||
|
<policy label_prefix="fs_rom" root="/" writeable="no"/>
|
||||||
|
</config>
|
||||||
|
<route>
|
||||||
|
<service name="File_system">
|
||||||
|
<child name="vfs_1"/> </service>
|
||||||
|
<any-service> <parent/> <any-child/> </any-service>
|
||||||
|
</route>
|
||||||
|
</start>
|
||||||
|
<start name="dynamic_rom">
|
||||||
|
<resource name="RAM" quantum="4M"/>
|
||||||
|
<provides><service name="ROM"/></provides>
|
||||||
|
<config verbose="yes">
|
||||||
|
<rom name="dynamic_rom">
|
||||||
|
<inline description="iteration 1">
|
||||||
|
<config iteration="1" />
|
||||||
|
</inline>
|
||||||
|
<sleep milliseconds="2000" />
|
||||||
|
<inline description="iteration 2">
|
||||||
|
<config iteration="2" />
|
||||||
|
</inline>
|
||||||
|
<sleep milliseconds="2000" />
|
||||||
|
<inline description="iteration 3">
|
||||||
|
<config iteration="3" />
|
||||||
|
</inline>
|
||||||
|
<sleep milliseconds="2000" />
|
||||||
|
<inline description="iteration 4">
|
||||||
|
<config iteration="4" />
|
||||||
|
</inline>
|
||||||
|
<sleep milliseconds="2000" />
|
||||||
|
</rom>
|
||||||
|
</config>
|
||||||
|
</start>
|
||||||
|
<start name="rom_to_file">
|
||||||
|
<resource name="RAM" quantum="2M"/>
|
||||||
|
<config rom="dynamic_rom"/>
|
||||||
|
<route>
|
||||||
|
<service name="ROM" label="dynamic_rom"> <child name="dynamic_rom"/> </service>
|
||||||
|
<service name="File_system" > <child name="vfs_1"/> </service>
|
||||||
|
<any-service> <parent/> <any-child/> </any-service>
|
||||||
|
</route>
|
||||||
|
</start>
|
||||||
|
|
||||||
|
<start name="fs_rom">
|
||||||
|
<resource name="RAM" quantum="2M"/>
|
||||||
|
<provides><service name="ROM"/></provides>
|
||||||
|
<route>
|
||||||
|
<service name="File_system" > <child name="vfs_2"/> </service>
|
||||||
|
<any-service> <parent/> </any-service>
|
||||||
|
</route>
|
||||||
|
</start>
|
||||||
|
<start name="rom_logger">
|
||||||
|
<resource name="RAM" quantum="1M"/>
|
||||||
|
<config rom="dynamic_rom"/>
|
||||||
|
<route>
|
||||||
|
<service name="ROM" label="dynamic_rom"> <child name="fs_rom"/> </service>
|
||||||
|
<any-service> <parent/> </any-service>
|
||||||
|
</route>
|
||||||
|
</start>
|
||||||
|
</config>}
|
||||||
|
|
||||||
|
install_config $config
|
||||||
|
|
||||||
|
#
|
||||||
|
# Boot modules
|
||||||
|
#
|
||||||
|
|
||||||
|
# generic modules
|
||||||
|
set boot_modules {
|
||||||
|
core ld.lib.so init
|
||||||
|
dynamic_rom
|
||||||
|
fs_rom
|
||||||
|
vfs
|
||||||
|
rom_logger
|
||||||
|
rom_to_file
|
||||||
|
timer
|
||||||
|
}
|
||||||
|
|
||||||
|
build_boot_image $boot_modules
|
||||||
|
|
||||||
|
append qemu_args " -nographic"
|
||||||
|
|
||||||
|
run_genode_until {.*<config iteration="4" />.*} 60
|
123
repos/os/run/fs_rom_update_ram.run
Normal file
123
repos/os/run/fs_rom_update_ram.run
Normal file
@ -0,0 +1,123 @@
|
|||||||
|
#
|
||||||
|
# Build
|
||||||
|
#
|
||||||
|
set build_components {
|
||||||
|
core init
|
||||||
|
app/rom_logger
|
||||||
|
app/rom_to_file
|
||||||
|
drivers/timer
|
||||||
|
server/dynamic_rom
|
||||||
|
server/fs_rom
|
||||||
|
server/vfs
|
||||||
|
}
|
||||||
|
|
||||||
|
build $build_components
|
||||||
|
|
||||||
|
create_boot_directory
|
||||||
|
|
||||||
|
#
|
||||||
|
# Generate config
|
||||||
|
#
|
||||||
|
append config {
|
||||||
|
<config>
|
||||||
|
<parent-provides>
|
||||||
|
<service name="CPU"/>
|
||||||
|
<service name="IO_MEM"/>
|
||||||
|
<service name="IO_PORT"/>
|
||||||
|
<service name="IRQ"/>
|
||||||
|
<service name="LOG"/>
|
||||||
|
<service name="PD"/>
|
||||||
|
<service name="RM"/>
|
||||||
|
<service name="ROM"/>
|
||||||
|
</parent-provides>
|
||||||
|
<default-route>
|
||||||
|
<any-service> <parent/> <any-child/> </any-service>
|
||||||
|
</default-route>
|
||||||
|
<default caps="100"/>
|
||||||
|
<start name="timer">
|
||||||
|
<resource name="RAM" quantum="1M"/>
|
||||||
|
<provides><service name="Timer"/></provides>
|
||||||
|
</start>
|
||||||
|
<start name="vfs">
|
||||||
|
<resource name="RAM" quantum="10M"/>
|
||||||
|
<provides><service name="File_system"/></provides>
|
||||||
|
<config>
|
||||||
|
<vfs> <ram/> </vfs>
|
||||||
|
<policy label_prefix="rom_to_file" root="/" writeable="yes"/>
|
||||||
|
<policy label_prefix="fs_rom" root="/" writeable="no"/>
|
||||||
|
</config>
|
||||||
|
</start>
|
||||||
|
<start name="dynamic_rom">
|
||||||
|
<resource name="RAM" quantum="4M"/>
|
||||||
|
<provides><service name="ROM"/></provides>
|
||||||
|
<config verbose="yes">
|
||||||
|
<rom name="dynamic_rom">
|
||||||
|
<inline description="iteration 1">
|
||||||
|
<config iteration="1" />
|
||||||
|
</inline>
|
||||||
|
<sleep milliseconds="2000" />
|
||||||
|
<inline description="iteration 2">
|
||||||
|
<config iteration="2" />
|
||||||
|
</inline>
|
||||||
|
<sleep milliseconds="2000" />
|
||||||
|
<inline description="iteration 3">
|
||||||
|
<config iteration="3" />
|
||||||
|
</inline>
|
||||||
|
<sleep milliseconds="2000" />
|
||||||
|
<inline description="iteration 4">
|
||||||
|
<config iteration="4" />
|
||||||
|
</inline>
|
||||||
|
<sleep milliseconds="2000" />
|
||||||
|
</rom>
|
||||||
|
</config>
|
||||||
|
</start>
|
||||||
|
<start name="rom_to_file">
|
||||||
|
<resource name="RAM" quantum="2M"/>
|
||||||
|
<config rom="dynamic_rom"/>
|
||||||
|
<route>
|
||||||
|
<service name="ROM" label="dynamic_rom"> <child name="dynamic_rom"/> </service>
|
||||||
|
<service name="File_system" > <child name="vfs"/> </service>
|
||||||
|
<any-service> <parent/> <any-child/> </any-service>
|
||||||
|
</route>
|
||||||
|
</start>
|
||||||
|
|
||||||
|
<start name="fs_rom">
|
||||||
|
<resource name="RAM" quantum="2M"/>
|
||||||
|
<provides><service name="ROM"/></provides>
|
||||||
|
<route>
|
||||||
|
<service name="File_system" > <child name="vfs"/> </service>
|
||||||
|
<any-service> <parent/> </any-service>
|
||||||
|
</route>
|
||||||
|
</start>
|
||||||
|
<start name="rom_logger">
|
||||||
|
<resource name="RAM" quantum="1M"/>
|
||||||
|
<config rom="dynamic_rom"/>
|
||||||
|
<route>
|
||||||
|
<service name="ROM" label="dynamic_rom"> <child name="fs_rom"/> </service>
|
||||||
|
<any-service> <parent/> </any-service>
|
||||||
|
</route>
|
||||||
|
</start>
|
||||||
|
</config>}
|
||||||
|
|
||||||
|
install_config $config
|
||||||
|
|
||||||
|
#
|
||||||
|
# Boot modules
|
||||||
|
#
|
||||||
|
|
||||||
|
# generic modules
|
||||||
|
set boot_modules {
|
||||||
|
core ld.lib.so init
|
||||||
|
dynamic_rom
|
||||||
|
fs_rom
|
||||||
|
vfs
|
||||||
|
rom_logger
|
||||||
|
rom_to_file
|
||||||
|
timer
|
||||||
|
}
|
||||||
|
|
||||||
|
build_boot_image $boot_modules
|
||||||
|
|
||||||
|
append qemu_args " -nographic"
|
||||||
|
|
||||||
|
run_genode_until {.*<config iteration="4" />.*} 60
|
@ -29,7 +29,6 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Lock used to serialize the interaction with the packet stream of the
|
* Lock used to serialize the interaction with the packet stream of the
|
||||||
* file-system session.
|
* file-system session.
|
||||||
@ -41,7 +40,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
|
|
||||||
Genode::Env &_env;
|
Genode::Env &_env;
|
||||||
Genode::Allocator_avl _fs_packet_alloc;
|
Genode::Allocator_avl _fs_packet_alloc;
|
||||||
Io_response_handler &_io_handler;
|
Io_response_handler &_event_handler;
|
||||||
|
|
||||||
typedef Genode::String<64> Label_string;
|
typedef Genode::String<64> Label_string;
|
||||||
Label_string _label;
|
Label_string _label;
|
||||||
@ -54,6 +53,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
typedef Genode::Id_space<::File_system::Node> Handle_space;
|
typedef Genode::Id_space<::File_system::Node> Handle_space;
|
||||||
|
|
||||||
Handle_space _handle_space { };
|
Handle_space _handle_space { };
|
||||||
|
Handle_space _watch_handle_space { };
|
||||||
|
|
||||||
struct Handle_state
|
struct Handle_state
|
||||||
{
|
{
|
||||||
@ -82,7 +82,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
friend class Genode::Id_space<::File_system::Node>;
|
friend class Genode::Id_space<::File_system::Node>;
|
||||||
|
|
||||||
::File_system::Connection &_fs;
|
::File_system::Connection &_fs;
|
||||||
Io_response_handler &_io_handler;
|
Io_response_handler &_event_handler;
|
||||||
|
|
||||||
bool _queue_read(file_size count, file_size const seek_offset)
|
bool _queue_read(file_size count, file_size const seek_offset)
|
||||||
{
|
{
|
||||||
@ -145,7 +145,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
* Notify anyone who might have failed on
|
* Notify anyone who might have failed on
|
||||||
* 'alloc_packet()' or 'submit_packet()'
|
* 'alloc_packet()' or 'submit_packet()'
|
||||||
*/
|
*/
|
||||||
_io_handler.handle_io_response(nullptr);
|
_event_handler.handle_io_response(nullptr);
|
||||||
|
|
||||||
return READ_OK;
|
return READ_OK;
|
||||||
}
|
}
|
||||||
@ -158,7 +158,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
:
|
:
|
||||||
Vfs_handle(fs, fs, alloc, status_flags),
|
Vfs_handle(fs, fs, alloc, status_flags),
|
||||||
Handle_space::Element(*this, space, node_handle),
|
Handle_space::Element(*this, space, node_handle),
|
||||||
_fs(fs_connection), _io_handler(io_handler)
|
_fs(fs_connection), _event_handler(io_handler)
|
||||||
{ }
|
{ }
|
||||||
|
|
||||||
::File_system::File_handle file_handle() const
|
::File_system::File_handle file_handle() const
|
||||||
@ -227,7 +227,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
* Notify anyone who might have failed on
|
* Notify anyone who might have failed on
|
||||||
* 'alloc_packet()' or 'submit_packet()'
|
* 'alloc_packet()' or 'submit_packet()'
|
||||||
*/
|
*/
|
||||||
_io_handler.handle_io_response(nullptr);
|
_event_handler.handle_io_response(nullptr);
|
||||||
|
|
||||||
return SYNC_OK;
|
return SYNC_OK;
|
||||||
}
|
}
|
||||||
@ -355,19 +355,40 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct Fs_vfs_watch_handle : Vfs_watch_handle,
|
||||||
|
private ::File_system::Node,
|
||||||
|
private Handle_space::Element
|
||||||
|
{
|
||||||
|
friend class Genode::Id_space<::File_system::Node>;
|
||||||
|
|
||||||
|
::File_system::Watch_handle const fs_handle;
|
||||||
|
|
||||||
|
Fs_vfs_watch_handle(Vfs::File_system &fs,
|
||||||
|
Allocator &alloc,
|
||||||
|
Handle_space &space,
|
||||||
|
::File_system::Watch_handle handle)
|
||||||
|
:
|
||||||
|
Vfs_watch_handle(fs, alloc),
|
||||||
|
Handle_space::Element(*this, space, handle),
|
||||||
|
fs_handle(handle)
|
||||||
|
{ }
|
||||||
|
};
|
||||||
|
|
||||||
struct Post_signal_hook : Genode::Entrypoint::Post_signal_hook
|
struct Post_signal_hook : Genode::Entrypoint::Post_signal_hook
|
||||||
{
|
{
|
||||||
Genode::Entrypoint &_ep;
|
Genode::Entrypoint &_ep;
|
||||||
Io_response_handler &_io_handler;
|
Io_response_handler &_event_handler;
|
||||||
List<Vfs_handle::Context> _context_list { };
|
List<Vfs_handle::Context> _context_list { };
|
||||||
|
List<Vfs_watch_handle::Context>
|
||||||
|
_watch_context_list { };
|
||||||
Lock _list_lock { };
|
Lock _list_lock { };
|
||||||
bool _notify_all { false };
|
bool _notify_all { false };
|
||||||
|
|
||||||
Post_signal_hook(Genode::Entrypoint &ep,
|
Post_signal_hook(Genode::Entrypoint &ep,
|
||||||
Io_response_handler &io_handler)
|
Io_response_handler &io_handler)
|
||||||
: _ep(ep), _io_handler(io_handler) { }
|
: _ep(ep), _event_handler(io_handler) { }
|
||||||
|
|
||||||
void arm(Vfs_handle::Context *context)
|
void arm_io_event(Vfs_handle::Context *context)
|
||||||
{
|
{
|
||||||
if (!context) {
|
if (!context) {
|
||||||
Lock::Guard list_guard(_list_lock);
|
Lock::Guard list_guard(_list_lock);
|
||||||
@ -391,6 +412,27 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
_ep.schedule_post_signal_hook(this);
|
_ep.schedule_post_signal_hook(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void arm_watch_event(Vfs_watch_handle::Context &context)
|
||||||
|
{
|
||||||
|
{
|
||||||
|
Lock::Guard list_guard(_list_lock);
|
||||||
|
|
||||||
|
for (Vfs_watch_handle::Context *list_context = _watch_context_list.first();
|
||||||
|
list_context;
|
||||||
|
list_context = list_context->next())
|
||||||
|
{
|
||||||
|
if (list_context == &context) {
|
||||||
|
/* already in list */
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_watch_context_list.insert(&context);
|
||||||
|
}
|
||||||
|
|
||||||
|
_ep.schedule_post_signal_hook(this);
|
||||||
|
}
|
||||||
|
|
||||||
void function() override
|
void function() override
|
||||||
{
|
{
|
||||||
Vfs_handle::Context *context = nullptr;
|
Vfs_handle::Context *context = nullptr;
|
||||||
@ -411,14 +453,26 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (context || notify_all)
|
if (context || notify_all)
|
||||||
_io_handler.handle_io_response(context);
|
_event_handler.handle_io_response(context);
|
||||||
|
|
||||||
/* done if no contexts and all notified */
|
/* done if no contexts and all notified */
|
||||||
} while (context);
|
} while (context);
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
Vfs_watch_handle::Context *context = nullptr;
|
||||||
|
{
|
||||||
|
Lock::Guard list_guard(_list_lock);
|
||||||
|
|
||||||
|
context = _watch_context_list.first();
|
||||||
|
if (!context) break;
|
||||||
|
_watch_context_list.remove(context);
|
||||||
|
_event_handler.handle_watch_response(context);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
Post_signal_hook _post_signal_hook { _env.ep(), _io_handler };
|
Post_signal_hook _post_signal_hook { _env.ep(), _event_handler };
|
||||||
|
|
||||||
file_size _read(Fs_vfs_handle &handle, void *buf,
|
file_size _read(Fs_vfs_handle &handle, void *buf,
|
||||||
file_size const count, file_size const seek_offset)
|
file_size const count, file_size const seek_offset)
|
||||||
@ -504,7 +558,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
void _ready_to_submit()
|
void _ready_to_submit()
|
||||||
{
|
{
|
||||||
/* notify anyone who might have failed on write() ready_to_submit */
|
/* notify anyone who might have failed on write() ready_to_submit */
|
||||||
_post_signal_hook.arm(nullptr);
|
_post_signal_hook.arm_io_event(nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _handle_ack()
|
void _handle_ack()
|
||||||
@ -519,7 +573,13 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
Handle_space::Id const id(packet.handle());
|
Handle_space::Id const id(packet.handle());
|
||||||
|
|
||||||
try {
|
try {
|
||||||
_handle_space.apply<Fs_vfs_handle>(id, [&] (Fs_vfs_handle &handle)
|
if (packet.operation() == Packet_descriptor::CONTENT_CHANGED) {
|
||||||
|
_watch_handle_space.apply<Fs_vfs_watch_handle>(id, [&] (Fs_vfs_watch_handle &handle) {
|
||||||
|
|
||||||
|
if (auto *ctx = handle.context())
|
||||||
|
_post_signal_hook.arm_watch_event(*ctx);
|
||||||
|
});
|
||||||
|
} else _handle_space.apply<Fs_vfs_handle>(id, [&] (Fs_vfs_handle &handle)
|
||||||
{
|
{
|
||||||
if (!packet.succeeded())
|
if (!packet.succeeded())
|
||||||
Genode::error("packet operation=", (int)packet.operation(), " failed");
|
Genode::error("packet operation=", (int)packet.operation(), " failed");
|
||||||
@ -527,13 +587,13 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
switch (packet.operation()) {
|
switch (packet.operation()) {
|
||||||
case Packet_descriptor::READ_READY:
|
case Packet_descriptor::READ_READY:
|
||||||
handle.read_ready_state = Handle_state::Read_ready_state::READY;
|
handle.read_ready_state = Handle_state::Read_ready_state::READY;
|
||||||
_post_signal_hook.arm(handle.context);
|
_post_signal_hook.arm_io_event(handle.context);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Packet_descriptor::READ:
|
case Packet_descriptor::READ:
|
||||||
handle.queued_read_packet = packet;
|
handle.queued_read_packet = packet;
|
||||||
handle.queued_read_state = Handle_state::Queued_state::ACK;
|
handle.queued_read_state = Handle_state::Queued_state::ACK;
|
||||||
_post_signal_hook.arm(handle.context);
|
_post_signal_hook.arm_io_event(handle.context);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Packet_descriptor::WRITE:
|
case Packet_descriptor::WRITE:
|
||||||
@ -541,18 +601,18 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
* Notify anyone who might have failed on
|
* Notify anyone who might have failed on
|
||||||
* 'alloc_packet()'
|
* 'alloc_packet()'
|
||||||
*/
|
*/
|
||||||
_post_signal_hook.arm(nullptr);
|
_post_signal_hook.arm_io_event(nullptr);
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Packet_descriptor::CONTENT_CHANGED:
|
|
||||||
_post_signal_hook.arm(handle.context);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case Packet_descriptor::SYNC:
|
case Packet_descriptor::SYNC:
|
||||||
handle.queued_sync_packet = packet;
|
handle.queued_sync_packet = packet;
|
||||||
handle.queued_sync_state = Handle_state::Queued_state::ACK;
|
handle.queued_sync_state = Handle_state::Queued_state::ACK;
|
||||||
_post_signal_hook.arm(handle.context);
|
_post_signal_hook.arm_io_event(handle.context);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Packet_descriptor::CONTENT_CHANGED:
|
||||||
|
/* previously handled */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -577,12 +637,12 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
Fs_file_system(Genode::Env &env,
|
Fs_file_system(Genode::Env &env,
|
||||||
Genode::Allocator &alloc,
|
Genode::Allocator &alloc,
|
||||||
Genode::Xml_node config,
|
Genode::Xml_node config,
|
||||||
Io_response_handler &io_handler,
|
Io_response_handler &event_handler,
|
||||||
File_system &)
|
File_system &)
|
||||||
:
|
:
|
||||||
_env(env),
|
_env(env),
|
||||||
_fs_packet_alloc(&alloc),
|
_fs_packet_alloc(&alloc),
|
||||||
_io_handler(io_handler),
|
_event_handler(event_handler),
|
||||||
_label(config.attribute_value("label", Label_string())),
|
_label(config.attribute_value("label", Label_string())),
|
||||||
_root( config.attribute_value("root", Root_string())),
|
_root( config.attribute_value("root", Root_string())),
|
||||||
_fs(env, _fs_packet_alloc,
|
_fs(env, _fs_packet_alloc,
|
||||||
@ -613,7 +673,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
try {
|
try {
|
||||||
::File_system::Node_handle node = _fs.node(path);
|
::File_system::Node_handle node = _fs.node(path);
|
||||||
Fs_handle_guard node_guard(*this, _fs, node, _handle_space,
|
Fs_handle_guard node_guard(*this, _fs, node, _handle_space,
|
||||||
_fs, _io_handler);
|
_fs, _event_handler);
|
||||||
status = _fs.status(node);
|
status = _fs.status(node);
|
||||||
}
|
}
|
||||||
catch (::File_system::Lookup_failed) { return STAT_ERR_NO_ENTRY; }
|
catch (::File_system::Lookup_failed) { return STAT_ERR_NO_ENTRY; }
|
||||||
@ -649,7 +709,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
try {
|
try {
|
||||||
::File_system::Dir_handle dir = _fs.dir(dir_path.base(), false);
|
::File_system::Dir_handle dir = _fs.dir(dir_path.base(), false);
|
||||||
Fs_handle_guard dir_guard(*this, _fs, dir, _handle_space, _fs,
|
Fs_handle_guard dir_guard(*this, _fs, dir, _handle_space, _fs,
|
||||||
_io_handler);
|
_event_handler);
|
||||||
|
|
||||||
_fs.unlink(dir, file_name.base() + 1);
|
_fs.unlink(dir, file_name.base() + 1);
|
||||||
}
|
}
|
||||||
@ -685,12 +745,12 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
_fs.dir(from_dir_path.base(), false);
|
_fs.dir(from_dir_path.base(), false);
|
||||||
|
|
||||||
Fs_handle_guard from_dir_guard(*this, _fs, from_dir,
|
Fs_handle_guard from_dir_guard(*this, _fs, from_dir,
|
||||||
_handle_space, _fs, _io_handler);
|
_handle_space, _fs, _event_handler);
|
||||||
|
|
||||||
::File_system::Dir_handle to_dir = _fs.dir(to_dir_path.base(),
|
::File_system::Dir_handle to_dir = _fs.dir(to_dir_path.base(),
|
||||||
false);
|
false);
|
||||||
Fs_handle_guard to_dir_guard(*this, _fs, to_dir, _handle_space,
|
Fs_handle_guard to_dir_guard(*this, _fs, to_dir, _handle_space,
|
||||||
_fs, _io_handler);
|
_fs, _event_handler);
|
||||||
|
|
||||||
_fs.move(from_dir, from_file_name.base() + 1,
|
_fs.move(from_dir, from_file_name.base() + 1,
|
||||||
to_dir, to_file_name.base() + 1);
|
to_dir, to_file_name.base() + 1);
|
||||||
@ -709,7 +769,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
::File_system::Node_handle node;
|
::File_system::Node_handle node;
|
||||||
try { node = _fs.node(path); } catch (...) { return 0; }
|
try { node = _fs.node(path); } catch (...) { return 0; }
|
||||||
Fs_handle_guard node_guard(*this, _fs, node, _handle_space, _fs,
|
Fs_handle_guard node_guard(*this, _fs, node, _handle_space, _fs,
|
||||||
_io_handler);
|
_event_handler);
|
||||||
|
|
||||||
::File_system::Status status = _fs.status(node);
|
::File_system::Status status = _fs.status(node);
|
||||||
|
|
||||||
@ -721,7 +781,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
try {
|
try {
|
||||||
::File_system::Node_handle node = _fs.node(path);
|
::File_system::Node_handle node = _fs.node(path);
|
||||||
Fs_handle_guard node_guard(*this, _fs, node, _handle_space,
|
Fs_handle_guard node_guard(*this, _fs, node, _handle_space,
|
||||||
_fs, _io_handler);
|
_fs, _event_handler);
|
||||||
|
|
||||||
::File_system::Status status = _fs.status(node);
|
::File_system::Status status = _fs.status(node);
|
||||||
|
|
||||||
@ -767,7 +827,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
try {
|
try {
|
||||||
::File_system::Dir_handle dir = _fs.dir(dir_path.base(), false);
|
::File_system::Dir_handle dir = _fs.dir(dir_path.base(), false);
|
||||||
Fs_handle_guard dir_guard(*this, _fs, dir, _handle_space, _fs,
|
Fs_handle_guard dir_guard(*this, _fs, dir, _handle_space, _fs,
|
||||||
_io_handler);
|
_event_handler);
|
||||||
|
|
||||||
::File_system::File_handle file = _fs.file(dir,
|
::File_system::File_handle file = _fs.file(dir,
|
||||||
file_name.base() + 1,
|
file_name.base() + 1,
|
||||||
@ -775,7 +835,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
|
|
||||||
*out_handle = new (alloc)
|
*out_handle = new (alloc)
|
||||||
Fs_vfs_file_handle(*this, alloc, vfs_mode, _handle_space,
|
Fs_vfs_file_handle(*this, alloc, vfs_mode, _handle_space,
|
||||||
file, _fs, _io_handler);
|
file, _fs, _event_handler);
|
||||||
}
|
}
|
||||||
catch (::File_system::Lookup_failed) { return OPEN_ERR_UNACCESSIBLE; }
|
catch (::File_system::Lookup_failed) { return OPEN_ERR_UNACCESSIBLE; }
|
||||||
catch (::File_system::Permission_denied) { return OPEN_ERR_NO_PERM; }
|
catch (::File_system::Permission_denied) { return OPEN_ERR_NO_PERM; }
|
||||||
@ -803,7 +863,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
|
|
||||||
*out_handle = new (alloc)
|
*out_handle = new (alloc)
|
||||||
Fs_vfs_dir_handle(*this, alloc, ::File_system::READ_ONLY,
|
Fs_vfs_dir_handle(*this, alloc, ::File_system::READ_ONLY,
|
||||||
_handle_space, dir, _fs, _io_handler);
|
_handle_space, dir, _fs, _event_handler);
|
||||||
}
|
}
|
||||||
catch (::File_system::Lookup_failed) { return OPENDIR_ERR_LOOKUP_FAILED; }
|
catch (::File_system::Lookup_failed) { return OPENDIR_ERR_LOOKUP_FAILED; }
|
||||||
catch (::File_system::Name_too_long) { return OPENDIR_ERR_NAME_TOO_LONG; }
|
catch (::File_system::Name_too_long) { return OPENDIR_ERR_NAME_TOO_LONG; }
|
||||||
@ -835,7 +895,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
false);
|
false);
|
||||||
|
|
||||||
Fs_handle_guard from_dir_guard(*this, _fs, dir_handle,
|
Fs_handle_guard from_dir_guard(*this, _fs, dir_handle,
|
||||||
_handle_space, _fs, _io_handler);
|
_handle_space, _fs, _event_handler);
|
||||||
|
|
||||||
::File_system::Symlink_handle symlink_handle =
|
::File_system::Symlink_handle symlink_handle =
|
||||||
_fs.symlink(dir_handle, symlink_name.base() + 1, create);
|
_fs.symlink(dir_handle, symlink_name.base() + 1, create);
|
||||||
@ -844,7 +904,7 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
Fs_vfs_symlink_handle(*this, alloc,
|
Fs_vfs_symlink_handle(*this, alloc,
|
||||||
::File_system::READ_ONLY,
|
::File_system::READ_ONLY,
|
||||||
_handle_space, symlink_handle, _fs,
|
_handle_space, symlink_handle, _fs,
|
||||||
_io_handler);
|
_event_handler);
|
||||||
|
|
||||||
return OPENLINK_OK;
|
return OPENLINK_OK;
|
||||||
}
|
}
|
||||||
@ -871,6 +931,41 @@ class Vfs::Fs_file_system : public File_system
|
|||||||
destroy(fs_handle->alloc(), fs_handle);
|
destroy(fs_handle->alloc(), fs_handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Watch_result watch(char const *path,
|
||||||
|
Vfs_watch_handle **handle,
|
||||||
|
Allocator &alloc)
|
||||||
|
{
|
||||||
|
using namespace ::File_system;
|
||||||
|
|
||||||
|
Watch_result res = WATCH_ERR_UNACCESSIBLE;
|
||||||
|
::File_system::Watch_handle fs_handle { -1U };
|
||||||
|
|
||||||
|
try { fs_handle = _fs.watch(path); }
|
||||||
|
catch (Lookup_failed) { return WATCH_ERR_UNACCESSIBLE; }
|
||||||
|
catch (Permission_denied) { return WATCH_ERR_STATIC; }
|
||||||
|
catch (Out_of_ram) { return WATCH_ERR_OUT_OF_RAM; }
|
||||||
|
catch (Out_of_caps) { return WATCH_ERR_OUT_OF_CAPS; }
|
||||||
|
|
||||||
|
try {
|
||||||
|
*handle = new (alloc)
|
||||||
|
Fs_vfs_watch_handle(
|
||||||
|
*this, alloc, _watch_handle_space, fs_handle);
|
||||||
|
return WATCH_OK;
|
||||||
|
}
|
||||||
|
catch (Out_of_ram) { res = WATCH_ERR_OUT_OF_RAM; }
|
||||||
|
catch (Out_of_caps) { res = WATCH_ERR_OUT_OF_CAPS; }
|
||||||
|
_fs.close(fs_handle);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
void close(Vfs_watch_handle *vfs_handle) override
|
||||||
|
{
|
||||||
|
Fs_vfs_watch_handle *handle =
|
||||||
|
static_cast<Fs_vfs_watch_handle *>(vfs_handle);
|
||||||
|
_fs.close(handle->fs_handle);
|
||||||
|
destroy(handle->alloc(), handle);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
/***************************
|
/***************************
|
||||||
** File_system interface **
|
** File_system interface **
|
||||||
|
@ -19,11 +19,16 @@
|
|||||||
#include <dataspace/client.h>
|
#include <dataspace/client.h>
|
||||||
#include <util/avl_tree.h>
|
#include <util/avl_tree.h>
|
||||||
|
|
||||||
|
namespace Vfs { class Ram_file_system; }
|
||||||
|
|
||||||
namespace Vfs_ram {
|
namespace Vfs_ram {
|
||||||
|
|
||||||
using namespace Genode;
|
using namespace Genode;
|
||||||
using namespace Vfs;
|
using namespace Vfs;
|
||||||
|
|
||||||
|
struct Io_handle;
|
||||||
|
struct Watch_handle;
|
||||||
|
|
||||||
class Node;
|
class Node;
|
||||||
class File;
|
class File;
|
||||||
class Symlink;
|
class Symlink;
|
||||||
@ -49,7 +54,39 @@ namespace Vfs_ram {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace Vfs { class Ram_file_system; }
|
|
||||||
|
struct Vfs_ram::Io_handle final : public Vfs_handle,
|
||||||
|
private Genode::List<Io_handle>::Element
|
||||||
|
{
|
||||||
|
friend Genode::List<Io_handle>;
|
||||||
|
|
||||||
|
Vfs_ram::Node &node;
|
||||||
|
|
||||||
|
/* Track if this handle has modified its node */
|
||||||
|
bool modifying = false;
|
||||||
|
|
||||||
|
Io_handle(Vfs::File_system &fs,
|
||||||
|
Allocator &alloc,
|
||||||
|
int status_flags,
|
||||||
|
Vfs_ram::Node &node)
|
||||||
|
: Vfs_handle(fs, fs, alloc, status_flags), node(node)
|
||||||
|
{ }
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
struct Vfs_ram::Watch_handle final : public Vfs_watch_handle,
|
||||||
|
private Genode::List<Watch_handle>::Element
|
||||||
|
{
|
||||||
|
friend Genode::List<Watch_handle>;
|
||||||
|
using Genode::List<Watch_handle>::Element::next;
|
||||||
|
|
||||||
|
Vfs_ram::Node &node;
|
||||||
|
|
||||||
|
Watch_handle(Vfs::File_system &fs,
|
||||||
|
Allocator &alloc,
|
||||||
|
Node &node)
|
||||||
|
: Vfs_watch_handle(fs, alloc), node(node) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
class Vfs_ram::Node : private Genode::Avl_node<Node>, private Genode::Lock
|
class Vfs_ram::Node : private Genode::Avl_node<Node>, private Genode::Lock
|
||||||
@ -58,10 +95,16 @@ class Vfs_ram::Node : private Genode::Avl_node<Node>, private Genode::Lock
|
|||||||
|
|
||||||
friend class Genode::Avl_node<Node>;
|
friend class Genode::Avl_node<Node>;
|
||||||
friend class Genode::Avl_tree<Node>;
|
friend class Genode::Avl_tree<Node>;
|
||||||
|
friend class Genode::List<Io_handle>;
|
||||||
|
friend class Genode::List<Io_handle>::Element;
|
||||||
|
friend class Genode::List<Watch_handle>;
|
||||||
|
friend class Genode::List<Watch_handle>::Element;
|
||||||
|
friend class Watch_handle;
|
||||||
friend class Directory;
|
friend class Directory;
|
||||||
|
|
||||||
char _name[MAX_NAME_LEN];
|
char _name[MAX_NAME_LEN];
|
||||||
int _open_handles = 0;
|
Genode::List<Io_handle> _io_handles { };
|
||||||
|
Genode::List<Watch_handle> _watch_handles { };
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate unique inode number
|
* Generate unique inode number
|
||||||
@ -89,37 +132,45 @@ class Vfs_ram::Node : private Genode::Avl_node<Node>, private Genode::Lock
|
|||||||
|
|
||||||
virtual Vfs::file_size length() = 0;
|
virtual Vfs::file_size length() = 0;
|
||||||
|
|
||||||
/**
|
void open(Io_handle &handle) { _io_handles.insert(&handle); }
|
||||||
* Increment reference counter
|
void open(Watch_handle &handle) { _watch_handles.insert(&handle); }
|
||||||
*/
|
|
||||||
void open() { ++_open_handles; }
|
|
||||||
|
|
||||||
bool close_but_keep()
|
bool opened() const
|
||||||
{
|
{
|
||||||
if (--_open_handles < 0) {
|
return _io_handles.first() != nullptr;
|
||||||
inode = 0;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual size_t read(char * /* dst */, size_t /* len */, file_size /* seek_offset */)
|
void close(Io_handle &handle) { _io_handles.remove(&handle); }
|
||||||
|
void close(Watch_handle &handle) { _watch_handles.remove(&handle); }
|
||||||
|
|
||||||
|
void notify(Io_response_handler &handler)
|
||||||
|
{
|
||||||
|
for (Watch_handle *h = _watch_handles.first(); h; h = h->next()) {
|
||||||
|
if (auto *ctx = h->context()) {
|
||||||
|
handler.handle_watch_response(ctx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void unlink() { inode = 0; }
|
||||||
|
bool unlinked() const { return inode == 0; }
|
||||||
|
|
||||||
|
virtual size_t read(char*, size_t, file_size)
|
||||||
{
|
{
|
||||||
Genode::error("Vfs_ram::Node::read() called");
|
Genode::error("Vfs_ram::Node::read() called");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual Vfs::File_io_service::Read_result complete_read(char * /* dst */,
|
virtual Vfs::File_io_service::Read_result complete_read(char *,
|
||||||
file_size /* count */,
|
file_size,
|
||||||
file_size /* seek_offset */,
|
file_size,
|
||||||
file_size & /* out_count */)
|
file_size &)
|
||||||
{
|
{
|
||||||
Genode::error("Vfs_ram::Node::complete_read() called");
|
Genode::error("Vfs_ram::Node::complete_read() called");
|
||||||
return Vfs::File_io_service::READ_ERR_INVALID;
|
return Vfs::File_io_service::READ_ERR_INVALID;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual size_t write(char const * /* src */, size_t /* len */,
|
virtual size_t write(char const *, size_t, file_size)
|
||||||
file_size /* seek_offset */)
|
|
||||||
{
|
{
|
||||||
Genode::error("Vfs_ram::Node::write() called");
|
Genode::error("Vfs_ram::Node::write() called");
|
||||||
return 0;
|
return 0;
|
||||||
@ -308,8 +359,8 @@ class Vfs_ram::Symlink : public Vfs_ram::Node
|
|||||||
|
|
||||||
Vfs::File_io_service::Read_result complete_read(char *dst,
|
Vfs::File_io_service::Read_result complete_read(char *dst,
|
||||||
file_size count,
|
file_size count,
|
||||||
file_size /* seek_offset */,
|
file_size,
|
||||||
file_size &out_count) override
|
file_size &out_count) override
|
||||||
{
|
{
|
||||||
out_count = get(dst, count);
|
out_count = get(dst, count);
|
||||||
return Vfs::File_io_service::READ_OK;
|
return Vfs::File_io_service::READ_OK;
|
||||||
@ -344,7 +395,7 @@ class Vfs_ram::Directory : public Vfs_ram::Node
|
|||||||
while (Node *node = _entries.first()) {
|
while (Node *node = _entries.first()) {
|
||||||
_entries.remove(node);
|
_entries.remove(node);
|
||||||
if (File *file = dynamic_cast<File*>(node)) {
|
if (File *file = dynamic_cast<File*>(node)) {
|
||||||
if (file->close_but_keep())
|
if (file->opened())
|
||||||
continue;
|
continue;
|
||||||
} else if (Directory *dir = dynamic_cast<Directory*>(node)) {
|
} else if (Directory *dir = dynamic_cast<Directory*>(node)) {
|
||||||
dir->empty(alloc);
|
dir->empty(alloc);
|
||||||
@ -426,23 +477,12 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
|
|
||||||
struct Ram_vfs_handle : Vfs_handle
|
friend class Genode::List<Vfs_ram::Watch_handle>;
|
||||||
{
|
|
||||||
Vfs_ram::Node &node;
|
|
||||||
|
|
||||||
Ram_vfs_handle(Ram_file_system &fs,
|
Genode::Env &_env;
|
||||||
Allocator &alloc,
|
Genode::Allocator &_alloc;
|
||||||
int status_flags,
|
Io_response_handler &_io_handler;
|
||||||
Vfs_ram::Node &node)
|
Vfs_ram::Directory _root = { "" };
|
||||||
: Vfs_handle(fs, fs, alloc, status_flags), node(node)
|
|
||||||
{
|
|
||||||
node.open();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Genode::Env &_env;
|
|
||||||
Genode::Allocator &_alloc;
|
|
||||||
Vfs_ram::Directory _root = { "" };
|
|
||||||
|
|
||||||
Vfs_ram::Node *lookup(char const *path, bool return_parent = false)
|
Vfs_ram::Node *lookup(char const *path, bool return_parent = false)
|
||||||
{
|
{
|
||||||
@ -493,8 +533,10 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
using namespace Vfs_ram;
|
using namespace Vfs_ram;
|
||||||
|
|
||||||
if (File *file = dynamic_cast<File*>(node)) {
|
if (File *file = dynamic_cast<File*>(node)) {
|
||||||
if (file->close_but_keep())
|
if (file->opened()) {
|
||||||
|
file->unlink();
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
} else if (Directory *dir = dynamic_cast<Directory*>(node)) {
|
} else if (Directory *dir = dynamic_cast<Directory*>(node)) {
|
||||||
dir->empty(_alloc);
|
dir->empty(_alloc);
|
||||||
}
|
}
|
||||||
@ -507,8 +549,9 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
Ram_file_system(Genode::Env &env,
|
Ram_file_system(Genode::Env &env,
|
||||||
Genode::Allocator &alloc,
|
Genode::Allocator &alloc,
|
||||||
Genode::Xml_node,
|
Genode::Xml_node,
|
||||||
Io_response_handler &, File_system &)
|
Io_response_handler &io_handler,
|
||||||
: _env(env), _alloc(alloc) { }
|
File_system &)
|
||||||
|
: _env(env), _alloc(alloc), _io_handler(io_handler) { }
|
||||||
|
|
||||||
~Ram_file_system() { _root.empty(_alloc); }
|
~Ram_file_system() { _root.empty(_alloc); }
|
||||||
|
|
||||||
@ -566,6 +609,7 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
try { file = new (_alloc) File(name, _alloc); }
|
try { file = new (_alloc) File(name, _alloc); }
|
||||||
catch (Out_of_memory) { return OPEN_ERR_NO_SPACE; }
|
catch (Out_of_memory) { return OPEN_ERR_NO_SPACE; }
|
||||||
parent->adopt(file);
|
parent->adopt(file);
|
||||||
|
parent->notify(_io_handler);
|
||||||
} else {
|
} else {
|
||||||
Node *node = lookup(path);
|
Node *node = lookup(path);
|
||||||
if (!node) return OPEN_ERR_UNACCESSIBLE;
|
if (!node) return OPEN_ERR_UNACCESSIBLE;
|
||||||
@ -575,7 +619,7 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
*handle = new (alloc) Ram_vfs_handle(*this, alloc, mode, *file);
|
*handle = new (alloc) Io_handle(*this, alloc, mode, *file);
|
||||||
return OPEN_OK;
|
return OPEN_OK;
|
||||||
} catch (Genode::Out_of_ram) {
|
} catch (Genode::Out_of_ram) {
|
||||||
if (create) {
|
if (create) {
|
||||||
@ -607,6 +651,8 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
Directory *dir;
|
Directory *dir;
|
||||||
|
|
||||||
if (create) {
|
if (create) {
|
||||||
|
if (*name == '\0')
|
||||||
|
return OPENDIR_ERR_NODE_ALREADY_EXISTS;
|
||||||
|
|
||||||
if (strlen(name) >= MAX_NAME_LEN)
|
if (strlen(name) >= MAX_NAME_LEN)
|
||||||
return OPENDIR_ERR_NAME_TOO_LONG;
|
return OPENDIR_ERR_NAME_TOO_LONG;
|
||||||
@ -618,7 +664,7 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
catch (Out_of_memory) { return OPENDIR_ERR_NO_SPACE; }
|
catch (Out_of_memory) { return OPENDIR_ERR_NO_SPACE; }
|
||||||
|
|
||||||
parent->adopt(dir);
|
parent->adopt(dir);
|
||||||
|
parent->notify(_io_handler);
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
Node *node = lookup(path);
|
Node *node = lookup(path);
|
||||||
@ -629,9 +675,8 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
*handle = new (alloc) Ram_vfs_handle(*this, alloc,
|
*handle = new (alloc) Io_handle(
|
||||||
Ram_vfs_handle::STATUS_RDONLY,
|
*this, alloc, Io_handle::STATUS_RDONLY, *dir);
|
||||||
*dir);
|
|
||||||
return OPENDIR_OK;
|
return OPENDIR_OK;
|
||||||
} catch (Genode::Out_of_ram) {
|
} catch (Genode::Out_of_ram) {
|
||||||
if (create) {
|
if (create) {
|
||||||
@ -677,7 +722,7 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
link->lock();
|
link->lock();
|
||||||
parent->adopt(link);
|
parent->adopt(link);
|
||||||
link->unlock();
|
link->unlock();
|
||||||
|
parent->notify(_io_handler);
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
if (!node) return OPENLINK_ERR_LOOKUP_FAILED;
|
if (!node) return OPENLINK_ERR_LOOKUP_FAILED;
|
||||||
@ -688,9 +733,8 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
*handle = new (alloc) Ram_vfs_handle(*this, alloc,
|
*handle = new (alloc)
|
||||||
Ram_vfs_handle::STATUS_RDWR,
|
Io_handle(*this, alloc, Io_handle::STATUS_RDWR, *link);
|
||||||
*link);
|
|
||||||
return OPENLINK_OK;
|
return OPENLINK_OK;
|
||||||
} catch (Genode::Out_of_ram) {
|
} catch (Genode::Out_of_ram) {
|
||||||
if (create) {
|
if (create) {
|
||||||
@ -709,13 +753,19 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
|
|
||||||
void close(Vfs_handle *vfs_handle) override
|
void close(Vfs_handle *vfs_handle) override
|
||||||
{
|
{
|
||||||
Ram_vfs_handle *ram_handle =
|
Vfs_ram::Io_handle *ram_handle =
|
||||||
static_cast<Ram_vfs_handle *>(vfs_handle);
|
static_cast<Vfs_ram::Io_handle *>(vfs_handle);
|
||||||
|
|
||||||
if (ram_handle) {
|
Vfs_ram::Node &node = ram_handle->node;
|
||||||
if (!ram_handle->node.close_but_keep())
|
bool node_modified = ram_handle->modifying;
|
||||||
destroy(_alloc, &ram_handle->node);
|
|
||||||
destroy(vfs_handle->alloc(), ram_handle);
|
ram_handle->node.close(*ram_handle);
|
||||||
|
destroy(vfs_handle->alloc(), ram_handle);
|
||||||
|
|
||||||
|
if (ram_handle->node.unlinked() && !ram_handle->node.opened()) {
|
||||||
|
destroy(_alloc, &ram_handle->node);
|
||||||
|
} else if (node_modified) {
|
||||||
|
node.notify(_io_handler);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -797,6 +847,9 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
from_node->name(new_name);
|
from_node->name(new_name);
|
||||||
to_dir->adopt(from_node);
|
to_dir->adopt(from_node);
|
||||||
|
|
||||||
|
from_dir->notify(_io_handler);
|
||||||
|
to_dir->notify(_io_handler);
|
||||||
|
|
||||||
return RENAME_OK;
|
return RENAME_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -813,6 +866,7 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
|
|
||||||
node->lock();
|
node->lock();
|
||||||
parent->release(node);
|
parent->release(node);
|
||||||
|
parent->notify(_io_handler);
|
||||||
remove(node);
|
remove(node);
|
||||||
return UNLINK_OK;
|
return UNLINK_OK;
|
||||||
}
|
}
|
||||||
@ -853,6 +907,35 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
static_cap_cast<Genode::Ram_dataspace>(ds_cap)); }
|
static_cap_cast<Genode::Ram_dataspace>(ds_cap)); }
|
||||||
|
|
||||||
|
|
||||||
|
Watch_result watch(char const *path,
|
||||||
|
Vfs_watch_handle **handle,
|
||||||
|
Allocator &alloc) override
|
||||||
|
{
|
||||||
|
using namespace Vfs_ram;
|
||||||
|
|
||||||
|
Node *node = lookup(path);
|
||||||
|
if (!node) return WATCH_ERR_UNACCESSIBLE;
|
||||||
|
Node::Guard guard(node);
|
||||||
|
|
||||||
|
try {
|
||||||
|
Vfs_ram::Watch_handle *watch_handle = new(alloc)
|
||||||
|
Vfs_ram::Watch_handle(*this, alloc, *node);
|
||||||
|
node->open(*watch_handle);
|
||||||
|
*handle = watch_handle;
|
||||||
|
return WATCH_OK;
|
||||||
|
}
|
||||||
|
catch (Genode::Out_of_ram) { return WATCH_ERR_OUT_OF_RAM; }
|
||||||
|
catch (Genode::Out_of_caps) { return WATCH_ERR_OUT_OF_CAPS; }
|
||||||
|
}
|
||||||
|
|
||||||
|
void close(Vfs_watch_handle *vfs_handle) override
|
||||||
|
{
|
||||||
|
Vfs_ram::Watch_handle *watch_handle =
|
||||||
|
static_cast<Vfs_ram::Watch_handle *>(vfs_handle);
|
||||||
|
watch_handle->node.close(*watch_handle);
|
||||||
|
destroy(watch_handle->alloc(), watch_handle);
|
||||||
|
};
|
||||||
|
|
||||||
/************************
|
/************************
|
||||||
** File I/O interface **
|
** File I/O interface **
|
||||||
************************/
|
************************/
|
||||||
@ -864,11 +947,12 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
if ((vfs_handle->status_flags() & OPEN_MODE_ACCMODE) == OPEN_MODE_RDONLY)
|
if ((vfs_handle->status_flags() & OPEN_MODE_ACCMODE) == OPEN_MODE_RDONLY)
|
||||||
return WRITE_ERR_INVALID;
|
return WRITE_ERR_INVALID;
|
||||||
|
|
||||||
Ram_vfs_handle const *handle =
|
Vfs_ram::Io_handle *handle =
|
||||||
static_cast<Ram_vfs_handle *>(vfs_handle);
|
static_cast<Vfs_ram::Io_handle *>(vfs_handle);
|
||||||
|
|
||||||
Vfs_ram::Node::Guard guard(&handle->node);
|
Vfs_ram::Node::Guard guard(&handle->node);
|
||||||
out = handle->node.write(buf, len, handle->seek());
|
out = handle->node.write(buf, len, handle->seek());
|
||||||
|
handle->modifying = true;
|
||||||
|
|
||||||
return WRITE_OK;
|
return WRITE_OK;
|
||||||
}
|
}
|
||||||
@ -878,8 +962,8 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
{
|
{
|
||||||
out_count = 0;
|
out_count = 0;
|
||||||
|
|
||||||
Ram_vfs_handle const *handle =
|
Vfs_ram::Io_handle const *handle =
|
||||||
static_cast<Ram_vfs_handle *>(vfs_handle);
|
static_cast<Vfs_ram::Io_handle *>(vfs_handle);
|
||||||
|
|
||||||
Vfs_ram::Node::Guard guard(&handle->node);
|
Vfs_ram::Node::Guard guard(&handle->node);
|
||||||
|
|
||||||
@ -893,8 +977,8 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
if ((vfs_handle->status_flags() & OPEN_MODE_ACCMODE) == OPEN_MODE_RDONLY)
|
if ((vfs_handle->status_flags() & OPEN_MODE_ACCMODE) == OPEN_MODE_RDONLY)
|
||||||
return FTRUNCATE_ERR_NO_PERM;
|
return FTRUNCATE_ERR_NO_PERM;
|
||||||
|
|
||||||
Ram_vfs_handle const *handle =
|
Vfs_ram::Io_handle const *handle =
|
||||||
static_cast<Ram_vfs_handle *>(vfs_handle);
|
static_cast<Vfs_ram::Io_handle *>(vfs_handle);
|
||||||
|
|
||||||
Vfs_ram::Node::Guard guard(&handle->node);
|
Vfs_ram::Node::Guard guard(&handle->node);
|
||||||
|
|
||||||
@ -903,6 +987,21 @@ class Vfs::Ram_file_system : public Vfs::File_system
|
|||||||
return FTRUNCATE_OK;
|
return FTRUNCATE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Notify other handles if this handle has modified the node
|
||||||
|
*/
|
||||||
|
Sync_result complete_sync(Vfs_handle *vfs_handle) override
|
||||||
|
{
|
||||||
|
Vfs_ram::Io_handle *handle =
|
||||||
|
static_cast<Vfs_ram::Io_handle *>(vfs_handle);
|
||||||
|
if (handle->modifying) {
|
||||||
|
handle->modifying = false;
|
||||||
|
handle->node.close(*handle);
|
||||||
|
handle->node.notify(_io_handler);
|
||||||
|
handle->node.open(*handle);
|
||||||
|
}
|
||||||
|
return SYNC_OK;
|
||||||
|
}
|
||||||
|
|
||||||
/***************************
|
/***************************
|
||||||
** File_system interface **
|
** File_system interface **
|
||||||
|
@ -35,7 +35,7 @@ namespace Vfs_server {
|
|||||||
|
|
||||||
class Session_component;
|
class Session_component;
|
||||||
class Root;
|
class Root;
|
||||||
class Io_response_handler;
|
class Event_response_handler;
|
||||||
|
|
||||||
typedef Genode::Registered<Session_component> Registered_session;
|
typedef Genode::Registered<Session_component> Registered_session;
|
||||||
typedef Genode::Registry<Registered_session> Session_registry;
|
typedef Genode::Registry<Registered_session> Session_registry;
|
||||||
@ -53,7 +53,7 @@ namespace Vfs_server {
|
|||||||
|
|
||||||
|
|
||||||
class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
||||||
public Node_io_handler
|
public Session_io_handler
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
|
|
||||||
@ -74,6 +74,8 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
*/
|
*/
|
||||||
Path const _root_path;
|
Path const _root_path;
|
||||||
|
|
||||||
|
Genode::Session_label const _label;
|
||||||
|
|
||||||
bool const _writable;
|
bool const _writable;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -92,11 +94,11 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
* \throw Invalid_handle
|
* \throw Invalid_handle
|
||||||
*/
|
*/
|
||||||
template <typename FUNC>
|
template <typename FUNC>
|
||||||
void _apply(Node_handle handle, FUNC const &fn)
|
void _apply_node(Node_handle handle, FUNC const &fn)
|
||||||
{
|
{
|
||||||
Node_space::Id id { handle.value };
|
Node_space::Id id { handle.value };
|
||||||
|
|
||||||
try { _node_space.apply<Node>(id, fn); }
|
try { return _node_space.apply<Node>(id, fn); }
|
||||||
catch (Node_space::Unknown_id) { throw Invalid_handle(); }
|
catch (Node_space::Unknown_id) { throw Invalid_handle(); }
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -155,7 +157,7 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
case Packet_descriptor::READ:
|
case Packet_descriptor::READ:
|
||||||
|
|
||||||
try {
|
try {
|
||||||
_apply(packet.handle(), [&] (Node &node) {
|
_apply(packet.handle(), [&] (Io_node &node) {
|
||||||
if (!node.read_ready()) {
|
if (!node.read_ready()) {
|
||||||
node.notify_read_ready(true);
|
node.notify_read_ready(true);
|
||||||
throw Not_ready();
|
throw Not_ready();
|
||||||
@ -178,7 +180,7 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
case Packet_descriptor::WRITE:
|
case Packet_descriptor::WRITE:
|
||||||
|
|
||||||
try {
|
try {
|
||||||
_apply(packet.handle(), [&] (Node &node) {
|
_apply(packet.handle(), [&] (Io_node &node) {
|
||||||
if (node.mode() & WRITE_ONLY) {
|
if (node.mode() & WRITE_ONLY) {
|
||||||
res_length = node.write((char const *)content, length, seek);
|
res_length = node.write((char const *)content, length, seek);
|
||||||
|
|
||||||
@ -213,7 +215,7 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case Packet_descriptor::CONTENT_CHANGED:
|
case Packet_descriptor::CONTENT_CHANGED:
|
||||||
/* The VFS does not track file changes yet */
|
Genode::warning("ignoring CONTENT_CHANGED packet from client");
|
||||||
throw Dont_ack();
|
throw Dont_ack();
|
||||||
|
|
||||||
case Packet_descriptor::SYNC:
|
case Packet_descriptor::SYNC:
|
||||||
@ -222,7 +224,7 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
* Sync the VFS and send any pending signals on the node.
|
* Sync the VFS and send any pending signals on the node.
|
||||||
*/
|
*/
|
||||||
try {
|
try {
|
||||||
_apply(packet.handle(), [&] (Node &node) {
|
_apply(packet.handle(), [&] (Io_node &node) {
|
||||||
node.sync();
|
node.sync();
|
||||||
succeeded = true;
|
succeeded = true;
|
||||||
});
|
});
|
||||||
@ -356,6 +358,8 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
destroy(_alloc, dir);
|
destroy(_alloc, dir);
|
||||||
else if (Symlink *link = dynamic_cast<Symlink*>(&node))
|
else if (Symlink *link = dynamic_cast<Symlink*>(&node))
|
||||||
destroy(_alloc, link);
|
destroy(_alloc, link);
|
||||||
|
else if (Watch_node *watch = dynamic_cast<Watch_node*>(&node))
|
||||||
|
destroy(_alloc, watch);
|
||||||
else
|
else
|
||||||
destroy(_alloc, &node);
|
destroy(_alloc, &node);
|
||||||
}
|
}
|
||||||
@ -372,6 +376,7 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
Session_component(Genode::Env &env,
|
Session_component(Genode::Env &env,
|
||||||
|
char const *label,
|
||||||
Genode::Ram_quota ram_quota,
|
Genode::Ram_quota ram_quota,
|
||||||
Genode::Cap_quota cap_quota,
|
Genode::Cap_quota cap_quota,
|
||||||
size_t tx_buf_size,
|
size_t tx_buf_size,
|
||||||
@ -387,6 +392,7 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
_process_packet_handler(env.ep(), *this, &Session_component::_process_packets),
|
_process_packet_handler(env.ep(), *this, &Session_component::_process_packets),
|
||||||
_vfs(vfs),
|
_vfs(vfs),
|
||||||
_root_path(root_path),
|
_root_path(root_path),
|
||||||
|
_label(label),
|
||||||
_writable(writable)
|
_writable(writable)
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
@ -419,14 +425,20 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
* node-specific, for example after 'release_packet()' to signal
|
* node-specific, for example after 'release_packet()' to signal
|
||||||
* that a previously failed 'alloc_packet()' may succeed now.
|
* that a previously failed 'alloc_packet()' may succeed now.
|
||||||
*/
|
*/
|
||||||
void handle_general_io()
|
void handle_general_io() {
|
||||||
{
|
_process_packets(); }
|
||||||
_process_packets();
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Node_io_handler interface */
|
|
||||||
void handle_node_io(Node &node) override
|
/********************************
|
||||||
|
** Node_io_handler interface **
|
||||||
|
********************************/
|
||||||
|
|
||||||
|
void handle_node_io(Io_node &node) override
|
||||||
{
|
{
|
||||||
|
if (!tx_sink()->ready_to_ack())
|
||||||
|
Genode::error(
|
||||||
|
"dropping I/O notfication, congested packet buffer to '", _label, "'");
|
||||||
|
|
||||||
if (node.notify_read_ready() && node.read_ready()
|
if (node.notify_read_ready() && node.read_ready()
|
||||||
&& tx_sink()->ready_to_ack()) {
|
&& tx_sink()->ready_to_ack()) {
|
||||||
Packet_descriptor packet(Packet_descriptor(),
|
Packet_descriptor packet(Packet_descriptor(),
|
||||||
@ -440,6 +452,22 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
_process_packets();
|
_process_packets();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void handle_node_watch(Watch_node &node) override
|
||||||
|
{
|
||||||
|
if (!tx_sink()->ready_to_ack())
|
||||||
|
Genode::error(
|
||||||
|
"dropping watch notfication, congested packet buffer to '", _label, "'");
|
||||||
|
|
||||||
|
if (tx_sink()->ready_to_ack()) {
|
||||||
|
Packet_descriptor packet(Packet_descriptor(),
|
||||||
|
Node_handle { node.id().value },
|
||||||
|
Packet_descriptor::CONTENT_CHANGED,
|
||||||
|
0, 0);
|
||||||
|
tx_sink()->acknowledge_packet(packet);
|
||||||
|
}
|
||||||
|
_process_packets();
|
||||||
|
}
|
||||||
|
|
||||||
/***************************
|
/***************************
|
||||||
** File_system interface **
|
** File_system interface **
|
||||||
***************************/
|
***************************/
|
||||||
@ -482,7 +510,7 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
_assert_valid_name(name_str);
|
_assert_valid_name(name_str);
|
||||||
|
|
||||||
return File_handle {
|
return File_handle {
|
||||||
dir.file(_node_space, _vfs, _alloc, *this, name_str, fs_mode, create).value
|
dir.file(_node_space, _vfs, _alloc, name_str, fs_mode, create).value
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -516,16 +544,47 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
|
|
||||||
Node *node;
|
Node *node;
|
||||||
|
|
||||||
try { node = new (_alloc) Node(_node_space, path_str, STAT_ONLY,
|
try { node = new (_alloc) Node(_node_space, path_str, *this); }
|
||||||
*this); }
|
|
||||||
catch (Out_of_memory) { throw Out_of_ram(); }
|
catch (Out_of_memory) { throw Out_of_ram(); }
|
||||||
|
|
||||||
return Node_handle { node->id().value };
|
return Node_handle { node->id().value };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Watch_handle watch(File_system::Path const &path) override
|
||||||
|
{
|
||||||
|
char const *path_str = path.string();
|
||||||
|
|
||||||
|
_assert_valid_path(path_str);
|
||||||
|
|
||||||
|
/* re-root the path */
|
||||||
|
Path sub_path(path_str+1, _root_path.base());
|
||||||
|
path_str = sub_path.base();
|
||||||
|
|
||||||
|
Vfs::Vfs_watch_handle *vfs_handle = nullptr;
|
||||||
|
typedef Directory_service::Watch_result Result;
|
||||||
|
switch (_vfs.watch(path_str, &vfs_handle, _alloc)) {
|
||||||
|
case Result::WATCH_OK: break;
|
||||||
|
case Result::WATCH_ERR_UNACCESSIBLE:
|
||||||
|
throw Lookup_failed();
|
||||||
|
case Result::WATCH_ERR_STATIC:
|
||||||
|
throw Unavailable();
|
||||||
|
case Result::WATCH_ERR_OUT_OF_RAM:
|
||||||
|
throw Out_of_ram();
|
||||||
|
case Result::WATCH_ERR_OUT_OF_CAPS:
|
||||||
|
throw Out_of_caps();
|
||||||
|
}
|
||||||
|
|
||||||
|
Node *node;
|
||||||
|
try { node = new (_alloc)
|
||||||
|
Watch_node(_node_space, path_str, *vfs_handle, *this); }
|
||||||
|
catch (Out_of_memory) { throw Out_of_ram(); }
|
||||||
|
|
||||||
|
return Watch_handle { node->id().value };
|
||||||
|
}
|
||||||
|
|
||||||
void close(Node_handle handle) override
|
void close(Node_handle handle) override
|
||||||
{
|
{
|
||||||
try { _apply(handle, [&] (Node &node) {
|
try { _apply_node(handle, [&] (Node &node) {
|
||||||
_close(node);
|
_close(node);
|
||||||
}); } catch (File_system::Invalid_handle) { }
|
}); } catch (File_system::Invalid_handle) { }
|
||||||
}
|
}
|
||||||
@ -534,7 +593,7 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
{
|
{
|
||||||
File_system::Status fs_stat;
|
File_system::Status fs_stat;
|
||||||
|
|
||||||
_apply(node_handle, [&] (Node &node) {
|
_apply_node(node_handle, [&] (Node &node) {
|
||||||
Directory_service::Stat vfs_stat;
|
Directory_service::Stat vfs_stat;
|
||||||
|
|
||||||
if (_vfs.stat(node.path(), vfs_stat) != Directory_service::STAT_OK)
|
if (_vfs.stat(node.path(), vfs_stat) != Directory_service::STAT_OK)
|
||||||
@ -615,15 +674,17 @@ class Vfs_server::Session_component : public File_system::Session_rpc_object,
|
|||||||
void control(Node_handle, Control) override { }
|
void control(Node_handle, Control) override { }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
struct Vfs_server::Io_response_handler : Vfs::Io_response_handler
|
* Global vfs event handler
|
||||||
|
*/
|
||||||
|
struct Vfs_server::Event_response_handler : Vfs::Io_response_handler
|
||||||
{
|
{
|
||||||
Session_registry &_session_registry;
|
Session_registry &_session_registry;
|
||||||
|
|
||||||
bool _in_progress { false };
|
bool _in_progress { false };
|
||||||
bool _handle_general_io { false };
|
bool _handle_general_io { false };
|
||||||
|
|
||||||
Io_response_handler(Session_registry &session_registry)
|
Event_response_handler(Session_registry &session_registry)
|
||||||
: _session_registry(session_registry) { }
|
: _session_registry(session_registry) { }
|
||||||
|
|
||||||
void handle_io_response(Vfs::Vfs_handle::Context *context) override
|
void handle_io_response(Vfs::Vfs_handle::Context *context) override
|
||||||
@ -637,7 +698,7 @@ struct Vfs_server::Io_response_handler : Vfs::Io_response_handler
|
|||||||
_in_progress = true;
|
_in_progress = true;
|
||||||
|
|
||||||
if (context)
|
if (context)
|
||||||
Node::node_by_context(*context).handle_io_response();
|
Io_node::node_by_context(*context).handle_io_response();
|
||||||
else
|
else
|
||||||
_handle_general_io = true;
|
_handle_general_io = true;
|
||||||
|
|
||||||
@ -650,6 +711,12 @@ struct Vfs_server::Io_response_handler : Vfs::Io_response_handler
|
|||||||
|
|
||||||
_in_progress = false;
|
_in_progress = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void handle_watch_response(Vfs::Vfs_watch_handle::Context *context) override
|
||||||
|
{
|
||||||
|
if (context)
|
||||||
|
Watch_node::node_by_context(*context).handle_watch_response();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -676,12 +743,12 @@ class Vfs_server::Root : public Genode::Root_component<Session_component>
|
|||||||
|
|
||||||
Session_registry _session_registry { };
|
Session_registry _session_registry { };
|
||||||
|
|
||||||
Io_response_handler _io_response_handler { _session_registry };
|
Event_response_handler _response_handler { _session_registry };
|
||||||
|
|
||||||
Vfs::Global_file_system_factory _global_file_system_factory { _vfs_heap };
|
Vfs::Global_file_system_factory _global_file_system_factory { _vfs_heap };
|
||||||
|
|
||||||
Vfs::Dir_file_system _vfs {
|
Vfs::Dir_file_system _vfs {
|
||||||
_env, _vfs_heap, vfs_config(), _io_response_handler,
|
_env, _vfs_heap, vfs_config(), _response_handler,
|
||||||
_global_file_system_factory };
|
_global_file_system_factory };
|
||||||
|
|
||||||
Genode::Signal_handler<Root> _config_handler {
|
Genode::Signal_handler<Root> _config_handler {
|
||||||
@ -774,7 +841,7 @@ class Vfs_server::Root : public Genode::Root_component<Session_component>
|
|||||||
}
|
}
|
||||||
|
|
||||||
Session_component *session = new (md_alloc())
|
Session_component *session = new (md_alloc())
|
||||||
Registered_session(_session_registry, _env,
|
Registered_session(_session_registry, _env, label.string(),
|
||||||
Genode::Ram_quota{ram_quota},
|
Genode::Ram_quota{ram_quota},
|
||||||
Genode::Cap_quota{cap_quota},
|
Genode::Cap_quota{cap_quota},
|
||||||
tx_buf_size, _vfs,
|
tx_buf_size, _vfs,
|
||||||
|
@ -29,16 +29,19 @@ namespace Vfs_server {
|
|||||||
using namespace File_system;
|
using namespace File_system;
|
||||||
using namespace Vfs;
|
using namespace Vfs;
|
||||||
|
|
||||||
struct Node;
|
class Node;
|
||||||
struct Directory;
|
class Io_node;
|
||||||
struct File;
|
class Watch_node;
|
||||||
struct Symlink;
|
class Directory;
|
||||||
|
class File;
|
||||||
|
class Symlink;
|
||||||
|
|
||||||
typedef Genode::Id_space<Node> Node_space;
|
typedef Genode::Id_space<Node> Node_space;
|
||||||
|
|
||||||
struct Node_io_handler : Interface
|
struct Session_io_handler : Interface
|
||||||
{
|
{
|
||||||
virtual void handle_node_io(Node &node) = 0;
|
virtual void handle_node_io(Io_node &node) = 0;
|
||||||
|
virtual void handle_node_watch(Watch_node &node) = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -59,20 +62,21 @@ namespace Vfs_server {
|
|||||||
* Type trait for determining the node type for a given handle type
|
* Type trait for determining the node type for a given handle type
|
||||||
*/
|
*/
|
||||||
template<typename T> struct Node_type;
|
template<typename T> struct Node_type;
|
||||||
template<> struct Node_type<Node_handle> { typedef Node Type; };
|
template<> struct Node_type<Node_handle> { typedef Io_node Type; };
|
||||||
template<> struct Node_type<Dir_handle> { typedef Directory Type; };
|
template<> struct Node_type<Dir_handle> { typedef Directory Type; };
|
||||||
template<> struct Node_type<File_handle> { typedef File Type; };
|
template<> struct Node_type<File_handle> { typedef File Type; };
|
||||||
template<> struct Node_type<Symlink_handle> { typedef Symlink Type; };
|
template<> struct Node_type<Symlink_handle> { typedef Symlink Type; };
|
||||||
|
template<> struct Node_type<Watch_handle> { typedef Watch_node Type; };
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Type trait for determining the handle type for a given node type
|
* Type trait for determining the handle type for a given node type
|
||||||
*/
|
*/
|
||||||
template<typename T> struct Handle_type;
|
template<typename T> struct Handle_type;
|
||||||
template<> struct Handle_type<Node> { typedef Node_handle Type; };
|
template<> struct Handle_type<Io_node> { typedef Node_handle Type; };
|
||||||
template<> struct Handle_type<Directory> { typedef Dir_handle Type; };
|
template<> struct Handle_type<Directory> { typedef Dir_handle Type; };
|
||||||
template<> struct Handle_type<File> { typedef File_handle Type; };
|
template<> struct Handle_type<File> { typedef File_handle Type; };
|
||||||
template<> struct Handle_type<Symlink> { typedef Symlink_handle Type; };
|
template<> struct Handle_type<Symlink> { typedef Symlink_handle Type; };
|
||||||
|
template<> struct Handle_type<Watch> { typedef Watch_handle Type; };
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Note that the file objects are created at the
|
* Note that the file objects are created at the
|
||||||
@ -82,10 +86,8 @@ namespace Vfs_server {
|
|||||||
*/
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class Vfs_server::Node : public File_system::Node_base,
|
class Vfs_server::Node : public File_system::Node_base,
|
||||||
private Node_space::Element,
|
private Node_space::Element
|
||||||
private Vfs::Vfs_handle::Context
|
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
|
||||||
@ -100,6 +102,52 @@ class Vfs_server::Node : public File_system::Node_base,
|
|||||||
Node &operator = (Node const &);
|
Node &operator = (Node const &);
|
||||||
|
|
||||||
Path const _path;
|
Path const _path;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
|
||||||
|
/**
|
||||||
|
* I/O handler for session context
|
||||||
|
*/
|
||||||
|
Session_io_handler &_session_io_handler;
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
Node(Node_space &space, char const *node_path,
|
||||||
|
Session_io_handler &io_handler)
|
||||||
|
:
|
||||||
|
Node_space::Element(*this, space),
|
||||||
|
_path(node_path),
|
||||||
|
_session_io_handler(io_handler)
|
||||||
|
{ }
|
||||||
|
|
||||||
|
virtual ~Node() { }
|
||||||
|
|
||||||
|
using Node_space::Element::id;
|
||||||
|
|
||||||
|
char const *path() const { return _path.base(); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Print for debugging
|
||||||
|
*/
|
||||||
|
void print(Genode::Output &out) const {
|
||||||
|
out.out_string(_path.base()); }
|
||||||
|
};
|
||||||
|
|
||||||
|
class Vfs_server::Io_node : public Vfs_server::Node,
|
||||||
|
private Vfs::Vfs_handle::Context
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
|
||||||
|
enum Op_state { IDLE, READ_QUEUED, SYNC_QUEUED };
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Noncopyable
|
||||||
|
*/
|
||||||
|
Io_node(Io_node const &);
|
||||||
|
Io_node &operator = (Io_node const &);
|
||||||
|
|
||||||
Mode const _mode;
|
Mode const _mode;
|
||||||
|
|
||||||
bool _notify_read_ready = false;
|
bool _notify_read_ready = false;
|
||||||
@ -108,7 +156,6 @@ class Vfs_server::Node : public File_system::Node_base,
|
|||||||
|
|
||||||
Vfs::Vfs_handle::Context &context() { return *this; }
|
Vfs::Vfs_handle::Context &context() { return *this; }
|
||||||
|
|
||||||
Node_io_handler &_node_io_handler;
|
|
||||||
Vfs::Vfs_handle *_handle { nullptr };
|
Vfs::Vfs_handle *_handle { nullptr };
|
||||||
Op_state op_state { Op_state::IDLE };
|
Op_state op_state { Op_state::IDLE };
|
||||||
|
|
||||||
@ -183,24 +230,19 @@ class Vfs_server::Node : public File_system::Node_base,
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
Node(Node_space &space, char const *node_path, Mode node_mode,
|
Io_node(Node_space &space, char const *node_path, Mode node_mode,
|
||||||
Node_io_handler &node_io_handler)
|
Session_io_handler &io_handler)
|
||||||
:
|
: Node(space, node_path, io_handler), _mode(node_mode) { }
|
||||||
Node_space::Element(*this, space),
|
|
||||||
_path(node_path), _mode(node_mode),
|
|
||||||
_node_io_handler(node_io_handler)
|
|
||||||
{ }
|
|
||||||
|
|
||||||
virtual ~Node() { }
|
virtual ~Io_node() { }
|
||||||
|
|
||||||
using Node_space::Element::id;
|
using Node_space::Element::id;
|
||||||
|
|
||||||
static Node &node_by_context(Vfs::Vfs_handle::Context &context)
|
static Io_node &node_by_context(Vfs::Vfs_handle::Context &context)
|
||||||
{
|
{
|
||||||
return static_cast<Node &>(context);
|
return static_cast<Io_node &>(context);
|
||||||
}
|
}
|
||||||
|
|
||||||
char const *path() { return _path.base(); }
|
|
||||||
Mode mode() const { return _mode; }
|
Mode mode() const { return _mode; }
|
||||||
|
|
||||||
virtual size_t read(char * /* dst */, size_t /* len */, seek_off_t)
|
virtual size_t read(char * /* dst */, size_t /* len */, seek_off_t)
|
||||||
@ -211,10 +253,13 @@ class Vfs_server::Node : public File_system::Node_base,
|
|||||||
|
|
||||||
bool read_ready() { return _handle->fs().read_ready(_handle); }
|
bool read_ready() { return _handle->fs().read_ready(_handle); }
|
||||||
|
|
||||||
void handle_io_response()
|
/**
|
||||||
{
|
* The global handler has drawn an association from an I/O
|
||||||
_node_io_handler.handle_node_io(*this);
|
* context and this open node, now process the event at the
|
||||||
}
|
* session for this node.
|
||||||
|
*/
|
||||||
|
void handle_io_response() {
|
||||||
|
_session_io_handler.handle_node_io(*this); }
|
||||||
|
|
||||||
void notify_read_ready(bool requested)
|
void notify_read_ready(bool requested)
|
||||||
{
|
{
|
||||||
@ -257,16 +302,67 @@ class Vfs_server::Node : public File_system::Node_base,
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Vfs_server::Symlink : Node
|
|
||||||
|
class Vfs_server::Watch_node final : public Vfs_server::Node,
|
||||||
|
private Vfs::Vfs_watch_handle::Context
|
||||||
{
|
{
|
||||||
Symlink(Node_space &space,
|
private:
|
||||||
Vfs::File_system &vfs,
|
|
||||||
Genode::Allocator &alloc,
|
/*
|
||||||
Node_io_handler &node_io_handler,
|
* Noncopyable
|
||||||
char const *link_path,
|
*/
|
||||||
Mode mode,
|
Watch_node(Watch_node const &);
|
||||||
bool create)
|
Watch_node &operator = (Watch_node const &);
|
||||||
: Node(space, link_path, mode, node_io_handler)
|
|
||||||
|
Vfs::Vfs_watch_handle &_watch_handle;
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
Watch_node(Node_space &space, char const *path,
|
||||||
|
Vfs::Vfs_watch_handle &handle,
|
||||||
|
Session_io_handler &io_handler)
|
||||||
|
:
|
||||||
|
Node(space, path, io_handler),
|
||||||
|
_watch_handle(handle)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* set the context so this Watch object
|
||||||
|
* is passed back thru the Io_handler
|
||||||
|
*/
|
||||||
|
_watch_handle.context(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
~Watch_node()
|
||||||
|
{
|
||||||
|
_watch_handle.context((Vfs::Vfs_watch_handle::Context*)~0ULL);
|
||||||
|
_watch_handle.fs().close(&_watch_handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
static Watch_node &node_by_context(Vfs::Vfs_watch_handle::Context &context)
|
||||||
|
{
|
||||||
|
return static_cast<Watch_node &>(context);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The global handler has drawn an association from a watch
|
||||||
|
* context and this open node, now process the event at the
|
||||||
|
* session for this node.
|
||||||
|
*/
|
||||||
|
void handle_watch_response() {
|
||||||
|
_session_io_handler.handle_node_watch(*this); }
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
struct Vfs_server::Symlink : Io_node
|
||||||
|
{
|
||||||
|
Symlink(Node_space &space,
|
||||||
|
Vfs::File_system &vfs,
|
||||||
|
Genode::Allocator &alloc,
|
||||||
|
Session_io_handler &node_io_handler,
|
||||||
|
char const *link_path,
|
||||||
|
Mode mode,
|
||||||
|
bool create)
|
||||||
|
: Io_node(space, link_path, mode, node_io_handler)
|
||||||
{
|
{
|
||||||
assert_openlink(vfs.openlink(link_path, create, &_handle, alloc));
|
assert_openlink(vfs.openlink(link_path, create, &_handle, alloc));
|
||||||
_handle->context = &context();
|
_handle->context = &context();
|
||||||
@ -316,7 +412,7 @@ struct Vfs_server::Symlink : Node
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
class Vfs_server::File : public Node
|
class Vfs_server::File : public Io_node
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
|
|
||||||
@ -330,15 +426,15 @@ class Vfs_server::File : public Node
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
File(Node_space &space,
|
File(Node_space &space,
|
||||||
Vfs::File_system &vfs,
|
Vfs::File_system &vfs,
|
||||||
Genode::Allocator &alloc,
|
Genode::Allocator &alloc,
|
||||||
Node_io_handler &node_io_handler,
|
Session_io_handler &node_io_handler,
|
||||||
char const *file_path,
|
char const *file_path,
|
||||||
Mode fs_mode,
|
Mode fs_mode,
|
||||||
bool create)
|
bool create)
|
||||||
:
|
:
|
||||||
Node(space, file_path, fs_mode, node_io_handler)
|
Io_node(space, file_path, fs_mode, node_io_handler)
|
||||||
{
|
{
|
||||||
unsigned vfs_mode =
|
unsigned vfs_mode =
|
||||||
(fs_mode-1) | (create ? Vfs::Directory_service::OPEN_MODE_CREATE : 0);
|
(fs_mode-1) | (create ? Vfs::Directory_service::OPEN_MODE_CREATE : 0);
|
||||||
@ -387,15 +483,15 @@ class Vfs_server::File : public Node
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
struct Vfs_server::Directory : Node
|
struct Vfs_server::Directory : Io_node
|
||||||
{
|
{
|
||||||
Directory(Node_space &space,
|
Directory(Node_space &space,
|
||||||
Vfs::File_system &vfs,
|
Vfs::File_system &vfs,
|
||||||
Genode::Allocator &alloc,
|
Genode::Allocator &alloc,
|
||||||
Node_io_handler &node_io_handler,
|
Session_io_handler &node_io_handler,
|
||||||
char const *dir_path,
|
char const *dir_path,
|
||||||
bool create)
|
bool create)
|
||||||
: Node(space, dir_path, READ_ONLY, node_io_handler)
|
: Io_node(space, dir_path, READ_ONLY, node_io_handler)
|
||||||
{
|
{
|
||||||
assert_opendir(vfs.opendir(dir_path, create, &_handle, alloc));
|
assert_opendir(vfs.opendir(dir_path, create, &_handle, alloc));
|
||||||
_handle->context = &context();
|
_handle->context = &context();
|
||||||
@ -406,7 +502,6 @@ struct Vfs_server::Directory : Node
|
|||||||
Node_space::Id file(Node_space &space,
|
Node_space::Id file(Node_space &space,
|
||||||
Vfs::File_system &vfs,
|
Vfs::File_system &vfs,
|
||||||
Genode::Allocator &alloc,
|
Genode::Allocator &alloc,
|
||||||
Node_io_handler &node_io_handler,
|
|
||||||
char const *file_path,
|
char const *file_path,
|
||||||
Mode mode,
|
Mode mode,
|
||||||
bool create)
|
bool create)
|
||||||
@ -416,8 +511,9 @@ struct Vfs_server::Directory : Node
|
|||||||
|
|
||||||
File *file;
|
File *file;
|
||||||
try {
|
try {
|
||||||
file = new (alloc)
|
file = new (alloc) File(space, vfs, alloc,
|
||||||
File(space, vfs, alloc, node_io_handler, path_str, mode, create);
|
_session_io_handler,
|
||||||
|
path_str, mode, create);
|
||||||
} catch (Out_of_memory) { throw Out_of_ram(); }
|
} catch (Out_of_memory) { throw Out_of_ram(); }
|
||||||
|
|
||||||
if (create)
|
if (create)
|
||||||
@ -436,7 +532,8 @@ struct Vfs_server::Directory : Node
|
|||||||
char const *path_str = subpath.base();
|
char const *path_str = subpath.base();
|
||||||
|
|
||||||
Symlink *link;
|
Symlink *link;
|
||||||
try { link = new (alloc) Symlink(space, vfs, alloc, _node_io_handler,
|
try { link = new (alloc) Symlink(space, vfs, alloc,
|
||||||
|
_session_io_handler,
|
||||||
path_str, mode, create); }
|
path_str, mode, create); }
|
||||||
catch (Out_of_memory) { throw Out_of_ram(); }
|
catch (Out_of_memory) { throw Out_of_ram(); }
|
||||||
if (create)
|
if (create)
|
||||||
|
Loading…
Reference in New Issue
Block a user