blob: db3ab08c8fbf227d04e223d42c4f0510e7ffbf30 [file] [log] [blame]
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved.
Pierre Ossman615d16b2019-05-03 10:53:06 +02002 * Copyright 2011-2019 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//
20// SConnection - class on the server side representing a connection to a
21// client. A derived class should override methods appropriately.
22//
23
24#ifndef __RFB_SCONNECTION_H__
25#define __RFB_SCONNECTION_H__
26
27#include <rdr/InStream.h>
28#include <rdr/OutStream.h>
29#include <rfb/SMsgHandler.h>
Adam Tkacbfd66c12010-10-01 08:33:29 +000030#include <rfb/SecurityServer.h>
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000031
32namespace rfb {
33
34 class SMsgReader;
35 class SMsgWriter;
36 class SSecurity;
37
38 class SConnection : public SMsgHandler {
39 public:
40
Pierre Ossman7069bdd2015-02-06 14:41:58 +010041 SConnection();
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000042 virtual ~SConnection();
43
44 // Methods to initialise the connection
45
46 // setStreams() sets the streams to be used for the connection. These must
47 // be set before initialiseProtocol() and processMsg() are called. The
48 // SSecurity object may call setStreams() again to provide alternative
49 // streams over which the RFB protocol is sent (i.e. encrypting/decrypting
50 // streams). Ownership of the streams remains with the caller
51 // (i.e. SConnection will not delete them).
52 void setStreams(rdr::InStream* is, rdr::OutStream* os);
53
54 // initialiseProtocol() should be called once the streams and security
55 // types are set. Subsequently, processMsg() should be called whenever
56 // there is data to read on the InStream.
57 void initialiseProtocol();
58
59 // processMsg() should be called whenever there is data to read on the
60 // InStream. You must have called initialiseProtocol() first.
61 void processMsg();
62
63 // approveConnection() is called to either accept or reject the connection.
64 // If accept is false, the reason string gives the reason for the
65 // rejection. It can either be called directly from queryConnection() or
66 // later, after queryConnection() has returned. It can only be called when
67 // in state RFBSTATE_QUERYING. On rejection, an AuthFailureException is
68 // thrown, so this must be handled appropriately by the caller.
69 void approveConnection(bool accept, const char* reason=0);
70
71
Pierre Ossman7d64b332018-10-08 15:59:02 +020072 // Methods to terminate the connection
73
74 // close() shuts down the connection to the client and awaits
75 // cleanup of the SConnection object by the server
76 virtual void close(const char* reason);
77
78
Pierre Ossman48700812014-09-17 17:11:56 +020079 // Overridden from SMsgHandler
80
Pierre Ossmanf38e2432015-02-11 13:47:58 +010081 virtual void setEncodings(int nEncodings, const rdr::S32* encodings);
Pierre Ossman48700812014-09-17 17:11:56 +020082
Pierre Ossman615d16b2019-05-03 10:53:06 +020083 virtual void clientCutText(const char* str);
84
Pierre Ossman0ff26552016-02-05 10:26:56 +010085 virtual void handleClipboardRequest(rdr::U32 flags);
86 virtual void handleClipboardPeek(rdr::U32 flags);
87 virtual void handleClipboardNotify(rdr::U32 flags);
88 virtual void handleClipboardProvide(rdr::U32 flags,
89 const size_t* lengths,
90 const rdr::U8* const* data);
91
Pierre Ossman5ae28212017-05-16 14:30:38 +020092 virtual void supportsQEMUKeyEvent();
Pierre Ossman48700812014-09-17 17:11:56 +020093
Pierre Ossman615d16b2019-05-03 10:53:06 +020094
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000095 // Methods to be overridden in a derived class
96
97 // versionReceived() indicates that the version number has just been read
98 // from the client. The version will already have been "cooked"
99 // to deal with unknown/bogus viewer protocol numbers.
100 virtual void versionReceived();
101
102 // authSuccess() is called when authentication has succeeded.
103 virtual void authSuccess();
104
Pierre Ossman88a94ed2019-04-01 14:22:01 +0200105 // authFailure() is called when authentication has failed. The default
106 // implementation will inform the client and throw a AuthFailureException.
107 virtual void authFailure(const char* reason);
108
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000109 // queryConnection() is called when authentication has succeeded, but
110 // before informing the client. It can be overridden to query a local user
111 // to accept the incoming connection, for example. The userName argument
112 // is the name of the user making the connection, or null (note that the
113 // storage for userName is owned by the caller). The connection must be
114 // accepted or rejected by calling approveConnection(), either directly
115 // from queryConnection() or some time later.
116 virtual void queryConnection(const char* userName);
117
118 // clientInit() is called when the ClientInit message is received. The
119 // derived class must call on to SConnection::clientInit().
120 virtual void clientInit(bool shared);
121
122 // setPixelFormat() is called when a SetPixelFormat message is received.
123 // The derived class must call on to SConnection::setPixelFormat().
124 virtual void setPixelFormat(const PixelFormat& pf);
125
126 // framebufferUpdateRequest() is called when a FramebufferUpdateRequest
127 // message is received. The derived class must call on to
128 // SConnection::framebufferUpdateRequest().
129 virtual void framebufferUpdateRequest(const Rect& r, bool incremental);
130
Pierre Ossmanc754cce2011-11-14 15:44:11 +0000131 // fence() is called when we get a fence request or response. By default
132 // it responds directly to requests (stating it doesn't support any
133 // synchronisation) and drops responses. Override to implement more proper
134 // support.
135 virtual void fence(rdr::U32 flags, unsigned len, const char data[]);
136
Pierre Ossmanc898d9a2011-11-14 16:22:23 +0000137 // enableContinuousUpdates() is called when the client wants to enable
138 // or disable continuous updates, or change the active area.
139 virtual void enableContinuousUpdates(bool enable,
140 int x, int y, int w, int h);
141
Pierre Ossman615d16b2019-05-03 10:53:06 +0200142 // handleClipboardRequest() is called whenever the client requests
143 // the server to send over its clipboard data. It will only be
144 // called after the server has first announced a clipboard change
145 // via announceClipboard().
146 virtual void handleClipboardRequest();
147
148 // handleClipboardAnnounce() is called to indicate a change in the
149 // clipboard on the client. Call requestClipboard() to access the
150 // actual data.
151 virtual void handleClipboardAnnounce(bool available);
152
153 // handleClipboardData() is called when the client has sent over
154 // the clipboard data as a result of a previous call to
155 // requestClipboard(). Note that this function might never be
156 // called if the clipboard data was no longer available when the
157 // client received the request.
158 virtual void handleClipboardData(const char* data);
159
160
Pierre Ossman7d64b332018-10-08 15:59:02 +0200161 // Other methods
162
Pierre Ossman615d16b2019-05-03 10:53:06 +0200163 // requestClipboard() will result in a request to the client to
164 // transfer its clipboard data. A call to handleClipboardData()
165 // will be made once the data is available.
166 virtual void requestClipboard();
167
168 // announceClipboard() informs the client of changes to the
169 // clipboard on the server. The client may later request the
170 // clipboard data via handleClipboardRequest().
171 virtual void announceClipboard(bool available);
172
173 // sendClipboardData() transfers the clipboard data to the client
174 // and should be called whenever the client has requested the
175 // clipboard via handleClipboardRequest().
176 virtual void sendClipboardData(const char* data);
177
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000178 // setAccessRights() allows a security package to limit the access rights
Pierre Ossman7d64b332018-10-08 15:59:02 +0200179 // of a SConnection to the server. How the access rights are treated
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000180 // is up to the derived class.
181
182 typedef rdr::U16 AccessRights;
Michal Srbb318b8f2014-11-24 13:18:28 +0200183 static const AccessRights AccessView; // View display contents
184 static const AccessRights AccessKeyEvents; // Send key events
185 static const AccessRights AccessPtrEvents; // Send pointer events
186 static const AccessRights AccessCutText; // Send/receive clipboard events
187 static const AccessRights AccessSetDesktopSize; // Change desktop size
Pierre Ossmane7be49b2014-12-02 14:33:17 +0100188 static const AccessRights AccessNonShared; // Exclusive access to the server
Michal Srbb318b8f2014-11-24 13:18:28 +0200189 static const AccessRights AccessDefault; // The default rights, INCLUDING FUTURE ONES
190 static const AccessRights AccessNoQuery; // Connect without local user accepting
191 static const AccessRights AccessFull; // All of the available AND FUTURE rights
Pierre Ossman7d64b332018-10-08 15:59:02 +0200192 virtual void setAccessRights(AccessRights ar);
193 virtual bool accessCheck(AccessRights ar) const;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000194
195 // authenticated() returns true if the client has authenticated
196 // successfully.
197 bool authenticated() { return (state_ == RFBSTATE_INITIALISATION ||
198 state_ == RFBSTATE_NORMAL); }
199
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000200 SMsgReader* reader() { return reader_; }
201 SMsgWriter* writer() { return writer_; }
202
203 rdr::InStream* getInStream() { return is; }
204 rdr::OutStream* getOutStream() { return os; }
205
206 enum stateEnum {
207 RFBSTATE_UNINITIALISED,
208 RFBSTATE_PROTOCOL_VERSION,
209 RFBSTATE_SECURITY_TYPE,
210 RFBSTATE_SECURITY,
Pierre Ossman88a94ed2019-04-01 14:22:01 +0200211 RFBSTATE_SECURITY_FAILURE,
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000212 RFBSTATE_QUERYING,
213 RFBSTATE_INITIALISATION,
214 RFBSTATE_NORMAL,
215 RFBSTATE_CLOSING,
216 RFBSTATE_INVALID
217 };
218
219 stateEnum state() { return state_; }
220
Pierre Ossman48700812014-09-17 17:11:56 +0200221 rdr::S32 getPreferredEncoding() { return preferredEncoding; }
222
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000223 protected:
Pierre Ossman2ebed0d2018-10-11 07:54:12 +0200224 // throwConnFailedException() prints a message to the log, sends a conn
225 // failed message to the client (if possible) and throws a
226 // ConnFailedException.
227 void throwConnFailedException(const char* format, ...) __printf_attr(2, 3);
228
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000229 void setState(stateEnum s) { state_ = s; }
Pierre Ossmanb7acf862015-02-06 14:44:32 +0100230
Pierre Ossman0144c532015-02-04 14:10:43 +0100231 void setReader(SMsgReader *r) { reader_ = r; }
232 void setWriter(SMsgWriter *w) { writer_ = w; }
233
Pierre Ossmanb7acf862015-02-06 14:44:32 +0100234 private:
Pierre Ossmanb6b4dc62014-01-20 15:05:21 +0100235 void writeFakeColourMap(void);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000236
237 bool readyForSetColourMapEntries;
238
239 void processVersionMsg();
240 void processSecurityTypeMsg();
Constantin Kaplinsky5fa9d222006-09-06 10:32:06 +0000241 void processSecurityType(int secType);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000242 void processSecurityMsg();
243 void processInitMsg();
244
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000245 int defaultMajorVersion, defaultMinorVersion;
246 rdr::InStream* is;
247 rdr::OutStream* os;
248 SMsgReader* reader_;
249 SMsgWriter* writer_;
Michal Srbdccb5f72017-03-27 13:55:46 +0300250 SecurityServer security;
Adam Tkaca6578bf2010-04-23 14:07:41 +0000251 SSecurity* ssecurity;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000252 stateEnum state_;
Pierre Ossman48700812014-09-17 17:11:56 +0200253 rdr::S32 preferredEncoding;
Pierre Ossman7d64b332018-10-08 15:59:02 +0200254 AccessRights accessRights;
Pierre Ossman615d16b2019-05-03 10:53:06 +0200255
256 char* clientClipboard;
Pierre Ossman0ff26552016-02-05 10:26:56 +0100257 bool hasLocalClipboard;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000258 };
259}
260#endif