glib: Fix crash when freeing lists with elements typed as interfaces

If a GList or a GSList had its element type set to a GInterface, and if
the elements were marked as owned, it would end up freeing those
elements with g_free(), which would lead to a crash.

They need to be unreffed with g_object_unref, but the criteria for that
was whether the element type is assignable to GLib.Object. This is not
true for GInterface types.

We now first check if the element type is an opaque. If not, and if it's
assignable to GLib.IWrapper, we then use g_object_unref.

From what I can see, all GLib.IWrapper subclasses that not opaque can be
unreffed with g_object_unref.
This commit is contained in:
Bertrand Lorentz 2012-11-18 16:06:34 +01:00
parent 91474fe1f1
commit ab61fbccbd

View file

@ -188,14 +188,17 @@ namespace GLib {
public void Empty () public void Empty ()
{ {
if (elements_owned) if (elements_owned) {
for (uint i = 0; i < Count; i++) for (uint i = 0; i < Count; i++) {
if (typeof (GLib.Object).IsAssignableFrom (element_type)) if (typeof (GLib.Opaque).IsAssignableFrom (element_type)) {
g_object_unref (NthData (i));
else if (typeof (GLib.Opaque).IsAssignableFrom (element_type))
GLib.Opaque.GetOpaque (NthData (i), element_type, true).Dispose (); GLib.Opaque.GetOpaque (NthData (i), element_type, true).Dispose ();
else } else if (typeof (GLib.IWrapper).IsAssignableFrom (element_type)) {
g_object_unref (NthData (i));
} else {
g_free (NthData (i)); g_free (NthData (i));
}
}
}
if (managed) if (managed)
FreeList (); FreeList ();