* compiles, but untested * works, but libtiff needs seek for write next: add seek methods to target * add target seek and read seem to work next: disc temps for disc output * add libnsgif COPYING oops, we were missing the COPYING file see https://github.com/libvips/libvips/issues/2800 thanks mika-fischer * tiffsave uses a disc temp if it can * revise temp target rules only make a disc temp if we are writing to a filesystem target * add new target methods to targetcustom
295 lines
6.0 KiB
C
295 lines
6.0 KiB
C
/* Some shared TIFF utilities.
|
|
*
|
|
* 14/10/16
|
|
* - from vips2tiff.c
|
|
*
|
|
* 26/8/17
|
|
* - add openout_read, to help tiffsave_buffer for pyramids
|
|
*/
|
|
|
|
/*
|
|
|
|
This file is part of VIPS.
|
|
|
|
VIPS is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
|
02110-1301 USA
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
These files are distributed with VIPS - http://www.vips.ecs.soton.ac.uk
|
|
|
|
*/
|
|
|
|
/*
|
|
#define DEBUG
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif /*HAVE_CONFIG_H*/
|
|
#include <glib/gi18n-lib.h>
|
|
|
|
#ifdef HAVE_TIFF
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif /*HAVE_UNISTD_H*/
|
|
#include <string.h>
|
|
|
|
#include <vips/vips.h>
|
|
#include <vips/internal.h>
|
|
|
|
#include <tiffio.h>
|
|
|
|
#include "tiff.h"
|
|
|
|
/* Handle TIFF errors here. Shared with vips2tiff.c. These can be called from
|
|
* more than one thread.
|
|
*/
|
|
static void
|
|
vips__thandler_error( const char *module, const char *fmt, va_list ap )
|
|
{
|
|
vips_verror( module, fmt, ap );
|
|
}
|
|
|
|
/* It'd be nice to be able to support the @fail option for the tiff loader, but
|
|
* there's no easy way to do this, since libtiff has a global warning handler.
|
|
*/
|
|
static void
|
|
vips__thandler_warning( const char *module, const char *fmt, va_list ap )
|
|
{
|
|
g_logv( G_LOG_DOMAIN, G_LOG_LEVEL_WARNING, fmt, ap );
|
|
}
|
|
|
|
/* Called during library init.
|
|
*
|
|
* libtiff error and warning handlers may be called from other threads
|
|
* running in other libs. Other libs may install error handlers and capture
|
|
* messages caused by us.
|
|
*/
|
|
void
|
|
vips__tiff_init( void )
|
|
{
|
|
TIFFSetErrorHandler( vips__thandler_error );
|
|
TIFFSetWarningHandler( vips__thandler_warning );
|
|
}
|
|
|
|
/* TIFF input from a vips source.
|
|
*/
|
|
|
|
static tsize_t
|
|
openin_source_read( thandle_t st, tdata_t data, tsize_t size )
|
|
{
|
|
VipsSource *source = VIPS_SOURCE( st );
|
|
|
|
return( vips_source_read( source, data, size ) );
|
|
}
|
|
|
|
static tsize_t
|
|
openin_source_write( thandle_t st, tdata_t buffer, tsize_t size )
|
|
{
|
|
g_assert_not_reached();
|
|
|
|
return( 0 );
|
|
}
|
|
|
|
static toff_t
|
|
openin_source_seek( thandle_t st, toff_t offset, int whence )
|
|
{
|
|
VipsSource *source = VIPS_SOURCE( st );
|
|
|
|
return( (toff_t) vips_source_seek( source, offset, whence ) );
|
|
}
|
|
|
|
static int
|
|
openin_source_close( thandle_t st )
|
|
{
|
|
VipsSource *source = VIPS_SOURCE( st );
|
|
|
|
VIPS_UNREF( source );
|
|
|
|
return( 0 );
|
|
}
|
|
|
|
static toff_t
|
|
openin_source_length( thandle_t st )
|
|
{
|
|
VipsSource *source = VIPS_SOURCE( st );
|
|
|
|
/* libtiff will use this to get file size if tags like StripByteCounts
|
|
* are missing.
|
|
*
|
|
* toff_t is usually uint64, with -1 cast to uint64 to indicate error.
|
|
*/
|
|
return( (toff_t) vips_source_length( source ) );
|
|
}
|
|
|
|
static int
|
|
openin_source_map( thandle_t st, tdata_t *start, toff_t *len )
|
|
{
|
|
g_assert_not_reached();
|
|
|
|
return( 0 );
|
|
}
|
|
|
|
static void
|
|
openin_source_unmap( thandle_t st, tdata_t start, toff_t len )
|
|
{
|
|
g_assert_not_reached();
|
|
|
|
return;
|
|
}
|
|
|
|
TIFF *
|
|
vips__tiff_openin_source( VipsSource *source )
|
|
{
|
|
TIFF *tiff;
|
|
|
|
#ifdef DEBUG
|
|
printf( "vips__tiff_openin_source:\n" );
|
|
#endif /*DEBUG*/
|
|
|
|
if( vips_source_rewind( source ) )
|
|
return( NULL );
|
|
|
|
/* Disable memory mapped input -- it chews up VM and the performance
|
|
* gain is very small.
|
|
*
|
|
* C enables strip chopping: very large uncompressed strips are
|
|
* chopped into c. 8kb chunks. This can reduce peak memory use for
|
|
* this type of file.
|
|
*/
|
|
if( !(tiff = TIFFClientOpen( "source input", "rmC",
|
|
(thandle_t) source,
|
|
openin_source_read,
|
|
openin_source_write,
|
|
openin_source_seek,
|
|
openin_source_close,
|
|
openin_source_length,
|
|
openin_source_map,
|
|
openin_source_unmap )) ) {
|
|
vips_error( "vips__tiff_openin_source", "%s",
|
|
_( "unable to open source for input" ) );
|
|
return( NULL );
|
|
}
|
|
|
|
/* Unreffed on close(), see above.
|
|
*/
|
|
g_object_ref( source );
|
|
|
|
return( tiff );
|
|
}
|
|
|
|
/* TIFF output to a target.
|
|
*/
|
|
|
|
/* libtiff needs this (!!?!?!) for writing multipage images.
|
|
*/
|
|
static tsize_t
|
|
openout_target_read( thandle_t st, tdata_t data, tsize_t size )
|
|
{
|
|
VipsTarget *target = (VipsTarget *) st;
|
|
|
|
return( vips_target_read( target, data, size ) );
|
|
}
|
|
|
|
static tsize_t
|
|
openout_target_write( thandle_t st, tdata_t data, tsize_t size )
|
|
{
|
|
VipsTarget *target = (VipsTarget *) st;
|
|
|
|
if( vips_target_write( target, data, size ) )
|
|
return( (tsize_t) -1 );
|
|
|
|
return( size );
|
|
}
|
|
|
|
static toff_t
|
|
openout_target_seek( thandle_t st, toff_t offset, int whence )
|
|
{
|
|
VipsTarget *target = (VipsTarget *) st;
|
|
|
|
return( vips_target_seek( target, offset, whence ) );
|
|
}
|
|
|
|
static int
|
|
openout_target_close( thandle_t st )
|
|
{
|
|
VipsTarget *target = (VipsTarget *) st;
|
|
|
|
if( vips_target_end( target ) )
|
|
return( -1 );
|
|
|
|
return( 0 );
|
|
}
|
|
|
|
static toff_t
|
|
openout_target_length( thandle_t st )
|
|
{
|
|
g_assert_not_reached();
|
|
|
|
return( (toff_t) -1 );
|
|
}
|
|
|
|
static int
|
|
openout_target_map( thandle_t st, tdata_t *start, toff_t *len )
|
|
{
|
|
g_assert_not_reached();
|
|
|
|
return( -1 );
|
|
}
|
|
|
|
static void
|
|
openout_target_unmap( thandle_t st, tdata_t start, toff_t len )
|
|
{
|
|
g_assert_not_reached();
|
|
|
|
return;
|
|
}
|
|
|
|
TIFF *
|
|
vips__tiff_openout_target( VipsTarget *target, gboolean bigtiff )
|
|
{
|
|
const char *mode = bigtiff ? "w8" : "w";
|
|
|
|
TIFF *tiff;
|
|
|
|
#ifdef DEBUG
|
|
printf( "vips__tiff_openout_buffer:\n" );
|
|
#endif /*DEBUG*/
|
|
|
|
if( !(tiff = TIFFClientOpen( "target output", mode,
|
|
(thandle_t) target,
|
|
openout_target_read,
|
|
openout_target_write,
|
|
openout_target_seek,
|
|
openout_target_close,
|
|
openout_target_length,
|
|
openout_target_map,
|
|
openout_target_unmap )) ) {
|
|
vips_error( "vips__tiff_openout_target", "%s",
|
|
_( "unable to open target for output" ) );
|
|
return( NULL );
|
|
}
|
|
|
|
return( tiff );
|
|
}
|
|
|
|
#endif /*HAVE_TIFF*/
|