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;
|
|
28
|
0
|
29 import com.glavsoft.drawing.Renderer;
|
|
30 import com.glavsoft.exceptions.CommonException;
|
|
31 import com.glavsoft.exceptions.ProtocolException;
|
|
32 import com.glavsoft.exceptions.TransportException;
|
|
33 import com.glavsoft.rfb.ClipboardController;
|
|
34 import com.glavsoft.rfb.IRepaintController;
|
|
35 import com.glavsoft.rfb.client.FramebufferUpdateRequestMessage;
|
|
36 import com.glavsoft.rfb.client.SetPixelFormatMessage;
|
|
37 import com.glavsoft.rfb.encoding.EncodingType;
|
|
38 import com.glavsoft.rfb.encoding.PixelFormat;
|
9
|
39 import com.glavsoft.rfb.encoding.decoder.AliceVNCMessage;
|
0
|
40 import com.glavsoft.rfb.encoding.decoder.Decoder;
|
|
41 import com.glavsoft.rfb.encoding.decoder.DecodersContainer;
|
|
42 import com.glavsoft.rfb.encoding.decoder.FramebufferUpdateRectangle;
|
|
43 import com.glavsoft.rfb.encoding.decoder.RichCursorDecoder;
|
|
44 import com.glavsoft.transport.Reader;
|
|
45
|
|
46 import java.io.PrintWriter;
|
|
47 import java.io.StringWriter;
|
|
48 import java.util.logging.Logger;
|
|
49
|
|
50 public class ReceiverTask implements Runnable {
|
|
51 private static final byte FRAMEBUFFER_UPDATE = 0;
|
|
52 private static final byte SET_COLOR_MAP_ENTRIES = 1;
|
|
53 private static final byte BELL = 2;
|
|
54 private static final byte SERVER_CUT_TEXT = 3;
|
|
55
|
|
56
|
|
57 private static Logger logger = Logger.getLogger("com.glavsoft.rfb.protocol.ReceiverTask");
|
|
58 private final Reader reader;
|
|
59 private volatile boolean isRunning = false;
|
|
60 private Renderer renderer;
|
|
61 private final IRepaintController repaintController;
|
|
62 private final ClipboardController clipboardController;
|
|
63 private final DecodersContainer decoders;
|
|
64 private FramebufferUpdateRequestMessage fullscreenFbUpdateIncrementalRequest;
|
|
65 private final ProtocolContext context;
|
|
66 private PixelFormat pixelFormat;
|
|
67 private boolean needSendPixelFormat;
|
|
68
|
|
69 public ReceiverTask(Reader reader,
|
|
70 IRepaintController repaintController, ClipboardController clipboardController,
|
|
71 DecodersContainer decoders, ProtocolContext context) {
|
|
72 this.reader = reader;
|
|
73 this.repaintController = repaintController;
|
|
74 this.clipboardController = clipboardController;
|
|
75 this.context = context;
|
|
76 this.decoders = decoders;
|
|
77 renderer = repaintController.createRenderer(reader, context.getFbWidth(), context.getFbHeight(),
|
|
78 context.getPixelFormat());
|
|
79 fullscreenFbUpdateIncrementalRequest =
|
|
80 new FramebufferUpdateRequestMessage(0, 0, context.getFbWidth(), context.getFbHeight(), true);
|
|
81 }
|
|
82
|
|
83 @Override
|
|
84 public void run() {
|
|
85 isRunning = true;
|
|
86 while (isRunning) {
|
|
87 try {
|
|
88 byte messageId = reader.readByte();
|
|
89 switch (messageId) {
|
|
90 case FRAMEBUFFER_UPDATE:
|
|
91 // logger.fine("Server message: FramebufferUpdate (0)");
|
|
92 framebufferUpdateMessage();
|
|
93 break;
|
|
94 case SET_COLOR_MAP_ENTRIES:
|
|
95 logger.severe("Server message SetColorMapEntries is not implemented. Skip.");
|
|
96 setColorMapEntries();
|
|
97 break;
|
|
98 case BELL:
|
|
99 logger.fine("Server message: Bell");
|
|
100 System.out.print("\0007");
|
|
101 System.out.flush();
|
|
102 break;
|
|
103 case SERVER_CUT_TEXT:
|
|
104 logger.fine("Server message: CutText (3)");
|
|
105 serverCutText();
|
|
106 break;
|
|
107 default:
|
|
108 logger.severe("Unsupported server message. Id = " + messageId);
|
|
109 }
|
|
110 } catch (TransportException e) {
|
|
111 if (isRunning) {
|
|
112 logger.severe("Close session: " + e.getMessage());
|
|
113 context.cleanUpSession("Connection closed.");
|
|
114 }
|
|
115 stopTask();
|
|
116 } catch (ProtocolException e) {
|
|
117 logger.severe(e.getMessage());
|
|
118 if (isRunning) {
|
|
119 context.cleanUpSession(e.getMessage() + "\nConnection closed.");
|
|
120 }
|
|
121 stopTask();
|
|
122 } catch (CommonException e) {
|
|
123 logger.severe(e.getMessage());
|
|
124 if (isRunning) {
|
|
125 context.cleanUpSession("Connection closed..");
|
|
126 }
|
|
127 stopTask();
|
|
128 } catch (Throwable te) {
|
|
129 StringWriter sw = new StringWriter();
|
|
130 PrintWriter pw = new PrintWriter(sw);
|
|
131 te.printStackTrace(pw);
|
|
132 if (isRunning) {
|
|
133 context.cleanUpSession(te.getMessage() + "\n" + sw.toString());
|
|
134 }
|
|
135 stopTask();
|
|
136 }
|
|
137 }
|
|
138 }
|
|
139
|
|
140 private void setColorMapEntries() throws TransportException {
|
|
141 reader.readByte(); // padding
|
|
142 reader.readUInt16(); // first color index
|
|
143 int length = reader.readUInt16();
|
|
144 while (length-- > 0) {
|
|
145 reader.readUInt16(); // R
|
|
146 reader.readUInt16(); // G
|
|
147 reader.readUInt16(); // B
|
|
148 }
|
|
149 }
|
|
150
|
|
151 private void serverCutText() throws TransportException {
|
|
152 reader.readByte(); // padding
|
|
153 reader.readInt16(); // padding
|
|
154 int length = reader.readInt32() & Integer.MAX_VALUE;
|
|
155 clipboardController.updateSystemClipboard(reader.readBytes(length));
|
|
156 }
|
|
157
|
|
158 public void framebufferUpdateMessage() throws CommonException {
|
|
159 reader.readByte(); // padding
|
|
160 int numberOfRectangles = reader.readUInt16();
|
|
161 while (numberOfRectangles-- > 0) {
|
|
162 FramebufferUpdateRectangle rect = new FramebufferUpdateRectangle();
|
|
163 rect.fill(reader);
|
|
164
|
|
165 Decoder decoder = decoders.getDecoderByType(rect.getEncodingType());
|
|
166 logger.finest(rect.toString() + (0 == numberOfRectangles ? "\n---" : ""));
|
|
167 if (decoder != null) {
|
|
168 decoder.decode(reader, renderer, rect);
|
|
169 repaintController.repaintBitmap(rect);
|
|
170 } else if (rect.getEncodingType() == EncodingType.RICH_CURSOR) {
|
|
171 RichCursorDecoder.getInstance().decode(reader, renderer, rect);
|
|
172 repaintController.repaintCursor();
|
|
173 } else if (rect.getEncodingType() == EncodingType.CURSOR_POS) {
|
|
174 renderer.decodeCursorPosition(rect);
|
|
175 repaintController.repaintCursor();
|
|
176 } else if (rect.getEncodingType() == EncodingType.DESKTOP_SIZE) {
|
|
177 fullscreenFbUpdateIncrementalRequest =
|
|
178 new FramebufferUpdateRequestMessage(0, 0, rect.width, rect.height, true);
|
|
179 synchronized (renderer.getLock()) {
|
|
180 renderer = repaintController.createRenderer(reader, rect.width, rect.height,
|
|
181 context.getPixelFormat());
|
|
182 }
|
9
|
183 AliceVNCMessage message = new AliceVNCMessage();
|
|
184 message.setRectangle(rect);
|
|
185 message.setPixelFormat(context.getPixelFormat());
|
17
|
186 DataSegment.getLocal().put("aliceVNCMessage", message, null);
|
9
|
187
|
0
|
188 context.sendMessage(new FramebufferUpdateRequestMessage(0, 0, rect.width, rect.height, false));
|
|
189 // repaintController.repaintCursor();
|
|
190 } else
|
|
191 throw new CommonException("Unprocessed encoding: " + rect.toString());
|
|
192 }
|
|
193 synchronized (this) {
|
|
194 if (needSendPixelFormat) {
|
|
195 needSendPixelFormat = false;
|
|
196 context.setPixelFormat(pixelFormat);
|
|
197 context.sendMessage(new SetPixelFormatMessage(pixelFormat));
|
|
198 logger.fine("sent: "+pixelFormat);
|
|
199 context.sendRefreshMessage();
|
|
200 logger.fine("sent: nonincremental fb update");
|
|
201 } else {
|
|
202 context.sendMessage(fullscreenFbUpdateIncrementalRequest);
|
|
203 }
|
|
204 }
|
|
205 }
|
|
206
|
|
207 public synchronized void queueUpdatePixelFormat(PixelFormat pf) {
|
|
208 pixelFormat = pf;
|
|
209 needSendPixelFormat = true;
|
|
210 // context.sendMessage(new FramebufferUpdateRequestMessage(0, 0, 1, 1, false));
|
|
211 }
|
|
212
|
|
213 public void stopTask() {
|
|
214 isRunning = false;
|
|
215 }
|
|
216
|
|
217 }
|