blob: d84c4cfbc887058b65eda638feb6e4f8930d2396 [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>
20#include <rfb/ImageGetter.h>
21#include <rfb/encodings.h>
22#include <rfb/ConnParams.h>
23#include <rfb/SMsgWriter.h>
24#include <rfb/ZRLEEncoder.h>
25#include <rfb/Configuration.h>
26
27using namespace rfb;
28
29rdr::MemOutStream* ZRLEEncoder::sharedMos = 0;
30int ZRLEEncoder::maxLen = 4097 * 1024; // enough for width 16384 32-bit pixels
31
32IntParameter zlibLevel("ZlibLevel","Zlib compression level",-1);
33
34#define EXTRA_ARGS ImageGetter* ig
35#define GET_IMAGE_INTO_BUF(r,buf) ig->getImage(buf, r);
36#define BPP 8
37#include <rfb/zrleEncode.h>
38#undef BPP
39#define BPP 16
40#include <rfb/zrleEncode.h>
41#undef BPP
42#define BPP 32
43#include <rfb/zrleEncode.h>
44#define CPIXEL 24A
45#include <rfb/zrleEncode.h>
46#undef CPIXEL
47#define CPIXEL 24B
48#include <rfb/zrleEncode.h>
49#undef CPIXEL
50#undef BPP
51
52Encoder* ZRLEEncoder::create(SMsgWriter* writer)
53{
54 return new ZRLEEncoder(writer);
55}
56
57ZRLEEncoder::ZRLEEncoder(SMsgWriter* writer_)
58 : writer(writer_), zos(0,0,zlibLevel)
59{
60 if (sharedMos)
61 mos = sharedMos;
62 else
63 mos = new rdr::MemOutStream(129*1024);
64}
65
66ZRLEEncoder::~ZRLEEncoder()
67{
68 if (!sharedMos)
69 delete mos;
70}
71
72bool ZRLEEncoder::writeRect(const Rect& r, ImageGetter* ig, Rect* actual)
73{
74 rdr::U8* imageBuf = writer->getImageBuf(64 * 64 * 4 + 4);
75 mos->clear();
76 bool wroteAll = true;
77 *actual = r;
78
79 switch (writer->bpp()) {
80 case 8:
81 wroteAll = zrleEncode8(r, mos, &zos, imageBuf, maxLen, actual, ig);
82 break;
83 case 16:
84 wroteAll = zrleEncode16(r, mos, &zos, imageBuf, maxLen, actual, ig);
85 break;
86 case 32:
87 {
88 const PixelFormat& pf = writer->getConnParams()->pf();
89
90 bool fitsInLS3Bytes = ((pf.redMax << pf.redShift) < (1<<24) &&
91 (pf.greenMax << pf.greenShift) < (1<<24) &&
92 (pf.blueMax << pf.blueShift) < (1<<24));
93
94 bool fitsInMS3Bytes = (pf.redShift > 7 &&
95 pf.greenShift > 7 &&
96 pf.blueShift > 7);
97
98 if ((fitsInLS3Bytes && !pf.bigEndian) ||
99 (fitsInMS3Bytes && pf.bigEndian))
100 {
101 wroteAll = zrleEncode24A(r, mos, &zos, imageBuf, maxLen, actual, ig);
102 }
103 else if ((fitsInLS3Bytes && pf.bigEndian) ||
104 (fitsInMS3Bytes && !pf.bigEndian))
105 {
106 wroteAll = zrleEncode24B(r, mos, &zos, imageBuf, maxLen, actual, ig);
107 }
108 else
109 {
110 wroteAll = zrleEncode32(r, mos, &zos, imageBuf, maxLen, actual, ig);
111 }
112 break;
113 }
114 }
115
116 writer->startRect(*actual, encodingZRLE);
117 rdr::OutStream* os = writer->getOutStream();
118 os->writeU32(mos->length());
119 os->writeBytes(mos->data(), mos->length());
120 writer->endRect();
121 return wroteAll;
122}