Mercurial > hg > Applications > TreeVNC
view src/main/java/com/glavsoft/rfb/protocol/ReceiverTask.java @ 579:5bc128c8e6aa
fix offset
author | Shinji KONO <kono@ie.u-ryukyu.ac.jp> |
---|---|
date | Fri, 07 Feb 2020 19:22:30 +0900 (2020-02-07) |
parents | c6893847c73a |
children | c7527f24e344 |
line wrap: on
line source
// Copyright (C) 2010, 2011, 2012, 2013 GlavSoft LLC. // All rights reserved. // //------------------------------------------------------------------------- // This file is part of the TightVNC software. Please visit our Web site: // // http://www.tightvnc.com/ // // This program 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 program 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 program; if not, write to the Free Software Foundation, Inc., // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. //------------------------------------------------------------------------- // package com.glavsoft.rfb.protocol; import com.glavsoft.drawing.Renderer; import com.glavsoft.exceptions.CommonException; import com.glavsoft.exceptions.ProtocolException; import com.glavsoft.exceptions.TransportException; import com.glavsoft.rfb.ClipboardController; import com.glavsoft.rfb.IRepaintController; import com.glavsoft.rfb.client.FramebufferUpdateRequestMessage; import com.glavsoft.rfb.client.SetPixelFormatMessage; import com.glavsoft.rfb.encoding.EncodingType; import com.glavsoft.rfb.encoding.PixelFormat; import com.glavsoft.rfb.encoding.decoder.*; import com.glavsoft.transport.Reader; import com.glavsoft.viewer.ConnectionPresenter; import com.glavsoft.viewer.ViewerInterface; import jp.ac.u_ryukyu.treevnc.CheckDelayReply; import jp.ac.u_ryukyu.treevnc.TreeRFBProto; import java.io.*; import java.nio.ByteBuffer; import java.util.Arrays; import java.util.LinkedList; import java.util.Timer; import java.util.logging.Logger; import java.util.Objects; public class ReceiverTask implements Runnable { public static final byte FRAMEBUFFER_UPDATE = 0; private static final byte SET_COLOR_MAP_ENTRIES = 1; private static final byte BELL = 2; private static final byte SERVER_CUT_TEXT = 3; private static Logger logger = Logger.getLogger("com.glavsoft.rfb.protocol.ReceiverTask"); private Reader reader; private volatile boolean isRunning = false; private Renderer renderer; private final IRepaintController repaintController; private final ClipboardController clipboardController; protected final DecodersContainer decoders; protected FramebufferUpdateRequestMessage fullscreenFbUpdateIncrementalRequest; protected final ProtocolContext context; protected PixelFormat pixelFormat; protected boolean needSendPixelFormat; private TreeRFBProto rfb; private long checkCounter = 0; public int numberOfRectangles = 0; private Timer timer = null; public ReceiverTask(Reader reader, IRepaintController repaintController, ClipboardController clipboardController, DecodersContainer decoders, final ProtocolContext context, TreeRFBProto _rfb) { rfb = _rfb; this.reader = reader; this.repaintController = repaintController; this.clipboardController = clipboardController; this.context = context; this.decoders = decoders; Decoder decoder = new ZRLEESender(rfb); decoders.setDecoderByType(EncodingType.ZLIB, decoder); decoders.setDecoderByType(EncodingType.ZRLE, decoder); decoders.setDecoderByType(EncodingType.ZRLEE, decoder); if(rfb.fixingSize) { context.setFbWidth(rfb.fixingSizeWidth); context.setFbHeight(rfb.fixingSizeHeight); } ConnectionPresenter cp = rfb.getViewer().getConnectionPresenter(); if(!rfb.getCuiVersion()) { if (cp.getSingleWidth()==0) { // request full screen for the first time cp.setSingleWidth(context.getFbWidth()); cp.setSingleHeight(context.getFbHeight()); } renderer = repaintController.createRenderer(reader, context.getFbWidth(), context.getFbHeight(), context.getPixelFormat()); } else { renderer = new NullRenderer(context.getPixelFormat().bitsPerPixel/8, context.getFbWidth(), context.getFbHeight(), context.getPixelFormat()); } if(rfb.isTreeManager()) { fullscreenFbUpdateIncrementalRequest = new FramebufferUpdateRequestMessage(cp.getX(), cp.getY(), cp.getSingleWidth(), cp.getSingleHeight(), false); connectionFinished(); } } @Override public void run() { isRunning = true; if (fullscreenFbUpdateIncrementalRequest != null) { fullscreenFbUpdateIncrementalRequest.sendFullScreenRequest(context); } while (isRunning) { try { // reader.available(); if (! isRunning) { // server Change in direct mode // pass the input stream to the TreeVNC protocol reader return; } byte messageId = getMessageId(reader); switch (messageId) { case FRAMEBUFFER_UPDATE: // logger.fine("Server message: FramebufferUpdate (0)"); framebufferUpdateMessage(reader); break; case SET_COLOR_MAP_ENTRIES: logger.severe("Server message SetColorMapEntries is not implemented. Skip."); setColorMapEntries(); break; case BELL: logger.fine("Server message: Bell"); // System.out.print("\0007"); // System.out.flush(); break; case SERVER_CUT_TEXT: logger.fine("Server message: CutText (3)"); serverCutText(); break; default: logger.fine("Unsupported server message. Id = " + messageId); continue; } } catch (TransportException e) { System.out.println(e.getMessage()); System.out.println(e.getCause().getClass().getSimpleName()); if (e.getCause().getClass().getSimpleName().equals("SocketTimeoutException")) { sendFrameBufferUpdateRequest(); continue; } logger.severe("Close session : ReceiverTask : " + e.getMessage()); if(!rfb.isTreeManager() && !(rfb.getTerminationType())) { System.out.println("death parent node, wait connect new parent node."); // close viewer ViewerInterface viewer = rfb.getViewer(); viewer.setVisible(false); try { // clean DataInputStream reader.close(); } catch (TransportException e1) { e1.printStackTrace(); } //} else { // rfb.sendDesktopSizeChange((short) -1); } stopTask(); } catch (ProtocolException e) { logger.severe(e.getMessage()); if (isRunning) { context.cleanUpSession(e.getMessage() + "\nConnection closed."); } stopTask(); } catch (CommonException e) { logger.severe(e.getMessage()); if (isRunning) { context.cleanUpSession("Connection closed.."); } stopTask(); } catch (Throwable te) { StringWriter sw = new StringWriter(); PrintWriter pw = new PrintWriter(sw); te.printStackTrace(pw); logger.severe("updateRectangle record failed : " + te); te.printStackTrace(); if (isRunning) { context.cleanUpSession(te.getMessage() + "\n" + sw.toString()); } stopTask(); } } } public byte getMessageId(Reader reader) throws Exception { if(! rfb.isTreeManager() && rfb.isAddSerialNum()) { // client has 8byte packet sequence number // add serial number flag (4byte) reader.mark(20+8+4); getLost(reader); //check seq consistency } else { reader.mark(20+4); } return reader.readByte(); } public void sendFrameBufferUpdateRequest() { if (rfb.isTreeManager()) { sendFrameBufferUpdateRequest0(); } } private void sleep(int i) { try { Thread.sleep(i); } catch (InterruptedException e) { // nothing. } } private void setColorMapEntries() throws TransportException { reader.readByte(); // padding reader.readUInt16(); // first color index int length = reader.readUInt16(); while (length-- > 0) { reader.readUInt16(); // R reader.readUInt16(); // G reader.readUInt16(); // B } } private void serverCutText() throws TransportException { reader.readByte(); // padding reader.readInt16(); // padding int length = reader.readInt32() & Integer.MAX_VALUE; clipboardController.updateSystemClipboard(reader.readBytes(length)); } public void framebufferUpdateMessage(Reader reader) throws Exception { int numberOfRectangles; try { reader.readByte(); // padding numberOfRectangles = reader.readUInt16(); this.numberOfRectangles = numberOfRectangles; if(numberOfRectangles > 3) { System.out.println("numberofrectangle : " + numberOfRectangles); if (rfb.getViewer().getUseMulticast()){ return; // Discard invalid packet } } while (numberOfRectangles-- > 0) { FramebufferUpdateRectangle rect = new FramebufferUpdateRectangle(); rect.fill(reader); Decoder decoder = decoders.getDecoderByType(rect.getEncodingType()); logger.finest(rect.toString() + (0 == numberOfRectangles ? "\n---" : "")); if (decoder != null) { try { System.out.println(rect); decoder.decode(reader, renderer, rect); // TreeVNC processing here if (rfb.getCuiVersion()) continue; repaintController.repaintBitmap(rect); } catch (Exception e) { e.printStackTrace(); break; } } else if (rect.getEncodingType() == EncodingType.RICH_CURSOR) { RichCursorDecoder.getInstance().decode(reader, renderer, rect); if(repaintController!=null) repaintController.repaintCursor(); } else if (rect.getEncodingType() == EncodingType.CURSOR_POS) { renderer.decodeCursorPosition(rect); repaintController.repaintCursor(); } else if (rect.getEncodingType() == EncodingType.EXTENDED_DESKTOP_SIZE) { int numberOfScreen = reader.readByte(); reader.readBytes(3); LinkedList<FramebufferUpdateRectangle> screens = new LinkedList<FramebufferUpdateRectangle>(); for (int i = 0; i < numberOfScreen; i++) { long id = reader.readUInt32(); int x = reader.readUInt16(); int y = reader.readUInt16(); int width = reader.readUInt16(); int height = reader.readUInt16(); long flag = reader.readUInt32(); FramebufferUpdateRectangle screen = new FramebufferUpdateRectangle(x, y, width, height); screen.port = (int) id; screen.time = flag; screens.add(screen); System.out.println("screen " + id + ":" + "x=" + x + " y=" + y + "width=" + width + "height=" + height); } return; } else if (rect.getEncodingType() == EncodingType.DESKTOP_SIZE) { System.out.println("DESKTOP_SIZE"); fullscreenFbUpdateIncrementalRequest = new FramebufferUpdateRequestMessage(rect.x, rect.y, rect.width, rect.height, false); renderer = repaintController.createRenderer(reader, rect.width, rect.height, context.getPixelFormat()); if (rfb.hasViewer()){ setScreenParameter(rect,rect.width,rect.height); } } else if (rect.getEncodingType() == EncodingType.INIT_DATA) { // VNCServer is changed, initiarize new screen. int length = reader.readInt32() - 6; short id = reader.readInt16(); int singleWidth = reader.readUInt16(); int singleHeight = reader.readUInt16(); byte[] initData = new byte[length]; reader.read(initData); String name = new String(initData, 24, length - 24, "UTF-8"); context.setRemoteDesktopName(name); context.setInitData(initData); context.setFbWidth(rect.width); context.setFbHeight(rect.height); repaintController.updateRemoteDesktopName(context); reader.reset(); System.out.println("INIT_DATA: "+rect); // request one screen fullscreenFbUpdateIncrementalRequest = new FramebufferUpdateRequestMessage(rect.x, rect.y, singleWidth, singleHeight, false); // All children multicastqueue should be discarded here. // rfb.clearChildrenTransmission(); rfb.readSendData(length + 16 + 10, reader, null, rect); // size of UpdateRectangleMessage with initData. if (!(rfb.getCuiVersion())) { // keep full frame buffer for multi screen renderer = repaintController.createRenderer(reader, rect.width, rect.height, context.getPixelFormat()); } rfb.setSharingId(id); if (rfb.hasViewer()){ System.out.println("setscreenparameter"); setScreenParameter(rect,singleWidth,singleHeight); } } else if (rect.getEncodingType() == EncodingType.CHECK_DELAY) { int checkDelaySize = 24; int port = rfb.acceptPort; String address = rfb.getMyAddress(); int dataLen = reader.readInt32(); sendCheckDelayReply(rect.time, port, address, dataLen); reader.reset(); reader.readBytes(24); decoder = decoders.getDecoderByType(EncodingType.ZRLEE); decoder.decode(reader, renderer, rect); // TreeVNC processing here repaintController.repaintBitmap(rect); } else if (rect.getEncodingType() == EncodingType.SOUND) { // SOUNDを受信した時の処理を } else if (rect.getEncodingType() == EncodingType.ERROR_ANNOUNCE) { short id = (short) rect.x; int length = reader.readInt32(); byte[] errorMessage = new byte[length]; reader.read(errorMessage); String errorMessageStr = new String(errorMessage, "UTF-8"); if (id == rfb.getId()) { rfb.getViewer().getConnectionPresenter().showPortErrorDialog(errorMessageStr); rfb.getViewer().getConnectionPresenter().clearMessage(); } else { reader.reset(); rfb.readSendData(length + 20, reader, null, rect); } } else throw new CommonException("Unprocessed encoding: " + rect.toString()); } } catch (UnsupportedEncodingException e) { e.getMessage(); e.printStackTrace(); reader.close(); } catch (Exception e) { System.out.println("FrameBufferUpdate: "+e); } sendFrameBufferUpdateRequest(); } public void checkFrameBufferRectanble(ByteBuffer c1, byte[] checkBytes, int flushOffset, int flushEnd) { FramebufferUpdateRectangle rect = new FramebufferUpdateRectangle(); Reader in = new Reader(new ByteArrayInputStream(c1.array())); try { if (getMessageId(in) != FRAMEBUFFER_UPDATE) { ; } in.readByte(); int numberOfRectangeles = in.readInt16(); if (true) { in.mark(c1.limit() - 4); rect.fill(in); if (rect.getEncodingType() == EncodingType.ZRLEE ) { int length = rect.width * rect.height * renderer.getBytesPerPixel(); int zippedLength = (int) in.readUInt32(); ZRLEDecoder decoder = new ZRLEDecoder(); ByteBuffer buf = decoder.unzip(in, zippedLength, length, rect.getEncodingType()); compareBytes(buf, checkBytes, flushOffset, flushEnd); } in.reset(); } while (numberOfRectangeles-- > 0) { rect.fill(in); System.out.println("check rect " + rect); if (rect.getEncodingType() == EncodingType.ZRLEE) { Decoder decoder = new ZRLEDecoder(); decoder.decode(in,new NullRenderer(context.getPixelFormat().bitsPerPixel/8, context.getFbWidth(), context.getFbHeight(), context.getPixelFormat()),rect); } } } catch (Exception e) { e.printStackTrace(); } } private void compareBytes(ByteBuffer buf, byte[] unCompressBytes, int flushOffset, int flushEnd) { int span = flushEnd - flushOffset; if (buf.remaining() == span) { }else { System.out.println("Bytes is not equal length "+buf.remaining()+" != "+span); } } private void setScreenParameter(FramebufferUpdateRectangle rect,int singleWidth ,int singleHeight) { ViewerInterface v = rfb.getViewer(); ConnectionPresenter cp = v.getConnectionPresenter(); cp.setX(rect.x); cp.setY(rect.y); cp.setFrameSizeWidth(rect.width); cp.setFrameSizeHeight(rect.height); cp.setSingleWidth(singleWidth); cp.setSingleHeight(singleHeight); rfb.setConnectionPresenter(cp); v.setFitScreen(); System.out.println("got INIT_DATA: myID = " + rfb.getId() + " sharingID = " + rfb.getSharingId()); if (rfb.isTreeManager()) { repaintController.setVisible(-1 != rfb.getSharingId() && rfb.getId() != rfb.getSharingId()); } else { if (rfb.getSharingId() != -1) repaintController.setVisible(rfb.getId() != rfb.getSharingId()); } } private void sendFrameBufferUpdateRequest0() { if (needSendPixelFormat) { needSendPixelFormat = false; context.setPixelFormat(pixelFormat); context.sendMessage(new SetPixelFormatMessage(pixelFormat)); logger.fine("sent: " + pixelFormat); context.sendMessage(fullscreenFbUpdateIncrementalRequest); logger.fine("sent: nonincremental fb update"); } else { context.sendMessage(fullscreenFbUpdateIncrementalRequest); } } private void sendCheckDelayReply(long time, int port, String address, int dataLen) throws UnsupportedEncodingException { context.sendMessage(new CheckDelayReply(time, port, address, dataLen)); } public synchronized void queueUpdatePixelFormat(PixelFormat pf) { pixelFormat = pf; needSendPixelFormat = true; // context.sendMessage(new FramebufferUpdateRequestMessage(0, 0, 1, 1, false)); } public void stopTask() { isRunning = false; } private void connectionFinished() { rfb.vncConnected(true); } private void getLost(Reader reader) throws Exception { int addSerialNumFlag = reader.readInt32(); long num = reader.readInt64(); if (addSerialNumFlag == 1) { if(num != ++checkCounter) { System.out.println("LostData" + (num - checkCounter)); checkCounter = num; } } } public void setReader(Reader reader) { this.reader = reader; } }