Files correlati : pdflib Ricompilazione Demo : [ ] Commento : Aggiornata PDFlib git-svn-id: svn://10.65.10.50/trunk@17433 c028cbd2-c16b-5b4b-a496-9718f37d4682
		
			
				
	
	
		
			1560 lines
		
	
	
		
			43 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			1560 lines
		
	
	
		
			43 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
/*---------------------------------------------------------------------------*
 | 
						|
 |              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.3 2008-10-20 14:34:15 guy 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)
 | 
						|
{
 | 
						|
    (void) imageslot;
 | 
						|
 | 
						|
    pdc_set_errmsg(p->pdc, PDF_E_UNSUPP_IMAGE, "JPEG", 0, 0, 0);
 | 
						|
 | 
						|
    return -1;
 | 
						|
}
 | 
						|
 | 
						|
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 */
 |