blob: 968edcfb36df2faa4ed51175b049bbc4c64e9140 [file] [log] [blame]
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +00001/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved.
2 *
3 * This is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation; either version 2 of the License, or
6 * (at your option) any later version.
7 *
8 * This software is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this software; if not, write to the Free Software
15 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
16 * USA.
17 */
18#include <rdr/OutStream.h>
19#include <rfb/Exception.h>
Pierre Ossman456b2c22014-01-15 13:22:03 +010020#include <rfb/TransImageGetter.h>
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000021#include <rfb/encodings.h>
22#include <rfb/ConnParams.h>
23#include <rfb/SMsgWriter.h>
Pierre Ossman668468b2014-01-31 12:37:32 +010024#include <rfb/SConnection.h>
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000025#include <rfb/ZRLEEncoder.h>
26#include <rfb/Configuration.h>
27
28using namespace rfb;
29
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000030IntParameter zlibLevel("ZlibLevel","Zlib compression level",-1);
31
Pierre Ossman7b5c0692014-03-17 14:35:51 +010032static inline void writeOpaque24A(rdr::OutStream* os, rdr::U32 u)
33{
34 os->check(3);
35 os->writeU8(((rdr::U8*)&u)[0]);
36 os->writeU8(((rdr::U8*)&u)[1]);
37 os->writeU8(((rdr::U8*)&u)[2]);
38}
39static inline void writeOpaque24B(rdr::OutStream* os, rdr::U32 u)
40{
41 os->check(3);
42 os->writeU8(((rdr::U8*)&u)[1]);
43 os->writeU8(((rdr::U8*)&u)[2]);
44 os->writeU8(((rdr::U8*)&u)[3]);
45}
46
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000047#define BPP 8
48#include <rfb/zrleEncode.h>
49#undef BPP
50#define BPP 16
51#include <rfb/zrleEncode.h>
52#undef BPP
53#define BPP 32
54#include <rfb/zrleEncode.h>
55#define CPIXEL 24A
56#include <rfb/zrleEncode.h>
57#undef CPIXEL
58#define CPIXEL 24B
59#include <rfb/zrleEncode.h>
60#undef CPIXEL
61#undef BPP
62
Pierre Ossman668468b2014-01-31 12:37:32 +010063ZRLEEncoder::ZRLEEncoder(SConnection* conn)
64 : Encoder(conn), zos(0,0,zlibLevel), mos(129*1024)
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000065{
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000066}
67
68ZRLEEncoder::~ZRLEEncoder()
69{
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000070}
71
Pierre Ossman717c07b2014-01-21 14:45:10 +010072void ZRLEEncoder::writeRect(const Rect& r, TransImageGetter* ig)
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000073{
Pierre Ossman668468b2014-01-31 12:37:32 +010074 rdr::U8* imageBuf = conn->writer()->getImageBuf(64 * 64 * 4 + 4);
Pierre Ossman4bca9112014-03-13 15:08:36 +010075 mos.clear();
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000076
Pierre Ossman668468b2014-01-31 12:37:32 +010077 switch (conn->cp.pf().bpp) {
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000078 case 8:
Pierre Ossman717c07b2014-01-21 14:45:10 +010079 zrleEncode8(r, &mos, &zos, imageBuf, ig);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000080 break;
81 case 16:
Pierre Ossman717c07b2014-01-21 14:45:10 +010082 zrleEncode16(r, &mos, &zos, imageBuf, ig);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000083 break;
84 case 32:
85 {
Pierre Ossman668468b2014-01-31 12:37:32 +010086 const PixelFormat& pf = conn->cp.pf();
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000087
Pierre Ossman67b2b2f2009-03-06 10:12:55 +000088 Pixel maxPixel = pf.pixelFromRGB((rdr::U16)-1, (rdr::U16)-1, (rdr::U16)-1);
89 bool fitsInLS3Bytes = maxPixel < (1<<24);
90 bool fitsInMS3Bytes = (maxPixel & 0xff) == 0;
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000091
Pierre Ossman67b2b2f2009-03-06 10:12:55 +000092 if ((fitsInLS3Bytes && pf.isLittleEndian()) ||
93 (fitsInMS3Bytes && pf.isBigEndian()))
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000094 {
Pierre Ossman717c07b2014-01-21 14:45:10 +010095 zrleEncode24A(r, &mos, &zos, imageBuf, ig);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000096 }
Pierre Ossman67b2b2f2009-03-06 10:12:55 +000097 else if ((fitsInLS3Bytes && pf.isBigEndian()) ||
98 (fitsInMS3Bytes && pf.isLittleEndian()))
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +000099 {
Pierre Ossman717c07b2014-01-21 14:45:10 +0100100 zrleEncode24B(r, &mos, &zos, imageBuf, ig);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000101 }
102 else
103 {
Pierre Ossman717c07b2014-01-21 14:45:10 +0100104 zrleEncode32(r, &mos, &zos, imageBuf, ig);
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000105 }
106 break;
107 }
108 }
109
Pierre Ossman668468b2014-01-31 12:37:32 +0100110 conn->writer()->startRect(r, encodingZRLE);
111 rdr::OutStream* os = conn->getOutStream();
Pierre Ossman4bca9112014-03-13 15:08:36 +0100112 os->writeU32(mos.length());
113 os->writeBytes(mos.data(), mos.length());
Pierre Ossman668468b2014-01-31 12:37:32 +0100114 conn->writer()->endRect();
Constantin Kaplinskya2adc8d2006-05-25 05:01:55 +0000115}