/* SPDX-License-Identifier: MIT OR X11 * * Copyright © 2024 Enrico Weigelt, metux IT consult */ #include #include #include #include #include #include #include #include #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; }