Software: Apache/2.0.54 (Fedora). PHP/5.0.4 uname -a: Linux mina-info.me 2.6.17-1.2142_FC4smp #1 SMP Tue Jul 11 22:57:02 EDT 2006 i686 uid=48(apache) gid=48(apache) groups=48(apache) Safe-mode: OFF (not secure) /usr/share/doc/swig-1.3.24/Manual/ drwxr-xr-x |
Viewing file: Select action/file-type: 11 Customization Features
11.1 Exception handling with %exceptionThe %exception directive allows you to define a general purpose exception handler. For example, you can specify the following: %exception { try { $action } catch (RangeError) { PyErr_SetString(PyExc_IndexError,"index out-of-bounds"); return NULL; } } When defined, the code enclosed in braces is inserted directly into the low-level wrapper functions. The special symbol $action gets replaced with the actual operation to be performed (a function call, method invocation, attribute access, etc.). An exception handler remains in effect until it is explicitly deleted. This is done by using either %exception or %noexception with no code. For example: %exception; // Deletes any previously defined handler Compatibility note: Previous versions of SWIG used a special directive %except for exception handling. That directive is still supported but is deprecated--%exception provides the same functionality, but is substantially more flexible. 11.1.1 Handling exceptions in C codeC has no formal exception handling mechanism so there are several approaches that might be used. A somewhat common technique is to simply set a special error code. For example: /* File : except.c */ static char error_message[256]; static int error_status = 0; void throw_exception(char *msg) { strncpy(error_message,msg,256); error_status = 1; } void clear_exception() { error_status = 0; } char *check_exception() { if (error_status) return error_message; else return NULL; } To use these functions, functions simply call throw_exception() to indicate an error occurred. For example : double inv(double x) { if (x != 0) return 1.0/x; else { throw_exception("Division by zero"); return 0; } } To catch the exception, you can write a simple exception handler such as the following (shown for Perl5) : %exception { char *err; clear_exception(); $action if ((err = check_exception())) { croak(err); } } In this case, when an error occurs, it is translated into a Perl error. 11.1.2 Exception handling with longjmp()Exception handling can also be added to C code using the <setjmp.h> library. Here is a minimalistic implementation that relies on the C preprocessor :/* File : except.c Just the declaration of a few global variables we're going to use */ #include <setjmp.h> jmp_buf exception_buffer; int exception_status; /* File : except.h */ #include <setjmp.h> extern jmp_buf exception_buffer; extern int exception_status; #define try if ((exception_status = setjmp(exception_buffer)) == 0) #define catch(val) else if (exception_status == val) #define throw(val) longjmp(exception_buffer,val) #define finally else /* Exception codes */ #define RangeError 1 #define DivisionByZero 2 #define OutOfMemory 3 Now, within a C program, you can do the following : double inv(double x) { if (x) return 1.0/x; else throw(DivisionByZero); } Finally, to create a SWIG exception handler, write the following : Note: This implementation is only intended to illustrate the general idea. To make it work better, you'll need to modify it to handle nested try declarations.%{ #include "except.h" %} %exception { try { $action } catch(RangeError) { croak("Range Error"); } catch(DivisionByZero) { croak("Division by zero"); } catch(OutOfMemory) { croak("Out of memory"); } finally { croak("Unknown exception"); } } 11.1.3 Handling C++ exceptionsHandling C++ exceptions is also straightforward. For example: %exception { try { $action } catch(RangeError) { croak("Range Error"); } catch(DivisionByZero) { croak("Division by zero"); } catch(OutOfMemory) { croak("Out of memory"); } catch(...) { croak("Unknown exception"); } } The exception types need to be declared as classes elsewhere, possibly in a header file : class RangeError {}; class DivisionByZero {}; class OutOfMemory {}; 11.1.4 Defining different exception handlersBy default, the %exception directive creates an exception handler that is used for all wrapper functions that follow it. Unless there is a well-defined (and simple) error handling mechanism in place, defining one universal exception handler may be unwieldy and result in excessive code bloat since the handler is inlined into each wrapper function.To fix this, you can be more selective about how you use the %exception directive. One approach is to only place it around critical pieces of code. For example: More precise control over exception handling can be obtained by attaching an exception handler to specific declaration name. For example:%exception { ... your exception handler ... } /* Define critical operations that can throw exceptions here */ %exception; /* Define non-critical operations that don't throw exceptions */ In this case, the exception handler is only attached to declarations named "allocate". This would include both global and member functions. The names supplied to %exception follow the same rules as for %rename described in the section on Ambiguity resolution and renaming. For example, if you wanted to define an exception handler for a specific class, you might write this:%exception allocate { try { $action } catch (MemoryError) { croak("Out of memory"); } } When a class prefix is supplied, the exception handler is applied to the corresponding declaration in the specified class as well as for identically named functions appearing in derived classes.%exception Object::allocate { try { $action } catch (MemoryError) { croak("Out of memory"); } } %exception can even be used to pinpoint a precise declaration when overloading is used. For example: Attaching exceptions to specific declarations is a good way to reduce code bloat. It can also be a useful way to attach exceptions to specific parts of a header file. For example:%exception Object::allocate(int) { try { $action } catch (MemoryError) { croak("Out of memory"); } } Compatibility note: The %exception directive replaces the functionality provided by the deprecated "except" typemap. The typemap would allow exceptions to be thrown in the target language based on the return type of a function and was intended to be a mechanism for pinpointing specific declarations. However, it never really worked that well and the new %exception directive is much better.%module example %{ #include "someheader.h" %} // Define a few exception handlers for specific declarations %exception Object::allocate(int) { try { $action } catch (MemoryError) { croak("Out of memory"); } } %exception Object::getitem { try { $action } catch (RangeError) { croak("Index out of range"); } } ... // Read a raw header file %include "someheader.h" 11.1.5 Using The SWIG exception libraryThe exception.i library file provides support for creating language independent exceptions in your interfaces. To use it, simply put an "%include exception.i" in your interface file. This creates a function SWIG_exception() that can be used to raise common scripting language exceptions in a portable manner. For example : // Language independent exception handler %include exception.i %exception { try { $action } catch(RangeError) { SWIG_exception(SWIG_ValueError, "Range Error"); } catch(DivisionByZero) { SWIG_exception(SWIG_DivisionByZero, "Division by zero"); } catch(OutOfMemory) { SWIG_exception(SWIG_MemoryError, "Out of memory"); } catch(...) { SWIG_exception(SWIG_RuntimeError,"Unknown exception"); } } As arguments, SWIG_exception() takes an error type code (an integer) and an error message string. The currently supported error types are : SWIG_MemoryError SWIG_IOError SWIG_RuntimeError SWIG_IndexError SWIG_TypeError SWIG_DivisionByZero SWIG_OverflowError SWIG_SyntaxError SWIG_ValueError SWIG_SystemError SWIG_UnknownError Since the SWIG_exception() function is defined at the C-level it can be used elsewhere in SWIG. This includes typemaps and helper functions. 11.2 Object ownership and %newobjectA common problem in some applications is managing proper ownership of objects. For example, consider a function like this:If you wrap the function blah(), SWIG has no idea that the return value is a newly allocated object. As a result, the resulting extension module may produce a memory leak (SWIG is conservative and will never delete objects unless it knows for certain that the returned object was newly created).Foo *blah() { Foo *f = new Foo(); return f; } To fix this, you can provide an extra hint to the code generator using the %newobject directive. For example: %newobject works exactly like %rename and %exception. In other words, you can attach it to class members and parameterized declarations as before. For example:%newobject blah; Foo *blah(); When %newobject is supplied, many language modules will arrange to take ownership of the return value. This allows the value to be automatically garbage-collected when it is no longer in use. However, this depends entirely on the target language (a language module may also choose to ignore the %newobject directive).%newobject ::blah(); // Only applies to global blah %newobject Object::blah(int,double); // Only blah(int,double) in Object %newobject *::copy; // Copy method in all classes ... Closely related to %newobject is a special typemap. The "newfree" typemap can be used to deallocate a newly allocated return value. It is only available on methods for which %newobject has been applied and is commonly used to clean-up string results. For example: In this case, the result of the function is a string in the target language. Since this string is a copy of the original result, the data returned by strdup() is no longer needed. The "newfree" typemap in the example simply releases this memory.%typemap(newfree) char * "free($1);"; ... %newobject strdup; ... char *strdup(const char *s); Compatibility note: Previous versions of SWIG had a special %new directive. However, unlike %newobject, it only applied to the next declaration. For example: For now this is still supported but is deprecated.%new char *strdup(const char *s); How to shoot yourself in the foot: The %newobject directive is not a declaration modifier like the old %new directive. Don't write code like this: The results might not be what you expect.%newobject char *strdup(const char *s); 11.3 Features and the %feature directiveBoth %exception and %newobject are examples of a more general purpose customization mechanism known as "features." A feature is simply a user-definable property that is attached to specific declarations in an interface file. Features are attached using the %feature directive. For example:In fact, the %exception and %newobject directives are really nothing more than macros involving %feature:%feature("except") Object::allocate { try { $action } catch (MemoryError) { croak("Out of memory"); } } %feature("new","1") *::copy; The %feature directive follows the same name matching rules as the %rename directive (which is in fact just a special form of %feature). This means that features can be applied with pinpoint accuracy to specific declarations if needed.#define %exception %feature("except") #define %newobject %feature("new","1") When a feature is defined, it is given a name and a value. Most commonly, the value is supplied after the declaration name as shown for the "except" example above. However, if the feature is simple, a value might be supplied as an extra argument as shown for the "new" feature. A feature stays in effect until it is explicitly disabled. A feature is disabled by supplying a %feature directive with no value. For example: %feature("except") Object::allocate; // Removes any previously defined feature If no declaration name is given, a global feature is defined. This feature is then attached to every declaration that follows. This is how global exception handlers are defined. For example: The %feature directive can be used with different syntax. The following are all equivalent:/* Define a global exception handler */ %feature("except") { try { $action } ... } ... bunch of declarations ... /* Disable the exception handler */ %feature("except"); The syntax in the first variation will generate the { } delimeters used whereas the other variations will not. The %feature directive also accepts XML style attributes in the same way that typemaps will. Any number of attributes can be specified. The following is the generic syntax for features:%feature("except") Object::method { $action }; %feature("except") Object::method %{ $action %}; %feature("except") Object::method " $action "; %feature("except","$action") Object::method; More than one attribute can be specified using a comma separated list. The Java module is an example that uses attributes in %feature("except"). The throws attribute specifies the name of a Java class to add to a proxy method's throws clause. In the following example, MyExceptionClass is the name of the Java class for adding to the throws clause.%feature("name","value", attribute1="AttibuteValue1") symbol; %feature("name", attribute1="AttibuteValue1") symbol {value}; %feature("name", attribute1="AttibuteValue1") symbol %{value%}; %feature("name", attribute1="AttibuteValue1") symbol "value"; %feature("except", throws="MyExceptionClass") Object::method { try { $action } catch (...) { ... code to throw a MyExceptionClass Java exception ... } }; Further details can be obtained from the Java exception handling section. 11.3.1 Features and default argumentsSWIG treats methods with default arguments as separate overloaded methods as detailed in the default arguments section. Any %feature targeting a method with default arguments will apply to all the extra overloaded methods that SWIG generates if the default arguments are specified in the feature. If the default arguments are not specified in the feature, then the feature will match that exact wrapper method only and not the extra overloaded methods that SWIG generates. For example: %feature("except") void hello(int i=0, double d=0.0); void hello(int i=0, double d=0.0); will apply the feature to all three wrapper methods, that is: void hello(int i, double d); void hello(int i); void hello(); If the default arguments are not specified in the feature: %feature("except") void hello(int i, double d); void hello(int i=0, double d=0.0); then the feature will only apply to this wrapper method: void hello(int i, double d); and not these wrapper methods: void hello(int i); void hello(); If compactdefaultargs are being used, then the difference between specifying or not specifying default arguments in a feature is not applicable as just one wrapper is generated. Compatibility note: The different behaviour of features specified with or without default arguments was introduced in SWIG-1.3.23 when the approach to wrapping methods with default arguments was changed. 11.3.2 Feature exampleAs has been shown earlier, the intended use for the %feature directive is as a highly flexible customization mechanism that can be used to annotate declarations with additional information for use by specific target language modules. Another example is in the Python module. You might use %feature to rewrite proxy/shadow class code as follows: Further details of %feature usage is described in the documentation for specific language modules.%module example %rename(bar_id) bar(int,double); // Rewrite bar() to allow some nice overloading %feature("shadow") Foo::bar(int) %{ def bar(*args): if len(args) == 3: return apply(examplec.Foo_bar_id,args) return apply(examplec.Foo_bar,args) %} class Foo { public: int bar(int x); int bar(int x, double y); } |
:: Command execute :: | |
--[ c99shell v. 1.0 pre-release build #16 powered by Captain Crunch Security Team | http://ccteam.ru | Generation time: 0.0029 ]-- |