/* * \brief Launchpad interface * \author Norman Feske * \date 2006-09-01 * * This class is a convenience-wrapper for starting and killing * child processes. To support the Launchpad application, is * also provides an interface to glues the child handling and * the GUI together. */ /* * Copyright (C) 2006-2013 Genode Labs GmbH * * This file is part of the Genode OS framework, which is distributed * under the terms of the GNU General Public License version 2. */ #ifndef _INCLUDE__LAUNCHPAD__LAUNCHPAD_H_ #define _INCLUDE__LAUNCHPAD__LAUNCHPAD_H_ #include #include #include #include #include #include #include #include class Launchpad_child_policy : public Genode::Child_policy, public Genode::Client { private: typedef Genode::String<64> Name; Name const _name; Genode::Server *_server; Genode::Service_registry *_parent_services; Genode::Service_registry *_child_services; Genode::Dataspace_capability _config_ds; Genode::Rpc_entrypoint *_parent_entrypoint; Init::Child_policy_enforce_labeling _labeling_policy; Init::Child_policy_provide_rom_file _config_policy; Init::Child_policy_provide_rom_file _binary_policy; public: Launchpad_child_policy(const char *name, Genode::Server *server, Genode::Service_registry *parent_services, Genode::Service_registry *child_services, Genode::Dataspace_capability config_ds, Genode::Dataspace_capability binary_ds, Genode::Rpc_entrypoint *parent_entrypoint) : _name(name), _server(server), _parent_services(parent_services), _child_services(child_services), _config_ds(config_ds), _parent_entrypoint(parent_entrypoint), _labeling_policy(_name.string()), _config_policy("config", config_ds, _parent_entrypoint), _binary_policy("binary", binary_ds, _parent_entrypoint) { } const char *name() const { return _name.string(); } Genode::Service *resolve_session_request(const char *service_name, const char *args) { Genode::Service *service; /* check for config file request */ if ((service = _config_policy.resolve_session_request(service_name, args))) return service; /* check for binary file request */ if ((service = _binary_policy.resolve_session_request(service_name, args))) return service; /* if service is provided by one of our children, use it */ if ((service = _child_services->find(service_name))) return service; /* * Handle special case of the demo scenario when the user uses * a nested Launchad for starting another Nitpicker instance * before starting Liquid_fb. In this case, we do not want to * delegate Nitpicker's session requests to the parent. The * parent may be a launchpad again trying to apply the same * policy. This instance will recognise that the session is not * available at init (because the actual input and framebuffer * drivers are singletons, and would therefore block for * Framebuffer or Input to become available at one of its own * children. The launchpad with which the user originally * interacted, however, would block at the parent interface * until this condition gets satisfied. */ if (Genode::strcmp(service_name, "Input") != 0 && Genode::strcmp(service_name, "Framebuffer") != 0 && (service = _parent_services->find(service_name))) return service; /* wait for the service to become available */ Genode::Client client; return _child_services->wait_for_service(service_name, &client, name()); } void filter_session_args(const char *service, char *args, Genode::size_t args_len) { _labeling_policy.filter_session_args(service, args, args_len); } bool announce_service(const char *service_name, Genode::Root_capability root, Genode::Allocator *alloc, Genode::Server * /*server*/) { if (_child_services->find(service_name)) { PWRN("%s: service %s is already registered", name(), service_name); return false; } /* XXX remove potential race between checking for and inserting service */ _child_services->insert(new (alloc) Genode::Child_service(service_name, root, _server)); Genode::printf("%s registered service %s\n", name(), service_name); return true; } void unregister_services() { Genode::Service *rs; while ((rs = _child_services->find_by_server(_server))) _child_services->remove(rs); } }; class Launchpad; class Launchpad_child : public Genode::List::Element { private: static Genode::Dataspace_capability _ldso_ds(); Launchpad *_launchpad; /* * Entry point used for serving the parent interface and the * locally provided ROM sessions for the 'config' and 'binary' * files. */ enum { ENTRYPOINT_STACK_SIZE = 12*1024 }; Genode::Rpc_entrypoint _entrypoint; Genode::Region_map_client _address_space; Genode::Rom_session_client _rom; Genode::Pd_session_client _pd; Genode::Ram_session_client _ram; Genode::Cpu_session_client _cpu; Genode::Child::Initial_thread _initial_thread; Genode::Server _server; Launchpad_child_policy _policy; Genode::Child _child; public: Launchpad_child(const char *name, Genode::Dataspace_capability elf_ds, Genode::Pd_session_capability pd, Genode::Ram_session_capability ram, Genode::Cpu_session_capability cpu, Genode::Rom_session_capability rom, Genode::Cap_session *cap_session, Genode::Service_registry *parent_services, Genode::Service_registry *child_services, Genode::Dataspace_capability config_ds, Launchpad *launchpad) : _launchpad(launchpad), _entrypoint(cap_session, ENTRYPOINT_STACK_SIZE, name, false), _address_space(Genode::Pd_session_client(pd).address_space()), _rom(rom), _pd(pd), _ram(ram), _cpu(cpu), _initial_thread(_cpu, _pd, name), _server(_ram), _policy(name, &_server, parent_services, child_services, config_ds, elf_ds, &_entrypoint), _child(elf_ds, _ldso_ds(), _pd, _pd, _ram, _ram, _cpu, _initial_thread, *Genode::env()->rm_session(), _address_space, _entrypoint, _policy) { _entrypoint.activate(); } /** * Required to forcefully kill client which blocks on a session * opening quest where the service is not up yet. */ void cancel_blocking() { _entrypoint.cancel_blocking(); } Genode::Rom_session_capability rom_session_cap() { return _rom; } Genode::Ram_session_capability ram_session_cap() { return _ram; } Genode::Cpu_session_capability cpu_session_cap() { return _cpu; } const char *name() const { return _policy.name(); } const Genode::Server *server() const { return &_server; } Genode::Allocator *heap() { return _child.heap(); } void revoke_server(const Genode::Server *server) { _child.revoke_server(server); } }; class Launchpad { private: unsigned long _initial_quota; Genode::Service_registry _parent_services; Genode::Service_registry _child_services; Genode::Lock _children_lock; Genode::List _children; bool _child_name_exists(const char *name); void _get_unique_child_name(const char *filename, char *dst, int dst_len); Genode::Sliced_heap _sliced_heap; /* cap session for allocating capabilities for parent interfaces */ Genode::Cap_connection _cap_session; protected: int _ypos; public: Genode::Lock gui_lock; Launchpad(unsigned long initial_quota); unsigned long initial_quota() { return _initial_quota; } virtual ~Launchpad() { } /** * Process launchpad XML configuration */ void process_config(); /************************* ** Configuring the GUI ** *************************/ virtual void quota(unsigned long quota) { } virtual void add_launcher(const char *filename, unsigned long default_quota, Genode::Dataspace_capability config_ds) { } virtual void add_child(const char *unique_name, unsigned long quota, Launchpad_child *launchpad_child, Genode::Allocator *alloc) { } virtual void remove_child(const char *name, Genode::Allocator *alloc) { } Launchpad_child *start_child(const char *prg_name, unsigned long quota, Genode::Dataspace_capability config_ds); /** * Exit child and close all its sessions * * \param timer Timer session to use for watchdog * mechanism. When using the default * value, a new timer session gets * created during the 'exit_child' * call. * \param session_close_timeout_ms Timeout in milliseconds until a an * unresponsive service->close call * gets canceled. */ void exit_child(Launchpad_child *child, Timer::Session *timer = 0, int session_close_timeout_ms = 2000); }; #endif /* _INCLUDE__LAUNCHPAD__LAUNCHPAD_H_ */