2016-12-16 05:59:08 +08:00
|
|
|
from ctypes import c_void_p
|
|
|
|
|
|
|
|
|
2017-01-19 15:39:46 +08:00
|
|
|
class CPointerBase:
|
2016-12-16 05:59:08 +08:00
|
|
|
"""
|
|
|
|
Base class for objects that have a pointer access property
|
|
|
|
that controls access to the underlying C pointer.
|
|
|
|
"""
|
|
|
|
_ptr = None # Initially the pointer is NULL.
|
|
|
|
ptr_type = c_void_p
|
|
|
|
destructor = None
|
|
|
|
null_ptr_exception_class = AttributeError
|
|
|
|
|
|
|
|
@property
|
|
|
|
def ptr(self):
|
|
|
|
# Raise an exception if the pointer isn't valid so that NULL pointers
|
|
|
|
# aren't passed to routines -- that's very bad.
|
|
|
|
if self._ptr:
|
|
|
|
return self._ptr
|
|
|
|
raise self.null_ptr_exception_class('NULL %s pointer encountered.' % self.__class__.__name__)
|
|
|
|
|
|
|
|
@ptr.setter
|
|
|
|
def ptr(self, ptr):
|
|
|
|
# Only allow the pointer to be set with pointers of the compatible
|
|
|
|
# type or None (NULL).
|
|
|
|
if not (ptr is None or isinstance(ptr, self.ptr_type)):
|
|
|
|
raise TypeError('Incompatible pointer type: %s.' % type(ptr))
|
|
|
|
self._ptr = ptr
|
|
|
|
|
|
|
|
def __del__(self):
|
|
|
|
"""
|
|
|
|
Free the memory used by the C++ object.
|
|
|
|
"""
|
|
|
|
if self.destructor and self._ptr:
|
|
|
|
try:
|
|
|
|
self.destructor(self.ptr)
|
2018-07-06 23:26:14 +08:00
|
|
|
except (AttributeError, ImportError, TypeError):
|
2016-12-16 05:59:08 +08:00
|
|
|
pass # Some part might already have been garbage collected
|