mirror of
https://github.com/genodelabs/genode.git
synced 2025-04-06 02:47:07 +00:00
vbox6: multi-monitor support
Support dynamic panorama of virtual displays. Virtual monitors for VirtualBox must be configured in two steps. - <Display monitorCount="<num>"> nodes in the .vbox file defines the number of _connectors_ and, thus, the maximum number of connected virtual monitors. - Each enabled VirtualBox monitor requests a dedicated GUI session. Enablement and labeling of these sessions is done via <monitor> nodes in the component configuration like follows. <monitor label="DP-1"/> <monitor label="HDMI-A-1"/> Labels can be used by the window manager to configure and place the corresponding window. The order of the nodes directly controls the connection order at the virtual graphics card. Fixes #5449
This commit is contained in:
parent
cec3a82401
commit
2cdcacf2fc
@ -4,15 +4,15 @@ VirtualBox configuration options
|
||||
The configuration requires an attribute named vbox_file with the name of the
|
||||
vbox configuration to be used (.vbox).
|
||||
|
||||
<config vbox_file="file.vbox">
|
||||
! <config vbox_file="file.vbox">
|
||||
|
||||
XHCI controller
|
||||
~~~~~~~~~~~~~~~
|
||||
|
||||
The virtual XHCI controller can be enabled with the following
|
||||
configuration option:
|
||||
configuration option.
|
||||
|
||||
<config xhci="yes">
|
||||
! <config xhci="yes">
|
||||
|
||||
CapsLock
|
||||
~~~~~~~~
|
||||
@ -22,7 +22,7 @@ operating systems. As guest and host state may diverge, the event
|
||||
filter optionally reports the capslock state as ROM, which can be
|
||||
monitored by VirtualBox if configured like follows.
|
||||
|
||||
<config capslock="rom">
|
||||
! <config capslock="rom">
|
||||
|
||||
VirtualBox requests a "capslock" ROM, which is interpreted as XML. The
|
||||
top-level node attribute 'enabled' is expected boolean and represents
|
||||
@ -30,3 +30,20 @@ the CapsLock state. If ROM and VM-internal CapsLock state differ, the
|
||||
VMM sends aritifical CapsLock key events to the VM. In this mode,
|
||||
incoming KEY_CAPSLOCK input events a dropped and not sent to the guest
|
||||
to prevent the state divergence.
|
||||
|
||||
Monitors
|
||||
~~~~~~~~
|
||||
|
||||
Virtual monitors for VirtualBox must be configured in two steps. First, the
|
||||
'<Display monitorCount="<num>">' node in the .vbox file defines the number
|
||||
of _connectors_ and, thus, the maximum number of connected virtual monitors.
|
||||
Each enabled VirtualBox monitor requests a dedicated GUI session. Enablement
|
||||
and labeling of these sessions is done via '<monitor>' nodes in the
|
||||
component configuration like follows.
|
||||
|
||||
! <monitor label="DP-1"/>
|
||||
! <monitor label="HDMI-A-1"/>
|
||||
|
||||
The label can be used by the window manager to configure and place the
|
||||
corresponding window. The order of the nodes directly controls the
|
||||
connection order at the virtual graphics card.
|
||||
|
@ -1,11 +1,12 @@
|
||||
/*
|
||||
* \brief Virtualbox framebuffer implementation for Genode
|
||||
* \author Alexander Boettcher
|
||||
* \author Christian Helmuth
|
||||
* \date 2013-10-16
|
||||
*/
|
||||
|
||||
/*
|
||||
* Copyright (C) 2013-2017 Genode Labs GmbH
|
||||
* Copyright (C) 2013-2024 Genode Labs GmbH
|
||||
*
|
||||
* This file is distributed under the terms of the GNU General Public License
|
||||
* version 2.
|
||||
@ -25,6 +26,9 @@
|
||||
#include <VirtualBoxBase.h>
|
||||
#include <DisplayWrap.h>
|
||||
|
||||
/* Genode port specific includes */
|
||||
#include <attempt.h>
|
||||
|
||||
class Genodefb :
|
||||
VBOX_SCRIPTABLE_IMPL(IFramebuffer)
|
||||
{
|
||||
@ -58,6 +62,8 @@ class Genodefb :
|
||||
ComPtr<IDisplay> _display;
|
||||
ComPtr<IDisplaySourceBitmap> _display_bitmap;
|
||||
|
||||
unsigned const _id;
|
||||
|
||||
void _clear_screen()
|
||||
{
|
||||
if (!_fb_base) return;
|
||||
@ -85,15 +91,15 @@ class Genodefb :
|
||||
|
||||
NS_DECL_ISUPPORTS
|
||||
|
||||
Genodefb(Genode::Env &env, Gui::Connection &gui, ComPtr<IDisplay> const &display)
|
||||
Genodefb(Genode::Env &env, Gui::Connection &gui, ComPtr<IDisplay> const &display, unsigned id)
|
||||
:
|
||||
_env(env),
|
||||
_gui(gui),
|
||||
_virtual_fb_area(_initial_setup()),
|
||||
_display(display)
|
||||
_display(display), _id(id)
|
||||
{
|
||||
int rc = RTCritSectInit(&_fb_lock);
|
||||
Assert(rc == VINF_SUCCESS); (void)rc;
|
||||
attempt([&] () { return RTCritSectInit(&_fb_lock); },
|
||||
"unable to initialize critsect");
|
||||
}
|
||||
|
||||
virtual ~Genodefb() { }
|
||||
@ -127,7 +133,7 @@ class Genodefb :
|
||||
return Global::vboxStatusCodeToCOM(RTCritSectLeave(&_fb_lock));
|
||||
}
|
||||
|
||||
STDMETHODIMP NotifyChange(PRUint32 screen, PRUint32, PRUint32,
|
||||
STDMETHODIMP NotifyChange(PRUint32 screen, PRUint32 ox, PRUint32 oy,
|
||||
PRUint32 w, PRUint32 h) override
|
||||
{
|
||||
HRESULT result = E_FAIL;
|
||||
@ -147,7 +153,7 @@ class Genodefb :
|
||||
Genode::log("fb resize : [", screen, "] ",
|
||||
_virtual_fb_area, " -> ",
|
||||
w, "x", h,
|
||||
" (host: ", _gui_win.area, ")");
|
||||
" (host: ", _gui_win.area, ") origin: ", ox, ",", oy);
|
||||
|
||||
if ((w < (ULONG)_gui_win.area.w) ||
|
||||
(h < (ULONG)_gui_win.area.h)) {
|
||||
@ -162,7 +168,7 @@ class Genodefb :
|
||||
Genode::log("fb resize : [", screen, "] ",
|
||||
_virtual_fb_area, " -> ",
|
||||
w, "x", h, " ignored"
|
||||
" (host: ", _gui_win.area, ")");
|
||||
" (host: ", _gui_win.area, ") origin: ", ox, ",", oy);
|
||||
}
|
||||
|
||||
Unlock();
|
||||
@ -217,7 +223,7 @@ class Genodefb :
|
||||
|
||||
Gui::Area const area_fb = Gui::Area(_gui_win.area.w,
|
||||
_gui_win.area.h);
|
||||
Gui::Area const area_vm = Gui::Area(ulWidth, ulHeight);
|
||||
Gui::Area const area_vm = Gui::Area(ulBytesPerLine/(ulBitsPerPixel/8), ulHeight);
|
||||
|
||||
using namespace Genode;
|
||||
|
||||
@ -227,13 +233,12 @@ class Genodefb :
|
||||
Texture<Pixel_src> texture((Pixel_src *)pAddress, nullptr, area_vm);
|
||||
Surface<Pixel_dst> surface((Pixel_dst *)_fb_base, area_fb);
|
||||
|
||||
surface.clip(Surface_base::Rect(Surface_base::Point(o_x, o_y),
|
||||
Surface_base::Area(width, height)));
|
||||
surface.clip(Gui::Rect(Gui::Point(o_x, o_y), Gui::Area(width, height)));
|
||||
|
||||
Texture_painter::paint(surface,
|
||||
texture,
|
||||
Genode::Color(0, 0, 0),
|
||||
Surface_base::Point(0, 0),
|
||||
Gui::Point(0, 0),
|
||||
Texture_painter::SOLID,
|
||||
false);
|
||||
|
||||
|
@ -6,7 +6,7 @@
|
||||
*/
|
||||
|
||||
/*
|
||||
* Copyright (C) 2020-2021 Genode Labs GmbH
|
||||
* Copyright (C) 2020-2024 Genode Labs GmbH
|
||||
*
|
||||
* This file is distributed under the terms of the GNU General Public License
|
||||
* version 2.
|
||||
@ -20,6 +20,8 @@
|
||||
|
||||
struct Input_adapter
|
||||
{
|
||||
using Point = Gui::Point;
|
||||
|
||||
struct Mouse
|
||||
{
|
||||
ComPtr<IMouse> _imouse;
|
||||
@ -32,8 +34,6 @@ struct Input_adapter
|
||||
|
||||
bool _key_status[Input::KEY_MAX + 1];
|
||||
|
||||
using Point = Genode::Surface_base::Point;
|
||||
|
||||
Point _abs_pos { 0, 0 };
|
||||
|
||||
bool _absolute { false };
|
||||
@ -47,7 +47,7 @@ struct Input_adapter
|
||||
|| keycode == Input::BTN_EXTRA;
|
||||
}
|
||||
|
||||
void handle_input_event(Input::Event const &);
|
||||
void handle_input_event(Input::Event const &, Point);
|
||||
|
||||
void absolute(bool absolute) { _absolute = absolute; }
|
||||
|
||||
@ -70,7 +70,7 @@ struct Input_adapter
|
||||
Input_adapter(ComPtr<IConsole> &iconsole)
|
||||
: _mouse(iconsole), _keyboard(iconsole) { }
|
||||
|
||||
void handle_input_event(Input::Event const &);
|
||||
void handle_input_event(Input::Event const &, Point origin = { 0, 0});
|
||||
|
||||
void mouse_absolute(bool absolute) { _mouse.absolute(absolute); }
|
||||
};
|
||||
@ -101,7 +101,7 @@ void Input_adapter::Keyboard::handle_input_event(Input::Event const &ev)
|
||||
}
|
||||
|
||||
|
||||
void Input_adapter::Mouse::handle_input_event(Input::Event const &ev)
|
||||
void Input_adapter::Mouse::handle_input_event(Input::Event const &ev, Point origin)
|
||||
{
|
||||
/* obtain bit mask of currently pressed mouse buttons */
|
||||
auto curr_mouse_button_bits = [&] () {
|
||||
@ -124,7 +124,7 @@ void Input_adapter::Mouse::handle_input_event(Input::Event const &ev)
|
||||
_key_status[key] = false; });
|
||||
|
||||
ev.handle_absolute_motion([&] (int ax, int ay) {
|
||||
_abs_pos = Point(ax, ay); });
|
||||
_abs_pos = origin + Point(ax, ay); });
|
||||
|
||||
int wheel_x = 0, wheel_y = 0;
|
||||
ev.handle_wheel([&] (int x, int y) { wheel_x = -x; wheel_y = -y; });
|
||||
@ -147,11 +147,11 @@ void Input_adapter::Mouse::handle_input_event(Input::Event const &ev)
|
||||
}
|
||||
|
||||
|
||||
void Input_adapter::handle_input_event(Input::Event const &ev)
|
||||
void Input_adapter::handle_input_event(Input::Event const &ev, Point origin)
|
||||
{
|
||||
/* present the event to potential consumers */
|
||||
_keyboard.handle_input_event(ev);
|
||||
_mouse.handle_input_event(ev);
|
||||
_mouse.handle_input_event(ev, origin);
|
||||
}
|
||||
|
||||
#endif /* _INPUT_ADAPTER_H_ */
|
||||
|
@ -7,7 +7,7 @@
|
||||
*/
|
||||
|
||||
/*
|
||||
* Copyright (C) 2013-2021 Genode Labs GmbH
|
||||
* Copyright (C) 2013-2024 Genode Labs GmbH
|
||||
*
|
||||
* This file is distributed under the terms of the GNU General Public License
|
||||
* version 2.
|
||||
@ -83,6 +83,16 @@ struct Main : Event_handler
|
||||
|
||||
Attached_rom_dataspace _config { _env, "config" };
|
||||
|
||||
Signal_handler<Main> _config_handler { _env.ep(), *this, &Main::_handle_config };
|
||||
|
||||
void _handle_config()
|
||||
{
|
||||
_config.update();
|
||||
Libc::with_libc([&] { _update_monitors(); });
|
||||
}
|
||||
|
||||
void _update_monitors();
|
||||
|
||||
struct Vbox_file_path
|
||||
{
|
||||
using Path = String<128>;
|
||||
@ -277,7 +287,63 @@ struct Main : Event_handler
|
||||
|
||||
Signal_handler<Main> _exit_handler { _env.ep(), *this, &Main::_handle_exit };
|
||||
|
||||
Registry<Registered<Gui::Connection>> _gui_connections { };
|
||||
struct Monitor : Registry<Monitor>::Element
|
||||
{
|
||||
using Label = String<32>;
|
||||
|
||||
struct Fb
|
||||
{
|
||||
unsigned const id;
|
||||
ComPtr<IDisplay> const display;
|
||||
ComPtr<Genodefb> const fb;
|
||||
|
||||
Bstr guid { };
|
||||
|
||||
Fb(Env &, Gui::Connection &, ComPtr<IDisplay> const &, unsigned id);
|
||||
~Fb();
|
||||
|
||||
int w() const { return fb->w(); }
|
||||
int h() const { return fb->h(); }
|
||||
|
||||
void update_mode(Gui::Rect gui_win) { fb->update_mode(gui_win); }
|
||||
};
|
||||
|
||||
Env &env;
|
||||
|
||||
Signal_context_capability const input_sigh;
|
||||
Signal_context_capability const mode_sigh;
|
||||
ComPtr<IDisplay> const display;
|
||||
unsigned const id;
|
||||
|
||||
Label label { };
|
||||
|
||||
Constructible<Gui::Connection> gui;
|
||||
Constructible<Fb> fb;
|
||||
|
||||
Gui::Rect rect { };
|
||||
|
||||
Monitor(Registry<Monitor> &, Env &,
|
||||
Signal_context_capability input_sigh,
|
||||
Signal_context_capability mode_sigh,
|
||||
ComPtr<IDisplay>, unsigned id, Label const &);
|
||||
~Monitor();
|
||||
|
||||
void update(Label const &label);
|
||||
};
|
||||
|
||||
Registry<Monitor> _monitors { };
|
||||
|
||||
Mutex _monitor_change_mutex;
|
||||
|
||||
struct Monitor_change_event
|
||||
{
|
||||
unsigned id;
|
||||
int origin_x, origin_y;
|
||||
} _monitor_change_event;
|
||||
|
||||
Signal_handler<Main> _monitor_handler { _env.ep(), *this, &Main::_handle_monitor };
|
||||
|
||||
void _handle_monitor();
|
||||
|
||||
Signal_handler<Main> _input_handler { _env.ep(), *this, &Main::_handle_input };
|
||||
|
||||
@ -289,30 +355,6 @@ struct Main : Event_handler
|
||||
|
||||
Input_adapter _input_adapter { _iconsole };
|
||||
|
||||
bool const _genode_gui_attached = ( _attach_genode_gui(), true );
|
||||
|
||||
void _attach_genode_gui()
|
||||
{
|
||||
Monitor_count const num_monitors = _machine.monitor_count();
|
||||
|
||||
for (unsigned i = 0; i < num_monitors.value; i++) {
|
||||
|
||||
String<3> label { i };
|
||||
|
||||
Gui::Connection &gui = *new Registered<Gui::Connection>(_gui_connections, _env, label.string());
|
||||
|
||||
gui.input.sigh(_input_handler);
|
||||
gui.info_sigh(_fb_mode_handler);
|
||||
|
||||
Genodefb *fb = new Genodefb(_env, gui, _idisplay);
|
||||
|
||||
Bstr fb_id { };
|
||||
|
||||
attempt([&] () { return _idisplay->AttachFramebuffer(i, fb, fb_id.asOutParam()); },
|
||||
"unable to attach framebuffer to virtual monitor ", i);
|
||||
}
|
||||
}
|
||||
|
||||
bool const _machine_powered_up = ( _power_up_machine(), true );
|
||||
|
||||
void _power_up_machine()
|
||||
@ -352,9 +394,7 @@ struct Main : Event_handler
|
||||
{
|
||||
_capslock.destruct_rom();
|
||||
|
||||
_gui_connections.for_each([&] (Gui::Connection &gui) {
|
||||
delete &gui;
|
||||
});
|
||||
_monitors.for_each([&] (Monitor &m) { delete &m; });
|
||||
|
||||
/* signal exit to main entrypoint */
|
||||
_exit_handler.local_submit();
|
||||
@ -382,12 +422,18 @@ struct Main : Event_handler
|
||||
event_types.push_back(VBoxEventType_OnKeyboardLedsChanged);
|
||||
event_types.push_back(VBoxEventType_OnStateChanged);
|
||||
event_types.push_back(VBoxEventType_OnAdditionsStateChanged);
|
||||
event_types.push_back(VBoxEventType_OnGuestMonitorChanged);
|
||||
if (0) /* activate for all events on debugging */
|
||||
event_types.push_back(VBoxEventType_Any);
|
||||
|
||||
ievent_source->RegisterListener(listener, ComSafeArrayAsInParam(event_types), true);
|
||||
}
|
||||
|
||||
Main(Genode::Env &env) : _env(env)
|
||||
{
|
||||
_config.sigh(_config_handler);
|
||||
_handle_config();
|
||||
|
||||
/*
|
||||
* Explicitly, adapt to current framebuffer/window size after
|
||||
* initialization finished. This ensures the use of the correct
|
||||
@ -398,6 +444,111 @@ struct Main : Event_handler
|
||||
};
|
||||
|
||||
|
||||
Main::Monitor::Fb::Fb(Env &env, Gui::Connection &gui,
|
||||
ComPtr<IDisplay> const &display, unsigned id)
|
||||
:
|
||||
id(id), display(display), fb(new Genodefb(env, gui, display, id))
|
||||
{
|
||||
display->AttachFramebuffer(id, fb, guid.asOutParam());
|
||||
}
|
||||
|
||||
|
||||
Main::Monitor::Fb::~Fb()
|
||||
{
|
||||
display->DetachFramebuffer(id, guid.raw());
|
||||
}
|
||||
|
||||
|
||||
void Main::Monitor::update(Label const &new_label)
|
||||
{
|
||||
if (new_label == label) return;
|
||||
|
||||
label = new_label;
|
||||
|
||||
fb.destruct();
|
||||
|
||||
gui.construct(env, label);
|
||||
gui->input.sigh(input_sigh);
|
||||
gui->info_sigh(mode_sigh);
|
||||
|
||||
fb.construct(env, *gui, display, id);
|
||||
|
||||
int origin_x, origin_y;
|
||||
unsigned w, h, bpp;
|
||||
GuestMonitorStatus_T monitor_status;
|
||||
display->GetScreenResolution(id, &w, &h, &bpp, &origin_x, &origin_y, &monitor_status);
|
||||
|
||||
rect = Gui::Rect({ origin_x, origin_y }, { w, h });
|
||||
|
||||
display->SetVideoModeHint(id, true, false, 0, 0, rect.area.w, rect.area.h, 32, true);
|
||||
}
|
||||
|
||||
|
||||
Main::Monitor::Monitor(Registry<Monitor> ®istry, Env &env,
|
||||
Signal_context_capability input_sigh,
|
||||
Signal_context_capability mode_sigh,
|
||||
ComPtr<IDisplay> display, unsigned id, Label const &label)
|
||||
:
|
||||
Registry<Monitor>::Element(registry, *this), env(env),
|
||||
input_sigh(input_sigh), mode_sigh(mode_sigh), display(display), id(id)
|
||||
{
|
||||
update(label);
|
||||
}
|
||||
|
||||
|
||||
Main::Monitor::~Monitor()
|
||||
{
|
||||
display->SetVideoModeHint(id, false, false, 0, 0, 0, 0, 0, true);
|
||||
}
|
||||
|
||||
|
||||
/* must be called in Libc::with_libc() context */
|
||||
void Main::_update_monitors()
|
||||
{
|
||||
if (!_config.xml().has_sub_node("monitor"))
|
||||
warning("no <monitor label=\"...\"/> config node found - running headless");
|
||||
|
||||
unsigned const max_id = _machine.monitor_count().value - 1;
|
||||
|
||||
unsigned id = 0;
|
||||
|
||||
/* handle new and updated monitors */
|
||||
_config.xml().for_each_sub_node("monitor", [&] (Xml_node const &xml) {
|
||||
|
||||
Monitor::Label label = xml.attribute_value("label", Monitor::Label(""));
|
||||
|
||||
if (id > max_id) {
|
||||
warning("ignoring ", xml, " for monitor id=", id, " max=", max_id,
|
||||
" (monitorCount in vbox file!)");
|
||||
return;
|
||||
}
|
||||
|
||||
bool updated = false;
|
||||
|
||||
/* update */
|
||||
_monitors.for_each([&] (Monitor &m) {
|
||||
if (m.id != id) return;
|
||||
|
||||
m.update(label);
|
||||
updated = true;
|
||||
});
|
||||
|
||||
/* create new monitor */
|
||||
if (!updated)
|
||||
new Monitor(_monitors, _env, _input_handler, _fb_mode_handler,
|
||||
_idisplay, id, label);
|
||||
|
||||
++id;
|
||||
});
|
||||
|
||||
/* disable excess monitors */
|
||||
_monitors.for_each([&] (Monitor &m) {
|
||||
if (m.id >= id) delete &m; });
|
||||
|
||||
_fb_mode_handler.local_submit();
|
||||
}
|
||||
|
||||
|
||||
/* must be called in Libc::with_libc() context */
|
||||
void Main::_sync_capslock()
|
||||
{
|
||||
@ -408,9 +559,29 @@ void Main::_sync_capslock()
|
||||
}
|
||||
|
||||
|
||||
void Main::_handle_monitor()
|
||||
{
|
||||
Monitor_change_event ev;
|
||||
{
|
||||
Mutex::Guard guard(_monitor_change_mutex);
|
||||
ev = _monitor_change_event;
|
||||
_monitor_change_event.id = ~0u;
|
||||
}
|
||||
|
||||
if (ev.id == ~0u)
|
||||
return;
|
||||
|
||||
_monitors.for_each([&] (Monitor &m) {
|
||||
if (m.id != ev.id) return;
|
||||
|
||||
m.rect = { { ev.origin_x, ev.origin_y }, m.rect.area };
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
void Main::_handle_input()
|
||||
{
|
||||
auto handle_one_event = [&] (Input::Event const &ev) {
|
||||
auto handle_one_event = [&] (Input::Event const &ev, Gui::Point origin) {
|
||||
/* don't confuse guests and drop CapsLock events in ROM mode */
|
||||
if (_capslock.mode == Capslock::Mode::ROM) {
|
||||
if (ev.key_press(Input::KEY_CAPSLOCK)
|
||||
@ -418,43 +589,44 @@ void Main::_handle_input()
|
||||
return;
|
||||
}
|
||||
|
||||
_input_adapter.handle_input_event(ev);
|
||||
_input_adapter.handle_input_event(ev, origin);
|
||||
};
|
||||
|
||||
Libc::with_libc([&] {
|
||||
_gui_connections.for_each([&] (Gui::Connection &gui) {
|
||||
gui.input.for_each_event([&] (Input::Event const &ev) {
|
||||
handle_one_event(ev); }); }); });
|
||||
_monitors.for_each([&] (Monitor &m) {
|
||||
m.gui->input.for_each_event([&] (Input::Event const &ev) {
|
||||
handle_one_event(ev, m.rect.at); }); }); });
|
||||
}
|
||||
|
||||
|
||||
void Main::_handle_fb_mode()
|
||||
{
|
||||
Libc::with_libc([&] {
|
||||
_gui_connections.for_each([&] (Gui::Connection &gui) {
|
||||
IFramebuffer *pFramebuffer = NULL;
|
||||
HRESULT rc = _idisplay->QueryFramebuffer(0, &pFramebuffer);
|
||||
Assert(SUCCEEDED(rc) && pFramebuffer); (void)rc;
|
||||
_monitors.for_each([&] (Monitor &m) {
|
||||
|
||||
Genodefb *fb = dynamic_cast<Genodefb *>(pFramebuffer);
|
||||
|
||||
Gui::Rect const gui_win = gui.window().convert<Gui::Rect>(
|
||||
Gui::Rect const gui_win = m.gui->window().convert<Gui::Rect>(
|
||||
[&] (Gui::Rect rect) { return rect; },
|
||||
[&] (Gui::Undefined) { return Gui::Rect { { }, { 1024, 768 } }; });
|
||||
|
||||
fb->update_mode(gui_win);
|
||||
m.rect = Gui::Rect(m.rect.at, gui_win.area);
|
||||
|
||||
if ((fb->w() <= 1) && (fb->h() <= 1)) {
|
||||
m.fb->update_mode(gui_win);
|
||||
|
||||
if ((m.fb->w() <= 1) && (m.fb->h() <= 1)) {
|
||||
/* interpret a size of 0x0 as indication to quit VirtualBox */
|
||||
if (_iconsole->PowerButton() != S_OK)
|
||||
Genode::error("ACPI shutdown failed");
|
||||
return;
|
||||
}
|
||||
|
||||
_idisplay->SetVideoModeHint(0 /*=display*/,
|
||||
/*
|
||||
* XXX May changeOrigin and originX/originY be used to hint guest
|
||||
* about panorama config?
|
||||
*/
|
||||
_idisplay->SetVideoModeHint(m.id /*=display*/,
|
||||
true /*=enabled*/, false /*=changeOrigin*/,
|
||||
0 /*=originX*/, 0 /*=originY*/,
|
||||
fb->w(), fb->h(),
|
||||
m.fb->w(), m.fb->h(),
|
||||
32, true);
|
||||
});
|
||||
});
|
||||
@ -540,7 +712,37 @@ void Main::handle_vbox_event(VBoxEventType_T ev_type, IEvent &ev)
|
||||
_sync_capslock();
|
||||
} break;
|
||||
|
||||
default: /* ignore other events */ break;
|
||||
case VBoxEventType_OnGuestMonitorChanged:
|
||||
{
|
||||
ComPtr<IGuestMonitorChangedEvent> mon_ev = &ev;
|
||||
GuestMonitorChangedEventType_T change_type;
|
||||
unsigned screen_id, origin_x, origin_y, width, height;
|
||||
mon_ev->COMGETTER(ChangeType)(&change_type);
|
||||
mon_ev->COMGETTER(ScreenId)(&screen_id);
|
||||
mon_ev->COMGETTER(OriginX)(&origin_x);
|
||||
mon_ev->COMGETTER(OriginY)(&origin_y);
|
||||
mon_ev->COMGETTER(Width)(&width);
|
||||
mon_ev->COMGETTER(Height)(&height);
|
||||
|
||||
unsigned bpp;
|
||||
GuestMonitorStatus_T gms;
|
||||
int ox, oy;
|
||||
_idisplay->GetScreenResolution(screen_id, &width, &height, &bpp, &ox, &oy, &gms);
|
||||
|
||||
/*
|
||||
* Prevent deadlock in VMMDev's critsect by defering calls to upper
|
||||
* layers to a signal handler.
|
||||
*/
|
||||
Mutex::Guard guard(_monitor_change_mutex);
|
||||
_monitor_change_event = { .id = screen_id,
|
||||
.origin_x = int(origin_x),
|
||||
.origin_y = int(origin_y) };
|
||||
_monitor_handler.local_submit();
|
||||
} break;
|
||||
|
||||
default:
|
||||
Genode::log("unexpected vbox event type ", int(ev_type), " will be ignored");
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user