/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved. * Copyright (C) 2004-2008 Constantin Kaplinsky. All Rights Reserved. * Copyright 2017 Peter Astrand for Cendio AB * * This is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This software is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this software; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, * USA. */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #ifdef HAVE_XTEST #include #endif #ifdef HAVE_XDAMAGE #include #endif #ifdef HAVE_XFIXES #include #include #endif #ifdef HAVE_XRANDR #include #include extern "C" { void vncSetGlueContext(Display *dpy, void *res); } #endif #include #include using namespace rfb; extern const unsigned short code_map_qnum_to_xorgevdev[]; extern const unsigned int code_map_qnum_to_xorgevdev_len; extern const unsigned short code_map_qnum_to_xorgkbd[]; extern const unsigned int code_map_qnum_to_xorgkbd_len; BoolParameter useShm("UseSHM", "Use MIT-SHM extension if available", true); BoolParameter rawKeyboard("RawKeyboard", "Send keyboard events straight through and " "avoid mapping them to the current keyboard " "layout", false); IntParameter queryConnectTimeout("QueryConnectTimeout", "Number of seconds to show the 'Accept connection' dialog before " "rejecting the connection", 10); static rfb::LogWriter vlog("XDesktop"); // order is important as it must match RFB extension static const char * ledNames[XDESKTOP_N_LEDS] = { "Scroll Lock", "Num Lock", "Caps Lock" }; XDesktop::XDesktop(Display* dpy_, Geometry *geometry_) : dpy(dpy_), geometry(geometry_), pb(nullptr), server(nullptr), queryConnectDialog(nullptr), queryConnectSock(nullptr), selection(dpy_, this), oldButtonMask(0), haveXtest(false), haveDamage(false), maxButtons(0), running(false), ledMasks(), ledState(0), codeMap(nullptr), codeMapLen(0) { int major, minor; int xkbOpcode, xkbErrorBase; major = XkbMajorVersion; minor = XkbMinorVersion; if (!XkbQueryExtension(dpy, &xkbOpcode, &xkbEventBase, &xkbErrorBase, &major, &minor)) { vlog.error("XKEYBOARD extension not present"); throw std::runtime_error("XKEYBOARD extension not present"); } XkbSelectEvents(dpy, XkbUseCoreKbd, XkbIndicatorStateNotifyMask, XkbIndicatorStateNotifyMask); // figure out bit masks for the indicators we are interested in for (int i = 0; i < XDESKTOP_N_LEDS; i++) { Atom a; int shift; Bool on; a = XInternAtom(dpy, ledNames[i], True); if (!a || !XkbGetNamedIndicator(dpy, a, &shift, &on, nullptr, nullptr)) continue; ledMasks[i] = 1u << shift; vlog.debug("Mask for '%s' is 0x%x", ledNames[i], ledMasks[i]); if (on) ledState |= 1u << i; } // X11 unfortunately uses keyboard driver specific keycodes and provides no // direct way to query this, so guess based on the keyboard mapping XkbDescPtr desc = XkbGetKeyboard(dpy, XkbAllComponentsMask, XkbUseCoreKbd); if (desc && desc->names) { char *keycodes = XGetAtomName(dpy, desc->names->keycodes); if (keycodes) { if (strncmp("evdev", keycodes, strlen("evdev")) == 0) { codeMap = code_map_qnum_to_xorgevdev; codeMapLen = code_map_qnum_to_xorgevdev_len; vlog.info("Using evdev codemap\n"); } else if (strncmp("xfree86", keycodes, strlen("xfree86")) == 0) { codeMap = code_map_qnum_to_xorgkbd; codeMapLen = code_map_qnum_to_xorgkbd_len; vlog.info("Using xorgkbd codemap\n"); } else { vlog.info("Unknown keycode '%s', no codemap\n", keycodes); } XFree(keycodes); } else { vlog.debug("Unable to get keycode map\n"); } XkbFreeKeyboard(desc, XkbAllComponentsMask, True); } #ifdef HAVE_XTEST int xtestEventBase; int xtestErrorBase; if (XTestQueryExtension(dpy, &xtestEventBase, &xtestErrorBase, &major, &minor)) { XTestGrabControl(dpy, True); vlog.info("XTest extension present - version %d.%d",major,minor); haveXtest = true; } else { #endif vlog.info("XTest extension not present"); vlog.info("Unable to inject events or display while server is grabbed"); #ifdef HAVE_XTEST } #endif #ifdef HAVE_XDAMAGE int xdamageErrorBase; if (XDamageQueryExtension(dpy, &xdamageEventBase, &xdamageErrorBase)) { haveDamage = true; } else { #endif vlog.info("DAMAGE extension not present"); vlog.info("Will have to poll screen for changes"); #ifdef HAVE_XDAMAGE } #endif #ifdef HAVE_XFIXES int xfixesErrorBase; if (XFixesQueryExtension(dpy, &xfixesEventBase, &xfixesErrorBase)) { XFixesSelectCursorInput(dpy, DefaultRootWindow(dpy), XFixesDisplayCursorNotifyMask); XFixesSelectSelectionInput(dpy, DefaultRootWindow(dpy), XA_PRIMARY, XFixesSetSelectionOwnerNotifyMask); XFixesSelectSelectionInput(dpy, DefaultRootWindow(dpy), xaCLIPBOARD, XFixesSetSelectionOwnerNotifyMask); } else { #endif vlog.info("XFIXES extension not present"); vlog.info("Will not be able to display cursors or monitor clipboard"); #ifdef HAVE_XFIXES } #endif #ifdef HAVE_XRANDR int xrandrErrorBase; randrSyncSerial = 0; if (XRRQueryExtension(dpy, &xrandrEventBase, &xrandrErrorBase)) { XRRSelectInput(dpy, DefaultRootWindow(dpy), RRScreenChangeNotifyMask | RRCrtcChangeNotifyMask); /* Override TXWindow::init input mask */ XSelectInput(dpy, DefaultRootWindow(dpy), PropertyChangeMask | StructureNotifyMask | ExposureMask | EnterWindowMask | LeaveWindowMask); } else { #endif vlog.info("RANDR extension not present"); vlog.info("Will not be able to handle session resize"); #ifdef HAVE_XRANDR } #endif TXWindow::setGlobalEventHandler(this); } XDesktop::~XDesktop() { if (running) stop(); } void XDesktop::poll() { if (pb and not haveDamage) pb->poll(server); if (running) { Window root, child; int x, y, wx, wy; unsigned int mask; if (XQueryPointer(dpy, DefaultRootWindow(dpy), &root, &child, &x, &y, &wx, &wy, &mask)) { x -= geometry->offsetLeft(); y -= geometry->offsetTop(); server->setCursorPos(rfb::Point(x, y), false); } } } void XDesktop::init(VNCServer* vs) { server = vs; } void XDesktop::start() { // Determine actual number of buttons of the X pointer device. unsigned char btnMap[9]; int numButtons = XGetPointerMapping(dpy, btnMap, 9); maxButtons = (numButtons > 9) ? 9 : numButtons; vlog.info("Enabling %d button%s of X pointer device", maxButtons, (maxButtons != 1) ? "s" : ""); // Create an ImageFactory instance for producing Image objects. ImageFactory factory((bool)useShm); // Create pixel buffer and provide it to the server object. pb = new XPixelBuffer(dpy, factory, geometry->getRect()); vlog.info("Allocated %s", pb->getImage()->classDesc()); server->setPixelBuffer(pb, computeScreenLayout()); #ifdef HAVE_XDAMAGE if (haveDamage) { damage = XDamageCreate(dpy, DefaultRootWindow(dpy), XDamageReportRawRectangles); } #endif #ifdef HAVE_XFIXES Window root, child; int x, y, wx, wy; unsigned int mask; // Check whether the cursor is initially on our screen if (XQueryPointer(dpy, DefaultRootWindow(dpy), &root, &child, &x, &y, &wx, &wy, &mask)) setCursor(); #endif server->setLEDState(ledState); running = true; } void XDesktop::stop() { running = false; #ifdef HAVE_XTEST // Delete added keycodes deleteAddedKeysyms(); #endif #ifdef HAVE_XDAMAGE if (haveDamage) XDamageDestroy(dpy, damage); #endif delete queryConnectDialog; queryConnectDialog = nullptr; server->setPixelBuffer(nullptr); delete pb; pb = nullptr; } void XDesktop::terminate() { kill(getpid(), SIGTERM); } bool XDesktop::isRunning() { return running; } void XDesktop::queryConnection(network::Socket* sock, const char* userName) { assert(isRunning()); // Someone already querying? if (queryConnectSock) { std::list sockets; // Check if this socket is still valid server->getSockets(&sockets); if (std::find(sockets.begin(), sockets.end(), queryConnectSock) != sockets.end()) { server->approveConnection(sock, false, "Another connection is currently being queried."); return; } } if (!userName) userName = "(anonymous)"; queryConnectSock = sock; delete queryConnectDialog; queryConnectDialog = new QueryConnectDialog(dpy, sock->getPeerAddress(), userName, queryConnectTimeout, this); queryConnectDialog->map(); } void XDesktop::pointerEvent(const Point& pos, uint16_t buttonMask) { #ifdef HAVE_XTEST if (!haveXtest) return; XTestFakeMotionEvent(dpy, DefaultScreen(dpy), geometry->offsetLeft() + pos.x, geometry->offsetTop() + pos.y, CurrentTime); if (buttonMask != oldButtonMask) { for (int i = 0; i < maxButtons; i++) { if ((buttonMask ^ oldButtonMask) & (1<min_key_code; keycode <= xkb->max_key_code; keycode++) { KeySym cursym; unsigned int out_mods; XkbTranslateKeyCode(xkb, keycode, mods, &out_mods, &cursym); if (cursym == keysym) break; } if (keycode > xkb->max_key_code) keycode = 0; XkbFreeKeyboard(xkb, XkbAllComponentsMask, True); // Shift+Tab is usually ISO_Left_Tab, but RFB hides this fact. Do // another attempt if we failed the initial lookup if ((keycode == 0) && (keysym == XK_Tab) && (mods & ShiftMask)) return XkbKeysymToKeycode(XK_ISO_Left_Tab); return keycode; } /* * Keeps the list in LRU order by moving the used key to front of the list. */ static void onKeyUsed(std::list &list, KeyCode usedKeycode) { if (list.empty() || list.front().keycode == usedKeycode) return; std::list::iterator it = list.begin(); ++it; for (; it != list.end(); ++it) { AddedKeySym item = *it; if (item.keycode == usedKeycode) { list.erase(it); list.push_front(item); break; } } } /* * Returns keycode of oldest item from list of manually added keysyms. * The item is removed from the list. * Returns 0 if no usable keycode is found. */ KeyCode XDesktop::getReusableKeycode(XkbDescPtr xkb) { while (!addedKeysyms.empty()) { AddedKeySym last = addedKeysyms.back(); addedKeysyms.pop_back(); // Make sure someone else hasn't modified the key if (XkbKeyNumGroups(xkb, last.keycode) > 0 && XkbKeySymsPtr(xkb, last.keycode)[0] == last.keysym) return last.keycode; } return 0; } KeyCode XDesktop::addKeysym(KeySym keysym) { int types[1]; unsigned int key; XkbDescPtr xkb; XkbMapChangesRec changes; KeySym *syms; KeySym upper, lower; xkb = XkbGetMap(dpy, XkbAllComponentsMask, XkbUseCoreKbd); if (!xkb) return 0; for (key = xkb->max_key_code; key >= xkb->min_key_code; key--) { if (XkbKeyNumGroups(xkb, key) == 0) break; } if (key < xkb->min_key_code) key = getReusableKeycode(xkb); if (!key) return 0; memset(&changes, 0, sizeof(changes)); XConvertCase(keysym, &lower, &upper); if (upper == lower) types[XkbGroup1Index] = XkbOneLevelIndex; else types[XkbGroup1Index] = XkbAlphabeticIndex; XkbChangeTypesOfKey(xkb, key, 1, XkbGroup1Mask, types, &changes); syms = XkbKeySymsPtr(xkb,key); if (upper == lower) syms[0] = keysym; else { syms[0] = lower; syms[1] = upper; } changes.changed |= XkbKeySymsMask; changes.first_key_sym = key; changes.num_key_syms = 1; if (XkbChangeMap(dpy, xkb, &changes)) { vlog.info("Added unknown keysym XK_%s (0x%04x) to keycode %d", XKeysymToString(keysym), (unsigned)keysym, key); addedKeysyms.push_front({ syms[0], (KeyCode)key }); return key; } return 0; } void XDesktop::deleteAddedKeysyms() { XkbDescPtr xkb; xkb = XkbGetMap(dpy, XkbAllComponentsMask, XkbUseCoreKbd); if (!xkb) return; XkbMapChangesRec changes; memset(&changes, 0, sizeof(changes)); KeyCode lowestKeyCode = xkb->max_key_code; KeyCode highestKeyCode = xkb->min_key_code; KeyCode keyCode = getReusableKeycode(xkb); while (keyCode != 0) { XkbChangeTypesOfKey(xkb, keyCode, 0, XkbGroup1Mask, nullptr, &changes); if (keyCode < lowestKeyCode) lowestKeyCode = keyCode; if (keyCode > highestKeyCode) highestKeyCode = keyCode; keyCode = getReusableKeycode(xkb); } // Did we actually find something to remove? if (highestKeyCode < lowestKeyCode) return; changes.changed |= XkbKeySymsMask; changes.first_key_sym = lowestKeyCode; changes.num_key_syms = highestKeyCode - lowestKeyCode + 1; XkbChangeMap(dpy, xkb, &changes); } KeyCode XDesktop::keysymToKeycode(KeySym keysym) { int keycode = 0; // XKeysymToKeycode() doesn't respect state, so we have to use // something slightly more complex keycode = XkbKeysymToKeycode(keysym); if (keycode != 0) return keycode; // TODO: try to further guess keycode with all possible mods as Xvnc does keycode = addKeysym(keysym); if (keycode == 0) vlog.error("Failure adding new keysym 0x%lx", keysym); return keycode; } #endif void XDesktop::keyEvent(uint32_t keysym, uint32_t xtcode, bool down) { #ifdef HAVE_XTEST int keycode = 0; if (!haveXtest) return; // Use scan code if provided and mapping exists if (codeMap && rawKeyboard && xtcode < codeMapLen) keycode = codeMap[xtcode]; if (!keycode) { if (pressedKeys.find(keysym) != pressedKeys.end()) keycode = pressedKeys[keysym]; else { keycode = keysymToKeycode(keysym); } } if (!keycode) { vlog.error("Could not map key event to X11 key code"); return; } if (down) pressedKeys[keysym] = keycode; else pressedKeys.erase(keysym); if (down) onKeyUsed(addedKeysyms, keycode); vlog.debug("%d %s", keycode, down ? "down" : "up"); XTestFakeKeyEvent(dpy, keycode, down, CurrentTime); #else (void)keysym; (void)xtcode; (void)down; #endif } ScreenSet XDesktop::computeScreenLayout() { ScreenSet layout; char buffer[2048]; #ifdef HAVE_XRANDR XRRScreenResources *res = XRRGetScreenResources(dpy, DefaultRootWindow(dpy)); if (!res) { vlog.error("XRRGetScreenResources failed"); return layout; } vncSetGlueContext(dpy, res); layout = ::computeScreenLayout(&outputIdMap); XRRFreeScreenResources(res); // Adjust the layout relative to the geometry ScreenSet::iterator iter, iter_next; Point offset(-geometry->offsetLeft(), -geometry->offsetTop()); for (iter = layout.begin();iter != layout.end();iter = iter_next) { iter_next = iter; ++iter_next; iter->dimensions = iter->dimensions.intersect(geometry->getRect()); if (iter->dimensions.is_empty()) layout.remove_screen(iter->id); else iter->dimensions = iter->dimensions.translate(offset); } #endif // Make sure that we have at least one screen if (layout.num_screens() == 0) layout.add_screen(rfb::Screen(0, 0, 0, geometry->width(), geometry->height(), 0)); vlog.debug("Detected screen layout:"); layout.print(buffer, sizeof(buffer)); vlog.debug("%s", buffer); return layout; } #ifdef HAVE_XRANDR /* Get the biggest mode which is equal or smaller to requested size. If no such mode exists, return the smallest. */ static void GetSmallerMode(XRRScreenResources *res, XRROutputInfo *output, unsigned int *width, unsigned int *height) { XRRModeInfo best = {}; XRRModeInfo smallest = {}; smallest.width = -1; smallest.height = -1; for (int i = 0; i < res->nmode; i++) { for (int j = 0; j < output->nmode; j++) { if (output->modes[j] == res->modes[i].id) { if ((res->modes[i].width > best.width && res->modes[i].width <= *width) && (res->modes[i].height > best.height && res->modes[i].height <= *height)) { best = res->modes[i]; } if ((res->modes[i].width < smallest.width) && res->modes[i].height < smallest.height) { smallest = res->modes[i]; } } } } if (best.id == 0 && smallest.id != 0) { best = smallest; } *width = best.width; *height = best.height; } #endif /* HAVE_XRANDR */ unsigned int XDesktop::setScreenLayout(int fb_width, int fb_height, const rfb::ScreenSet& layout) { #ifdef HAVE_XRANDR XRRScreenResources *res = XRRGetScreenResources(dpy, DefaultRootWindow(dpy)); if (!res) { vlog.error("XRRGetScreenResources failed"); return rfb::resultProhibited; } vncSetGlueContext(dpy, res); /* The client may request a screen layout which is not supported by the Xserver. This happens, for example, when adjusting the size of a non-fullscreen vncviewer window. To handle this and other cases, we first call tryScreenLayout. If this fails, we try to adjust the request to one screen with a smaller mode. */ vlog.debug("Testing screen layout"); unsigned int tryresult = ::tryScreenLayout(fb_width, fb_height, layout, &outputIdMap); rfb::ScreenSet adjustedLayout; if (tryresult == rfb::resultSuccess) { adjustedLayout = layout; } else { vlog.debug("Impossible layout - trying to adjust"); ScreenSet::const_iterator firstscreen = layout.begin(); adjustedLayout.add_screen(*firstscreen); ScreenSet::iterator iter = adjustedLayout.begin(); RROutput outputId = None; for (int i = 0;i < vncRandRGetOutputCount();i++) { unsigned int oi = vncRandRGetOutputId(i); /* Known? */ if (outputIdMap.count(oi) == 0) continue; /* Find the corresponding screen... */ if (iter->id == outputIdMap[oi]) { outputId = oi; } else { outputIdMap.erase(oi); } } /* New screen */ if (outputId == None) { int i = getPreferredScreenOutput(&outputIdMap, std::set()); if (i != -1) { outputId = vncRandRGetOutputId(i); } } if (outputId == None) { vlog.debug("Resize adjust: Could not find corresponding screen"); XRRFreeScreenResources(res); return rfb::resultInvalid; } XRROutputInfo *output = XRRGetOutputInfo(dpy, res, outputId); if (!output) { vlog.debug("Resize adjust: XRRGetOutputInfo failed"); XRRFreeScreenResources(res); return rfb::resultInvalid; } if (!output->crtc) { vlog.debug("Resize adjust: Selected output has no CRTC"); XRRFreeScreenResources(res); XRRFreeOutputInfo(output); return rfb::resultInvalid; } XRRCrtcInfo *crtc = XRRGetCrtcInfo(dpy, res, output->crtc); if (!crtc) { vlog.debug("Resize adjust: XRRGetCrtcInfo failed"); XRRFreeScreenResources(res); XRRFreeOutputInfo(output); return rfb::resultInvalid; } unsigned int swidth = iter->dimensions.width(); unsigned int sheight = iter->dimensions.height(); switch (crtc->rotation) { case RR_Rotate_90: case RR_Rotate_270: unsigned int swap = swidth; swidth = sheight; sheight = swap; break; } GetSmallerMode(res, output, &swidth, &sheight); XRRFreeOutputInfo(output); switch (crtc->rotation) { case RR_Rotate_90: case RR_Rotate_270: unsigned int swap = swidth; swidth = sheight; sheight = swap; break; } XRRFreeCrtcInfo(crtc); if (sheight != 0 && swidth != 0) { vlog.debug("Adjusted resize request to %dx%d", swidth, sheight); iter->dimensions.setXYWH(0, 0, swidth, sheight); fb_width = swidth; fb_height = sheight; } else { vlog.error("Failed to find smaller or equal screen size"); XRRFreeScreenResources(res); return rfb::resultInvalid; } } vlog.debug("Changing screen layout"); unsigned int ret = ::setScreenLayout(fb_width, fb_height, adjustedLayout, &outputIdMap); XRRFreeScreenResources(res); /* Send a dummy event to the root window. When this event is seen, earlier change events (ConfigureNotify and/or CrtcChange) have been processed. An Expose event is used for simplicity; does not require any Atoms, and will not affect other applications. */ unsigned long serial = XNextRequest(dpy); XExposeEvent ev = {}; /* zero x, y, width, height, count */ ev.type = Expose; ev.display = dpy; ev.window = DefaultRootWindow(dpy); if (XSendEvent(dpy, DefaultRootWindow(dpy), False, ExposureMask, (XEvent*)&ev)) { while (randrSyncSerial < serial) { TXWindow::handleXEvents(dpy); } } else { vlog.error("XSendEvent failed"); } /* The protocol requires that an error is returned if the requested layout could not be set. This is checked by VNCSConnectionST::setDesktopSize. Another ExtendedDesktopSize with reason=0 will be sent in response to the changes seen by the event handler. */ if (adjustedLayout != layout) return rfb::resultInvalid; // Explicitly update the server state with the result as there // can be corner cases where we don't get feedback from the X server server->setScreenLayout(computeScreenLayout()); return ret; #else (void)fb_width; (void)fb_height; (void)layout; return rfb::resultProhibited; #endif /* HAVE_XRANDR */ } bool XDesktop::handleGlobalEvent(XEvent* ev) { if (ev->type == xkbEventBase + XkbEventCode) { XkbEvent *kb = (XkbEvent *)ev; if (kb->any.xkb_type != XkbIndicatorStateNotify) return false; vlog.debug("Got indicator update, mask is now 0x%x", kb->indicators.state); ledState = 0; for (int i = 0; i < XDESKTOP_N_LEDS; i++) { if (kb->indicators.state & ledMasks[i]) ledState |= 1u << i; } if (running) server->setLEDState(ledState); return true; #ifdef HAVE_XDAMAGE } else if (ev->type == xdamageEventBase) { XDamageNotifyEvent* dev; Rect rect; if (!running) return true; dev = (XDamageNotifyEvent*)ev; rect.setXYWH(dev->area.x, dev->area.y, dev->area.width, dev->area.height); rect = rect.translate(Point(-geometry->offsetLeft(), -geometry->offsetTop())); server->add_changed(rect); return true; #endif #ifdef HAVE_XFIXES } else if (ev->type == xfixesEventBase + XFixesCursorNotify) { XFixesCursorNotifyEvent* cev; if (!running) return true; cev = (XFixesCursorNotifyEvent*)ev; if (cev->subtype != XFixesDisplayCursorNotify) return false; Window root, child; int x, y, wx, wy; unsigned int mask; // Check whether the cursor is initially on our screen if (!XQueryPointer(dpy, DefaultRootWindow(dpy), &root, &child, &x, &y, &wx, &wy, &mask)) return false; return setCursor(); } else if (ev->type == xfixesEventBase + XFixesSelectionNotify) { XFixesSelectionNotifyEvent* sev = (XFixesSelectionNotifyEvent*)ev; if (!running) return true; if (sev->subtype != XFixesSetSelectionOwnerNotify) return false; selection.handleSelectionOwnerChange(sev->owner, sev->selection, sev->timestamp); return true; #endif #ifdef HAVE_XRANDR } else if (ev->type == Expose) { XExposeEvent* eev = (XExposeEvent*)ev; randrSyncSerial = eev->serial; return false; } else if (ev->type == ConfigureNotify) { XConfigureEvent* cev = (XConfigureEvent*)ev; if (cev->window != DefaultRootWindow(dpy)) { return false; } XRRUpdateConfiguration(ev); geometry->recalc(cev->width, cev->height); if (!running) { return false; } if ((cev->width != pb->width() || (cev->height != pb->height()))) { // Recreate pixel buffer ImageFactory factory((bool)useShm); delete pb; pb = new XPixelBuffer(dpy, factory, geometry->getRect()); server->setPixelBuffer(pb, computeScreenLayout()); // Mark entire screen as changed server->add_changed(rfb::Region(Rect(0, 0, cev->width, cev->height))); } return true; } else if (ev->type == xrandrEventBase + RRNotify) { XRRNotifyEvent* rev = (XRRNotifyEvent*)ev; if (rev->window != DefaultRootWindow(dpy)) { return false; } if (!running) return false; if (rev->subtype == RRNotify_CrtcChange) { server->setScreenLayout(computeScreenLayout()); } return true; #endif #ifdef HAVE_XFIXES } else if (ev->type == EnterNotify) { XCrossingEvent* cev; if (!running) return true; cev = (XCrossingEvent*)ev; if (cev->window != cev->root) return false; return setCursor(); } else if (ev->type == LeaveNotify) { XCrossingEvent* cev; if (!running) return true; cev = (XCrossingEvent*)ev; if (cev->window == cev->root) return false; server->setCursor(0, 0, Point(), nullptr); return true; #endif } return false; } void XDesktop::queryApproved() { assert(isRunning()); server->approveConnection(queryConnectSock, true, nullptr); queryConnectSock = nullptr; } void XDesktop::queryRejected() { assert(isRunning()); server->approveConnection(queryConnectSock, false, "Connection rejected by local user"); queryConnectSock = nullptr; } #ifdef HAVE_XFIXES bool XDesktop::setCursor() { XFixesCursorImage *cim; cim = XFixesGetCursorImage(dpy); if (cim == nullptr) return false; // Copied from XserverDesktop::setCursor() in // unix/xserver/hw/vnc/XserverDesktop.cc and adapted to // handle long -> uint32_t conversion for 64-bit Xlib uint8_t* cursorData; uint8_t *out; const unsigned long *pixels; cursorData = new uint8_t[cim->width * cim->height * 4]; // Un-premultiply alpha pixels = cim->pixels; out = cursorData; for (int y = 0; y < cim->height; y++) { for (int x = 0; x < cim->width; x++) { uint8_t alpha; uint32_t pixel = *pixels++; alpha = (pixel >> 24) & 0xff; if (alpha == 0) alpha = 1; // Avoid division by zero *out++ = ((pixel >> 16) & 0xff) * 255/alpha; *out++ = ((pixel >> 8) & 0xff) * 255/alpha; *out++ = ((pixel >> 0) & 0xff) * 255/alpha; *out++ = ((pixel >> 24) & 0xff); } } try { server->setCursor(cim->width, cim->height, Point(cim->xhot, cim->yhot), cursorData); } catch (std::exception& e) { vlog.error("XserverDesktop::setCursor: %s",e.what()); } delete [] cursorData; XFree(cim); return true; } #endif // X selection availability changed, let VNC clients know void XDesktop::handleXSelectionAnnounce(bool available) { server->announceClipboard(available); } // A VNC client wants data, send request to selection owner void XDesktop::handleClipboardRequest() { selection.requestSelectionData(); } // Data is available, send it to clients void XDesktop::handleXSelectionData(const char* data) { server->sendClipboardData(data); } // When a client says it has clipboard data, request it void XDesktop::handleClipboardAnnounce(bool available) { if(available) server->requestClipboard(); } // Client has sent the data void XDesktop::handleClipboardData(const char* data) { if (data) selection.handleClientClipboardData(data); }