577 lines
15 KiB
C
577 lines
15 KiB
C
/* SPDX-License-Identifier: MIT OR X11
|
|
*
|
|
* Copyright © 2024 Enrico Weigelt, metux IT consult <info@metux.net>
|
|
*/
|
|
#include <dix-config.h>
|
|
|
|
#include <xcb/xcb.h>
|
|
#include <xcb/xcb_aux.h>
|
|
#include <xcb/xcb_icccm.h>
|
|
|
|
#include <X11/X.h>
|
|
#include <X11/Xdefs.h>
|
|
#include <X11/Xproto.h>
|
|
#include <xcb/xkb.h>
|
|
|
|
#include "include/gc.h"
|
|
#include "include/servermd.h"
|
|
|
|
#include "xnest-xcb.h"
|
|
#include "xnest-xkb.h"
|
|
#include "XNGC.h"
|
|
#include "Display.h"
|
|
|
|
struct xnest_upstream_info xnestUpstreamInfo = { 0 };
|
|
xnest_visual_t *xnestVisualMap;
|
|
int xnestNumVisualMap;
|
|
|
|
Bool xnest_upstream_setup(const char* displayName)
|
|
{
|
|
xnestUpstreamInfo.conn = xcb_connect(displayName, &xnestUpstreamInfo.screenId);
|
|
if (!xnestUpstreamInfo.conn)
|
|
return FALSE;
|
|
|
|
/* retrieve setup data for our screen */
|
|
xnestUpstreamInfo.setup = xcb_get_setup(xnestUpstreamInfo.conn);
|
|
xcb_screen_iterator_t iter = xcb_setup_roots_iterator (xnestUpstreamInfo.setup);
|
|
|
|
for (int i = 0; i < xnestUpstreamInfo.screenId; ++i)
|
|
xcb_screen_next (&iter);
|
|
xnestUpstreamInfo.screenInfo = iter.data;
|
|
|
|
xorg_list_init(&xnestUpstreamInfo.eventQueue.entry);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* retrieve upstream GC XID for our xserver GC */
|
|
uint32_t xnest_upstream_gc(GCPtr pGC) {
|
|
if (pGC == NULL) return 0;
|
|
|
|
xnestPrivGC *priv = dixLookupPrivate(&(pGC)->devPrivates, xnestGCPrivateKey);
|
|
if (priv == NULL) return 0;
|
|
|
|
return priv->gc;
|
|
}
|
|
|
|
const char WM_COLORMAP_WINDOWS[] = "WM_COLORMAP_WINDOWS";
|
|
|
|
void xnest_wm_colormap_windows(
|
|
xcb_connection_t *conn,
|
|
xcb_window_t w,
|
|
xcb_window_t *windows,
|
|
int count)
|
|
{
|
|
xcb_intern_atom_reply_t *reply = xcb_intern_atom_reply(
|
|
conn,
|
|
xcb_intern_atom(
|
|
conn, 0,
|
|
sizeof(WM_COLORMAP_WINDOWS)-1,
|
|
WM_COLORMAP_WINDOWS),
|
|
NULL);
|
|
|
|
if (!reply)
|
|
return;
|
|
|
|
xcb_icccm_set_wm_colormap_windows_checked(
|
|
conn,
|
|
w,
|
|
reply->atom,
|
|
count,
|
|
(xcb_window_t*)windows);
|
|
|
|
free(reply);
|
|
}
|
|
|
|
uint32_t xnest_create_bitmap_from_data(
|
|
xcb_connection_t *conn,
|
|
uint32_t drawable,
|
|
const char *data,
|
|
uint32_t width,
|
|
uint32_t height)
|
|
{
|
|
uint32_t pix = xcb_generate_id(xnestUpstreamInfo.conn);
|
|
xcb_create_pixmap(conn, 1, pix, drawable, width, height);
|
|
|
|
uint32_t gc = xcb_generate_id(xnestUpstreamInfo.conn);
|
|
xcb_create_gc(conn, gc, pix, 0, NULL);
|
|
|
|
const int leftPad = 0;
|
|
|
|
xcb_put_image(conn,
|
|
XYPixmap,
|
|
pix,
|
|
gc,
|
|
width,
|
|
height,
|
|
0 /* dst_x */,
|
|
0 /* dst_y */,
|
|
leftPad,
|
|
1 /* depth */,
|
|
BitmapBytePad(width + leftPad) * height,
|
|
(uint8_t*)data);
|
|
|
|
xcb_free_gc(conn, gc);
|
|
return pix;
|
|
}
|
|
|
|
uint32_t xnest_create_pixmap_from_bitmap_data(
|
|
xcb_connection_t *conn,
|
|
uint32_t drawable,
|
|
const char *data,
|
|
uint32_t width,
|
|
uint32_t height,
|
|
uint32_t fg,
|
|
uint32_t bg,
|
|
uint16_t depth)
|
|
{
|
|
uint32_t pix = xcb_generate_id(xnestUpstreamInfo.conn);
|
|
xcb_create_pixmap(conn, depth, pix, drawable, width, height);
|
|
|
|
uint32_t gc = xcb_generate_id(xnestUpstreamInfo.conn);
|
|
xcb_create_gc(conn, gc, pix, 0, NULL);
|
|
|
|
xcb_params_gc_t gcv = {
|
|
.foreground = fg,
|
|
.background = bg
|
|
};
|
|
|
|
xcb_aux_change_gc(conn, gc, XCB_GC_FOREGROUND | XCB_GC_BACKGROUND, &gcv);
|
|
|
|
const int leftPad = 0;
|
|
xcb_put_image(conn,
|
|
XYBitmap,
|
|
pix,
|
|
gc,
|
|
width,
|
|
height,
|
|
0 /* dst_x */,
|
|
0 /* dst_y */,
|
|
leftPad,
|
|
1 /* depth */,
|
|
BitmapBytePad(width + leftPad) * height,
|
|
(uint8_t*)data);
|
|
|
|
xcb_free_gc(conn, gc);
|
|
return pix;
|
|
}
|
|
|
|
void xnest_set_command(
|
|
xcb_connection_t *conn,
|
|
xcb_window_t window,
|
|
char **argv,
|
|
int argc)
|
|
{
|
|
int i = 0, nbytes = 0;
|
|
|
|
for (i = 0, nbytes = 0; i < argc; i++)
|
|
nbytes += strlen(argv[i]) + 1;
|
|
|
|
if (nbytes >= (1<<16) - 1)
|
|
return;
|
|
|
|
char *buf = calloc(1, nbytes+1);
|
|
if (!buf)
|
|
return; // BadAlloc
|
|
|
|
char *bp = buf;
|
|
|
|
/* copy arguments into single buffer */
|
|
for (i = 0; i < argc; i++) {
|
|
strcpy(bp, argv[i]);
|
|
bp += strlen(argv[i]) + 1;
|
|
}
|
|
|
|
xcb_change_property(conn,
|
|
XCB_PROP_MODE_REPLACE,
|
|
window,
|
|
XCB_ATOM_WM_COMMAND,
|
|
XCB_ATOM_STRING,
|
|
8,
|
|
nbytes,
|
|
buf);
|
|
free(buf);
|
|
}
|
|
|
|
void xnest_xkb_init(xcb_connection_t *conn)
|
|
{
|
|
xcb_generic_error_t *err = NULL;
|
|
xcb_xkb_use_extension_reply_t *reply = xcb_xkb_use_extension_reply(
|
|
xnestUpstreamInfo.conn,
|
|
xcb_xkb_use_extension(
|
|
xnestUpstreamInfo.conn,
|
|
XCB_XKB_MAJOR_VERSION,
|
|
XCB_XKB_MINOR_VERSION),
|
|
&err);
|
|
|
|
if (err) {
|
|
ErrorF("failed query xkb extension: %d\n", err->error_code);
|
|
free(err);
|
|
} else {
|
|
free(reply);
|
|
}
|
|
}
|
|
|
|
#define XkbGBN_AllComponentsMask_2 ( \
|
|
XCB_XKB_GBN_DETAIL_TYPES | \
|
|
XCB_XKB_GBN_DETAIL_COMPAT_MAP | \
|
|
XCB_XKB_GBN_DETAIL_CLIENT_SYMBOLS | \
|
|
XCB_XKB_GBN_DETAIL_SERVER_SYMBOLS | \
|
|
XCB_XKB_GBN_DETAIL_INDICATOR_MAPS | \
|
|
XCB_XKB_GBN_DETAIL_KEY_NAMES | \
|
|
XCB_XKB_GBN_DETAIL_GEOMETRY | \
|
|
XCB_XKB_GBN_DETAIL_OTHER_NAMES)
|
|
|
|
int xnest_xkb_device_id(xcb_connection_t *conn)
|
|
{
|
|
int device_id = -1;
|
|
uint8_t xlen[6] = { 0 };
|
|
xcb_generic_error_t *err = NULL;
|
|
|
|
xcb_xkb_get_kbd_by_name_reply_t *reply = xcb_xkb_get_kbd_by_name_reply(
|
|
xnestUpstreamInfo.conn,
|
|
xcb_xkb_get_kbd_by_name_2(
|
|
xnestUpstreamInfo.conn,
|
|
XCB_XKB_ID_USE_CORE_KBD,
|
|
XkbGBN_AllComponentsMask_2,
|
|
XkbGBN_AllComponentsMask_2,
|
|
0,
|
|
sizeof(xlen),
|
|
xlen),
|
|
&err);
|
|
|
|
if (err) {
|
|
ErrorF("failed retrieving core keyboard: %d\n", err->error_code);
|
|
free(err);
|
|
return -1;
|
|
}
|
|
|
|
if (!reply) {
|
|
ErrorF("failed retrieving core keyboard: no reply");
|
|
return -1;
|
|
}
|
|
|
|
device_id = reply->deviceID;
|
|
free(reply);
|
|
return device_id;
|
|
}
|
|
|
|
xcb_get_keyboard_mapping_reply_t *xnest_get_keyboard_mapping(
|
|
xcb_connection_t *conn,
|
|
int min_keycode,
|
|
int count
|
|
) {
|
|
xcb_generic_error_t *err= NULL;
|
|
xcb_get_keyboard_mapping_reply_t * reply = xcb_get_keyboard_mapping_reply(
|
|
xnestUpstreamInfo.conn,
|
|
xcb_get_keyboard_mapping(conn, min_keycode, count),
|
|
&err);
|
|
|
|
if (err) {
|
|
ErrorF("Couldn't get keyboard mapping: %d\n", err->error_code);
|
|
free(err);
|
|
}
|
|
|
|
return reply;
|
|
}
|
|
|
|
void xnest_get_pointer_control(
|
|
xcb_connection_t *conn,
|
|
int *acc_num,
|
|
int *acc_den,
|
|
int *threshold)
|
|
{
|
|
xcb_generic_error_t *err = NULL;
|
|
xcb_get_pointer_control_reply_t *reply = xcb_get_pointer_control_reply(
|
|
xnestUpstreamInfo.conn,
|
|
xcb_get_pointer_control(xnestUpstreamInfo.conn),
|
|
&err);
|
|
|
|
if (err) {
|
|
ErrorF("error retrieving pointer control data: %d\n", err->error_code);
|
|
free(err);
|
|
}
|
|
|
|
if (!reply) {
|
|
ErrorF("error retrieving pointer control data: no reply\n");
|
|
return;
|
|
}
|
|
|
|
*acc_num = reply->acceleration_numerator;
|
|
*acc_den = reply->acceleration_denominator;
|
|
*threshold = reply->threshold;
|
|
free(reply);
|
|
}
|
|
|
|
xRectangle xnest_get_geometry(xcb_connection_t *conn, uint32_t window)
|
|
{
|
|
xcb_generic_error_t *err = NULL;
|
|
xcb_get_geometry_reply_t *reply = xcb_get_geometry_reply(
|
|
xnestUpstreamInfo.conn,
|
|
xcb_get_geometry(xnestUpstreamInfo.conn, window),
|
|
&err);
|
|
|
|
if (err) {
|
|
ErrorF("failed getting window attributes for %d: %d\n", window, err->error_code);
|
|
free(err);
|
|
return (xRectangle) { 0 };
|
|
}
|
|
|
|
if (!reply) {
|
|
ErrorF("failed getting window attributes for %d: no reply\n", window);
|
|
return (xRectangle) { 0 };
|
|
}
|
|
|
|
return (xRectangle) {
|
|
.x = reply->x,
|
|
.y = reply->y,
|
|
.width = reply->width,
|
|
.height = reply->height };
|
|
}
|
|
|
|
static int __readint(const char *str, const char **next)
|
|
{
|
|
int res = 0, sign = 1;
|
|
|
|
if (*str=='+')
|
|
str++;
|
|
else if (*str=='-') {
|
|
str++;
|
|
sign = -1;
|
|
}
|
|
|
|
for (; (*str>='0') && (*str<='9'); str++)
|
|
res = (res * 10) + (*str-'0');
|
|
|
|
*next = str;
|
|
return sign * res;
|
|
}
|
|
|
|
int xnest_parse_geometry(const char *string, xRectangle *geometry)
|
|
{
|
|
int mask = 0;
|
|
const char *next;
|
|
xRectangle temp = { 0 };
|
|
|
|
if ((string == NULL) || (*string == '\0')) return 0;
|
|
|
|
if (*string == '=')
|
|
string++; /* ignore possible '=' at beg of geometry spec */
|
|
|
|
if (*string != '+' && *string != '-' && *string != 'x') {
|
|
temp.width = __readint(string, &next);
|
|
if (string == next)
|
|
return 0;
|
|
string = next;
|
|
mask |= XCB_CONFIG_WINDOW_WIDTH;
|
|
}
|
|
|
|
if (*string == 'x' || *string == 'X') {
|
|
string++;
|
|
temp.height = __readint(string, &next);
|
|
if (string == next)
|
|
return 0;
|
|
string = next;
|
|
mask |= XCB_CONFIG_WINDOW_HEIGHT;
|
|
}
|
|
|
|
if ((*string == '+') || (*string== '-')) {
|
|
if (*string== '-') {
|
|
string++;
|
|
temp.x = -__readint(string, &next);
|
|
if (string == next)
|
|
return 0;
|
|
string = next;
|
|
}
|
|
else
|
|
{
|
|
string++;
|
|
temp.x = __readint(string, &next);
|
|
if (string == next)
|
|
return 0;
|
|
string = next;
|
|
}
|
|
mask |= XCB_CONFIG_WINDOW_X;
|
|
if ((*string == '+') || (*string== '-')) {
|
|
if (*string== '-') {
|
|
string++;
|
|
temp.y = -__readint(string, &next);
|
|
if (string == next)
|
|
return 0;
|
|
string = next;
|
|
}
|
|
else
|
|
{
|
|
string++;
|
|
temp.y = __readint(string, &next);
|
|
if (string == next)
|
|
return 0;
|
|
string = next;
|
|
}
|
|
mask |= XCB_CONFIG_WINDOW_Y;
|
|
}
|
|
}
|
|
|
|
if (*string != '\0') return 0;
|
|
|
|
if (mask & XCB_CONFIG_WINDOW_X)
|
|
geometry->x = temp.x;
|
|
if (mask & XCB_CONFIG_WINDOW_Y)
|
|
geometry->y = temp.y;
|
|
if (mask & XCB_CONFIG_WINDOW_WIDTH)
|
|
geometry->width = temp.width;
|
|
if (mask & XCB_CONFIG_WINDOW_HEIGHT)
|
|
geometry->height = temp.height;
|
|
|
|
return mask;
|
|
}
|
|
|
|
uint32_t xnest_visual_map_to_upstream(VisualID visual)
|
|
{
|
|
for (int i = 0; i < xnestNumVisualMap; i++) {
|
|
if (xnestVisualMap[i].ourXID == visual) {
|
|
return xnestVisualMap[i].upstreamVisual->visual_id;
|
|
}
|
|
}
|
|
return XCB_NONE;
|
|
}
|
|
|
|
uint32_t xnest_upstream_visual_to_cmap(uint32_t upstreamVisual)
|
|
{
|
|
for (int i = 0; i < xnestNumVisualMap; i++) {
|
|
if (xnestVisualMap[i].upstreamVisual->visual_id == upstreamVisual) {
|
|
return xnestVisualMap[i].upstreamCMap;
|
|
}
|
|
}
|
|
return XCB_COLORMAP_NONE;
|
|
}
|
|
|
|
uint32_t xnest_visual_to_upstream_cmap(uint32_t visual)
|
|
{
|
|
for (int i = 0; i < xnestNumVisualMap; i++) {
|
|
if (xnestVisualMap[i].ourXID == visual) {
|
|
return xnestVisualMap[i].upstreamCMap;
|
|
}
|
|
}
|
|
return XCB_COLORMAP_NONE;
|
|
}
|
|
|
|
static inline char XN_CI_NONEXISTCHAR(xcb_charinfo_t *cs)
|
|
{
|
|
return ((cs->character_width == 0) && \
|
|
((cs->right_side_bearing | cs->left_side_bearing | cs->ascent | cs->descent) == 0));
|
|
}
|
|
|
|
#define XN_CI_GET_CHAR_INFO_1D(font,col,def,cs) \
|
|
do { \
|
|
cs = def; \
|
|
if (col >= font->font_reply->min_char_or_byte2 && col <= font->font_reply->max_char_or_byte2) { \
|
|
if (font->chars == NULL) { \
|
|
cs = &font->font_reply->min_bounds; \
|
|
} else { \
|
|
cs = (xcb_charinfo_t *)&font->chars[(col - font->font_reply->min_char_or_byte2)]; \
|
|
if (XN_CI_NONEXISTCHAR(cs)) cs = def; \
|
|
} \
|
|
} \
|
|
} while (0)
|
|
|
|
#define XN_CI_GET_CHAR_INFO_2D(font,row,col,def,cs) \
|
|
do { \
|
|
cs = def; \
|
|
if (row >= font->font_reply->min_byte1 && row <= font->font_reply->max_byte1 && \
|
|
col >= font->font_reply->min_char_or_byte2 && col <= font->font_reply->max_char_or_byte2) { \
|
|
if (font->chars == NULL) { \
|
|
cs = &font->font_reply->min_bounds; \
|
|
} else { \
|
|
cs = (xcb_charinfo_t*)&font->chars[((row - font->font_reply->min_byte1) * \
|
|
(font->font_reply->max_char_or_byte2 - \
|
|
font->font_reply->min_char_or_byte2 + 1)) + \
|
|
(col - font->font_reply->min_char_or_byte2)]; \
|
|
if (XN_CI_NONEXISTCHAR(cs)) cs = def; \
|
|
} \
|
|
} \
|
|
} while (0)
|
|
|
|
#define XN_CI_GET_DEFAULT_INFO_2D(font,cs) \
|
|
do { \
|
|
unsigned int r = (font->font_reply->default_char >> 8); \
|
|
unsigned int c = (font->font_reply->default_char & 0xff); \
|
|
XN_CI_GET_CHAR_INFO_2D (font, r, c, NULL, cs); \
|
|
} while (0)
|
|
|
|
#define XN_CI_GET_ROWZERO_CHAR_INFO_2D(font,col,def,cs) \
|
|
do { \
|
|
cs = def; \
|
|
if (font->font_reply->min_byte1 == 0 && \
|
|
col >= font->font_reply->min_char_or_byte2 && col <= font->font_reply->max_char_or_byte2) { \
|
|
if (font->chars == NULL) { \
|
|
cs = &font->font_reply->min_bounds; \
|
|
} else { \
|
|
cs = (xcb_charinfo_t*)&font->chars[(col - font->font_reply->min_char_or_byte2)]; \
|
|
if (XN_CI_NONEXISTCHAR(cs)) cs = def; \
|
|
} \
|
|
} \
|
|
} while (0)
|
|
|
|
int xnest_text_width(xnestPrivFont *font, const char *string, int count)
|
|
{
|
|
xcb_charinfo_t *def;
|
|
|
|
if (font->font_reply->max_byte1 == 0)
|
|
XN_CI_GET_CHAR_INFO_1D (font, font->font_reply->default_char, NULL, def);
|
|
else
|
|
XN_CI_GET_DEFAULT_INFO_2D (font, def);
|
|
|
|
if (def && font->font_reply->min_bounds.character_width == font->font_reply->max_bounds.character_width)
|
|
return (font->font_reply->min_bounds.character_width * count);
|
|
|
|
int width = 0, i = 0;
|
|
unsigned char *us;
|
|
for (i = 0, us = (unsigned char *) string; i < count; i++, us++) {
|
|
unsigned uc = (unsigned) *us;
|
|
xcb_charinfo_t *cs;
|
|
|
|
if (font->font_reply->max_byte1 == 0) {
|
|
XN_CI_GET_CHAR_INFO_1D (font, uc, def, cs);
|
|
} else {
|
|
XN_CI_GET_ROWZERO_CHAR_INFO_2D (font, uc, def, cs);
|
|
}
|
|
|
|
if (cs) width += cs->character_width;
|
|
}
|
|
|
|
return width;
|
|
}
|
|
|
|
int xnest_text_width_16 (xnestPrivFont *font, const uint16_t* str, int count)
|
|
{
|
|
xcb_charinfo_t *def;
|
|
xcb_char2b_t *string = (xcb_char2b_t*)str;
|
|
|
|
if (font->font_reply->max_byte1 == 0)
|
|
XN_CI_GET_CHAR_INFO_1D (font, font->font_reply->default_char, NULL, def);
|
|
else
|
|
XN_CI_GET_DEFAULT_INFO_2D (font, def);
|
|
|
|
if (def && font->font_reply->min_bounds.character_width == font->font_reply->max_bounds.character_width)
|
|
return (font->font_reply->min_bounds.character_width * count);
|
|
|
|
int width = 0;
|
|
for (int i = 0; i < count; i++, string++) {
|
|
xcb_charinfo_t *cs;
|
|
unsigned int r = (unsigned int) string->byte1;
|
|
unsigned int c = (unsigned int) string->byte2;
|
|
|
|
if (font->font_reply->max_byte1 == 0) {
|
|
unsigned int ind = ((r << 8) | c);
|
|
XN_CI_GET_CHAR_INFO_1D (font, ind, def, cs);
|
|
} else {
|
|
XN_CI_GET_CHAR_INFO_2D (font, r, c, def, cs);
|
|
}
|
|
|
|
if (cs) width += cs->character_width;
|
|
}
|
|
|
|
return width;
|
|
}
|