/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved. * Copyright (C) 2004-2008 Constantin Kaplinsky. All Rights Reserved. * * 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. */ // FIXME: Check cases when screen width/height is not a multiply of 32. // e.g. 800x600. #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef HAVE_LIBSYSTEMD # include #endif #include #include #include #include #include #include #include #include extern char buildtime[]; using namespace rfb; using namespace network; static LogWriter vlog("Main"); static const char* defaultDesktopName(); IntParameter pollingCycle("PollingCycle", "Milliseconds per one polling " "cycle; actual interval may be dynamically " "adjusted to satisfy MaxProcessorUsage setting", 30); IntParameter maxProcessorUsage("MaxProcessorUsage", "Maximum percentage of " "CPU time to be consumed", 35); StringParameter desktopName("desktop", "Name of VNC desktop", defaultDesktopName()); StringParameter displayname("display", "The X display", ""); IntParameter rfbport("rfbport", "TCP port to listen for RFB protocol",5900); StringParameter rfbunixpath("rfbunixpath", "Unix socket to listen for RFB protocol", ""); IntParameter rfbunixmode("rfbunixmode", "Unix socket access mode", 0600); StringParameter hostsFile("HostsFile", "File with IP access control rules", ""); BoolParameter localhostOnly("localhost", "Only allow connections from localhost", false); StringParameter interface("interface", "listen on the specified network address", "all"); static const char* defaultDesktopName() { long host_max = sysconf(_SC_HOST_NAME_MAX); if (host_max < 0) return ""; std::vector hostname(host_max + 1); if (gethostname(hostname.data(), hostname.size()) == -1) return ""; struct passwd* pwent = getpwuid(getuid()); if (pwent == nullptr) return ""; int len = snprintf(nullptr, 0, "%s@%s", pwent->pw_name, hostname.data()); if (len < 0) return ""; char* name = new char[len + 1]; snprintf(name, len + 1, "%s@%s", pwent->pw_name, hostname.data()); return name; } // // Allow the main loop terminate itself gracefully on receiving a signal. // static bool caughtSignal = false; static void CleanupSignalHandler(int /*sig*/) { caughtSignal = true; } static bool hasSystemdListeners() { #ifdef HAVE_LIBSYSTEMD // This also returns true on errors, because we then assume we were // meant to use systemd but failed somewhere return sd_listen_fds(0) != 0; #else return false; #endif } static int createSystemdListeners(std::list *listeners) { #ifdef HAVE_LIBSYSTEMD int count = sd_listen_fds(0); if (count < 0) { vlog.error("Error getting listening sockets from systemd: %s", strerror(-count)); return count; } for (int i = 0; i < count; ++i) listeners->push_back(new TcpListener(SD_LISTEN_FDS_START + i)); return count; #else (void)listeners; return 0; #endif } class FileTcpFilter : public TcpFilter { public: FileTcpFilter(const char *fname) : TcpFilter("-"), fileName(nullptr), lastModTime(0) { if (fname != nullptr) fileName = strdup((char *)fname); } virtual ~FileTcpFilter() { if (fileName != nullptr) free(fileName); } bool verifyConnection(Socket* s) override { if (!reloadRules()) { vlog.error("Could not read IP filtering rules: rejecting all clients"); filter.clear(); filter.push_back(parsePattern("-")); return false; } return TcpFilter::verifyConnection(s); } protected: bool reloadRules() { if (fileName == nullptr) return true; struct stat st; if (stat(fileName, &st) != 0) return false; if (st.st_mtime != lastModTime) { // Actually reload only if the file was modified FILE *fp = fopen(fileName, "r"); if (fp == nullptr) return false; // Remove all the rules from the parent class filter.clear(); // Parse the file contents adding rules to the parent class char buf[32]; while (readLine(buf, 32, fp)) { if (buf[0] && strchr("+-?", buf[0])) { filter.push_back(parsePattern(buf)); } } fclose(fp); lastModTime = st.st_mtime; } return true; } protected: char *fileName; time_t lastModTime; private: // // NOTE: we silently truncate long lines in this function. // bool readLine(char *buf, int bufSize, FILE *fp) { if (fp == nullptr || buf == nullptr || bufSize == 0) return false; if (fgets(buf, bufSize, fp) == nullptr) return false; char *ptr = strchr(buf, '\n'); if (ptr != nullptr) { *ptr = '\0'; // remove newline at the end } else { if (!feof(fp)) { int c; do { // skip the rest of a long line c = getc(fp); } while (c != '\n' && c != EOF); } } return true; } }; char* programName; static void printVersion(FILE *fp) { fprintf(fp, "TigerVNC Server version %s, built %s\n", PACKAGE_VERSION, buildtime); } static void usage() { printVersion(stderr); fprintf(stderr, "\nUsage: %s []\n", programName); fprintf(stderr, " %s --version\n", programName); fprintf(stderr,"\n" "Parameters can be turned on with - or off with -=0\n" "Parameters which take a value can be specified as " "- \n" "Other valid forms are = -= " "--=\n" "Parameter names are case-insensitive. The parameters are:\n\n"); Configuration::listParams(79, 14); exit(1); } int main(int argc, char** argv) { initStdIOLoggers(); LogWriter::setLogParams("*:stderr:30"); programName = argv[0]; Display* dpy; Configuration::enableServerParams(); // FIXME: We don't support clipboard yet Configuration::removeParam("AcceptCutText"); Configuration::removeParam("SendCutText"); Configuration::removeParam("MaxCutText"); // Assume different defaults when socket activated if (hasSystemdListeners()) rfbport.setParam(-1); for (int i = 1; i < argc; i++) { if (Configuration::setParam(argv[i])) continue; if (argv[i][0] == '-') { if (i+1 < argc) { if (Configuration::setParam(&argv[i][1], argv[i+1])) { i++; continue; } } if (strcmp(argv[i], "-v") == 0 || strcmp(argv[i], "-version") == 0 || strcmp(argv[i], "--version") == 0) { printVersion(stdout); return 0; } usage(); } usage(); } if (!(dpy = XOpenDisplay(displayname))) { // FIXME: Why not vlog.error(...)? fprintf(stderr,"%s: unable to open display \"%s\"\r\n", programName, XDisplayName(displayname)); exit(1); } signal(SIGHUP, CleanupSignalHandler); signal(SIGINT, CleanupSignalHandler); signal(SIGTERM, CleanupSignalHandler); std::list listeners; try { TXWindow::init(dpy,"x0vncserver"); Geometry geo(DisplayWidth(dpy, DefaultScreen(dpy)), DisplayHeight(dpy, DefaultScreen(dpy))); if (geo.getRect().is_empty()) { vlog.error("Exiting with error"); return 1; } XDesktop desktop(dpy, &geo); VNCServerST server(desktopName, &desktop); if (createSystemdListeners(&listeners) > 0) { // When systemd is in charge of listeners, do not listen to anything else vlog.info("Listening on systemd sockets"); } if (rfbunixpath.getValueStr()[0] != '\0') { listeners.push_back(new network::UnixListener(rfbunixpath, rfbunixmode)); vlog.info("Listening on %s (mode %04o)", (const char*)rfbunixpath, (int)rfbunixmode); } if ((int)rfbport != -1) { std::list tcp_listeners; const char *addr = interface; if (strcasecmp(addr, "all") == 0) addr = nullptr; if (localhostOnly) createLocalTcpListeners(&tcp_listeners, (int)rfbport); else createTcpListeners(&tcp_listeners, addr, (int)rfbport); if (!tcp_listeners.empty()) { listeners.splice (listeners.end(), tcp_listeners); vlog.info("Listening for VNC connections on %s interface(s), port %d", localhostOnly ? "local" : (const char*)interface, (int)rfbport); } FileTcpFilter fileTcpFilter(hostsFile); if (strlen(hostsFile) != 0) for (SocketListener* listener : listeners) listener->setFilter(&fileTcpFilter); } if (listeners.empty()) { vlog.error("No path or port configured for incoming connections"); return -1; } PollingScheduler sched((int)pollingCycle, (int)maxProcessorUsage); while (!caughtSignal) { int wait_ms, nextTimeout; struct timeval tv; fd_set rfds, wfds; std::list sockets; std::list::iterator i; // Process any incoming X events TXWindow::handleXEvents(dpy); FD_ZERO(&rfds); FD_ZERO(&wfds); FD_SET(ConnectionNumber(dpy), &rfds); for (SocketListener* listener : listeners) FD_SET(listener->getFd(), &rfds); server.getSockets(&sockets); int clients_connected = 0; for (i = sockets.begin(); i != sockets.end(); i++) { if ((*i)->isShutdown()) { server.removeSocket(*i); delete (*i); } else { FD_SET((*i)->getFd(), &rfds); if ((*i)->outStream().hasBufferedData()) FD_SET((*i)->getFd(), &wfds); clients_connected++; } } if (!clients_connected) sched.reset(); wait_ms = -1; if (sched.isRunning()) { wait_ms = sched.millisRemaining(); if (wait_ms > 500) { wait_ms = 500; } } // Trigger timers and check when the next will expire nextTimeout = Timer::checkTimeouts(); if (nextTimeout >= 0 && (wait_ms == -1 || nextTimeout < wait_ms)) wait_ms = nextTimeout; tv.tv_sec = wait_ms / 1000; tv.tv_usec = (wait_ms % 1000) * 1000; // Do the wait... sched.sleepStarted(); int n = select(FD_SETSIZE, &rfds, &wfds, nullptr, wait_ms >= 0 ? &tv : nullptr); sched.sleepFinished(); if (n < 0) { if (errno == EINTR) { vlog.debug("Interrupted select() system call"); continue; } else { throw rdr::SystemException("select", errno); } } // Accept new VNC connections for (SocketListener* listener : listeners) { if (FD_ISSET(listener->getFd(), &rfds)) { Socket* sock = listener->accept(); if (sock) { server.addSocket(sock); } else { vlog.status("Client connection rejected"); } } } Timer::checkTimeouts(); // Client list could have been changed. server.getSockets(&sockets); // Nothing more to do if there are no client connections. if (sockets.empty()) continue; // Process events on existing VNC connections for (i = sockets.begin(); i != sockets.end(); i++) { if (FD_ISSET((*i)->getFd(), &rfds)) server.processSocketReadEvent(*i); if (FD_ISSET((*i)->getFd(), &wfds)) server.processSocketWriteEvent(*i); } if (desktop.isRunning() && sched.goodTimeToPoll()) { sched.newPass(); desktop.poll(); } } } catch (rdr::Exception &e) { vlog.error("%s", e.str()); return 1; } TXWindow::handleXEvents(dpy); // Run listener destructors; remove UNIX sockets etc for (SocketListener* listener : listeners) delete listener; vlog.info("Terminated"); return 0; }