diff --git a/repos/os/run/cpufreq.run b/repos/os/run/cpufreq.run
index 80020e611d..3eafa67d9b 100644
--- a/repos/os/run/cpufreq.run
+++ b/repos/os/run/cpufreq.run
@@ -54,7 +54,6 @@ set config {
-
}
install_config $config
@@ -74,5 +73,11 @@ set boot_modules {
}
build_boot_image $boot_modules
-run_genode_until forever
+run_genode_until {.*Round 04: A A.*\n.*Setting CPU frequency low.*\n.*Setting CPU frequency high.*\n.*Round 05: A A} 60
+grep_output {^\[init -> test-cpufreq\] }
+
+compare_output_to {
+[init -> test-cpufreq] Setting CPU frequency low
+[init -> test-cpufreq] Setting CPU frequency high
+}
diff --git a/repos/os/run/dynamic_config_loader.run b/repos/os/run/dynamic_config_loader.run
index c86a676f5a..86406f836c 100644
--- a/repos/os/run/dynamic_config_loader.run
+++ b/repos/os/run/dynamic_config_loader.run
@@ -24,6 +24,12 @@ install_config {
+
+
+
+
+
+
diff --git a/repos/os/run/fb_bench.run b/repos/os/run/fb_bench.run
index 58c5fbb265..4725fee601 100644
--- a/repos/os/run/fb_bench.run
+++ b/repos/os/run/fb_bench.run
@@ -69,19 +69,13 @@ install_config $config
# Boot modules
#
-# generic modules
-set boot_modules {
- core ld.lib.so init timer
- test-fb_bench
-}
+set boot_modules { core ld.lib.so init timer test-fb_bench }
-# platform-specific modules
append_platform_drv_boot_modules
-lappend_if [have_spec sdl] boot_modules fb_sdl
-lappend_if [have_spec framebuffer] boot_modules fb_drv
+lappend_if [have_spec sdl] boot_modules fb_sdl
+lappend_if [have_spec framebuffer] boot_modules fb_drv
build_boot_image $boot_modules
-run_genode_until forever
-
+run_genode_until {.*--- Framebuffer benchmark finished ---.*\n} 30
diff --git a/repos/os/run/ram_fs_chunk.run b/repos/os/run/ram_fs_chunk.run
index 09e9a407e3..bc1ab73edb 100644
--- a/repos/os/run/ram_fs_chunk.run
+++ b/repos/os/run/ram_fs_chunk.run
@@ -30,13 +30,13 @@ build_boot_image "core ld.lib.so init test-ram_fs_chunk"
append qemu_args "-nographic -m 64"
-run_genode_until {child "test-ram_fs_chunk" exited with exit value 0.*\n} 10
+run_genode_until {.*--- RAM filesystem chunk test finished ---.*\n} 10
grep_output {^\[init -> test-ram_fs_chunk\]}
unify_output { sizeof=[0-9]+} {}
compare_output_to {
- [init -> test-ram_fs_chunk] --- ram_fs_chunk test ---
+ [init -> test-ram_fs_chunk] --- RAM filesystem chunk test ---
[init -> test-ram_fs_chunk] chunk sizes
[init -> test-ram_fs_chunk] level 0: payload=120
[init -> test-ram_fs_chunk] level 1: payload=24
@@ -77,6 +77,7 @@ compare_output_to {
[init -> test-ram_fs_chunk] trunc(2) -> content (size=2): "fi"
[init -> test-ram_fs_chunk] trunc(1) -> content (size=1): "f"
[init -> test-ram_fs_chunk] allocator: sum=0
+ [init -> test-ram_fs_chunk] --- RAM filesystem chunk test finished ---
}
diff --git a/repos/os/run/reconstructible.run b/repos/os/run/reconstructible.run
index 1541cb939c..1abad5d697 100644
--- a/repos/os/run/reconstructible.run
+++ b/repos/os/run/reconstructible.run
@@ -25,43 +25,43 @@ build_boot_image "core ld.lib.so init test-reconstructible"
append qemu_args "-nographic -m 64"
-run_genode_until {child "test-reconstructible" exited with exit value 0.*\n} 10
+run_genode_until {.*--- Reconstructible utility test finished ---.*\n} 10
grep_output {-> test-reconstructible}
compare_output_to {
-[init -> test-reconstructible] --- test-reconstructible started ---
+[init -> test-reconstructible] --- Reconstructible utility test ---
[init -> test-reconstructible] construct Object 1
[init -> test-reconstructible] construct Object 2
-[init -> test-reconstructible] -- create Compound object --
+[init -> test-reconstructible] create Compound object
[init -> test-reconstructible] construct Member_with_reference
[init -> test-reconstructible] construct Compound
[init -> test-reconstructible] compound.member.constructed returns 1
[init -> test-reconstructible] compound.lazy_member.constructed returns 0
-[init -> test-reconstructible] -- construct lazy member --
+[init -> test-reconstructible] construct lazy member
[init -> test-reconstructible] construct Member_with_reference
[init -> test-reconstructible] compound.lazy_member.constructed returns 1
-[init -> test-reconstructible] -- call method on member (with reference to Object 1) --
+[init -> test-reconstructible] call method on member (with reference to Object 1)
[init -> test-reconstructible] const method called on Object 1
-[init -> test-reconstructible] -- reconstruct member with Object 2 as reference --
+[init -> test-reconstructible] reconstruct member with Object 2 as reference
[init -> test-reconstructible] destruct Member_with_reference
[init -> test-reconstructible] construct Member_with_reference
-[init -> test-reconstructible] -- call method on member --
+[init -> test-reconstructible] call method on member
[init -> test-reconstructible] const method called on Object 2
-[init -> test-reconstructible] -- destruct member --
+[init -> test-reconstructible] destruct member
[init -> test-reconstructible] destruct Member_with_reference
-[init -> test-reconstructible] -- try to call method on member, catch exception --
+[init -> test-reconstructible] try to call method on member, catch exception
[init -> test-reconstructible] got exception, as expected
-[init -> test-reconstructible] -- destruct Compound and Objects 1 and 2 --
+[init -> test-reconstructible] destruct Compound and Objects 1 and 2
[init -> test-reconstructible] destruct Compound
[init -> test-reconstructible] destruct Member_with_reference
[init -> test-reconstructible] destruct Object 2
[init -> test-reconstructible] destruct Object 1
-[init -> test-reconstructible] -- construct Throwing object
+[init -> test-reconstructible] construct Throwing object
[init -> test-reconstructible] construct Throwing -> don't throw
[init -> test-reconstructible] destruct Throwing
[init -> test-reconstructible] construct Throwing -> throw exception
-[init -> test-reconstructible] -- catched exception as expected
-[init -> test-reconstructible] --- test-reconstructible finished ---
+[init -> test-reconstructible] got exception, as expected
+[init -> test-reconstructible] --- Reconstructible utility test finished ---
}
diff --git a/repos/os/run/rom_blk.run b/repos/os/run/rom_blk.run
index c710785414..3cbbebb78d 100644
--- a/repos/os/run/rom_blk.run
+++ b/repos/os/run/rom_blk.run
@@ -1,9 +1,3 @@
-#
-# \brief Test for 'rom_blk' service
-# \author Stefan Kalkowski
-# \date 2011-05-10
-#
-
build "core init server/rom_blk test/rom_blk"
create_boot_directory
@@ -37,4 +31,4 @@ build_boot_image "core ld.lib.so init rom_blk test-rom_blk"
append qemu_args "-m 64 -nographic "
-run_genode_until "all done, finished!" 10
+run_genode_until {.*--- ROM Block test finished ---.*\n} 10
diff --git a/repos/os/run/signal.run b/repos/os/run/signal.run
index a79ba937eb..7c227d2aeb 100644
--- a/repos/os/run/signal.run
+++ b/repos/os/run/signal.run
@@ -32,4 +32,4 @@ build_boot_image "core ld.lib.so init timer test-signal"
append qemu_args "-nographic -m 64"
-run_genode_until {.*child "test-signal" exited with exit value 0.*} 200
+run_genode_until {.*--- Signalling test finished ---.*\n} 200
diff --git a/repos/os/run/synced_interface.run b/repos/os/run/synced_interface.run
index 0731238137..f91b1e503f 100644
--- a/repos/os/run/synced_interface.run
+++ b/repos/os/run/synced_interface.run
@@ -25,14 +25,15 @@ build_boot_image "core ld.lib.so init test-synced_interface"
append qemu_args "-nographic -m 64"
-run_genode_until {child "test-synced_interface" exited with exit value 0} 10
+run_genode_until {.*--- Synced interface test finished ---.*\n} 10
grep_output {-> test-synced_interface}
compare_output_to {
+[init -> test-synced_interface] --- Synced interface test ---
[init -> test-synced_interface] lock
[init -> test-synced_interface] adding 13 + 14
[init -> test-synced_interface] unlock
[init -> test-synced_interface] result is 27
+[init -> test-synced_interface] --- Synced interface test finished ---
}
-
diff --git a/repos/os/run/thread_join.run b/repos/os/run/thread_join.run
index f52e9d4df7..e5b9cdcdea 100644
--- a/repos/os/run/thread_join.run
+++ b/repos/os/run/thread_join.run
@@ -32,4 +32,4 @@ build_boot_image "core ld.lib.so init timer test-thread_join"
append qemu_args "-nographic -m 64"
-run_genode_until {child "test-thread_join" exited with exit value 0.*\n} 10
+run_genode_until {.*--- Thread join test finished ---.*\n} 10
diff --git a/repos/os/run/timed_semaphore.run b/repos/os/run/timed_semaphore.run
index c19ce26664..1035704ee9 100644
--- a/repos/os/run/timed_semaphore.run
+++ b/repos/os/run/timed_semaphore.run
@@ -54,4 +54,4 @@ build_boot_image $boot_modules
append qemu_args " -m 64 -nographic "
-run_genode_until "end of timed-semaphore test" 10
+run_genode_until {.*--- Timed semaphore test finished ---.*\n} 10
diff --git a/repos/os/run/trace.run b/repos/os/run/trace.run
index 3f2ca8c67e..1d4faae806 100644
--- a/repos/os/run/trace.run
+++ b/repos/os/run/trace.run
@@ -64,5 +64,4 @@ build_boot_image $boot_modules
append qemu_args " -nographic -serial mon:stdio -m 256 "
-run_genode_until forever
-#{.*child exited with exit value 0.* } 60
+run_genode_until "--- test-trace finished ---" 30
diff --git a/repos/os/src/test/cpufreq/main.cc b/repos/os/src/test/cpufreq/main.cc
index f092518b8c..a4551721a6 100644
--- a/repos/os/src/test/cpufreq/main.cc
+++ b/repos/os/src/test/cpufreq/main.cc
@@ -1,42 +1,49 @@
/*
* \brief Test for changing the CPU frequency
* \author Stefan Kalkowski
+ * \author Martin Stein
* \date 2013-06-14
*/
/*
- * Copyright (C) 2013 Genode Labs GmbH
+ * Copyright (C) 2013-2017 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.
*/
/* Genode includes */
-#include
-#include
-#include
-
+#include
#include
#include
#include
-int main(int argc, char **argv)
+using namespace Genode;
+
+struct Main
{
- using namespace Genode;
+ enum { PERIOD_US = 8 * 1000 * 1000 };
- log("--- test-cpufreq started ---");
+ Env &env;
+ Timer::Connection timer { env };
+ Regulator::Connection cpu_regulator { env, Regulator::CLK_CPU };
+ Signal_handler timer_handler { env.ep(), *this, &Main::handle_timer };
+ bool high { true };
- static Timer::Connection timer;
- static Regulator::Connection cpu_regulator(Regulator::CLK_CPU);
- bool high = true;
-
- while (true) {
- timer.msleep(10000);
+ void handle_timer()
+ {
log("Setting CPU frequency ", high ? "low" : "high");
- cpu_regulator.level(high ? Regulator::CPU_FREQ_200
- : Regulator::CPU_FREQ_1600);
+ cpu_regulator.level(high ? Regulator::CPU_FREQ_200 :
+ Regulator::CPU_FREQ_1600);
high = !high;
+ timer.trigger_once(PERIOD_US);
}
- return 1;
-}
+ Main(Env &env) : env(env)
+ {
+ timer.sigh(timer_handler);
+ timer.trigger_once(PERIOD_US);
+ }
+};
+
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/dynamic_config/loader/main.cc b/repos/os/src/test/dynamic_config/loader/main.cc
index 27dfe22115..92d412431f 100644
--- a/repos/os/src/test/dynamic_config/loader/main.cc
+++ b/repos/os/src/test/dynamic_config/loader/main.cc
@@ -12,53 +12,40 @@
*/
/* Genode includes */
-#include
-#include
#include
#include
-
+#include
using namespace Genode;
-
-enum { CONFIG_SIZE = 100 };
-
-static Loader::Connection loader(8*1024*1024);
-
-
-static void update_config(int counter)
+struct Main
{
- Dataspace_capability config_ds =
- loader.alloc_rom_module("config", CONFIG_SIZE);
+ enum { CONFIG_SIZE = 100 };
- char *config_ds_addr = env()->rm_session()->attach(config_ds);
+ Env &env;
+ int counter { -1 };
+ Loader::Connection loader { env, 8 * 1024 * 1024 };
+ Timer::Connection timer { env };
+ Signal_handler timer_handler { env.ep(), *this, &Main::handle_timer };
- snprintf(config_ds_addr, CONFIG_SIZE,
- "%d",
- counter);
+ void handle_timer()
+ {
+ char *config_ds_addr =
+ env.rm().attach(loader.alloc_rom_module("config", CONFIG_SIZE));
- env()->rm_session()->detach(config_ds_addr);
-
- loader.commit_rom_module("config");
-}
-
-
-int main(int, char **)
-{
- update_config(-1);
-
- loader.start("test-dynamic_config", "test-label");
-
- /* update slave config at regular intervals */
- int counter = 0;
- for (;;) {
-
- static Timer::Connection timer;
- timer.msleep(250);
- update_config(counter++);
+ String<100> config("", counter++, "");
+ strncpy(config_ds_addr, config.string(), CONFIG_SIZE);
+ env.rm().detach(config_ds_addr);
+ loader.commit_rom_module("config");
+ timer.trigger_once(250 * 1000);
}
- sleep_forever();
+ Main(Env &env) : env(env)
+ {
+ timer.sigh(timer_handler);
+ handle_timer();
+ loader.start("test-dynamic_config", "test-label");
+ }
+};
- return 0;
-}
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/dynamic_config/main.cc b/repos/os/src/test/dynamic_config/main.cc
index b680fceb3a..4d3b428ca7 100644
--- a/repos/os/src/test/dynamic_config/main.cc
+++ b/repos/os/src/test/dynamic_config/main.cc
@@ -1,49 +1,45 @@
/*
* \brief Test for changing configuration at runtime
* \author Norman Feske
+ * \author Martin Stein
* \date 2012-04-04
*/
/*
- * Copyright (C) 2012-2013 Genode Labs GmbH
+ * Copyright (C) 2012-2017 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.
*/
/* Genode includes */
-#include
+#include
+#include
+using namespace Genode;
-void parse_config()
+struct Main
{
- try {
- long counter = 1;
- Genode::config()->xml_node().sub_node("counter").value(&counter);
- Genode::log("obtained counter value ", counter, " from config");
+ Env &env;
+ Attached_rom_dataspace config { env, "config" };
+ Signal_handler config_handler { env.ep(), *this, &Main::handle_config };
- } catch (...) {
- Genode::error("could not parse configuration");
+ void handle_config()
+ {
+ config.update();
+ try {
+ long counter = 1;
+ config.xml().sub_node("counter").value(&counter);
+ log("obtained counter value ", counter, " from config");
+ }
+ catch (...) { error("could not parse configuration"); }
}
-}
-
-int main(int, char **)
-{
- parse_config();
-
- /* register signal handler for config changes */
- Genode::Signal_receiver sig_rec;
- Genode::Signal_context sig_ctx;
- Genode::config()->sigh(sig_rec.manage(&sig_ctx));
-
- for (;;) {
-
- /* wait for config change */
- sig_rec.wait_for_signal();
-
- Genode::config()->reload();
- parse_config();
+ Main(Env &env) : env(env)
+ {
+ handle_config();
+ config.sigh(config_handler);
}
- return 0;
-}
+};
+
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/dynamic_config/server/main.cc b/repos/os/src/test/dynamic_config/server/main.cc
index 4d9d9920b7..c4ffbc42d7 100644
--- a/repos/os/src/test/dynamic_config/server/main.cc
+++ b/repos/os/src/test/dynamic_config/server/main.cc
@@ -1,6 +1,7 @@
/*
* \brief Test for changing configuration at runtime (server-side)
* \author Norman Feske
+ * \author Martin Stein
* \date 2012-04-04
*
* This program provides a generated config file as ROM service. After
@@ -8,65 +9,56 @@
*/
/*
- * Copyright (C) 2012-2013 Genode Labs GmbH
+ * Copyright (C) 2012-2017 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.
*/
/* Genode includes */
-#include
#include
#include
#include
#include
-#include
+#include
+using namespace Genode;
/*
* The implementation of this class follows the lines of
* 'os/include/os/child_policy_dynamic_rom.h'.
*/
-class Rom_session_component : public Genode::Rpc_object
+class Rom_session_component : public Rpc_object
{
private:
- Genode::Attached_ram_dataspace _fg;
- Genode::Attached_ram_dataspace _bg;
-
- bool _bg_has_pending_data;
-
- Genode::Lock _lock;
-
- Genode::Signal_context_capability _sigh;
+ Env &_env;
+ Attached_ram_dataspace _fg { _env.ram(), _env.rm(), 0 };
+ Attached_ram_dataspace _bg { _env.ram(), _env.rm(), 0 };
+ bool _bg_pending_data { false };
+ Signal_context_capability _sigh;
public:
- /**
- * Constructor
- */
- Rom_session_component()
- : _fg(0, 0), _bg(0, 0), _bg_has_pending_data(false) { }
+ Rom_session_component(Env &env) : _env(env) { }
/**
* Update the config file
*/
void configure(char const *data)
{
- Genode::Lock::Guard guard(_lock);
-
- Genode::size_t const data_len = Genode::strlen(data) + 1;
+ size_t const data_len = strlen(data) + 1;
/* let background buffer grow if needed */
if (_bg.size() < data_len)
- _bg.realloc(Genode::env()->ram_session(), data_len);
+ _bg.realloc(&_env.ram(), data_len);
- Genode::strncpy(_bg.local_addr(), data, data_len);
- _bg_has_pending_data = true;
+ strncpy(_bg.local_addr(), data, data_len);
+ _bg_pending_data = true;
/* inform client about the changed data */
if (_sigh.valid())
- Genode::Signal_transmitter(_sigh).submit();
+ Signal_transmitter(_sigh).submit();
}
@@ -74,67 +66,51 @@ class Rom_session_component : public Genode::Rpc_object
** ROM session interface **
***************************/
- Genode::Rom_dataspace_capability dataspace()
+ Rom_dataspace_capability dataspace() override
{
- Genode::Lock::Guard guard(_lock);
-
- if (!_fg.size() && !_bg_has_pending_data) {
- Genode::error("no data loaded");
- return Genode::Rom_dataspace_capability();
+ if (!_fg.size() && !_bg_pending_data) {
+ error("no data loaded");
+ return Rom_dataspace_capability();
}
-
/*
* Keep foreground if no background exists. Otherwise, use old
* background as new foreground.
*/
- if (_bg_has_pending_data) {
+ if (_bg_pending_data) {
_fg.swap(_bg);
- _bg_has_pending_data = false;
+ _bg_pending_data = false;
}
-
- Genode::Dataspace_capability ds_cap = _fg.cap();
- return Genode::static_cap_cast(ds_cap);
+ Dataspace_capability ds_cap = _fg.cap();
+ return static_cap_cast(ds_cap);
}
- void sigh(Genode::Signal_context_capability sigh_cap)
- {
- Genode::Lock::Guard guard(_lock);
- _sigh = sigh_cap;
- }
+ void sigh(Signal_context_capability sigh_cap) override { _sigh = sigh_cap; }
};
-
-int main(int argc, char **argv)
+struct Main
{
- using namespace Genode;
+ enum { STACK_SIZE = 2 * 1024 * sizeof(addr_t) };
- /* connection to capability service needed to create capabilities */
- static Cap_connection cap;
+ Env &env;
+ Rom_session_component rom_session { env };
+ Static_root rom_root { env.ep().manage(rom_session) };
+ int counter { -1 };
+ Timer::Connection timer { env };
+ Signal_handler timer_handler { env.ep(), *this, &Main::handle_timer };
- enum { STACK_SIZE = 8*1024 };
- static Rpc_entrypoint ep(&cap, STACK_SIZE, "rom_ep");
-
- static Rom_session_component rom_session;
- static Static_root rom_root(ep.manage(&rom_session));
-
- rom_session.configure("-1");
-
- /* announce server */
- env()->parent()->announce(ep.manage(&rom_root));
-
- int counter = 0;
- for (;;) {
-
- static Timer::Connection timer;
- timer.msleep(250);
-
- /* re-generate configuration */
- char buf[100];
- Genode::snprintf(buf, sizeof(buf),
- "%d",
- counter++);
-
- rom_session.configure(buf);
+ void handle_timer()
+ {
+ String<100> config("", counter++, "");
+ rom_session.configure(config.string());
+ timer.trigger_once(250 * 1000);
+ }
+
+ Main(Env &env) : env(env)
+ {
+ timer.sigh(timer_handler);
+ handle_timer();
+ env.parent().announce(env.ep().manage(rom_root));
}
- return 0;
};
+
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/fb_bench/main.cc b/repos/os/src/test/fb_bench/main.cc
index 6039f7b250..02f73c2677 100644
--- a/repos/os/src/test/fb_bench/main.cc
+++ b/repos/os/src/test/fb_bench/main.cc
@@ -1,138 +1,141 @@
/*
* \brief Framebuffer throughput test
* \author Norman Feske
+ * \author Martin Stein
* \date 2015-06-05
*/
/*
- * Copyright (C) 2012-2014 Genode Labs GmbH
+ * Copyright (C) 2012-2017 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.
*/
/* Genode includes */
-#include
-#include
+#include
+#include
#include
#include
#include
#include
+using namespace Genode;
-static unsigned long now_ms()
+struct Test
{
- static Timer::Connection timer;
- return timer.elapsed_ms();
-}
+ enum { DURATION_MS = 2000 };
+ Env &env;
+ int id;
+ Timer::Connection timer { env };
+ Heap heap { env.ram(), env.rm() };
+ Framebuffer::Connection fb { env, Framebuffer::Mode() };
+ Attached_dataspace fb_ds { env.rm(), fb.dataspace() };
+ Framebuffer::Mode const fb_mode { fb.mode() };
+ char *buf[2];
-int main(int argc, char **argv)
+ Test(Env &env, int id, char const *brief) : env(env), id(id)
+ {
+ log("\nTEST ", id, ": ", brief, "\n");
+ for (unsigned i = 0; i < 2; i++) {
+ if (!heap.alloc(fb_ds.size(), (void **)&buf[i])) {
+ env.parent().exit(-1); }
+ }
+ /* fill one memory buffer with white pixels */
+ memset(buf[1], ~0, fb_ds.size());
+ }
+
+ void conclusion(unsigned kib, unsigned start_ms, unsigned end_ms) {
+ log("throughput: ", kib / (end_ms - start_ms), " MiB/sec"); }
+
+ ~Test() { log("\nTEST ", id, " finished\n"); }
+};
+
+struct Bytewise_ram_test : Test
{
- using namespace Genode;
+ static constexpr char const *brief = "byte-wise memcpy from RAM to RAM";
- printf("--- test-fb_bench started ---\n");
-
- static Framebuffer::Connection fb;
-
- static Attached_dataspace fb_ds(fb.dataspace());
- static Framebuffer::Mode const fb_mode = fb.mode();
-
- /*
- * Allocate two memory buffers as big as the framebuffer.
- */
- char *src_buf[2];
- for (unsigned i = 0; i < 2; i++)
- src_buf[i] = (char *)env()->heap()->alloc(fb_ds.size());
-
- /* duration of individual test, in milliseconds */
- unsigned long duration_ms = 2000;
-
- printf("byte-wise memcpy from RAM to RAM...\n");
+ Bytewise_ram_test(Env &env, int id) : Test(env, id, brief)
{
- unsigned long transferred_kib = 0;
- unsigned long const start_ms = now_ms();
-
- for (; now_ms() - start_ms < duration_ms;) {
- memcpy(src_buf[0], src_buf[1], fb_ds.size());
- transferred_kib += fb_ds.size() / 1024;
+ unsigned kib = 0;
+ unsigned const start_ms = timer.elapsed_ms();
+ for (; timer.elapsed_ms() - start_ms < DURATION_MS;) {
+ memcpy(buf[0], buf[1], fb_ds.size());
+ kib += fb_ds.size() / 1024;
}
-
- unsigned long const end_ms = now_ms();
-
- printf("-> %ld MiB/sec\n",
- (transferred_kib)/(end_ms - start_ms));
+ conclusion(kib, start_ms, timer.elapsed_ms());
}
+};
- /*
- * Fill one memory buffer with white pixels.
- */
- memset(src_buf[1], ~0, fb_ds.size());
+struct Bytewise_fb_test : Test
+{
+ static constexpr char const *brief = "byte-wise memcpy from RAM to FB";
- printf("byte-wise memcpy from RAM to framebuffer...\n");
+ Bytewise_fb_test(Env &env, int id) : Test(env, id, brief)
{
- unsigned long transferred_kib = 0;
- unsigned long const start_ms = now_ms();
-
- for (unsigned i = 0; now_ms() - start_ms < duration_ms; i++) {
- memcpy(fb_ds.local_addr(), src_buf[i % 2], fb_ds.size());
- transferred_kib += fb_ds.size() / 1024;
+ unsigned kib = 0;
+ unsigned const start_ms = timer.elapsed_ms();
+ for (unsigned i = 0; timer.elapsed_ms() - start_ms < DURATION_MS; i++) {
+ memcpy(fb_ds.local_addr(), buf[i % 2], fb_ds.size());
+ kib += fb_ds.size() / 1024;
}
-
- unsigned long const end_ms = now_ms();
-
- printf("-> %ld MiB/sec\n",
- (transferred_kib)/(end_ms - start_ms));
+ conclusion(kib, start_ms, timer.elapsed_ms());
}
+};
- /*
- * Blitting via the blit library from RAM to framebuffer
- */
- printf("copy via blit library from RAM to framebuffer...\n");
+struct Blit_test : Test
+{
+ static constexpr char const *brief = "copy via blit library from RAM to FB";
+
+ Blit_test(Env &env, int id) : Test(env, id, brief)
{
- unsigned long transferred_kib = 0;
- unsigned long const start_ms = now_ms();
-
- /* line width in bytes */
- unsigned const w = fb_mode.width() * fb_mode.bytes_per_pixel();
- unsigned const h = fb_mode.height();
-
- for (unsigned i = 0; now_ms() - start_ms < duration_ms; i++) {
- blit(src_buf[i % 2], w, fb_ds.local_addr(), w, w, h);
-
- transferred_kib += (w*h) / 1024;
+ unsigned kib = 0;
+ unsigned const start_ms = timer.elapsed_ms();
+ unsigned const w = fb_mode.width() * fb_mode.bytes_per_pixel();
+ unsigned const h = fb_mode.height();
+ for (unsigned i = 0; timer.elapsed_ms() - start_ms < DURATION_MS; i++) {
+ blit(buf[i % 2], w, fb_ds.local_addr(), w, w, h);
+ kib += (w * h) / 1024;
}
-
- unsigned long const end_ms = now_ms();
-
- printf("-> %ld MiB/sec\n",
- (transferred_kib)/(end_ms - start_ms));
+ conclusion(kib, start_ms, timer.elapsed_ms());
}
+};
- /*
- * Unaligned blitting via the blit library from RAM to framebuffer
- */
- printf("unaligned copy via blit library from RAM to framebuffer...\n");
+struct Unaligned_blit_test : Test
+{
+ static constexpr char const *brief = "unaligned copy via blit library from RAM to FB";
+
+ Unaligned_blit_test(Env &env, int id) : Test(env, id, brief)
{
- unsigned long transferred_kib = 0;
- unsigned long const start_ms = now_ms();
-
- /* line width in bytes */
- unsigned const w = fb_mode.width() * fb_mode.bytes_per_pixel();
- unsigned const h = fb_mode.height();
-
- for (unsigned i = 0; now_ms() - start_ms < duration_ms; i++) {
- blit(src_buf[i % 2] + 2, w, fb_ds.local_addr() + 2, w, w - 2, h);
-
- transferred_kib += (w*h) / 1024;
+ unsigned kib = 0;
+ unsigned const start_ms = timer.elapsed_ms();
+ unsigned const w = fb_mode.width() * fb_mode.bytes_per_pixel();
+ unsigned const h = fb_mode.height();
+ for (unsigned i = 0; timer.elapsed_ms() - start_ms < DURATION_MS; i++) {
+ blit(buf[i % 2] + 2, w, fb_ds.local_addr() + 2, w, w - 2, h);
+ kib += (w * h) / 1024;
}
-
- unsigned long const end_ms = now_ms();
-
- printf("-> %ld MiB/sec\n",
- (transferred_kib)/(end_ms - start_ms));
+ conclusion(kib, start_ms, timer.elapsed_ms());
}
+};
- printf("--- test-fb_bench finished ---\n");
- return 0;
-}
+struct Main
+{
+ Constructible test_1;
+ Constructible test_2;
+ Constructible test_3;
+ Constructible test_4;
+
+ Main(Env &env)
+ {
+ log("--- Framebuffer benchmark ---");
+ test_1.construct(env, 1); test_1.destruct();
+ test_2.construct(env, 2); test_2.destruct();
+ test_3.construct(env, 3); test_3.destruct();
+ test_4.construct(env, 4); test_4.destruct();
+ log("--- Framebuffer benchmark finished ---");
+ }
+};
+
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/input/test.cc b/repos/os/src/test/input/test.cc
index 3534ac5e7d..88ee9eb4a4 100644
--- a/repos/os/src/test/input/test.cc
+++ b/repos/os/src/test/input/test.cc
@@ -5,22 +5,20 @@
*/
/*
- * Copyright (C) 2010-2016 Genode Labs GmbH
+ * Copyright (C) 2010-2017 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.
*/
+/* Genode includes */
#include
-#include
-#include
#include
#include
-#include
-
using namespace Genode;
+
static char const * ev_type(Input::Event::Type type)
{
switch (type) {
@@ -47,62 +45,42 @@ static char const * key_name(Input::Event const &ev)
}
-class Test_environment
+struct Main
{
- private:
+ Env &env;
+ Input::Connection input { env };
+ Signal_handler input_sigh { env.ep(), *this, &Main::handle_input };
+ unsigned event_cnt { 0 };
- Genode::Env &_env;
+ void handle_input();
- Input::Connection _input;
-
- Genode::Signal_handler _input_sigh;
-
- unsigned int event_count = 0;
-
- void _handle_input();
-
- public:
-
- Test_environment(Genode::Env &env)
- : _env(env),
- _input_sigh(env.ep(), *this, &Test_environment::_handle_input)
- {
- log("--- Input test is up ---");
-
- _input.sigh(_input_sigh);
- }
+ Main(Env &env) : env(env)
+ {
+ log("--- Input test ---");
+ input.sigh(input_sigh);
+ }
};
-void Test_environment::_handle_input()
+void Main::handle_input()
{
- /*
- * Handle input events
- */
int key_cnt = 0;
-
- _input.for_each_event([&] (Input::Event const &ev) {
- event_count++;
+ input.for_each_event([&] (Input::Event const &ev) {
+ event_cnt++;
if (ev.type() == Input::Event::PRESS) key_cnt++;
if (ev.type() == Input::Event::RELEASE) key_cnt--;
- /* log event */
- log("Input event #", event_count, "\t"
+ log("Input event #", event_cnt, "\t"
"type=", ev_type(ev.type()), "\t"
"code=", ev.code(), "\t"
"rx=", ev.rx(), "\t"
"ry=", ev.ry(), "\t"
"ax=", ev.ax(), "\t"
"ay=", ev.ay(), "\t"
- "key_cnt=", key_cnt, "\t", key_name(ev));
+ "key_cnt=", key_cnt, "\t", key_name(ev));
});
}
-void Component::construct(Genode::Env &env)
-{
- using namespace Genode;
- log("--- Test input ---\n");
- static Test_environment te(env);
-}
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/nic_loopback/main.cc b/repos/os/src/test/nic_loopback/main.cc
index 49329f3948..174a9ad2ac 100644
--- a/repos/os/src/test/nic_loopback/main.cc
+++ b/repos/os/src/test/nic_loopback/main.cc
@@ -61,7 +61,7 @@ struct Test::Base
enum { BUF_SIZE = Nic::Packet_allocator::DEFAULT_PACKET_SIZE * 128 };
- Nic::Connection _nic { &_tx_block_alloc, BUF_SIZE, BUF_SIZE };
+ Nic::Connection _nic { _env, &_tx_block_alloc, BUF_SIZE, BUF_SIZE };
void _handle_nic() { if (!_done) handle_nic(); }
diff --git a/repos/os/src/test/ram_fs_chunk/main.cc b/repos/os/src/test/ram_fs_chunk/main.cc
index a458fa3d9b..afe410479c 100644
--- a/repos/os/src/test/ram_fs_chunk/main.cc
+++ b/repos/os/src/test/ram_fs_chunk/main.cc
@@ -1,152 +1,137 @@
/*
* \brief Unit test for RAM fs chunk data structure
* \author Norman Feske
+ * \author Martin Stein
* \date 2012-04-19
*/
+/*
+ * Copyright (C) 2012-2017 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.
+ */
+
/* Genode includes */
-#include
-#include
+#include
+#include
#include
-namespace File_system {
- typedef Chunk<2> Chunk_level_3;
- typedef Chunk_index<3, Chunk_level_3> Chunk_level_2;
- typedef Chunk_index<4, Chunk_level_2> Chunk_level_1;
- typedef Chunk_index<5, Chunk_level_1> Chunk_level_0;
-}
+using namespace File_system;
+using namespace Genode;
+using Chunk_level_3 = Chunk<2>;
+using Chunk_level_2 = Chunk_index<3, Chunk_level_3>;
+using Chunk_level_1 = Chunk_index<4, Chunk_level_2>;
-namespace Genode {
+struct Chunk_level_0 : Chunk_index<5, Chunk_level_1>
+{
+ Chunk_level_0(Allocator &alloc, seek_off_t off) : Chunk_index(alloc, off) { }
- struct Allocator_tracer : Allocator
+ void print(Output &out) const
{
- struct Alloc
- {
- using Id = Id_space::Id;
+ static char read_buf[Chunk_level_0::SIZE];
+ if (used_size() > Chunk_level_0::SIZE) {
+ throw Index_out_of_range(); }
- Id_space::Element id_space_elem;
- size_t size;
-
- Alloc(Id_space &space, Id id, size_t size)
- : id_space_elem(*this, space, id), size(size) { }
- };
-
- Id_space _allocs;
- size_t _sum;
- Allocator &_wrapped;
-
- Allocator_tracer(Allocator &wrapped) : _sum(0), _wrapped(wrapped) { }
-
- size_t sum() const { return _sum; }
-
- bool alloc(size_t size, void **out_addr)
- {
- _sum += size;
- bool result = _wrapped.alloc(size, out_addr);
- new (_wrapped) Alloc(_allocs, Alloc::Id { (addr_t)*out_addr }, size);
- return result;
+ read(read_buf, used_size(), 0);
+ Genode::print(out, "content (size=", used_size(), "): ");
+ Genode::print(out, "\"");
+ for (unsigned i = 0; i < used_size(); i++) {
+ char const c = read_buf[i];
+ if (c) {
+ Genode::print(out, Char(c)); }
+ else {
+ Genode::print(out, "."); }
}
-
- void free(void *addr, size_t size)
- {
- _allocs.apply(Alloc::Id { (addr_t)addr }, [&] (Alloc &alloc) {
- _sum -= alloc.size;
- destroy(_wrapped, &alloc);
- _wrapped.free(addr, size);
- });
- }
-
- size_t overhead(size_t size) const override { return _wrapped.overhead(size); }
- bool need_size_for_free() const override { return _wrapped.need_size_for_free(); }
- };
+ Genode::print(out, "\"");
+ }
};
-
-namespace Genode {
-
- /**
- * Helper for the formatted output of a chunk state
- */
- static inline void print(Output &out, File_system::Chunk_level_0 const &chunk)
+struct Allocator_tracer : Allocator
+{
+ struct Alloc
{
- using namespace File_system;
-
- static char read_buf[Chunk_level_0::SIZE];
-
- size_t used_size = chunk.used_size();
-
- struct File_size_out_of_bounds { };
- if (used_size > Chunk_level_0::SIZE)
- throw File_size_out_of_bounds();
-
- chunk.read(read_buf, used_size, 0);
-
- Genode::print(out, "content (size=", used_size, "): ");
+ using Id = Id_space::Id;
- Genode::print(out, "\"");
- for (unsigned i = 0; i < used_size; i++) {
- char const c = read_buf[i];
- if (c)
- Genode::print(out, Genode::Char(c));
- else
- Genode::print(out, ".");
+ Id_space::Element id_space_elem;
+ size_t size;
+
+ Alloc(Id_space &space, Id id, size_t size)
+ : id_space_elem(*this, space, id), size(size) { }
+ };
+
+ Id_space allocs;
+ size_t sum { 0 };
+ Allocator &wrapped;
+
+ Allocator_tracer(Allocator &wrapped) : wrapped(wrapped) { }
+
+ bool alloc(size_t size, void **out_addr) override
+ {
+ sum += size;
+ bool result = wrapped.alloc(size, out_addr);
+ new (wrapped) Alloc(allocs, Alloc::Id { (addr_t)*out_addr }, size);
+ return result;
+ }
+
+ void free(void *addr, size_t size) override
+ {
+ allocs.apply(Alloc::Id { (addr_t)addr }, [&] (Alloc &alloc) {
+ sum -= alloc.size;
+ destroy(wrapped, &alloc);
+ wrapped.free(addr, size);
+ });
+ }
+
+ size_t overhead(size_t size) const override { return wrapped.overhead(size); }
+ bool need_size_for_free() const override { return wrapped.need_size_for_free(); }
+};
+
+struct Main
+{
+ Env &env;
+ Heap heap { env.ram(), env.rm() };
+ Allocator_tracer alloc { heap };
+
+ Main(Env &env) : env(env)
+ {
+ log("--- RAM filesystem chunk test ---");
+ log("chunk sizes");
+ log(" level 0: payload=", (int)Chunk_level_0::SIZE, " sizeof=", sizeof(Chunk_level_0));
+ log(" level 1: payload=", (int)Chunk_level_1::SIZE, " sizeof=", sizeof(Chunk_level_1));
+ log(" level 2: payload=", (int)Chunk_level_2::SIZE, " sizeof=", sizeof(Chunk_level_2));
+ log(" level 3: payload=", (int)Chunk_level_3::SIZE, " sizeof=", sizeof(Chunk_level_3));
+ {
+ Chunk_level_0 chunk(alloc, 0);
+ write(chunk, "five-o-one", 0);
+
+ /* overwrite part of the file */
+ write(chunk, "five", 7);
+
+ /* write to position beyond current file length */
+ write(chunk, "Nuance", 17);
+ write(chunk, "YM-2149", 35);
+
+ truncate(chunk, 30);
+ for (unsigned i = 29; i > 0; i--)
+ truncate(chunk, i);
}
- Genode::print(out, "\"");
+ log("allocator: sum=", alloc.sum);
+ log("--- RAM filesystem chunk test finished ---");
}
-}
-
-
-static void write(File_system::Chunk_level_0 &chunk,
- char const *str, Genode::off_t seek_offset)
-{
- chunk.write(str, Genode::strlen(str), seek_offset);
- Genode::log("write \"", str, "\" at offset ", seek_offset, " -> ", chunk);
-}
-
-
-static void truncate(File_system::Chunk_level_0 &chunk,
- File_system::file_size_t size)
-{
- chunk.truncate(size);
- Genode::log("trunc(", size, ") -> ", chunk);
-}
-
-
-int main(int, char **)
-{
- using namespace File_system;
- using namespace Genode;
-
- log("--- ram_fs_chunk test ---");
-
- log("chunk sizes");
- log(" level 0: payload=", (int)Chunk_level_0::SIZE, " sizeof=", sizeof(Chunk_level_0));
- log(" level 1: payload=", (int)Chunk_level_1::SIZE, " sizeof=", sizeof(Chunk_level_1));
- log(" level 2: payload=", (int)Chunk_level_2::SIZE, " sizeof=", sizeof(Chunk_level_2));
- log(" level 3: payload=", (int)Chunk_level_3::SIZE, " sizeof=", sizeof(Chunk_level_3));
-
- static Allocator_tracer alloc(*env()->heap());
+ void write(Chunk_level_0 &chunk, char const *str, off_t seek_offset)
{
- Chunk_level_0 chunk(alloc, 0);
-
- write(chunk, "five-o-one", 0);
-
- /* overwrite part of the file */
- write(chunk, "five", 7);
-
- /* write to position beyond current file length */
- write(chunk, "Nuance", 17);
- write(chunk, "YM-2149", 35);
-
- truncate(chunk, 30);
-
- for (unsigned i = 29; i > 0; i--)
- truncate(chunk, i);
+ chunk.write(str, strlen(str), seek_offset);
+ log("write \"", str, "\" at offset ", seek_offset, " -> ", chunk);
}
- log("allocator: sum=", alloc.sum());
+ void truncate(Chunk_level_0 &chunk, file_size_t size)
+ {
+ chunk.truncate(size);
+ log("trunc(", size, ") -> ", chunk);
+ }
+};
- return 0;
-}
+void Component::construct(Env &env) { struct Main main(env); }
diff --git a/repos/os/src/test/reconstructible/main.cc b/repos/os/src/test/reconstructible/main.cc
index fc831b719e..ca0c404c57 100644
--- a/repos/os/src/test/reconstructible/main.cc
+++ b/repos/os/src/test/reconstructible/main.cc
@@ -13,26 +13,16 @@
/* Genode includes */
#include
-#include
-
-using Genode::Reconstructible;
-using Genode::Constructible;
-using Genode::log;
+#include
+using namespace Genode;
struct Object
{
unsigned const id;
- Object(unsigned id) : id(id)
- {
- log("construct Object ", id);
- }
-
- ~Object()
- {
- log("destruct Object ", id);
- }
+ Object(unsigned id) : id(id) { log("construct Object ", id); }
+ ~Object() { log("destruct Object ", id); }
void method() { log("method called on Object ", id); }
void const_method() const { log("const method called on Object ", id); }
@@ -41,38 +31,25 @@ struct Object
struct Member_with_reference
{
- Object &reference;
+ Object &reference;
+ int const c = 13;
- const int c = 13;
+ Member_with_reference(Object &reference) : reference(reference) {
+ log("construct Member_with_reference"); }
- Member_with_reference(Object &reference) : reference(reference)
- {
- log("construct Member_with_reference");
- }
-
- ~Member_with_reference()
- {
- log("destruct Member_with_reference");
- }
+ ~Member_with_reference() { log("destruct Member_with_reference"); }
};
struct Compound
{
- Reconstructible member;
- Constructible lazy_member;
+ Reconstructible member;
+ Constructible lazy_member;
- Compound(Object &object)
- :
- member(object)
- {
- log("construct Compound");
- }
+ Compound(Object &object) : member(object) {
+ log("construct Compound"); }
- ~Compound()
- {
- log("destruct Compound");
- }
+ ~Compound() { log("destruct Compound"); }
};
@@ -94,79 +71,68 @@ struct Throwing
log("construct Throwing -> throw exception");
throw -1;
} else {
- log("construct Throwing -> don't throw");
- }
+ log("construct Throwing -> don't throw"); }
}
- virtual ~Throwing()
- {
- log("destruct Throwing");
- }
+ virtual ~Throwing() { log("destruct Throwing"); }
};
-static void call_const_method(Compound const &compound)
+struct Main
{
- compound.member->reference.const_method();
-}
-
-
-int main(int, char **)
-{
- using namespace Genode;
-
- log("--- test-reconstructible started ---");
+ void call_const_method(Compound const &compound) {
+ compound.member->reference.const_method(); }
+ Main(Env &env)
{
- Object object_1(1);
- Object object_2(2);
+ log("--- Reconstructible utility test ---");
+ {
+ Object object_1(1);
+ Object object_2(2);
- log("-- create Compound object --");
- Compound compound(object_1);
+ log("create Compound object");
+ Compound compound(object_1);
- log("compound.member.constructed returns ",
- compound.member.constructed());
- log("compound.lazy_member.constructed returns ",
- compound.lazy_member.constructed());
+ log("compound.member.constructed returns ",
+ compound.member.constructed());
+ log("compound.lazy_member.constructed returns ",
+ compound.lazy_member.constructed());
- log("-- construct lazy member --");
- compound.lazy_member.construct(object_2);
- log("compound.lazy_member.constructed returns ",
- compound.lazy_member.constructed());
+ log("construct lazy member");
+ compound.lazy_member.construct(object_2);
+ log("compound.lazy_member.constructed returns ",
+ compound.lazy_member.constructed());
- log("-- call method on member (with reference to Object 1) --");
- call_const_method(compound);
+ log("call method on member (with reference to Object 1)");
+ call_const_method(compound);
- log("-- reconstruct member with Object 2 as reference --");
- compound.member.construct(object_2);
+ log("reconstruct member with Object 2 as reference");
+ compound.member.construct(object_2);
- log("-- call method on member --");
- call_const_method(compound);
+ log("call method on member");
+ call_const_method(compound);
- log("-- destruct member --");
- compound.member.destruct();
+ log("destruct member");
+ compound.member.destruct();
- log("-- try to call method on member, catch exception --");
+ log("try to call method on member, catch exception");
+ try { call_const_method(compound); }
+ catch (typename Reconstructible::Deref_unconstructed_object) {
+ log("got exception, as expected"); }
+
+ log("destruct Compound and Objects 1 and 2");
+ }
try {
- call_const_method(compound); }
- catch (typename Reconstructible::Deref_unconstructed_object) {
+ log("construct Throwing object");
+ Bool const b_false(false), b_true(true);
+ Reconstructible inst(b_false);
+ inst.construct(b_true);
+ Genode::error("expected contructor to throw");
+ } catch (int i) {
log("got exception, as expected"); }
- log("-- destruct Compound and Objects 1 and 2 --");
+ log("--- Reconstructible utility test finished ---");
}
+};
- try {
- log("-- construct Throwing object");
- Bool const b_false(false), b_true(true);
-
- Reconstructible inst(b_false);
- inst.construct(b_true);
- Genode::error("expected contructor to throw");
- } catch (int i) {
- log("-- catched exception as expected");
- }
-
- log("--- test-reconstructible finished ---");
-
- return 0;
-}
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/rom_blk/main.cc b/repos/os/src/test/rom_blk/main.cc
index 0c527bf067..bac814d429 100644
--- a/repos/os/src/test/rom_blk/main.cc
+++ b/repos/os/src/test/rom_blk/main.cc
@@ -1,6 +1,7 @@
/*
- * \brief Rom-file to block-session client test implementation
+ * \brief ROM-file to block-session client test implementation
* \author Stefan Kalkowski
+ * \author Martin Stein
* \date 2010-07-07
*
* The test program compares the values delivered by the block-service,
@@ -8,120 +9,72 @@
*/
/*
- * Copyright (C) 2010-2013 Genode Labs GmbH
+ * Copyright (C) 2010-2017 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.
*/
+/* Genode includes */
#include
-#include
-#include
-#include
-#include
-#include
+#include
+#include
#include
-#include
+#include
+using namespace Genode;
-class Comparer : public Genode::Thread_deprecated<8192>
+struct Main
{
- private:
+ enum { REQ_PARALLEL = 10 };
- Block::Connection _blk_con;
- Genode::Rom_connection _rom;
- Genode::addr_t _addr;
+ using File_name = String<64>;
+ using Packet_descriptor = Block::Packet_descriptor;
+ struct Files_differ : Exception { };
+ struct Device_not_readable : Exception { };
+ struct Read_request_failed : Exception { };
- class Block_file_differ : Genode::Exception {};
+ Env &env;
+ Attached_rom_dataspace config { env, "config" };
+ File_name file_name { config.xml().attribute_value("file", File_name()) };
+ Heap heap { env.ram(), env.rm() };
+ Allocator_avl block_alloc { &heap };
+ Block::Connection block { env, &block_alloc };
+ Attached_rom_dataspace rom { env, file_name.string() };
- public:
+ Main(Env &env) : env(env)
+ {
+ log("--- ROM Block test ---");
- enum {
- BLOCK_REQ_PARALLEL = 10 /* number of blocks to handle per block-request */
- };
+ Block::Session::Tx::Source &src = *block.tx();
+ size_t blk_sz;
+ Block::sector_t blk_cnt;
+ Block::Session::Operations ops;
- Comparer(Genode::Allocator_avl *block_alloc,
- const char* filename)
- : Thread_deprecated("comparer"), _blk_con(block_alloc), _rom(filename),
- _addr(Genode::env()->rm_session()->attach(_rom.dataspace())) { }
+ block.info(&blk_cnt, &blk_sz, &ops);
+ if (!ops.supported(Packet_descriptor::READ)) {
+ throw Device_not_readable(); }
- void entry()
- {
- using namespace Genode;
+ log("We have ", blk_cnt, " blocks with a size of ", blk_sz, " bytes");
+ for (size_t i = 0; i < blk_cnt; i += REQ_PARALLEL) {
+ size_t cnt = (blk_cnt - i > REQ_PARALLEL) ? REQ_PARALLEL : blk_cnt - i;
+ Packet_descriptor pkt(src.alloc_packet(cnt * blk_sz),
+ Packet_descriptor::READ, i, cnt);
- Block::Session::Tx::Source *source = _blk_con.tx();
- size_t blk_size = 0;
- Block::sector_t blk_cnt = 0;
- Genode::addr_t end =
- _addr + Dataspace_client(_rom.dataspace()).size();
- Block::Session::Operations ops;
- _blk_con.info(&blk_cnt, &blk_size, &ops);
+ log("Check blocks ", i, "..", i + cnt - 1);
+ src.submit_packet(pkt);
+ pkt = src.get_acked_packet();
+ if (!pkt.succeeded()) {
+ throw Read_request_failed(); }
- if (!ops.supported(Block::Packet_descriptor::READ)) {
- error("Block device not readable!");
- }
+ char const *rom_src = rom.local_addr() + i * blk_sz;
+ if (strcmp(rom_src, src.packet_content(pkt), rom.size())) {
+ throw Files_differ(); }
- log("We have ", blk_cnt, " blocks with a "
- "size of ", Hex(blk_size), " bytes");
-
- for (size_t i = 0; i < blk_cnt; i += BLOCK_REQ_PARALLEL) {
- try {
- size_t cnt = (blk_cnt - i > BLOCK_REQ_PARALLEL)
- ? BLOCK_REQ_PARALLEL : blk_cnt - i;
- Block::Packet_descriptor p(source->alloc_packet(cnt * blk_size),
- Block::Packet_descriptor::READ, i, cnt);
-
- source->submit_packet(p);
- p = source->get_acked_packet();
-
- if (!p.succeeded()) {
- error("could not read block ", Hex(i), "-", Hex(i + cnt));
- return;
- }
-
- char* blk_src = source->packet_content(p);
- char* rom_src = (char*) _addr + i * blk_size;
- bool differ = false;
- for (size_t j = 0; j < cnt; j++)
- for (size_t k = 0; k < blk_size; k++) {
- if (&rom_src[j*blk_size+k] >= (char*)end) {
- error("end of image file reached!");
- return;
- }
- if (blk_src[j*blk_size+k] != rom_src[j*blk_size+k])
- differ = true;
- }
- if (differ) {
- warning("block ", i, " differs!");
- throw Block_file_differ();
- }
- source->release_packet(p);
- } catch (Block::Session::Tx::Source::Packet_alloc_failed) {
- error("Mmh, strange we run out of packets");
- return;
- }
- }
- log("all done, finished!");
+ src.release_packet(pkt);
}
+ log("--- ROM Block test finished ---");
+ }
};
-
-int main(int argc, char **argv)
-{
- using namespace Genode;
-
- log("--- Block session test ---");
-
- try {
- static char filename[64];
- config()->xml_node().attribute("file").value(filename, sizeof(filename));
- Allocator_avl block_alloc(env()->heap());
- Comparer th(&block_alloc, filename);
- th.start();
- sleep_forever();
- } catch (Rom_connection::Rom_connection_failed) {
- error("config file or file given by tag is missing.");
- }
- log("An error occured, exit now ...");
- return -1;
-}
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/rtc/main.cc b/repos/os/src/test/rtc/main.cc
index 0b00e88e65..66506e29e1 100644
--- a/repos/os/src/test/rtc/main.cc
+++ b/repos/os/src/test/rtc/main.cc
@@ -13,11 +13,11 @@
#include
#include
-#include
-#include
#include
#include
+using namespace Genode;
+
struct Main
{
Main(Genode::Env &env)
@@ -31,9 +31,8 @@ struct Main
for (unsigned i = 0; i < 4; ++i) {
Rtc::Timestamp now = rtc.current_time();
- Genode::printf("RTC: %u-%02u-%02u %02u:%02u:%02u\n",
- now.year, now.month, now.day,
- now.hour, now.minute, now.second);
+ log("RTC: ", now.year, "-", now.month, "-", now.day, " ",
+ now.hour, ":", now.minute, ":", now.second);
timer.msleep(1000);
}
diff --git a/repos/os/src/test/sd_card_bench/main.cc b/repos/os/src/test/sd_card_bench/main.cc
index 46ab6340ec..1ec6f272fc 100644
--- a/repos/os/src/test/sd_card_bench/main.cc
+++ b/repos/os/src/test/sd_card_bench/main.cc
@@ -16,7 +16,7 @@
#include
#include
#include
-#include
+#include
/* local includes */
#include
@@ -51,14 +51,14 @@ struct Main
Env &env;
Packet_descriptor pkt;
unsigned long time_before_ms;
- Timer::Connection timer;
+ Timer::Connection timer { env };
Operation operation { READ };
Signal_handler ack_handler { env.ep(), *this, &Main::update_state };
Driver_session drv_session { ack_handler };
Sd_card::Driver drv { env };
- size_t const buf_size_kib { config()->xml_node()
- .attribute_value("buffer_size_kib",
- (size_t)0) };
+ size_t const buf_size_kib { Attached_rom_dataspace(env, "config")
+ .xml().attribute_value("buffer_size_kib",
+ (size_t)0) };
size_t const buf_size { buf_size_kib * 1024 };
Attached_ram_dataspace buf { &env.ram(), buf_size, UNCACHED };
char *buf_virt { buf.local_addr() };
diff --git a/repos/os/src/test/signal/main.cc b/repos/os/src/test/signal/main.cc
index babe01d30b..c739ef0e80 100644
--- a/repos/os/src/test/signal/main.cc
+++ b/repos/os/src/test/signal/main.cc
@@ -1,142 +1,100 @@
/*
* \brief Test for signalling framework
* \author Norman Feske
+ * \author Martin Stein
* \date 2008-09-06
*/
/*
- * Copyright (C) 2008-2013 Genode Labs GmbH
+ * Copyright (C) 2008-2017 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.
*/
+/* Genode includes */
#include
#include
-#include
-#include
-#include
#include
+#include
#include
-#include
using namespace Genode;
-
/**
- * Transmit signals in a periodic fashion
+ * A thread that submits a signal context in a periodic fashion
*/
class Sender : Thread
{
private:
- Timer::Connection _timer; /* timer connection for local use */
+ Timer::Connection _timer;
Signal_transmitter _transmitter;
- unsigned const _interval_ms; /* interval between signals in milliseconds */
- bool _stop; /* state for destruction protocol */
- unsigned _submit_cnt; /* statistics */
- bool _idle; /* suppress the submission of signals */
- bool const _verbose; /* print activities */
+ unsigned const _interval_ms;
+ bool const _verbose;
+ bool _stop { false };
+ unsigned _submit_cnt { 0 };
+ bool _idle { false };
- /**
- * Sender thread submits signals every '_interval_ms' milliseconds
- */
void entry()
{
while (!_stop) {
-
if (!_idle) {
_submit_cnt++;
-
- if (_verbose)
- log("submit signal ", _submit_cnt);
+ if (_verbose) {
+ log("submit signal ", _submit_cnt); }
_transmitter.submit();
-
- if (_interval_ms)
- _timer.msleep(_interval_ms);
- } else
- _timer.msleep(100);
+ if (_interval_ms) {
+ _timer.msleep(_interval_ms); }
+ } else {
+ _timer.msleep(100); }
}
}
public:
- /**
- * Constructor
- *
- * \param context signal destination
- * \param interval_ms interval between signals
- * \param verbose print status information
- */
- Sender(Env &env, Signal_context_capability context,
- unsigned interval_ms, bool verbose = true)
+ Sender(Env &env,
+ Signal_context_capability context,
+ unsigned interval_ms,
+ bool verbose)
:
- Thread(env, "sender", 4096*sizeof(long)),
- _timer(env),
- _transmitter(context),
- _interval_ms(interval_ms),
- _stop(false),
- _submit_cnt(0),
- _idle(0),
- _verbose(verbose)
+ Thread(env, "sender", 4096 * sizeof(addr_t)), _timer(env),
+ _transmitter(context), _interval_ms(interval_ms), _verbose(verbose)
{
- /* start thread at 'entry' function */
- start();
+ Thread::start();
}
- /**
- * Destructor
- */
- ~Sender()
- {
- /* tell thread to stop iterating */
- _stop = true;
+ /***************
+ ** Accessors **
+ ***************/
- /* wait for current 'msleep' call of the thread to finish */
- _timer.msleep(0);
- }
-
- /**
- * Suppress the transmission of further signals
- */
- void idle(bool idle = true) { _idle = idle; }
-
- /**
- * Return total number of submitted notifications
- */
- unsigned submit_cnt() { return _submit_cnt; }
+ void idle(bool idle) { _idle = idle; }
+ unsigned submit_cnt() const { return _submit_cnt; }
};
-
/**
- * Signal handler receives signals and takes some time to handle each
+ * A thread that receives signals and takes some time to handle each
*/
class Handler : Thread
{
private:
- Timer::Connection _timer; /* timer connection for local use */
- unsigned const _dispatch_ms; /* time needed for dispatching a signal */
- unsigned const _id; /* unique ID of signal handler for debug output */
- static unsigned _id_cnt; /* counter for producing unique IDs */
- Signal_receiver &_receiver; /* signal endpoint */
- bool _stop; /* state for destruction protocol */
- unsigned _receive_cnt; /* number of received notifications */
- unsigned _activation_cnt; /* number of invocations of the signal handler */
- bool _idle; /* suppress the further handling of signals */
- bool const _verbose; /* print status information */
+ Timer::Connection _timer;
+ unsigned const _dispatch_ms;
+ unsigned const _id;
+ bool const _verbose;
+ Signal_receiver &_receiver;
+ bool _stop { false };
+ unsigned _receive_cnt { 0 };
+ unsigned _activation_cnt { 0 };
+ bool _idle { false };
- /**
- * Signal handler needs '_dispatch_ms' milliseconds for each signal
- */
void entry()
{
while (!_stop) {
-
if (!_idle) {
Signal signal = _receiver.wait_for_signal();
-
if (_verbose)
log("handler ", _id, " got ", signal.num(), " "
"signal", (signal.num() == 1 ? "" : "s"), " "
@@ -145,7 +103,6 @@ class Handler : Thread
_receive_cnt += signal.num();
_activation_cnt++;
}
-
if (_dispatch_ms)
_timer.msleep(_dispatch_ms);
}
@@ -153,486 +110,351 @@ class Handler : Thread
public:
- /**
- * Constructor
- *
- * \param receiver receiver to request signals from
- * \param dispatch_ms duration of signal-handler activity
- * \param verbose print status information
- */
- Handler(Env &env, Signal_receiver &receiver, unsigned dispatch_ms, bool verbose = true)
+ Handler(Env &env,
+ Signal_receiver &receiver,
+ unsigned dispatch_ms,
+ bool verbose,
+ unsigned id)
:
- Thread(env, "handler", 4096*sizeof(long)),
- _timer(env),
- _dispatch_ms(dispatch_ms),
- _id(++_id_cnt),
- _receiver(receiver),
- _stop(false),
- _receive_cnt(0),
- _activation_cnt(0),
- _idle(0),
- _verbose(verbose)
+ Thread(env, "handler", 4096 * sizeof(addr_t)), _timer(env),
+ _dispatch_ms(dispatch_ms), _id(id), _verbose(verbose),
+ _receiver(receiver)
{
- start();
+ Thread::start();
}
- /**
- * Destructor
- */
- ~Handler()
- {
- /* tell thread to stop iterating */
- _stop = true;
+ void print(Output &output) const { Genode::print(output, "handler ", _id); }
- /* wait for current 'msleep' call of the thread to finish */
- _timer.msleep(0);
- }
+ /***************
+ ** Accessors **
+ ***************/
- /**
- * Suppress the handling of further signals
- */
- void idle(bool idle = true) { _idle = idle; }
-
- /**
- * Return total number of received notifications
- */
- unsigned receive_cnt() const { return _receive_cnt; }
-
- /**
- * Return total number of signal-handler activations
- */
+ void idle(bool idle) { _idle = idle; }
+ unsigned receive_cnt() const { return _receive_cnt; }
unsigned activation_cnt() const { return _activation_cnt; }
};
-
/**
- * Counter for generating unique signal-handler IDs
+ * Base of all signalling tests
*/
-unsigned Handler::_id_cnt = 0;
-
-
-/**
- * Counter for enumerating the tests
- */
-static unsigned test_cnt = 0;
-
-
-/**
- * Symbolic error codes
- */
-class Test_failed { };
-class Test_failed_with_unequal_sent_and_received_signals : public Test_failed { };
-class Test_failed_with_unequal_activation_of_handlers : public Test_failed { };
-
-
-class Id_signal_context : public Signal_context
+struct Signal_test
{
- private:
+ enum { SPEED = 10 };
- int _id;
+ int id;
- public:
+ Signal_test(int id, char const *brief) : id(id) {
+ log("\nTEST ", id, ": ", brief, "\n"); }
- Id_signal_context(int id) : _id(id) { }
-
- int id() const { return _id; }
+ ~Signal_test() { log("\nTEST ", id, " finished\n"); }
};
-
-/**
- * Test for reliable notification delivery
- *
- * For this test, the produce more notification than that can be handled.
- * Still, the total number of notifications gets transmitted because of the
- * batching of notifications in one signal. This test fails if the number of
- * submitted notifications on the sender side does not match the number of
- * notifications received at the signal handler.
- */
-static void fast_sender_test(Env &env)
+struct Fast_sender_test : Signal_test
{
- enum { SPEED = 10 };
- enum { TEST_DURATION = 50*SPEED };
- enum { HANDLER_INTERVAL = 10*SPEED };
- enum { SENDER_INTERVAL = 2*SPEED };
- enum { FINISH_IDLE_TIME = 2*HANDLER_INTERVAL };
+ static constexpr char const *brief =
+ "reliable delivery if the sender is faster than the handlers";
- log("");
- log("TEST ", ++test_cnt, ": one sender, one handler, sender is faster than handler");
- log("");
+ enum { HANDLER_INTERVAL_MS = 10 * SPEED,
+ SENDER_INTERVAL_MS = 2 * SPEED,
+ DURATION_MS = 50 * SPEED,
+ FINISH_IDLE_MS = 2 * HANDLER_INTERVAL_MS };
- Signal_receiver receiver;
- Id_signal_context context_123(123);
+ struct Unequal_sent_and_received_signals : Exception { };
- Heap heap(env.ram(), env.rm());
- Timer::Connection timer(env);
-
- Handler *handler = new (heap) Handler(env, receiver, HANDLER_INTERVAL, false);
- Sender *sender = new (heap) Sender(env, receiver.manage(&context_123),
- SENDER_INTERVAL, false);
-
- timer.msleep(TEST_DURATION);
-
- /* stop emitting signals */
- log("deactivate sender");
- sender->idle();
- timer.msleep(FINISH_IDLE_TIME);
-
- log("");
- log("sender submitted a total of ", sender->submit_cnt(), " signals");
- log("handler received a total of ", handler->receive_cnt(), " signals");
- log("");
-
- if (sender->submit_cnt() != handler->receive_cnt())
- throw Test_failed();
-
- receiver.dissolve(&context_123);
-
- destroy(heap, sender);
- destroy(heap, handler);
-
- log("TEST ", test_cnt, " FINISHED");
-}
-
-
-/**
- * Fairness test if multiple signal-handler threads are present at one receiver
- *
- * We expect that all handler threads get activated in a fair manner. The test
- * fails if the number of activations per handler differs by more than one.
- * Furthermore, if operating in non-descrete mode, the total number of sent and
- * handled notifications is checked.
- */
-static void multiple_handlers_test(Env &env)
-{
- enum { SPEED = 10 };
- enum { TEST_DURATION = 50*SPEED };
- enum { HANDLER_INTERVAL = 8*SPEED };
- enum { SENDER_INTERVAL = 1*SPEED };
- enum { FINISH_IDLE_TIME = 2*HANDLER_INTERVAL };
- enum { NUM_HANDLERS = 4 };
-
- log("");
- log("TEST ", ++test_cnt, ": one busy sender, ", (int)NUM_HANDLERS, " handlers");
- log("");
-
- Heap heap(env.ram(), env.rm());
- Timer::Connection timer(env);
- Signal_receiver receiver;
-
- Handler *handler[NUM_HANDLERS];
- for (int i = 0; i < NUM_HANDLERS; i++)
- handler[i] = new (heap) Handler(env, receiver, HANDLER_INTERVAL);
-
- Id_signal_context context_123(123);
- Sender *sender = new (heap) Sender(env, receiver.manage(&context_123), SENDER_INTERVAL);
-
- timer.msleep(TEST_DURATION);
-
- /* stop emitting signals */
- log("stop generating new notifications");
- sender->idle();
- timer.msleep(FINISH_IDLE_TIME);
-
- /* let handlers settle down */
- for (int i = 0; i < NUM_HANDLERS; i++)
- handler[i]->idle();
- timer.msleep(FINISH_IDLE_TIME);
-
- /* print signal delivery statistics */
- log("");
- log("sender submitted a total of ", sender->submit_cnt(), " signals");
- unsigned total_receive_cnt = 0;
- for (int i = 0; i < NUM_HANDLERS; i++) {
- log("handler ", i, " "
- "received a total of ", handler[i]->receive_cnt(), " signals");
- total_receive_cnt += handler[i]->receive_cnt();
- }
- log("all handlers received a total of ", total_receive_cnt, " signals");
-
- /* check if number of sent notifications match the received ones */
- if (sender->submit_cnt() != total_receive_cnt)
- throw Test_failed_with_unequal_sent_and_received_signals();
-
- /* print activation statistics */
- log("");
- for (int i = 0; i < NUM_HANDLERS; i++)
- log("handler ", i, " was activated ", handler[i]->activation_cnt(), " times");
- log("");
-
- /* check if handlers had been activated equally (tolerating a difference of one) */
- for (int i = 0; i < NUM_HANDLERS; i++) {
-
- int diff = handler[0]->activation_cnt()
- - handler[(i + 1)/NUM_HANDLERS]->activation_cnt();
-
- if (abs(diff) > 1)
- throw Test_failed_with_unequal_activation_of_handlers();
- }
-
- /* cleanup */
- receiver.dissolve(&context_123);
- destroy(heap, sender);
- for (int i = 0; i < NUM_HANDLERS; i++)
- destroy(heap, handler[i]);
-
- log("TEST ", test_cnt, " FINISHED");
-}
-
-
-/**
- * Stress test to estimate signal throughput
- *
- * For this test, we disable status output and any simulated wait times.
- * We produce and handle notifications as fast as possible via spinning
- * loops at the sender and handler side.
- */
-static void stress_test(Env &env)
-{
- enum { SPEED = 10 };
- enum { DURATION_SECONDS = 5 };
- enum { FINISH_IDLE_TIME = 100*SPEED };
-
- log("");
- log("TEST ", ++test_cnt, ": stress test, busy signal transmission and handling");
- log("");
-
- Timer::Connection timer(env);
- Heap heap(env.ram(), env.rm());
- Signal_receiver receiver;
- Id_signal_context context_123(123);
-
- Handler *handler = new (heap) Handler(env, receiver, 0, false);
- Sender *sender = new (heap) Sender(env, receiver.manage(&context_123),
- 0, false);
-
- for (int i = 1; i <= DURATION_SECONDS; i++) {
- log(i, "/", (int)DURATION_SECONDS);
- timer.msleep(1000);
- }
-
- /* stop emitting signals */
- log("deactivate sender");
- sender->idle();
-
- while (handler->receive_cnt() < sender->submit_cnt()) {
- log("waiting for signals still in flight...");
- timer.msleep(FINISH_IDLE_TIME);
- }
-
- log("");
- log("sender submitted a total of ", sender->submit_cnt(), " signals");
- log("handler received a total of ", handler->receive_cnt(), " signals");
- log("");
- log("processed ", (handler->receive_cnt()/DURATION_SECONDS), " notifications per second");
- log("handler was activated ", (handler->activation_cnt()/DURATION_SECONDS), " times per second");
- log("");
-
- if (sender->submit_cnt() != handler->receive_cnt())
- throw Test_failed_with_unequal_sent_and_received_signals();
-
- receiver.dissolve(&context_123);
- destroy(heap, sender);
- destroy(heap, handler);
-
- log("TEST ", test_cnt, " FINISHED");
-}
-
-
-static void lazy_receivers_test(Env &env)
-{
- log("");
- log("TEST ", ++test_cnt, ": lazy and out-of-order signal reception test");
- log("");
-
- Signal_receiver rec_1, rec_2;
- Signal_context rec_context_1, rec_context_2;
-
- Signal_transmitter transmitter_1(rec_1.manage(&rec_context_1));
- Signal_transmitter transmitter_2(rec_2.manage(&rec_context_2));
-
- log("submit and receive signals with multiple receivers in order");
- transmitter_1.submit();
- transmitter_2.submit();
+ Env &env;
+ Timer::Connection timer { env };
+ Signal_context context;
+ Signal_receiver receiver;
+ Handler handler { env, receiver, HANDLER_INTERVAL_MS, false, 1 };
+ Sender sender { env, receiver.manage(&context),
+ SENDER_INTERVAL_MS, false };
+ Fast_sender_test(Env &env, int id) : Signal_test(id, brief), env(env)
{
- Signal signal = rec_1.wait_for_signal();
- log("returned from wait_for_signal for receiver 1");
+ timer.msleep(DURATION_MS);
- signal = rec_2.wait_for_signal();
- log("returned from wait_for_signal for receiver 2");
+ /* stop emitting signals */
+ log("deactivate sender");
+ sender.idle(true);
+ timer.msleep(FINISH_IDLE_MS);
+ log("sender submitted a total of ", sender.submit_cnt(), " signals");
+ log("handler received a total of ", handler.receive_cnt(), " signals");
+
+ if (sender.submit_cnt() != handler.receive_cnt()) {
+ throw Unequal_sent_and_received_signals(); }
}
+};
- log("submit and receive signals with multiple receivers out of order");
- transmitter_1.submit();
- transmitter_2.submit();
-
- {
- Signal signal = rec_2.wait_for_signal();
- log("returned from wait_for_signal for receiver 2");
-
- signal = rec_1.wait_for_signal();
- log("returned from wait_for_signal for receiver 1");
- }
-
- rec_1.dissolve(&rec_context_1);
- rec_2.dissolve(&rec_context_2);
-
- log("TEST ", test_cnt, " FINISHED");
-}
-
-
-/**
- * Try correct initialization and cleanup of receiver/context
- */
-static void check_context_management(Env &env)
+struct Multiple_handlers_test : Signal_test
{
- Id_signal_context *context;
- Signal_receiver *rec;
- Signal_context_capability cap;
+ static constexpr char const *brief =
+ "get multiple handlers at one sender activated in a fair manner";
- Timer::Connection timer(env);
- Heap heap(env.ram(), env.rm());
+ enum { HANDLER_INTERVAL_MS = 8 * SPEED,
+ SENDER_INTERVAL_MS = 1 * SPEED,
+ FINISH_IDLE_MS = 2 * HANDLER_INTERVAL_MS,
+ DURATION_MS = 50 * SPEED,
+ NR_OF_HANDLERS = 4 };
- /* setup receiver side */
- context = new (heap) Id_signal_context(321);
- rec = new (heap) Signal_receiver;
- cap = rec->manage(context);
+ struct Unequal_sent_and_received_signals : Exception { };
+ struct Unequal_activation_of_handlers : Exception { };
- /* spawn sender */
- Sender *sender = new (heap) Sender(env, cap, 500);
+ Env &env;
+ Heap heap { env.ram(), env.rm() };
+ Timer::Connection timer { env };
+ Signal_context context;
+ Signal_receiver receiver;
+ Registry > handlers;
+ Sender sender { env, receiver.manage(&context),
+ SENDER_INTERVAL_MS, true};
- /* stop sender after timeout */
- timer.msleep(1000);
- log("suspend sender");
- sender->idle();
-
- /* collect pending signals and dissolve context from receiver */
+ Multiple_handlers_test(Env &env, int id) : Signal_test(id, brief), env(env)
{
- Signal signal = rec->wait_for_signal();
- log("got ", signal.num(), " signal(s) from ", signal.context());
+ for (unsigned i = 0; i < NR_OF_HANDLERS; i++)
+ new (heap) Registered(handlers, env, receiver,
+ HANDLER_INTERVAL_MS, true, i);
+ timer.msleep(DURATION_MS);
+
+ /* stop emitting signals */
+ log("stop generating new signals");
+ sender.idle(true);
+ timer.msleep(FINISH_IDLE_MS);
+
+ /* let handlers settle down */
+ handlers.for_each([&] (Handler &handler) { handler.idle(true); });
+ timer.msleep(FINISH_IDLE_MS);
+
+ /* print statistics and clean up */
+ unsigned total_rcv = 0, max_act = 0, min_act = ~0;;
+ handlers.for_each([&] (Handler &handler) {
+ unsigned const rcv = handler.receive_cnt();
+ unsigned const act = handler.activation_cnt();
+ log(handler, " received ", rcv, " signals, was activated ", act, " times");
+ total_rcv += rcv;
+ if (act > max_act) { max_act = act; }
+ if (act < min_act) { min_act = act; }
+ destroy(heap, &handler);
+ });
+ log("sender submitted a total of ", sender.submit_cnt(), " signals");
+ log("handlers received a total of ", total_rcv, " signals");
+
+ /* check if number of sent signals match the received ones */
+ if (sender.submit_cnt() != total_rcv) {
+ throw Unequal_sent_and_received_signals(); }
+
+ /* check if handlers had been activated equally (tolerance of one) */
+ if (max_act - min_act > 1) {
+ throw Unequal_activation_of_handlers(); }
}
- rec->dissolve(context);
+};
- /* let sender spin for some time */
- log("resume sender");
- sender->idle(false);
- timer.msleep(1000);
- log("suspend sender");
- sender->idle();
-
- log("destroy sender");
- destroy(heap, sender);
-
- destroy(heap, context);
- destroy(heap, rec);
-}
-
-
-/**
- * Test if 'Signal_receiver::dissolve()' blocks as long as the signal context
- * is still referenced by one or more 'Signal' objects
- */
-
-static Lock signal_context_destroyer_lock(Lock::LOCKED);
-static bool signal_context_destroyed = false;
-
-class Signal_context_destroyer : public Thread_deprecated<4096>
+struct Stress_test : Signal_test
{
- private:
+ static constexpr char const *brief =
+ "throughput when submitting/handling as fast as possible";
- Signal_receiver *_receiver;
- Signal_context *_context;
+ enum { DURATION_SEC = 5 };
- public:
+ struct Unequal_sent_and_received_signals : Exception { };
- Signal_context_destroyer(Signal_receiver *receiver, Signal_context *context)
- : Thread_deprecated("signal_context_destroyer"),
- _receiver(receiver), _context(context) { }
+ Env &env;
+ Timer::Connection timer { env };
+ Signal_context context;
+ Signal_receiver receiver;
+ Handler handler { env, receiver, 0, false, 1 };
+ Sender sender { env, receiver.manage(&context), 0, false };
- void entry()
+ Stress_test(Env &env, int id) : Signal_test(id, brief), env(env)
+ {
+ for (unsigned i = 1; i <= DURATION_SEC; i++) {
+ log(i, "/", (unsigned)DURATION_SEC);
+ timer.msleep(1000);
+ }
+ log("deactivate sender");
+ sender.idle(true);
+
+ while (handler.receive_cnt() < sender.submit_cnt()) {
+ log("waiting for signals still in flight...");
+ timer.msleep(1000);
+ }
+ log("");
+ log("sender submitted a total of ", sender.submit_cnt(), " signals");
+ log("handler received a total of ", handler.receive_cnt(), " signals");
+ log("");
+ log("handler received ", handler.receive_cnt() / DURATION_SEC, " signals per second");
+ log("handler was activated ", handler.activation_cnt() / DURATION_SEC, " times per second");
+ log("");
+
+ if (sender.submit_cnt() != handler.receive_cnt())
+ throw Unequal_sent_and_received_signals();
+ }
+};
+
+struct Lazy_receivers_test : Signal_test
+{
+ static constexpr char const *brief = "lazy and out-of-order signal reception";
+
+ Signal_context context_1, context_2;
+ Signal_receiver receiver_1, receiver_2;
+ Signal_transmitter transmitter_1 { receiver_1.manage(&context_1) };
+ Signal_transmitter transmitter_2 { receiver_2.manage(&context_2) };
+
+ Lazy_receivers_test(Env &env, int id) : Signal_test(id, brief)
+ {
+ log("submit and receive signals with multiple receivers in order");
+ transmitter_1.submit();
+ transmitter_2.submit();
{
- signal_context_destroyer_lock.lock();
- _receiver->dissolve(_context);
- signal_context_destroyed = true;
- destroy(env()->heap(), _context);
+ Signal signal = receiver_1.wait_for_signal();
+ log("returned from wait_for_signal for receiver 1");
+
+ signal = receiver_2.wait_for_signal();
+ log("returned from wait_for_signal for receiver 2");
}
+ log("submit and receive signals with multiple receivers out of order");
+ transmitter_1.submit();
+ transmitter_2.submit();
+ {
+ Signal signal = receiver_2.wait_for_signal();
+ log("returned from wait_for_signal for receiver 2");
+
+ signal = receiver_1.wait_for_signal();
+ log("returned from wait_for_signal for receiver 1");
+ }
+ }
};
-
-static void synchronized_context_destruction_test(Env &env)
+struct Context_management_test : Signal_test
{
- Signal_receiver receiver;
- Timer::Connection timer(env);
- static Heap heap(env.ram(), env.rm());
+ static constexpr char const *brief =
+ "correct initialization and cleanup of receiver and context";
- Signal_context *context = new (heap) Signal_context;
+ Env &env;
+ Timer::Connection timer { env };
+ Signal_context context;
+ Signal_receiver receiver;
+ Signal_context_capability context_cap { receiver.manage(&context) };
+ Sender sender { env, context_cap, 500, true };
- Signal_transmitter transmitter(receiver.manage(context));
- transmitter.submit();
-
- Signal_context_destroyer signal_context_destroyer(&receiver, context);
- signal_context_destroyer.start();
-
- /* The signal context destroyer thread should not be able to destroy the
- * signal context during the 'Signal' objects life time. */
+ Context_management_test(Env &env, int id) : Signal_test(id, brief), env(env)
{
- Signal signal = receiver.wait_for_signal();
-
- /* let the signal context destroyer thread try to destroy the signal context */
- signal_context_destroyer_lock.unlock();
+ /* stop sender after timeout */
timer.msleep(1000);
+ log("suspend sender");
+ sender.idle(true);
- Signal signal_copy = signal;
- Signal signal_copy2 = signal;
-
- signal_copy = signal_copy2;
-
- if (signal_context_destroyed) {
- error("signal context destroyed too early");
- sleep_forever();
+ /* collect pending signals and dissolve context from receiver */
+ {
+ Signal signal = receiver.wait_for_signal();
+ log("got ", signal.num(), " signal(s) from ", signal.context());
}
+ receiver.dissolve(&context);
+
+ /* let sender spin for some time */
+ log("resume sender");
+ sender.idle(false);
+ timer.msleep(1000);
+ log("suspend sender");
+ sender.idle(true);
+ log("destroy sender");
+ }
+};
+
+struct Synchronized_destruction_test : Signal_test, Thread
+{
+ static constexpr char const *brief =
+ "does 'dissolve' block as long as the signal context is referenced?";
+
+ struct Failed : Exception { };
+
+ Env &env;
+ Timer::Connection timer { env };
+ Heap heap { env.ram(), env.rm() };
+ Signal_context &context { *new (heap) Signal_context };
+ Signal_receiver receiver;
+ Signal_transmitter transmitter { receiver.manage(&context) };
+ bool destroyed { false };
+
+ void entry()
+ {
+ receiver.dissolve(&context);
+ log("dissolve finished");
+ destroyed = true;
+ destroy(heap, &context);
}
- signal_context_destroyer.join();
- signal_context_destroyed = false;
-}
+ Synchronized_destruction_test(Env &env, int id)
+ : Signal_test(id, brief), Thread(env, "destroyer", 1024 * sizeof(addr_t)), env(env)
+ {
+ transmitter.submit();
+ {
+ Signal signal = receiver.wait_for_signal();
+ log("start dissolving");
+ Thread::start();
+ timer.msleep(2000);
+ Signal signal_copy_1 = signal;
+ Signal signal_copy_2 = signal;
+ signal_copy_1 = signal_copy_2;
+ if (destroyed) {
+ throw Failed(); }
+ log("destruct signal");
+ }
+ Thread::join();
+ }
+};
-
-static void many_managed_contexts(Env &env)
+struct Many_contexts_test : Signal_test
{
- static Heap heap(env.ram(), env.rm());
- for (unsigned round = 0; round < 10; ++round) {
+ static constexpr char const *brief = "create and manage many contexts";
- unsigned const num_contexts = 200 + 5*round;
- log("round ", round, ": create and manage ", num_contexts, " contexts");
+ struct Manage_failed : Exception { };
- Signal_receiver rec;
+ Env &env;
+ Heap heap { env.ram(), env.rm() };
+ Registry > contexts;
- for (unsigned i = 0; i < num_contexts; ++i) {
- Id_signal_context *context = new (heap) Id_signal_context(i);
- if (!rec.manage(context).valid()) {
- error("failed to manage signal context");
- sleep_forever();
+ Many_contexts_test(Env &env, int id) : Signal_test(id, brief), env(env)
+ {
+ for (unsigned round = 0; round < 10; round++) {
+
+ unsigned const nr_of_contexts = 200 + 5 * round;
+ log("round ", round, ": manage ", nr_of_contexts, " contexts");
+
+ Signal_receiver receiver;
+ for (unsigned i = 0; i < nr_of_contexts; i++) {
+ if (!receiver.manage(new (heap) Registered(contexts)).valid()) {
+ throw Manage_failed(); }
}
+ contexts.for_each([&] (Registered &context) {
+ receiver.dissolve(&context);
+ destroy(heap, &context);
+ });
}
}
+};
- log("many contexts finished");
-}
-
-
-void Component::construct(Genode::Env &env)
+struct Main
{
- log("--- signalling test ---");
+ Constructible test_1;
+ Constructible test_2;
+ Constructible test_3;
+ Constructible test_4;
+ Constructible test_5;
+ Constructible test_6;
+ Constructible test_7;
- fast_sender_test(env);
- multiple_handlers_test(env);
- stress_test(env);
- lazy_receivers_test(env);
- check_context_management(env);
- synchronized_context_destruction_test(env);
- many_managed_contexts(env);
-
- log("--- signalling test finished ---");
- env.parent().exit(0);
+ Main(Env &env)
+ {
+ log("--- Signalling test ---");
+ test_1.construct(env, 1); test_1.destruct();
+ test_2.construct(env, 2); test_2.destruct();
+ test_3.construct(env, 3); test_3.destruct();
+ test_4.construct(env, 4); test_4.destruct();
+ test_5.construct(env, 5); test_5.destruct();
+ test_6.construct(env, 6); test_6.destruct();
+ test_7.construct(env, 7); test_7.destruct();
+ log("--- Signalling test finished ---");
}
+};
+
+void Component::construct(Genode::Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/synced_interface/main.cc b/repos/os/src/test/synced_interface/main.cc
index 10766875d4..21b8d6f2ab 100644
--- a/repos/os/src/test/synced_interface/main.cc
+++ b/repos/os/src/test/synced_interface/main.cc
@@ -13,14 +13,16 @@
/* Genode includes */
#include
-#include
+#include
+
+using namespace Genode;
struct Adder
{
int add(int a, int b)
{
- Genode::log("adding ", a, " + ", b);
+ log("adding ", a, " + ", b);
return a + b;
}
};
@@ -28,22 +30,25 @@ struct Adder
struct Pseudo_lock
{
- void lock() { Genode::log("lock"); }
- void unlock() { Genode::log("unlock"); }
+ void lock() { log("lock"); }
+ void unlock() { log("unlock"); }
};
-int main(int, char **)
+struct Main
{
- using namespace Genode;
+ Pseudo_lock lock;
+ Adder adder;
+ Synced_interface synced_adder { lock, &adder };
- Pseudo_lock lock;
- Adder adder;
+ Main(Env &env)
+ {
+ log("--- Synced interface test ---");
+ int const res = synced_adder()->add(13, 14);
+ log("result is ", res);
+ log("--- Synced interface test finished ---");
+ }
+};
- Synced_interface synced_adder(lock, &adder);
- int const res = synced_adder()->add(13, 14);
-
- log("result is ", res);
- return 0;
-}
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/terminal_echo/main.cc b/repos/os/src/test/terminal_echo/main.cc
index 12bc0a08b4..8f8b9fb950 100644
--- a/repos/os/src/test/terminal_echo/main.cc
+++ b/repos/os/src/test/terminal_echo/main.cc
@@ -1,57 +1,49 @@
/*
* \brief Terminal echo program
* \author Norman Feske
+ * \author Martin Stein
* \date 2009-10-16
*/
/*
- * Copyright (C) 2009-2013 Genode Labs GmbH
+ * Copyright (C) 2009-2017 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.
*/
-#include
-#include
+/* Genode includes */
+#include
#include
using namespace Genode;
-static const bool verbose = false;
-
-int main(int, char **)
+struct Main
{
- static Terminal::Connection terminal;
+ Terminal::Connection terminal;
+ Signal_handler read_avail;
+ char read_buffer[100];
- enum { READ_BUFFER_SIZE = 100 };
- static char read_buffer[READ_BUFFER_SIZE];
+ String<128> intro {
+ "--- Terminal echo test started - now you can type characters to be echoed. ---\r\n" };
- Signal_receiver sig_rec;
- Signal_context sig_ctx;
-
- terminal.read_avail_sigh(sig_rec.manage(&sig_ctx));
-
- static const char *intro_text =
- "--- Terminal echo test started - now you can type characters to be echoed. ---\r\n";
- terminal.write(intro_text, strlen(intro_text) + 1);
-
- for (;;) {
-
- sig_rec.wait_for_signal();
-
- int num_bytes = terminal.read(read_buffer, sizeof(read_buffer));
-
- if (verbose && (num_bytes > 0))
- log("got ", num_bytes, " bytes");
-
- for (int i = 0; i < num_bytes; i++) {
+ void handle_read_avail()
+ {
+ unsigned num_bytes = terminal.read(read_buffer, sizeof(read_buffer));
+ log("got ", num_bytes, " byte(s)");
+ for (unsigned i = 0; i < num_bytes; i++) {
if (read_buffer[i] == '\r') {
- terminal.write("\n", 1);
- }
-
+ terminal.write("\n", 1); }
terminal.write(&read_buffer[i], 1);
}
}
- return 0;
-}
+ Main(Env &env) : terminal(env),
+ read_avail(env.ep(), *this, &Main::handle_read_avail)
+ {
+ terminal.read_avail_sigh(read_avail);
+ terminal.write(intro.string(), intro.length() + 1);
+ }
+};
+
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/thread_join/main.cc b/repos/os/src/test/thread_join/main.cc
index ef5fdbf9d8..e910074871 100644
--- a/repos/os/src/test/thread_join/main.cc
+++ b/repos/os/src/test/thread_join/main.cc
@@ -11,65 +11,63 @@
* under the terms of the GNU General Public License version 2.
*/
-#include
+/* Genode includes */
+#include
#include
#include
using namespace Genode;
-struct Worker : Genode::Thread_deprecated<4096>
+struct Worker : Thread
{
- Timer::Session &timer;
- unsigned const result_value;
- unsigned volatile result;
+ Timer::Session &timer;
+ unsigned const result_value;
+ unsigned volatile result;
void entry()
{
- log("worker thread is up");
+ log("Worker thread is up");
timer.msleep(250);
- log("worker is leaving the entry function with "
- "result=", result_value, "...");
+ log("Worker is leaving the entry function with result=", result_value);
result = result_value;
}
- Worker(Timer::Session &timer, int result_value)
- :
- Thread_deprecated("worker"),
- timer(timer), result_value(result_value), result(~0)
+ Worker(Env &env, Timer::Session &timer, unsigned result_value)
+ : Thread(env, "worker", 1024 * sizeof(addr_t)), timer(timer),
+ result_value(result_value), result(~0)
{
- start();
+ Thread::start();
}
};
-/**
- * Main program
- */
-int main(int, char **)
+struct Main
{
- log("--- thread join test ---");
+ struct Worker_unfinished_after_join : Exception { };
Timer::Connection timer;
- for (unsigned i = 0; i < 10; i++) {
+ Main(Env &env) : timer(env)
+ {
+ log("--- Thread join test ---");
+ for (unsigned i = 0; i < 10; i++) {
- /*
- * A worker thread is created in each iteration. Just before
- * leaving the entry function, the worker assigns the result
- * to 'Worker::result' variable. By validating this value,
- * we determine whether the worker has finished or not.
- */
- Worker worker(timer, i);
- worker.join();
-
- if (worker.result != i) {
- error("work remains unfinished after 'join()' returned");
- return -1;
+ /*
+ * A worker thread is created in each iteration. Just before
+ * leaving the entry function, the worker assigns the result
+ * to 'Worker::result' variable. By validating this value,
+ * we determine whether the worker has finished or not.
+ */
+ Worker worker(env, timer, i);
+ worker.join();
+ if (worker.result != i) {
+ throw Worker_unfinished_after_join(); }
}
+ log("--- Thread join test finished ---");
}
+};
- log("--- signalling test finished ---");
- return 0;
-}
+
+void Component::construct(Genode::Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/timed_semaphore/main.cc b/repos/os/src/test/timed_semaphore/main.cc
index 91003cee7c..b0331a0671 100644
--- a/repos/os/src/test/timed_semaphore/main.cc
+++ b/repos/os/src/test/timed_semaphore/main.cc
@@ -1,6 +1,7 @@
/*
* \brief Test for the timed-semaphore
* \author Stefan Kalkowski
+ * \author Martin Stein
* \date 2010-03-05
*/
@@ -11,93 +12,67 @@
* under the terms of the GNU General Public License version 2.
*/
+/* Genode includes */
#include
#include
#include
+#include
using namespace Genode;
-class Wakeup_thread : public Thread_deprecated<4096>
+
+struct Test : Thread
{
- private:
+ struct Failed : Exception { };
- Timed_semaphore *_sem;
- Timer::Session *_timer;
- int _timeout;
- Lock _lock;
- bool _stop;
+ unsigned id;
+ Timer::Connection wakeup_timer;
+ unsigned const wakeup_period;
+ Timed_semaphore sem;
+ bool stop_wakeup { false };
+ Lock wakeup_stopped { Lock::LOCKED };
+ bool got_timeouts { false };
- public:
+ void entry()
+ {
+ do {
+ wakeup_timer.msleep(wakeup_period);
+ sem.up();
+ } while (!stop_wakeup);
+ wakeup_stopped.unlock();
+ }
- Wakeup_thread(Timed_semaphore *sem,
- Timer::Session *timer,
- Alarm::Time timeout)
- : Thread_deprecated("wakeup"), _sem(sem), _timer(timer), _timeout(timeout),
- _lock(Lock::LOCKED), _stop(false) { }
+ Test(Env &env, bool timeouts, unsigned id, char const *brief)
+ : Thread(env, "wakeup", 1024 * sizeof(addr_t)), id(id), wakeup_timer(env),
+ wakeup_period(timeouts ? 1000 : 100)
+ {
+ log("\nTEST ", id, ": ", brief, "\n");
+ Thread::start();
+ try { for (int i = 0; i < 10; i++) { sem.down(timeouts ? 100 : 1000); } }
+ catch (Timeout_exception) { got_timeouts = true; }
+ if (timeouts != got_timeouts) {
+ throw Failed(); }
- void entry()
- {
- while(true) {
- _timer->msleep(_timeout);
- _sem->up();
-
- if (_stop) {
- _lock.unlock();
- return;
- }
- }
- }
-
- void stop() { _stop = true; _lock.lock(); }
+ stop_wakeup = true;
+ wakeup_stopped.lock();
+ }
+ ~Test() { log("\nTEST ", id, " finished\n"); }
};
-bool test_loop(Timer::Session *timer, Alarm::Time timeout1, Alarm::Time timeout2, int loops)
+struct Main
{
- Timed_semaphore sem;
- Wakeup_thread thread(&sem, timer, timeout2);
- bool ret = true;
+ Constructible test;
- thread.start();
- try{
- for (int i = 0; i < loops; i++)
- sem.down(timeout1);
- } catch (Timeout_exception) {
- ret = false;
+ Main(Env &env)
+ {
+ log("--- Timed semaphore test ---");
+ test.construct(env, false, 1, "without timeouts"); test.destruct();
+ test.construct(env, true, 2, "with timeouts"); test.destruct();
+ log("--- Timed semaphore test finished ---");
}
-
- /*
- * Explicitly stop the thread, so the destructor does not get called in
- * unfavourable situations, e.g. where the semaphore-meta lock is still
- * held and the semaphore destructor stalls afterwards
- */
- thread.stop();
-
- return ret;
-}
+};
-int main(int, char **)
-{
- log("--- timed-semaphore test ---");
-
- Timer::Connection timer;
-
- log("--- test 1: good case, no timeout triggers --");
- if(!test_loop(&timer, 1000, 100, 10)) {
- error("Test 1 failed!");
- return -1;
- }
- log("--- everything went ok --");
-
- log("--- test 2: triggers timeouts --");
- if(test_loop(&timer, 100, 1000, 10)) {
- error("Test 2 failed!");
- return -2;
- }
- log("--- everything went ok --");
-
- log("--- end of timed-semaphore test ---");
- return 0;
-}
+void Component::construct(Genode::Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/trace/main.cc b/repos/os/src/test/trace/main.cc
index 35e8290f64..7cdaf27efd 100644
--- a/repos/os/src/test/trace/main.cc
+++ b/repos/os/src/test/trace/main.cc
@@ -2,11 +2,12 @@
* \brief Low-level test for TRACE service
* \author Norman Feske
* \author Josef Soentgen
+ * \author Martin Stein
* \date 2013-08-12
*/
/*
- * Copyright (C) 2013 Genode Labs GmbH
+ * Copyright (C) 2013-2017 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.
@@ -15,75 +16,61 @@
/* Genode includes */
#include
#include
-#include
-#include
+#include
+#include
+#include
-static char const *state_name(Genode::Trace::Subject_info::State state)
+using namespace Genode;
+
+
+struct Test_thread : Thread
{
- switch (state) {
- case Genode::Trace::Subject_info::INVALID: return "INVALID";
- case Genode::Trace::Subject_info::UNTRACED: return "UNTRACED";
- case Genode::Trace::Subject_info::TRACED: return "TRACED";
- case Genode::Trace::Subject_info::FOREIGN: return "FOREIGN";
- case Genode::Trace::Subject_info::ERROR: return "ERROR";
- case Genode::Trace::Subject_info::DEAD: return "DEAD";
- }
- return "undefined";
-}
-
-
-struct Test_thread : Genode::Thread_deprecated<1024 * sizeof (unsigned long)>
-{
- Timer::Connection _timer;
+ Env &env;
+ Timer::Connection timer { env };
void entry()
{
- using namespace Genode;
-
- for (size_t i = 0; ; i++) {
+ for (unsigned i = 0; ; i++) {
if (i & 0x3) {
- Ram_dataspace_capability ds_cap = env()->ram_session()->alloc(1024);
- env()->ram_session()->free(ds_cap);
+ Ram_dataspace_capability ds_cap = env.ram().alloc(1024);
+ env.ram().free(ds_cap);
}
-
- _timer.msleep(250);
+ timer.msleep(250);
}
}
- Test_thread(const char *name)
- : Thread_deprecated(name) { start(); }
+ Test_thread(Env &env, Name &name)
+ : Thread(env, name, 1024 * sizeof(addr_t)), env(env) { start(); }
};
-using namespace Genode;
-
-
class Trace_buffer_monitor
{
private:
enum { MAX_ENTRY_BUF = 256 };
- char _buf[MAX_ENTRY_BUF];
+ char _buf[MAX_ENTRY_BUF];
+ Region_map &_rm;
Trace::Subject_id _id;
Trace::Buffer *_buffer;
- Trace::Buffer::Entry _curr_entry;
+ Trace::Buffer::Entry _curr_entry;
const char *_terminate_entry(Trace::Buffer::Entry const &entry)
{
size_t len = min(entry.length() + 1, MAX_ENTRY_BUF);
memcpy(_buf, entry.data(), len);
_buf[len-1] = '\0';
-
return _buf;
}
public:
- Trace_buffer_monitor(Trace::Subject_id id, Dataspace_capability ds_cap)
+ Trace_buffer_monitor(Region_map &rm,
+ Trace::Subject_id id,
+ Dataspace_capability ds_cap)
:
- _id(id),
- _buffer(env()->rm_session()->attach(ds_cap)),
+ _rm(rm), _id(id), _buffer(rm.attach(ds_cap)),
_curr_entry(_buffer->first())
{
log("monitor subject:", _id.id, " buffer:", Hex((addr_t)_buffer));
@@ -92,7 +79,7 @@ class Trace_buffer_monitor
~Trace_buffer_monitor()
{
if (_buffer)
- env()->rm_session()->detach(_buffer);
+ _rm.detach(_buffer);
}
Trace::Subject_id id() { return _id; };
@@ -100,7 +87,6 @@ class Trace_buffer_monitor
void dump()
{
log("overflows: ", _buffer->wrapped());
-
log("read all remaining events");
for (; !_curr_entry.last(); _curr_entry = _buffer->next(_curr_entry)) {
/* omit empty entries */
@@ -111,163 +97,193 @@ class Trace_buffer_monitor
if (data)
log(data);
}
-
/* reset after we read all available entries */
_curr_entry = _buffer->first();
}
};
-static void test_out_of_metadata()
+struct Test_out_of_metadata
{
- log("test Out_of_metadata exception of Trace::Session::subjects call");
+ Env &env;
- /*
- * The call of 'subjects' will prompt core's TRACE service to import those
- * threads as trace subjects into the TRACE session. This step should fail
- * because we dimensioned the TRACE session with a very low amount of
- * session quota. The allocation failure is propagated to the TRACE client
- * by the 'Out_of_metadata' exception. The test validates this
- * error-handling procedure.
- */
+ Test_out_of_metadata(Env &env) : env(env)
+ {
+ log("test Out_of_metadata exception of Trace::Session::subjects call");
- enum { MAX_SUBJECT_IDS = 16 };
- Genode::Trace::Subject_id subject_ids[MAX_SUBJECT_IDS];
+ /*
+ * The call of 'subjects' will prompt core's TRACE service to import those
+ * threads as trace subjects into the TRACE session. This step should fail
+ * because we dimensioned the TRACE session with a very low amount of
+ * session quota. The allocation failure is propagated to the TRACE client
+ * by the 'Out_of_metadata' exception. The test validates this
+ * error-handling procedure.
+ */
- try {
- Genode::Trace::Connection trace(sizeof(subject_ids) + 4096, sizeof(subject_ids), 0);
+ enum { MAX_SUBJECT_IDS = 16 };
+ Trace::Subject_id subject_ids[MAX_SUBJECT_IDS];
- /* we should never arrive here */
- struct Unexpectedly_got_no_exception{};
- throw Unexpectedly_got_no_exception();
- } catch (Genode::Parent::Service_denied) {
- log("got Genode::Parent::Service_denied exception as expected");
+ try {
+ Trace::Connection trace(env, sizeof(subject_ids) + 4096,
+ sizeof(subject_ids), 0);
+
+ /* we should never arrive here */
+ struct Unexpectedly_got_no_exception{};
+ throw Unexpectedly_got_no_exception();
+ } catch (Parent::Service_denied) {
+ log("got Parent::Service_denied exception as expected"); }
+
+ try {
+ Trace::Connection trace(env, sizeof(subject_ids) + 5*4096,
+ sizeof(subject_ids), 0);
+ trace.subjects(subject_ids, MAX_SUBJECT_IDS);
+
+ /* we should never arrive here */
+ struct Unexpectedly_got_no_exception{};
+ throw Unexpectedly_got_no_exception();
+
+ } catch (Trace::Out_of_metadata) {
+ log("got Trace::Out_of_metadata exception as expected"); }
+
+ log("passed Out_of_metadata test");
}
-
- try {
- Genode::Trace::Connection trace(sizeof(subject_ids) + 5*4096, sizeof(subject_ids), 0);
- trace.subjects(subject_ids, MAX_SUBJECT_IDS);
-
- /* we should never arrive here */
- struct Unexpectedly_got_no_exception{};
- throw Unexpectedly_got_no_exception();
-
- } catch (Trace::Out_of_metadata) {
- log("got Trace::Out_of_metadata exception as expected");
- }
-
- log("passed Out_of_metadata test");
-}
+};
-int main(int argc, char **argv)
+struct Test_tracing
{
- using namespace Genode;
+ Env &env;
+ Attached_rom_dataspace config { env, "config" };
+ Heap heap { env.ram(), env.rm() };
+ Trace::Connection trace { env, 1024*1024, 64*1024, 0 };
+ Timer::Connection timer { env };
+ Test_thread::Name thread_name { "test-thread" };
+ Test_thread thread { env, thread_name };
+ Trace_buffer_monitor *test_monitor { nullptr };
+ bool policy_set { false };
+ Trace::Policy_id policy_id;
+ char policy_label[64];
+ char policy_module[64];
+ Rom_dataspace_capability policy_module_rom_ds;
- log("--- test-trace started ---");
-
- test_out_of_metadata();
-
- static Genode::Trace::Connection trace(1024*1024, 64*1024, 0);
-
- static Timer::Connection timer;
-
- static Test_thread test("test-thread");
-
- static Trace_buffer_monitor *test_monitor = 0;
-
- Genode::Trace::Policy_id policy_id;
- bool policy_set = false;
-
- char policy_label[64];
- char policy_module[64];
- Rom_dataspace_capability policy_module_rom_ds;
-
- try {
- Xml_node policy = config()->xml_node().sub_node("trace_policy");
- for (;; policy = policy.next("trace_policy")) {
- try {
- policy.attribute("label").value(policy_label, sizeof (policy_label));
- policy.attribute("module").value(policy_module, sizeof (policy_module));
-
- static Rom_connection policy_rom(policy_module);
- policy_module_rom_ds = policy_rom.dataspace();
-
- size_t rom_size = Dataspace_client(policy_module_rom_ds).size();
-
- policy_id = trace.alloc_policy(rom_size);
- Dataspace_capability ds_cap = trace.policy(policy_id);
-
- if (ds_cap.valid()) {
- void *ram = env()->rm_session()->attach(ds_cap);
- void *rom = env()->rm_session()->attach(policy_module_rom_ds);
- memcpy(ram, rom, rom_size);
-
- env()->rm_session()->detach(ram);
- env()->rm_session()->detach(rom);
- }
- } catch (...) {
- error("could not load module '", Cstring(policy_module), "' for "
- "label '", Cstring(policy_label), "'");
- }
-
- log("load module: '", Cstring(policy_module), "' for "
- "label: '", Cstring(policy_label), "'");
-
- if (policy.last("trace_policy")) break;
+ char const *state_name(Trace::Subject_info::State state)
+ {
+ switch (state) {
+ case Trace::Subject_info::INVALID: return "INVALID";
+ case Trace::Subject_info::UNTRACED: return "UNTRACED";
+ case Trace::Subject_info::TRACED: return "TRACED";
+ case Trace::Subject_info::FOREIGN: return "FOREIGN";
+ case Trace::Subject_info::ERROR: return "ERROR";
+ case Trace::Subject_info::DEAD: return "DEAD";
}
+ return "undefined";
+ }
- } catch (...) { }
-
- for (size_t cnt = 0; cnt < 5; cnt++) {
-
- timer.msleep(3000);
-
- Trace::Subject_id subjects[32];
- size_t num_subjects = trace.subjects(subjects, 32);
-
- log(num_subjects, " tracing subjects present");
-
- for (size_t i = 0; i < num_subjects; i++) {
-
- Trace::Subject_info info = trace.subject_info(subjects[i]);
- log("ID:", subjects[i].id, " "
- "label:\"", info.session_label(), "\" "
- "name:\"", info.thread_name(), "\" "
- "state:", state_name(info.state()), " "
- "policy:", info.policy_id().id, " "
- "time:", info.execution_time().value);
-
- /* enable tracing */
- if (!policy_set
- && strcmp(info.session_label().string(), policy_label) == 0
- && strcmp(info.thread_name().string(), "test-thread") == 0) {
+ Test_tracing(Env &env) : env(env)
+ {
+ try {
+ Xml_node policy = config.xml().sub_node("trace_policy");
+ for (;; policy = policy.next("trace_policy")) {
try {
- log("enable tracing for "
- "thread:'", info.thread_name().string(), "' with "
- "policy:", policy_id.id);
+ policy.attribute("label").value(policy_label, sizeof (policy_label));
+ policy.attribute("module").value(policy_module, sizeof (policy_module));
- trace.trace(subjects[i].id, policy_id, 16384U);
+ Rom_connection policy_rom(env, policy_module);
+ policy_module_rom_ds = policy_rom.dataspace();
- Dataspace_capability ds_cap = trace.buffer(subjects[i].id);
- test_monitor = new (env()->heap()) Trace_buffer_monitor(subjects[i].id, ds_cap);
+ size_t rom_size = Dataspace_client(policy_module_rom_ds).size();
- } catch (Trace::Source_is_dead) { error("source is dead"); }
+ policy_id = trace.alloc_policy(rom_size);
+ Dataspace_capability ds_cap = trace.policy(policy_id);
- policy_set = true;
+ if (ds_cap.valid()) {
+ void *ram = env.rm().attach(ds_cap);
+ void *rom = env.rm().attach(policy_module_rom_ds);
+ memcpy(ram, rom, rom_size);
+
+ env.rm().detach(ram);
+ env.rm().detach(rom);
+ }
+ } catch (...) {
+ error("could not load module '", Cstring(policy_module), "' for "
+ "label '", Cstring(policy_label), "'");
+ }
+
+ log("load module: '", Cstring(policy_module), "' for "
+ "label: '", Cstring(policy_label), "'");
+
+ if (policy.last("trace_policy")) break;
}
- /* read events from trace buffer */
- if (test_monitor) {
- if (subjects[i].id == test_monitor->id().id)
- test_monitor->dump();
+ } catch (...) { }
+
+ for (size_t cnt = 0; cnt < 5; cnt++) {
+
+ timer.msleep(3000);
+
+ Trace::Subject_id subjects[32];
+ size_t num_subjects = trace.subjects(subjects, 32);
+
+ log(num_subjects, " tracing subjects present");
+
+ for (size_t i = 0; i < num_subjects; i++) {
+
+ Trace::Subject_info info = trace.subject_info(subjects[i]);
+ log("ID:", subjects[i].id, " "
+ "label:\"", info.session_label(), "\" "
+ "name:\"", info.thread_name(), "\" "
+ "state:", state_name(info.state()), " "
+ "policy:", info.policy_id().id, " "
+ "time:", info.execution_time().value);
+
+ /* enable tracing */
+ if (!policy_set
+ && strcmp(info.session_label().string(), policy_label) == 0
+ && strcmp(info.thread_name().string(), "test-thread") == 0) {
+ try {
+ log("enable tracing for "
+ "thread:'", info.thread_name().string(), "' with "
+ "policy:", policy_id.id);
+
+ trace.trace(subjects[i].id, policy_id, 16384U);
+
+ Dataspace_capability ds_cap = trace.buffer(subjects[i].id);
+ test_monitor = new (heap)
+ Trace_buffer_monitor(env.rm(), subjects[i].id, ds_cap);
+
+ } catch (Trace::Source_is_dead) { error("source is dead"); }
+
+ policy_set = true;
+ }
+
+ /* read events from trace buffer */
+ if (test_monitor) {
+ if (subjects[i].id == test_monitor->id().id)
+ test_monitor->dump();
+ }
}
}
+
+ if (test_monitor)
+ destroy(heap, test_monitor);
}
+};
- if (test_monitor)
- destroy(env()->heap(), test_monitor);
+struct Main
+{
+ Constructible test_1;
+ Constructible test_2;
- log("--- test-trace finished ---");
- return 0;
-}
+ Main(Env &env)
+ {
+ log("--- test-trace started ---");
+ test_1.construct(env);
+ test_1.destruct();
+ test_2.construct(env);
+ test_2.destruct();
+ log("--- test-trace finished ---");
+ }
+};
+
+
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/uart/main.cc b/repos/os/src/test/uart/main.cc
index 39d4bcc6a8..3375515903 100644
--- a/repos/os/src/test/uart/main.cc
+++ b/repos/os/src/test/uart/main.cc
@@ -5,35 +5,37 @@
*/
/*
- * Copyright (C) 2011-2013 Genode Labs GmbH
+ * Copyright (C) 2011-2017 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.
*/
+/* Genode includes */
#include
-#include
+#include
#include
#include
-
using namespace Genode;
-int main()
+
+struct Main
{
- log("--- UART test started ---");
+ Timer::Connection timer;
+ Uart::Connection uart;
+ char buf[100];
- static Timer::Connection timer;
- static Uart::Connection uart;
+ Main(Env &env)
+ {
+ log("--- UART test started ---");
- for (unsigned i = 0; ; ++i) {
-
- static char buf[100];
- int n = snprintf(buf, sizeof(buf), "UART test message %d\n", i);
- uart.write(buf, n);
-
- timer.msleep(2000);
+ for (unsigned i = 0; ; ++i) {
+ int n = snprintf(buf, sizeof(buf), "UART test message %d\n", i);
+ uart.write(buf, n);
+ timer.msleep(2000);
+ }
}
+};
- return 0;
-}
+void Component::construct(Env &env) { static Main main(env); }
diff --git a/repos/os/src/test/vfs_stress/main.cc b/repos/os/src/test/vfs_stress/main.cc
index bf5f98aaa6..ad808c0d41 100644
--- a/repos/os/src/test/vfs_stress/main.cc
+++ b/repos/os/src/test/vfs_stress/main.cc
@@ -477,7 +477,7 @@ void Component::construct(Genode::Env &env)
MAX_DEPTH = config_xml.attribute_value("depth", 16U);
unsigned long elapsed_ms;
- Timer::Connection timer;
+ Timer::Connection timer(env);
/* populate the directory file system at / */
vfs_root.num_dirent("/");