0
|
1 // Copyright (C) 2010, 2011, 2012, 2013 GlavSoft LLC.
|
|
2 // All rights reserved.
|
|
3 //
|
|
4 //-------------------------------------------------------------------------
|
|
5 // This file is part of the TightVNC software. Please visit our Web site:
|
|
6 //
|
|
7 // http://www.tightvnc.com/
|
|
8 //
|
|
9 // This program is free software; you can redistribute it and/or modify
|
|
10 // it under the terms of the GNU General Public License as published by
|
|
11 // the Free Software Foundation; either version 2 of the License, or
|
|
12 // (at your option) any later version.
|
|
13 //
|
|
14 // This program is distributed in the hope that it will be useful,
|
|
15 // but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
16 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
17 // GNU General Public License for more details.
|
|
18 //
|
|
19 // You should have received a copy of the GNU General Public License along
|
|
20 // with this program; if not, write to the Free Software Foundation, Inc.,
|
|
21 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
22 //-------------------------------------------------------------------------
|
|
23 //
|
|
24
|
|
25 package com.glavsoft.rfb.protocol;
|
|
26
|
9
|
27 import alice.datasegment.DataSegment;
|
18
|
28 import alice.datasegment.ReceiveData;
|
9
|
29
|
0
|
30 import com.glavsoft.drawing.Renderer;
|
|
31 import com.glavsoft.exceptions.CommonException;
|
|
32 import com.glavsoft.exceptions.ProtocolException;
|
|
33 import com.glavsoft.exceptions.TransportException;
|
|
34 import com.glavsoft.rfb.ClipboardController;
|
|
35 import com.glavsoft.rfb.IRepaintController;
|
|
36 import com.glavsoft.rfb.client.FramebufferUpdateRequestMessage;
|
|
37 import com.glavsoft.rfb.client.SetPixelFormatMessage;
|
|
38 import com.glavsoft.rfb.encoding.EncodingType;
|
|
39 import com.glavsoft.rfb.encoding.PixelFormat;
|
9
|
40 import com.glavsoft.rfb.encoding.decoder.AliceVNCMessage;
|
0
|
41 import com.glavsoft.rfb.encoding.decoder.Decoder;
|
|
42 import com.glavsoft.rfb.encoding.decoder.DecodersContainer;
|
|
43 import com.glavsoft.rfb.encoding.decoder.FramebufferUpdateRectangle;
|
|
44 import com.glavsoft.rfb.encoding.decoder.RichCursorDecoder;
|
|
45 import com.glavsoft.transport.Reader;
|
|
46
|
|
47 import java.io.PrintWriter;
|
|
48 import java.io.StringWriter;
|
|
49 import java.util.logging.Logger;
|
|
50
|
|
51 public class ReceiverTask implements Runnable {
|
|
52 private static final byte FRAMEBUFFER_UPDATE = 0;
|
|
53 private static final byte SET_COLOR_MAP_ENTRIES = 1;
|
|
54 private static final byte BELL = 2;
|
|
55 private static final byte SERVER_CUT_TEXT = 3;
|
|
56
|
|
57
|
|
58 private static Logger logger = Logger.getLogger("com.glavsoft.rfb.protocol.ReceiverTask");
|
|
59 private final Reader reader;
|
|
60 private volatile boolean isRunning = false;
|
|
61 private Renderer renderer;
|
|
62 private final IRepaintController repaintController;
|
|
63 private final ClipboardController clipboardController;
|
|
64 private final DecodersContainer decoders;
|
|
65 private FramebufferUpdateRequestMessage fullscreenFbUpdateIncrementalRequest;
|
|
66 private final ProtocolContext context;
|
|
67 private PixelFormat pixelFormat;
|
|
68 private boolean needSendPixelFormat;
|
|
69
|
|
70 public ReceiverTask(Reader reader,
|
|
71 IRepaintController repaintController, ClipboardController clipboardController,
|
|
72 DecodersContainer decoders, ProtocolContext context) {
|
|
73 this.reader = reader;
|
|
74 this.repaintController = repaintController;
|
|
75 this.clipboardController = clipboardController;
|
|
76 this.context = context;
|
|
77 this.decoders = decoders;
|
|
78 renderer = repaintController.createRenderer(reader, context.getFbWidth(), context.getFbHeight(),
|
|
79 context.getPixelFormat());
|
|
80 fullscreenFbUpdateIncrementalRequest =
|
|
81 new FramebufferUpdateRequestMessage(0, 0, context.getFbWidth(), context.getFbHeight(), true);
|
|
82 }
|
|
83
|
|
84 @Override
|
|
85 public void run() {
|
|
86 isRunning = true;
|
|
87 while (isRunning) {
|
|
88 try {
|
|
89 byte messageId = reader.readByte();
|
|
90 switch (messageId) {
|
|
91 case FRAMEBUFFER_UPDATE:
|
|
92 // logger.fine("Server message: FramebufferUpdate (0)");
|
|
93 framebufferUpdateMessage();
|
|
94 break;
|
|
95 case SET_COLOR_MAP_ENTRIES:
|
|
96 logger.severe("Server message SetColorMapEntries is not implemented. Skip.");
|
|
97 setColorMapEntries();
|
|
98 break;
|
|
99 case BELL:
|
|
100 logger.fine("Server message: Bell");
|
|
101 System.out.print("\0007");
|
|
102 System.out.flush();
|
|
103 break;
|
|
104 case SERVER_CUT_TEXT:
|
|
105 logger.fine("Server message: CutText (3)");
|
|
106 serverCutText();
|
|
107 break;
|
|
108 default:
|
|
109 logger.severe("Unsupported server message. Id = " + messageId);
|
|
110 }
|
|
111 } catch (TransportException e) {
|
|
112 if (isRunning) {
|
|
113 logger.severe("Close session: " + e.getMessage());
|
|
114 context.cleanUpSession("Connection closed.");
|
|
115 }
|
|
116 stopTask();
|
|
117 } catch (ProtocolException e) {
|
|
118 logger.severe(e.getMessage());
|
|
119 if (isRunning) {
|
|
120 context.cleanUpSession(e.getMessage() + "\nConnection closed.");
|
|
121 }
|
|
122 stopTask();
|
|
123 } catch (CommonException e) {
|
|
124 logger.severe(e.getMessage());
|
|
125 if (isRunning) {
|
|
126 context.cleanUpSession("Connection closed..");
|
|
127 }
|
|
128 stopTask();
|
|
129 } catch (Throwable te) {
|
|
130 StringWriter sw = new StringWriter();
|
|
131 PrintWriter pw = new PrintWriter(sw);
|
|
132 te.printStackTrace(pw);
|
|
133 if (isRunning) {
|
|
134 context.cleanUpSession(te.getMessage() + "\n" + sw.toString());
|
|
135 }
|
|
136 stopTask();
|
|
137 }
|
|
138 }
|
|
139 }
|
|
140
|
|
141 private void setColorMapEntries() throws TransportException {
|
|
142 reader.readByte(); // padding
|
|
143 reader.readUInt16(); // first color index
|
|
144 int length = reader.readUInt16();
|
|
145 while (length-- > 0) {
|
|
146 reader.readUInt16(); // R
|
|
147 reader.readUInt16(); // G
|
|
148 reader.readUInt16(); // B
|
|
149 }
|
|
150 }
|
|
151
|
|
152 private void serverCutText() throws TransportException {
|
|
153 reader.readByte(); // padding
|
|
154 reader.readInt16(); // padding
|
|
155 int length = reader.readInt32() & Integer.MAX_VALUE;
|
|
156 clipboardController.updateSystemClipboard(reader.readBytes(length));
|
|
157 }
|
|
158
|
|
159 public void framebufferUpdateMessage() throws CommonException {
|
|
160 reader.readByte(); // padding
|
|
161 int numberOfRectangles = reader.readUInt16();
|
|
162 while (numberOfRectangles-- > 0) {
|
|
163 FramebufferUpdateRectangle rect = new FramebufferUpdateRectangle();
|
|
164 rect.fill(reader);
|
|
165
|
|
166 Decoder decoder = decoders.getDecoderByType(rect.getEncodingType());
|
|
167 logger.finest(rect.toString() + (0 == numberOfRectangles ? "\n---" : ""));
|
|
168 if (decoder != null) {
|
|
169 decoder.decode(reader, renderer, rect);
|
|
170 repaintController.repaintBitmap(rect);
|
|
171 } else if (rect.getEncodingType() == EncodingType.RICH_CURSOR) {
|
|
172 RichCursorDecoder.getInstance().decode(reader, renderer, rect);
|
|
173 repaintController.repaintCursor();
|
|
174 } else if (rect.getEncodingType() == EncodingType.CURSOR_POS) {
|
|
175 renderer.decodeCursorPosition(rect);
|
|
176 repaintController.repaintCursor();
|
|
177 } else if (rect.getEncodingType() == EncodingType.DESKTOP_SIZE) {
|
|
178 fullscreenFbUpdateIncrementalRequest =
|
|
179 new FramebufferUpdateRequestMessage(0, 0, rect.width, rect.height, true);
|
|
180 synchronized (renderer.getLock()) {
|
|
181 renderer = repaintController.createRenderer(reader, rect.width, rect.height,
|
|
182 context.getPixelFormat());
|
|
183 }
|
9
|
184 AliceVNCMessage message = new AliceVNCMessage();
|
|
185 message.setRectangle(rect);
|
|
186 message.setPixelFormat(context.getPixelFormat());
|
19
|
187
|
|
188 ReceiveData rData = new ReceiveData("dummy", false, false);
|
|
189 DataSegment.getLocal().put("pixelByteArray", rData, null);
|
|
190 rData = new ReceiveData(message, false, false);
|
18
|
191 DataSegment.getLocal().put("aliceVNCMessage", rData, null);
|
9
|
192
|
0
|
193 context.sendMessage(new FramebufferUpdateRequestMessage(0, 0, rect.width, rect.height, false));
|
|
194 // repaintController.repaintCursor();
|
|
195 } else
|
|
196 throw new CommonException("Unprocessed encoding: " + rect.toString());
|
|
197 }
|
|
198 synchronized (this) {
|
|
199 if (needSendPixelFormat) {
|
|
200 needSendPixelFormat = false;
|
|
201 context.setPixelFormat(pixelFormat);
|
|
202 context.sendMessage(new SetPixelFormatMessage(pixelFormat));
|
|
203 logger.fine("sent: "+pixelFormat);
|
|
204 context.sendRefreshMessage();
|
|
205 logger.fine("sent: nonincremental fb update");
|
|
206 } else {
|
|
207 context.sendMessage(fullscreenFbUpdateIncrementalRequest);
|
|
208 }
|
|
209 }
|
|
210 }
|
|
211
|
|
212 public synchronized void queueUpdatePixelFormat(PixelFormat pf) {
|
|
213 pixelFormat = pf;
|
|
214 needSendPixelFormat = true;
|
|
215 // context.sendMessage(new FramebufferUpdateRequestMessage(0, 0, 1, 1, false));
|
|
216 }
|
|
217
|
|
218 public void stopTask() {
|
|
219 isRunning = false;
|
|
220 }
|
|
221
|
|
222 }
|