/******************************************************************************* * Copyright (c) 2016 comtel inc. * * Licensed under the Apache License, version 2.0 (the "License"); you may not use this file except * in compliance with the License. You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software distributed under the License * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express * or implied. See the License for the specific language governing permissions and limitations under * the License. *******************************************************************************/ package org.jfxvnc.net.rfb.codec.handshaker; import java.nio.charset.Charset; import java.nio.charset.StandardCharsets; import java.util.Arrays; import java.util.List; import org.jfxvnc.net.rfb.codec.PixelFormat; import org.jfxvnc.net.rfb.codec.handshaker.event.SecurityResultEvent; import org.jfxvnc.net.rfb.codec.handshaker.event.SecurityTypesEvent; import org.jfxvnc.net.rfb.codec.handshaker.event.ServerInitEvent; import org.jfxvnc.net.rfb.codec.security.SecurityType; import org.jfxvnc.net.rfb.exception.ProtocolException; import org.jfxvnc.net.rfb.exception.SecurityException; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import io.netty.buffer.ByteBuf; import io.netty.channel.ChannelHandlerContext; import io.netty.handler.codec.ReplayingDecoder; class RfbClient38Decoder extends ReplayingDecoder<RfbClient38Decoder.State> implements RfbClientDecoder { private static Logger logger = LoggerFactory.getLogger(RfbClient38Decoder.class); protected final Charset ASCII = StandardCharsets.US_ASCII; public RfbClient38Decoder() { super(State.SEC_TYPES); } enum State { SEC_TYPES, SEC_AUTH, SEC_RESULT, SERVER_INIT } @Override protected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception { switch (state()) { case SEC_TYPES: int numberOfSecurtiyTypes = in.readUnsignedByte(); if (numberOfSecurtiyTypes == 0) { logger.error("no security types available"); decodeErrorMessage(ctx, in); return; } SecurityType[] serverSecTypes = new SecurityType[numberOfSecurtiyTypes]; for (int i = 0; i < numberOfSecurtiyTypes; i++) { int sec = in.readUnsignedByte(); serverSecTypes[i] = SecurityType.valueOf(sec); if (serverSecTypes[i] == SecurityType.UNKNOWN) { logger.error("un supported security types: {}", sec); } } logger.debug("supported security types: {}", Arrays.toString(serverSecTypes)); checkpoint(State.SEC_RESULT); out.add(new SecurityTypesEvent(true, serverSecTypes)); break; case SEC_RESULT: int secResult = in.readInt(); logger.debug("server login {}", secResult == 0 ? "succeed" : "failed"); if (secResult == 1) { int length = in.readInt(); if (length == 0) { out.add(new SecurityResultEvent(false, new ProtocolException("decode error message failed"))); return; } byte[] text = new byte[length]; in.readBytes(text); out.add(new SecurityResultEvent(false, new SecurityException(new String(text, ASCII)))); return; } out.add(new SecurityResultEvent(true)); checkpoint(State.SERVER_INIT); break; case SERVER_INIT: ServerInitEvent initMsg = new ServerInitEvent(); initMsg.setFrameBufferWidth(in.readUnsignedShort()); initMsg.setFrameBufferHeight(in.readUnsignedShort()); initMsg.setPixelFormat(parsePixelFormat(in)); byte[] name = new byte[in.readInt()]; in.readBytes(name); initMsg.setServerName(new String(name, ASCII)); out.add(initMsg); break; default: break; } } private void decodeErrorMessage(ChannelHandlerContext ctx, ByteBuf in) { int length = in.readInt(); if (length == 0) { ctx.fireExceptionCaught(new ProtocolException("decode error message failed")); return; } byte[] text = new byte[length]; in.readBytes(text); ctx.fireExceptionCaught(new ProtocolException(new String(text, ASCII))); } private PixelFormat parsePixelFormat(ByteBuf m) { PixelFormat pf = new PixelFormat(); pf.setBitPerPixel(m.readUnsignedByte()); pf.setDepth(m.readUnsignedByte()); pf.setBigEndian(m.readUnsignedByte() == 1); pf.setTrueColor(m.readUnsignedByte() == 1); pf.setRedMax(m.readUnsignedShort()); pf.setGreenMax(m.readUnsignedShort()); pf.setBlueMax(m.readUnsignedShort()); pf.setRedShift(m.readUnsignedByte()); pf.setGreenShift(m.readUnsignedByte()); pf.setBlueShift(m.readUnsignedByte()); m.skipBytes(3); return pf; } }