blob: cf37fddd6db346276753332dc77459a223fc33d4 [file] [log] [blame]
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved.
Pierre Ossmancef3cf72016-11-25 10:06:34 +01002 * Copyright 2009-2016 Pierre Ossman for Cendio AB
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00003 *
4 * This is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; either version 2 of the License, or
7 * (at your option) any later version.
8 *
9 * This software is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this software; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
17 * USA.
18 */
19
Pierre Ossman1b478e52011-11-15 12:08:30 +000020// Debug output on what the congestion control is up to
21#undef CONGESTION_DEBUG
22
23#include <sys/time.h>
24
25#ifdef CONGESTION_DEBUG
26#include <sys/socket.h>
27#include <netinet/in.h>
28#include <netinet/tcp.h>
29#endif
30
Pierre Ossmana830bec2011-11-08 12:12:02 +000031#include <network/TcpSocket.h>
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000032#include <rfb/VNCSConnectionST.h>
33#include <rfb/LogWriter.h>
Adam Tkac5a0caed2010-04-23 13:58:10 +000034#include <rfb/Security.h>
Pierre Ossmanc5e25602009-03-20 12:59:05 +000035#include <rfb/screenTypes.h>
Pierre Ossman2c764942011-11-14 15:54:30 +000036#include <rfb/fenceTypes.h>
Pierre Ossmanbb305ca2016-12-11 12:41:26 +010037#include <rfb/ledStates.h>
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000038#include <rfb/ServerCore.h>
39#include <rfb/ComparingUpdateTracker.h>
40#include <rfb/KeyRemapper.h>
Pierre Ossmanfdba3fe2014-01-31 13:12:18 +010041#include <rfb/Encoder.h>
Pierre Ossmanbb305ca2016-12-11 12:41:26 +010042#define XK_LATIN1
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000043#define XK_MISCELLANY
44#define XK_XKB_KEYS
45#include <rfb/keysymdef.h>
46
47using namespace rfb;
48
49static LogWriter vlog("VNCSConnST");
50
Pierre Ossman1b478e52011-11-15 12:08:30 +000051// This window should get us going fairly fast on a decent bandwidth network.
52// If it's too high, it will rapidly be reduced and stay low.
53static const unsigned INITIAL_WINDOW = 16384;
54
55// TCP's minimal window is 3*MSS. But since we don't know the MSS, we
klemens0536d092017-01-28 20:56:56 +010056// make a guess at 4 KiB (it's probably a bit higher).
Pierre Ossman1b478e52011-11-15 12:08:30 +000057static const unsigned MINIMUM_WINDOW = 4096;
58
59// The current default maximum window for Linux (4 MiB). Should be a good
60// limit for now...
61static const unsigned MAXIMUM_WINDOW = 4194304;
62
63struct RTTInfo {
64 struct timeval tv;
65 int offset;
66 unsigned inFlight;
67};
68
Pierre Ossman71ca8d52017-09-15 11:03:12 +020069static Cursor emptyCursor(0, 0, Point(0, 0), NULL);
70
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000071VNCSConnectionST::VNCSConnectionST(VNCServerST* server_, network::Socket *s,
72 bool reverse)
Pierre Ossman7069bdd2015-02-06 14:41:58 +010073 : sock(s), reverseConnection(reverse),
Pierre Ossmanf8e3b342015-01-26 14:37:04 +010074 queryConnectTimer(this), inProcessMessages(false),
Pierre Ossmanb8b1e962012-07-20 10:47:00 +000075 pendingSyncFence(false), syncFence(false), fenceFlags(0),
76 fenceDataLen(0), fenceData(NULL),
Pierre Ossmanb1cd6ca2015-03-03 16:37:43 +010077 baseRTT(-1), congWindow(0), ackedOffset(0), sentOffset(0),
78 minRTT(-1), seenCongestion(false),
79 pingCounter(0), congestionTimer(this),
Pierre Ossman0c9bd4b2014-07-09 16:44:11 +020080 server(server_), updates(false),
Pierre Ossman671d2ef2015-06-09 16:09:06 +020081 updateRenderedCursor(false), removeRenderedCursor(false),
Pierre Ossmana40ab202016-04-29 15:35:56 +020082 continuousUpdates(false), encodeManager(this), pointerEventTime(0),
Pierre Ossman1bb8b6c2011-10-25 15:20:05 +000083 accessRights(AccessDefault), startTime(time(0))
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000084{
85 setStreams(&sock->inStream(), &sock->outStream());
86 peerEndpoint.buf = sock->getPeerEndpoint();
87 VNCServerST::connectionsLog.write(1,"accepted: %s", peerEndpoint.buf);
88
89 // Configure the socket
90 setSocketTimeouts();
91 lastEventTime = time(0);
92
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000093 server->clients.push_front(this);
94}
95
96
97VNCSConnectionST::~VNCSConnectionST()
98{
99 // If we reach here then VNCServerST is deleting us!
100 VNCServerST::connectionsLog.write(1,"closed: %s (%s)",
101 peerEndpoint.buf,
102 (closeReason.buf) ? closeReason.buf : "");
103
104 // Release any keys the client still had pressed
Pierre Ossman16e1dcb2017-05-16 14:33:43 +0200105 while (!pressedKeys.empty()) {
106 rdr::U32 keysym, keycode;
107
108 keysym = pressedKeys.begin()->second;
109 keycode = pressedKeys.begin()->first;
110 pressedKeys.erase(pressedKeys.begin());
111
112 vlog.debug("Releasing key 0x%x / 0x%x on client disconnect",
113 keysym, keycode);
114 server->desktop->keyEvent(keysym, keycode, false);
Pierre Ossman9a153b02015-08-31 10:01:14 +0200115 }
Pierre Ossman16e1dcb2017-05-16 14:33:43 +0200116
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000117 if (server->pointerClient == this)
118 server->pointerClient = 0;
119
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000120 // Remove this client from the server
121 server->clients.remove(this);
122
Pierre Ossman2c764942011-11-14 15:54:30 +0000123 delete [] fenceData;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000124}
125
126
127// Methods called from VNCServerST
128
129bool VNCSConnectionST::init()
130{
131 try {
132 initialiseProtocol();
133 } catch (rdr::Exception& e) {
134 close(e.str());
135 return false;
136 }
137 return true;
138}
139
140void VNCSConnectionST::close(const char* reason)
141{
142 // Log the reason for the close
143 if (!closeReason.buf)
Adam Tkacd36b6262009-09-04 10:57:20 +0000144 closeReason.buf = strDup(reason);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000145 else
146 vlog.debug("second close: %s (%s)", peerEndpoint.buf, reason);
147
148 if (authenticated()) {
149 server->lastDisconnectTime = time(0);
150 }
151
152 // Just shutdown the socket and mark our state as closing. Eventually the
153 // calling code will call VNCServerST's removeSocket() method causing us to
154 // be deleted.
155 sock->shutdown();
156 setState(RFBSTATE_CLOSING);
157}
158
159
160void VNCSConnectionST::processMessages()
161{
162 if (state() == RFBSTATE_CLOSING) return;
163 try {
164 // - Now set appropriate socket timeouts and process data
165 setSocketTimeouts();
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000166
167 inProcessMessages = true;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000168
Pierre Ossmana830bec2011-11-08 12:12:02 +0000169 // Get the underlying TCP layer to build large packets if we send
170 // multiple small responses.
171 network::TcpSocket::cork(sock->getFd(), true);
172
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000173 while (getInStream()->checkNoWait(1)) {
Pierre Ossmanb8b1e962012-07-20 10:47:00 +0000174 if (pendingSyncFence) {
175 syncFence = true;
176 pendingSyncFence = false;
177 }
178
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000179 processMsg();
Pierre Ossmanb8b1e962012-07-20 10:47:00 +0000180
Pierre Ossman2c764942011-11-14 15:54:30 +0000181 if (syncFence) {
182 writer()->writeFence(fenceFlags, fenceDataLen, fenceData);
183 syncFence = false;
184 }
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000185 }
186
Pierre Ossmana830bec2011-11-08 12:12:02 +0000187 // Flush out everything in case we go idle after this.
188 network::TcpSocket::cork(sock->getFd(), false);
189
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000190 inProcessMessages = false;
Constantin Kaplinsky2ef66952008-08-29 11:33:46 +0000191
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000192 // If there were anything requiring an update, try to send it here.
193 // We wait until now with this to aggregate responses and to give
194 // higher priority to user actions such as keyboard and pointer events.
195 writeFramebufferUpdate();
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000196 } catch (rdr::EndOfStream&) {
197 close("Clean disconnection");
198 } catch (rdr::Exception &e) {
199 close(e.str());
200 }
201}
202
Pierre Ossmand408ca52016-04-29 14:26:05 +0200203void VNCSConnectionST::flushSocket()
204{
205 if (state() == RFBSTATE_CLOSING) return;
206 try {
207 setSocketTimeouts();
208 sock->outStream().flush();
Pierre Ossmana40ab202016-04-29 15:35:56 +0200209 // Flushing the socket might release an update that was previously
210 // delayed because of congestion.
211 if (sock->outStream().bufferUsage() == 0)
212 writeFramebufferUpdate();
Pierre Ossmand408ca52016-04-29 14:26:05 +0200213 } catch (rdr::Exception &e) {
214 close(e.str());
215 }
216}
217
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000218void VNCSConnectionST::pixelBufferChange()
219{
220 try {
221 if (!authenticated()) return;
222 if (cp.width && cp.height && (server->pb->width() != cp.width ||
223 server->pb->height() != cp.height))
224 {
225 // We need to clip the next update to the new size, but also add any
226 // extra bits if it's bigger. If we wanted to do this exactly, something
227 // like the code below would do it, but at the moment we just update the
Pierre Ossman671d2ef2015-06-09 16:09:06 +0200228 // entire new size. However, we do need to clip the damagedCursorRegion
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000229 // because that might be added to updates in writeFramebufferUpdate().
230
231 //updates.intersect(server->pb->getRect());
232 //
233 //if (server->pb->width() > cp.width)
234 // updates.add_changed(Rect(cp.width, 0, server->pb->width(),
235 // server->pb->height()));
236 //if (server->pb->height() > cp.height)
237 // updates.add_changed(Rect(0, cp.height, cp.width,
238 // server->pb->height()));
239
Pierre Ossman671d2ef2015-06-09 16:09:06 +0200240 damagedCursorRegion.assign_intersect(server->pb->getRect());
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000241
242 cp.width = server->pb->width();
243 cp.height = server->pb->height();
Pierre Ossman34e62f32009-03-20 21:46:12 +0000244 cp.screenLayout = server->screenLayout;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000245 if (state() == RFBSTATE_NORMAL) {
Pierre Ossman2ee430a2009-05-28 12:54:24 +0000246 // We should only send EDS to client asking for both
247 if (!writer()->writeExtendedDesktopSize()) {
248 if (!writer()->writeSetDesktopSize()) {
249 close("Client does not support desktop resize");
250 return;
251 }
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000252 }
253 }
254 }
255 // Just update the whole screen at the moment because we're too lazy to
256 // work out what's actually changed.
257 updates.clear();
258 updates.add_changed(server->pb->getRect());
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000259 writeFramebufferUpdate();
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000260 } catch(rdr::Exception &e) {
261 close(e.str());
262 }
263}
264
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000265void VNCSConnectionST::writeFramebufferUpdateOrClose()
Pierre Ossman04e62db2009-03-23 16:57:07 +0000266{
267 try {
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000268 writeFramebufferUpdate();
269 } catch(rdr::Exception &e) {
270 close(e.str());
271 }
272}
Pierre Ossman04e62db2009-03-23 16:57:07 +0000273
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000274void VNCSConnectionST::screenLayoutChangeOrClose(rdr::U16 reason)
275{
276 try {
277 screenLayoutChange(reason);
Pierre Ossman04e62db2009-03-23 16:57:07 +0000278 } catch(rdr::Exception &e) {
279 close(e.str());
280 }
281}
282
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000283void VNCSConnectionST::bellOrClose()
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000284{
285 try {
286 if (state() == RFBSTATE_NORMAL) writer()->writeBell();
287 } catch(rdr::Exception& e) {
288 close(e.str());
289 }
290}
291
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000292void VNCSConnectionST::serverCutTextOrClose(const char *str, int len)
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000293{
294 try {
295 if (!(accessRights & AccessCutText)) return;
296 if (!rfb::Server::sendCutText) return;
297 if (state() == RFBSTATE_NORMAL)
298 writer()->writeServerCutText(str, len);
299 } catch(rdr::Exception& e) {
300 close(e.str());
301 }
302}
303
Peter Ã…strandc39e0782009-01-15 12:21:42 +0000304
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000305void VNCSConnectionST::setDesktopNameOrClose(const char *name)
Peter Ã…strandc39e0782009-01-15 12:21:42 +0000306{
Peter Ã…strandc39e0782009-01-15 12:21:42 +0000307 try {
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000308 setDesktopName(name);
Peter Ã…strandc39e0782009-01-15 12:21:42 +0000309 } catch(rdr::Exception& e) {
310 close(e.str());
311 }
312}
313
314
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000315void VNCSConnectionST::setCursorOrClose()
316{
317 try {
318 setCursor();
319 } catch(rdr::Exception& e) {
320 close(e.str());
321 }
322}
323
324
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100325void VNCSConnectionST::setLEDStateOrClose(unsigned int state)
326{
327 try {
328 setLEDState(state);
329 } catch(rdr::Exception& e) {
330 close(e.str());
331 }
332}
333
334
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000335int VNCSConnectionST::checkIdleTimeout()
336{
337 int idleTimeout = rfb::Server::idleTimeout;
338 if (idleTimeout == 0) return 0;
339 if (state() != RFBSTATE_NORMAL && idleTimeout < 15)
340 idleTimeout = 15; // minimum of 15 seconds while authenticating
341 time_t now = time(0);
342 if (now < lastEventTime) {
343 // Someone must have set the time backwards. Set lastEventTime so that the
344 // idleTimeout will count from now.
345 vlog.info("Time has gone backwards - resetting idle timeout");
346 lastEventTime = now;
347 }
348 int timeLeft = lastEventTime + idleTimeout - now;
349 if (timeLeft < -60) {
350 // Our callback is over a minute late - someone must have set the time
351 // forwards. Set lastEventTime so that the idleTimeout will count from
352 // now.
353 vlog.info("Time has gone forwards - resetting idle timeout");
354 lastEventTime = now;
355 return secsToMillis(idleTimeout);
356 }
357 if (timeLeft <= 0) {
358 close("Idle timeout");
359 return 0;
360 }
361 return secsToMillis(timeLeft);
362}
363
Pierre Ossmanb114cec2011-11-20 15:36:11 +0000364
365bool VNCSConnectionST::getComparerState()
366{
367 // We interpret a low compression level as an indication that the client
368 // wants to prioritise CPU usage over bandwidth, and hence disable the
369 // comparing update tracker.
370 return (cp.compressLevel == -1) || (cp.compressLevel > 1);
371}
372
373
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000374// renderedCursorChange() is called whenever the server-side rendered cursor
375// changes shape or position. It ensures that the next update will clean up
376// the old rendered cursor and if necessary draw the new rendered cursor.
377
378void VNCSConnectionST::renderedCursorChange()
379{
380 if (state() != RFBSTATE_NORMAL) return;
Pierre Ossman71ca8d52017-09-15 11:03:12 +0200381 // Are we switching between client-side and server-side cursor?
382 if (damagedCursorRegion.is_empty() != needRenderedCursor())
383 setCursorOrClose();
Pierre Ossman671d2ef2015-06-09 16:09:06 +0200384 if (!damagedCursorRegion.is_empty())
Pierre Ossman5c9e1e52011-11-08 10:32:05 +0000385 removeRenderedCursor = true;
Pierre Ossman6b0bc292011-12-21 13:17:54 +0000386 if (needRenderedCursor()) {
Pierre Ossman671d2ef2015-06-09 16:09:06 +0200387 updateRenderedCursor = true;
Pierre Ossman6b0bc292011-12-21 13:17:54 +0000388 writeFramebufferUpdateOrClose();
389 }
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000390}
391
392// needRenderedCursor() returns true if this client needs the server-side
393// rendered cursor. This may be because it does not support local cursor or
394// because the current cursor position has not been set by this client.
395// Unfortunately we can't know for sure when the current cursor position has
396// been set by this client. We guess that this is the case when the current
397// cursor position is the same as the last pointer event from this client, or
398// if it is a very short time since this client's last pointer event (up to a
399// second). [ Ideally we should do finer-grained timing here and make the time
400// configurable, but I don't think it's that important. ]
401
402bool VNCSConnectionST::needRenderedCursor()
403{
Pierre Ossman77ede0a2016-12-05 16:57:30 +0100404 if (state() != RFBSTATE_NORMAL)
405 return false;
406
Pierre Ossman324043e2017-08-16 16:26:11 +0200407 if (!cp.supportsLocalCursorWithAlpha &&
408 !cp.supportsLocalCursor && !cp.supportsLocalXCursor)
Pierre Ossman77ede0a2016-12-05 16:57:30 +0100409 return true;
410 if (!server->cursorPos.equals(pointerEventPos) &&
411 (time(0) - pointerEventTime) > 0)
412 return true;
413
414 return false;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000415}
416
417
418void VNCSConnectionST::approveConnectionOrClose(bool accept,
419 const char* reason)
420{
421 try {
422 approveConnection(accept, reason);
423 } catch (rdr::Exception& e) {
424 close(e.str());
425 }
426}
427
428
429
430// -=- Callbacks from SConnection
431
432void VNCSConnectionST::authSuccess()
433{
434 lastEventTime = time(0);
435
436 server->startDesktop();
437
438 // - Set the connection parameters appropriately
439 cp.width = server->pb->width();
440 cp.height = server->pb->height();
Pierre Ossman34e62f32009-03-20 21:46:12 +0000441 cp.screenLayout = server->screenLayout;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000442 cp.setName(server->getName());
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100443 cp.setLEDState(server->ledState);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000444
445 // - Set the default pixel format
446 cp.setPF(server->pb->getPF());
447 char buffer[256];
448 cp.pf().print(buffer, 256);
449 vlog.info("Server default pixel format %s", buffer);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000450
451 // - Mark the entire display as "dirty"
452 updates.add_changed(server->pb->getRect());
453 startTime = time(0);
Pierre Ossman1b478e52011-11-15 12:08:30 +0000454
455 // - Bootstrap the congestion control
456 ackedOffset = sock->outStream().length();
457 congWindow = INITIAL_WINDOW;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000458}
459
460void VNCSConnectionST::queryConnection(const char* userName)
461{
462 // - Authentication succeeded - clear from blacklist
463 CharArray name; name.buf = sock->getPeerAddress();
464 server->blHosts->clearBlackmark(name.buf);
465
466 // - Special case to provide a more useful error message
467 if (rfb::Server::neverShared && !rfb::Server::disconnectClients &&
468 server->authClientCount() > 0) {
469 approveConnection(false, "The server is already in use");
470 return;
471 }
472
473 // - Does the client have the right to bypass the query?
474 if (reverseConnection ||
475 !(rfb::Server::queryConnect || sock->requiresQuery()) ||
476 (accessRights & AccessNoQuery))
477 {
478 approveConnection(true);
479 return;
480 }
481
482 // - Get the server to display an Accept/Reject dialog, if required
483 // If a dialog is displayed, the result will be PENDING, and the
484 // server will call approveConnection at a later time
485 CharArray reason;
486 VNCServerST::queryResult qr = server->queryConnection(sock, userName,
487 &reason.buf);
Pierre Ossmanf8e3b342015-01-26 14:37:04 +0100488 if (qr == VNCServerST::PENDING) {
489 queryConnectTimer.start(rfb::Server::queryConnectTimeout * 1000);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000490 return;
Pierre Ossmanf8e3b342015-01-26 14:37:04 +0100491 }
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000492
493 // - If server returns ACCEPT/REJECT then pass result to SConnection
494 approveConnection(qr == VNCServerST::ACCEPT, reason.buf);
495}
496
497void VNCSConnectionST::clientInit(bool shared)
498{
499 lastEventTime = time(0);
500 if (rfb::Server::alwaysShared || reverseConnection) shared = true;
Pierre Ossmane7be49b2014-12-02 14:33:17 +0100501 if (!(accessRights & AccessNonShared)) shared = true;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000502 if (rfb::Server::neverShared) shared = false;
503 if (!shared) {
Pierre Ossmane7be49b2014-12-02 14:33:17 +0100504 if (rfb::Server::disconnectClients && (accessRights & AccessNonShared)) {
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000505 // - Close all the other connected clients
506 vlog.debug("non-shared connection - closing clients");
507 server->closeClients("Non-shared connection requested", getSock());
508 } else {
509 // - Refuse this connection if there are existing clients, in addition to
510 // this one
511 if (server->authClientCount() > 1) {
512 close("Server is already in use");
513 return;
514 }
515 }
516 }
517 SConnection::clientInit(shared);
518}
519
520void VNCSConnectionST::setPixelFormat(const PixelFormat& pf)
521{
522 SConnection::setPixelFormat(pf);
523 char buffer[256];
524 pf.print(buffer, 256);
525 vlog.info("Client pixel format %s", buffer);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000526 setCursor();
527}
528
529void VNCSConnectionST::pointerEvent(const Point& pos, int buttonMask)
530{
531 pointerEventTime = lastEventTime = time(0);
532 server->lastUserInputTime = lastEventTime;
533 if (!(accessRights & AccessPtrEvents)) return;
534 if (!rfb::Server::acceptPointerEvents) return;
535 if (!server->pointerClient || server->pointerClient == this) {
536 pointerEventPos = pos;
537 if (buttonMask)
538 server->pointerClient = this;
539 else
540 server->pointerClient = 0;
541 server->desktop->pointerEvent(pointerEventPos, buttonMask);
542 }
543}
544
545
546class VNCSConnectionSTShiftPresser {
547public:
548 VNCSConnectionSTShiftPresser(SDesktop* desktop_)
549 : desktop(desktop_), pressed(false) {}
550 ~VNCSConnectionSTShiftPresser() {
Pierre Ossman9a153b02015-08-31 10:01:14 +0200551 if (pressed) {
552 vlog.debug("Releasing fake Shift_L");
Pierre Ossman5ae28212017-05-16 14:30:38 +0200553 desktop->keyEvent(XK_Shift_L, 0, false);
Pierre Ossman9a153b02015-08-31 10:01:14 +0200554 }
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000555 }
556 void press() {
Pierre Ossman9a153b02015-08-31 10:01:14 +0200557 vlog.debug("Pressing fake Shift_L");
Pierre Ossman5ae28212017-05-16 14:30:38 +0200558 desktop->keyEvent(XK_Shift_L, 0, true);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000559 pressed = true;
560 }
561 SDesktop* desktop;
562 bool pressed;
563};
564
565// keyEvent() - record in the pressedKeys which keys were pressed. Allow
566// multiple down events (for autorepeat), but only allow a single up event.
Pierre Ossman5ae28212017-05-16 14:30:38 +0200567void VNCSConnectionST::keyEvent(rdr::U32 keysym, rdr::U32 keycode, bool down) {
Pierre Ossman16e1dcb2017-05-16 14:33:43 +0200568 rdr::U32 lookup;
569
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000570 lastEventTime = time(0);
571 server->lastUserInputTime = lastEventTime;
572 if (!(accessRights & AccessKeyEvents)) return;
573 if (!rfb::Server::acceptKeyEvents) return;
574
Pierre Ossman9a153b02015-08-31 10:01:14 +0200575 if (down)
Pierre Ossman5ae28212017-05-16 14:30:38 +0200576 vlog.debug("Key pressed: 0x%x / 0x%x", keysym, keycode);
Pierre Ossman9a153b02015-08-31 10:01:14 +0200577 else
Pierre Ossman5ae28212017-05-16 14:30:38 +0200578 vlog.debug("Key released: 0x%x / 0x%x", keysym, keycode);
Pierre Ossman9a153b02015-08-31 10:01:14 +0200579
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000580 // Remap the key if required
Pierre Ossman9a153b02015-08-31 10:01:14 +0200581 if (server->keyRemapper) {
582 rdr::U32 newkey;
Pierre Ossman5ae28212017-05-16 14:30:38 +0200583 newkey = server->keyRemapper->remapKey(keysym);
584 if (newkey != keysym) {
Pierre Ossman9a153b02015-08-31 10:01:14 +0200585 vlog.debug("Key remapped to 0x%x", newkey);
Pierre Ossman5ae28212017-05-16 14:30:38 +0200586 keysym = newkey;
Pierre Ossman9a153b02015-08-31 10:01:14 +0200587 }
588 }
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000589
Pierre Ossmanbb305ca2016-12-11 12:41:26 +0100590 // Avoid lock keys if we don't know the server state
591 if ((server->ledState == ledUnknown) &&
Pierre Ossman5ae28212017-05-16 14:30:38 +0200592 ((keysym == XK_Caps_Lock) ||
593 (keysym == XK_Num_Lock) ||
594 (keysym == XK_Scroll_Lock))) {
Pierre Ossmanbb305ca2016-12-11 12:41:26 +0100595 vlog.debug("Ignoring lock key (e.g. caps lock)");
596 return;
597 }
Pierre Ossmanbb305ca2016-12-11 12:41:26 +0100598
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100599 // Lock key heuristics
600 // (only for clients that do not support the LED state extension)
601 if (!cp.supportsLEDState) {
602 // Always ignore ScrollLock as we don't have a heuristic
603 // for that
Pierre Ossman5ae28212017-05-16 14:30:38 +0200604 if (keysym == XK_Scroll_Lock) {
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100605 vlog.debug("Ignoring lock key (e.g. caps lock)");
606 return;
Pierre Ossmanbb305ca2016-12-11 12:41:26 +0100607 }
608
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100609 if (down && (server->ledState != ledUnknown)) {
610 // CapsLock synchronisation heuristic
611 // (this assumes standard interaction between CapsLock the Shift
612 // keys and normal characters)
Pierre Ossman5ae28212017-05-16 14:30:38 +0200613 if (((keysym >= XK_A) && (keysym <= XK_Z)) ||
614 ((keysym >= XK_a) && (keysym <= XK_z))) {
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100615 bool uppercase, shift, lock;
Pierre Ossmanbb305ca2016-12-11 12:41:26 +0100616
Pierre Ossman5ae28212017-05-16 14:30:38 +0200617 uppercase = (keysym >= XK_A) && (keysym <= XK_Z);
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100618 shift = pressedKeys.find(XK_Shift_L) != pressedKeys.end() ||
619 pressedKeys.find(XK_Shift_R) != pressedKeys.end();
620 lock = server->ledState & ledCapsLock;
Pierre Ossmanbb305ca2016-12-11 12:41:26 +0100621
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100622 if (lock == (uppercase == shift)) {
623 vlog.debug("Inserting fake CapsLock to get in sync with client");
Pierre Ossman5ae28212017-05-16 14:30:38 +0200624 server->desktop->keyEvent(XK_Caps_Lock, 0, true);
625 server->desktop->keyEvent(XK_Caps_Lock, 0, false);
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100626 }
627 }
628
629 // NumLock synchronisation heuristic
630 // (this is more cautious because of the differences between Unix,
631 // Windows and macOS)
Pierre Ossman5ae28212017-05-16 14:30:38 +0200632 if (((keysym >= XK_KP_Home) && (keysym <= XK_KP_Delete)) ||
633 ((keysym >= XK_KP_0) && (keysym <= XK_KP_9)) ||
634 (keysym == XK_KP_Separator) || (keysym == XK_KP_Decimal)) {
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100635 bool number, shift, lock;
636
Pierre Ossman5ae28212017-05-16 14:30:38 +0200637 number = ((keysym >= XK_KP_0) && (keysym <= XK_KP_9)) ||
638 (keysym == XK_KP_Separator) || (keysym == XK_KP_Decimal);
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100639 shift = pressedKeys.find(XK_Shift_L) != pressedKeys.end() ||
640 pressedKeys.find(XK_Shift_R) != pressedKeys.end();
641 lock = server->ledState & ledNumLock;
642
643 if (shift) {
644 // We don't know the appropriate NumLock state for when Shift
645 // is pressed as it could be one of:
646 //
647 // a) A Unix client where Shift negates NumLock
648 //
649 // b) A Windows client where Shift only cancels NumLock
650 //
651 // c) A macOS client where Shift doesn't have any effect
652 //
653 } else if (lock == (number == shift)) {
654 vlog.debug("Inserting fake NumLock to get in sync with client");
Pierre Ossman5ae28212017-05-16 14:30:38 +0200655 server->desktop->keyEvent(XK_Num_Lock, 0, true);
656 server->desktop->keyEvent(XK_Num_Lock, 0, false);
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100657 }
Pierre Ossmanbb305ca2016-12-11 12:41:26 +0100658 }
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000659 }
660 }
661
662 // Turn ISO_Left_Tab into shifted Tab.
663 VNCSConnectionSTShiftPresser shiftPresser(server->desktop);
Pierre Ossman5ae28212017-05-16 14:30:38 +0200664 if (keysym == XK_ISO_Left_Tab) {
Pierre Ossman16e1dcb2017-05-16 14:33:43 +0200665 std::map<rdr::U32, rdr::U32>::const_iterator iter;
666 bool shifted;
667
668 shifted = false;
669 for (iter = pressedKeys.begin(); iter != pressedKeys.end(); ++iter) {
670 if ((iter->second == XK_Shift_L) ||
671 (iter->second == XK_Shift_R)) {
672 shifted = true;
673 break;
674 }
675 }
676
677 if (!shifted)
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000678 shiftPresser.press();
Pierre Ossman16e1dcb2017-05-16 14:33:43 +0200679
Pierre Ossman5ae28212017-05-16 14:30:38 +0200680 keysym = XK_Tab;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000681 }
682
Pierre Ossman16e1dcb2017-05-16 14:33:43 +0200683 // We need to be able to track keys, so generate a fake index when we
684 // aren't given a keycode
685 if (keycode == 0)
686 lookup = 0x80000000 | keysym;
687 else
688 lookup = keycode;
689
690 // We force the same keysym for an already down key for the
691 // sake of sanity
692 if (pressedKeys.find(lookup) != pressedKeys.end())
693 keysym = pressedKeys[lookup];
694
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000695 if (down) {
Pierre Ossman16e1dcb2017-05-16 14:33:43 +0200696 pressedKeys[lookup] = keysym;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000697 } else {
Pierre Ossman16e1dcb2017-05-16 14:33:43 +0200698 if (!pressedKeys.erase(lookup))
Pierre Ossman5ae28212017-05-16 14:30:38 +0200699 return;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000700 }
Pierre Ossman16e1dcb2017-05-16 14:33:43 +0200701
Pierre Ossman5ae28212017-05-16 14:30:38 +0200702 server->desktop->keyEvent(keysym, keycode, down);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000703}
704
705void VNCSConnectionST::clientCutText(const char* str, int len)
706{
707 if (!(accessRights & AccessCutText)) return;
708 if (!rfb::Server::acceptCutText) return;
709 server->desktop->clientCutText(str, len);
710}
711
712void VNCSConnectionST::framebufferUpdateRequest(const Rect& r,bool incremental)
713{
Pierre Ossmane9962f72009-04-23 12:31:42 +0000714 Rect safeRect;
715
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000716 if (!(accessRights & AccessView)) return;
717
718 SConnection::framebufferUpdateRequest(r, incremental);
719
Pierre Ossmand9a59ba2009-03-20 15:55:37 +0000720 // Check that the client isn't sending crappy requests
721 if (!r.enclosed_by(Rect(0, 0, cp.width, cp.height))) {
722 vlog.error("FramebufferUpdateRequest %dx%d at %d,%d exceeds framebuffer %dx%d",
723 r.width(), r.height(), r.tl.x, r.tl.y, cp.width, cp.height);
Pierre Ossmane9962f72009-04-23 12:31:42 +0000724 safeRect = r.intersect(Rect(0, 0, cp.width, cp.height));
725 } else {
726 safeRect = r;
Pierre Ossmand9a59ba2009-03-20 15:55:37 +0000727 }
728
Constantin Kaplinsky2ef66952008-08-29 11:33:46 +0000729 // Just update the requested region.
730 // Framebuffer update will be sent a bit later, see processMessages().
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000731 Region reqRgn(r);
Pierre Ossman1b478e52011-11-15 12:08:30 +0000732 if (!incremental || !continuousUpdates)
733 requested.assign_union(reqRgn);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000734
735 if (!incremental) {
736 // Non-incremental update - treat as if area requested has changed
737 updates.add_changed(reqRgn);
Pierre Ossman53125a72009-04-22 15:37:51 +0000738
739 // And send the screen layout to the client (which, unlike the
740 // framebuffer dimensions, the client doesn't get during init)
Pierre Ossmanc5e25602009-03-20 12:59:05 +0000741 writer()->writeExtendedDesktopSize();
Pierre Ossman53125a72009-04-22 15:37:51 +0000742
743 // We do not send a DesktopSize since it only contains the
744 // framebuffer size (which the client already should know) and
745 // because some clients don't handle extra DesktopSize events
746 // very well.
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000747 }
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000748}
749
Pierre Ossman34bb0612009-03-21 21:16:14 +0000750void VNCSConnectionST::setDesktopSize(int fb_width, int fb_height,
751 const ScreenSet& layout)
Pierre Ossmanc5e25602009-03-20 12:59:05 +0000752{
Pierre Ossman04e62db2009-03-23 16:57:07 +0000753 unsigned int result;
754
Michal Srbb318b8f2014-11-24 13:18:28 +0200755 if (!(accessRights & AccessSetDesktopSize)) return;
756 if (!rfb::Server::acceptSetDesktopSize) return;
757
Pierre Ossman04e62db2009-03-23 16:57:07 +0000758 // Don't bother the desktop with an invalid configuration
759 if (!layout.validate(fb_width, fb_height)) {
760 writer()->writeExtendedDesktopSize(reasonClient, resultInvalid,
761 fb_width, fb_height, layout);
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000762 writeFramebufferUpdate();
Pierre Ossman04e62db2009-03-23 16:57:07 +0000763 return;
764 }
765
766 // FIXME: the desktop will call back to VNCServerST and an extra set
767 // of ExtendedDesktopSize messages will be sent. This is okay
768 // protocol-wise, but unnecessary.
769 result = server->desktop->setScreenLayout(fb_width, fb_height, layout);
770
Pierre Ossman04e62db2009-03-23 16:57:07 +0000771 writer()->writeExtendedDesktopSize(reasonClient, result,
772 fb_width, fb_height, layout);
Pierre Ossman04e62db2009-03-23 16:57:07 +0000773
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000774 // Only notify other clients on success
Pierre Ossman04e62db2009-03-23 16:57:07 +0000775 if (result == resultSuccess) {
776 if (server->screenLayout != layout)
777 throw Exception("Desktop configured a different screen layout than requested");
778 server->notifyScreenLayoutChange(this);
779 }
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000780
781 // but always send back a reply to the requesting client
782 // (do this last as it might throw an exception on socket errors)
783 writeFramebufferUpdate();
Pierre Ossmanc5e25602009-03-20 12:59:05 +0000784}
785
Pierre Ossman2c764942011-11-14 15:54:30 +0000786void VNCSConnectionST::fence(rdr::U32 flags, unsigned len, const char data[])
787{
788 if (flags & fenceFlagRequest) {
789 if (flags & fenceFlagSyncNext) {
Pierre Ossmanb8b1e962012-07-20 10:47:00 +0000790 pendingSyncFence = true;
Pierre Ossman2c764942011-11-14 15:54:30 +0000791
792 fenceFlags = flags & (fenceFlagBlockBefore | fenceFlagBlockAfter | fenceFlagSyncNext);
793 fenceDataLen = len;
794 delete [] fenceData;
Michal Srbf3afa242017-03-27 19:02:15 +0300795 fenceData = NULL;
Pierre Ossman2c764942011-11-14 15:54:30 +0000796 if (len > 0) {
797 fenceData = new char[len];
798 memcpy(fenceData, data, len);
799 }
800
801 return;
802 }
803
804 // We handle everything synchronously so we trivially honor these modes
805 flags = flags & (fenceFlagBlockBefore | fenceFlagBlockAfter);
806
807 writer()->writeFence(flags, len, data);
808 return;
809 }
810
Pierre Ossman1b478e52011-11-15 12:08:30 +0000811 struct RTTInfo rttInfo;
812
Pierre Ossman2c764942011-11-14 15:54:30 +0000813 switch (len) {
814 case 0:
815 // Initial dummy fence;
816 break;
Pierre Ossman1b478e52011-11-15 12:08:30 +0000817 case sizeof(struct RTTInfo):
818 memcpy(&rttInfo, data, sizeof(struct RTTInfo));
819 handleRTTPong(rttInfo);
820 break;
Pierre Ossman2c764942011-11-14 15:54:30 +0000821 default:
822 vlog.error("Fence response of unexpected size received");
823 }
824}
825
Pierre Ossman1b478e52011-11-15 12:08:30 +0000826void VNCSConnectionST::enableContinuousUpdates(bool enable,
827 int x, int y, int w, int h)
828{
829 Rect rect;
830
831 if (!cp.supportsFence || !cp.supportsContinuousUpdates)
832 throw Exception("Client tried to enable continuous updates when not allowed");
833
834 continuousUpdates = enable;
835
836 rect.setXYWH(x, y, w, h);
837 cuRegion.reset(rect);
838
839 if (enable) {
840 requested.clear();
841 writeFramebufferUpdate();
842 } else {
843 writer()->writeEndOfContinuousUpdates();
844 }
845}
846
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000847// supportsLocalCursor() is called whenever the status of
848// cp.supportsLocalCursor has changed. If the client does now support local
849// cursor, we make sure that the old server-side rendered cursor is cleaned up
850// and the cursor is sent to the client.
851
852void VNCSConnectionST::supportsLocalCursor()
853{
Pierre Ossman324043e2017-08-16 16:26:11 +0200854 if (cp.supportsLocalCursorWithAlpha ||
855 cp.supportsLocalCursor || cp.supportsLocalXCursor) {
Pierre Ossman671d2ef2015-06-09 16:09:06 +0200856 if (!damagedCursorRegion.is_empty())
Pierre Ossman5c9e1e52011-11-08 10:32:05 +0000857 removeRenderedCursor = true;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000858 setCursor();
859 }
860}
861
Pierre Ossman2c764942011-11-14 15:54:30 +0000862void VNCSConnectionST::supportsFence()
863{
864 writer()->writeFence(fenceFlagRequest, 0, NULL);
865}
866
Pierre Ossman1b478e52011-11-15 12:08:30 +0000867void VNCSConnectionST::supportsContinuousUpdates()
868{
869 // We refuse to use continuous updates if we cannot monitor the buffer
870 // usage using fences.
871 if (!cp.supportsFence)
872 return;
873
874 writer()->writeEndOfContinuousUpdates();
875}
876
Pierre Ossmanb45a84f2016-12-12 16:59:15 +0100877void VNCSConnectionST::supportsLEDState()
878{
879 writer()->writeLEDState();
880}
881
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000882
Pierre Ossman1bb8b6c2011-10-25 15:20:05 +0000883bool VNCSConnectionST::handleTimeout(Timer* t)
884{
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000885 try {
Pierre Ossmana40ab202016-04-29 15:35:56 +0200886 if (t == &congestionTimer)
Pierre Ossman1b478e52011-11-15 12:08:30 +0000887 updateCongestion();
Pierre Ossmanf8e3b342015-01-26 14:37:04 +0100888 else if (t == &queryConnectTimer) {
889 if (state() == RFBSTATE_QUERYING)
890 approveConnection(false, "The attempt to prompt the user to accept the connection failed");
891 }
Pierre Ossmana3ac01e2011-11-07 21:13:54 +0000892 } catch (rdr::Exception& e) {
893 close(e.str());
894 }
Pierre Ossman1bb8b6c2011-10-25 15:20:05 +0000895
896 return false;
897}
898
899
Pierre Ossman1b478e52011-11-15 12:08:30 +0000900void VNCSConnectionST::writeRTTPing()
901{
902 struct RTTInfo rttInfo;
903
904 if (!cp.supportsFence)
905 return;
906
907 memset(&rttInfo, 0, sizeof(struct RTTInfo));
908
909 gettimeofday(&rttInfo.tv, NULL);
910 rttInfo.offset = sock->outStream().length();
911 rttInfo.inFlight = rttInfo.offset - ackedOffset;
912
913 // We need to make sure any old update are already processed by the
914 // time we get the response back. This allows us to reliably throttle
915 // back on client overload, as well as network overload.
916 writer()->writeFence(fenceFlagRequest | fenceFlagBlockBefore,
917 sizeof(struct RTTInfo), (const char*)&rttInfo);
918
919 pingCounter++;
920
921 sentOffset = rttInfo.offset;
922
923 // Let some data flow before we adjust the settings
924 if (!congestionTimer.isStarted())
925 congestionTimer.start(__rfbmin(baseRTT * 2, 100));
926}
927
928void VNCSConnectionST::handleRTTPong(const struct RTTInfo &rttInfo)
929{
930 unsigned rtt, delay;
Pierre Ossman1b478e52011-11-15 12:08:30 +0000931
932 pingCounter--;
933
934 rtt = msSince(&rttInfo.tv);
935 if (rtt < 1)
936 rtt = 1;
937
938 ackedOffset = rttInfo.offset;
939
940 // Try to estimate wire latency by tracking lowest seen latency
941 if (rtt < baseRTT)
942 baseRTT = rtt;
943
944 if (rttInfo.inFlight > congWindow) {
945 seenCongestion = true;
946
947 // Estimate added delay because of overtaxed buffers
948 delay = (rttInfo.inFlight - congWindow) * baseRTT / congWindow;
949
950 if (delay < rtt)
951 rtt -= delay;
952 else
953 rtt = 1;
954
955 // If we underestimate the congestion window, then we'll get a latency
956 // that's less than the wire latency, which will confuse other portions
957 // of the code.
958 if (rtt < baseRTT)
959 rtt = baseRTT;
960 }
961
962 // We only keep track of the minimum latency seen (for a given interval)
klemens0536d092017-01-28 20:56:56 +0100963 // on the basis that we want to avoid continuous buffer issue, but don't
Pierre Ossman1b478e52011-11-15 12:08:30 +0000964 // mind (or even approve of) bursts.
965 if (rtt < minRTT)
966 minRTT = rtt;
967}
968
Pierre Ossman1bb8b6c2011-10-25 15:20:05 +0000969bool VNCSConnectionST::isCongested()
970{
Pierre Ossman1b478e52011-11-15 12:08:30 +0000971 int offset;
972
973 // Stuff still waiting in the send buffer?
Pierre Ossman352d0622016-04-29 15:50:54 +0200974 sock->outStream().flush();
Pierre Ossman1bb8b6c2011-10-25 15:20:05 +0000975 if (sock->outStream().bufferUsage() > 0)
976 return true;
977
Pierre Ossman1b478e52011-11-15 12:08:30 +0000978 if (!cp.supportsFence)
979 return false;
980
981 // Idle for too long? (and no data on the wire)
982 //
983 // FIXME: This should really just be one baseRTT, but we're getting
984 // problems with triggering the idle timeout on each update.
985 // Maybe we need to use a moving average for the wire latency
986 // instead of baseRTT.
987 if ((sentOffset == ackedOffset) &&
988 (sock->outStream().getIdleTime() > 2 * baseRTT)) {
989
990#ifdef CONGESTION_DEBUG
991 if (congWindow > INITIAL_WINDOW)
992 fprintf(stderr, "Reverting to initial window (%d KiB) after %d ms\n",
993 INITIAL_WINDOW / 1024, sock->outStream().getIdleTime());
994#endif
995
996 // Close congestion window and allow a transfer
997 // FIXME: Reset baseRTT like Linux Vegas?
998 congWindow = __rfbmin(INITIAL_WINDOW, congWindow);
999
1000 return false;
1001 }
1002
1003 offset = sock->outStream().length();
1004
1005 // FIXME: Should we compensate for non-update data?
1006 // (i.e. use sentOffset instead of offset)
1007 if ((offset - ackedOffset) < congWindow)
1008 return false;
1009
1010 // If we just have one outstanding "ping", that means the client has
1011 // started receiving our update. In order to not regress compared to
1012 // before we had congestion avoidance, we allow another update here.
1013 // This could further clog up the tubes, but congestion control isn't
1014 // really working properly right now anyway as the wire would otherwise
1015 // be idle for at least RTT/2.
1016 if (pingCounter == 1)
1017 return false;
1018
1019 return true;
1020}
1021
1022
1023void VNCSConnectionST::updateCongestion()
1024{
1025 unsigned diff;
1026
1027 if (!seenCongestion)
1028 return;
1029
1030 diff = minRTT - baseRTT;
1031
1032 if (diff > __rfbmin(100, baseRTT)) {
1033 // Way too fast
1034 congWindow = congWindow * baseRTT / minRTT;
1035 } else if (diff > __rfbmin(50, baseRTT/2)) {
1036 // Slightly too fast
1037 congWindow -= 4096;
1038 } else if (diff < 5) {
1039 // Way too slow
1040 congWindow += 8192;
1041 } else if (diff < 25) {
1042 // Too slow
1043 congWindow += 4096;
1044 }
1045
1046 if (congWindow < MINIMUM_WINDOW)
1047 congWindow = MINIMUM_WINDOW;
1048 if (congWindow > MAXIMUM_WINDOW)
1049 congWindow = MAXIMUM_WINDOW;
1050
1051#ifdef CONGESTION_DEBUG
1052 fprintf(stderr, "RTT: %d ms (%d ms), Window: %d KiB, Bandwidth: %g Mbps\n",
1053 minRTT, baseRTT, congWindow / 1024,
1054 congWindow * 8.0 / baseRTT / 1000.0);
1055
1056#ifdef TCP_INFO
1057 struct tcp_info tcp_info;
1058 socklen_t tcp_info_length;
1059
1060 tcp_info_length = sizeof(tcp_info);
1061 if (getsockopt(sock->getFd(), SOL_TCP, TCP_INFO,
1062 (void *)&tcp_info, &tcp_info_length) == 0) {
1063 fprintf(stderr, "Socket: RTT: %d ms (+/- %d ms) Window %d KiB\n",
1064 tcp_info.tcpi_rtt / 1000, tcp_info.tcpi_rttvar / 1000,
1065 tcp_info.tcpi_snd_mss * tcp_info.tcpi_snd_cwnd / 1024);
1066 }
1067#endif
1068
1069#endif
1070
1071 minRTT = -1;
1072 seenCongestion = false;
Pierre Ossman1bb8b6c2011-10-25 15:20:05 +00001073}
1074
1075
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001076void VNCSConnectionST::writeFramebufferUpdate()
1077{
Pierre Ossman2c764942011-11-14 15:54:30 +00001078 // We're in the middle of processing a command that's supposed to be
1079 // synchronised. Allowing an update to slip out right now might violate
1080 // that synchronisation.
1081 if (syncFence)
1082 return;
1083
Pierre Ossmana3ac01e2011-11-07 21:13:54 +00001084 // We try to aggregate responses, so don't send out anything whilst we
1085 // still have incoming messages. processMessages() will give us another
1086 // chance to run once things are idle.
1087 if (inProcessMessages)
1088 return;
1089
Pierre Ossman1b478e52011-11-15 12:08:30 +00001090 if (state() != RFBSTATE_NORMAL)
1091 return;
1092 if (requested.is_empty() && !continuousUpdates)
Pierre Ossmane9962f72009-04-23 12:31:42 +00001093 return;
Pierre Ossmand9a59ba2009-03-20 15:55:37 +00001094
Pierre Ossman1bb8b6c2011-10-25 15:20:05 +00001095 // Check that we actually have some space on the link and retry in a
1096 // bit if things are congested.
Pierre Ossmana40ab202016-04-29 15:35:56 +02001097 if (isCongested())
Pierre Ossman1bb8b6c2011-10-25 15:20:05 +00001098 return;
Pierre Ossman1bb8b6c2011-10-25 15:20:05 +00001099
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001100 // Updates often consists of many small writes, and in continuous
1101 // mode, we will also have small fence messages around the update. We
1102 // need to aggregate these in order to not clog up TCP's congestion
1103 // window.
Pierre Ossman36dadf82011-11-15 12:11:32 +00001104 network::TcpSocket::cork(sock->getFd(), true);
1105
Pierre Ossmane9962f72009-04-23 12:31:42 +00001106 // First take care of any updates that cannot contain framebuffer data
1107 // changes.
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001108 writeNoDataUpdate();
1109
1110 // Then real data (if possible)
1111 writeDataUpdate();
1112
1113 network::TcpSocket::cork(sock->getFd(), false);
1114}
1115
1116void VNCSConnectionST::writeNoDataUpdate()
1117{
1118 if (!writer()->needNoDataUpdate())
1119 return;
1120
1121 writer()->writeNoDataUpdate();
1122
1123 // Make sure no data update is sent until next request
1124 requested.clear();
1125}
1126
1127void VNCSConnectionST::writeDataUpdate()
1128{
1129 Region req;
1130 UpdateInfo ui;
1131 bool needNewUpdateInfo;
Pierre Ossman24684e52016-12-05 16:58:19 +01001132 const RenderedCursor *cursor;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001133
Constantin Kaplinsky604d7812007-08-31 15:50:37 +00001134 updates.enable_copyrect(cp.useCopyRect);
1135
Pierre Ossman6e49e952016-10-07 15:59:38 +02001136 // See if we are allowed to send anything right now (the framebuffer
1137 // might have changed in ways we haven't yet been informed of).
Pierre Ossmanbbf955e2011-11-08 12:44:10 +00001138 if (!server->checkUpdate())
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001139 return;
Constantin Kaplinskya09dc142008-12-18 12:08:15 +00001140
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001141 // See what the client has requested (if anything)
Pierre Ossman1b478e52011-11-15 12:08:30 +00001142 if (continuousUpdates)
1143 req = cuRegion.union_(requested);
1144 else
1145 req = requested;
1146
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001147 if (req.is_empty())
1148 return;
1149
1150 // Get the lists of updates. Prior to exporting the data to the `ui' object,
1151 // getUpdateInfo() will normalize the `updates' object such way that its
1152 // `changed' and `copied' regions would not intersect.
Pierre Ossman1b478e52011-11-15 12:08:30 +00001153 updates.getUpdateInfo(&ui, req);
1154 needNewUpdateInfo = false;
Constantin Kaplinsky604d7812007-08-31 15:50:37 +00001155
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001156 // If the previous position of the rendered cursor overlaps the source of the
1157 // copy, then when the copy happens the corresponding rectangle in the
1158 // destination will be wrong, so add it to the changed region.
1159
Pierre Ossman671d2ef2015-06-09 16:09:06 +02001160 if (!ui.copied.is_empty() && !damagedCursorRegion.is_empty()) {
1161 Region bogusCopiedCursor;
1162
1163 bogusCopiedCursor.copyFrom(damagedCursorRegion);
1164 bogusCopiedCursor.translate(ui.copy_delta);
1165 bogusCopiedCursor.assign_intersect(server->pb->getRect());
Constantin Kaplinsky45517c82007-08-31 15:56:33 +00001166 if (!ui.copied.intersect(bogusCopiedCursor).is_empty()) {
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001167 updates.add_changed(bogusCopiedCursor);
Constantin Kaplinsky604d7812007-08-31 15:50:37 +00001168 needNewUpdateInfo = true;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001169 }
1170 }
1171
Pierre Ossman671d2ef2015-06-09 16:09:06 +02001172 // If we need to remove the old rendered cursor, just add the region to
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001173 // the changed region.
1174
1175 if (removeRenderedCursor) {
Pierre Ossman671d2ef2015-06-09 16:09:06 +02001176 updates.add_changed(damagedCursorRegion);
Constantin Kaplinsky604d7812007-08-31 15:50:37 +00001177 needNewUpdateInfo = true;
Pierre Ossman671d2ef2015-06-09 16:09:06 +02001178 damagedCursorRegion.clear();
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001179 removeRenderedCursor = false;
1180 }
1181
1182 // Return if there is nothing to send the client.
1183
Pierre Ossman671d2ef2015-06-09 16:09:06 +02001184 if (updates.is_empty() && !writer()->needFakeUpdate() && !updateRenderedCursor)
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001185 return;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001186
Constantin Kaplinsky604d7812007-08-31 15:50:37 +00001187 // The `updates' object could change, make sure we have valid update info.
1188
1189 if (needNewUpdateInfo)
Pierre Ossman1b478e52011-11-15 12:08:30 +00001190 updates.getUpdateInfo(&ui, req);
Constantin Kaplinsky604d7812007-08-31 15:50:37 +00001191
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001192 // If the client needs a server-side rendered cursor, work out the cursor
1193 // rectangle. If it's empty then don't bother drawing it, but if it overlaps
1194 // with the update region, we need to draw the rendered cursor regardless of
1195 // whether it has changed.
1196
Pierre Ossman24684e52016-12-05 16:58:19 +01001197 cursor = NULL;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001198 if (needRenderedCursor()) {
Pierre Ossman671d2ef2015-06-09 16:09:06 +02001199 Rect renderedCursorRect;
1200
Pierre Ossman24684e52016-12-05 16:58:19 +01001201 cursor = server->getRenderedCursor();
1202
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001203 renderedCursorRect
Pierre Ossman24684e52016-12-05 16:58:19 +01001204 = cursor->getEffectiveRect().intersect(req.get_bounding_rect());
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001205
1206 if (renderedCursorRect.is_empty()) {
Pierre Ossman24684e52016-12-05 16:58:19 +01001207 cursor = NULL;
1208 } else if (!updateRenderedCursor &&
1209 ui.changed.union_(ui.copied)
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001210 .intersect(renderedCursorRect).is_empty()) {
Pierre Ossman24684e52016-12-05 16:58:19 +01001211 cursor = NULL;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001212 }
1213
Pierre Ossman5c037202016-12-05 17:00:35 +01001214 if (cursor) {
1215 updates.subtract(renderedCursorRect);
1216 updates.getUpdateInfo(&ui, req);
1217 }
Pierre Ossman671d2ef2015-06-09 16:09:06 +02001218
1219 damagedCursorRegion.assign_union(renderedCursorRect);
1220 updateRenderedCursor = false;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001221 }
1222
Pierre Ossman24684e52016-12-05 16:58:19 +01001223 if (ui.is_empty() && !writer()->needFakeUpdate() && !cursor)
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001224 return;
Pierre Ossmanfdba3fe2014-01-31 13:12:18 +01001225
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001226 writeRTTPing();
Pierre Ossman1b478e52011-11-15 12:08:30 +00001227
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001228 encodeManager.writeUpdate(ui, server->getPixelBuffer(), cursor);
Pierre Ossman1b478e52011-11-15 12:08:30 +00001229
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001230 writeRTTPing();
Pierre Ossmanc0397262014-03-14 15:59:46 +01001231
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001232 // The request might be for just part of the screen, so we cannot
1233 // just clear the entire update tracker.
1234 updates.subtract(req);
Pierre Ossmanc0397262014-03-14 15:59:46 +01001235
Pierre Ossmancef3cf72016-11-25 10:06:34 +01001236 requested.clear();
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001237}
1238
1239
Pierre Ossmana3ac01e2011-11-07 21:13:54 +00001240void VNCSConnectionST::screenLayoutChange(rdr::U16 reason)
1241{
1242 if (!authenticated())
1243 return;
1244
1245 cp.screenLayout = server->screenLayout;
1246
1247 if (state() != RFBSTATE_NORMAL)
1248 return;
1249
1250 writer()->writeExtendedDesktopSize(reason, 0, cp.width, cp.height,
1251 cp.screenLayout);
1252 writeFramebufferUpdate();
1253}
1254
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001255
1256// setCursor() is called whenever the cursor has changed shape or pixel format.
1257// If the client supports local cursor then it will arrange for the cursor to
1258// be sent to the client.
1259
1260void VNCSConnectionST::setCursor()
1261{
Pierre Ossmana3ac01e2011-11-07 21:13:54 +00001262 if (state() != RFBSTATE_NORMAL)
1263 return;
Pierre Ossmana3ac01e2011-11-07 21:13:54 +00001264
Pierre Ossman71ca8d52017-09-15 11:03:12 +02001265 // We need to blank out the client's cursor or there will be two
1266 if (needRenderedCursor())
1267 cp.setCursor(emptyCursor);
1268 else
1269 cp.setCursor(*server->cursor);
Pierre Ossman126e5642014-02-13 14:40:25 +01001270
Pierre Ossman8053c8e2017-02-21 12:59:04 +01001271 if (!writer()->writeSetCursorWithAlpha()) {
1272 if (!writer()->writeSetCursor()) {
1273 if (!writer()->writeSetXCursor()) {
1274 // No client support
1275 return;
1276 }
Pierre Ossman126e5642014-02-13 14:40:25 +01001277 }
1278 }
1279
Pierre Ossmana3ac01e2011-11-07 21:13:54 +00001280 writeFramebufferUpdate();
1281}
1282
1283void VNCSConnectionST::setDesktopName(const char *name)
1284{
1285 cp.setName(name);
1286
1287 if (state() != RFBSTATE_NORMAL)
1288 return;
1289
1290 if (!writer()->writeSetDesktopName()) {
1291 fprintf(stderr, "Client does not support desktop rename\n");
1292 return;
1293 }
1294
1295 writeFramebufferUpdate();
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001296}
1297
Pierre Ossmanb45a84f2016-12-12 16:59:15 +01001298void VNCSConnectionST::setLEDState(unsigned int ledstate)
1299{
1300 if (state() != RFBSTATE_NORMAL)
1301 return;
1302
1303 cp.setLEDState(ledstate);
1304
1305 if (!writer()->writeLEDState()) {
1306 // No client support
1307 return;
1308 }
1309
1310 writeFramebufferUpdate();
1311}
1312
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001313void VNCSConnectionST::setSocketTimeouts()
1314{
1315 int timeoutms = rfb::Server::clientWaitTimeMillis;
1316 soonestTimeout(&timeoutms, secsToMillis(rfb::Server::idleTimeout));
1317 if (timeoutms == 0)
1318 timeoutms = -1;
1319 sock->inStream().setTimeout(timeoutms);
1320 sock->outStream().setTimeout(timeoutms);
1321}
1322
1323char* VNCSConnectionST::getStartTime()
1324{
1325 char* result = ctime(&startTime);
1326 result[24] = '\0';
1327 return result;
1328}
1329
1330void VNCSConnectionST::setStatus(int status)
1331{
1332 switch (status) {
1333 case 0:
1334 accessRights = accessRights | AccessPtrEvents | AccessKeyEvents | AccessView;
1335 break;
1336 case 1:
Pierre Ossman7728be22015-03-03 16:39:37 +01001337 accessRights = (accessRights & ~(AccessPtrEvents | AccessKeyEvents)) | AccessView;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001338 break;
1339 case 2:
Adam Tkac8e985062011-02-07 11:33:57 +00001340 accessRights = accessRights & ~(AccessPtrEvents | AccessKeyEvents | AccessView);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001341 break;
1342 }
1343 framebufferUpdateRequest(server->pb->getRect(), false);
1344}
1345int VNCSConnectionST::getStatus()
1346{
1347 if ((accessRights & (AccessPtrEvents | AccessKeyEvents | AccessView)) == 0x0007)
1348 return 0;
1349 if ((accessRights & (AccessPtrEvents | AccessKeyEvents | AccessView)) == 0x0001)
1350 return 1;
1351 if ((accessRights & (AccessPtrEvents | AccessKeyEvents | AccessView)) == 0x0000)
1352 return 2;
1353 return 4;
1354}
1355