mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-18 05:16:05 +00:00
f0f8e19694
Also fix some previous errors with managed types.
159 lines
4.6 KiB
C#
159 lines
4.6 KiB
C#
// Gst.GLib.ToggleRef.cs - Gst.GLib ToggleRef class implementation
|
|
//
|
|
// Author: Mike Kestner <mkestner@novell.com>
|
|
//
|
|
// Copyright <c> 2007 Novell, Inc.
|
|
//
|
|
// This program is free software; you can redistribute it and/or
|
|
// modify it under the terms of version 2 of the Lesser GNU General
|
|
// Public License as published by the Free Software Foundation.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
// Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
// License along with this program; if not, write to the
|
|
// Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
// Boston, MA 02111-1307, USA.
|
|
|
|
|
|
namespace Gst.GLib {
|
|
|
|
using System;
|
|
using System.Collections;
|
|
using System.Runtime.InteropServices;
|
|
|
|
internal class ToggleRef {
|
|
|
|
bool hardened;
|
|
IntPtr handle;
|
|
object reference;
|
|
GCHandle gch;
|
|
Hashtable signals;
|
|
|
|
public ToggleRef (Gst.GLib.Object target)
|
|
{
|
|
handle = target.Handle;
|
|
gch = GCHandle.Alloc (this);
|
|
reference = target;
|
|
g_object_add_toggle_ref (target.Handle, ToggleNotifyCallback, (IntPtr) gch);
|
|
g_object_unref (target.Handle);
|
|
}
|
|
|
|
public bool IsAlive {
|
|
get {
|
|
if (reference is WeakReference) {
|
|
WeakReference weak = reference as WeakReference;
|
|
return weak.IsAlive;
|
|
} else if (reference == null)
|
|
return false;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
public IntPtr Handle {
|
|
get {
|
|
return handle;
|
|
}
|
|
}
|
|
|
|
public Hashtable Signals {
|
|
get {
|
|
if (signals == null)
|
|
signals = new Hashtable ();
|
|
return signals;
|
|
}
|
|
}
|
|
|
|
public Gst.GLib.Object Target {
|
|
get {
|
|
if (reference == null)
|
|
return null;
|
|
else if (reference is Gst.GLib.Object)
|
|
return reference as Gst.GLib.Object;
|
|
|
|
WeakReference weak = reference as WeakReference;
|
|
return weak.Target as Gst.GLib.Object;
|
|
}
|
|
}
|
|
|
|
public void Free ()
|
|
{
|
|
Signal[] signals = new Signal [Signals.Count];
|
|
Signals.Values.CopyTo (signals, 0);
|
|
foreach (Signal s in signals)
|
|
s.Free ();
|
|
if (hardened)
|
|
g_object_unref (handle);
|
|
else
|
|
g_object_remove_toggle_ref (handle, ToggleNotifyCallback, (IntPtr) gch);
|
|
reference = null;
|
|
gch.Free ();
|
|
}
|
|
|
|
internal void Harden ()
|
|
{
|
|
// Added for the benefit of GnomeProgram. It releases a final ref in
|
|
// an atexit handler which causes toggle ref notifications to occur after
|
|
// our delegates are gone, so we need a mechanism to override the
|
|
// notifications. This method effectively leaks all objects which invoke it,
|
|
// but since it is only used by Gnome.Program, which is a singleton object
|
|
// with program duration persistence, who cares.
|
|
|
|
g_object_ref (handle);
|
|
g_object_remove_toggle_ref (handle, ToggleNotifyCallback, (IntPtr) gch);
|
|
if (reference is WeakReference)
|
|
reference = (reference as WeakReference).Target;
|
|
hardened = true;
|
|
}
|
|
|
|
void Toggle (bool is_last_ref)
|
|
{
|
|
if (is_last_ref && reference is Gst.GLib.Object)
|
|
reference = new WeakReference (reference);
|
|
else if (!is_last_ref && reference is WeakReference) {
|
|
WeakReference weak = reference as WeakReference;
|
|
if (weak.IsAlive)
|
|
reference = weak.Target;
|
|
}
|
|
}
|
|
|
|
[UnmanagedFunctionPointer (CallingConvention.Cdecl)]
|
|
delegate void ToggleNotifyHandler (IntPtr data, IntPtr handle, bool is_last_ref);
|
|
|
|
static void RefToggled (IntPtr data, IntPtr handle, bool is_last_ref)
|
|
{
|
|
try {
|
|
GCHandle gch = (GCHandle) data;
|
|
ToggleRef tref = gch.Target as ToggleRef;
|
|
tref.Toggle (is_last_ref);
|
|
} catch (Exception e) {
|
|
ExceptionManager.RaiseUnhandledException (e, false);
|
|
}
|
|
}
|
|
|
|
static ToggleNotifyHandler toggle_notify_callback;
|
|
static ToggleNotifyHandler ToggleNotifyCallback {
|
|
get {
|
|
if (toggle_notify_callback == null)
|
|
toggle_notify_callback = new ToggleNotifyHandler (RefToggled);
|
|
return toggle_notify_callback;
|
|
}
|
|
}
|
|
|
|
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
|
|
static extern void g_object_add_toggle_ref (IntPtr raw, ToggleNotifyHandler notify_cb, IntPtr data);
|
|
|
|
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
|
|
static extern void g_object_remove_toggle_ref (IntPtr raw, ToggleNotifyHandler notify_cb, IntPtr data);
|
|
|
|
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
|
|
static extern IntPtr g_object_ref (IntPtr raw);
|
|
|
|
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
|
|
static extern void g_object_unref (IntPtr raw);
|
|
|
|
}
|
|
}
|