Mercurial > hg > Applications > TreeVNC
view src/main/java/com/glavsoft/rfb/protocol/Protocol.java @ 326:1d4d5055a288
add error message, add assure stream close.
author | oc |
---|---|
date | Sun, 01 Feb 2015 15:30:17 +0900 |
parents | a237c7e3b7ca |
children | 42fcc9419498 |
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.core.SettingsChangedEvent; import com.glavsoft.exceptions.*; import com.glavsoft.rfb.*; import com.glavsoft.rfb.client.ClientToServerMessage; import com.glavsoft.rfb.client.FramebufferUpdateRequestMessage; import com.glavsoft.rfb.client.SetEncodingsMessage; import com.glavsoft.rfb.client.SetPixelFormatMessage; import com.glavsoft.rfb.encoding.PixelFormat; import com.glavsoft.rfb.encoding.decoder.DecodersContainer; import com.glavsoft.rfb.protocol.state.HandshakeState; import com.glavsoft.rfb.protocol.state.ProtocolState; import com.glavsoft.transport.Reader; import com.glavsoft.transport.Writer; import java.util.logging.Logger; import jp.ac.u_ryukyu.treevnc.TreeRFBProto; public class Protocol implements ProtocolContext, IChangeSettingsListener { private ProtocolState state; private final Logger logger; private final IPasswordRetriever passwordRetriever; private final ProtocolSettings settings; private int fbWidth; private int fbHeight; private PixelFormat pixelFormat; private final Reader reader; private final Writer writer; private String remoteDesktopName; private MessageQueue messageQueue; private final DecodersContainer decoders; private SenderTask senderTask; private ReceiverTask receiverTask; private IRfbSessionListener rfbSessionListener; private IRepaintController repaintController; private PixelFormat serverPixelFormat; private Thread senderThread; private Thread receiverThread; private boolean isTight; private String protocolVersion; private byte[] initData; private boolean isRetina = false; private TreeRFBProto rfb; public Protocol(Reader reader, Writer writer, IPasswordRetriever passwordRetriever, ProtocolSettings settings, TreeRFBProto myRfb) { this.reader = reader; this.writer = writer; this.passwordRetriever = passwordRetriever; this.settings = settings; decoders = new DecodersContainer(); decoders.instantiateDecodersWhenNeeded(settings.encodings); state = new HandshakeState(this); logger = Logger.getLogger(getClass().getName()); this.rfb = myRfb; } @Override public void changeStateTo(ProtocolState state) { this.state = state; } public void handshake() throws UnsupportedProtocolVersionException, UnsupportedSecurityTypeException, AuthenticationFailedException, TransportException, FatalException { while (state.next()) { // continue; } this.messageQueue = new MessageQueue(); // ここでsoundpacketqueueを作成する } @Override public PixelFormat getPixelFormat() { return pixelFormat; } @Override public void setPixelFormat(PixelFormat pixelFormat) { this.pixelFormat = pixelFormat; if (repaintController != null) { repaintController.setPixelFormat(pixelFormat); } } @Override public String getRemoteDesktopName() { return remoteDesktopName; } @Override public void setRemoteDesktopName(String name) { remoteDesktopName = name; } @Override public int getFbWidth() { return fbWidth; } @Override public void setFbWidth(int fbWidth) { if(!isRetina) this.fbWidth = fbWidth; } @Override public int getFbHeight() { return fbHeight; } @Override public byte[] getInitData() { return initData; } @Override public void setInitData(byte[] initData) { this.initData = initData; } @Override public void setFbHeight(int fbHeight) { if(!isRetina) this.fbHeight = fbHeight; } @Override public IPasswordRetriever getPasswordRetriever() { return passwordRetriever; } @Override public ProtocolSettings getSettings() { return settings; } @Override public Writer getWriter() { return writer; } @Override public Reader getReader() { return reader; } /** * Following the server initialisation message it's up to the client to send * whichever protocol messages it wants. Typically it will send a * SetPixelFormat message and a SetEncodings message, followed by a * FramebufferUpdateRequest. From then on the server will send * FramebufferUpdate messages in response to the client's * FramebufferUpdateRequest messages. The client should send * FramebufferUpdateRequest messages with incremental set to true when it has * finished processing one FramebufferUpdate and is ready to process another. * With a fast client, the rate at which FramebufferUpdateRequests are sent * should be regulated to avoid hogging the network. */ public void startNormalHandling(IRfbSessionListener rfbSessionListener, IRepaintController repaintController, ClipboardController clipboardController, TreeRFBProto rfb) { receiverTask = new ReceiverTask( reader, repaintController, clipboardController, decoders, this, rfb); startNormalHandling1(rfbSessionListener, repaintController, clipboardController); } public void startNormalHandling1(IRfbSessionListener rfbSessionListener, IRepaintController repaintController, ClipboardController clipboardController) { this.rfbSessionListener = rfbSessionListener; this.repaintController = repaintController; // if (settings.getColorDepth() == 0) { // settings.setColorDepth(pixelFormat.depth); // the same the server sent when not initialized yet // } serverPixelFormat = pixelFormat; correctServerPixelFormat(); setPixelFormat(createPixelFormat(settings)); sendMessage(new SetPixelFormatMessage(pixelFormat)); logger.fine("sent: " + pixelFormat); sendSupportedEncodingsMessage(settings); settings.addListener(this); // to support pixel format (color depth), and encodings changes settings.addListener(repaintController); sendRefreshMessage(); senderTask = new SenderTask(messageQueue, writer, this); senderThread = new Thread(senderTask, "RfbSenderTask"); senderThread.start(); decoders.resetDecoders(); receiverThread = new Thread(receiverTask, "RfbReceiverTask"); receiverThread.start(); } private void correctServerPixelFormat() { // correct true color flag - we don't support color maps, so always set it up serverPixelFormat.trueColourFlag = 1; // correct .depth to use actual depth 24 instead of incorrect 32, used by ex. UltraVNC server, that cause // protocol incompatibility in ZRLE encoding final long significant = serverPixelFormat.redMax << serverPixelFormat.redShift | serverPixelFormat.greenMax << serverPixelFormat.greenShift | serverPixelFormat.blueMax << serverPixelFormat.blueShift; if (32 == serverPixelFormat.bitsPerPixel && ((significant & 0x00ff000000L) == 0 || (significant & 0x000000ffL) == 0) && 32 == serverPixelFormat.depth) { serverPixelFormat.depth = 24; } } @Override public void sendMessage(ClientToServerMessage message) { messageQueue.put(message); } private void sendSupportedEncodingsMessage(ProtocolSettings settings) { decoders.instantiateDecodersWhenNeeded(settings.encodings); SetEncodingsMessage encodingsMessage = new SetEncodingsMessage(settings.encodings); sendMessage(encodingsMessage); logger.fine("sent: " + encodingsMessage.toString()); } /** * create pixel format by bpp */ private PixelFormat createPixelFormat(ProtocolSettings settings) { int serverBigEndianFlag = serverPixelFormat.bigEndianFlag; switch (settings.getColorDepth()) { case ProtocolSettings.COLOR_DEPTH_24: return PixelFormat.create24bitColorDepthPixelFormat(serverBigEndianFlag); case ProtocolSettings.COLOR_DEPTH_16: return PixelFormat.create16bitColorDepthPixelFormat(serverBigEndianFlag); case ProtocolSettings.COLOR_DEPTH_8: return PixelFormat.create8bitColorDepthBGRPixelFormat(serverBigEndianFlag); case ProtocolSettings.COLOR_DEPTH_6: return PixelFormat.create6bitColorDepthPixelFormat(serverBigEndianFlag); case ProtocolSettings.COLOR_DEPTH_3: return PixelFormat.create3bppPixelFormat(serverBigEndianFlag); case ProtocolSettings.COLOR_DEPTH_SERVER_SETTINGS: return serverPixelFormat; default: // unsupported bpp, use default return PixelFormat.create24bitColorDepthPixelFormat(serverBigEndianFlag); } } @Override public void settingsChanged(SettingsChangedEvent e) { ProtocolSettings settings = (ProtocolSettings) e.getSource(); if (settings.isChangedEncodings()) { sendSupportedEncodingsMessage(settings); } if (settings.isChangedColorDepth() && receiverTask != null) { receiverTask.queueUpdatePixelFormat(createPixelFormat(settings)); } } @Override public void sendRefreshMessage() { sendMessage(new FramebufferUpdateRequestMessage(0, 0, fbWidth, fbHeight, false)); // sendMessage(new FramebufferUpdateRequestMessage(0, 0, 1920, 1080, false)); logger.fine("sent: full FB Refresh"); } @Override public void cleanUpSession(String message) { cleanUpSession(); if (message!=null) rfbSessionListener.rfbSessionStopped(message); } public synchronized void cleanUpSession() { if (senderTask != null) { senderTask.stopTask(); try { senderThread.join(); } catch (InterruptedException e) { System.out.println("any thread has interrupted the current thread"); } senderTask = null; } stopReceiverTask(); } @Override public void stopReceiverTask() { if (receiverTask != null) { receiverTask.stopTask(); try { receiverThread.join(); } catch (InterruptedException e) { System.out.println("any thread has interrupted the current thread"); } receiverTask = null; } } @Override public int getNumberOfRectangle() { int numberOfRectangles = receiverTask.numberOfRectangles; return numberOfRectangles; } @Override public TreeRFBProto getRfb() { return rfb; } @Override public void setTight(boolean isTight) { this.isTight = isTight; } @Override public boolean isTight() { return isTight; } @Override public void setProtocolVersion(String protocolVersion) { this.protocolVersion = protocolVersion; } @Override public String getProtocolVersion() { return protocolVersion; } @Override public void resetDecoder() { decoders.resetDecoders(); } }