diff options
author | David Robillard <d@drobilla.net> | 2012-04-24 22:06:24 +0000 |
---|---|---|
committer | David Robillard <d@drobilla.net> | 2012-04-24 22:06:24 +0000 |
commit | 3723d115a6f4d30f652957d131c2a6ea56ad1dd0 (patch) | |
tree | 53839c92e3d8f4530ccc16959cdd6a6e820bb78d /src/x11_in_gtk2.c | |
parent | d8df0d21ab8f2d28c8b0e32670594fa079425a1d (diff) | |
download | suil-3723d115a6f4d30f652957d131c2a6ea56ad1dd0.tar.gz suil-3723d115a6f4d30f652957d131c2a6ea56ad1dd0.tar.bz2 suil-3723d115a6f4d30f652957d131c2a6ea56ad1dd0.zip |
Fix crashes when wrapper widget is destroyed by toolkit before suil cleanup
function is called.
git-svn-id: http://svn.drobilla.net/lad/trunk/suil@4264 a436a847-0d15-0410-975c-d299462d15a1
Diffstat (limited to 'src/x11_in_gtk2.c')
-rw-r--r-- | src/x11_in_gtk2.c | 43 |
1 files changed, 37 insertions, 6 deletions
diff --git a/src/x11_in_gtk2.c b/src/x11_in_gtk2.c index 99308e7..e7b88ed 100644 --- a/src/x11_in_gtk2.c +++ b/src/x11_in_gtk2.c @@ -29,6 +29,7 @@ typedef struct _SuilX11WrapperClass SuilX11WrapperClass; struct _SuilX11Wrapper { GtkSocket socket; GtkPlug* plug; + SuilWrapper* wrapper; SuilInstance* instance; }; @@ -40,10 +41,28 @@ GType suil_x11_wrapper_get_type(void); // Accessor for SUIL_TYPE_X11_WRAPPER G_DEFINE_TYPE(SuilX11Wrapper, suil_x11_wrapper, GTK_TYPE_SOCKET) +static gboolean +on_plug_removed(GtkSocket* sock, gpointer data) +{ + SuilX11Wrapper* const self = SUIL_X11_WRAPPER(sock); + + if (self->instance->handle) { + self->instance->descriptor->cleanup(self->instance->handle); + self->instance->handle = NULL; + } + + self->plug = NULL; + return TRUE; +} + static void -wrap_widget_dispose(GObject* gobject) +suil_x11_wrapper_finalize(GObject* gobject) { - G_OBJECT_CLASS(suil_x11_wrapper_parent_class)->dispose(gobject); + SuilX11Wrapper* const self = SUIL_X11_WRAPPER(gobject); + + self->wrapper->impl = NULL; + + G_OBJECT_CLASS(suil_x11_wrapper_parent_class)->finalize(gobject); } static void @@ -51,7 +70,7 @@ suil_x11_wrapper_class_init(SuilX11WrapperClass* klass) { GObjectClass* const gobject_class = G_OBJECT_CLASS(klass); - gobject_class->dispose = wrap_widget_dispose; + gobject_class->finalize = suil_x11_wrapper_finalize; } static void @@ -85,6 +104,7 @@ wrapper_wrap(SuilWrapper* wrapper, SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(wrapper->impl); instance->host_widget = GTK_WIDGET(wrap); + wrap->wrapper = wrapper; wrap->instance = instance; g_signal_connect_after(G_OBJECT(wrap), @@ -92,15 +112,24 @@ wrapper_wrap(SuilWrapper* wrapper, G_CALLBACK(suil_x11_wrapper_realize), NULL); + g_signal_connect(G_OBJECT(wrap), + "plug-removed", + G_CALLBACK(on_plug_removed), + NULL); + return 0; } static void wrapper_free(SuilWrapper* wrapper) { - free(wrapper); + if (wrapper->impl) { + SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER(wrapper->impl); + gtk_object_destroy(GTK_OBJECT(wrap)); + } } + SUIL_API SuilWrapper* suil_wrapper_new(SuilHost* host, @@ -110,12 +139,14 @@ suil_wrapper_new(SuilHost* host, unsigned n_features) { SuilWrapper* wrapper = (SuilWrapper*)malloc(sizeof(SuilWrapper)); - wrapper->wrap = wrapper_wrap; - wrapper->free = wrapper_free; + wrapper->wrap = wrapper_wrap; + wrapper->free = wrapper_free; SuilX11Wrapper* const wrap = SUIL_X11_WRAPPER( g_object_new(SUIL_TYPE_X11_WRAPPER, NULL)); + wrap->wrapper = NULL; + wrapper->impl = wrap; wrapper->resize.handle = wrap; wrapper->resize.ui_resize = wrapper_resize; |