no more submodule
This commit is contained in:
431
GtkSharp/Source/Libs/GLibSharp/Marshaller.cs
Normal file
431
GtkSharp/Source/Libs/GLibSharp/Marshaller.cs
Normal file
@@ -0,0 +1,431 @@
|
||||
// GLibSharp.Marshaller.cs : Marshalling utils
|
||||
//
|
||||
// Author: Rachel Hestilow <rachel@nullenvoid.com>
|
||||
// Mike Kestner <mkestner@ximian.com>
|
||||
//
|
||||
// Copyright (c) 2002, 2003 Rachel Hestilow
|
||||
// Copyright (c) 2004 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 GLib {
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Runtime.InteropServices;
|
||||
|
||||
public class Marshaller {
|
||||
|
||||
private Marshaller () {}
|
||||
|
||||
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
|
||||
delegate void d_g_free(IntPtr mem);
|
||||
static d_g_free g_free = FuncLoader.LoadFunction<d_g_free>(FuncLoader.GetProcAddress(GLibrary.Load(Library.GLib), "g_free"));
|
||||
|
||||
public static void Free (IntPtr ptr)
|
||||
{
|
||||
g_free (ptr);
|
||||
}
|
||||
|
||||
public static void Free (IntPtr[] ptrs)
|
||||
{
|
||||
if (ptrs == null)
|
||||
return;
|
||||
|
||||
for (int i = 0; i < ptrs.Length; i++)
|
||||
g_free (ptrs [i]);
|
||||
}
|
||||
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
|
||||
delegate IntPtr d_g_filename_to_utf8(IntPtr mem, IntPtr len, IntPtr read, out IntPtr written, out IntPtr error);
|
||||
static d_g_filename_to_utf8 g_filename_to_utf8 = FuncLoader.LoadFunction<d_g_filename_to_utf8>(FuncLoader.GetProcAddress(GLibrary.Load(Library.GLib), "g_filename_to_utf8"));
|
||||
|
||||
public static string FilenamePtrToString (IntPtr ptr)
|
||||
{
|
||||
if (ptr == IntPtr.Zero) return null;
|
||||
|
||||
IntPtr utf8 = g_filename_to_utf8 (ptr, (IntPtr)(-1), IntPtr.Zero, out _, out var error);
|
||||
|
||||
if (error != IntPtr.Zero)
|
||||
throw new GLib.GException (error);
|
||||
return Utf8PtrToString (utf8);
|
||||
}
|
||||
|
||||
public static string FilenamePtrToStringGFree (IntPtr ptr)
|
||||
{
|
||||
string ret = FilenamePtrToString (ptr);
|
||||
g_free (ptr);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static unsafe ulong strlen (IntPtr s)
|
||||
{
|
||||
ulong cnt = 0;
|
||||
byte *b = (byte *)s;
|
||||
while (*b != 0) {
|
||||
b++;
|
||||
cnt++;
|
||||
}
|
||||
return cnt;
|
||||
}
|
||||
|
||||
public static string Utf8PtrToString (IntPtr ptr)
|
||||
{
|
||||
if (ptr == IntPtr.Zero)
|
||||
return null;
|
||||
|
||||
int len = (int) (uint) strlen (ptr);
|
||||
byte[] bytes = new byte [len];
|
||||
Marshal.Copy (ptr, bytes, 0, len);
|
||||
return System.Text.Encoding.UTF8.GetString (bytes);
|
||||
}
|
||||
|
||||
public static string[] Utf8PtrToString (IntPtr[] ptrs) {
|
||||
// The last pointer is a null terminator.
|
||||
string[] ret = new string[ptrs.Length - 1];
|
||||
for (int i = 0; i < ret.Length; i++)
|
||||
ret[i] = Utf8PtrToString (ptrs[i]);
|
||||
return ret;
|
||||
}
|
||||
|
||||
public static string PtrToStringGFree (IntPtr ptr)
|
||||
{
|
||||
string ret = Utf8PtrToString (ptr);
|
||||
g_free (ptr);
|
||||
return ret;
|
||||
}
|
||||
|
||||
public static string[] PtrToStringGFree (IntPtr[] ptrs) {
|
||||
// The last pointer is a null terminator.
|
||||
string[] ret = new string[ptrs.Length - 1];
|
||||
for (int i = 0; i < ret.Length; i++) {
|
||||
ret[i] = Utf8PtrToString (ptrs[i]);
|
||||
g_free (ptrs[i]);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
|
||||
delegate IntPtr d_g_filename_from_utf8(IntPtr mem, IntPtr len, IntPtr read, out IntPtr written, out IntPtr error);
|
||||
static d_g_filename_from_utf8 g_filename_from_utf8 = FuncLoader.LoadFunction<d_g_filename_from_utf8>(FuncLoader.GetProcAddress(GLibrary.Load(Library.GLib), "g_filename_from_utf8"));
|
||||
|
||||
public static IntPtr StringToFilenamePtr (string str)
|
||||
{
|
||||
if (str == null)
|
||||
return IntPtr.Zero;
|
||||
|
||||
IntPtr utf8 = StringToPtrGStrdup (str);
|
||||
IntPtr result = g_filename_from_utf8 (utf8, (IntPtr)(-1), IntPtr.Zero, out _, out var error);
|
||||
|
||||
g_free (utf8);
|
||||
if (error != IntPtr.Zero)
|
||||
throw new GException (error);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public static IntPtr StringToPtrGStrdup (string str) {
|
||||
if (str == null)
|
||||
return IntPtr.Zero;
|
||||
byte[] bytes = System.Text.Encoding.UTF8.GetBytes (str);
|
||||
IntPtr result = g_malloc (new UIntPtr ((ulong)bytes.Length + 1));
|
||||
Marshal.Copy (bytes, 0, result, bytes.Length);
|
||||
Marshal.WriteByte (result, bytes.Length, 0);
|
||||
return result;
|
||||
}
|
||||
|
||||
public static string StringFormat (string format, params object[] args) {
|
||||
string ret = String.Format (format, args);
|
||||
if (ret.IndexOf ('%') == -1)
|
||||
return ret;
|
||||
else
|
||||
return ret.Replace ("%", "%%");
|
||||
}
|
||||
|
||||
public static IntPtr StringArrayToStrvPtr (string[] strs)
|
||||
{
|
||||
IntPtr[] ptrs = StringArrayToNullTermPointer (strs);
|
||||
IntPtr ret = g_malloc (new UIntPtr ((ulong) (ptrs.Length * IntPtr.Size)));
|
||||
Marshal.Copy (ptrs, 0, ret, ptrs.Length);
|
||||
return ret;
|
||||
}
|
||||
|
||||
public static IntPtr StringArrayToNullTermStrvPointer (string[] strs)
|
||||
{
|
||||
return StringArrayToStrvPtr (strs);
|
||||
}
|
||||
|
||||
public static IntPtr[] StringArrayToNullTermPointer (string[] strs)
|
||||
{
|
||||
if (strs == null)
|
||||
return null;
|
||||
IntPtr[] result = new IntPtr [strs.Length + 1];
|
||||
for (int i = 0; i < strs.Length; i++)
|
||||
result [i] = StringToPtrGStrdup (strs [i]);
|
||||
result [strs.Length] = IntPtr.Zero;
|
||||
return result;
|
||||
}
|
||||
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
|
||||
delegate void d_g_strfreev(IntPtr mem);
|
||||
static d_g_strfreev g_strfreev = FuncLoader.LoadFunction<d_g_strfreev>(FuncLoader.GetProcAddress(GLibrary.Load(Library.GLib), "g_strfreev"));
|
||||
|
||||
public static void StrFreeV (IntPtr null_term_array)
|
||||
{
|
||||
g_strfreev (null_term_array);
|
||||
}
|
||||
|
||||
public static string[] NullTermPtrToStringArray (IntPtr null_term_array, bool owned)
|
||||
{
|
||||
if (null_term_array == IntPtr.Zero)
|
||||
return new string [0];
|
||||
|
||||
int count = 0;
|
||||
var result = new List<string> ();
|
||||
IntPtr s = Marshal.ReadIntPtr (null_term_array, count++ * IntPtr.Size);
|
||||
while (s != IntPtr.Zero) {
|
||||
result.Add (Utf8PtrToString (s));
|
||||
s = Marshal.ReadIntPtr (null_term_array, count++ * IntPtr.Size);
|
||||
}
|
||||
|
||||
if (owned)
|
||||
g_strfreev (null_term_array);
|
||||
|
||||
return result.ToArray ();
|
||||
}
|
||||
|
||||
public static string[] PtrToStringArrayGFree (IntPtr string_array)
|
||||
{
|
||||
if (string_array == IntPtr.Zero)
|
||||
return new string [0];
|
||||
|
||||
int count = 0;
|
||||
while (Marshal.ReadIntPtr (string_array, count*IntPtr.Size) != IntPtr.Zero)
|
||||
++count;
|
||||
|
||||
string[] members = new string[count];
|
||||
for (int i = 0; i < count; ++i) {
|
||||
IntPtr s = Marshal.ReadIntPtr (string_array, i * IntPtr.Size);
|
||||
members[i] = PtrToStringGFree (s);
|
||||
}
|
||||
Free (string_array);
|
||||
return members;
|
||||
}
|
||||
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
|
||||
delegate IntPtr d_g_malloc(UIntPtr size);
|
||||
static d_g_malloc g_malloc = FuncLoader.LoadFunction<d_g_malloc>(FuncLoader.GetProcAddress(GLibrary.Load(Library.GLib), "g_malloc"));
|
||||
|
||||
public static IntPtr Malloc (ulong size)
|
||||
{
|
||||
return g_malloc (new UIntPtr (size));
|
||||
}
|
||||
|
||||
static System.DateTime local_epoch = new System.DateTime (1970, 1, 1, 0, 0, 0);
|
||||
static int utc_offset = (int) (System.TimeZone.CurrentTimeZone.GetUtcOffset (System.DateTime.Now)).TotalSeconds;
|
||||
|
||||
public static IntPtr DateTimeTotime_t (System.DateTime time)
|
||||
{
|
||||
return new IntPtr (((long)time.Subtract (local_epoch).TotalSeconds) - utc_offset);
|
||||
}
|
||||
|
||||
public static System.DateTime time_tToDateTime (IntPtr time_t)
|
||||
{
|
||||
return local_epoch.AddSeconds (time_t.ToInt64 () + utc_offset);
|
||||
}
|
||||
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
|
||||
delegate IntPtr d_g_malloc0(UIntPtr size);
|
||||
static d_g_malloc0 g_malloc0 = FuncLoader.LoadFunction<d_g_malloc0>(FuncLoader.GetProcAddress(GLibrary.Load(Library.GLib), "g_malloc0"));
|
||||
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
|
||||
delegate int d_g_unichar_to_utf8(uint c, IntPtr buf);
|
||||
static d_g_unichar_to_utf8 g_unichar_to_utf8 = FuncLoader.LoadFunction<d_g_unichar_to_utf8>(FuncLoader.GetProcAddress(GLibrary.Load(Library.GLib), "g_unichar_to_utf8"));
|
||||
|
||||
public static char GUnicharToChar (uint ucs4_char)
|
||||
{
|
||||
if (ucs4_char == 0)
|
||||
return (char) 0;
|
||||
|
||||
string ret = GUnicharToString (ucs4_char);
|
||||
if (ret.Length != 1)
|
||||
throw new ArgumentOutOfRangeException ("ucs4char is not representable by a char.");
|
||||
|
||||
return ret [0];
|
||||
}
|
||||
|
||||
public static string GUnicharToString (uint ucs4_char)
|
||||
{
|
||||
if (ucs4_char == 0)
|
||||
return String.Empty;
|
||||
|
||||
IntPtr buf = g_malloc0 (new UIntPtr (7));
|
||||
g_unichar_to_utf8 (ucs4_char, buf);
|
||||
return PtrToStringGFree (buf);
|
||||
}
|
||||
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
|
||||
delegate IntPtr d_g_utf16_to_ucs4(ref ushort c, IntPtr len, IntPtr d1, IntPtr d2, IntPtr d3);
|
||||
static d_g_utf16_to_ucs4 g_utf16_to_ucs4 = FuncLoader.LoadFunction<d_g_utf16_to_ucs4>(FuncLoader.GetProcAddress(GLibrary.Load(Library.GLib), "g_utf16_to_ucs4"));
|
||||
|
||||
public static uint CharToGUnichar (char c)
|
||||
{
|
||||
ushort val = (ushort) c;
|
||||
IntPtr ucs4_str = g_utf16_to_ucs4 (ref val, new IntPtr (1), IntPtr.Zero, IntPtr.Zero, IntPtr.Zero);
|
||||
uint result = (uint) Marshal.ReadInt32 (ucs4_str);
|
||||
g_free (ucs4_str);
|
||||
return result;
|
||||
}
|
||||
|
||||
public static IntPtr StructureToPtrAlloc (object o)
|
||||
{
|
||||
IntPtr result = Marshal.AllocHGlobal (Marshal.SizeOf (o));
|
||||
Marshal.StructureToPtr (o, result, false);
|
||||
return result;
|
||||
}
|
||||
|
||||
public static IntPtr ArrayToArrayPtr (byte[] array)
|
||||
{
|
||||
IntPtr ret = Malloc ((ulong) array.Length);
|
||||
Marshal.Copy (array, 0, ret, array.Length);
|
||||
return ret;
|
||||
}
|
||||
|
||||
public static Array ArrayPtrToArray (IntPtr array_ptr, Type element_type, int length, bool owned)
|
||||
{
|
||||
Array result = null;
|
||||
if (element_type == typeof (byte)) {
|
||||
byte[] ret = new byte [length];
|
||||
Marshal.Copy (array_ptr, ret, 0, length);
|
||||
result = ret;
|
||||
} else {
|
||||
throw new InvalidOperationException ("Marshaling of " + element_type + " arrays is not supported");
|
||||
}
|
||||
if (owned)
|
||||
Free (array_ptr);
|
||||
return result;
|
||||
}
|
||||
|
||||
public static byte[] ArrayPtrToArray<TElement> (IntPtr array_ptr, int length, bool owned)
|
||||
{
|
||||
byte[] result = null;
|
||||
if (typeof(TElement) == typeof (byte)) {
|
||||
byte[] ret = new byte [length];
|
||||
Marshal.Copy (array_ptr, ret, 0, length);
|
||||
result = ret;
|
||||
} else {
|
||||
throw new InvalidOperationException ("Marshaling of " + typeof(TElement) + " arrays is not supported");
|
||||
}
|
||||
if (owned)
|
||||
Free (array_ptr);
|
||||
return result;
|
||||
}
|
||||
|
||||
public static Array ListPtrToArray (IntPtr list_ptr, Type list_type, bool owned, bool elements_owned, Type elem_type)
|
||||
{
|
||||
Type array_type = elem_type == typeof (ListBase.FilenameString) ? typeof (string) : elem_type;
|
||||
ListBase list;
|
||||
if (list_type == typeof(GLib.List))
|
||||
list = new GLib.List (list_ptr, elem_type, owned, elements_owned);
|
||||
else
|
||||
list = new GLib.SList (list_ptr, elem_type, owned, elements_owned);
|
||||
|
||||
using (list)
|
||||
return ListToArray (list, array_type);
|
||||
}
|
||||
|
||||
public static TElement[] ListPtrToArray<TElement, TListElement> (IntPtr list_ptr, bool owned, bool elements_owned)
|
||||
{
|
||||
using (GLib.List list = new GLib.List (list_ptr, typeof(TListElement), owned, elements_owned))
|
||||
return ListToArray<TElement> (list);
|
||||
}
|
||||
|
||||
public static TElement[] SListPtrToArray<TElement, TListElement> (IntPtr list_ptr, bool owned, bool elements_owned)
|
||||
{
|
||||
using (GLib.SList list = new GLib.SList (list_ptr, typeof(TListElement), owned, elements_owned))
|
||||
return ListToArray<TElement> (list);
|
||||
}
|
||||
|
||||
public static Array PtrArrayToArray (IntPtr list_ptr, bool owned, bool elements_owned, Type elem_type)
|
||||
{
|
||||
GLib.PtrArray array = new GLib.PtrArray (list_ptr, elem_type, owned, elements_owned);
|
||||
Array ret = Array.CreateInstance (elem_type, array.Count);
|
||||
array.CopyTo (ret, 0);
|
||||
array.Dispose ();
|
||||
return ret;
|
||||
}
|
||||
|
||||
public static TElement[] PtrArrayToArray<TElement> (IntPtr list_ptr, bool owned, bool elements_owned)
|
||||
{
|
||||
GLib.PtrArray array = new GLib.PtrArray (list_ptr, typeof(TElement), owned, elements_owned);
|
||||
TElement[] ret = new TElement[array.Count];
|
||||
array.CopyTo (ret, 0);
|
||||
array.Dispose ();
|
||||
return ret;
|
||||
}
|
||||
|
||||
public static Array ListToArray (ListBase list, System.Type type)
|
||||
{
|
||||
Array result = Array.CreateInstance (type, list.Count);
|
||||
if (list.Count > 0)
|
||||
list.CopyTo (result, 0);
|
||||
|
||||
if (type.IsSubclassOf (typeof (GLib.Opaque)))
|
||||
list.elements_owned = false;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public static TElement[] ListToArray<TElement> (ListBase list)
|
||||
{
|
||||
TElement[] result = new TElement[list.Count];
|
||||
if (list.Count > 0)
|
||||
list.CopyTo (result, 0);
|
||||
|
||||
if (typeof(TElement).IsSubclassOf (typeof (GLib.Opaque)))
|
||||
list.elements_owned = false;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public static T[] StructArrayFromNullTerminatedIntPtr<T> (IntPtr array)
|
||||
{
|
||||
var res = new List<T> ();
|
||||
IntPtr current = array;
|
||||
T currentStruct = default(T);
|
||||
|
||||
while (current != IntPtr.Zero) {
|
||||
Marshal.PtrToStructure (current, currentStruct);
|
||||
res.Add (currentStruct);
|
||||
current = (IntPtr) ((long)current + Marshal.SizeOf<T> ());
|
||||
}
|
||||
|
||||
return res.ToArray ();
|
||||
}
|
||||
|
||||
public static unsafe IntPtr StructArrayToNullTerminatedStructArrayIntPtr<T> (T[] InputArray)
|
||||
{
|
||||
int intPtrSize = sizeof (IntPtr);
|
||||
IntPtr mem = Marshal.AllocHGlobal ((InputArray.Length + 1) * intPtrSize);
|
||||
|
||||
for (int i = 0; i < InputArray.Length; i++) {
|
||||
IntPtr structPtr = Marshal.AllocHGlobal (Marshal.SizeOf<T> ());
|
||||
Marshal.StructureToPtr (InputArray[i], structPtr, false);
|
||||
// jump to next pointer
|
||||
Marshal.WriteIntPtr (mem, structPtr);
|
||||
mem = (IntPtr) ((long)mem + intPtrSize);
|
||||
}
|
||||
// null terminate
|
||||
Marshal.WriteIntPtr (mem, IntPtr.Zero);
|
||||
|
||||
return mem;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user