monado/src/xrt/compositor/main/comp_window_xcb.cpp

411 lines
11 KiB
C++
Raw Normal View History

2019-03-18 05:52:32 +00:00
// Copyright 2019, Collabora, Ltd.
// SPDX-License-Identifier: BSL-1.0
/*!
* @file
* @brief XCB window code.
* @author Lubosz Sarnecki <lubosz.sarnecki@collabora.com>
* @author Jakob Bornecrantz <jakob@collabora.com>
2020-03-01 10:31:21 +00:00
* @ingroup comp_main
2019-03-18 05:52:32 +00:00
*/
#include <xcb/xcb.h>
#include <xcb/randr.h>
#include <string>
#include <vector>
#include <utility>
#include <cstring>
#include "util/u_misc.h"
2019-03-18 05:52:32 +00:00
#include "xrt/xrt_compiler.h"
#include "main/comp_window.h"
/*
*
* Private structs.
*
*/
/*!
* Xcb display, xrandr output.
*/
struct comp_window_xcb_display
{
std::string name;
std::pair<int16_t, int16_t> position;
std::pair<uint16_t, uint16_t> size;
};
/*!
* A xcb connection and window.
*/
struct comp_window_xcb
{
struct comp_window base = comp_window();
xcb_connection_t *connection = nullptr;
2019-03-18 05:52:32 +00:00
xcb_window_t window = XCB_NONE;
xcb_screen_t *screen = nullptr;
2019-03-18 05:52:32 +00:00
xcb_atom_t atom_wm_protocols = XCB_NONE;
xcb_atom_t atom_wm_delete_window = XCB_NONE;
std::vector<comp_window_xcb_display> displays = {};
};
/*
*
* Pre declare functions.
*
*/
static void
comp_window_xcb_destroy(struct comp_window *w);
2019-03-18 05:52:32 +00:00
static void
comp_window_xcb_flush(struct comp_window *w);
2019-03-18 05:52:32 +00:00
XRT_MAYBE_UNUSED static void
comp_window_xcb_list_screens(comp_window_xcb *w, xcb_screen_t *screen);
2019-03-18 05:52:32 +00:00
static bool
comp_window_xcb_init(struct comp_window *w);
2019-03-18 05:52:32 +00:00
static comp_window_xcb_display *
comp_window_xcb_current_display(comp_window_xcb *w);
2019-03-18 05:52:32 +00:00
static bool
comp_window_xcb_init_swapchain(struct comp_window *w,
2019-03-18 05:52:32 +00:00
uint32_t width,
uint32_t height);
static int
comp_window_xcb_connect(comp_window_xcb *w);
2019-03-18 05:52:32 +00:00
static void
comp_window_xcb_create_window(comp_window_xcb *w,
2019-03-18 05:52:32 +00:00
uint32_t width,
uint32_t height);
static void
comp_window_xcb_get_randr_outputs(comp_window_xcb *w);
2019-03-18 05:52:32 +00:00
static void
comp_window_xcb_connect_delete_event(comp_window_xcb *w);
2019-03-18 05:52:32 +00:00
static void
comp_window_xcb_set_full_screen(comp_window_xcb *w);
2019-03-18 05:52:32 +00:00
static xcb_atom_t
comp_window_xcb_get_atom(comp_window_xcb *w, const char *name);
2019-03-18 05:52:32 +00:00
static VkResult
comp_window_xcb_create_surface(comp_window_xcb *w, VkSurfaceKHR *surface);
2019-03-18 05:52:32 +00:00
static void
comp_window_xcb_update_window_title(struct comp_window *w, const char *title);
2019-03-18 05:52:32 +00:00
/*
*
* Functions.
*
*/
extern "C" struct comp_window *
comp_window_xcb_create(struct comp_compositor *c)
2019-03-18 05:52:32 +00:00
{
auto w = new comp_window_xcb();
w->base.name = "xcb";
w->base.destroy = comp_window_xcb_destroy;
w->base.flush = comp_window_xcb_flush;
w->base.init = comp_window_xcb_init;
w->base.init_swapchain = comp_window_xcb_init_swapchain;
w->base.update_window_title = comp_window_xcb_update_window_title;
w->base.c = c;
return &w->base;
}
static void
comp_window_xcb_destroy(struct comp_window *w)
2019-03-18 05:52:32 +00:00
{
comp_window_xcb *w_xcb = (comp_window_xcb *)w;
2019-03-18 05:52:32 +00:00
xcb_destroy_window(w_xcb->connection, w_xcb->window);
xcb_disconnect(w_xcb->connection);
delete w;
}
static void
comp_window_xcb_list_screens(comp_window_xcb *w, xcb_screen_t *screen)
2019-03-18 05:52:32 +00:00
{
COMP_DEBUG(w->base.c, "Screen 0 %dx%d", screen->width_in_pixels,
screen->height_in_pixels);
comp_window_xcb_get_randr_outputs(w);
int display_i = 0;
for (const comp_window_xcb_display &d : w->displays) {
2019-03-18 05:52:32 +00:00
COMP_DEBUG(w->base.c, "%d: %s %dx%d [%d, %d]", display_i,
d.name.c_str(), d.size.first, d.size.second,
d.position.first, d.position.second);
display_i++;
}
}
static bool
comp_window_xcb_init(struct comp_window *w)
2019-03-18 05:52:32 +00:00
{
struct comp_window_xcb *w_xcb = (struct comp_window_xcb *)w;
2019-03-18 05:52:32 +00:00
if (!comp_window_xcb_connect(w_xcb)) {
return false;
}
xcb_screen_iterator_t iter =
xcb_setup_roots_iterator(xcb_get_setup(w_xcb->connection));
w_xcb->screen = iter.data;
if (w->c->settings.fullscreen) {
comp_window_xcb_get_randr_outputs(w_xcb);
if (w->c->settings.display > (int)w_xcb->displays.size() - 1) {
COMP_DEBUG(w->c,
"Requested display %d, but only %d "
"displays are available.",
w->c->settings.display,
(int)w_xcb->displays.size());
w->c->settings.display = 0;
comp_window_xcb_display *d =
2019-03-18 05:52:32 +00:00
comp_window_xcb_current_display(w_xcb);
COMP_DEBUG(w->c, "Selecting '%s' instead.",
d->name.c_str());
}
if (w->c->settings.display == -1)
w->c->settings.display = 0;
comp_window_xcb_display *d =
2019-03-18 05:52:32 +00:00
comp_window_xcb_current_display(w_xcb);
w->c->settings.width = d->size.first;
w->c->settings.height = d->size.second;
// TODO: size cb
// set_size_cb(settings->width, settings->height);
}
comp_window_xcb_create_window(w_xcb, w->c->settings.width,
w->c->settings.height);
comp_window_xcb_connect_delete_event(w_xcb);
if (w->c->settings.fullscreen)
comp_window_xcb_set_full_screen(w_xcb);
xcb_map_window(w_xcb->connection, w_xcb->window);
return true;
}
static comp_window_xcb_display *
comp_window_xcb_current_display(comp_window_xcb *w)
2019-03-18 05:52:32 +00:00
{
return &w->displays[w->base.c->settings.display];
}
static void
comp_window_xcb_flush(struct comp_window *w)
2019-03-18 05:52:32 +00:00
{}
static bool
comp_window_xcb_init_swapchain(struct comp_window *w,
2019-03-18 05:52:32 +00:00
uint32_t width,
uint32_t height)
{
comp_window_xcb *w_xcb = (comp_window_xcb *)w;
2019-03-18 05:52:32 +00:00
VkResult ret;
ret = comp_window_xcb_create_surface(w_xcb, &w->swapchain.surface);
if (ret != VK_SUCCESS) {
return false;
}
// vk_swapchain_set_dimension_cb(&swapchain, set_size_cb);
// vk_swapchain_set_settings(&w->swapchain, w->settings);
vk_swapchain_create(
&w->swapchain, width, height, w->c->settings.color_format,
w->c->settings.color_space, w->c->settings.present_mode);
return true;
}
static int
comp_window_xcb_connect(comp_window_xcb *w)
2019-03-18 05:52:32 +00:00
{
w->connection = xcb_connect(nullptr, nullptr);
return !xcb_connection_has_error(w->connection);
}
static void
comp_window_xcb_create_window(comp_window_xcb *w,
2019-03-18 05:52:32 +00:00
uint32_t width,
uint32_t height)
{
w->window = xcb_generate_id(w->connection);
int x = 0;
int y = 0;
if (w->base.c->settings.fullscreen) {
x = comp_window_xcb_current_display(w)->position.first;
y = comp_window_xcb_current_display(w)->position.second;
}
uint32_t value_list = XCB_EVENT_MASK_STRUCTURE_NOTIFY;
xcb_create_window(w->connection, XCB_COPY_FROM_PARENT, w->window,
w->screen->root, x, y, width, height, 0,
XCB_WINDOW_CLASS_INPUT_OUTPUT, w->screen->root_visual,
XCB_CW_EVENT_MASK, &value_list);
}
static void
comp_window_xcb_get_randr_outputs(comp_window_xcb *w)
2019-03-18 05:52:32 +00:00
{
xcb_randr_get_screen_resources_cookie_t resources_cookie =
xcb_randr_get_screen_resources(w->connection, w->screen->root);
xcb_randr_get_screen_resources_reply_t *resources_reply =
2019-03-18 05:52:32 +00:00
xcb_randr_get_screen_resources_reply(w->connection,
resources_cookie, nullptr);
xcb_randr_output_t *xcb_outputs =
2019-03-18 05:52:32 +00:00
xcb_randr_get_screen_resources_outputs(resources_reply);
int count =
xcb_randr_get_screen_resources_outputs_length(resources_reply);
if (count < 1) {
COMP_ERROR(w->base.c, "Failed to retrieve randr outputs");
}
for (int i = 0; i < count; i++) {
xcb_randr_get_output_info_cookie_t output_cookie =
xcb_randr_get_output_info(w->connection, xcb_outputs[i],
XCB_CURRENT_TIME);
xcb_randr_get_output_info_reply_t *output_reply =
2019-03-18 05:52:32 +00:00
xcb_randr_get_output_info_reply(w->connection,
output_cookie, nullptr);
if (output_reply->connection !=
XCB_RANDR_CONNECTION_CONNECTED ||
output_reply->crtc == XCB_NONE) {
free(output_reply);
2019-03-18 05:52:32 +00:00
continue;
}
xcb_randr_get_crtc_info_cookie_t crtc_cookie =
xcb_randr_get_crtc_info(w->connection, output_reply->crtc,
XCB_CURRENT_TIME);
xcb_randr_get_crtc_info_reply_t *crtc_reply =
2019-03-18 05:52:32 +00:00
xcb_randr_get_crtc_info_reply(w->connection, crtc_cookie,
nullptr);
uint8_t *name = xcb_randr_get_output_info_name(output_reply);
2019-03-18 05:52:32 +00:00
int name_len =
xcb_randr_get_output_info_name_length(output_reply);
char *name_str = U_TYPED_ARRAY_CALLOC(char, name_len + 1);
2019-03-18 05:52:32 +00:00
memcpy(name_str, name, name_len);
name_str[name_len] = '\0';
struct comp_window_xcb_display d = {};
d.name = std::string(name_str);
d.position = {crtc_reply->x, crtc_reply->y};
d.size = {crtc_reply->width, crtc_reply->height};
w->displays.push_back(d);
free(crtc_reply);
free(output_reply);
2019-03-18 05:52:32 +00:00
free(name_str);
}
free(resources_reply);
2019-03-18 05:52:32 +00:00
}
static void
comp_window_xcb_connect_delete_event(comp_window_xcb *w)
2019-03-18 05:52:32 +00:00
{
w->atom_wm_protocols = comp_window_xcb_get_atom(w, "WM_PROTOCOLS");
w->atom_wm_delete_window =
comp_window_xcb_get_atom(w, "WM_DELETE_WINDOW");
xcb_change_property(w->connection, XCB_PROP_MODE_REPLACE, w->window,
w->atom_wm_protocols, XCB_ATOM_ATOM, 32, 1,
&w->atom_wm_delete_window);
}
static void
comp_window_xcb_set_full_screen(comp_window_xcb *w)
2019-03-18 05:52:32 +00:00
{
xcb_atom_t atom_wm_state = comp_window_xcb_get_atom(w, "_NET_WM_STATE");
xcb_atom_t atom_wm_fullscreen =
comp_window_xcb_get_atom(w, "_NET_WM_STATE_FULLSCREEN");
xcb_change_property(w->connection, XCB_PROP_MODE_REPLACE, w->window,
atom_wm_state, XCB_ATOM_ATOM, 32, 1,
&atom_wm_fullscreen);
}
static xcb_atom_t
comp_window_xcb_get_atom(comp_window_xcb *w, const char *name)
2019-03-18 05:52:32 +00:00
{
xcb_intern_atom_cookie_t cookie;
xcb_intern_atom_reply_t *reply;
2019-03-18 05:52:32 +00:00
xcb_atom_t atom;
cookie = xcb_intern_atom(w->connection, 0, strlen(name), name);
reply = xcb_intern_atom_reply(w->connection, cookie, NULL);
if (reply) {
atom = reply->atom;
} else {
atom = XCB_NONE;
}
free(reply);
return atom;
}
static VkResult
comp_window_xcb_create_surface(comp_window_xcb *w, VkSurfaceKHR *surface)
2019-03-18 05:52:32 +00:00
{
struct vk_bundle *vk = w->base.swapchain.vk;
2019-03-18 05:52:32 +00:00
VkResult ret;
VkXcbSurfaceCreateInfoKHR surface_info = {
.sType = VK_STRUCTURE_TYPE_XCB_SURFACE_CREATE_INFO_KHR,
.pNext = nullptr,
.flags = 0,
.connection = w->connection,
.window = w->window,
};
ret = vk->vkCreateXcbSurfaceKHR(vk->instance, &surface_info, NULL,
surface);
if (ret != VK_SUCCESS) {
COMP_ERROR(w->base.c, "vkCreateXcbSurfaceKHR: %s",
vk_result_string(ret));
return ret;
}
return VK_SUCCESS;
}
static void
comp_window_xcb_update_window_title(struct comp_window *w, const char *title)
2019-03-18 05:52:32 +00:00
{
comp_window_xcb *w_xcb = (comp_window_xcb *)w;
2019-03-18 05:52:32 +00:00
xcb_change_property(w_xcb->connection, XCB_PROP_MODE_REPLACE,
w_xcb->window, XCB_ATOM_WM_NAME, XCB_ATOM_STRING, 8,
strlen(title), title);
}