add dbuf object
dynamically expanding buffer also, escape "<>& appropriately when we write xml
This commit is contained in:
parent
5614330cc4
commit
e1b9c789cb
@ -36,7 +36,6 @@
|
|||||||
more tightly ... this makes seq more versatile and more reliable
|
more tightly ... this makes seq more versatile and more reliable
|
||||||
- use expat, not libxml, for XML load ... removes a required dependency, since
|
- use expat, not libxml, for XML load ... removes a required dependency, since
|
||||||
we get expat as part of glib
|
we get expat as part of glib
|
||||||
- use printf for xml write
|
|
||||||
|
|
||||||
8/12/16 started 8.4.5
|
8/12/16 started 8.4.5
|
||||||
- allow libgsf-1.14.26 to help centos, thanks tdiprima
|
- allow libgsf-1.14.26 to help centos, thanks tdiprima
|
||||||
|
134
TODO
134
TODO
@ -1,123 +1,7 @@
|
|||||||
- remove libxml from configure.ac
|
- use VipsDbuf for tiffsave_buffer etc.
|
||||||
|
|
||||||
we don't escape " inside xml attr strings, or < > inside xml content
|
- verify xml data against master for vips save and dzsave
|
||||||
|
|
||||||
break Buffer out into a separate class that does this? VipsDbuf, for dynamic
|
|
||||||
buffer? or can it be a mode for VipsBuf?
|
|
||||||
|
|
||||||
buffer should expand by 30% on each fill to avoid lots of tiny realloc
|
|
||||||
|
|
||||||
use for tiffsave_buffer etc.
|
|
||||||
|
|
||||||
have things like write-escaped-string as extra append operations
|
|
||||||
|
|
||||||
need one that escapes ", a separate one that escapes < >
|
|
||||||
|
|
||||||
- dzsave vips.properties is currently
|
|
||||||
|
|
||||||
|
|
||||||
<?xml version="1.0"?>
|
|
||||||
<image xmlns="http://www.vips.ecs.soton.ac.uk/dzsave" date="2017-02-25T17:32:07.565137Z" version="8.5.0">
|
|
||||||
<properties>
|
|
||||||
<property>
|
|
||||||
<name>width</name>
|
|
||||||
<value type="gint">1450</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>height</name>
|
|
||||||
<value type="gint">2048</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>bands</name>
|
|
||||||
<value type="gint">3</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>xoffset</name>
|
|
||||||
<value type="gint">0</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>yoffset</name>
|
|
||||||
<value type="gint">0</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>xres</name>
|
|
||||||
<value type="gdouble">0.99999999990686772</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>yres</name>
|
|
||||||
<value type="gdouble">0.99999999990686772</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>vips-loader</name>
|
|
||||||
<value type="VipsRefString">jpegload</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>jpeg-multiscan</name>
|
|
||||||
<value type="gint">0</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-data</name>
|
|
||||||
<value type="VipsBlob">
|
|
||||||
RXhpZgAASUkqAAgAAAAGABIBAwABAAAAAQAAABoBBQABAAAAVgAAABsBBQABAAAAXgAAACgBAwAB
|
|
||||||
AAAAAgAAABMCAwABAAAAAQAAAGmHBAABAAAAZgAAAAAAAAAR/P//KSgUChH8//8pKBQKBgAAkAcA
|
|
||||||
BAAAADAyMTABkQcABAAAAAECAwAAoAcABAAAADAxMDABoAMAAQAAAP//AAACoAQAAQAAAKoFAAAD
|
|
||||||
oAQAAQAAAAAIAAAAAAAA
|
|
||||||
</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>resolution-unit</name>
|
|
||||||
<value type="VipsRefString">in</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd0-Orientation</name>
|
|
||||||
<value type="VipsRefString">1 (Top-left, Short, 1 components, 2 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd0-XResolution</name>
|
|
||||||
<value type="VipsRefString">4294966289/169093161 (25.399999998, Rational, 1 components, 8 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd0-YResolution</name>
|
|
||||||
<value type="VipsRefString">4294966289/169093161 (25.399999998, Rational, 1 components, 8 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd0-ResolutionUnit</name>
|
|
||||||
<value type="VipsRefString">2 (Inch, Short, 1 components, 2 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd0-YCbCrPositioning</name>
|
|
||||||
<value type="VipsRefString">1 (Centred, Short, 1 components, 2 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd2-ExifVersion</name>
|
|
||||||
<value type="VipsRefString">Exif Version 2.1 (Exif Version 2.1, Undefined, 4 components, 4 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd2-ComponentsConfiguration</name>
|
|
||||||
<value type="VipsRefString">Y Cb Cr - (Y Cb Cr -, Undefined, 4 components, 4 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd2-FlashPixVersion</name>
|
|
||||||
<value type="VipsRefString">FlashPix Version 1.0 (FlashPix Version 1.0, Undefined, 4 components, 4 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd2-ColorSpace</name>
|
|
||||||
<value type="VipsRefString">65535 (Internal error (unknown value 65535), Short, 1 components, 2 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd2-PixelXDimension</name>
|
|
||||||
<value type="VipsRefString">1450 (1450, Long, 1 components, 4 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>exif-ifd2-PixelYDimension</name>
|
|
||||||
<value type="VipsRefString">2048 (2048, Long, 1 components, 4 bytes)</value>
|
|
||||||
</property>
|
|
||||||
<property>
|
|
||||||
<name>orientation</name>
|
|
||||||
<value type="gint">1</value>
|
|
||||||
</property>
|
|
||||||
</properties>
|
|
||||||
</image>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -246,20 +130,6 @@ DEBUG:gi.overrides.Vips:assigning sequential
|
|||||||
output imagevec
|
output imagevec
|
||||||
output blob
|
output blob
|
||||||
|
|
||||||
- fix up aconv
|
|
||||||
|
|
||||||
- rewrite im_conv() etc. as vips_conv(), also the mosaicing functions
|
|
||||||
|
|
||||||
finally finish --disable-deprecated option
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
- can we use postbuild elsewhere? look at use of "preclose" / "written", etc.
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
- test draw_mask on labq images
|
|
||||||
|
|
||||||
we probably need to unpack the ink back to double before blending
|
we probably need to unpack the ink back to double before blending
|
||||||
|
|
||||||
|
|
||||||
|
@ -6,6 +6,7 @@ pkginclude_HEADERS = \
|
|||||||
deprecated.h \
|
deprecated.h \
|
||||||
arithmetic.h \
|
arithmetic.h \
|
||||||
buf.h \
|
buf.h \
|
||||||
|
dbuf.h \
|
||||||
colour.h \
|
colour.h \
|
||||||
conversion.h \
|
conversion.h \
|
||||||
convolution.h \
|
convolution.h \
|
||||||
|
72
libvips/include/vips/dbuf.h
Normal file
72
libvips/include/vips/dbuf.h
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
/* A dynamic memory buffer that expands as you write.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef VIPS_DBUF_H
|
||||||
|
#define VIPS_DBUF_H
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif /*__cplusplus*/
|
||||||
|
|
||||||
|
#include <vips/vips.h>
|
||||||
|
|
||||||
|
/* A buffer in the process of being written to ... multiple calls to
|
||||||
|
* vips_dbuf_append add to it.
|
||||||
|
*/
|
||||||
|
|
||||||
|
typedef struct _VipsDbuf {
|
||||||
|
/* All fields are private.
|
||||||
|
*/
|
||||||
|
/*< private >*/
|
||||||
|
|
||||||
|
/* The current base, and the size of the allocated memory area.
|
||||||
|
*/
|
||||||
|
char *data;
|
||||||
|
size_t max_size;
|
||||||
|
|
||||||
|
/* And the write point.
|
||||||
|
*/
|
||||||
|
size_t write_point;
|
||||||
|
} VipsDbuf;
|
||||||
|
|
||||||
|
void vips_dbuf_destroy( VipsDbuf *buf );
|
||||||
|
void vips_dbuf_init( VipsDbuf *buf );
|
||||||
|
gboolean vips_dbuf_append( VipsDbuf *dbuf, const char *data, size_t size );
|
||||||
|
gboolean vips_dbuf_appendf( VipsDbuf *dbuf, const char *fmt, ... );
|
||||||
|
void vips_dbuf_rewind( VipsDbuf *dbuf );
|
||||||
|
void vips_dbuf_destroy( VipsDbuf *dbuf );
|
||||||
|
char *vips_dbuf_string( VipsDbuf *dbuf, size_t *size );
|
||||||
|
char *vips_dbuf_steal( VipsDbuf *dbuf, size_t *size );
|
||||||
|
|
||||||
|
#endif /*VIPS_DBUF_H*/
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif /*__cplusplus*/
|
@ -106,6 +106,7 @@ extern "C" {
|
|||||||
#include <vips/basic.h>
|
#include <vips/basic.h>
|
||||||
|
|
||||||
#include <vips/buf.h>
|
#include <vips/buf.h>
|
||||||
|
#include <vips/dbuf.h>
|
||||||
#include <vips/util.h>
|
#include <vips/util.h>
|
||||||
#include <vips/object.h>
|
#include <vips/object.h>
|
||||||
#include <vips/type.h>
|
#include <vips/type.h>
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
noinst_LTLIBRARIES = libiofuncs.la
|
noinst_LTLIBRARIES = libiofuncs.la
|
||||||
|
|
||||||
libiofuncs_la_SOURCES = \
|
libiofuncs_la_SOURCES = \
|
||||||
|
dbuf.c \
|
||||||
reorder.c \
|
reorder.c \
|
||||||
vipsmarshal.h \
|
vipsmarshal.h \
|
||||||
vipsmarshal.c \
|
vipsmarshal.c \
|
||||||
|
206
libvips/iofuncs/dbuf.c
Normal file
206
libvips/iofuncs/dbuf.c
Normal file
@ -0,0 +1,206 @@
|
|||||||
|
/* A dynamic memory buffer that expands as you write.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (C) 1991-2003 The National Gallery
|
||||||
|
|
||||||
|
This library 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.1 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
This library 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 library; 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 <vips/intl.h>
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include <vips/vips.h>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vips_dbuf_init:
|
||||||
|
* @dbuf: the buffer
|
||||||
|
*
|
||||||
|
* Initialize a buffer.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
vips_dbuf_init( VipsDbuf *dbuf )
|
||||||
|
{
|
||||||
|
dbuf->data = NULL;
|
||||||
|
dbuf->max_size = 0;
|
||||||
|
dbuf->write_point = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vips_dbuf_append:
|
||||||
|
* @dbuf: the buffer
|
||||||
|
* @data: the data to append to the buffer
|
||||||
|
* @size: the size of the len to append
|
||||||
|
*
|
||||||
|
* Append len bytes from @data to the buffer. The buffer expands if necessary.
|
||||||
|
*
|
||||||
|
* Returns: %FALSE on out of memory, %TRUE otherwise.
|
||||||
|
*/
|
||||||
|
gboolean
|
||||||
|
vips_dbuf_append( VipsDbuf *dbuf, const char *data, size_t size )
|
||||||
|
{
|
||||||
|
size_t new_write_point = dbuf->write_point + size;
|
||||||
|
|
||||||
|
if( new_write_point > dbuf->max_size ) {
|
||||||
|
size_t new_max_size = 3 * (16 + new_write_point) / 2;
|
||||||
|
|
||||||
|
char *new_data;
|
||||||
|
|
||||||
|
if( !(new_data = g_try_realloc( dbuf->data, new_max_size )) ) {
|
||||||
|
vips_error( "VipsDbuf", "%s", _( "out of memory" ) );
|
||||||
|
return( FALSE );
|
||||||
|
}
|
||||||
|
|
||||||
|
dbuf->data = new_data;
|
||||||
|
dbuf->max_size = new_max_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy( dbuf->data + dbuf->write_point, data, size );
|
||||||
|
dbuf->write_point = new_write_point;
|
||||||
|
|
||||||
|
return( TRUE );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vips_dbuf_appendf:
|
||||||
|
* @dbuf: the buffer
|
||||||
|
* @fmt: <function>printf()</function>-style format string
|
||||||
|
* @...: arguments to format string
|
||||||
|
*
|
||||||
|
* Format the string and append to @dbuf.
|
||||||
|
*
|
||||||
|
* Returns: %FALSE on out of memory, %TRUE otherwise.
|
||||||
|
*/
|
||||||
|
gboolean
|
||||||
|
vips_dbuf_appendf( VipsDbuf *dbuf, const char *fmt, ... )
|
||||||
|
{
|
||||||
|
va_list ap;
|
||||||
|
char *line;
|
||||||
|
|
||||||
|
va_start( ap, fmt );
|
||||||
|
line = g_strdup_vprintf( fmt, ap );
|
||||||
|
va_end( ap );
|
||||||
|
|
||||||
|
if( vips_dbuf_append( dbuf, line, strlen( line ) ) ) {
|
||||||
|
g_free( line );
|
||||||
|
return( FALSE );
|
||||||
|
}
|
||||||
|
g_free( line );
|
||||||
|
|
||||||
|
return( TRUE );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vips_dbuf_rewind:
|
||||||
|
* @dbuf: the buffer
|
||||||
|
*
|
||||||
|
* Reset the buffer to empty. No memory is freed, just the write pointer is
|
||||||
|
* reset.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
vips_dbuf_rewind( VipsDbuf *dbuf )
|
||||||
|
{
|
||||||
|
dbuf->write_point = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vips_dbuf_destroy:
|
||||||
|
* @dbuf: the buffer
|
||||||
|
*
|
||||||
|
* Destroy a buffer. This frees any allocated memory.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
vips_dbuf_destroy( VipsDbuf *dbuf )
|
||||||
|
{
|
||||||
|
VIPS_FREE( dbuf->data );
|
||||||
|
dbuf->max_size = 0;
|
||||||
|
dbuf->write_point = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vips_dbuf_steal:
|
||||||
|
* @dbuf: the buffer
|
||||||
|
* @size: (allow-none): optionally return length in bytes here
|
||||||
|
*
|
||||||
|
* Destroy a buffer, but rather than freeing memory, a pointer is returned.
|
||||||
|
* This must be freed with g_free().
|
||||||
|
*
|
||||||
|
* A `\0` is appended, but not included in the character count. This is so the
|
||||||
|
* pointer can be safely treated as a C string.
|
||||||
|
*
|
||||||
|
* Returns: (transfer full): The pointer held by @dbuf.
|
||||||
|
*/
|
||||||
|
char *
|
||||||
|
vips_dbuf_steal( VipsDbuf *dbuf, size_t *size )
|
||||||
|
{
|
||||||
|
char *data;
|
||||||
|
|
||||||
|
vips_dbuf_append( dbuf, "", 1 );
|
||||||
|
dbuf->write_point -= 1;
|
||||||
|
|
||||||
|
data = dbuf->data;
|
||||||
|
|
||||||
|
if( size )
|
||||||
|
*size = dbuf->write_point;
|
||||||
|
|
||||||
|
dbuf->data = NULL;
|
||||||
|
dbuf->max_size = 0;
|
||||||
|
dbuf->write_point = 0;
|
||||||
|
|
||||||
|
return( data );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vips_dbuf_data:
|
||||||
|
* @dbuf: the buffer
|
||||||
|
* @size: (allow-none): optionally return length in bytes here
|
||||||
|
*
|
||||||
|
* Return a pointer to @dbuf's internal data.
|
||||||
|
*
|
||||||
|
* A `\0` is appended, but not included in the character count. This is so the
|
||||||
|
* pointer can be safely treated as a C string.
|
||||||
|
*
|
||||||
|
* Returns: (transfer none): The pointer held by @dbuf.
|
||||||
|
*/
|
||||||
|
char *
|
||||||
|
vips_dbuf_string( VipsDbuf *dbuf, size_t *size )
|
||||||
|
{
|
||||||
|
vips_dbuf_append( dbuf, "", 1 );
|
||||||
|
dbuf->write_point -= 1;
|
||||||
|
|
||||||
|
if( size )
|
||||||
|
*size = dbuf->write_point;
|
||||||
|
|
||||||
|
return( dbuf->data );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
@ -49,8 +49,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
#define DEBUG
|
|
||||||
#define SHOW_HEADER
|
#define SHOW_HEADER
|
||||||
|
#define DEBUG
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
@ -492,56 +492,6 @@ parser_read_fd( XML_Parser parser, int fd )
|
|||||||
|
|
||||||
#define MAX_PARSE_ATTR (256)
|
#define MAX_PARSE_ATTR (256)
|
||||||
|
|
||||||
/* A memory buffer that expands as we write to it.
|
|
||||||
*/
|
|
||||||
typedef struct _Buffer {
|
|
||||||
char *data;
|
|
||||||
size_t current_size;
|
|
||||||
} Buffer;
|
|
||||||
|
|
||||||
static void
|
|
||||||
buffer_init( Buffer *buffer )
|
|
||||||
{
|
|
||||||
buffer->data = NULL;
|
|
||||||
buffer->current_size = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
buffer_append( Buffer *buffer, const char *data, int len )
|
|
||||||
{
|
|
||||||
size_t new_size = buffer->current_size + len;
|
|
||||||
|
|
||||||
buffer->data = g_realloc( buffer->data, new_size );
|
|
||||||
memcpy( buffer->data + buffer->current_size, data, len );
|
|
||||||
buffer->current_size = new_size;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
buffer_appendf( Buffer *buffer, const char *fmt, ... )
|
|
||||||
{
|
|
||||||
va_list ap;
|
|
||||||
char line[256];
|
|
||||||
|
|
||||||
va_start( ap, fmt );
|
|
||||||
(void) vips_vsnprintf( line, 256, fmt, ap );
|
|
||||||
va_end( ap );
|
|
||||||
|
|
||||||
buffer_append( buffer, line, strlen( line ) );
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
buffer_rewind( Buffer *buffer )
|
|
||||||
{
|
|
||||||
buffer->current_size = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
buffer_destroy( Buffer *buffer )
|
|
||||||
{
|
|
||||||
VIPS_FREE( buffer->data );
|
|
||||||
buffer->current_size = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* What we track during expat parse.
|
/* What we track during expat parse.
|
||||||
*/
|
*/
|
||||||
typedef struct _VipsExpatParse {
|
typedef struct _VipsExpatParse {
|
||||||
@ -562,7 +512,7 @@ typedef struct _VipsExpatParse {
|
|||||||
|
|
||||||
/* Accumulate data here.
|
/* Accumulate data here.
|
||||||
*/
|
*/
|
||||||
Buffer buffer;
|
VipsDbuf dbuf;
|
||||||
} VipsExpatParse;
|
} VipsExpatParse;
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -586,7 +536,7 @@ parser_element_start_handler( void *user_data,
|
|||||||
vips_strncpy( vep->type, p[1], MAX_PARSE_ATTR );
|
vips_strncpy( vep->type, p[1], MAX_PARSE_ATTR );
|
||||||
}
|
}
|
||||||
|
|
||||||
buffer_rewind( &vep->buffer );
|
vips_dbuf_rewind( &vep->dbuf );
|
||||||
}
|
}
|
||||||
else if( strcmp( name, "header" ) == 0 )
|
else if( strcmp( name, "header" ) == 0 )
|
||||||
vep->header = TRUE;
|
vep->header = TRUE;
|
||||||
@ -666,16 +616,10 @@ parser_element_end_handler( void *user_data, const XML_Char *name )
|
|||||||
#endif /*DEBUG*/
|
#endif /*DEBUG*/
|
||||||
|
|
||||||
if( strcmp( name, "field" ) == 0 ) {
|
if( strcmp( name, "field" ) == 0 ) {
|
||||||
buffer_append( &vep->buffer, "", 1 );
|
|
||||||
|
|
||||||
#ifdef DEBUG
|
|
||||||
printf( "parser_element_end_handler: %zd bytes\n",
|
|
||||||
vep->current_size );
|
|
||||||
#endif /*DEBUG*/
|
|
||||||
|
|
||||||
if( vep->header ) {
|
if( vep->header ) {
|
||||||
if( strcmp( name, "Hist" ) == 0 )
|
if( strcmp( name, "Hist" ) == 0 )
|
||||||
set_history( vep->image, vep->buffer.data );
|
set_history( vep->image,
|
||||||
|
vips_dbuf_string( &vep->dbuf, NULL ) );
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
GType gtype = g_type_from_name( vep->type );
|
GType gtype = g_type_from_name( vep->type );
|
||||||
@ -686,7 +630,8 @@ parser_element_end_handler( void *user_data, const XML_Char *name )
|
|||||||
g_value_type_transformable(
|
g_value_type_transformable(
|
||||||
VIPS_TYPE_SAVE_STRING, gtype ) &&
|
VIPS_TYPE_SAVE_STRING, gtype ) &&
|
||||||
set_meta( vep->image,
|
set_meta( vep->image,
|
||||||
gtype, vep->name, vep->buffer.data ) )
|
gtype, vep->name,
|
||||||
|
vips_dbuf_string( &vep->dbuf, NULL ) ) )
|
||||||
vep->error = TRUE;
|
vep->error = TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -701,7 +646,7 @@ parser_data_handler( void *user_data, const XML_Char *data, int len )
|
|||||||
printf( "parser_data_handler: %d bytes\n", len );
|
printf( "parser_data_handler: %d bytes\n", len );
|
||||||
#endif /*DEBUG*/
|
#endif /*DEBUG*/
|
||||||
|
|
||||||
buffer_append( &vep->buffer, data, len );
|
vips_dbuf_append( &vep->dbuf, data, len );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Called at the end of vips open ... get any XML after the pixel data
|
/* Called at the end of vips open ... get any XML after the pixel data
|
||||||
@ -719,7 +664,7 @@ readhist( VipsImage *im )
|
|||||||
parser = XML_ParserCreate( "UTF-8" );
|
parser = XML_ParserCreate( "UTF-8" );
|
||||||
|
|
||||||
vep.image = im;
|
vep.image = im;
|
||||||
buffer_init( &vep.buffer );
|
vips_dbuf_init( &vep.dbuf );
|
||||||
vep.error = FALSE;
|
vep.error = FALSE;
|
||||||
XML_SetUserData( parser, &vep );
|
XML_SetUserData( parser, &vep );
|
||||||
|
|
||||||
@ -729,12 +674,12 @@ readhist( VipsImage *im )
|
|||||||
|
|
||||||
if( parser_read_fd( parser, im->fd ) ||
|
if( parser_read_fd( parser, im->fd ) ||
|
||||||
vep.error ) {
|
vep.error ) {
|
||||||
buffer_destroy( &vep.buffer );
|
vips_dbuf_destroy( &vep.dbuf );
|
||||||
XML_ParserFree( parser );
|
XML_ParserFree( parser );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
buffer_destroy( &vep.buffer );
|
vips_dbuf_destroy( &vep.dbuf );
|
||||||
XML_ParserFree( parser );
|
XML_ParserFree( parser );
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
@ -768,8 +713,59 @@ vips__write_extension_block( VipsImage *im, void *buf, int size )
|
|||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Append a string to a buffer, but escape " as \".
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
dbuf_append_quotes( VipsDbuf *dbuf, const char *str )
|
||||||
|
{
|
||||||
|
const char *p;
|
||||||
|
size_t len;
|
||||||
|
|
||||||
|
for( p = str; *p; p += len ) {
|
||||||
|
len = strcspn( p, "\"" );
|
||||||
|
|
||||||
|
vips_dbuf_append( dbuf, p, len );
|
||||||
|
if( p[len] == '"' )
|
||||||
|
vips_dbuf_append( dbuf, "\\", 1 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Append a string to a buffer, but escape &<>.
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
dbuf_append_amp( VipsDbuf *dbuf, const char *str )
|
||||||
|
{
|
||||||
|
const char *p;
|
||||||
|
size_t len;
|
||||||
|
|
||||||
|
for( p = str; *p; p += len ) {
|
||||||
|
len = strcspn( p, "&<>" );
|
||||||
|
|
||||||
|
vips_dbuf_append( dbuf, p, len );
|
||||||
|
switch( p[len] ) {
|
||||||
|
case '&':
|
||||||
|
vips_dbuf_append( dbuf, "&", 5 );
|
||||||
|
len += 1;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '<':
|
||||||
|
vips_dbuf_append( dbuf, "<", 4 );
|
||||||
|
len += 1;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '>':
|
||||||
|
vips_dbuf_append( dbuf, ">", 4 );
|
||||||
|
len += 1;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void *
|
static void *
|
||||||
build_xml_meta( VipsMeta *meta, Buffer *buffer )
|
build_xml_meta( VipsMeta *meta, VipsDbuf *dbuf )
|
||||||
{
|
{
|
||||||
GType type = G_VALUE_TYPE( &meta->value );
|
GType type = G_VALUE_TYPE( &meta->value );
|
||||||
|
|
||||||
@ -786,14 +782,16 @@ build_xml_meta( VipsMeta *meta, Buffer *buffer )
|
|||||||
if( !g_value_transform( &meta->value, &save_value ) ) {
|
if( !g_value_transform( &meta->value, &save_value ) ) {
|
||||||
vips_error( "VipsImage", "%s",
|
vips_error( "VipsImage", "%s",
|
||||||
_( "error transforming to save format" ) );
|
_( "error transforming to save format" ) );
|
||||||
return( buffer );
|
return( meta );
|
||||||
}
|
}
|
||||||
|
|
||||||
str = vips_value_get_save_string( &save_value );
|
str = vips_value_get_save_string( &save_value );
|
||||||
buffer_appendf( buffer, " <field type=\"%s\" name=\"%s\">",
|
vips_dbuf_appendf( dbuf, " <field type=\"%s\" name=\"",
|
||||||
g_type_name( type ), meta->name );
|
g_type_name( type ) );
|
||||||
buffer_append( buffer, str, strlen( str ) );
|
dbuf_append_quotes( dbuf, meta->name );
|
||||||
buffer_appendf( buffer, "</field>\n" );
|
vips_dbuf_appendf( dbuf, "\">" );
|
||||||
|
dbuf_append_amp( dbuf, str );
|
||||||
|
vips_dbuf_appendf( dbuf, "</field>\n" );
|
||||||
|
|
||||||
g_value_unset( &save_value );
|
g_value_unset( &save_value );
|
||||||
}
|
}
|
||||||
@ -806,43 +804,43 @@ build_xml_meta( VipsMeta *meta, Buffer *buffer )
|
|||||||
static char *
|
static char *
|
||||||
build_xml( VipsImage *image )
|
build_xml( VipsImage *image )
|
||||||
{
|
{
|
||||||
Buffer buffer;
|
VipsDbuf dbuf;
|
||||||
const char *str;
|
const char *str;
|
||||||
|
|
||||||
buffer_init( &buffer );
|
vips_dbuf_init( &dbuf );
|
||||||
|
|
||||||
buffer_appendf( &buffer, "<?xml version=\"1.0\"?>\n" );
|
vips_dbuf_appendf( &dbuf, "<?xml version=\"1.0\"?>\n" );
|
||||||
buffer_appendf( &buffer, "<root xmlns=\"%s/vips/%d.%d.%d\">\n",
|
vips_dbuf_appendf( &dbuf, "<root xmlns=\"%s/vips/%d.%d.%d\">\n",
|
||||||
NAMESPACE_URI,
|
NAMESPACE_URI,
|
||||||
VIPS_MAJOR_VERSION, VIPS_MINOR_VERSION, VIPS_MICRO_VERSION );
|
VIPS_MAJOR_VERSION, VIPS_MINOR_VERSION, VIPS_MICRO_VERSION );
|
||||||
buffer_appendf( &buffer, " <header>\n" );
|
vips_dbuf_appendf( &dbuf, " <header>\n" );
|
||||||
|
|
||||||
str = vips_image_get_history( image );
|
str = vips_image_get_history( image );
|
||||||
buffer_appendf( &buffer, " <field type=\"%s\" name=\"Hist\">",
|
vips_dbuf_appendf( &dbuf, " <field type=\"%s\" name=\"Hist\">",
|
||||||
g_type_name( VIPS_TYPE_REF_STRING ) );
|
g_type_name( VIPS_TYPE_REF_STRING ) );
|
||||||
buffer_append( &buffer, str, strlen( str ) );
|
dbuf_append_amp( &dbuf, str );
|
||||||
buffer_appendf( &buffer, "</field>\n" );
|
vips_dbuf_appendf( &dbuf, "</field>\n" );
|
||||||
|
|
||||||
buffer_appendf( &buffer, " </header>\n" );
|
vips_dbuf_appendf( &dbuf, " </header>\n" );
|
||||||
buffer_appendf( &buffer, " <meta>\n" );
|
vips_dbuf_appendf( &dbuf, " <meta>\n" );
|
||||||
|
|
||||||
if( vips_slist_map2( image->meta_traverse,
|
if( vips_slist_map2( image->meta_traverse,
|
||||||
(VipsSListMap2Fn) build_xml_meta, &buffer, NULL ) ) {
|
(VipsSListMap2Fn) build_xml_meta, &dbuf, NULL ) ) {
|
||||||
buffer_destroy( &buffer );
|
vips_dbuf_destroy( &dbuf );
|
||||||
return( NULL );
|
return( NULL );
|
||||||
}
|
}
|
||||||
|
|
||||||
buffer_appendf( &buffer, " </meta>\n" );
|
vips_dbuf_appendf( &dbuf, " </meta>\n" );
|
||||||
buffer_appendf( &buffer, "</root>\n" );
|
vips_dbuf_appendf( &dbuf, "</root>\n" );
|
||||||
|
|
||||||
return( buffer.data );
|
return( vips_dbuf_steal( &dbuf, NULL ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
static void *
|
static void *
|
||||||
vips__xml_properties_meta( VipsImage *image,
|
vips__xml_properties_meta( VipsImage *image,
|
||||||
const char *field, GValue *value, void *a )
|
const char *field, GValue *value, void *a )
|
||||||
{
|
{
|
||||||
Buffer *buffer = (Buffer *) a;
|
VipsDbuf *dbuf = (VipsDbuf *) a;
|
||||||
GType type = G_VALUE_TYPE( value );
|
GType type = G_VALUE_TYPE( value );
|
||||||
|
|
||||||
const char *str;
|
const char *str;
|
||||||
@ -858,18 +856,20 @@ vips__xml_properties_meta( VipsImage *image,
|
|||||||
if( !g_value_transform( value, &save_value ) ) {
|
if( !g_value_transform( value, &save_value ) ) {
|
||||||
vips_error( "VipsImage", "%s",
|
vips_error( "VipsImage", "%s",
|
||||||
_( "error transforming to save format" ) );
|
_( "error transforming to save format" ) );
|
||||||
return( buffer );
|
return( dbuf );
|
||||||
}
|
}
|
||||||
str = vips_value_get_save_string( &save_value );
|
str = vips_value_get_save_string( &save_value );
|
||||||
g_value_unset( &save_value );
|
g_value_unset( &save_value );
|
||||||
|
|
||||||
buffer_appendf( buffer, " <property>\n" );
|
vips_dbuf_appendf( dbuf, " <property>\n" );
|
||||||
buffer_appendf( buffer, " <name>%s</name>\n", field );
|
vips_dbuf_appendf( dbuf, " <name>" );
|
||||||
buffer_appendf( buffer, " <value type=\"%s\">",
|
dbuf_append_amp( dbuf, field );
|
||||||
|
vips_dbuf_appendf( dbuf, "</name>\n" );
|
||||||
|
vips_dbuf_appendf( dbuf, " <value type=\"%s\">",
|
||||||
g_type_name( type ) );
|
g_type_name( type ) );
|
||||||
buffer_append( buffer, str, strlen( str ) );
|
dbuf_append_amp( dbuf, str );
|
||||||
buffer_appendf( buffer, "</value>\n" );
|
vips_dbuf_appendf( dbuf, "</value>\n" );
|
||||||
buffer_appendf( buffer, " </property>\n" );
|
vips_dbuf_appendf( dbuf, " </property>\n" );
|
||||||
}
|
}
|
||||||
|
|
||||||
return( NULL );
|
return( NULL );
|
||||||
@ -881,32 +881,32 @@ vips__xml_properties_meta( VipsImage *image,
|
|||||||
char *
|
char *
|
||||||
vips__xml_properties( VipsImage *image )
|
vips__xml_properties( VipsImage *image )
|
||||||
{
|
{
|
||||||
Buffer buffer;
|
VipsDbuf dbuf;
|
||||||
GTimeVal now;
|
GTimeVal now;
|
||||||
char *date;
|
char *date;
|
||||||
|
|
||||||
buffer_init( &buffer );
|
vips_dbuf_init( &dbuf );
|
||||||
|
|
||||||
g_get_current_time( &now );
|
g_get_current_time( &now );
|
||||||
date = g_time_val_to_iso8601( &now );
|
date = g_time_val_to_iso8601( &now );
|
||||||
buffer_appendf( &buffer, "<?xml version=\"1.0\"?>\n" );
|
vips_dbuf_appendf( &dbuf, "<?xml version=\"1.0\"?>\n" );
|
||||||
buffer_appendf( &buffer, "<image xmlns=\"%s/dzsave\" "
|
vips_dbuf_appendf( &dbuf, "<image xmlns=\"%s/dzsave\" "
|
||||||
"date=\"%s\" version=\"%d.%d.%d\">\n",
|
"date=\"%s\" version=\"%d.%d.%d\">\n",
|
||||||
NAMESPACE_URI,
|
NAMESPACE_URI,
|
||||||
date,
|
date,
|
||||||
VIPS_MAJOR_VERSION, VIPS_MINOR_VERSION, VIPS_MICRO_VERSION );
|
VIPS_MAJOR_VERSION, VIPS_MINOR_VERSION, VIPS_MICRO_VERSION );
|
||||||
g_free( date );
|
g_free( date );
|
||||||
buffer_appendf( &buffer, " <properties>\n" );
|
vips_dbuf_appendf( &dbuf, " <properties>\n" );
|
||||||
|
|
||||||
if( vips_image_map( image, vips__xml_properties_meta, &buffer ) ) {
|
if( vips_image_map( image, vips__xml_properties_meta, &dbuf ) ) {
|
||||||
buffer_destroy( &buffer );
|
vips_dbuf_destroy( &dbuf );
|
||||||
return( NULL );
|
return( NULL );
|
||||||
}
|
}
|
||||||
|
|
||||||
buffer_appendf( &buffer, " </properties>\n" );
|
vips_dbuf_appendf( &dbuf, " </properties>\n" );
|
||||||
buffer_appendf( &buffer, "</image>\n" );
|
vips_dbuf_appendf( &dbuf, "</image>\n" );
|
||||||
|
|
||||||
return( buffer.data );
|
return( vips_dbuf_steal( &dbuf, NULL ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Append XML to output fd.
|
/* Append XML to output fd.
|
||||||
|
Loading…
Reference in New Issue
Block a user