summaryrefslogtreecommitdiff
path: root/src/pdflib/pdflib/p_jpeg.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/pdflib/pdflib/p_jpeg.c')
-rw-r--r--src/pdflib/pdflib/p_jpeg.c1560
1 files changed, 1560 insertions, 0 deletions
diff --git a/src/pdflib/pdflib/p_jpeg.c b/src/pdflib/pdflib/p_jpeg.c
new file mode 100644
index 0000000..467f282
--- /dev/null
+++ b/src/pdflib/pdflib/p_jpeg.c
@@ -0,0 +1,1560 @@
+/*---------------------------------------------------------------------------*
+ | PDFlib - A library for generating PDF on the fly |
+ +---------------------------------------------------------------------------+
+ | Copyright (c) 1997-2006 Thomas Merz and PDFlib GmbH. All rights reserved. |
+ +---------------------------------------------------------------------------+
+ | |
+ | This software is subject to the PDFlib license. It is NOT in the |
+ | public domain. Extended versions and commercial licenses are |
+ | available, please check http://www.pdflib.com. |
+ | |
+ *---------------------------------------------------------------------------*/
+
+/* $Id: p_jpeg.c,v 1.1 2008/10/17 06:11:49 scuri Exp $
+ *
+ * JPEG processing for PDFlib
+ *
+ */
+
+#include "p_intern.h"
+#include "p_color.h"
+#include "p_image.h"
+
+#ifndef PDF_JPEG_SUPPORTED
+
+pdc_bool
+pdf_is_JPEG_file(PDF *p, pdc_file *fp)
+{
+ (void) p;
+ (void) fp;
+
+ return pdc_false;
+}
+
+int
+pdf_process_JPEG_data(
+ PDF *p,
+ int imageslot)
+{
+ pdf_image *image = &p->images[imageslot];
+
+ pdc_set_errmsg(p->pdc, PDF_E_UNSUPP_IMAGE, "JPEG", 0, 0, 0);
+
+ return -1;
+}
+
+/* CDPDF - added missing function */
+void
+pdf_cleanup_jpeg(PDF *p, pdf_image *image)
+{
+ (void) p;
+ (void) image;
+}
+
+#else
+
+#include "jinclude.h"
+#include "jpeglib.h"
+
+/*
+ * The following enum is stolen from the IJG JPEG library
+ * Comments added by tm.
+ * This table contains far too many names since PDFlib
+ * is rather simple-minded about markers.
+ */
+
+typedef enum { /* JPEG marker codes */
+ M_SOF0 = 0xc0, /* baseline DCT */
+ M_SOF1 = 0xc1, /* extended sequential DCT */
+ M_SOF2 = 0xc2, /* progressive DCT */
+ M_SOF3 = 0xc3, /* lossless (sequential) */
+
+ M_SOF5 = 0xc5, /* differential sequential DCT */
+ M_SOF6 = 0xc6, /* differential progressive DCT */
+ M_SOF7 = 0xc7, /* differential lossless */
+
+ M_JPG = 0xc8, /* JPEG extensions */
+ M_SOF9 = 0xc9, /* extended sequential DCT */
+ M_SOF10 = 0xca, /* progressive DCT */
+ M_SOF11 = 0xcb, /* lossless (sequential) */
+
+ M_SOF13 = 0xcd, /* differential sequential DCT */
+ M_SOF14 = 0xce, /* differential progressive DCT */
+ M_SOF15 = 0xcf, /* differential lossless */
+
+ M_DHT = 0xc4, /* define Huffman tables */
+
+ M_DAC = 0xcc, /* define arithmetic conditioning table */
+
+ M_RST0 = 0xd0, /* restart */
+ M_RST1 = 0xd1, /* restart */
+ M_RST2 = 0xd2, /* restart */
+ M_RST3 = 0xd3, /* restart */
+ M_RST4 = 0xd4, /* restart */
+ M_RST5 = 0xd5, /* restart */
+ M_RST6 = 0xd6, /* restart */
+ M_RST7 = 0xd7, /* restart */
+
+ M_SOI = 0xd8, /* start of image */
+ M_EOI = 0xd9, /* end of image */
+ M_SOS = 0xda, /* start of scan */
+ M_DQT = 0xdb, /* define quantization tables */
+ M_DNL = 0xdc, /* define number of lines */
+ M_DRI = 0xdd, /* define restart interval */
+ M_DHP = 0xde, /* define hierarchical progression */
+ M_EXP = 0xdf, /* expand reference image(s) */
+
+ M_APP0 = 0xe0, /* application marker, used for JFIF */
+ M_APP1 = 0xe1, /* application marker, used for Exif */
+ M_APP2 = 0xe2, /* application marker, used for FlashPix*
+ * and ICC Profiles */
+ M_APP3 = 0xe3, /* application marker */
+ M_APP4 = 0xe4, /* application marker */
+ M_APP5 = 0xe5, /* application marker */
+ M_APP6 = 0xe6, /* application marker */
+ M_APP7 = 0xe7, /* application marker */
+ M_APP8 = 0xe8, /* application marker, used for SPIFF */
+ M_APP9 = 0xe9, /* application marker */
+ M_APP10 = 0xea, /* application marker */
+ M_APP11 = 0xeb, /* application marker */
+ M_APP12 = 0xec, /* application marker */
+ M_APP13 = 0xed, /* application marker, used by Photoshop*/
+ M_APP14 = 0xee, /* application marker, used by Adobe */
+ M_APP15 = 0xef, /* application marker */
+
+ M_JPG0 = 0xf0, /* reserved for JPEG extensions */
+ M_JPG13 = 0xfd, /* reserved for JPEG extensions */
+ M_COM = 0xfe, /* comment */
+
+ M_TEM = 0x01 /* temporary use */
+
+} JPEG_MARKER;
+
+#define JPEG_SEGLIST_CHUNKSIZE 64
+#define JPEG_MARKER_LEN 2
+#define JPEG_LENGTH_LEN 2
+#define JPEG_BUFSIZE 0xFFFF
+
+struct pdf_jpeg_segment_s
+{
+ long pos; /* position of segment */
+ size_t length; /* length of segement in byte */
+};
+
+static void
+pdf_register_JPEG_segment(PDF *p, pdf_image *image, long pos, size_t length)
+{
+ static const char fn[] = "pdf_register_JPEG_segment";
+ pdf_jpeg_info *jpeg = &image->info.jpeg;
+ size_t len;
+
+ pdc_logg_cond(p->pdc, 5, trc_image,
+ "\t\tKeep segment, position = 0x%lX, length = 0x%lX(%ld)\n",
+ pos, length, length);
+
+ while(length > 0)
+ {
+ len = length;
+ if (len > JPEG_BUFSIZE)
+ len = JPEG_BUFSIZE;
+
+ if (jpeg->number >= jpeg->capacity)
+ {
+ if (jpeg->capacity == 0)
+ {
+ jpeg->capacity = JPEG_SEGLIST_CHUNKSIZE;
+ jpeg->seglist = (pdf_jpeg_segment *) pdc_malloc(p->pdc,
+ jpeg->capacity * sizeof(pdf_jpeg_segment), fn);
+ }
+ else
+ {
+ jpeg->capacity += JPEG_SEGLIST_CHUNKSIZE;
+ jpeg->seglist = (pdf_jpeg_segment *) pdc_realloc(p->pdc,
+ jpeg->seglist, jpeg->capacity* sizeof(pdf_jpeg_segment), fn);
+ }
+ }
+ jpeg->seglist[jpeg->number].pos = pos;
+ jpeg->seglist[jpeg->number].length = len;
+ jpeg->number++;
+
+ length -= len;
+ pos += len;
+ }
+}
+
+static void
+pdf_data_source_JPEG_init(PDF *p, PDF_data_source *src)
+{
+ static const char fn[] = "pdf_data_source_JPEG_init";
+ pdf_image *image;
+ pdf_jpeg_info *jpeg;
+
+ image = (pdf_image *) src->private_data;
+ jpeg = &image->info.jpeg;
+
+ jpeg->capacity = jpeg->number;
+ jpeg->number = 0;
+
+ src->buffer_start = (pdc_byte *) pdc_malloc(p->pdc, JPEG_BUFSIZE, fn);
+ src->buffer_length = JPEG_BUFSIZE;
+}
+
+static pdc_bool
+pdf_data_source_JPEG_fill(PDF *p, PDF_data_source *src)
+{
+ pdf_image *image;
+ pdf_jpeg_info *jpeg;
+ size_t length;
+ long pos;
+
+ (void) p;
+
+ image = (pdf_image *) src->private_data;
+ jpeg = &image->info.jpeg;
+
+ if (jpeg->number < jpeg->capacity)
+ {
+ pos = jpeg->seglist[jpeg->number].pos;
+ length = jpeg->seglist[jpeg->number].length;
+ jpeg->number++;
+
+ pdc_fseek(image->fp, pos, SEEK_SET);
+ src->next_byte = src->buffer_start;
+ src->bytes_available =
+ pdc_fread(src->buffer_start, 1, length, image->fp);
+ }
+ else
+ {
+ src->bytes_available = 0;
+ }
+
+ if (src->bytes_available == 0)
+ return pdc_false;
+ else
+ return pdc_true;
+}
+
+static void
+pdf_data_source_JPEG_terminate(PDF *p, PDF_data_source *src)
+{
+ pdc_free(p->pdc, (void *) src->buffer_start);
+}
+
+/**********************************************************************
+ *
+ * Decompression data source routines for the case of
+ * reading JPEG data from a PDFlib virtual file in
+ * JPEG library - analogous to ../libs/jpeg/jdatasrc.c
+ *
+ **********************************************************************/
+
+typedef struct
+{
+ struct jpeg_source_mgr pub; /* public fields */
+ pdc_file *infile;
+ PDF *p; /* required for logging only */
+ pdf_image *image; /* required for access to the filename */
+}
+pdf_source_mgr;
+
+typedef pdf_source_mgr * pdf_src_ptr;
+
+static void
+pdf_init_JPEG_source (j_decompress_ptr cinfo)
+{
+ (void) cinfo;
+}
+
+static boolean
+pdf_fill_JPEG_input_buffer (j_decompress_ptr cinfo)
+{
+ pdf_src_ptr src = (pdf_src_ptr) cinfo->src;
+ JOCTET *buffer;
+ size_t nbytes;
+
+ buffer = (JOCTET *) pdc_freadall(src->infile, &nbytes, NULL);
+
+ src->pub.next_input_byte = buffer;
+ src->pub.bytes_in_buffer = nbytes;
+
+ return TRUE;
+}
+
+static void
+pdf_skip_JPEG_input_data (j_decompress_ptr cinfo, long num_bytes)
+{
+ pdf_src_ptr src = (pdf_src_ptr) cinfo->src;
+
+ src->pub.next_input_byte += (size_t) num_bytes;
+ src->pub.bytes_in_buffer -= (size_t) num_bytes;
+}
+
+static void
+pdf_term_JPEG_source (j_decompress_ptr cinfo)
+{
+ (void) cinfo;
+}
+
+static void
+pdf_jpeg_pdcread_src(j_decompress_ptr cinfo,
+ PDF *p, pdc_file *infile, pdf_image *image)
+{
+ pdf_src_ptr src;
+
+ cinfo->src = (struct jpeg_source_mgr *)
+ (*cinfo->mem->alloc_small) ((j_common_ptr) cinfo, JPOOL_IMAGE,
+ SIZEOF(pdf_source_mgr));
+
+ src = (pdf_src_ptr) cinfo->src;
+ src->pub.init_source = pdf_init_JPEG_source;
+ src->pub.fill_input_buffer = pdf_fill_JPEG_input_buffer;
+ src->pub.skip_input_data = pdf_skip_JPEG_input_data;
+ src->pub.resync_to_restart = jpeg_resync_to_restart;
+ src->pub.term_source = pdf_term_JPEG_source;
+ src->infile = infile;
+ src->p = p;
+ src->image = image;
+ src->pub.bytes_in_buffer = 0;
+ src->pub.next_input_byte = NULL;
+}
+
+/**********************************************************************
+ *
+ * Compression data destination routines for the case of
+ * emitting JPEG data to a open PDFlib PDF file in
+ * JPEG library - analogous to ../libs/jpeg/jdatadst.c
+ *
+ **********************************************************************/
+
+typedef struct
+{
+ struct jpeg_destination_mgr pub;
+ PDF *p;
+ pdf_image *image; /* required for access to the filename */
+ JOCTET *buffer;
+}
+pdf_destination_mgr;
+
+typedef pdf_destination_mgr * pdf_dest_ptr;
+
+#define OUTPUT_BUF_SIZE 4096
+
+static void
+pdf_init_JPEG_destination (j_compress_ptr cinfo)
+{
+ pdf_dest_ptr dest = (pdf_dest_ptr) cinfo->dest;
+
+ dest->buffer = (JOCTET *)
+ (*cinfo->mem->alloc_small) ((j_common_ptr) cinfo, JPOOL_IMAGE,
+ OUTPUT_BUF_SIZE * SIZEOF(JOCTET));
+
+ dest->pub.next_output_byte = dest->buffer;
+ dest->pub.free_in_buffer = OUTPUT_BUF_SIZE;
+}
+
+static boolean
+pdf_empty_JPEG_output_buffer (j_compress_ptr cinfo)
+{
+ pdf_dest_ptr dest = (pdf_dest_ptr) cinfo->dest;
+
+ pdc_write(dest->p->out, dest->buffer, OUTPUT_BUF_SIZE);
+
+ dest->pub.next_output_byte = dest->buffer;
+ dest->pub.free_in_buffer = OUTPUT_BUF_SIZE;
+
+ return TRUE;
+}
+
+static void
+pdf_term_JPEG_destination (j_compress_ptr cinfo)
+{
+ pdf_dest_ptr dest = (pdf_dest_ptr) cinfo->dest;
+ size_t datacount = OUTPUT_BUF_SIZE - dest->pub.free_in_buffer;
+
+ if (datacount)
+ pdc_write(dest->p->out, dest->buffer, datacount);
+}
+
+static void
+pdf_jpeg_pdcwrite_dest(j_compress_ptr cinfo, PDF *p, pdf_image *image)
+{
+ pdf_dest_ptr dest;
+
+ cinfo->dest = (struct jpeg_destination_mgr *)
+ (*cinfo->mem->alloc_small) ((j_common_ptr) cinfo, JPOOL_IMAGE,
+ SIZEOF(pdf_destination_mgr));
+
+ dest = (pdf_dest_ptr) cinfo->dest;
+ dest->pub.init_destination = pdf_init_JPEG_destination;
+ dest->pub.empty_output_buffer = pdf_empty_JPEG_output_buffer;
+ dest->pub.term_destination = pdf_term_JPEG_destination;
+ dest->p = p;
+ dest->image = image;
+}
+
+/**********************************************************************/
+
+#define PDF_JMSG_LENGTH_MAX 200
+
+/*
+ * Private replacements for libjpeg's error message function.
+ * They serve two purposes:
+ * - avoid libjpeg writing to stderr
+ * - write the message to the log file if logging is enabled
+ * One function is required for each source and destination.
+ */
+
+static void
+pdf_output_message_src(j_common_ptr cinfo)
+{
+ char buffer[PDF_JMSG_LENGTH_MAX];
+
+ /* we use this method only for decompression objects */
+ j_decompress_ptr cinfo2 = (j_decompress_ptr) cinfo;
+ pdf_source_mgr *src = (pdf_source_mgr *) cinfo2->src;
+
+ if (!pdc_logg_is_enabled(src->p->pdc, 5, trc_image))
+ return;
+
+ /* Create the message */
+ (*cinfo->err->format_message) (cinfo, buffer);
+
+ pdc_logg(src->p->pdc, "\tlibjpeg src: %s\n", buffer);
+}
+
+static void
+pdf_output_message_dst(j_common_ptr cinfo)
+{
+ char buffer[PDF_JMSG_LENGTH_MAX];
+
+ /* we use this method only for compression objects */
+ j_compress_ptr cinfo2 = (j_compress_ptr) cinfo;
+ pdf_destination_mgr *dst = (pdf_destination_mgr *) cinfo2->dest;
+
+ if (!pdc_logg_is_enabled(dst->p->pdc, 5, trc_image))
+ return;
+
+ /* Create the message */
+ (*cinfo->err->format_message) (cinfo, buffer);
+
+ pdc_logg(dst->p->pdc, "\tlibjpeg dst: %s\n", buffer);
+}
+
+/*
+ * Private replacements for libjpeg's error_exit function.
+ * They serve three purposes:
+ * - avoid libjpeg exiting
+ * - write a message to the log file if logging is enabled
+ * - return control from libjpeg by raising an exception
+ * One function is required for each source and destination.
+ */
+
+static void
+pdf_error_exit_src(j_common_ptr cinfo)
+{
+ PDF *p;
+ pdf_image *image;
+ char buffer[PDF_JMSG_LENGTH_MAX];
+
+ /* we use this method only for decompression objects */
+ j_decompress_ptr cinfo2 = (j_decompress_ptr) cinfo;
+ pdf_source_mgr *src = (pdf_source_mgr *) cinfo2->src;
+
+ p = src->p;
+ image = src->image;
+
+ (*cinfo->err->output_message) (cinfo);
+ (*cinfo->err->format_message) (cinfo, buffer);
+
+ if (pdc_logg_is_enabled(p->pdc, 5, trc_image))
+ pdc_logg(p->pdc, "\tlibjpeg (src) called error_exit routine\n");
+
+ /* clean up libjpeg */
+ jpeg_destroy(cinfo);
+
+ pdc_error(p->pdc, PDF_E_JPEG_TRANSCODE,
+ pdc_errprintf(p->pdc, "%.*s", PDC_ERR_MAXSTRLEN, image->filename),
+ buffer, 0, 0);
+}
+
+static void
+pdf_error_exit_dst(j_common_ptr cinfo)
+{
+ PDF *p;
+ pdf_image *image;
+ char buffer[PDF_JMSG_LENGTH_MAX];
+
+ /* we use this method only for compression objects */
+ j_compress_ptr cinfo2 = (j_compress_ptr) cinfo;
+ pdf_destination_mgr *dst = (pdf_destination_mgr *) cinfo2->dest;
+
+ p = dst->p;
+ image = dst->image;
+
+ (*cinfo->err->output_message) (cinfo);
+ (*cinfo->err->format_message) (cinfo, buffer);
+
+ if (pdc_logg_is_enabled(p->pdc, 5, trc_image))
+ pdc_logg(p->pdc, "\tlibjpeg (dst) called error_exit routine\n");
+
+ /* clean up libjpeg */
+ jpeg_destroy(cinfo);
+
+ pdc_error(p->pdc, PDF_E_JPEG_TRANSCODE,
+ pdc_errprintf(p->pdc, "%.*s", PDC_ERR_MAXSTRLEN, image->filename),
+ buffer, 0, 0);
+}
+
+static pdc_bool
+pdf_data_source_JPEG_fill_transcode(PDF *p, PDF_data_source *src)
+{
+ pdf_image *image = (pdf_image *) src->private_data;
+ pdc_bool logg5 = pdc_logg_is_enabled(p->pdc, 5, trc_image);
+
+ struct jpeg_decompress_struct srcinfo;
+ struct jpeg_compress_struct dstinfo;
+ jvirt_barray_ptr * src_coef_arrays;
+ struct jpeg_error_mgr jsrcerr, jdsterr;
+
+ /* ---------- Setup for decompression ---------- */
+ /* Initialize the JPEG decompression object with default error handling. */
+ srcinfo.err = jpeg_std_error(&jsrcerr);
+
+ /* Hook up our own message handler for logging */
+ srcinfo.err->output_message = pdf_output_message_src;
+
+ /* Hook up our own fatal error handler */
+ srcinfo.err->error_exit = pdf_error_exit_src;
+
+ /* Extended libjpeg tracing if PDFlib logging is enabled */
+ if (logg5)
+ srcinfo.err->trace_level = 5;
+
+ jpeg_create_decompress(&srcinfo);
+
+ /* Specify data source for decompression analogous to jpeg_stdio_src */
+ pdf_jpeg_pdcread_src(&srcinfo, p, image->fp, image);
+
+ /* ---------- Setup for compression ---------- */
+ /* Initialize the JPEG compression object with default error handling. */
+ dstinfo.err = jpeg_std_error(&jdsterr);
+
+ /* Hook up our own message handler for logging */
+ dstinfo.err->output_message = pdf_output_message_dst;
+
+ /* Hook up our own fatal error handler */
+ dstinfo.err->error_exit = pdf_error_exit_dst;
+
+ /* Extended libjpeg tracing if PDFlib logging is enabled */
+ if (logg5)
+ dstinfo.err->trace_level = 5;
+
+ jpeg_create_compress(&dstinfo);
+
+ PDC_TRY(p->pdc)
+ {
+ /* ---------- start transcoding ---------- */
+
+ /* Read file header */
+ if (jpeg_read_header(&srcinfo, TRUE) != JPEG_HEADER_OK)
+ {
+ if (logg5)
+ pdc_logg(p->pdc, "\tlibjpeg couldn't read header\n");
+
+ pdc_error(p->pdc, PDF_E_IMAGE_CORRUPT, "JPEG",
+ pdc_errprintf(p->pdc, "%.*s", PDC_ERR_MAXSTRLEN,
+ image->filename), 0, 0);
+ }
+
+ /* Read source file as DCT coefficients */
+ src_coef_arrays = jpeg_read_coefficients(&srcinfo);
+ if (src_coef_arrays == NULL)
+ {
+ if (logg5)
+ pdc_logg(p->pdc, "\tlibjpeg couldn't read coefficients\n");
+
+ pdc_error(p->pdc, PDF_E_IMAGE_CORRUPT, "JPEG",
+ pdc_errprintf(p->pdc, "%.*s", PDC_ERR_MAXSTRLEN,
+ image->filename), 0, 0);
+ }
+
+ /* Initialize destination compression parameters from source values */
+ jpeg_copy_critical_parameters(&srcinfo, &dstinfo);
+
+ /* Specify data destination for compression analogous to
+ * jpeg_stdio_dest
+ */
+ pdf_jpeg_pdcwrite_dest(&dstinfo, p, image);
+
+ /* Start compressor (note no image data is actually written here) */
+ jpeg_write_coefficients(&dstinfo, src_coef_arrays);
+
+ /* Finish compression */
+ /* DON'T change the order! */
+ jpeg_finish_compress(&dstinfo);
+ (void) jpeg_finish_decompress(&srcinfo);
+ }
+ PDC_CATCH(p->pdc)
+ {
+ image->corrupt = pdc_true;
+ }
+
+ /* Release memory */
+ jpeg_destroy_compress(&dstinfo);
+ jpeg_destroy_decompress(&srcinfo);
+
+ /* All done. Check for errors */
+ if (jsrcerr.num_warnings != 0 && logg5)
+ {
+ /*
+ * We don't really care about problems in the input since
+ * they will be fixed by transcoding. Log them, but don't throw an
+ * exception.
+ */
+ pdc_logg(p->pdc,
+ "\tlibjpeg total: %d corrupt data warning(s)\n",
+ jsrcerr.num_warnings);
+ }
+
+ if (jdsterr.num_warnings != 0)
+ {
+ char buffer[PDF_JMSG_LENGTH_MAX];
+
+ /*
+ * Errors in the output are rare, but fatal. Log them,
+ * and unconditionally throw an exception.
+ */
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tlibjpeg: %d warning(s) for output\n",
+ jdsterr.num_warnings);
+ }
+
+ (dstinfo.err->format_message) ((j_common_ptr) &dstinfo, buffer);
+ pdc_set_errmsg(p->pdc, PDF_E_JPEG_TRANSCODE,
+ pdf_get_image_filename(p, image), buffer, 0, 0);
+
+ image->corrupt = pdc_true;
+ }
+
+ return pdc_false;
+}
+
+static pdc_ushort
+get_ushort(pdc_file *fp)
+{
+ pdc_byte c[2];
+
+ c[0] = (pdc_byte) pdc_fgetc(fp);
+ c[1] = (pdc_byte) pdc_fgetc(fp);
+
+ return pdc_get_be_ushort(c);
+}
+
+#define CHECK_LENGTH 1024L
+
+pdc_bool
+pdf_is_JPEG_file(PDF *p, pdc_file *fp)
+{
+ long pos = 0L;
+ int c;
+ long start = (long) pdc_ftell(fp);
+ long check_length = start + CHECK_LENGTH;
+
+ pdc_logg_cond(p->pdc, 1, trc_image, "\tChecking image type JPEG...\n");
+
+#if !defined(MVS) || !defined(I370)
+ /* Tommy's special trick for Macintosh JPEGs: simply skip some */
+ /* hundred bytes at the beginning of the file! */
+ do
+ {
+ do /* skip if not FF */
+ {
+ c = pdc_fgetc(fp);
+ pos++;
+
+ }
+ while (!pdc_feof(fp) && c != 0xFF && pos < check_length);
+
+ if (pdc_feof(fp) || pos >= check_length)
+ {
+ pdc_fseek(fp, start, SEEK_SET);
+ return pdc_false;
+ }
+
+ do /* skip repeated FFs */
+ {
+ c = pdc_fgetc(fp);
+ pos++;
+ }
+ while (c == 0xFF && pos < check_length);
+
+ /* remember start position */
+ pos = (pdc_off_t1) pdc_ftell(fp);
+ if (pos < 0L || pos >= check_length)
+ {
+ pdc_fseek(fp, start, SEEK_SET);
+ return pdc_false;
+ }
+
+ pos -= JPEG_MARKER_LEN; /* subtract marker length */
+
+ if (c == M_SOI)
+ {
+ pdc_fseek(fp, pos, SEEK_SET);
+ break;
+ }
+ }
+ while (!pdc_feof(fp));
+#endif /* !MVS || !I370 */
+
+#define BOGUS_LENGTH 768
+ /* If we are that far from the start we consider the image as damaged if:
+ * - OJPEG-TIFF: it does not start at the alleged data offset
+ * - any other flavor: it has too much garbage at the beginning
+ */
+ if (pdc_feof(fp) || pos > (start ? start : BOGUS_LENGTH))
+ {
+ pdc_fseek(fp, start, SEEK_SET);
+ return pdc_false;
+ }
+
+ return pdc_true;
+}
+
+/* This function should be moved to p_color.c once it gets used by other
+ * image modules as well.
+ */
+
+static void
+pdf_log_colorspace(PDF *p, int slot)
+{
+ pdf_colorspace *cs;
+
+ if (slot < 0 || slot >= p->colorspaces_number)
+ {
+ pdc_logg(p->pdc, " Bad color space slot %d", slot);
+ }
+
+ cs = &p->colorspaces[slot];
+
+ switch (cs->type) {
+ case DeviceGray:
+ pdc_logg(p->pdc, "/DeviceGray");
+ break;
+
+ case DeviceRGB:
+ pdc_logg(p->pdc, "/DeviceRGB");
+ break;
+
+ case DeviceCMYK:
+ pdc_logg(p->pdc, "/DeviceCMYK");
+ break;
+
+
+ case Indexed:
+ pdc_logg(p->pdc, "/Indexed");
+ break;
+
+ case PatternCS:
+ pdc_logg(p->pdc, "/Pattern");
+ break;
+
+ default:
+ pdc_logg(p->pdc, "%d (unknown)", cs->type);
+ }
+}
+
+/* open JPEG image and analyze marker */
+int
+pdf_process_JPEG_data(
+ PDF *p,
+ int imageslot)
+{
+ int c, unit;
+ unsigned long length, len = 0, slen;
+#define APP_MAX 255
+ pdc_byte appstring[APP_MAX];
+ const char *filename = NULL;
+ pdc_bool ismem = pdc_false;
+ void *filebase = NULL;
+ size_t filelen;
+ pdf_image *image;
+ int transform = 0;
+ pdc_bool marker_found = pdc_false;
+ pdc_bool markers_done = pdc_false;
+ pdc_bool need_transcode = pdc_false;
+ pdc_bool logg5 = pdc_logg_is_enabled(p->pdc, 5, trc_image);
+ long pos = 0, endpos = 0;
+ long adobe_pos = 0, adobe_len = 0;
+ int errint = 0;
+ int errcode = 0;
+
+ image = &p->images[imageslot];
+ image->compression = pdf_comp_dct;
+ image->use_raw = pdc_true;
+ image->info.jpeg.virtfile = NULL;
+ image->info.jpeg.seglist = NULL;
+ image->info.jpeg.capacity = 0;
+ image->info.jpeg.number = 0;
+
+ need_transcode = !image->passthrough;
+
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tjpegoptimize = %s\n",
+ image->jpegoptimize ? "true" : "false");
+ if (need_transcode)
+ pdc_logg(p->pdc, "\ttranscoding...\n");
+ else
+ pdc_logg(p->pdc, "\ttranscoding disabled by passthrough option\n");
+ }
+
+ /* jpeg file not available */
+ if (image->reference != pdf_ref_direct)
+ {
+
+
+ image->in_use = pdc_true; /* mark slot as used */
+ pdf_put_image(p, imageslot, pdc_true, pdc_true);
+ return imageslot;
+ }
+
+ if (!pdc_file_isvirtual(image->fp))
+ {
+ /* read whole file */
+ filebase = (void *) pdc_freadall(image->fp, &filelen, &ismem);
+ if (filebase == NULL)
+ {
+ errcode = PDC_E_IO_READ;
+ goto PDF_JPEG_ERROR;
+ }
+ pdc_fclose(image->fp);
+
+ /* temporary memory */
+ pdc_insert_mem_tmp(p->pdc, filebase, 0, 0);
+
+ /* create virtual image file */
+ filename = "__jpeg__image__data__";
+ pdc__create_pvf(p->pdc, filename, filebase, filelen, "");
+ image->info.jpeg.virtfile = filename;
+
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tVirtual file created, "
+ "length = 0x%lX(%ld)\n", filelen, filelen);
+ }
+
+ /* open virtual file */
+ image->fp = pdc_fsearch_fopen(p->pdc, filename, NULL, "",
+ PDC_FILE_BINARY);
+ }
+
+ if (image->info.jpeg.jpegifoffset)
+ {
+ /* Just to be sure: if we were handed a OJPEG-compressed TIFF with
+ * an offset we let libjpeg transcode.
+ */
+ need_transcode = pdc_true;
+
+ if (logg5)
+ {
+ pdc_logg(p->pdc,
+ "\ttranscoding because of OJPEG-compressed TIFF\n");
+ pdc_logg(p->pdc,
+ "\tseeking to base offset 0x%lX(%ld) (TIFF with OJPEG)\n",
+ image->info.jpeg.jpegifoffset, image->info.jpeg.jpegifoffset);
+ }
+ pdc_fseek(image->fp, image->info.jpeg.jpegifoffset, SEEK_SET);
+ }
+
+ if (pdf_is_JPEG_file(p, image->fp) == pdc_false)
+ {
+ errcode = PDF_E_IMAGE_CORRUPT;
+ goto PDF_JPEG_ERROR;
+ }
+
+ /* JPEG marker loop */
+ while (1)
+ {
+ /* look for next JPEG Marker */
+ if (!markers_done)
+ {
+ do /* repeat if FF/00 */
+ {
+ do /* skip to FF */
+ {
+ if (pdc_feof(image->fp))
+ {
+ errcode = PDF_E_IMAGE_CORRUPT;
+ goto PDF_JPEG_ERROR;
+ }
+ c = pdc_fgetc(image->fp);
+ }
+ while (c != 0xFF);
+
+ do /* skip repeated FFs */
+ {
+ if (pdc_feof(image->fp))
+ {
+ errcode = PDF_E_IMAGE_CORRUPT;
+ goto PDF_JPEG_ERROR;
+ }
+ c = pdc_fgetc(image->fp);
+ }
+ while (c == 0xFF);
+ }
+ while (c == 0);
+
+ /* start of new segment */
+ pos = (pdc_off_t1) pdc_ftell(image->fp) - JPEG_MARKER_LEN;
+
+ /* skip garbage at the start of image data */
+ if (!marker_found && pos > 0)
+ {
+ if (logg5 && pos > (long) image->info.jpeg.jpegifoffset)
+ {
+ pdc_logg(p->pdc, "\t0x%lX(%ld) bytes garbage "
+ "at start of image\n", pos, pos);
+ }
+
+ /* we must create a new virtual file */
+ if (image->info.jpeg.virtfile == 0)
+ {
+ /* read whole file */
+ filebase = (void *) pdc_freadall(image->fp,
+ &filelen, &ismem);
+ if (filebase == NULL)
+ {
+ errcode = PDC_E_IO_READ;
+ goto PDF_JPEG_ERROR;
+ }
+
+ /* temporary memory */
+ pdc_insert_mem_tmp(p->pdc, filebase, 0, 0);
+
+ filename = "__jpeg__image__data__";
+ }
+ else
+ {
+ /* delete virtual file */
+ pdc__delete_pvf(p->pdc, image->info.jpeg.virtfile);
+ }
+
+ /* [re]create virtual file */
+ filelen -= pos;
+ memmove(filebase, (char *) filebase + pos, filelen);
+ pdc__create_pvf(p->pdc, filename, filebase, filelen, "");
+ image->info.jpeg.virtfile = filename;
+
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tVirtual file created, "
+ "length = 0x%lX(%ld)\n",
+ filelen, filelen);
+ }
+ /* [re]open virtual file */
+ pdc_fclose(image->fp);
+ image->fp = pdc_fsearch_fopen(p->pdc, filename, NULL, "",
+ PDC_FILE_BINARY);
+
+ /* restart with the cleaned file */
+ continue;
+ }
+ length = 0;
+ marker_found = pdc_true;
+ }
+ else
+ {
+ /* enforcing end of image */
+ pos = (pdc_off_t1) pdc_ftell(image->fp);
+ pdc_fseek(image->fp, 0L, SEEK_END);
+ endpos = (pdc_off_t1) pdc_ftell(image->fp) - JPEG_MARKER_LEN;
+ length = endpos - pos;
+ c = M_EOI;
+ }
+
+ /* analyzing JPEG Marker */
+ switch (c)
+ {
+ /* markers which are not supported in PDF 1.3 and above */
+ case M_SOF3:
+ case M_SOF5:
+ case M_SOF6:
+ case M_SOF7:
+ case M_SOF9:
+ case M_SOF11:
+ case M_SOF13:
+ case M_SOF14:
+ case M_SOF15:
+ {
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tMarker 0x%X(SOF%d) found - "
+ "not supported\n", c, c - M_SOF0);
+ }
+ errint = c;
+ errcode = PDF_E_JPEG_COMPRESSION;
+ }
+ goto PDF_JPEG_ERROR;
+
+ /* markers without any parameters */
+ case M_SOI:
+ case M_TEM:
+ case M_EOI:
+ case M_RST0:
+ case M_RST1:
+ case M_RST2:
+ case M_RST3:
+ case M_RST4:
+ case M_RST5:
+ case M_RST6:
+ case M_RST7:
+ {
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tMarker 0x%X", c);
+ if (c == M_EOI)
+ pdc_logg(p->pdc, "(EOI)");
+ pdc_logg(p->pdc, " found - no contents\n");
+ }
+ pdf_register_JPEG_segment(p, image, pos,
+ (size_t) (length + JPEG_MARKER_LEN));
+ }
+ break;
+
+ /* skip segment if jpegoptimize = true, otherwise keep */
+ case M_APP0:
+ case M_APP1:
+ case M_APP2:
+ case M_APP3:
+ case M_APP4:
+ case M_APP5:
+ case M_APP6:
+ case M_APP7:
+ case M_APP8:
+ case M_APP9:
+ case M_APP10:
+ case M_APP11:
+ case M_APP12:
+ case M_APP13:
+ case M_APP14:
+ case M_APP15:
+ case M_COM:
+ {
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tMarker 0x%X", c);
+ if (c == M_COM)
+ pdc_logg(p->pdc, "(COM) found\n");
+ else
+ pdc_logg(p->pdc, "(APP%d) found\n", c - M_APP0);
+ }
+
+ length = get_ushort(image->fp);
+ if (!image->jpegoptimize)
+ pdf_register_JPEG_segment(p, image, pos,
+ (size_t) (length + JPEG_MARKER_LEN));
+ else if (logg5)
+ pdc_logg(p->pdc, "\t\tSkip segment, position=0x%lX, "
+ "length=0x%lX(%ld)\n",
+ pos, length, length);
+
+ /* We may have to register the Adobe marker later */
+ if (c == M_APP14)
+ {
+ adobe_pos = pos;
+ adobe_len = length;
+ }
+
+ length -= JPEG_LENGTH_LEN;
+ }
+ break;
+
+ /* keep segment unconditionally */
+ case M_SOF0:
+ case M_SOF1:
+ case M_SOF2:
+ case M_SOF10:
+ case M_SOS:
+ default:
+ {
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tMarker 0x%X", c);
+ if (c == M_SOS)
+ pdc_logg(p->pdc, "(SOS) found\n");
+ else if (c <= M_SOF15)
+ pdc_logg(p->pdc, "(SOF%d) found\n", c - M_SOF0);
+ else
+ pdc_logg(p->pdc, " found\n");
+ }
+
+ length = get_ushort(image->fp);
+ pdf_register_JPEG_segment(p, image, pos,
+ (size_t) (length + JPEG_MARKER_LEN));
+ length -= JPEG_LENGTH_LEN;
+ }
+ break;
+ }
+
+ /* end of image */
+ if (c == M_EOI)
+ {
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tEnd of image\n");
+ }
+ break;
+ }
+
+ /* processing JPEG Marker */
+ switch (c)
+ {
+ /* check for frame header markers */
+ case M_SOF0:
+ case M_SOF1:
+ case M_SOF2:
+ case M_SOF10:
+ {
+ int comp;
+
+ image->bpc = pdc_fgetc(image->fp);
+ image->height = (pdc_scalar) get_ushort(image->fp);
+ image->width = (pdc_scalar) get_ushort(image->fp);
+ image->components = pdc_fgetc(image->fp);
+ length -= 6;
+
+ for (comp=0; comp<image->components; comp++)
+ {
+ pdc_byte b;
+
+ /* We don't support more than 4 components */
+ if (comp==JPEG_MAX_COMPS) break;
+
+ image->info.jpeg.id[comp] = pdc_fgetc(image->fp);
+ b = pdc_fgetc(image->fp);
+ image->info.jpeg.hsamp[comp] = (b >> 4) & 0x0F;
+ image->info.jpeg.vsamp[comp] = b & 0x0F;
+ image->info.jpeg.table[comp] = pdc_fgetc(image->fp);
+ length -= 3;
+ }
+
+ /*
+ * No need to read more markers since multiscan detection
+ * not required for single-component images.
+ */
+ if (image->components == 1)
+ markers_done = pdc_true;
+
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\t\tbpc = %d\n", image->bpc);
+ pdc_logg(p->pdc, "\t\theight = %g\n", image->height);
+ pdc_logg(p->pdc, "\t\twidth = %g\n", image->width);
+ pdc_logg(p->pdc, "\t\tcomponents = %d\n",
+ image->components);
+
+ for (comp=0; comp<image->components; comp++)
+ {
+ if (comp==JPEG_MAX_COMPS)
+ {
+ pdc_logg(p->pdc, "\t\tMore components found\n");
+ break;
+ }
+
+ if (pdc_logg_isprint((int) image->info.jpeg.id[comp]))
+ {
+ pdc_logg(p->pdc,
+ "\t\tcomponent 0x%x (name='%c'): "
+ "%dhx%dv table=%d\n",
+ image->info.jpeg.id[comp],
+ image->info.jpeg.id[comp],
+ image->info.jpeg.hsamp[comp],
+ image->info.jpeg.vsamp[comp],
+ image->info.jpeg.table[comp]);
+ }
+ else
+ {
+ pdc_logg(p->pdc,
+ "\t\tcomponent 0x%x: %dhx%dv table=%d\n",
+ image->info.jpeg.id[comp],
+ image->info.jpeg.hsamp[comp],
+ image->info.jpeg.vsamp[comp],
+ image->info.jpeg.table[comp]);
+ }
+ }
+ }
+ }
+ break;
+
+ /* check for JFIF marker with resolution */
+ case M_APP0:
+ {
+ len = MIN(APP_MAX, length);
+ if (!PDC_OK_FREAD(image->fp, appstring, len))
+ {
+ errcode = PDF_E_IMAGE_CORRUPT;
+ goto PDF_JPEG_ERROR;
+ }
+ length -= len;
+
+ /* Check for JFIF application marker and read density values
+ * per JFIF spec version 1.02.
+ */
+
+#define PDF_STRING_JFIF "\x4A\x46\x49\x46"
+
+ slen = strlen(PDF_STRING_JFIF);
+ if (len > slen &&
+ !strncmp(PDF_STRING_JFIF, (char *) appstring, slen))
+ {
+ /* resolution unit and resolution */
+ unit = appstring[7];
+ image->dpi_x = (pdc_scalar)
+ pdc_get_be_ushort(&appstring[8]);
+ image->dpi_y = (pdc_scalar)
+ pdc_get_be_ushort(&appstring[10]);
+
+#define JFIF_ASPECT_RATIO 0 /* JFIF unit byte: aspect ratio only */
+#define JFIF_DOTS_PER_INCH 1 /* JFIF unit byte: dots per inch */
+#define JFIF_DOTS_PER_CM 2 /* JFIF unit byte: dots per cm */
+
+ switch (unit)
+ {
+ case JFIF_DOTS_PER_INCH:
+ break;
+
+ case JFIF_DOTS_PER_CM:
+ image->dpi_x *= 100 * PDC_INCH2METER;
+ image->dpi_y *= 100 * PDC_INCH2METER;
+ break;
+
+ case JFIF_ASPECT_RATIO:
+ image->dpi_x *= -1;
+ image->dpi_y *= -1;
+ break;
+
+ /* unknown ==> ignore */
+ default:
+ break;
+ }
+
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\t\tJFIF marker found\n");
+ pdc_logg(p->pdc, "\t\tJFIF density unit: %d", unit);
+
+ switch (unit)
+ {
+ case JFIF_DOTS_PER_INCH:
+ pdc_logg(p->pdc, " (inch)\n");
+ break;
+
+ case JFIF_DOTS_PER_CM:
+ pdc_logg(p->pdc, " (cm)\n");
+ break;
+
+ case JFIF_ASPECT_RATIO:
+ pdc_logg(p->pdc, " (aspect ratio)\n");
+ break;
+
+ default:
+ pdc_logg(p->pdc, " (unknown; ignored)\n");
+ break;
+ }
+ pdc_logg(p->pdc, "\t\tJFIF x resolution = %g\n",
+ image->dpi_x);
+ pdc_logg(p->pdc, "\t\tJFIF y resolution = %g\n",
+ image->dpi_y);
+ }
+ }
+ }
+ break;
+
+
+ /* check for Adobe marker */
+ case M_APP14:
+ {
+ len = MIN(APP_MAX, length);
+ if (!PDC_OK_FREAD(image->fp, appstring, len))
+ {
+ errcode = PDF_E_IMAGE_CORRUPT;
+ goto PDF_JPEG_ERROR;
+ }
+ length -= len;
+
+ /*
+ * Check for Adobe application marker. It is known
+ * (per Adobe's TN5116)
+ * to contain the string "Adobe" at the start
+ * of the APP14 marker.
+ */
+
+#define PDF_STRING_Adobe "\x41\x64\x6F\x62\x65"
+
+ slen = strlen(PDF_STRING_Adobe);
+ if (len > slen &&
+ !strncmp(PDF_STRING_Adobe, (char *) appstring, slen))
+ {
+ if (logg5)
+ {
+ pdc_byte *val = appstring+slen;
+
+ pdc_logg(p->pdc, "\t\tAdobe marker found\n");
+
+ if (len >= 12)
+ {
+ pdc_logg(p->pdc, "\t\tversion = 0x%02X 0x%02X\n",
+ (unsigned char) val[0], (unsigned char) val[1]);
+ pdc_logg(p->pdc, "\t\tflags0 = 0x%02X 0x%02X\n",
+ (unsigned char) val[2], (unsigned char) val[3]);
+ pdc_logg(p->pdc, "\t\tflags1 = 0x%02X 0x%02X\n",
+ (unsigned char) val[4], (unsigned char) val[5]);
+ pdc_logg(p->pdc, "\t\tcolor transform = 0x%02X\n",
+ val[6]);
+ }
+ }
+ if (len >= 12)
+ transform = appstring[slen+6];
+
+ /* Keep Adobe marker for transform == 2 (YCCK) */
+ if (transform == 2)
+ {
+ if (logg5)
+ pdc_logg(p->pdc,
+ "\t\tYCCK color space: Keep Adobe marker\n");
+
+ pdf_register_JPEG_segment(p, image,
+ adobe_pos, (size_t) (adobe_len + JPEG_MARKER_LEN));
+ }
+ }
+ }
+ break;
+
+ /* check for start of scan marker */
+ case M_SOS:
+ {
+ pdc_byte comps = pdc_fgetc(image->fp);
+ length -= 1;
+
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\t\tNumber of components in scan = "
+ "%d\n", comps);
+ }
+
+ /*
+ * If the scan doesn't contain all components it must be
+ * a multiscan image, which doesn't work in Acrobat.
+ */
+
+ if (comps < image->components)
+ {
+ need_transcode = pdc_true;
+ if (logg5)
+ {
+ pdc_logg(p->pdc,
+ "\ttranscoding because of multiscan\n");
+ }
+ }
+
+ markers_done = pdc_true;
+ }
+ break;
+
+ default:
+ break;
+ }
+
+ /* jump to the next marker */
+ if (length > 0)
+ {
+ if (pdc_fseek(image->fp, (long) length, SEEK_CUR) == -1)
+ {
+ errcode = PDF_E_IMAGE_CORRUPT;
+ goto PDF_JPEG_ERROR;
+ }
+ }
+ }
+
+ /* do some sanity checks with the parameters */
+ if (image->height <= 0 || image->width <= 0 || image->components <= 0)
+ {
+ errcode = PDF_E_IMAGE_CORRUPT;
+ goto PDF_JPEG_ERROR;
+ }
+
+ if (image->bpc != 8)
+ {
+ errint = image->bpc;
+ errcode = PDF_E_IMAGE_BADDEPTH;
+ goto PDF_JPEG_ERROR;
+ }
+
+ {
+ switch (image->components) {
+ case 1:
+ /* spot color may have been applied */
+ if (image->colorspace == pdc_undef)
+ image->colorspace = DeviceGray;
+ break;
+
+ case 3:
+ image->colorspace = DeviceRGB;
+ break;
+
+ case 4:
+ image->colorspace = DeviceCMYK;
+ break;
+
+ default:
+ errint = image->components;
+ errcode = PDF_E_IMAGE_BADCOMP;
+ goto PDF_JPEG_ERROR;
+ }
+ }
+
+
+
+ if (image->imagemask)
+ {
+ if (image->components != 1)
+ {
+ errcode = PDF_E_IMAGE_BADMASK;
+ goto PDF_JPEG_ERROR;
+ }
+
+ if (p->compatibility <= PDC_1_3)
+ {
+ errcode = PDF_E_IMAGE_MASK1BIT13;
+ goto PDF_JPEG_ERROR;
+ }
+ else
+ {
+ /* images with more than one bit will be written as /SMask,
+ * and don't require an /ImageMask entry.
+ */
+ image->imagemask = pdc_false;
+ }
+ }
+
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tColorspace=");
+ pdf_log_colorspace(p, image->colorspace);
+ pdc_logg(p->pdc, "\n");
+ }
+
+ /* special handling for CMYK JPEG files */
+ if (image->components == 4)
+ {
+ /* CMYK JPEGs use inverse polarity */
+ image->invert = !image->invert;
+ if (logg5)
+ pdc_logg(p->pdc,
+ "\tinverting image because of 4 components\n");
+
+ /* Adobe and other CMYK JPEGs always require transcoding */
+ need_transcode = pdc_true;
+ if (logg5)
+ pdc_logg(p->pdc,
+ "\ttranscoding image because of 4 components\n");
+
+ }
+
+ image->in_use = pdc_true; /* mark slot as used */
+
+ if (need_transcode)
+ {
+ if (logg5)
+ {
+ pdc_logg(p->pdc, "\tcalling libjpeg for transcoding\n");
+ }
+ image->src.init = NULL;
+ image->src.fill = pdf_data_source_JPEG_fill_transcode;
+ image->src.terminate = NULL;
+ }
+ else
+ {
+ image->src.init = pdf_data_source_JPEG_init;
+ image->src.fill = pdf_data_source_JPEG_fill;
+ image->src.terminate = pdf_data_source_JPEG_terminate;
+ }
+
+ image->src.private_data = (void *) image;
+
+ if (image->doinline)
+ pdf_put_inline_image(p, imageslot);
+ else
+ pdf_put_image(p, imageslot, pdc_true, pdc_true);
+
+ if (!image->corrupt)
+ {
+ pdf_cleanup_jpeg(p, image);
+
+ return imageslot;
+ }
+
+ PDF_JPEG_ERROR:
+ {
+ const char *stemp = NULL;
+
+ if (errcode)
+ stemp = pdf_get_image_filename(p, image);
+
+
+ switch (errcode)
+ {
+ case PDC_E_IO_READ:
+ case PDF_E_IMAGE_ICC:
+ case PDF_E_IMAGE_ICC2:
+ case PDF_E_IMAGE_COLORIZE:
+ case PDF_E_IMAGE_BADMASK:
+ case PDF_E_IMAGE_MASK1BIT13:
+ pdc_set_errmsg(p->pdc, errcode, stemp, 0, 0, 0);
+ break;
+
+ case PDC_E_IO_BADFORMAT:
+ pdc_set_errmsg(p->pdc, errcode, stemp, "JPEG", 0, 0);
+ break;
+
+ case PDF_E_IMAGE_CORRUPT:
+ pdc_set_errmsg(p->pdc, errcode, "JPEG", stemp, 0, 0);
+ break;
+
+ case PDF_E_JPEG_COMPRESSION:
+ case PDF_E_IMAGE_BADDEPTH:
+ case PDF_E_IMAGE_BADCOMP:
+ pdc_set_errmsg(p->pdc, errcode,
+ pdc_errprintf(p->pdc, "%d", errint), stemp, 0, 0);
+ break;
+
+ case 0: /* error code and message already set */
+ break;
+ }
+ }
+
+ pdf_cleanup_jpeg(p, image);
+
+ return -1;
+}
+
+void
+pdf_cleanup_jpeg(PDF *p, pdf_image *image)
+{
+ if (image->info.jpeg.virtfile != NULL)
+ {
+ (void) pdc__delete_pvf(p->pdc, image->info.jpeg.virtfile);
+ image->info.jpeg.virtfile = NULL;
+ }
+
+ if (image->info.jpeg.seglist != NULL)
+ {
+ pdc_free(p->pdc, image->info.jpeg.seglist);
+ image->info.jpeg.seglist = NULL;
+ }
+}
+
+
+#endif /* PDF_JPEG_SUPPORTED */