summaryrefslogtreecommitdiffstats
path: root/src/x11_in_gtk2.c
diff options
context:
space:
mode:
authorDavid Robillard <d@drobilla.net>2012-04-24 22:06:24 +0000
committerDavid Robillard <d@drobilla.net>2012-04-24 22:06:24 +0000
commit3723d115a6f4d30f652957d131c2a6ea56ad1dd0 (patch)
tree53839c92e3d8f4530ccc16959cdd6a6e820bb78d /src/x11_in_gtk2.c
parentd8df0d21ab8f2d28c8b0e32670594fa079425a1d (diff)
downloadsuil-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.c43
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;