/* * \brief Runtime for hosting GUI dialogs in child components * \author Norman Feske * \date 2023-03-24 */ /* * Copyright (C) 2023 Genode Labs GmbH * * This file is part of the Genode OS framework, which is distributed * under the terms of the GNU Affero General Public License version 3. */ #ifndef _INCLUDE__DIALOG__SANDBOXED_RUNTIME_H_ #define _INCLUDE__DIALOG__SANDBOXED_RUNTIME_H_ #include #include #include #include #include #include namespace Dialog { struct Sandboxed_runtime; } class Dialog::Sandboxed_runtime : Noncopyable { public: class View; struct Event_handler_base : Interface, Noncopyable { virtual void handle_event(Event const &event) = 0; }; template class Event_handler; private: Env &_env; Allocator &_alloc; Sandbox &_sandbox; struct Optional_event_handler { Event_handler_base *_ptr; void handle_event(Event const &event) { if (_ptr) _ptr->handle_event(event); } } _optional_event_handler { }; using Views = Dictionary; Event::Seq_number _global_seq_number { 1 }; Views _views { }; struct Gui_session; struct Report_session; using Gui_service = Sandbox::Local_service; using Rom_service = Sandbox::Local_service; using Report_service = Sandbox::Local_service; void _handle_gui_service(); void _handle_rom_service(); void _handle_report_service(); struct Service_handler : Sandbox::Local_service_base::Wakeup { Sandboxed_runtime &_runtime; using Member = void (Sandboxed_runtime::*) (); Member _member; void wakeup_local_service() override { (_runtime.*_member)(); } Service_handler(Sandboxed_runtime &runtime, Member member) : _runtime(runtime), _member(member) { } }; Service_handler _gui_handler { *this, &Sandboxed_runtime::_handle_gui_service }; Service_handler _rom_handler { *this, &Sandboxed_runtime::_handle_rom_service }; Service_handler _report_handler { *this, &Sandboxed_runtime::_handle_report_service }; Gui_service _gui_service; Rom_service _rom_service; Report_service _report_service; public: Sandboxed_runtime(Env &, Allocator &, Sandbox &); /** * Respond to sandbox state changes * * \return true if the sandbox configuration needs to be updated */ bool apply_sandbox_state(Xml_node const &); void gen_start_nodes(Xml_generator &) const; }; class Dialog::Sandboxed_runtime::Report_session : public Session_object { public: struct Handler : Interface, Genode::Noncopyable { virtual void handle_report() = 0; }; private: Attached_ram_dataspace _client_ds; Attached_ram_dataspace _local_ds; Constructible _xml { }; /* points inside _local_ds */ Handler &_handler; /******************************* ** Report::Session interface ** *******************************/ Dataspace_capability dataspace() override { return _client_ds.cap(); } void submit(size_t length) override { size_t const num_bytes = min(_client_ds.size(), length); memcpy(_local_ds.local_addr(), _client_ds.local_addr(), num_bytes); _xml.destruct(); try { _xml.construct(_local_ds.local_addr(), num_bytes); } catch (...) { } _handler.handle_report(); } void response_sigh(Signal_context_capability) override { } size_t obtain_response() override { return 0; } public: template Report_session(Env &env, Handler &handler, Entrypoint &ep, Resources const &resources, ARGS &&... args) : Session_object(ep, resources, args...), _client_ds(env.ram(), env.rm(), resources.ram_quota.value/2), _local_ds (env.ram(), env.rm(), resources.ram_quota.value/2), _handler(handler) { } template void with_xml(FN const &fn) const { if (_xml.constructed()) fn(*_xml); else fn(Xml_node("")); } }; class Dialog::Sandboxed_runtime::View : private Views::Element { private: /* needed for privately inheriting 'Views::Element' */ friend class Dictionary; friend class Avl_node; friend class Avl_tree; public: Env &_env; Allocator &_alloc; Event::Seq_number &_global_seq_number; Optional_event_handler &_optional_event_handler; Top_level_dialog &_dialog; bool _dialog_hovered = false; /* used to cut hover feedback loop */ /* sequence numbers to correlate hover info with click/clack events */ Event::Seq_number _hover_seq_number { }; Constructible _click_seq_number { }; Constructible _clack_seq_number { }; bool _click_delivered = false; /* used to deliver each click only once */ bool _dragged() const { return _click_seq_number.constructed() && *_click_seq_number == _global_seq_number && _click_delivered; } bool _hover_observable_without_click = false; struct Rom_producer : Dynamic_rom_session::Xml_producer { View const &_view; Rom_producer(View const &view) : Dynamic_rom_session::Xml_producer("dialog"), _view(view) { } void produce_xml(Xml_generator &xml) override { _view._with_dialog_hover([&] (Xml_node const &hover) { Event::Dragged const dragged { _view._dragged() }; bool const supply_hover = _view._hover_observable_without_click || dragged.value; static Xml_node omitted_hover(""); At const at { _view._global_seq_number, supply_hover ? hover : omitted_hover }; Scope<> top_level_scope(xml, at, dragged, { _view._dialog.name }); _view._dialog.view(top_level_scope); }); } } _dialog_producer { *this }; Dynamic_rom_session _dialog_rom_session { _env.ep(), _env.ram(), _env.rm(), _dialog_producer }; template struct Hover_handler : Report_session::Handler { T &_obj; void (T::*_member) (); Hover_handler(T &obj, void (T::*member)()) : _obj(obj), _member(member) { } void handle_report() override { (_obj.*_member)(); } }; Constructible _hover_report_session { }; template void _with_dialog_hover(FN const &fn) const { bool done = false; if (_hover_report_session.constructed()) _hover_report_session->with_xml([&] (Xml_node const &hover) { hover.with_optional_sub_node("dialog", [&] (Xml_node const &dialog) { fn(dialog); done = true; }); }); if (!done) fn(Xml_node("")); } void _handle_input_event(Input::Event const &); void _handle_hover(); Hover_handler _hover_handler { *this, &View::_handle_hover }; void _try_handle_click_and_clack(); struct Menu_view_state { using Start_name = String<128>; Start_name const _name; Ram_quota const _initial_ram; Cap_quota const _initial_caps; Ram_quota _ram = _initial_ram; Cap_quota _caps = _initial_caps; unsigned _version = 0; void trigger_restart() { _version++; _ram = _initial_ram; _caps = _initial_caps; } /** * Adapt runtime state information to the child * * This method responds to RAM and cap-resource requests by increasing * the resource quotas as needed. * * \param child child node of the sandbox state report * \return true if runtime must be reconfigured so that the changes * can take effect */ bool apply_child_state_report(Xml_node const &child) { bool result = false; if (child.attribute_value("name", Start_name()) != _name) return false; if (child.has_sub_node("ram") && child.sub_node("ram").has_attribute("requested")) { _ram.value *= 2; result = true; } if (child.has_sub_node("caps") && child.sub_node("caps").has_attribute("requested")) { _caps.value += 100; result = true; } return result; } Menu_view_state(Top_level_dialog::Name const &name, Ram_quota ram, Cap_quota caps) : _name(name), _initial_ram(ram), _initial_caps(caps) { } void gen_start_node(Xml_generator &) const; } _menu_view_state; Registry _gui_sessions { }; public: View(Sandboxed_runtime &runtime, Top_level_dialog &dialog) : Views::Element(runtime._views, dialog.name), _env(runtime._env), _alloc(runtime._alloc), _global_seq_number(runtime._global_seq_number), _optional_event_handler(runtime._optional_event_handler), _dialog(dialog), _menu_view_state(dialog.name, Ram_quota { 4*1024*1024 }, Cap_quota { 200 }) { } ~View(); void refresh() { _dialog_rom_session.trigger_update(); } }; template class Dialog::Sandboxed_runtime::Event_handler : Event_handler_base { private: Sandboxed_runtime &_runtime; T &_obj; void (T::*_member) (Event const &); void handle_event(Event const &event) override { (_obj.*_member)(event); } public: Event_handler(Sandboxed_runtime &runtime, T &obj, void (T::*member)(Event const &)) : _runtime(runtime), _obj(obj), _member(member) { /* register event handler at runtime */ _runtime._optional_event_handler._ptr = this; } ~Event_handler() { _runtime._optional_event_handler._ptr = nullptr; } }; #endif /* _INCLUDE__DIALOG__SANDBOXED_RUNTIME_H_ */