genode/repos/os/src/server/rom_prefetcher/main.cc

143 lines
3.0 KiB
C++
Raw Normal View History

2011-12-22 16:19:25 +01:00
/*
* \brief ROM prefetching service
* \author Norman Feske
* \date 2011-01-24
*/
/*
* Copyright (C) 2011-2016 Genode Labs GmbH
2011-12-22 16:19:25 +01:00
*
* 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 <root/component.h>
2016-11-25 16:54:49 +01:00
#include <base/component.h>
#include <base/log.h>
#include <base/heap.h>
2016-11-25 16:54:49 +01:00
#include <base/attached_rom_dataspace.h>
#include <timer_session/connection.h>
#include <base/session_label.h>
2011-12-22 16:19:25 +01:00
2016-11-25 16:54:49 +01:00
namespace Rom_prefetcher {
class Rom_session_component;
class Rom_root;
struct Main;
}
2011-12-22 16:19:25 +01:00
volatile int dummy;
2016-11-25 16:54:49 +01:00
static void prefetch_dataspace(Genode::Region_map &rm, Genode::Dataspace_capability cap)
2011-12-22 16:19:25 +01:00
{
2016-11-25 16:54:49 +01:00
Genode::Attached_dataspace ds(rm, cap);
2011-12-22 16:19:25 +01:00
/*
* Modify global volatile 'dummy' variable to prevent the compiler
* from removing the prefetch loop.
*/
2011-12-22 16:19:25 +01:00
enum { PREFETCH_STEP = 4096 };
2016-11-25 16:54:49 +01:00
for (Genode::size_t i = 0; i < ds.size(); i += PREFETCH_STEP)
dummy += ds.local_addr<char>()[i];
}
2011-12-22 16:19:25 +01:00
2016-11-25 16:54:49 +01:00
class Rom_prefetcher::Rom_session_component : public Genode::Rpc_object<Genode::Rom_session>
{
private:
2011-12-22 16:19:25 +01:00
Genode::Rom_connection _rom;
2011-12-22 16:19:25 +01:00
public:
/**
* Constructor
*
* \param filename name of the requested file
*/
2016-11-25 16:54:49 +01:00
Rom_session_component(Genode::Env &env, Genode::Session_label const &label)
:
_rom(label.string())
2011-12-22 16:19:25 +01:00
{
2016-11-25 16:54:49 +01:00
prefetch_dataspace(env.rm(), _rom.dataspace());
2011-12-22 16:19:25 +01:00
}
2016-11-25 16:54:49 +01:00
2011-12-22 16:19:25 +01:00
/***************************
** ROM session interface **
***************************/
2016-11-25 16:54:49 +01:00
Genode::Rom_dataspace_capability dataspace() { return _rom.dataspace(); }
Support for dynamic ROM sessions, fix #170 This patch introduces support for ROM sessions that update their provided data during the lifetime of the session. The 'Rom_session' interface had been extended with the new 'release()' and 'sigh()' functions, which are needed to support the new protocol. All ROM services have been updated to the new interface. Furthermore, the patch changes the child policy of init with regard to the handling of configuration files. The 'Init::Child' used to always provide the ROM dataspace with the child's config file via a locally implemented ROM service. However, for dynamic ROM sessions, we need to establish a session to the real supplier of the ROM data. This is achieved by using a new 'Child_policy_redirect_rom_file' policy to handle the 'configfile' rather than handling the 'configfile' case entirely within 'Child_config'. To see the new facility in action, the new 'os/run/dynamic_config.run' script provides a simple scenario. The config file of the test program is provided by a service, which generates and updates the config data at regular intervals. In addition, new support has been added to let slaves use dynamic reconfiguration. By using the new 'Child_policy_dynamic_rom_file', the configuration of a slave can be changed dynamically at runtime via the new 'configure()' function. The config is provided as plain null-terminated string (instead of a dataspace capability) because we need to buffer the config data anyway. So there is no benefit of using a dataspace. For buffering configuration data, a 'Ram_session' must be supplied. If no 'Ram_session' is specified at construction time of a 'Slave_policy', no config is supplied to the slave (which is still a common case). An example for dynamically reconfiguring a slave is provided by 'os/run/dynamic_config_slave.run'.
2012-04-04 17:07:19 +02:00
void sigh(Genode::Signal_context_capability) { }
2011-12-22 16:19:25 +01:00
};
2016-11-25 16:54:49 +01:00
class Rom_prefetcher::Rom_root : public Genode::Root_component<Rom_session_component>
2011-12-22 16:19:25 +01:00
{
private:
2016-11-25 16:54:49 +01:00
Genode::Env &_env;
2011-12-22 16:19:25 +01:00
Rom_session_component *_create_session(const char *args)
{
Genode::Session_label const label = Genode::label_from_args(args);
2011-12-22 16:19:25 +01:00
/* create new session for the requested file */
return new (md_alloc())
2016-11-25 16:54:49 +01:00
Rom_session_component(_env, label.last_element());
2011-12-22 16:19:25 +01:00
}
public:
2016-11-25 16:54:49 +01:00
Rom_root(Genode::Env &env, Genode::Allocator &md_alloc)
2011-12-22 16:19:25 +01:00
:
2016-11-25 16:54:49 +01:00
Genode::Root_component<Rom_session_component>(env.ep(), md_alloc),
_env(env)
2011-12-22 16:19:25 +01:00
{ }
};
2016-11-25 16:54:49 +01:00
struct Rom_prefetcher::Main
2011-12-22 16:19:25 +01:00
{
2016-11-25 16:54:49 +01:00
Genode::Env &_env;
2011-12-22 16:19:25 +01:00
2016-11-25 16:54:49 +01:00
Genode::Attached_rom_dataspace _config { _env, "config" };
2011-12-22 16:19:25 +01:00
2016-11-25 16:54:49 +01:00
Genode::Sliced_heap _sliced_heap { _env.ram(), _env.rm() };
Rom_root _root { _env, _sliced_heap };
Main(Genode::Env &env) : _env(env)
{
Timer::Connection timer(_env);
2016-11-25 16:54:49 +01:00
_config.xml().for_each_sub_node("rom", [&] (Genode::Xml_node entry) {
typedef Genode::String<64> Name;
Name const name = entry.attribute_value("name", Name());
try {
2016-11-25 16:54:49 +01:00
Genode::Rom_connection rom(_env, name.string());
log("prefetching ROM module ", name);
prefetch_dataspace(_env.rm(), rom.dataspace());
} catch (...) {
2016-11-25 16:54:49 +01:00
error("could not open ROM module ", name);
}
/* yield */
timer.msleep(1);
2016-11-25 16:54:49 +01:00
});
2016-11-25 16:54:49 +01:00
/* announce server */
_env.parent().announce(_env.ep().manage(_root));
}
};
2011-12-22 16:19:25 +01:00
2016-11-25 16:54:49 +01:00
void Component::construct(Genode::Env &env) { static Rom_prefetcher::Main main(env); }
2011-12-22 16:19:25 +01:00