Mercurial > hg > Applications > TreeVNC
diff src/main/java/com/glavsoft/rfb/encoding/decoder/TightDecoder.java @ 0:4689cc86d6cb
create TreeViewer2 Repository
author | Yu Taninari <you@cr.ie.u-ryukyu.ac.jp> |
---|---|
date | Tue, 03 Jul 2012 13:20:49 +0900 |
parents | |
children | 0c08cdc4b572 17b702648079 |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/main/java/com/glavsoft/rfb/encoding/decoder/TightDecoder.java Tue Jul 03 13:20:49 2012 +0900 @@ -0,0 +1,301 @@ +// Copyright (C) 2010, 2011 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.encoding.decoder; + +import java.util.logging.Logger; +import java.util.zip.DataFormatException; +import java.util.zip.Inflater; + +import com.glavsoft.drawing.ColorDecoder; +import com.glavsoft.drawing.Renderer; +import com.glavsoft.exceptions.TransportException; +import com.glavsoft.transport.Reader; + +/** + * Tight protocol extention decoder + */ +public class TightDecoder extends Decoder { + private static Logger logger = Logger.getLogger("com.glavsoft.rfb.encoding.decoder"); + + private static final int FILL_TYPE = 0x08; + private static final int JPEG_TYPE = 0x09; + + private static final int FILTER_ID_MASK = 0x40; + private static final int STREAM_ID_MASK = 0x30; + + private static final int BASIC_FILTER = 0x00; + private static final int PALETTE_FILTER = 0x01; + private static final int GRADIENT_FILTER = 0x02; + private static final int MIN_SIZE_TO_COMPRESS = 12; + + static final int DECODERS_NUM = 4; + Inflater[] decoders; + + private int decoderId; + + final static int tightZlibBufferSize = 512; + + public TightDecoder() { + reset(); + } + + @Override + public void decode(Reader reader, Renderer renderer, + FramebufferUpdateRectangle rect) throws TransportException { + int bytesPerPixel = renderer.getBytesPerPixelSignificant(); + + /** + * bits + * 7 - FILL or JPEG type + * 6 - filter presence flag + * 5, 4 - decoder to use when Basic type (bit 7 not set) + * or + * 4 - JPEG type when set bit 7 + * 3 - reset decoder #3 + * 2 - reset decoder #2 + * 1 - reset decoder #1 + * 0 - reset decoder #0 + */ + int compControl = reader.readUInt8(); + resetDecoders(compControl); + + int compType = compControl >> 4 & 0x0F; + switch (compType) { + case FILL_TYPE: + int color = renderer.readTightPixelColor(reader); + renderer.fillRect(color, rect); + break; + case JPEG_TYPE: + if (bytesPerPixel != 3) { +// throw new EncodingException( +// "Tight doesn't support JPEG subencoding while depth not equal to 24bpp is used"); + } + processJpegType(reader, renderer, rect); + break; + default: + if (compType > JPEG_TYPE) { +// throw new EncodingException( +// "Compression control byte is incorrect!"); + } else { + processBasicType(compControl, reader, renderer, rect); + } + } + } + + private void processBasicType(int compControl, Reader reader, + Renderer renderer, FramebufferUpdateRectangle rect) throws TransportException { + decoderId = (compControl & STREAM_ID_MASK) >> 4; + + int filterId = 0; + if ((compControl & FILTER_ID_MASK) > 0) { // filter byte presence + filterId = reader.readUInt8(); + } + int bytesPerCPixel = renderer.getBytesPerPixelSignificant(); + int lengthCurrentbpp = bytesPerCPixel * rect.width * rect.height; + byte [] buffer; + switch (filterId) { + case BASIC_FILTER: + buffer = readTightData(lengthCurrentbpp, reader); + renderer.drawTightBytes(buffer, 0, rect.x, rect.y, rect.width, rect.height); + break; + case PALETTE_FILTER: + int paletteSize = reader.readUInt8() + 1; + int[] palette = readPalette(paletteSize, reader, renderer); + int dataLength = paletteSize == 2 ? + rect.height * ((rect.width + 7) / 8) : + rect.width * rect.height; + buffer = readTightData(dataLength, reader); + renderer.drawBytesWithPalette(buffer, rect, palette); + break; + case GRADIENT_FILTER: +/* + * The "gradient" filter pre-processes pixel data with a simple algorithm + * which converts each color component to a difference between a "predicted" + * intensity and the actual intensity. Such a technique does not affect + * uncompressed data size, but helps to compress photo-like images better. + * Pseudo-code for converting intensities to differences is the following: + * + * P[i,j] := V[i-1,j] + V[i,j-1] - V[i-1,j-1]; + * if (P[i,j] < 0) then P[i,j] := 0; + * if (P[i,j] > MAX) then P[i,j] := MAX; + * D[i,j] := V[i,j] - P[i,j]; + * + * Here V[i,j] is the intensity of a color component for a pixel at + * coordinates (i,j). MAX is the maximum value of intensity for a color + * component.*/ + buffer = readTightData(bytesPerCPixel * rect.width * rect.height, reader); + byte [][] opRows = new byte[2][rect.width * 3 + 3]; + int opRowIndex = 0; + byte [] components = new byte[3]; + int pixelOffset = 0; + ColorDecoder colorDecoder = renderer.getColorDecoder(); + for (int i = 0; i < rect.height; ++i) { + // exchange thisRow and prevRow: + byte [] thisRow = opRows[opRowIndex]; + byte [] prevRow = opRows[opRowIndex = (opRowIndex + 1) % 2]; + for (int j = 3; j < rect.width * 3 + 3; j += 3) { + colorDecoder.fillRawComponents(components, buffer, pixelOffset); + pixelOffset += bytesPerCPixel; + int + d = (0xff & prevRow[j + 0]) + // "upper" pixel (from prev row) + (0xff & thisRow[j + 0 - 3]) - // prev pixel + (0xff & prevRow[j + 0 - 3]); // "diagonal" prev pixel + thisRow[j + 0] = (byte) (components[0] + (d < 0 ? 0 : d > colorDecoder.redMax ? colorDecoder.redMax: d) & colorDecoder.redMax); + d = (0xff & prevRow[j + 1]) + + (0xff & thisRow[j + 1 - 3]) - + (0xff & prevRow[j + 1 - 3]); + thisRow[j + 1] = (byte) (components[1] + (d < 0 ? 0 : d > colorDecoder.greenMax ? colorDecoder.greenMax: d) & colorDecoder.greenMax); + d = (0xff & prevRow[j + 2]) + + (0xff & thisRow[j + 2 - 3]) - + (0xff & prevRow[j + 2 - 3]); + thisRow[j + 2] = (byte) (components[2] + (d < 0 ? 0 : d > colorDecoder.blueMax ? colorDecoder.blueMax: d) & colorDecoder.blueMax); + } + renderer.drawUncaliberedRGBLine(thisRow, rect.x, rect.y + i, rect.width); + } + + break; + default: + break; + } + } + + /** + * Read palette from reader + */ + private int[] readPalette(int paletteSize, Reader reader, Renderer renderer) throws TransportException { + /** + * When bytesPerPixel == 1 && paletteSize == 2 read 2 bytes of palette + * When bytesPerPixel == 1 && paletteSize != 2 - error + * When bytesPerPixel == 3 (4) read (paletteSize * 3) bytes of palette + * so use renderer.readPixelColor + */ + int[] palette = new int[paletteSize]; + for (int i = 0; i < palette.length; ++i) { + palette[i] = renderer.readTightPixelColor(reader); + } + return palette; + } + + /** + * Reads compressed (expected length >= MIN_SIZE_TO_COMPRESS) or + * uncompressed data. When compressed decompresses it. + * + * @param expectedLength expected data length in bytes + * @param reader + * @return result data + * @throws TransportException + */ + private byte[] readTightData(int expectedLength, Reader reader) throws TransportException { + if (expectedLength < MIN_SIZE_TO_COMPRESS) { + byte [] buffer = ByteBuffer.getInstance().getBuffer(expectedLength); + reader.readBytes(buffer, 0, expectedLength); + return buffer; + } else + return readCompressedData(expectedLength, reader); + } + + /** + * Reads compressed data length, then read compressed data into rawBuffer + * and decompress data with expected length == length + * + * Note: returned data contains not only decompressed data but raw data at array tail + * which need to be ignored. Use only first expectedLength bytes. + * + * @param expectedLength expected data length + * @param reader + * @return decompressed data (length == expectedLength) / + followed raw data (ignore, please) + * @throws TransportException + */ + private byte[] readCompressedData(int expectedLength, Reader reader) throws TransportException { + int rawDataLength = readCompactSize(reader); + + byte [] buffer = ByteBuffer.getInstance().getBuffer(expectedLength + rawDataLength); + // read compressed (raw) data behind space allocated for decompressed data + reader.readBytes(buffer, expectedLength, rawDataLength); + if (null == decoders[decoderId]) { + decoders[decoderId] = new Inflater(); + } + Inflater decoder = decoders[decoderId]; + decoder.setInput(buffer, expectedLength, rawDataLength); + try { + decoder.inflate(buffer, 0, expectedLength); + } catch (DataFormatException e) { + logger.throwing("TightDecoder", "readCompressedData", e); + throw new TransportException("cannot inflate tight compressed data", e); + } + return buffer; + } + + private void processJpegType(Reader reader, Renderer renderer, + FramebufferUpdateRectangle rect) throws TransportException { + int jpegBufferLength = readCompactSize(reader); + byte [] bytes = ByteBuffer.getInstance().getBuffer(jpegBufferLength); + reader.readBytes(bytes, 0, jpegBufferLength); + renderer.drawJpegImage(bytes, 0, jpegBufferLength, rect); + } + + /** + * Read an integer from reader in compact representation (from 1 to 3 bytes). + * Highest bit of read byte set to 1 means next byte contains data. + * Lower 7 bit of each byte contains significant data. Max bytes = 3. + * Less significant bytes first order. + * + * @param reader + * @return int value + * @throws TransportException + */ + private int readCompactSize(Reader reader) throws TransportException { + int b = reader.readUInt8(); + int size = b & 0x7F; + if ((b & 0x80) != 0) { + b = reader.readUInt8(); + size += (b & 0x7F) << 7; + if ((b & 0x80) != 0) { + size += reader.readUInt8() << 14; + } + } + return size; + } + + /** + * Flush (reset) zlib decoders when bits 3, 2, 1, 0 of compControl is set + * @param compControl + */ + private void resetDecoders(int compControl) { + for (int i=0; i < DECODERS_NUM; ++i) { + if ((compControl & 1) != 0 && decoders[i] != null) { + decoders[i].reset(); + } + compControl >>= 1; + } + + } + + @Override + public void reset() { + decoders = new Inflater[DECODERS_NUM]; + } + +}