2003-09-17 Rachel Hestilow <rachel@nullenvoid.com>
* glib/DelegateWrapper.cs: Remove 'RemoveIfNotAlive' and revamp the memory management to use destroy notification. * generator/CallbackGen.cs: Do not generate the call to RemoveIfNotAlive. * gtk/GtkSharp.GtkClipboardGetFuncNative, GtkSharp.GtkClipboardClearFuncNative: Do not call RemoveIfNotAlive. svn path=/trunk/gtk-sharp/; revision=18163
This commit is contained in:
parent
9ff3497076
commit
69e51261af
5 changed files with 70 additions and 28 deletions
13
ChangeLog
13
ChangeLog
|
@ -1,3 +1,16 @@
|
||||||
|
2003-09-17 Rachel Hestilow <rachel@nullenvoid.com>
|
||||||
|
|
||||||
|
* glib/DelegateWrapper.cs: Remove 'RemoveIfNotAlive'
|
||||||
|
and revamp the memory management to use destroy
|
||||||
|
notification.
|
||||||
|
|
||||||
|
* generator/CallbackGen.cs: Do not generate the call to
|
||||||
|
RemoveIfNotAlive.
|
||||||
|
|
||||||
|
* gtk/GtkSharp.GtkClipboardGetFuncNative,
|
||||||
|
GtkSharp.GtkClipboardClearFuncNative: Do not call
|
||||||
|
RemoveIfNotAlive.
|
||||||
|
|
||||||
2003-09-17 Rachel Hestilow <rachel@nullenvoid.com>
|
2003-09-17 Rachel Hestilow <rachel@nullenvoid.com>
|
||||||
|
|
||||||
* generator/StructBase.cs: Make pointer, wrapped,
|
* generator/StructBase.cs: Make pointer, wrapped,
|
||||||
|
|
|
@ -114,13 +114,6 @@ namespace GtkSharp.Generation {
|
||||||
|
|
||||||
sw.WriteLine ("\t\tpublic " + m_ret + " NativeCallback (" + import_sig + ")");
|
sw.WriteLine ("\t\tpublic " + m_ret + " NativeCallback (" + import_sig + ")");
|
||||||
sw.WriteLine ("\t\t{");
|
sw.WriteLine ("\t\t{");
|
||||||
sw.Write ("\t\t\tif (RemoveIfNotAlive ()) return ");
|
|
||||||
if (table.IsStruct (rettype) || table.IsBoxed (rettype))
|
|
||||||
sw.WriteLine ("IntPtr.Zero;");
|
|
||||||
else if (ret_wrapper != null && (ret_wrapper is ObjectGen || ret_wrapper is OpaqueGen))
|
|
||||||
sw.WriteLine ("IntPtr.Zero;");
|
|
||||||
else
|
|
||||||
sw.WriteLine (m_ret != "void" ? "_dummy;" : ";");
|
|
||||||
|
|
||||||
int count = (parms != null) ? parms.Count : 0;
|
int count = (parms != null) ? parms.Count : 0;
|
||||||
if (count > 0)
|
if (count > 0)
|
||||||
|
|
|
@ -11,6 +11,7 @@ namespace GLib {
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Collections;
|
using System.Collections;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// DelegateWrapper Class
|
/// DelegateWrapper Class
|
||||||
|
@ -23,9 +24,14 @@ namespace GLib {
|
||||||
public class DelegateWrapper
|
public class DelegateWrapper
|
||||||
{
|
{
|
||||||
// Keys in the hashtable are instances of classes derived from this one.
|
// Keys in the hashtable are instances of classes derived from this one.
|
||||||
// Values are WeakReference instances to the object that creates the
|
// Values are each instance's destroy notification delegate
|
||||||
// delegate or instances of derived classes (if created from static methods).
|
static Hashtable instances = new Hashtable ();
|
||||||
static Hashtable weakReferences = new Hashtable ();
|
|
||||||
|
// This list holds references to wrappers for static
|
||||||
|
// methods. These will never expire.
|
||||||
|
static ArrayList static_instances = new ArrayList ();
|
||||||
|
|
||||||
|
static int notify_count = 0;
|
||||||
|
|
||||||
// The object 'o' is the object that creates the instance of the DelegateWrapper
|
// The object 'o' is the object that creates the instance of the DelegateWrapper
|
||||||
// derived class or null if created from a static method.
|
// derived class or null if created from a static method.
|
||||||
|
@ -33,26 +39,58 @@ namespace GLib {
|
||||||
// method.
|
// method.
|
||||||
protected DelegateWrapper (object o)
|
protected DelegateWrapper (object o)
|
||||||
{
|
{
|
||||||
if (o == null)
|
if (o != null) {
|
||||||
o = this; // Never expires. Used in static methods.
|
// If o is a GObject, we can get
|
||||||
|
// destroy notification. Otherwise
|
||||||
|
// no additional references to
|
||||||
|
// the wrapper are kept.
|
||||||
|
// FIXME: This should work because
|
||||||
|
// currently only GObjects store
|
||||||
|
// callbacks over the long-term
|
||||||
|
|
||||||
weakReferences [this] = new WeakReference (o);
|
if (o is GLib.Object) {
|
||||||
|
AddDestroyNotify ((GLib.Object) o);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// If o is null, we cannot ask for a destroy
|
||||||
|
// notification, so the wrapper never expires.
|
||||||
|
|
||||||
|
lock (typeof (DelegateWrapper)) {
|
||||||
|
static_instances.Add (this);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// IMPORTANT: this method must be the first one called from the callback methods that
|
private delegate void DestroyNotify (IntPtr data);
|
||||||
// are invoked from unmanaged code.
|
|
||||||
// If this method returns true, the object that created the delegate wrapper no longer
|
|
||||||
// exists and the instance of the delegate itself is removed from the hash table.
|
|
||||||
protected bool RemoveIfNotAlive ()
|
|
||||||
{
|
|
||||||
WeakReference r = null;
|
|
||||||
r = weakReferences [this] as WeakReference;
|
|
||||||
if (r != null && !r.IsAlive) {
|
|
||||||
weakReferences.Remove (this);
|
|
||||||
r = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (r == null);
|
[DllImport("libgobject-2.0-0.dll")]
|
||||||
|
private static extern void g_object_set_data (IntPtr obj, string name, IntPtr data, DestroyNotify destroy);
|
||||||
|
|
||||||
|
private void AddDestroyNotify (GLib.Object o) {
|
||||||
|
// This is a bit of an ugly hack. There is no
|
||||||
|
// way of getting a destroy notification
|
||||||
|
// explicitly, so we set some data and ask
|
||||||
|
// for notification when it is removed
|
||||||
|
|
||||||
|
string name = String.Format ("_GtkSharpDelegateWrapper_{0}", notify_count);
|
||||||
|
DestroyNotify destroy = new DestroyNotify (this.OnDestroy);
|
||||||
|
|
||||||
|
g_object_set_data (o.Handle, name, IntPtr.Zero, destroy);
|
||||||
|
lock (typeof (DelegateWrapper)) {
|
||||||
|
instances[this] = destroy;
|
||||||
|
notify_count++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// This callback is invoked by GLib to indicate that the
|
||||||
|
// object that owned the native delegate wrapper no longer
|
||||||
|
// exists and the instance of the delegate itself is removed from the hash table.
|
||||||
|
private void OnDestroy (IntPtr data) {
|
||||||
|
lock (typeof (DelegateWrapper)) {
|
||||||
|
if (instances.ContainsKey (this)) {
|
||||||
|
instances.Remove (this);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,7 +9,6 @@ namespace GtkSharp {
|
||||||
|
|
||||||
public void NativeCallback (IntPtr clipboard, uint objid)
|
public void NativeCallback (IntPtr clipboard, uint objid)
|
||||||
{
|
{
|
||||||
if (RemoveIfNotAlive ()) return;
|
|
||||||
object[] _args = new object[2];
|
object[] _args = new object[2];
|
||||||
_args[0] = (Gtk.Clipboard) GLib.Object.GetObject(clipboard, false);
|
_args[0] = (Gtk.Clipboard) GLib.Object.GetObject(clipboard, false);
|
||||||
if (_args[0] == null)
|
if (_args[0] == null)
|
||||||
|
|
|
@ -9,7 +9,6 @@ namespace GtkSharp {
|
||||||
|
|
||||||
public void NativeCallback (IntPtr clipboard, ref Gtk.SelectionData selection_data, uint info, uint obj_id)
|
public void NativeCallback (IntPtr clipboard, ref Gtk.SelectionData selection_data, uint info, uint obj_id)
|
||||||
{
|
{
|
||||||
if (RemoveIfNotAlive ()) return;
|
|
||||||
object[] _args = new object[4];
|
object[] _args = new object[4];
|
||||||
_args[0] = (Gtk.Clipboard) GLib.Object.GetObject(clipboard, false);
|
_args[0] = (Gtk.Clipboard) GLib.Object.GetObject(clipboard, false);
|
||||||
if (_args[0] == null)
|
if (_args[0] == null)
|
||||||
|
|
Loading…
Reference in a new issue