add pbm save
previously vips could load but not save pbm (one bit) images fix a few bugs in float load and save as well, and improve one-bit load
This commit is contained in:
parent
be4ffa6d8a
commit
cce90b4fba
@ -1,5 +1,7 @@
|
|||||||
4/11/14 started 7.42.0
|
4/11/14 started 7.42.0
|
||||||
- better default resolution for png load
|
- better default resolution for png load
|
||||||
|
- better pbm (one bit) load, better pfm (float) load/save
|
||||||
|
- added pbm (one bit) save
|
||||||
|
|
||||||
25/7/14 started 7.41.0
|
25/7/14 started 7.41.0
|
||||||
- start working on --disable-deprecated
|
- start working on --disable-deprecated
|
||||||
|
8
TODO
8
TODO
@ -1,11 +1,3 @@
|
|||||||
- try:
|
|
||||||
|
|
||||||
$ vipsthumbnail 360mp.png --vips-leak --vips-concurrency=1 --vips-progress
|
|
||||||
|
|
||||||
works in tiles ... should it be strips?
|
|
||||||
|
|
||||||
no %complete feedback
|
|
||||||
|
|
||||||
- try:
|
- try:
|
||||||
|
|
||||||
vips gaussnoise r.v 64 64
|
vips gaussnoise r.v 64 64
|
||||||
|
@ -29,6 +29,8 @@
|
|||||||
* - add PFM (portable float map) support
|
* - add PFM (portable float map) support
|
||||||
* 19/12/11
|
* 19/12/11
|
||||||
* - rework as a set of fns ready to be called from a class
|
* - rework as a set of fns ready to be called from a class
|
||||||
|
* 8/11/14
|
||||||
|
* - add 1 bit write
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -68,6 +70,7 @@
|
|||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
#include <vips/vips.h>
|
#include <vips/vips.h>
|
||||||
#include <vips/internal.h>
|
#include <vips/internal.h>
|
||||||
@ -216,7 +219,8 @@ read_header( FILE *fp, VipsImage *out, int *bits, int *ascii, int *msb_first )
|
|||||||
/* For binary images, there is always exactly 1 more whitespace
|
/* For binary images, there is always exactly 1 more whitespace
|
||||||
* character before the data starts.
|
* character before the data starts.
|
||||||
*/
|
*/
|
||||||
if( !*ascii && !isspace( fgetc( fp ) ) ) {
|
if( !*ascii &&
|
||||||
|
!isspace( fgetc( fp ) ) ) {
|
||||||
vips_error( "ppm2vips", "%s",
|
vips_error( "ppm2vips", "%s",
|
||||||
_( "not whitespace before start of binary data" ) );
|
_( "not whitespace before start of binary data" ) );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
@ -291,7 +295,7 @@ read_mmap( FILE *fp, const char *filename, int msb_first, VipsImage *out )
|
|||||||
"coding", out->Coding,
|
"coding", out->Coding,
|
||||||
NULL ) ||
|
NULL ) ||
|
||||||
vips_copy( t[1], &t[2],
|
vips_copy( t[1], &t[2],
|
||||||
"swap", !vips_amiMSBfirst(),
|
"swap", vips_amiMSBfirst() != msb_first,
|
||||||
NULL ) ||
|
NULL ) ||
|
||||||
vips_image_write( t[2], out ) ) {
|
vips_image_write( t[2], out ) ) {
|
||||||
g_object_unref( x );
|
g_object_unref( x );
|
||||||
@ -364,7 +368,7 @@ read_1bit_ascii( FILE *fp, VipsImage *out )
|
|||||||
if( read_int( fp, &val ) )
|
if( read_int( fp, &val ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
if( val == 1 )
|
if( val )
|
||||||
buf[x] = 0;
|
buf[x] = 0;
|
||||||
else
|
else
|
||||||
buf[x] = 255;
|
buf[x] = 255;
|
||||||
@ -382,7 +386,7 @@ read_1bit_ascii( FILE *fp, VipsImage *out )
|
|||||||
static int
|
static int
|
||||||
read_1bit_binary( FILE *fp, VipsImage *out )
|
read_1bit_binary( FILE *fp, VipsImage *out )
|
||||||
{
|
{
|
||||||
int x, y, i;
|
int x, y;
|
||||||
int bits;
|
int bits;
|
||||||
VipsPel *buf;
|
VipsPel *buf;
|
||||||
|
|
||||||
@ -390,14 +394,19 @@ read_1bit_binary( FILE *fp, VipsImage *out )
|
|||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
bits = fgetc( fp );
|
bits = fgetc( fp );
|
||||||
for( i = 0, y = 0; y < out->Ysize; y++ ) {
|
for( y = 0; y < out->Ysize; y++ ) {
|
||||||
for( x = 0; x < out->Xsize * out->Bands; x++, i++ ) {
|
for( x = 0; x < out->Xsize * out->Bands; x++ ) {
|
||||||
buf[x] = (bits & 128) ? 255 : 0;
|
buf[x] = (bits & 128) ? 0 : 255;
|
||||||
bits <<= 1;
|
bits <<= 1;
|
||||||
if( (i & 7) == 7 )
|
if( (x & 7) == 7 )
|
||||||
bits = fgetc( fp );
|
bits = fgetc( fp );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Skip any unaligned bits at end of line.
|
||||||
|
*/
|
||||||
|
if( (x & 7) != 0 )
|
||||||
|
bits = fgetc( fp );
|
||||||
|
|
||||||
if( vips_image_write_line( out, y, buf ) )
|
if( vips_image_write_line( out, y, buf ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
@ -529,11 +538,13 @@ vips__ppm_flags( const char *filename )
|
|||||||
|
|
||||||
const char *vips__ppm_suffs[] = { ".ppm", ".pgm", ".pbm", ".pfm", NULL };
|
const char *vips__ppm_suffs[] = { ".ppm", ".pgm", ".pbm", ".pfm", NULL };
|
||||||
|
|
||||||
typedef int (*write_fn)( VipsImage *in, FILE *fp, VipsPel *p );
|
struct _Write;
|
||||||
|
|
||||||
|
typedef int (*write_fn)( struct _Write *write, VipsPel *p );
|
||||||
|
|
||||||
/* What we track during a PPM write.
|
/* What we track during a PPM write.
|
||||||
*/
|
*/
|
||||||
typedef struct {
|
typedef struct _Write {
|
||||||
VipsImage *in;
|
VipsImage *in;
|
||||||
FILE *fp;
|
FILE *fp;
|
||||||
char *name;
|
char *name;
|
||||||
@ -570,24 +581,27 @@ write_new( VipsImage *in, const char *name )
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
write_ppm_line_ascii( VipsImage *in, FILE *fp, VipsPel *p )
|
write_ppm_line_ascii( Write *write, VipsPel *p )
|
||||||
{
|
{
|
||||||
const int sk = VIPS_IMAGE_SIZEOF_PEL( in );
|
const int sk = VIPS_IMAGE_SIZEOF_PEL( write->in );
|
||||||
int x, k;
|
int x, k;
|
||||||
|
|
||||||
for( x = 0; x < in->Xsize; x++ ) {
|
for( x = 0; x < write->in->Xsize; x++ ) {
|
||||||
for( k = 0; k < in->Bands; k++ ) {
|
for( k = 0; k < write->in->Bands; k++ ) {
|
||||||
switch( in->BandFmt ) {
|
switch( write->in->BandFmt ) {
|
||||||
case VIPS_FORMAT_UCHAR:
|
case VIPS_FORMAT_UCHAR:
|
||||||
fprintf( fp, "%d ", p[k] );
|
fprintf( write->fp,
|
||||||
|
"%d ", p[k] );
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case VIPS_FORMAT_USHORT:
|
case VIPS_FORMAT_USHORT:
|
||||||
fprintf( fp, "%d ", ((unsigned short *) p)[k] );
|
fprintf( write->fp,
|
||||||
|
"%d ", ((unsigned short *) p)[k] );
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case VIPS_FORMAT_UINT:
|
case VIPS_FORMAT_UINT:
|
||||||
fprintf( fp, "%d ", ((unsigned int *) p)[k] );
|
fprintf( write->fp,
|
||||||
|
"%d ", ((unsigned int *) p)[k] );
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@ -595,14 +609,12 @@ write_ppm_line_ascii( VipsImage *in, FILE *fp, VipsPel *p )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fprintf( fp, " " );
|
|
||||||
|
|
||||||
p += sk;
|
p += sk;
|
||||||
}
|
}
|
||||||
|
|
||||||
if( !fprintf( fp, "\n" ) ) {
|
if( !fprintf( write->fp, "\n" ) ) {
|
||||||
vips_error( "vips2ppm",
|
vips_error_system( errno, "vips2ppm",
|
||||||
"%s", _( "write error ... disc full?" ) );
|
"%s", _( "write error" ) );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -610,17 +622,71 @@ write_ppm_line_ascii( VipsImage *in, FILE *fp, VipsPel *p )
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
write_ppm_line_binary( VipsImage *in, FILE *fp, VipsPel *p )
|
write_ppm_line_ascii_squash( Write *write, VipsPel *p )
|
||||||
{
|
{
|
||||||
if( !fwrite( p, VIPS_IMAGE_SIZEOF_LINE( in ), 1, fp ) ) {
|
int x;
|
||||||
vips_error( "vips2ppm",
|
|
||||||
"%s", _( "write error ... disc full?" ) );
|
for( x = 0; x < write->in->Xsize; x++ )
|
||||||
|
fprintf( write->fp, "%d ", p[x] ? 0 : 1 );
|
||||||
|
|
||||||
|
if( !fprintf( write->fp, "\n" ) ) {
|
||||||
|
vips_error_system( errno, "vips2ppm",
|
||||||
|
"%s", _( "write error" ) );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
write_ppm_line_binary( Write *write, VipsPel *p )
|
||||||
|
{
|
||||||
|
if( vips__file_write( p, VIPS_IMAGE_SIZEOF_LINE( write->in ), 1,
|
||||||
|
write->fp ) )
|
||||||
|
return( -1 );
|
||||||
|
|
||||||
|
return( 0 );
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
write_ppm_line_binary_squash( Write *write, VipsPel *p )
|
||||||
|
{
|
||||||
|
int x;
|
||||||
|
int bits;
|
||||||
|
int n_bits;
|
||||||
|
|
||||||
|
bits = 0;
|
||||||
|
n_bits = 0;
|
||||||
|
for( x = 0; x < write->in->Xsize; x++ ) {
|
||||||
|
bits <<= 1;
|
||||||
|
n_bits += 1;
|
||||||
|
bits |= p[x] ? 0 : 1;
|
||||||
|
|
||||||
|
if( n_bits == 8 ) {
|
||||||
|
if( fputc( bits, write->fp ) == EOF ) {
|
||||||
|
vips_error_system( errno, "vips2ppm",
|
||||||
|
"%s", _( "write error" ) );
|
||||||
|
return( -1 );
|
||||||
|
}
|
||||||
|
|
||||||
|
bits = 0;
|
||||||
|
n_bits = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Flush any remaining bits in this line.
|
||||||
|
*/
|
||||||
|
if( n_bits ) {
|
||||||
|
if( fputc( bits, write->fp ) == EOF ) {
|
||||||
|
vips_error_system( errno, "vips2ppm",
|
||||||
|
"%s", _( "write error" ) );
|
||||||
|
return( -1 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return( 0 );
|
||||||
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
write_ppm_block( VipsRegion *region, VipsRect *area, void *a )
|
write_ppm_block( VipsRegion *region, VipsRect *area, void *a )
|
||||||
{
|
{
|
||||||
@ -630,7 +696,7 @@ write_ppm_block( VipsRegion *region, VipsRect *area, void *a )
|
|||||||
for( i = 0; i < area->height; i++ ) {
|
for( i = 0; i < area->height; i++ ) {
|
||||||
VipsPel *p = VIPS_REGION_ADDR( region, 0, area->top + i );
|
VipsPel *p = VIPS_REGION_ADDR( region, 0, area->top + i );
|
||||||
|
|
||||||
if( write->fn( write->in, write->fp, p ) )
|
if( write->fn( write, p ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -638,7 +704,7 @@ write_ppm_block( VipsRegion *region, VipsRect *area, void *a )
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
write_ppm( Write *write, int ascii )
|
write_ppm( Write *write, gboolean ascii, gboolean squash )
|
||||||
{
|
{
|
||||||
VipsImage *in = write->in;
|
VipsImage *in = write->in;
|
||||||
|
|
||||||
@ -650,8 +716,12 @@ write_ppm( Write *write, int ascii )
|
|||||||
magic = "PF";
|
magic = "PF";
|
||||||
else if( in->BandFmt == VIPS_FORMAT_FLOAT && in->Bands == 1 )
|
else if( in->BandFmt == VIPS_FORMAT_FLOAT && in->Bands == 1 )
|
||||||
magic = "Pf";
|
magic = "Pf";
|
||||||
|
else if( in->Bands == 1 && ascii && squash )
|
||||||
|
magic = "P1";
|
||||||
else if( in->Bands == 1 && ascii )
|
else if( in->Bands == 1 && ascii )
|
||||||
magic = "P2";
|
magic = "P2";
|
||||||
|
else if( in->Bands == 1 && !ascii && squash )
|
||||||
|
magic = "P4";
|
||||||
else if( in->Bands == 1 && !ascii )
|
else if( in->Bands == 1 && !ascii )
|
||||||
magic = "P5";
|
magic = "P5";
|
||||||
else if( in->Bands == 3 && ascii )
|
else if( in->Bands == 3 && ascii )
|
||||||
@ -666,6 +736,7 @@ write_ppm( Write *write, int ascii )
|
|||||||
fprintf( write->fp, "#vips2ppm - %s\n", ctime( &timebuf ) );
|
fprintf( write->fp, "#vips2ppm - %s\n", ctime( &timebuf ) );
|
||||||
fprintf( write->fp, "%d %d\n", in->Xsize, in->Ysize );
|
fprintf( write->fp, "%d %d\n", in->Xsize, in->Ysize );
|
||||||
|
|
||||||
|
if( !squash )
|
||||||
switch( in->BandFmt ) {
|
switch( in->BandFmt ) {
|
||||||
case VIPS_FORMAT_UCHAR:
|
case VIPS_FORMAT_UCHAR:
|
||||||
fprintf( write->fp, "%d\n", UCHAR_MAX );
|
fprintf( write->fp, "%d\n", UCHAR_MAX );
|
||||||
@ -695,7 +766,14 @@ write_ppm( Write *write, int ascii )
|
|||||||
g_assert( 0 );
|
g_assert( 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
write->fn = ascii ? write_ppm_line_ascii : write_ppm_line_binary;
|
if( squash )
|
||||||
|
write->fn = ascii ?
|
||||||
|
write_ppm_line_ascii_squash :
|
||||||
|
write_ppm_line_binary_squash;
|
||||||
|
else
|
||||||
|
write->fn = ascii ?
|
||||||
|
write_ppm_line_ascii :
|
||||||
|
write_ppm_line_binary;
|
||||||
|
|
||||||
if( vips_sink_disc( write->in, write_ppm_block, write ) )
|
if( vips_sink_disc( write->in, write_ppm_block, write ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
@ -704,7 +782,8 @@ write_ppm( Write *write, int ascii )
|
|||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
vips__ppm_save( VipsImage *in, const char *filename, gboolean ascii )
|
vips__ppm_save( VipsImage *in, const char *filename,
|
||||||
|
gboolean ascii, gboolean squash )
|
||||||
{
|
{
|
||||||
Write *write;
|
Write *write;
|
||||||
|
|
||||||
@ -714,20 +793,29 @@ vips__ppm_save( VipsImage *in, const char *filename, gboolean ascii )
|
|||||||
vips_image_pio_input( in ) )
|
vips_image_pio_input( in ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
/* We can only write >8 bit binary images in float.
|
if( ascii &&
|
||||||
|
in->BandFmt == VIPS_FORMAT_FLOAT ) {
|
||||||
|
vips_warn( "vips2ppm",
|
||||||
|
"%s", _( "float images must be binary -- "
|
||||||
|
"disabling ascii" ) );
|
||||||
|
ascii = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* One bit images must come from a 8 bit, one band source.
|
||||||
*/
|
*/
|
||||||
if( vips_format_sizeof( in->BandFmt ) > 1 &&
|
if( squash &&
|
||||||
!ascii &&
|
(in->Bands != 1 ||
|
||||||
in->BandFmt != VIPS_FORMAT_FLOAT ) {
|
in->BandFmt != VIPS_FORMAT_UCHAR) ) {
|
||||||
vips_error( "vips2ppm",
|
vips_warn( "vips2ppm",
|
||||||
"%s", _( "binary >8 bit images must be float" ) );
|
"%s", _( "can only squash 1 band uchar images -- "
|
||||||
return( -1 );
|
"disabling squash" ) );
|
||||||
|
squash = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if( !(write = write_new( in, filename )) )
|
if( !(write = write_new( in, filename )) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
if( write_ppm( write, ascii ) ) {
|
if( write_ppm( write, ascii, squash ) ) {
|
||||||
write_destroy( write );
|
write_destroy( write );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
@ -42,7 +42,7 @@ VipsForeignFlags vips__ppm_flags( const char *filename );
|
|||||||
extern const char *vips__ppm_suffs[];
|
extern const char *vips__ppm_suffs[];
|
||||||
|
|
||||||
int vips__ppm_save( VipsImage *in, const char *filename,
|
int vips__ppm_save( VipsImage *in, const char *filename,
|
||||||
gboolean ascii );
|
gboolean ascii, gboolean squash );
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
|
@ -54,6 +54,7 @@ typedef struct _VipsForeignSavePpm {
|
|||||||
|
|
||||||
char *filename;
|
char *filename;
|
||||||
gboolean ascii;
|
gboolean ascii;
|
||||||
|
gboolean squash;
|
||||||
} VipsForeignSavePpm;
|
} VipsForeignSavePpm;
|
||||||
|
|
||||||
typedef VipsForeignSaveClass VipsForeignSavePpmClass;
|
typedef VipsForeignSaveClass VipsForeignSavePpmClass;
|
||||||
@ -71,7 +72,8 @@ vips_foreign_save_ppm_build( VipsObject *object )
|
|||||||
build( object ) )
|
build( object ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
if( vips__ppm_save( save->ready, ppm->filename, ppm->ascii ) )
|
if( vips__ppm_save( save->ready, ppm->filename,
|
||||||
|
ppm->ascii, ppm->squash ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
@ -128,6 +130,13 @@ vips_foreign_save_ppm_class_init( VipsForeignSavePpmClass *class )
|
|||||||
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
||||||
G_STRUCT_OFFSET( VipsForeignSavePpm, ascii ),
|
G_STRUCT_OFFSET( VipsForeignSavePpm, ascii ),
|
||||||
FALSE );
|
FALSE );
|
||||||
|
|
||||||
|
VIPS_ARG_BOOL( class, "squash", 11,
|
||||||
|
_( "Squash" ),
|
||||||
|
_( "save as one bit" ),
|
||||||
|
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
||||||
|
G_STRUCT_OFFSET( VipsForeignSavePpm, squash ),
|
||||||
|
FALSE );
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -144,8 +153,9 @@ vips_foreign_save_ppm_init( VipsForeignSavePpm *ppm )
|
|||||||
* Optional arguments:
|
* Optional arguments:
|
||||||
*
|
*
|
||||||
* @ascii: save as ASCII rather than binary
|
* @ascii: save as ASCII rather than binary
|
||||||
|
* @squash: squash 8-bit images down to one bit
|
||||||
*
|
*
|
||||||
* Write a VIPS image to a file as PPM. It can write 8, 16 or
|
* Write a VIPS image to a file as PPM. It can write 1, 8, 16 or
|
||||||
* 32 bit unsigned integer images, float images, colour or monochrome,
|
* 32 bit unsigned integer images, float images, colour or monochrome,
|
||||||
* stored as binary or ASCII.
|
* stored as binary or ASCII.
|
||||||
* Integer images of more than 8 bits can only be stored in ASCII.
|
* Integer images of more than 8 bits can only be stored in ASCII.
|
||||||
@ -156,10 +166,10 @@ vips_foreign_save_ppm_init( VipsForeignSavePpm *ppm )
|
|||||||
* Set @ascii to %TRUE to write as human-readable ASCII. Normally data is
|
* Set @ascii to %TRUE to write as human-readable ASCII. Normally data is
|
||||||
* written in binary.
|
* written in binary.
|
||||||
*
|
*
|
||||||
* The storage format is indicated by a filename extension. Use one of .pbm,
|
* Set @squash to %TRUE to squash 8-bit images down to one bit. The saver does
|
||||||
* .pgm, .ppm, .pfm.
|
* no dithering, that's up to you.
|
||||||
*
|
*
|
||||||
* See also: vips_image_write_file().
|
* See also: vips_image_write_to_file().
|
||||||
*
|
*
|
||||||
* Returns: 0 on success, -1 on error.
|
* Returns: 0 on success, -1 on error.
|
||||||
*/
|
*/
|
||||||
|
@ -725,9 +725,9 @@ vips__file_write( void *data, size_t size, size_t nmemb, FILE *stream )
|
|||||||
return( 0 );
|
return( 0 );
|
||||||
|
|
||||||
if( (n = fwrite( data, size, nmemb, stream )) != nmemb ) {
|
if( (n = fwrite( data, size, nmemb, stream )) != nmemb ) {
|
||||||
vips_error( "vips__file_write",
|
vips_error_system( errno, "vips__file_write",
|
||||||
_( "write error (%zd out of %zd blocks written) "
|
_( "write error (%zd out of %zd blocks written)" ),
|
||||||
"... disc full?" ), n, nmemb );
|
n, nmemb );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user