blob: 086afcb59d222f016deb5223228d28cd9656e854 [file] [log] [blame]
Peter Åstrand462753d2004-11-16 15:23:25 +00001/* Copyright (C) 2000-2003 Constantin Kaplinsky. All Rights Reserved.
Peter Åstrandb04748d2004-12-07 14:13:51 +00002 * Copyright (C) 2004 Peter Astrand, Cendio AB. All Rights Reserved.
Peter Åstrand462753d2004-11-16 15:23:25 +00003 *
4 * This is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; either version 2 of the License, or
7 * (at your option) any later version.
8 *
9 * This software is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this software; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
17 * USA.
18 */
19#include <rfb/CMsgReader.h>
20#include <rfb/CMsgHandler.h>
21#include <rfb/TightDecoder.h>
Peter Åstranda6bb7702004-12-07 08:22:42 +000022#include <stdio.h> /* jpeglib.h needs FILE */
Peter Åstrandb4a23162004-12-07 13:52:13 +000023extern "C" {
Peter Åstranda6bb7702004-12-07 08:22:42 +000024#include <jpeglib.h>
Peter Åstrandb4a23162004-12-07 13:52:13 +000025}
Peter Åstrand462753d2004-11-16 15:23:25 +000026
27using namespace rfb;
28
Peter Åstrand0af24d42004-12-09 20:01:00 +000029#define RGB_TO_PIXEL(r,g,b) \
30 (((PIXEL_T)(r) & myFormat.redMax) << myFormat.redShift | \
31 ((PIXEL_T)(g) & myFormat.greenMax) << myFormat.greenShift | \
32 ((PIXEL_T)(b) & myFormat.blueMax) << myFormat.blueShift)
33
34#define RGB24_TO_PIXEL(r,g,b) \
Peter Åstranda6bb7702004-12-07 08:22:42 +000035 ((((PIXEL_T)(r) & 0xFF) * myFormat.redMax + 127) / 255 \
36 << myFormat.redShift | \
37 (((PIXEL_T)(g) & 0xFF) * myFormat.greenMax + 127) / 255 \
38 << myFormat.greenShift | \
39 (((PIXEL_T)(b) & 0xFF) * myFormat.blueMax + 127) / 255 \
40 << myFormat.blueShift)
41
Peter Åstrand1154ebf2004-12-12 13:14:00 +000042#define RGB24_TO_PIXEL32(r,g,b) \
43 (((rdr::U32)(r) & 0xFF) << myFormat.redShift | \
44 ((rdr::U32)(g) & 0xFF) << myFormat.greenShift | \
45 ((rdr::U32)(b) & 0xFF) << myFormat.blueShift)
46
Peter Åstranda6bb7702004-12-07 08:22:42 +000047#define TIGHT_MAX_WIDTH 2048
48
49static void JpegSetSrcManager(j_decompress_ptr cinfo, char *compressedData,
50 int compressedLen);
51static bool jpegError;
52
Peter Åstrand462753d2004-11-16 15:23:25 +000053#define EXTRA_ARGS CMsgHandler* handler
54#define FILL_RECT(r, p) handler->fillRect(r, p)
55#define IMAGE_RECT(r, p) handler->imageRect(r, p)
56#define BPP 8
57#include <rfb/tightDecode.h>
58#undef BPP
59#define BPP 16
60#include <rfb/tightDecode.h>
61#undef BPP
62#define BPP 32
63#include <rfb/tightDecode.h>
64#undef BPP
65
66Decoder* TightDecoder::create(CMsgReader* reader)
67{
68 return new TightDecoder(reader);
69}
70
71TightDecoder::TightDecoder(CMsgReader* reader_) : reader(reader_)
72{
73}
74
75TightDecoder::~TightDecoder()
76{
77}
78
79void TightDecoder::readRect(const Rect& r, CMsgHandler* handler)
80{
81 rdr::InStream* is = reader->getInStream();
Peter Åstranda6bb7702004-12-07 08:22:42 +000082 /* Uncompressed RGB24 JPEG data, before translated, can be up to 3
83 times larger, if VNC bpp is 8. */
84 rdr::U8* buf = reader->getImageBuf(r.area()*3);
Peter Åstrand462753d2004-11-16 15:23:25 +000085 switch (reader->bpp()) {
86 case 8:
87 tightDecode8 (r, is, zis, (rdr::U8*) buf, handler); break;
88 case 16:
89 tightDecode16(r, is, zis, (rdr::U16*)buf, handler); break;
90 case 32:
91 tightDecode32(r, is, zis, (rdr::U32*)buf, handler); break;
92 }
93}
Peter Åstranda6bb7702004-12-07 08:22:42 +000094
95
96//
97// A "Source manager" for the JPEG library.
98//
99
100static struct jpeg_source_mgr jpegSrcManager;
101static JOCTET *jpegBufferPtr;
102static size_t jpegBufferLen;
103
104static void JpegInitSource(j_decompress_ptr cinfo);
105static boolean JpegFillInputBuffer(j_decompress_ptr cinfo);
106static void JpegSkipInputData(j_decompress_ptr cinfo, long num_bytes);
107static void JpegTermSource(j_decompress_ptr cinfo);
108
109static void
110JpegInitSource(j_decompress_ptr cinfo)
111{
112 jpegError = false;
113}
114
115static boolean
116JpegFillInputBuffer(j_decompress_ptr cinfo)
117{
118 jpegError = true;
119 jpegSrcManager.bytes_in_buffer = jpegBufferLen;
120 jpegSrcManager.next_input_byte = (JOCTET *)jpegBufferPtr;
121
122 return TRUE;
123}
124
125static void
126JpegSkipInputData(j_decompress_ptr cinfo, long num_bytes)
127{
128 if (num_bytes < 0 || (size_t)num_bytes > jpegSrcManager.bytes_in_buffer) {
129 jpegError = true;
130 jpegSrcManager.bytes_in_buffer = jpegBufferLen;
131 jpegSrcManager.next_input_byte = (JOCTET *)jpegBufferPtr;
132 } else {
133 jpegSrcManager.next_input_byte += (size_t) num_bytes;
134 jpegSrcManager.bytes_in_buffer -= (size_t) num_bytes;
135 }
136}
137
138static void
139JpegTermSource(j_decompress_ptr cinfo)
140{
141 /* No work necessary here. */
142}
143
144static void
145JpegSetSrcManager(j_decompress_ptr cinfo, char *compressedData, int compressedLen)
146{
147 jpegBufferPtr = (JOCTET *)compressedData;
148 jpegBufferLen = (size_t)compressedLen;
149
150 jpegSrcManager.init_source = JpegInitSource;
151 jpegSrcManager.fill_input_buffer = JpegFillInputBuffer;
152 jpegSrcManager.skip_input_data = JpegSkipInputData;
153 jpegSrcManager.resync_to_restart = jpeg_resync_to_restart;
154 jpegSrcManager.term_source = JpegTermSource;
155 jpegSrcManager.next_input_byte = jpegBufferPtr;
156 jpegSrcManager.bytes_in_buffer = jpegBufferLen;
157
158 cinfo->src = &jpegSrcManager;
159}