more cpp tinkering

This commit is contained in:
John Cupitt 2014-10-24 14:57:01 +01:00
parent ceb143a22a
commit 66445c359d

View File

@ -6,8 +6,36 @@
*/ */
#include <list> #include <list>
#include <string>
#include <iosfwd>
#include <exception>
#include <vips/vips.h> #include <vips/vips.h>
#include <vips/debug.h>
class VError : public std::exception {
std::string _what;
public:
VError( std::string what ) : _what( what ) {}
VError() : _what( vips_error_buffer() ) {}
virtual ~VError() throw() {}
// Extract string
virtual const char *what() const throw() { return _what.c_str(); }
void ostream_print( std::ostream & ) const;
};
inline std::ostream &operator<<( std::ostream &file, const VError &err )
{
err.ostream_print( file );
return( file );
}
void VError::ostream_print( std::ostream &file ) const
{
file << _what;
}
enum VSteal { enum VSteal {
NOSTEAL = 0, NOSTEAL = 0,
@ -17,80 +45,120 @@ enum VSteal {
class VObject class VObject
{ {
private: private:
GObject *gobject; // can be NULL, see eg. VObject()
VipsObject *vobject;
public: public:
VObject( GObject *new_gobject, VSteal steal = STEAL ) : VObject( VipsObject *new_vobject, VSteal steal = STEAL ) :
gobject( new_gobject ) vobject( new_vobject )
{ {
g_assert( new_vobject &&
VIPS_IS_OBJECT( new_vobject ) );
printf( "VObject constructor, obj = %p, steal = %d\n", printf( "VObject constructor, obj = %p, steal = %d\n",
new_gobject, steal ); new_vobject, steal );
if( !steal ) { if( !steal ) {
printf( " reffing object\n" ); printf( " reffing object\n" );
g_object_ref( gobject ); g_object_ref( vobject );
} }
} }
VObject() :
vobject( 0 )
{
}
// copy constructor // copy constructor
VObject( const VObject &vobject ) : VObject( const VObject &a ) :
gobject( vobject.gobject ) vobject( a.vobject )
{ {
g_assert( VIPS_IS_OBJECT( a.vobject ) );
printf( "VObject copy constructor, obj = %p\n", printf( "VObject copy constructor, obj = %p\n",
gobject ); vobject );
g_object_ref( gobject ); g_object_ref( vobject );
printf( " reffing object\n" ); printf( " reffing object\n" );
} }
// assignment ... we must delete the old ref // assignment ... we must delete the old ref
// old can be NULL, new must not be NULL
VObject &operator=( const VObject &a ) VObject &operator=( const VObject &a )
{ {
GObject *old_gobject; VipsObject *old_vobject;
printf( "VObject assignment\n" ); printf( "VObject assignment\n" );
printf( " reffing %p\n", a.gobject ); printf( " reffing %p\n", a.vobject );
printf( " unreffing %p\n", gobject ); printf( " unreffing %p\n", vobject );
g_assert( !vobject ||
VIPS_IS_OBJECT( vobject ) );
g_assert( a.vobject &&
VIPS_IS_OBJECT( a.vobject ) );
// delete the old ref at the end ... otherwise "a = a;" could // delete the old ref at the end ... otherwise "a = a;" could
// unref before reffing again // unref before reffing again
old_gobject = gobject; old_vobject = vobject;
gobject = a.gobject; vobject = a.vobject;
g_object_ref( gobject ); g_object_ref( vobject );
g_object_unref( old_gobject ); if( old_vobject )
g_object_unref( old_vobject );
return( *this ); return( *this );
} }
// this mustn't be virtual: we want this class to only be a pointer
~VObject() ~VObject()
{ {
printf( "VObject destructor\n" ); printf( "VObject destructor\n" );
printf( " unreffing %p\n", gobject ); printf( " unreffing %p\n", vobject );
g_object_unref( gobject ); g_assert( !vobject ||
VIPS_IS_OBJECT( vobject ) );
if( vobject )
g_object_unref( vobject );
} }
GObject *get_object() VipsObject *get_object()
{ {
return( gobject ); g_assert( !vobject ||
VIPS_IS_OBJECT( vobject ) );
return( vobject );
} }
}; };
class VImage;
class VOption;
class VOption class VOption
{ {
private: private:
struct Pair { struct Pair {
const char *name; const char *name;
GValue value = {0, };
Pair( const char *name ) : name( name ) // the thing we pass to VipsOperation
GValue value;
// we need to box and unbox VImage ... keep a pointer to the
// VImage from C++ here
VImage *vimage;
// keep a VipsImage pointer from VipsOperation here
VipsImage *vips_image;
Pair( const char *name ) :
name( name ), vimage( 0 ), vips_image( 0 )
{ {
G_VALUE_TYPE( &value ) = 0;
} }
~Pair() ~Pair()
{ {
g_value_unset( &value ) g_value_unset( &value );
}
} }
};
std::list<Pair *> options; std::list<Pair *> options;
@ -99,65 +167,29 @@ public:
{ {
} }
~VOption() virtual ~VOption();
{
std::list<Pair *>::iterator i;
for( i = options.begin(); i != options.end(); i++ ) VOption set( const char *name, const char *value );
delete *i; VOption set( const char *name, int value );
} VOption set( const char *name, VImage value );
VOption set( const char *name, VImage *value );
VOption set( const char *name, const char *value ) int set_operation( VipsOperation *operation );
{ int get_operation( VipsOperation *operation );
Pair *pair = new Pair( name );
g_value_init( &pair->value, G_TYPE_STRING ); };
g_value_set_string( &pair->value, value );
options.push_back( pair );
return( this );
}
VOption set( const char *name, int value )
{
Pair *pair = new Pair( name );
g_value_init( &pair->value, G_TYPE_INT );
g_value_set_int( &pair->value, value );
options.push_back( pair );
return( this );
}
VOption set( const char *name, VImage value )
{
Pair *pair = new Pair( name );
g_value_init( &pair->value, G_TYPE_OBJECT );
g_value_set_object( &pair->value, value );
options.push_back( pair );
return( this );
}
VOption set( const char *name, VImage *value )
{
Pair *pair = new Pair( name );
g_value_init( &pair->value, G_TYPE_POINTER );
g_value_set_pointer( &pair->value, value );
options.push_back( pair );
return( this );
}
}
class VImage : VObject class VImage : VObject
{ {
public: public:
VImage( VipsImage *image, VSteal steal = STEAL ) : VImage( VipsImage *image, VSteal steal = STEAL ) :
VObject( (GObject *) image, steal ) VObject( (VipsObject *) image, steal )
{
}
// an empty (NULL) VImage, eg. "VImage a;"
VImage() :
VObject( 0 )
{ {
} }
@ -166,12 +198,177 @@ public:
return( (VipsImage *) VObject::get_object() ); return( (VipsImage *) VObject::get_object() );
} }
VImage::VOption *option() static VOption *option()
{ {
return( new VOption() ); return( new VOption() );
} }
void call_option_string( const char *operation_name,
const char *option_string, VOption *options = 0 )
throw( VError );
void call( const char *operation_name, VOption *options = 0 )
throw( VError );
VImage new_from_file( const char *name, VOption *options = 0 ) VImage new_from_file( const char *name, VOption *options = 0 )
throw( VError );
void write_to_file( const char *name, VOption *options = 0 )
throw( VError );
VImage invert( VOption *options = 0 )
throw( VError );
};
VOption::~VOption()
{
std::list<Pair *>::iterator i;
for( i = options.begin(); i != options.end(); i++ )
delete *i;
}
VOption VOption::set( const char *name, const char *value )
{
Pair *pair = new Pair( name );
g_value_init( &pair->value, G_TYPE_STRING );
g_value_set_string( &pair->value, value );
options.push_back( pair );
return( *this );
}
VOption VOption::set( const char *name, int value )
{
Pair *pair = new Pair( name );
g_value_init( &pair->value, G_TYPE_INT );
g_value_set_int( &pair->value, value );
options.push_back( pair );
return( *this );
}
VOption VOption::set( const char *name, VImage value )
{
Pair *pair = new Pair( name );
// we need to unbox
g_value_init( &pair->value, G_TYPE_OBJECT );
g_value_set_pointer( &pair->value, value.get_image() );
options.push_back( pair );
return( *this );
}
VOption VOption::set( const char *name, VImage *value )
{
Pair *pair = new Pair( name );
// where we will write the VImage on success
pair->vimage = value;
// get VipsOperation to write the VipsImage * here
g_value_init( &pair->value, G_TYPE_POINTER );
g_value_set_pointer( &pair->value, &pair->vips_image );
options.push_back( pair );
return( *this );
}
// walk the options and set props on the operation
int VOption::set_operation( VipsOperation *operation )
{
std::list<Pair *>::iterator i;
for( i = options.begin(); i != options.end(); i++ )
if( vips_object_set( VIPS_OBJECT( operation ),
(*i)->name, &(*i)->value,
NULL ) )
return( -1 );
return( 0 );
}
// walk the options and do any processing needed for output objects
int VOption::get_operation( VipsOperation *operation )
{
std::list<Pair *>::iterator i;
for( i = options.begin(); i != options.end(); i++ )
if( (*i)->vimage )
*((*i)->vimage) = VImage( (*i)->vips_image );
return( 0 );
}
void VImage::call_option_string( const char *operation_name,
const char *option_string, VOption *options )
throw( VError )
{
VipsOperation *operation;
VIPS_DEBUG_MSG( "vips_call_by_name: starting for %s ...\n",
operation_name );
if( !(operation = vips_operation_new( operation_name )) ) {
if( options )
delete options;
throw( VError() );
}
/* Set str options before vargs options, so the user can't
* override things we set deliberately.
*/
if( option_string &&
vips_object_set_from_string( VIPS_OBJECT( operation ),
option_string ) ) {
vips_object_unref_outputs( VIPS_OBJECT( operation ) );
g_object_unref( operation );
delete options;
throw( VError() );
}
if( options &&
options->set_operation( operation ) ) {
g_object_unref( operation );
delete options;
throw( VError() );
}
/* Build from cache.
*/
if( vips_cache_operation_buildp( &operation ) ) {
vips_object_unref_outputs( VIPS_OBJECT( operation ) );
delete options;
throw( VError() );
}
/* Walk args again, writing output.
*/
if( options &&
options->get_operation( operation ) ) {
vips_object_unref_outputs( VIPS_OBJECT( operation ) );
delete options;
throw( VError() );
}
/* The operation we have built should now have been reffed by
* one of its arguments or have finished its work. Either
* way, we can unref.
*/
g_object_unref( operation );
}
void VImage::call( const char *operation_name, VOption *options )
throw( VError )
{
call_option_string( operation_name, NULL, options );
}
VImage VImage::new_from_file( const char *name, VOption *options )
throw( VError )
{ {
char filename[VIPS_PATH_MAX]; char filename[VIPS_PATH_MAX];
char option_string[VIPS_PATH_MAX]; char option_string[VIPS_PATH_MAX];
@ -185,17 +382,14 @@ public:
throw VError(); throw VError();
} }
if( call_option_string( operation_name, option_string, call_option_string( operation_name, option_string,
options.set( "filename", filename ). (options ? options : VImage::option())->
set( "out", &out ) ) ) { set( "filename", filename )->
delete options; set( "out", &out ) );
throw VError();
} }
delete options; void VImage::write_to_file( const char *name, VOption *options )
} throw( VError )
void write_to_file( const char *name, VOption *options = 0 )
{ {
char filename[VIPS_PATH_MAX]; char filename[VIPS_PATH_MAX];
char option_string[VIPS_PATH_MAX]; char option_string[VIPS_PATH_MAX];
@ -207,33 +401,25 @@ public:
throw VError(); throw VError();
} }
if( call_option_string( operation_name, option_string, call_option_string( operation_name, option_string,
options.set( "in", this ). (options ? options : VImage::option())->
set( "filename", filename ) ) ) { set( "in", this )->
delete options; set( "filename", filename ) );
throw VError();
} }
delete options; VImage VImage::invert( VOption *options )
} throw( VError )
VImage invert( VOption *options = 0 )
{ {
VImage out; VImage out;
if( call( "invert", call( "invert",
options.set( "in", this ). (options ? options : VImage::option())->
set( "out", &out ) ) ) { set( "in", this )->
delete options; set( "out", &out ) );
throw VError();
}
delete options;
return( out ); return( out );
} }
};
int int
main( int argc, char **argv ) main( int argc, char **argv )
{ {