mirror of
https://github.com/genodelabs/genode.git
synced 2025-05-31 22:50:54 +00:00
Noux terminal file system
This patch adds a new "terminal" file system type to Noux, which allows to create a "character device" file that is connected to a Genode 'Terminal' service. The 'Terminal' session created by the file system has the label "noux(terminal_fs)" to distinguish it from the 'Terminal' session created by Noux itself. Fixes #244.
This commit is contained in:
parent
7d86edc355
commit
1bf7967463
@ -120,6 +120,7 @@ namespace Noux {
|
|||||||
DIRENT_TYPE_FILE,
|
DIRENT_TYPE_FILE,
|
||||||
DIRENT_TYPE_DIRECTORY,
|
DIRENT_TYPE_DIRECTORY,
|
||||||
DIRENT_TYPE_FIFO,
|
DIRENT_TYPE_FIFO,
|
||||||
|
DIRENT_TYPE_CHARDEV,
|
||||||
DIRENT_TYPE_SYMLINK,
|
DIRENT_TYPE_SYMLINK,
|
||||||
DIRENT_TYPE_END
|
DIRENT_TYPE_END
|
||||||
};
|
};
|
||||||
|
219
ports/run/noux_terminal_fs.run
Normal file
219
ports/run/noux_terminal_fs.run
Normal file
@ -0,0 +1,219 @@
|
|||||||
|
#
|
||||||
|
# \brief Test for the Noux terminal file system
|
||||||
|
# \author Christian Prochaska
|
||||||
|
# \date 2012-06-11
|
||||||
|
#
|
||||||
|
|
||||||
|
#
|
||||||
|
# The test shows two framebuffer terminals. The upper terminal runs an
|
||||||
|
# interatctive shell and the lower terminal can be accessed from the shell via
|
||||||
|
# /dev/terminal.
|
||||||
|
#
|
||||||
|
# Example for writing to the lower terminal: echo "test" > /dev/terminal
|
||||||
|
# Example for reading a line from the lower terminal: head -n 1 /dev/terminal
|
||||||
|
#
|
||||||
|
|
||||||
|
set build_components {
|
||||||
|
core init drivers/timer noux lib/libc_noux
|
||||||
|
drivers/framebuffer drivers/pci drivers/input
|
||||||
|
server/terminal server/ram_fs server/nitpicker
|
||||||
|
server/nit_fb
|
||||||
|
}
|
||||||
|
|
||||||
|
#
|
||||||
|
# Build Noux packages only once
|
||||||
|
#
|
||||||
|
foreach pkg {bash coreutils} {
|
||||||
|
lappend_if [expr ![file exists bin/$pkg]] build_components noux-pkg/$pkg }
|
||||||
|
|
||||||
|
build $build_components
|
||||||
|
|
||||||
|
# strip all binaries prior archiving
|
||||||
|
exec sh -c "find bin/bash/ bin/coreutils/ -type f | (xargs strip || true) 2>/dev/null"
|
||||||
|
|
||||||
|
exec tar cfv bin/bash.tar -h -C bin/bash .
|
||||||
|
exec tar cfv bin/coreutils.tar -h -C bin/coreutils .
|
||||||
|
|
||||||
|
create_boot_directory
|
||||||
|
|
||||||
|
append config {
|
||||||
|
<config verbose="yes">
|
||||||
|
<parent-provides>
|
||||||
|
<service name="ROM"/>
|
||||||
|
<service name="LOG"/>
|
||||||
|
<service name="CAP"/>
|
||||||
|
<service name="RAM"/>
|
||||||
|
<service name="RM"/>
|
||||||
|
<service name="CPU"/>
|
||||||
|
<service name="PD"/>
|
||||||
|
<service name="IRQ"/>
|
||||||
|
<service name="IO_PORT"/>
|
||||||
|
<service name="IO_MEM"/>
|
||||||
|
<service name="SIGNAL"/>
|
||||||
|
</parent-provides>
|
||||||
|
<default-route>
|
||||||
|
<any-service> <any-child/> <parent/> </any-service>
|
||||||
|
</default-route>
|
||||||
|
<start name="timer">
|
||||||
|
<resource name="RAM" quantum="1M"/>
|
||||||
|
<provides><service name="Timer"/></provides>
|
||||||
|
</start> }
|
||||||
|
|
||||||
|
append_if [have_spec sdl] config {
|
||||||
|
<start name="fb_sdl">
|
||||||
|
<resource name="RAM" quantum="4M"/>
|
||||||
|
<provides>
|
||||||
|
<service name="Input"/>
|
||||||
|
<service name="Framebuffer"/>
|
||||||
|
</provides>
|
||||||
|
</start> }
|
||||||
|
|
||||||
|
append_if [have_spec pci] config {
|
||||||
|
<start name="pci_drv">
|
||||||
|
<resource name="RAM" quantum="1M"/>
|
||||||
|
<provides><service name="PCI"/></provides>
|
||||||
|
</start> }
|
||||||
|
|
||||||
|
append_if [have_spec vesa] config {
|
||||||
|
<start name="vesa_drv">
|
||||||
|
<resource name="RAM" quantum="1M"/>
|
||||||
|
<provides><service name="Framebuffer"/></provides>
|
||||||
|
</start> }
|
||||||
|
|
||||||
|
append_if [have_spec pl11x] config {
|
||||||
|
<start name="pl11x_drv">
|
||||||
|
<resource name="RAM" quantum="2M"/>
|
||||||
|
<provides><service name="Framebuffer"/></provides>
|
||||||
|
</start> }
|
||||||
|
|
||||||
|
append_if [have_spec ps2] config {
|
||||||
|
<start name="ps2_drv">
|
||||||
|
<resource name="RAM" quantum="1M"/>
|
||||||
|
<provides><service name="Input"/></provides>
|
||||||
|
</start> }
|
||||||
|
|
||||||
|
append config {
|
||||||
|
<start name="nitpicker">
|
||||||
|
<resource name="RAM" quantum="1M"/>
|
||||||
|
<provides><service name="Nitpicker"/></provides>
|
||||||
|
<route> }
|
||||||
|
append_if [have_spec ps2] config {
|
||||||
|
<service name="Input"><child name="ps2_drv"/></service> }
|
||||||
|
append_if [have_spec vesa] config {
|
||||||
|
<service name="Framebuffer"><child name="vesa_drv"/> </service> }
|
||||||
|
append_if [have_spec pl11x] config {
|
||||||
|
<service name="Framebuffer"><child name="pl11x_drv"/> </service> }
|
||||||
|
append config {
|
||||||
|
<any-service><parent/><any-child/></any-service>
|
||||||
|
</route>
|
||||||
|
</start>
|
||||||
|
<start name="terminal_noux_fb">
|
||||||
|
<binary name="nit_fb"/>
|
||||||
|
<resource name="RAM" quantum="2M"/>
|
||||||
|
<provides>
|
||||||
|
<service name="Framebuffer"/>
|
||||||
|
<service name="Input"/>
|
||||||
|
</provides>
|
||||||
|
<config xpos="0" ypos="0" width="1024" height="380" refresh_rate="25"/>
|
||||||
|
</start>
|
||||||
|
<start name="terminal_test_fb">
|
||||||
|
<binary name="nit_fb"/>
|
||||||
|
<resource name="RAM" quantum="2M"/>
|
||||||
|
<provides>
|
||||||
|
<service name="Framebuffer"/>
|
||||||
|
<service name="Input"/>
|
||||||
|
</provides>
|
||||||
|
<config xpos="0" ypos="388" width="1024" height="380" refresh_rate="25"/>
|
||||||
|
</start>
|
||||||
|
<start name="terminal_noux">
|
||||||
|
<binary name="terminal"/>
|
||||||
|
<resource name="RAM" quantum="2M"/>
|
||||||
|
<provides><service name="Terminal"/></provides>
|
||||||
|
<route>
|
||||||
|
<service name="Input"><child name="terminal_noux_fb"/></service>
|
||||||
|
<service name="Framebuffer"><child name="terminal_noux_fb"/> </service>
|
||||||
|
<any-service><parent/><any-child/></any-service>
|
||||||
|
</route>
|
||||||
|
<config>
|
||||||
|
<keyboard layout="de"/>
|
||||||
|
</config>
|
||||||
|
</start>
|
||||||
|
<start name="terminal_test">
|
||||||
|
<binary name="terminal"/>
|
||||||
|
<resource name="RAM" quantum="2M"/>
|
||||||
|
<provides><service name="Terminal"/></provides>
|
||||||
|
<route>
|
||||||
|
<service name="Input"><child name="terminal_test_fb"/></service>
|
||||||
|
<service name="Framebuffer"><child name="terminal_test_fb"/> </service>
|
||||||
|
<any-service><parent/><any-child/></any-service>
|
||||||
|
</route>
|
||||||
|
<config>
|
||||||
|
<keyboard layout="de"/>
|
||||||
|
</config>
|
||||||
|
</start>
|
||||||
|
<start name="ram_fs">
|
||||||
|
<resource name="RAM" quantum="10M"/>
|
||||||
|
<provides><service name="File_system"/></provides>
|
||||||
|
<config>
|
||||||
|
<!-- constrain sessions according to their labels -->
|
||||||
|
<policy label="noux -> root" root="/" />
|
||||||
|
</config>
|
||||||
|
</start>
|
||||||
|
<start name="noux">
|
||||||
|
<resource name="RAM" quantum="1G" />
|
||||||
|
<route>
|
||||||
|
<service name="Terminal">
|
||||||
|
<if-arg key="label" value="noux"/><child name="terminal_noux"/>
|
||||||
|
</service>
|
||||||
|
<service name="Terminal">
|
||||||
|
<if-arg key="label" value="noux -> noux(terminal_fs)"/><child name="terminal_test"/>
|
||||||
|
</service>
|
||||||
|
<any-service><parent/><any-child/></any-service>
|
||||||
|
</route>
|
||||||
|
<config>
|
||||||
|
<fstab>
|
||||||
|
<tar name="coreutils.tar" />
|
||||||
|
<tar name="bash.tar" />
|
||||||
|
|
||||||
|
<dir name="ram"> <fs label="root" /> </dir>
|
||||||
|
<dir name="dev"> <terminal name="terminal" /> </dir>
|
||||||
|
</fstab>
|
||||||
|
<start name="/bin/bash">
|
||||||
|
<env name="TERM" value="linux" />
|
||||||
|
</start>
|
||||||
|
</config>
|
||||||
|
</start>
|
||||||
|
</config>
|
||||||
|
}
|
||||||
|
|
||||||
|
install_config $config
|
||||||
|
|
||||||
|
|
||||||
|
#
|
||||||
|
# Boot modules
|
||||||
|
#
|
||||||
|
|
||||||
|
# generic modules
|
||||||
|
set boot_modules {
|
||||||
|
core init timer ld.lib.so noux terminal ram_fs nitpicker nit_fb
|
||||||
|
libc.lib.so libm.lib.so libc_noux.lib.so
|
||||||
|
bash.tar coreutils.tar
|
||||||
|
}
|
||||||
|
|
||||||
|
# platform-specific modules
|
||||||
|
lappend_if [have_spec linux] boot_modules fb_sdl
|
||||||
|
lappend_if [have_spec pci] boot_modules pci_drv
|
||||||
|
lappend_if [have_spec vesa] boot_modules vesa_drv
|
||||||
|
lappend_if [have_spec ps2] boot_modules ps2_drv
|
||||||
|
lappend_if [have_spec pl11x] boot_modules pl11x_drv
|
||||||
|
|
||||||
|
build_boot_image $boot_modules
|
||||||
|
|
||||||
|
if {[have_spec x86_64]} {
|
||||||
|
# bash.tar is really huge when built for x86_64
|
||||||
|
append qemu_args " -m 300 "
|
||||||
|
}
|
||||||
|
|
||||||
|
run_genode_until forever
|
||||||
|
|
||||||
|
exec rm bin/bash.tar
|
@ -931,6 +931,7 @@ namespace {
|
|||||||
case Noux::Sysio::DIRENT_TYPE_FILE: dirent->d_type = DT_REG; break;
|
case Noux::Sysio::DIRENT_TYPE_FILE: dirent->d_type = DT_REG; break;
|
||||||
case Noux::Sysio::DIRENT_TYPE_SYMLINK: dirent->d_type = DT_LNK; break;
|
case Noux::Sysio::DIRENT_TYPE_SYMLINK: dirent->d_type = DT_LNK; break;
|
||||||
case Noux::Sysio::DIRENT_TYPE_FIFO: dirent->d_type = DT_FIFO; break;
|
case Noux::Sysio::DIRENT_TYPE_FIFO: dirent->d_type = DT_FIFO; break;
|
||||||
|
case Noux::Sysio::DIRENT_TYPE_CHARDEV: dirent->d_type = DT_CHR; break;
|
||||||
case Noux::Sysio::DIRENT_TYPE_END: return 0;
|
case Noux::Sysio::DIRENT_TYPE_END: return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
#include <noux_session/sysio.h>
|
#include <noux_session/sysio.h>
|
||||||
#include <tar_file_system.h>
|
#include <tar_file_system.h>
|
||||||
#include <fs_file_system.h>
|
#include <fs_file_system.h>
|
||||||
|
#include <terminal_file_system.h>
|
||||||
|
|
||||||
namespace Noux {
|
namespace Noux {
|
||||||
|
|
||||||
@ -90,6 +91,11 @@ namespace Noux {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (sub_node.has_type("terminal")) {
|
||||||
|
_append_file_system(new Terminal_file_system(sub_node));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
char type_name[64];
|
char type_name[64];
|
||||||
sub_node.type_name(type_name, sizeof(type_name));
|
sub_node.type_name(type_name, sizeof(type_name));
|
||||||
|
210
ports/src/noux/terminal_file_system.h
Normal file
210
ports/src/noux/terminal_file_system.h
Normal file
@ -0,0 +1,210 @@
|
|||||||
|
/*
|
||||||
|
* \brief Terminal file system
|
||||||
|
* \author Christian Prochaska
|
||||||
|
* \date 2012-05-23
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2012 Genode Labs GmbH
|
||||||
|
*
|
||||||
|
* This file is part of the Genode OS framework, which is distributed
|
||||||
|
* under the terms of the GNU General Public License version 2.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _NOUX__TERMINAL_FILE_SYSTEM_H_
|
||||||
|
#define _NOUX__TERMINAL_FILE_SYSTEM_H_
|
||||||
|
|
||||||
|
/* Genode includes */
|
||||||
|
#include <base/printf.h>
|
||||||
|
#include <base/lock.h>
|
||||||
|
#include <terminal_session/connection.h>
|
||||||
|
#include <util/string.h>
|
||||||
|
|
||||||
|
/* Noux includes */
|
||||||
|
#include <noux_session/sysio.h>
|
||||||
|
#include "file_system.h"
|
||||||
|
|
||||||
|
|
||||||
|
namespace Noux {
|
||||||
|
|
||||||
|
class Terminal_file_system : public File_system
|
||||||
|
{
|
||||||
|
private:
|
||||||
|
|
||||||
|
Terminal::Session_client _terminal;
|
||||||
|
Signal_context _read_avail_sig_ctx;
|
||||||
|
Signal_receiver _read_avail_sig_rec;
|
||||||
|
|
||||||
|
enum { FILENAME_MAX_LEN = 64 };
|
||||||
|
char _filename[FILENAME_MAX_LEN];
|
||||||
|
|
||||||
|
bool _is_root(const char *path)
|
||||||
|
{
|
||||||
|
return (strcmp(path, "") == 0) || (strcmp(path, "/") == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool _is_terminal_file(const char *path)
|
||||||
|
{
|
||||||
|
return (strlen(path) == (strlen(_filename) + 1)) &&
|
||||||
|
(strcmp(&path[1], _filename) == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
Terminal_file_system(Xml_node config)
|
||||||
|
: _terminal(env()->parent()->session<Terminal::Session>
|
||||||
|
("ram_quota=8192, label=\"noux(terminal_fs)\""))
|
||||||
|
{
|
||||||
|
_filename[0] = 0;
|
||||||
|
try { config.attribute("name").value(_filename, sizeof(_filename)); }
|
||||||
|
catch (...) { }
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Wait for connection-established signal
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* create signal receiver, just for the single signal */
|
||||||
|
Signal_context sig_ctx;
|
||||||
|
Signal_receiver sig_rec;
|
||||||
|
Signal_context_capability sig_cap = sig_rec.manage(&sig_ctx);
|
||||||
|
|
||||||
|
/* register signal handler */
|
||||||
|
_terminal.connected_sigh(sig_cap);
|
||||||
|
|
||||||
|
/* wati for signal */
|
||||||
|
sig_rec.wait_for_signal();
|
||||||
|
sig_rec.dissolve(&sig_ctx);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Register "read available" signal handler
|
||||||
|
*/
|
||||||
|
_terminal.read_avail_sigh(_read_avail_sig_rec.manage(&_read_avail_sig_ctx));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*********************************
|
||||||
|
** Directory-service interface **
|
||||||
|
*********************************/
|
||||||
|
|
||||||
|
Dataspace_capability dataspace(char const *path)
|
||||||
|
{
|
||||||
|
/* not supported */
|
||||||
|
return Dataspace_capability();
|
||||||
|
}
|
||||||
|
|
||||||
|
void release(char const *path, Dataspace_capability ds_cap)
|
||||||
|
{
|
||||||
|
/* not supported */
|
||||||
|
}
|
||||||
|
|
||||||
|
bool stat(Sysio *sysio, char const *path)
|
||||||
|
{
|
||||||
|
Sysio::Stat st = { 0, 0, 0, 0, 0, 0 };
|
||||||
|
|
||||||
|
if (_is_root(path))
|
||||||
|
st.mode = Sysio::STAT_MODE_DIRECTORY;
|
||||||
|
else if (_is_terminal_file(path)) {
|
||||||
|
st.mode = Sysio::STAT_MODE_CHARDEV;
|
||||||
|
} else {
|
||||||
|
sysio->error.stat = Sysio::STAT_ERR_NO_ENTRY;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
sysio->stat_out.st = st;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool dirent(Sysio *sysio, char const *path, off_t index)
|
||||||
|
{
|
||||||
|
if (_is_root(path)) {
|
||||||
|
if (index == 0) {
|
||||||
|
sysio->dirent_out.entry.type = Sysio::DIRENT_TYPE_CHARDEV;
|
||||||
|
strncpy(sysio->dirent_out.entry.name,
|
||||||
|
_filename,
|
||||||
|
sizeof(sysio->dirent_out.entry.name));
|
||||||
|
} else {
|
||||||
|
sysio->dirent_out.entry.type = Sysio::DIRENT_TYPE_END;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t num_dirent(char const *path)
|
||||||
|
{
|
||||||
|
if (_is_root(path))
|
||||||
|
return 1;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool is_directory(char const *path)
|
||||||
|
{
|
||||||
|
if (_is_root(path))
|
||||||
|
return true;
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
char const *leaf_path(char const *path)
|
||||||
|
{
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
|
Vfs_handle *open(Sysio *sysio, char const *path)
|
||||||
|
{
|
||||||
|
if (!_is_terminal_file(path)) {
|
||||||
|
sysio->error.open = Sysio::OPEN_ERR_UNACCESSIBLE;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new (env()->heap()) Vfs_handle(this, this, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool unlink(Sysio *sysio, char const *path)
|
||||||
|
{
|
||||||
|
/* not supported */
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool rename(Sysio *sysio, char const *from_path, char const *to_path)
|
||||||
|
{
|
||||||
|
/* not supported */
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool mkdir(Sysio *sysio, char const *path)
|
||||||
|
{
|
||||||
|
/* not supported */
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/***************************
|
||||||
|
** File_system interface **
|
||||||
|
***************************/
|
||||||
|
|
||||||
|
char const *name() const { return "terminal"; }
|
||||||
|
|
||||||
|
|
||||||
|
/********************************
|
||||||
|
** File I/O service interface **
|
||||||
|
********************************/
|
||||||
|
|
||||||
|
bool write(Sysio *sysio, Vfs_handle *handle)
|
||||||
|
{
|
||||||
|
sysio->write_out.count = _terminal.write(sysio->write_in.chunk, sysio->write_in.count);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool read(Sysio *sysio, Vfs_handle *vfs_handle)
|
||||||
|
{
|
||||||
|
_read_avail_sig_rec.wait_for_signal();
|
||||||
|
sysio->read_out.count = _terminal.read(sysio->read_out.chunk, sysio->read_in.count);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* _NOUX__TERMINAL_FILE_SYSTEM_H_ */
|
Loading…
x
Reference in New Issue
Block a user