b732e377c9
From ~200ms down to ~65ms on a rMBP with Nvidia 650M and Mac OS X 10.9.2.
171 lines
5.8 KiB
C#
171 lines
5.8 KiB
C#
#region License
|
|
//
|
|
// NS.cs
|
|
//
|
|
// Author:
|
|
// Stefanos A. <stapostol@gmail.com>
|
|
//
|
|
// Copyright (c) 2006-2013 Stefanos Apostolopoulos
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
// in the Software without restriction, including without limitation the rights
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
// THE SOFTWARE.
|
|
//
|
|
|
|
#endregion
|
|
|
|
using System;
|
|
using System.Runtime.InteropServices;
|
|
|
|
namespace OpenTK.Platform.MacOS
|
|
{
|
|
[Flags]
|
|
enum AddImageFlags
|
|
{
|
|
ReturnOnError = 1,
|
|
WithSearching = 2,
|
|
ReturnOnlyIfLoaded = 4
|
|
}
|
|
|
|
[Flags]
|
|
enum SymbolLookupFlags
|
|
{
|
|
Bind = 0,
|
|
BindNow = 1,
|
|
BindFully = 2,
|
|
ReturnOnError = 4
|
|
}
|
|
|
|
internal class NS
|
|
{
|
|
const string Library = "libdl.dylib";
|
|
|
|
[DllImport(Library, EntryPoint = "NSAddImage")]
|
|
internal static extern IntPtr AddImage(string s, AddImageFlags flags);
|
|
[DllImport(Library, EntryPoint = "NSAddressOfSymbol")]
|
|
internal static extern IntPtr AddressOfSymbol(IntPtr symbol);
|
|
[DllImport(Library, EntryPoint = "NSIsSymbolNameDefined")]
|
|
internal static extern bool IsSymbolNameDefined(string s);
|
|
[DllImport(Library, EntryPoint = "NSIsSymbolNameDefined")]
|
|
internal static extern bool IsSymbolNameDefined(IntPtr s);
|
|
[DllImport(Library, EntryPoint = "NSLookupAndBindSymbol")]
|
|
internal static extern IntPtr LookupAndBindSymbol(string s);
|
|
[DllImport(Library, EntryPoint = "NSLookupAndBindSymbol")]
|
|
internal static extern IntPtr LookupAndBindSymbol(IntPtr s);
|
|
[DllImport(Library, EntryPoint = "NSLookupSymbolInImage")]
|
|
internal static extern IntPtr LookupSymbolInImage(IntPtr image, IntPtr symbolName, SymbolLookupFlags options);
|
|
|
|
// Unfortunately, these are slower even if they are more
|
|
// portable and simpler to use.
|
|
[DllImport(Library)]
|
|
internal static extern IntPtr dlopen(String fileName, int flags);
|
|
[DllImport(Library)]
|
|
internal static extern int dlclose(IntPtr handle);
|
|
[DllImport (Library)]
|
|
internal static extern IntPtr dlsym (IntPtr handle, string symbol);
|
|
[DllImport (Library)]
|
|
internal static extern IntPtr dlsym (IntPtr handle, IntPtr symbol);
|
|
|
|
public static IntPtr GetAddress(string function)
|
|
{
|
|
// Instead of allocating and combining strings in managed memory
|
|
// we do that directly in unmanaged memory. This way, we avoid
|
|
// 2 string allocations every time this function is called.
|
|
|
|
// must add a '_' prefix and null-terminate the function name,
|
|
// hence we allocate +2 bytes
|
|
IntPtr ptr = Marshal.AllocHGlobal(function.Length + 2);
|
|
try
|
|
{
|
|
Marshal.WriteByte(ptr, (byte)'_');
|
|
for (int i = 0; i < function.Length; i++)
|
|
{
|
|
Marshal.WriteByte(ptr, i + 1, (byte)function[i]);
|
|
}
|
|
Marshal.WriteByte(ptr, function.Length + 1, 0); // null-terminate
|
|
|
|
IntPtr symbol = GetAddressInternal(ptr);
|
|
return symbol;
|
|
}
|
|
finally
|
|
{
|
|
Marshal.FreeHGlobal(ptr);
|
|
}
|
|
}
|
|
|
|
public static IntPtr GetAddress(IntPtr function)
|
|
{
|
|
unsafe
|
|
{
|
|
const int max = 64;
|
|
byte* symbol = stackalloc byte[max];
|
|
byte* ptr = symbol;
|
|
byte* cur = (byte*)function.ToPointer();
|
|
int i = 0;
|
|
|
|
*ptr++ = (byte)'_';
|
|
while (*cur != 0 && ++i < max)
|
|
{
|
|
*ptr++ = *cur++;
|
|
}
|
|
|
|
if (i >= max - 1)
|
|
{
|
|
throw new NotSupportedException(String.Format(
|
|
"Function {0} is too long. Please report a bug at https://github.com/opentk/issues/issues",
|
|
Marshal.PtrToStringAnsi(function)));
|
|
}
|
|
|
|
return GetAddressInternal(new IntPtr(symbol));
|
|
}
|
|
}
|
|
|
|
static IntPtr GetAddressInternal(IntPtr function)
|
|
{
|
|
IntPtr symbol = IntPtr.Zero;
|
|
if (IsSymbolNameDefined(function))
|
|
{
|
|
symbol = LookupAndBindSymbol(function);
|
|
if (symbol != IntPtr.Zero)
|
|
symbol = AddressOfSymbol(symbol);
|
|
}
|
|
return symbol;
|
|
}
|
|
|
|
public static IntPtr GetSymbol(IntPtr handle, string symbol)
|
|
{
|
|
return dlsym(handle, symbol);
|
|
}
|
|
|
|
public static IntPtr GetSymbol(IntPtr handle, IntPtr symbol)
|
|
{
|
|
return dlsym(handle, symbol);
|
|
}
|
|
|
|
public static IntPtr LoadLibrary(string fileName)
|
|
{
|
|
const int RTLD_NOW = 2;
|
|
return dlopen(fileName, RTLD_NOW);
|
|
}
|
|
|
|
public static void FreeLibrary(IntPtr handle)
|
|
{
|
|
dlclose(handle);
|
|
}
|
|
}
|
|
}
|
|
|