2011-12-22 16:19:25 +01:00
|
|
|
/*
|
2020-06-11 16:06:21 +02:00
|
|
|
* \brief GUI session interface
|
2011-12-22 16:19:25 +01:00
|
|
|
* \author Norman Feske
|
|
|
|
* \date 2006-08-10
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2017-02-20 13:23:52 +01:00
|
|
|
* Copyright (C) 2006-2017 Genode Labs GmbH
|
2011-12-22 16:19:25 +01:00
|
|
|
*
|
|
|
|
* This file is part of the Genode OS framework, which is distributed
|
2017-02-20 13:23:52 +01:00
|
|
|
* under the terms of the GNU Affero General Public License version 3.
|
2011-12-22 16:19:25 +01:00
|
|
|
*/
|
|
|
|
|
2020-06-11 14:27:20 +02:00
|
|
|
#ifndef _INCLUDE__GUI_SESSION__GUI_SESSION_H_
|
|
|
|
#define _INCLUDE__GUI_SESSION__GUI_SESSION_H_
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2024-08-07 17:07:43 +02:00
|
|
|
#include <base/id_space.h>
|
2011-12-22 16:19:25 +01:00
|
|
|
#include <session/session.h>
|
2014-06-06 17:26:53 +02:00
|
|
|
#include <os/surface.h>
|
2011-12-22 16:19:25 +01:00
|
|
|
#include <framebuffer_session/capability.h>
|
|
|
|
#include <input_session/capability.h>
|
|
|
|
|
2020-06-12 11:23:57 +02:00
|
|
|
namespace Gui {
|
Streamline exception types
This patch reduces the number of exception types by facilitating
globally defined exceptions for common usage patterns shared by most
services. In particular, RPC functions that demand a session-resource
upgrade not longer reflect this condition via a session-specific
exception but via the 'Out_of_ram' or 'Out_of_caps' types.
Furthermore, the 'Parent::Service_denied', 'Parent::Unavailable',
'Root::Invalid_args', 'Root::Unavailable', 'Service::Invalid_args',
'Service::Unavailable', and 'Local_service::Factory::Denied' types have
been replaced by the single 'Service_denied' exception type defined in
'session/session.h'.
This consolidation eases the error handling (there are fewer exceptions
to handle), alleviates the need to convert exceptions along the
session-creation call chain, and avoids possible aliasing problems
(catching the wrong type with the same name but living in a different
scope).
2017-05-07 22:03:22 +02:00
|
|
|
|
2024-06-10 16:10:45 +02:00
|
|
|
using namespace Genode;
|
Streamline exception types
This patch reduces the number of exception types by facilitating
globally defined exceptions for common usage patterns shared by most
services. In particular, RPC functions that demand a session-resource
upgrade not longer reflect this condition via a session-specific
exception but via the 'Out_of_ram' or 'Out_of_caps' types.
Furthermore, the 'Parent::Service_denied', 'Parent::Unavailable',
'Root::Invalid_args', 'Root::Unavailable', 'Service::Invalid_args',
'Service::Unavailable', and 'Local_service::Factory::Denied' types have
been replaced by the single 'Service_denied' exception type defined in
'session/session.h'.
This consolidation eases the error handling (there are fewer exceptions
to handle), alleviates the need to convert exceptions along the
session-creation call chain, and avoids possible aliasing problems
(catching the wrong type with the same name but living in a different
scope).
2017-05-07 22:03:22 +02:00
|
|
|
|
2016-11-06 14:27:26 +01:00
|
|
|
struct Session_client;
|
2014-06-06 17:26:53 +02:00
|
|
|
struct View;
|
2013-10-14 21:31:14 +02:00
|
|
|
struct Session;
|
Streamline exception types
This patch reduces the number of exception types by facilitating
globally defined exceptions for common usage patterns shared by most
services. In particular, RPC functions that demand a session-resource
upgrade not longer reflect this condition via a session-specific
exception but via the 'Out_of_ram' or 'Out_of_caps' types.
Furthermore, the 'Parent::Service_denied', 'Parent::Unavailable',
'Root::Invalid_args', 'Root::Unavailable', 'Service::Invalid_args',
'Service::Unavailable', and 'Local_service::Factory::Denied' types have
been replaced by the single 'Service_denied' exception type defined in
'session/session.h'.
This consolidation eases the error handling (there are fewer exceptions
to handle), alleviates the need to convert exceptions along the
session-creation call chain, and avoids possible aliasing problems
(catching the wrong type with the same name but living in a different
scope).
2017-05-07 22:03:22 +02:00
|
|
|
|
2024-08-07 17:07:43 +02:00
|
|
|
/*
|
|
|
|
* View capabilities are used as tokens to share views between sessions.
|
|
|
|
* There is no RPC interface associated with a view. View operations refer
|
|
|
|
* to views via session-local IDs.
|
|
|
|
*/
|
|
|
|
struct View : Interface { GENODE_RPC_INTERFACE(); };
|
|
|
|
struct View_ref : Interface { };
|
|
|
|
|
2024-06-10 16:10:45 +02:00
|
|
|
using View_capability = Capability<View>;
|
2024-08-09 14:27:26 +02:00
|
|
|
using View_ids = Id_space<View_ref>;
|
2024-08-08 17:38:36 +02:00
|
|
|
using View_id = Id_space<View_ref>::Id;
|
Streamline exception types
This patch reduces the number of exception types by facilitating
globally defined exceptions for common usage patterns shared by most
services. In particular, RPC functions that demand a session-resource
upgrade not longer reflect this condition via a session-specific
exception but via the 'Out_of_ram' or 'Out_of_caps' types.
Furthermore, the 'Parent::Service_denied', 'Parent::Unavailable',
'Root::Invalid_args', 'Root::Unavailable', 'Service::Invalid_args',
'Service::Unavailable', and 'Local_service::Factory::Denied' types have
been replaced by the single 'Service_denied' exception type defined in
'session/session.h'.
This consolidation eases the error handling (there are fewer exceptions
to handle), alleviates the need to convert exceptions along the
session-creation call chain, and avoids possible aliasing problems
(catching the wrong type with the same name but living in a different
scope).
2017-05-07 22:03:22 +02:00
|
|
|
|
2024-08-09 10:59:49 +02:00
|
|
|
using Title = String<64>;
|
2024-06-10 16:10:45 +02:00
|
|
|
using Rect = Surface_base::Rect;
|
|
|
|
using Point = Surface_base::Point;
|
|
|
|
using Area = Surface_base::Area;
|
2013-10-14 21:31:14 +02:00
|
|
|
}
|
2011-12-22 16:19:25 +01:00
|
|
|
|
|
|
|
|
2020-06-12 11:23:57 +02:00
|
|
|
struct Gui::Session : Genode::Session
|
2013-10-14 21:31:14 +02:00
|
|
|
{
|
2017-05-24 14:41:19 +02:00
|
|
|
/**
|
|
|
|
* \noapi
|
|
|
|
*/
|
2020-06-11 16:06:21 +02:00
|
|
|
static const char *service_name() { return "Gui"; }
|
2013-10-14 21:31:14 +02:00
|
|
|
|
2017-05-07 22:03:25 +02:00
|
|
|
/*
|
|
|
|
* A nitpicker session consumes a dataspace capability for the server's
|
|
|
|
* session-object allocation, a session capability, a dataspace capability
|
|
|
|
* for the command buffer, and the capabilities needed for the aggregated
|
|
|
|
* 'Framebuffer' and 'Input' sessions.
|
|
|
|
*/
|
2023-05-15 17:33:44 +02:00
|
|
|
static constexpr unsigned CAP_QUOTA = Framebuffer::Session::CAP_QUOTA
|
|
|
|
+ Input::Session::CAP_QUOTA + 3;
|
2017-05-07 22:03:25 +02:00
|
|
|
|
2014-06-06 17:26:53 +02:00
|
|
|
struct Command
|
|
|
|
{
|
2024-08-06 15:05:10 +02:00
|
|
|
enum Opcode { GEOMETRY, OFFSET, FRONT, BACK, FRONT_OF, BEHIND_OF,
|
|
|
|
BACKGROUND, TITLE, NOP };
|
2014-06-06 17:26:53 +02:00
|
|
|
|
2024-08-06 15:05:10 +02:00
|
|
|
struct Nop { static constexpr Opcode opcode = NOP; };
|
2014-06-06 17:26:53 +02:00
|
|
|
|
2024-08-06 15:05:10 +02:00
|
|
|
/**
|
|
|
|
* Operation that takes a view as first argument
|
|
|
|
*/
|
|
|
|
template <Opcode OPCODE>
|
|
|
|
struct View_op
|
2014-06-06 17:26:53 +02:00
|
|
|
{
|
2024-08-06 15:05:10 +02:00
|
|
|
static constexpr Opcode opcode = OPCODE;
|
2024-08-08 17:38:36 +02:00
|
|
|
View_id view;
|
2014-06-06 17:26:53 +02:00
|
|
|
};
|
|
|
|
|
2024-08-06 15:05:10 +02:00
|
|
|
struct Geometry : View_op<GEOMETRY> { Rect rect; };
|
|
|
|
struct Offset : View_op<OFFSET> { Point offset; };
|
|
|
|
struct Front : View_op<FRONT> { };
|
|
|
|
struct Back : View_op<BACK> { };
|
2024-08-08 17:38:36 +02:00
|
|
|
struct Front_of : View_op<FRONT_OF> { View_id neighbor; };
|
|
|
|
struct Behind_of : View_op<BEHIND_OF> { View_id neighbor; };
|
2024-08-06 15:05:10 +02:00
|
|
|
struct Background : View_op<BACKGROUND> { };
|
2024-08-09 10:59:49 +02:00
|
|
|
struct Title : View_op<TITLE> { Gui::Title title; };
|
2014-06-06 17:26:53 +02:00
|
|
|
|
|
|
|
Opcode opcode;
|
|
|
|
union
|
|
|
|
{
|
|
|
|
Nop nop;
|
|
|
|
Geometry geometry;
|
|
|
|
Offset offset;
|
2024-08-06 15:05:10 +02:00
|
|
|
Front front;
|
|
|
|
Back back;
|
|
|
|
Front_of front_of;
|
|
|
|
Behind_of behind_of;
|
2014-06-06 17:26:53 +02:00
|
|
|
Background background;
|
|
|
|
Title title;
|
|
|
|
};
|
|
|
|
|
2024-08-06 15:05:10 +02:00
|
|
|
Command() : opcode(Opcode::NOP) { }
|
2014-06-06 17:26:53 +02:00
|
|
|
|
|
|
|
template <typename ARGS>
|
2024-08-06 15:05:10 +02:00
|
|
|
Command(ARGS args) : opcode(ARGS::opcode)
|
2014-06-06 17:26:53 +02:00
|
|
|
{
|
|
|
|
reinterpret_cast<ARGS &>(nop) = args;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Command buffer shared between nitpicker and client
|
|
|
|
*/
|
|
|
|
class Command_buffer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
enum { MAX_COMMANDS = 64 };
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
unsigned _num = 0;
|
|
|
|
|
|
|
|
Command _commands[MAX_COMMANDS];
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
2016-05-11 18:21:47 +02:00
|
|
|
bool full() const { return _num >= MAX_COMMANDS; }
|
|
|
|
|
2014-06-06 17:26:53 +02:00
|
|
|
unsigned num() const
|
|
|
|
{
|
|
|
|
/* copy out _num value to avoid use-after-check problems */
|
|
|
|
unsigned const num = _num;
|
|
|
|
return num <= MAX_COMMANDS ? num : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void reset() { _num = 0; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enqueue command
|
|
|
|
*
|
|
|
|
* The command will be dropped if the buffer is full. Check for this
|
2016-05-11 18:21:47 +02:00
|
|
|
* condition by calling 'full()' prior calling this method.
|
2014-06-06 17:26:53 +02:00
|
|
|
*/
|
|
|
|
void enqueue(Command const &command)
|
|
|
|
{
|
2016-05-11 18:21:47 +02:00
|
|
|
if (!full())
|
2014-06-06 17:26:53 +02:00
|
|
|
_commands[_num++] = command;
|
|
|
|
}
|
|
|
|
|
|
|
|
Command get(unsigned i)
|
|
|
|
{
|
2024-06-10 16:10:45 +02:00
|
|
|
return (i < MAX_COMMANDS) ? _commands[i] : Command(Command::Nop());
|
2014-06-06 17:26:53 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-10-14 21:31:14 +02:00
|
|
|
virtual ~Session() { }
|
|
|
|
|
|
|
|
/**
|
2024-08-06 12:28:15 +02:00
|
|
|
* Request framebuffer RPC interface
|
2013-10-14 21:31:14 +02:00
|
|
|
*/
|
2024-08-06 12:28:15 +02:00
|
|
|
virtual Framebuffer::Session_capability framebuffer() = 0;
|
2013-10-14 21:31:14 +02:00
|
|
|
|
|
|
|
/**
|
2024-08-06 12:28:15 +02:00
|
|
|
* Request input RPC interface
|
2013-10-14 21:31:14 +02:00
|
|
|
*/
|
2024-08-06 12:28:15 +02:00
|
|
|
virtual Input::Session_capability input() = 0;
|
2013-10-14 21:31:14 +02:00
|
|
|
|
2024-08-09 14:27:26 +02:00
|
|
|
struct View_attr
|
|
|
|
{
|
|
|
|
Title title;
|
|
|
|
Rect rect;
|
|
|
|
bool front;
|
|
|
|
};
|
|
|
|
|
|
|
|
enum class View_result { OK, OUT_OF_RAM, OUT_OF_CAPS };
|
2024-08-05 13:47:22 +02:00
|
|
|
|
2013-10-14 21:31:14 +02:00
|
|
|
/**
|
2024-06-10 16:03:40 +02:00
|
|
|
* Create a new top-level view at the buffer
|
|
|
|
*/
|
2024-08-09 14:27:26 +02:00
|
|
|
virtual View_result view(View_id, View_attr const &) = 0;
|
2024-08-05 13:47:22 +02:00
|
|
|
|
2024-08-09 14:27:26 +02:00
|
|
|
enum class Child_view_result { OK, OUT_OF_RAM, OUT_OF_CAPS, INVALID };
|
2024-06-10 16:03:40 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new child view at the buffer
|
2013-10-14 21:31:14 +02:00
|
|
|
*
|
2014-02-14 10:36:04 +01:00
|
|
|
* The 'parent' argument allows the client to use the location of an
|
|
|
|
* existing view as the coordinate origin for the to-be-created view.
|
2013-10-14 21:31:14 +02:00
|
|
|
*/
|
2024-08-09 14:27:26 +02:00
|
|
|
virtual Child_view_result child_view(View_id, View_id parent, View_attr const &) = 0;
|
2013-10-14 21:31:14 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Destroy view
|
|
|
|
*/
|
2024-08-08 17:38:36 +02:00
|
|
|
virtual void destroy_view(View_id) = 0;
|
2013-10-14 21:31:14 +02:00
|
|
|
|
2024-08-14 14:43:37 +02:00
|
|
|
enum class Associate_result { OK, OUT_OF_RAM, OUT_OF_CAPS, INVALID };
|
2024-08-06 16:44:04 +02:00
|
|
|
|
|
|
|
/**
|
2024-08-14 14:43:37 +02:00
|
|
|
* Associate existing view with the specified ID
|
2024-08-06 16:44:04 +02:00
|
|
|
*/
|
2024-08-14 14:43:37 +02:00
|
|
|
virtual Associate_result associate(View_id, View_capability) = 0;
|
2014-06-06 17:26:53 +02:00
|
|
|
|
2024-08-14 15:14:33 +02:00
|
|
|
enum class View_capability_error { OUT_OF_RAM, OUT_OF_CAPS };
|
|
|
|
using View_capability_result = Attempt<View_capability, View_capability_error>;
|
|
|
|
|
2014-06-06 17:26:53 +02:00
|
|
|
/**
|
2024-08-08 17:38:36 +02:00
|
|
|
* Request view capability for a given ID
|
2014-06-06 17:26:53 +02:00
|
|
|
*
|
|
|
|
* The returned view capability can be used to share the view with another
|
|
|
|
* session.
|
|
|
|
*/
|
2024-08-14 15:14:33 +02:00
|
|
|
virtual View_capability_result view_capability(View_id) = 0;
|
2014-06-06 17:26:53 +02:00
|
|
|
|
|
|
|
/**
|
2024-08-08 17:38:36 +02:00
|
|
|
* Release session-local view ID
|
2014-06-06 17:26:53 +02:00
|
|
|
*/
|
2024-08-08 17:38:36 +02:00
|
|
|
virtual void release_view_id(View_id) = 0;
|
2014-06-06 17:26:53 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Request dataspace used for issuing view commands to nitpicker
|
|
|
|
*/
|
2024-06-10 16:10:45 +02:00
|
|
|
virtual Dataspace_capability command_dataspace() = 0;
|
2014-06-06 17:26:53 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Execution batch of commands contained in the command dataspace
|
|
|
|
*/
|
|
|
|
virtual void execute() = 0;
|
2013-10-14 21:31:14 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return physical screen mode
|
|
|
|
*/
|
|
|
|
virtual Framebuffer::Mode mode() = 0;
|
|
|
|
|
2014-07-09 10:34:02 +02:00
|
|
|
/**
|
|
|
|
* Register signal handler to be notified about mode changes
|
|
|
|
*/
|
2024-06-10 16:10:45 +02:00
|
|
|
virtual void mode_sigh(Signal_context_capability) = 0;
|
2014-07-09 10:34:02 +02:00
|
|
|
|
2024-08-05 13:47:22 +02:00
|
|
|
enum class Buffer_result { OK, OUT_OF_RAM, OUT_OF_CAPS };
|
|
|
|
|
2013-10-14 21:31:14 +02:00
|
|
|
/**
|
|
|
|
* Define dimensions of virtual framebuffer
|
|
|
|
*/
|
2024-08-05 13:47:22 +02:00
|
|
|
virtual Buffer_result buffer(Framebuffer::Mode mode, bool use_alpha) = 0;
|
2013-10-14 21:31:14 +02:00
|
|
|
|
2014-06-04 15:58:49 +02:00
|
|
|
/**
|
|
|
|
* Set focused session
|
|
|
|
*
|
2018-04-05 19:50:56 +02:00
|
|
|
* Normally, the focused session is defined by the 'focus' ROM, which is
|
|
|
|
* driven by an external policy component. However, in cases where one
|
|
|
|
* application consists of multiple nitpicker sessions, it is desirable to
|
|
|
|
* let the application manage the focus among its sessions by applying an
|
|
|
|
* application-local policy. The 'focus' RPC function allows a client to
|
|
|
|
* specify another client that should receive the focus whenever the
|
|
|
|
* session becomes focused. As the designated receiver of the focus is
|
|
|
|
* referred to by its session capability, a common parent can manage the
|
|
|
|
* focus among its children. But unrelated sessions cannot interfere.
|
2014-06-04 15:58:49 +02:00
|
|
|
*/
|
2024-06-10 16:10:45 +02:00
|
|
|
virtual void focus(Capability<Session> focused) = 0;
|
2014-10-04 18:50:06 +02:00
|
|
|
|
2013-10-14 21:31:14 +02:00
|
|
|
/**
|
|
|
|
* Return number of bytes needed for virtual framebuffer of specified size
|
|
|
|
*/
|
|
|
|
static size_t ram_quota(Framebuffer::Mode mode, bool use_alpha)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* If alpha blending is used, each pixel requires an additional byte
|
|
|
|
* for the alpha value and a byte holding the input mask.
|
2011-12-22 16:19:25 +01:00
|
|
|
*/
|
2020-06-16 15:46:59 +02:00
|
|
|
return (mode.bytes_per_pixel() + 2*use_alpha)*mode.area.count();
|
2013-10-14 21:31:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************
|
|
|
|
** RPC declaration **
|
|
|
|
*********************/
|
|
|
|
|
2024-08-06 12:28:15 +02:00
|
|
|
GENODE_RPC(Rpc_framebuffer, Framebuffer::Session_capability, framebuffer);
|
|
|
|
GENODE_RPC(Rpc_input, Input::Session_capability, input);
|
2024-08-09 14:27:26 +02:00
|
|
|
GENODE_RPC(Rpc_view, View_result, view, View_id, View_attr const &);
|
|
|
|
GENODE_RPC(Rpc_child_view, Child_view_result, child_view, View_id, View_id, View_attr const &);
|
2024-08-08 17:38:36 +02:00
|
|
|
GENODE_RPC(Rpc_destroy_view, void, destroy_view, View_id);
|
2024-08-14 14:43:37 +02:00
|
|
|
GENODE_RPC(Rpc_associate, Associate_result, associate, View_id, View_capability);
|
2024-08-14 15:14:33 +02:00
|
|
|
GENODE_RPC(Rpc_view_capability, View_capability_result, view_capability, View_id);
|
2024-08-08 17:38:36 +02:00
|
|
|
GENODE_RPC(Rpc_release_view_id, void, release_view_id, View_id);
|
2024-06-10 16:10:45 +02:00
|
|
|
GENODE_RPC(Rpc_command_dataspace, Dataspace_capability, command_dataspace);
|
2014-06-06 17:26:53 +02:00
|
|
|
GENODE_RPC(Rpc_execute, void, execute);
|
2013-10-14 21:31:14 +02:00
|
|
|
GENODE_RPC(Rpc_background, int, background, View_capability);
|
|
|
|
GENODE_RPC(Rpc_mode, Framebuffer::Mode, mode);
|
2024-06-10 16:10:45 +02:00
|
|
|
GENODE_RPC(Rpc_mode_sigh, void, mode_sigh, Signal_context_capability);
|
|
|
|
GENODE_RPC(Rpc_focus, void, focus, Capability<Session>);
|
2024-08-05 13:47:22 +02:00
|
|
|
GENODE_RPC(Rpc_buffer, Buffer_result, buffer, Framebuffer::Mode, bool);
|
2013-10-14 21:31:14 +02:00
|
|
|
|
2024-08-06 12:28:15 +02:00
|
|
|
GENODE_RPC_INTERFACE(Rpc_framebuffer, Rpc_input,
|
2024-08-14 14:43:37 +02:00
|
|
|
Rpc_view, Rpc_child_view, Rpc_destroy_view, Rpc_associate,
|
2024-08-08 17:38:36 +02:00
|
|
|
Rpc_view_capability, Rpc_release_view_id,
|
2016-05-11 17:23:51 +02:00
|
|
|
Rpc_command_dataspace, Rpc_execute, Rpc_mode,
|
2024-05-30 14:23:41 +02:00
|
|
|
Rpc_mode_sigh, Rpc_buffer, Rpc_focus);
|
2013-10-14 21:31:14 +02:00
|
|
|
};
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2020-06-11 14:27:20 +02:00
|
|
|
#endif /* _INCLUDE__GUI_SESSION__GUI_SESSION_H_ */
|