/* GMODULE - GLIB wrapper code for dynamic module loading * Copyright (C) 1998, 2000 Tim Janik * * dyld (Darwin) GMODULE implementation * Copyright (C) 2001 Dan Winship * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 library; if not, see <http://www.gnu.org/licenses/>. */ #include "config.h" #include <mach-o/dyld.h> static gpointer self_module = GINT_TO_POINTER (1); static gpointer _g_module_open (const gchar *file_name, gboolean bind_lazy, gboolean bind_local) { NSObjectFileImage image; NSObjectFileImageReturnCode ret; NSModule module; unsigned long options; char *msg; ret = NSCreateObjectFileImageFromFile (file_name, &image); if (ret != NSObjectFileImageSuccess) { switch (ret) { case NSObjectFileImageInappropriateFile: case NSObjectFileImageFormat: msg = g_strdup_printf ("%s is not a loadable module", file_name); break; case NSObjectFileImageArch: msg = g_strdup_printf ("%s is not built for this architecture", file_name); break; case NSObjectFileImageAccess: if (access (file_name, F_OK) == 0) msg = g_strdup_printf ("%s: permission denied", file_name); else msg = g_strdup_printf ("%s: no such file or directory", file_name); break; default: msg = g_strdup_printf ("unknown error for %s", file_name); break; } g_module_set_error (msg); g_free (msg); return NULL; } options = NSLINKMODULE_OPTION_RETURN_ON_ERROR; if (bind_local) options |= NSLINKMODULE_OPTION_PRIVATE; if (!bind_lazy) options |= NSLINKMODULE_OPTION_BINDNOW; module = NSLinkModule (image, file_name, options); NSDestroyObjectFileImage (image); if (!module) { NSLinkEditErrors c; int error_number; const char *file, *error; NSLinkEditError (&c, &error_number, &file, &error); msg = g_strdup_printf ("could not link %s: %s", file_name, error); g_module_set_error (msg); g_free (msg); return NULL; } return module; } static gpointer _g_module_self (void) { return &self_module; } static void _g_module_close (gpointer handle, gboolean is_unref) { if (handle == &self_module) return; if (!NSUnLinkModule (handle, 0)) g_module_set_error ("could not unlink module"); } static gpointer _g_module_symbol (gpointer handle, const gchar *symbol_name) { NSSymbol sym; char *msg; if (handle == &self_module) { if (NSIsSymbolNameDefined (symbol_name)) sym = NSLookupAndBindSymbol (symbol_name); else sym = NULL; } else sym = NSLookupSymbolInModule (handle, symbol_name); if (!sym) { msg = g_strdup_printf ("no such symbol %s", symbol_name); g_module_set_error (msg); g_free (msg); return NULL; } return NSAddressOfSymbol (sym); } static gchar* _g_module_build_path (const gchar *directory, const gchar *module_name) { if (directory && *directory) { if (strncmp (module_name, "lib", 3) == 0) return g_strconcat (directory, "/", module_name, NULL); else return g_strconcat (directory, "/lib", module_name, "." G_MODULE_SUFFIX, NULL); } else if (strncmp (module_name, "lib", 3) == 0) return g_strdup (module_name); else return g_strconcat ("lib", module_name, "." G_MODULE_SUFFIX, NULL); }