Mercurial > hg > Applications > TreeVNC
view src/main/java/jp/ac/u_ryukyu/treevnc/MyRfbProto.java @ 77:cdf7bbe45885
remove some warnings
author | Shinji KONO <kono@ie.u-ryukyu.ac.jp> |
---|---|
date | Tue, 29 Apr 2014 18:00:34 +0900 |
parents | f1938dd3b518 |
children | 432dac0b9dae |
line wrap: on
line source
package jp.ac.u_ryukyu.treevnc; import java.io.DataOutputStream; import java.io.IOException; import java.net.Socket; import java.nio.ByteBuffer; import java.util.LinkedList; import java.util.concurrent.atomic.AtomicInteger; import jp.ac.u_ryukyu.treevnc.client.EchoClient; import jp.ac.u_ryukyu.treevnc.server.RequestScreenThread; import com.glavsoft.exceptions.TransportException; import com.glavsoft.rfb.protocol.Protocol; import com.glavsoft.rfb.protocol.ProtocolContext; import com.glavsoft.transport.Reader; import com.glavsoft.transport.Writer; import com.glavsoft.viewer.ViewerImpl; public class MyRfbProto { final static int FramebufferUpdateRequest = 3; final static int CheckDelay = 11; final static int FramebufferUpdate = 0; private ProtocolContext context; //final static String versionMsg_3_855 = "RFB 003.855\n"; final static String versionMsg_3_856 = "RFB 003.856\n"; private int clients; protected MulticastQueue<LinkedList<ByteBuffer>> multicastqueue = new MulticastQueue<LinkedList<ByteBuffer>>(); private RequestScreenThread rThread; private boolean proxyFlag = true; private EchoClient echo; private String proxyAddr; public int acceptPort; protected boolean readyReconnect; private boolean cuiVersion; public MyRfbProto() { rThread = new RequestScreenThread(this); } public void newClient(AcceptThread acceptThread, final Socket newCli, final Writer os, final Reader is) throws IOException { // createBimgFlag = true; // rfb.addSockTmp(newCli); // addSock(newCli); final int myId = clients; final MulticastQueue.Client<LinkedList<ByteBuffer>> c = multicastqueue.newClient(); final AtomicInteger writerRunning = new AtomicInteger(); writerRunning.set(1); /** * Timeout thread. If a client is suspended, it has top of queue * indefinitely, which caused memory overflow. After the timeout, we * poll the queue and discard it. Start long wait if writer is running. */ final Runnable timer = new Runnable() { public void run() { int count = 0; for (;;) { long timeout = 50000 / 8; try { synchronized (this) { int state, flag; writerRunning.set(0); wait(timeout); flag = 0; while ((state = writerRunning.get()) == 0) { c.poll(); // discard, should be timeout count++; if (flag == 0) { System.out.println("Discarding " + myId + " count=" + count); flag = 1; } wait(10); // if this is too short, writer cannot // take the poll, if this is too // long, memory will overflow... } if (flag == 1) System.out.println("Resuming " + myId + " count=" + count); if (state != 1) { System.out.println("Client died " + myId); break; } } } catch (InterruptedException e) { } } } }; new Thread(timer).start(); /** * discard all incoming from clients */ final Runnable reader = new Runnable() { public void run() { byte b[] = new byte[4096]; for (;;) { try { int c = is.readByte(b); if (c <= 0) throw new IOException(); // System.out.println("client read "+c); } catch (IOException e) { try { writerRunning.set(2); os.close(); is.close(); break; } catch (IOException e1) { } catch (TransportException e1) { e1.printStackTrace(); } return; } catch (TransportException e) { e.printStackTrace(); } } } }; /** * send packets to a client */ Runnable sender = new Runnable() { public void run() { writerRunning.set(1); try { requestThreadNotify(); // rThread.checkDelay(); /** * initial connection of RFB protocol */ sendRfbVersion(os); // readVersionMsg(is); readVersionMsg(is, os); sendSecurityType(os); readSecType(is); sendSecResult(os); readClientInit(is); sendInitData(os); new Thread(reader).start(); // discard incoming packet here // after. // writeFramebufferUpdateRequest(0,0, framebufferWidth, // framebufferHeight, false ); for (;;) { LinkedList<ByteBuffer> bufs = c.poll(); int inputIndex = 0; ByteBuffer header = bufs.get(inputIndex); if (header == null) continue; else if (header.get(0) == CheckDelay) { writeToClient(os, bufs, inputIndex); continue; } else if (header.get(0) == FramebufferUpdate) { //System.out.println("client "+ myId); } /* * if(i%20==0){ sendDataCheckDelay(); } i++; */ writeToClient(os, bufs, inputIndex); writerRunning.set(1); // yes my client is awaking. } } catch (IOException e) { try { writerRunning.set(2); os.close(); } catch (IOException e1) { } /* if socket closed cliList.remove(newCli); */ } catch (TransportException e) { e.printStackTrace(); } } public void writeToClient(final Writer os, LinkedList<ByteBuffer> bufs, int inputIndex) throws TransportException { while (inputIndex < bufs.size()) { ByteBuffer b = bufs.get(inputIndex++); os.write(b.array(), b.position(), b.limit()); } os.flush(); bufs = null; multicastqueue.heapAvailable(); } }; clients++; new Thread(sender).start(); } public synchronized void requestThreadNotify() { rThread.reStart(); } private void sendRfbVersion(Writer writer) throws IOException, TransportException { // os.write(versionMsg_3_8.getBytes()); writer.write(versionMsg_3_856.getBytes()); } private int readVersionMsg(Reader reader, Writer writer) throws IOException, TransportException { byte[] b = new byte[12]; reader.readBytes(b); if ((b[0] != 'R') || (b[1] != 'F') || (b[2] != 'B') || (b[3] != ' ') || (b[4] < '0') || (b[4] > '9') || (b[5] < '0') || (b[5] > '9') || (b[6] < '0') || (b[6] > '9') || (b[7] != '.') || (b[8] < '0') || (b[8] > '9') || (b[9] < '0') || (b[9] > '9') || (b[10] < '0') || (b[10] > '9') || (b[11] != '\n')) { throw new IOException("this is not an RFB server"); } int rfbMajor = (b[4] - '0') * 100 + (b[5] - '0') * 10 + (b[6] - '0'); int rfbMinor = (b[8] - '0') * 100 + (b[9] - '0') * 10 + (b[10] - '0'); if (rfbMajor < 3) { throw new IOException( "RFB server does not support protocol version 3"); } if (rfbMinor == 855) { sendProxyFlag(writer); if (proxyFlag) sendPortNumber(writer); } return rfbMinor; } public void screenChangeRequest() throws IOException { Socket echoSocket = new Socket(proxyAddr, 10002); DataOutputStream os = new DataOutputStream(echoSocket.getOutputStream()); os.writeBytes(echo.getMyAddress()+"\n"); //os.writeBytes(String.valueOf(echo.client.getFrameWidth())+"\n"); temp comment out for rebuild //os.writeBytes(String.valueOf(echo.client.getFrameHeight())+"\n"); temp comment out for rebuild os.close(); echoSocket.close(); } private void sendProxyFlag(Writer writer) throws TransportException { if (proxyFlag) writer.writeInt(1); else writer.writeInt(0); } private void sendPortNumber(Writer writer) throws TransportException { byte[] b = new byte[4]; //b = castIntByte(getHost.getPort()); b = castIntByte(9999); writer.write(b); } private byte[] castIntByte(int len) { byte[] b = new byte[4]; b[0] = (byte) ((len >>> 24) & 0xFF); b[1] = (byte) ((len >>> 16) & 0xFF); b[2] = (byte) ((len >>> 8) & 0xFF); b[3] = (byte) ((len >>> 0) & 0xFF); return b; } private void readSecType(Reader reader) throws TransportException { byte[] b = new byte[1]; reader.read(b); } private void sendSecurityType(Writer os) throws TransportException { // number-of-security-types os.writeInt(1); // security-types // 1:None os.writeInt(1); /* * os.write(4); os.write(30); os.write(31); os.write(32); os.write(35); * os.flush(); */ } private void sendSecResult(Writer os) throws TransportException { byte[] b = castIntByte(0); os.write(b); } private void readClientInit(Reader in) throws TransportException { byte[] b = new byte[0]; in.readBytes(b); } private void sendInitData(Writer os) throws TransportException { os.write(context.getInitData()); } public void setProtocolContext(Protocol workingProtocol) { context = workingProtocol; } public void readSendData(int dataLen, Reader reader) throws TransportException { } public Socket accept() throws IOException { return null; } public int selectPort(int port) { return port; } public void writeFramebufferUpdateRequest(int x, int y, int w, int h, boolean incremental) throws TransportException { byte[] b = new byte[10]; b[0] = (byte) FramebufferUpdateRequest; // 3 is FrameBufferUpdateRequest b[1] = (byte) (incremental ? 1 : 0); b[2] = (byte) ((x >> 8) & 0xff); b[3] = (byte) (x & 0xff); b[4] = (byte) ((y >> 8) & 0xff); b[5] = (byte) (y & 0xff); b[6] = (byte) ((w >> 8) & 0xff); b[7] = (byte) (w & 0xff); b[8] = (byte) ((h >> 8) & 0xff); b[9] = (byte) (h & 0xff); // os.write(b); } public void notProxy() { proxyFlag = false; } public void setEcho(EchoClient _echo) { echo = _echo; } public void setViewer(ViewerImpl v) { echo.setViewer(v); } public ViewerImpl getViewer() { return echo.getViewer(); } public EchoClient getEcho() { return echo; } public void setTerminationType(boolean setType) { /*nop*/ } public boolean getTerminationType() { /*nop*/ return true; } public void setProxyAddr(String proxyAddr) { this.proxyAddr = proxyAddr; } public void close() { //nothing } public int getAcceptPort() { return 0; } public boolean getReadyReconnect() { return readyReconnect; } public void setReadyReconnect(boolean ready) { } public boolean getCuiVersion() { return cuiVersion; } public void setCuiVersion(boolean flag) { cuiVersion = flag; } public void readCheckDelay(Reader reader) throws TransportException { } public String getProxyAddr() { return proxyAddr; } }