1
/* Copyright (C) 2001-2006 Artifex Software, Inc.
4
This software is provided AS-IS with no warranty, either express or
7
This software is distributed under license and may not be copied, modified
8
or distributed except as expressly authorized under the terms of that
9
license. Refer to licensing information at http://www.artifex.com/
10
or contact Artifex Software, Inc., 7 Mt. Lassen Drive - Suite A-134,
11
San Rafael, CA 94903, U.S.A., +1(415)492-9861, for further information.
14
/* $Id: sdctd.c 8250 2007-09-25 13:31:24Z giles $ */
15
/* DCT decoding filter stream */
26
/* ------ DCTDecode ------ */
28
/* JPEG source manager procedures */
30
dctd_init_source(j_decompress_ptr dinfo)
33
static const JOCTET fake_eoi[2] =
36
dctd_fill_input_buffer(j_decompress_ptr dinfo)
38
jpeg_decompress_data *jddp =
39
(jpeg_decompress_data *) ((char *)dinfo -
40
offset_of(jpeg_decompress_data, dinfo));
43
return FALSE; /* normal case: suspend processing */
44
/* Reached end of source data without finding EOI */
45
WARNMS(dinfo, JWRN_JPEG_EOF);
46
/* Insert a fake EOI marker */
47
dinfo->src->next_input_byte = fake_eoi;
48
dinfo->src->bytes_in_buffer = 2;
49
jddp->faked_eoi = true; /* so process routine doesn't use next_input_byte */
53
dctd_skip_input_data(j_decompress_ptr dinfo, long num_bytes)
55
struct jpeg_source_mgr *src = dinfo->src;
56
jpeg_decompress_data *jddp =
57
(jpeg_decompress_data *) ((char *)dinfo -
58
offset_of(jpeg_decompress_data, dinfo));
61
if (num_bytes > src->bytes_in_buffer) {
62
jddp->skip += num_bytes - src->bytes_in_buffer;
63
src->next_input_byte += src->bytes_in_buffer;
64
src->bytes_in_buffer = 0;
67
src->next_input_byte += num_bytes;
68
src->bytes_in_buffer -= num_bytes;
72
dctd_term_source(j_decompress_ptr dinfo)
76
/* Set the defaults for the DCTDecode filter. */
78
s_DCTD_set_defaults(stream_state * st)
80
s_DCT_set_defaults(st);
83
/* Initialize DCTDecode filter */
85
s_DCTD_init(stream_state * st)
87
stream_DCT_state *const ss = (stream_DCT_state *) st;
88
struct jpeg_source_mgr *src = &ss->data.decompress->source;
90
src->init_source = dctd_init_source;
91
src->fill_input_buffer = dctd_fill_input_buffer;
92
src->skip_input_data = dctd_skip_input_data;
93
src->term_source = dctd_term_source;
94
src->resync_to_restart = jpeg_resync_to_restart; /* use default method */
95
ss->data.common->memory = ss->jpeg_memory;
96
ss->data.decompress->dinfo.src = src;
97
ss->data.decompress->skip = 0;
98
ss->data.decompress->input_eod = false;
99
ss->data.decompress->faked_eoi = false;
104
/* Process a buffer */
106
s_DCTD_process(stream_state * st, stream_cursor_read * pr,
107
stream_cursor_write * pw, bool last)
109
stream_DCT_state *const ss = (stream_DCT_state *) st;
110
jpeg_decompress_data *jddp = ss->data.decompress;
111
struct jpeg_source_mgr *src = jddp->dinfo.src;
114
if_debug3('w', "[wdd]process avail=%u, skip=%u, last=%d\n",
115
(uint) (pr->limit - pr->ptr), (uint) jddp->skip, last);
116
if (jddp->skip != 0) {
117
long avail = pr->limit - pr->ptr;
119
if (avail < jddp->skip) {
123
return 0; /* need more data */
124
jddp->skip = 0; /* don't skip past input EOD */
126
pr->ptr += jddp->skip;
129
src->next_input_byte = pr->ptr + 1;
130
src->bytes_in_buffer = pr->limit - pr->ptr;
131
jddp->input_eod = last;
133
case 0: /* not initialized yet */
135
* Adobe implementations seem to ignore leading garbage bytes,
136
* even though neither the standard nor Adobe's own
137
* documentation mention this.
139
while (pr->ptr < pr->limit && pr->ptr[1] != 0xff)
141
if (pr->ptr == pr->limit)
143
src->next_input_byte = pr->ptr + 1;
144
src->bytes_in_buffer = pr->limit - pr->ptr;
147
case 1: /* reading header markers */
148
if ((code = gs_jpeg_read_header(ss, TRUE)) < 0)
151
(jddp->faked_eoi ? pr->limit : src->next_input_byte - 1);
155
/*case JPEG_HEADER_OK: */
157
/* If we have a ColorTransform parameter, and it's not
158
* overridden by an Adobe marker in the data, set colorspace.
160
if (ss->ColorTransform >= 0 &&
161
!jddp->dinfo.saw_Adobe_marker) {
162
switch (jddp->dinfo.num_components) {
164
jddp->dinfo.jpeg_color_space =
165
(ss->ColorTransform ? JCS_YCbCr : JCS_RGB);
166
/* out_color_space will default to JCS_RGB */
169
jddp->dinfo.jpeg_color_space =
170
(ss->ColorTransform ? JCS_YCCK : JCS_CMYK);
171
/* out_color_space will default to JCS_CMYK */
177
case 2: /* start_decompress */
178
if ((code = gs_jpeg_start_decompress(ss)) < 0)
181
(jddp->faked_eoi ? pr->limit : src->next_input_byte - 1);
185
jddp->dinfo.output_width * jddp->dinfo.output_components;
186
if_debug4('w', "[wdd]width=%u, components=%d, scan_line_size=%u, min_out_size=%u\n",
187
jddp->dinfo.output_width,
188
jddp->dinfo.output_components,
189
ss->scan_line_size, jddp->template.min_out_size);
190
if (ss->scan_line_size > (uint) jddp->template.min_out_size) {
191
/* Create a spare buffer for oversize scanline */
192
jddp->scanline_buffer =
193
gs_alloc_bytes_immovable(gs_memory_stable(jddp->memory),
195
"s_DCTD_process(scanline_buffer)");
196
if (jddp->scanline_buffer == NULL)
199
jddp->bytes_in_scanline = 0;
202
case 3: /* reading data */
204
if (jddp->bytes_in_scanline != 0) {
205
uint avail = pw->limit - pw->ptr;
206
uint tomove = min(jddp->bytes_in_scanline,
209
if_debug2('w', "[wdd]moving %u/%u\n",
211
memcpy(pw->ptr + 1, jddp->scanline_buffer +
212
(ss->scan_line_size - jddp->bytes_in_scanline),
215
jddp->bytes_in_scanline -= tomove;
216
if (jddp->bytes_in_scanline != 0)
217
return 1; /* need more room */
219
while (jddp->dinfo.output_height > jddp->dinfo.output_scanline) {
223
if (jddp->scanline_buffer != NULL)
224
samples = jddp->scanline_buffer;
226
if ((uint) (pw->limit - pw->ptr) < ss->scan_line_size)
227
return 1; /* need more room */
228
samples = pw->ptr + 1;
230
read = gs_jpeg_read_scanlines(ss, &samples, 1);
233
if_debug3('w', "[wdd]read returns %d, used=%u, faked_eoi=%d\n",
235
(uint) (src->next_input_byte - 1 - pr->ptr),
236
(int)jddp->faked_eoi);
238
(jddp->faked_eoi ? pr->limit : src->next_input_byte - 1);
240
return 0; /* need more data */
241
if (jddp->scanline_buffer != NULL) {
242
jddp->bytes_in_scanline = ss->scan_line_size;
245
pw->ptr += ss->scan_line_size;
249
case 4: /* end of image; scan for EOI */
250
if ((code = gs_jpeg_finish_decompress(ss)) < 0)
253
(jddp->faked_eoi ? pr->limit : src->next_input_byte - 1);
258
case 5: /* we are DONE */
261
/* Default case can't happen.... */
265
/* Release the stream */
267
s_DCTD_release(stream_state * st)
269
stream_DCT_state *const ss = (stream_DCT_state *) st;
272
if (ss->data.decompress->scanline_buffer != NULL)
273
gs_free_object(gs_memory_stable(ss->data.common->memory),
274
ss->data.decompress->scanline_buffer,
275
"s_DCTD_release(scanline_buffer)");
276
gs_free_object(ss->data.common->memory, ss->data.decompress,
278
/* Switch the template pointer back in case we still need it. */
279
st->template = &s_DCTD_template;
282
/* Stream template */
283
const stream_template s_DCTD_template =
284
{&st_DCT_state, s_DCTD_init, s_DCTD_process, 2000, 4000, s_DCTD_release,