Утечка памяти rsvg python в osx (ctypes?)

Я использую следующий код для чтения svg:

from ctypes import CDLL, POINTER, Structure, byref, util
from ctypes import c_bool, c_byte, c_void_p, c_int, c_double, c_uint32, c_char_p

class _PycairoContext(Structure):
    _fields_ = [("PyObject_HEAD", c_byte * object.__basicsize__),
                ("ctx", c_void_p),
                ("base", c_void_p)]

class _RsvgProps(Structure):
    _fields_ = [("width", c_int), ("height", c_int),
                ("em", c_double), ("ex", c_double)]

class _GError(Structure):
    _fields_ = [("domain", c_uint32), ("code", c_int), ("message", c_char_p)]


def _load_rsvg(rsvg_lib_path=None, gobject_lib_path=None):
    if rsvg_lib_path is None:
        rsvg_lib_path = util.find_library('rsvg-2')
    if gobject_lib_path is None:
        gobject_lib_path = util.find_library('gobject-2.0')
    l = CDLL(rsvg_lib_path)
    g = CDLL(gobject_lib_path)
    g.g_type_init()

    l.rsvg_handle_new_from_file.argtypes = [c_char_p, POINTER(POINTER(_GError))]
    l.rsvg_handle_new_from_file.restype = c_void_p
    l.rsvg_handle_render_cairo.argtypes = [c_void_p, c_void_p]
    l.rsvg_handle_render_cairo.restype = c_bool
    l.rsvg_handle_get_dimensions.argtypes = [c_void_p, POINTER(_RsvgProps)]

    return l    

_librsvg = _load_rsvg()


class Handle(object):
    def __init__(self, path):
        lib = _librsvg
        err = POINTER(_GError)()
        self.handle = lib.rsvg_handle_new_from_file(path, byref(err))
        if self.handle is None:
            gerr = err.contents
            raise Exception(gerr.message)
        self.props = _RsvgProps()
        lib.rsvg_handle_get_dimensions(self.handle, byref(self.props))

    def render_cairo(self, ctx):
        """Returns True is drawing succeeded."""
        z = _PycairoContext.from_address(id(ctx))
        return _librsvg.rsvg_handle_render_cairo(self.handle, z.ctx)

(из Ошибка с ctypes Python и librsvg)

Я вызываю img = Handle(path) и это приводит к утечке памяти. Я почти уверен, что это связано с неправильным использованием ctypes и указателей, но я не могу найти способ это исправить.


person user3663263    schedule 22.05.2014    source источник
comment
Похоже, это на правильном пути. Я попробовал код g_object_unref и получил segfault. Я также пытался передавать gobject и звонить оттуда, но также получил segfault.   -  person user3663263    schedule 22.05.2014


Ответы (1)


Согласно документации rsvg_handle_new, освободите дескриптор с помощью g_object_unref. Кроме того, если неудачный вызов выделяет GError, после того, как вы получите код и сообщение, вы должны устранить ошибку с помощью g_error_free.

from ctypes import *
from ctypes.util import find_library

_gobj = CDLL(find_library("gobject-2.0"))
_glib = CDLL(find_library("glib-2.0"))

class _GError(Structure):
    _fields_ = [("domain", c_uint32), 
                ("code", c_int), 
                ("message", c_char_p)]

_GErrorP = POINTER(_GError)

_glib.g_error_free.restype = None
_glib.g_error_free.argtypes = [_GErrorP]

_gobj.g_object_unref.restype = None
_gobj.g_object_unref.argtypes = [c_void_p]

Вы можете освободить дескриптор в методе __del__ класса Handle:

class Handle(object):
    _gobj = _gobj # keep a valid ref for module teardown

    # ...

    def __del__(self):
        if self.handle:
            self._gobj.g_object_unref(self.handle)
            self.handle = None
person Eryk Sun    schedule 23.05.2014