cmake/Source/kwsys/DynamicLoader.cxx

479 lines
13 KiB
C++
Raw Normal View History

2017-04-14 19:02:05 +02:00
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
file Copyright.txt or https://cmake.org/licensing#kwsys for details. */
2019-11-11 23:01:05 +01:00
#if defined(_WIN32)
# define NOMINMAX // hide min,max to not conflict with <limits>
#endif
#include "kwsysPrivate.h"
#include KWSYS_HEADER(DynamicLoader.hxx)
#include KWSYS_HEADER(Configure.hxx)
2019-11-11 23:01:05 +01:00
#include KWSYS_HEADER(Encoding.hxx)
// Work-around CMake dependency scanning limitation. This must
// duplicate the above list of headers.
#if 0
2018-08-09 18:06:22 +02:00
# include "Configure.hxx.in"
# include "DynamicLoader.hxx.in"
#endif
2017-04-14 19:02:05 +02:00
// This file actually contains several different implementations:
// * NOOP for environments without dynamic libs
// * HP machines which uses shl_load
// * Mac OS X 10.2.x and earlier which uses NSLinkModule
// * Windows which uses LoadLibrary
// * BeOS / Haiku
// * FreeMiNT for Atari
// * Default implementation for *NIX systems (including Mac OS X 10.3 and
// later) which use dlopen
//
// Each part of the ifdef contains a complete implementation for
// the static methods of DynamicLoader.
2019-11-11 23:01:05 +01:00
#define CHECK_OPEN_FLAGS(var, supported, ret) \
do { \
/* Check for unknown flags. */ \
if ((var & AllOpenFlags) != var) { \
return ret; \
} \
\
/* Check for unsupported flags. */ \
if ((var & (supported)) != var) { \
return ret; \
} \
} while (0)
namespace KWSYS_NAMESPACE {
DynamicLoader::LibraryHandle DynamicLoader::OpenLibrary(
const std::string& libname)
{
return DynamicLoader::OpenLibrary(libname, 0);
}
}
2017-04-14 19:02:05 +02:00
#if !KWSYS_SUPPORTS_SHARED_LIBS
// Implementation for environments without dynamic libs
2018-08-09 18:06:22 +02:00
# include <string.h> // for strerror()
2017-04-14 19:02:05 +02:00
namespace KWSYS_NAMESPACE {
DynamicLoader::LibraryHandle DynamicLoader::OpenLibrary(
2019-11-11 23:01:05 +01:00
const std::string& libname, int flags)
2017-04-14 19:02:05 +02:00
{
return 0;
}
int DynamicLoader::CloseLibrary(DynamicLoader::LibraryHandle lib)
{
if (!lib) {
return 0;
}
return 1;
}
2017-04-14 19:02:05 +02:00
DynamicLoader::SymbolPointer DynamicLoader::GetSymbolAddress(
DynamicLoader::LibraryHandle lib, const std::string& sym)
{
2017-04-14 19:02:05 +02:00
return 0;
}
2017-04-14 19:02:05 +02:00
const char* DynamicLoader::LastError()
{
return "General error";
}
} // namespace KWSYS_NAMESPACE
#elif defined(__hpux)
2018-04-23 21:13:27 +02:00
// Implementation for HPUX machines
2018-08-09 18:06:22 +02:00
# include <dl.h>
# include <errno.h>
2017-04-14 19:02:05 +02:00
namespace KWSYS_NAMESPACE {
DynamicLoader::LibraryHandle DynamicLoader::OpenLibrary(
2019-11-11 23:01:05 +01:00
const std::string& libname, int flags)
{
2019-11-11 23:01:05 +01:00
CHECK_OPEN_FLAGS(flags, 0, 0);
2015-04-27 22:25:09 +02:00
return shl_load(libname.c_str(), BIND_DEFERRED | DYNAMIC_PATH, 0L);
}
int DynamicLoader::CloseLibrary(DynamicLoader::LibraryHandle lib)
{
2017-04-14 19:02:05 +02:00
if (!lib) {
2015-08-17 11:37:30 +02:00
return 0;
2017-04-14 19:02:05 +02:00
}
return !shl_unload(lib);
}
2017-04-14 19:02:05 +02:00
DynamicLoader::SymbolPointer DynamicLoader::GetSymbolAddress(
DynamicLoader::LibraryHandle lib, const std::string& sym)
{
void* addr;
int status;
/* TYPE_PROCEDURE Look for a function or procedure. (This used to be default)
2017-04-14 19:02:05 +02:00
* TYPE_DATA Look for a symbol in the data segment (for example,
* variables).
* TYPE_UNDEFINED Look for any symbol.
*/
2017-04-14 19:02:05 +02:00
status = shl_findsym(&lib, sym.c_str(), TYPE_UNDEFINED, &addr);
void* result = (status < 0) ? (void*)0 : addr;
// Hack to cast pointer-to-data to pointer-to-function.
return *reinterpret_cast<DynamicLoader::SymbolPointer*>(&result);
}
const char* DynamicLoader::LastError()
{
// TODO: Need implementation with errno/strerror
/* If successful, shl_findsym returns an integer (int) value zero. If
* shl_findsym cannot find sym, it returns -1 and sets errno to zero.
* If any other errors occur, shl_findsym returns -1 and sets errno to one
* of these values (defined in <errno.h>):
* ENOEXEC
* A format error was detected in the specified library.
* ENOSYM
* A symbol on which sym depends could not be found.
* EINVAL
* The specified handle is invalid.
*/
2017-04-14 19:02:05 +02:00
if (errno == ENOEXEC || errno == ENOSYM || errno == EINVAL) {
return strerror(errno);
2017-04-14 19:02:05 +02:00
}
// else
return 0;
}
} // namespace KWSYS_NAMESPACE
2017-04-14 19:02:05 +02:00
#elif defined(__APPLE__) && (MAC_OS_X_VERSION_MAX_ALLOWED < 1030)
// Implementation for Mac OS X 10.2.x and earlier
2018-08-09 18:06:22 +02:00
# include <mach-o/dyld.h>
# include <string.h> // for strlen
2017-04-14 19:02:05 +02:00
namespace KWSYS_NAMESPACE {
2017-04-14 19:02:05 +02:00
DynamicLoader::LibraryHandle DynamicLoader::OpenLibrary(
2019-11-11 23:01:05 +01:00
const std::string& libname, int flags)
{
2019-11-11 23:01:05 +01:00
CHECK_OPEN_FLAGS(flags, 0, 0);
NSObjectFileImageReturnCode rc;
NSObjectFileImage image = 0;
2015-04-27 22:25:09 +02:00
rc = NSCreateObjectFileImageFromFile(libname.c_str(), &image);
// rc == NSObjectFileImageInappropriateFile when trying to load a dylib file
2017-04-14 19:02:05 +02:00
if (rc != NSObjectFileImageSuccess) {
return 0;
2017-04-14 19:02:05 +02:00
}
2018-08-09 18:06:22 +02:00
NSModule handle = NSLinkModule(image, libname.c_str(),
NSLINKMODULE_OPTION_BINDNOW |
NSLINKMODULE_OPTION_RETURN_ON_ERROR);
NSDestroyObjectFileImage(image);
return handle;
}
2017-04-14 19:02:05 +02:00
int DynamicLoader::CloseLibrary(DynamicLoader::LibraryHandle lib)
{
// NSUNLINKMODULE_OPTION_KEEP_MEMORY_MAPPED
// With this option the memory for the module is not deallocated
// allowing pointers into the module to still be valid.
// You should use this option instead if your code experience some problems
// reported against Panther 10.3.9 (fixed in Tiger 10.4.2 and up)
bool success = NSUnLinkModule(lib, NSUNLINKMODULE_OPTION_NONE);
return success;
}
DynamicLoader::SymbolPointer DynamicLoader::GetSymbolAddress(
2015-11-17 17:22:37 +01:00
DynamicLoader::LibraryHandle lib, const std::string& sym)
{
2017-04-14 19:02:05 +02:00
void* result = 0;
// Need to prepend symbols with '_' on Apple-gcc compilers
2018-04-23 21:13:27 +02:00
std::string rsym = '_' + sym;
2018-04-23 21:13:27 +02:00
NSSymbol symbol = NSLookupSymbolInModule(lib, rsym.c_str());
2017-04-14 19:02:05 +02:00
if (symbol) {
result = NSAddressOfSymbol(symbol);
2017-04-14 19:02:05 +02:00
}
// Hack to cast pointer-to-data to pointer-to-function.
return *reinterpret_cast<DynamicLoader::SymbolPointer*>(&result);
}
const char* DynamicLoader::LastError()
{
return 0;
}
} // namespace KWSYS_NAMESPACE
2017-04-14 19:02:05 +02:00
#elif defined(_WIN32) && !defined(__CYGWIN__)
// Implementation for Windows win32 code but not cygwin
2018-08-09 18:06:22 +02:00
# include <windows.h>
2019-11-11 23:01:05 +01:00
# include <stdio.h>
2017-04-14 19:02:05 +02:00
namespace KWSYS_NAMESPACE {
2017-04-14 19:02:05 +02:00
DynamicLoader::LibraryHandle DynamicLoader::OpenLibrary(
2019-11-11 23:01:05 +01:00
const std::string& libname, int flags)
{
2020-02-01 23:06:01 +01:00
CHECK_OPEN_FLAGS(flags, SearchBesideLibrary, nullptr);
2019-11-11 23:01:05 +01:00
DWORD llFlags = 0;
if (flags & SearchBesideLibrary) {
llFlags |= LOAD_WITH_ALTERED_SEARCH_PATH;
}
2020-02-01 23:06:01 +01:00
return LoadLibraryExW(Encoding::ToWindowsExtendedPath(libname).c_str(),
nullptr, llFlags);
}
int DynamicLoader::CloseLibrary(DynamicLoader::LibraryHandle lib)
{
return (int)FreeLibrary(lib);
}
DynamicLoader::SymbolPointer DynamicLoader::GetSymbolAddress(
2015-11-17 17:22:37 +01:00
DynamicLoader::LibraryHandle lib, const std::string& sym)
{
// TODO: The calling convention affects the name of the symbol. We
// should have a tool to help get the symbol with the desired
// calling convention. Currently we assume cdecl.
//
// MSVC:
// __cdecl = "func" (default)
// __fastcall = "@_func@X"
// __stdcall = "_func@X"
//
// Note that the "@X" part of the name above is the total size (in
// bytes) of the arguments on the stack.
2017-04-14 19:02:05 +02:00
void* result;
const char* rsym = sym.c_str();
result = (void*)GetProcAddress(lib, rsym);
return *reinterpret_cast<DynamicLoader::SymbolPointer*>(&result);
}
2019-11-11 23:01:05 +01:00
# define DYNLOAD_ERROR_BUFFER_SIZE 1024
const char* DynamicLoader::LastError()
{
2019-11-11 23:01:05 +01:00
wchar_t lpMsgBuf[DYNLOAD_ERROR_BUFFER_SIZE + 1];
DWORD error = GetLastError();
DWORD length = FormatMessageW(
2020-02-01 23:06:01 +01:00
FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, nullptr, error,
2019-11-11 23:01:05 +01:00
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
2020-02-01 23:06:01 +01:00
lpMsgBuf, DYNLOAD_ERROR_BUFFER_SIZE, nullptr);
2019-11-11 23:01:05 +01:00
static char str[DYNLOAD_ERROR_BUFFER_SIZE + 1];
if (length < 1) {
/* FormatMessage failed. Use a default message. */
2022-03-29 21:10:50 +02:00
snprintf(str, DYNLOAD_ERROR_BUFFER_SIZE,
"DynamicLoader encountered error 0x%lX. "
"FormatMessage failed with error 0x%lX",
error, GetLastError());
2019-11-11 23:01:05 +01:00
return str;
}
2017-04-14 19:02:05 +02:00
2019-11-11 23:01:05 +01:00
if (!WideCharToMultiByte(CP_UTF8, 0, lpMsgBuf, -1, str,
2020-02-01 23:06:01 +01:00
DYNLOAD_ERROR_BUFFER_SIZE, nullptr, nullptr)) {
2019-11-11 23:01:05 +01:00
/* WideCharToMultiByte failed. Use a default message. */
2022-03-29 21:10:50 +02:00
snprintf(str, DYNLOAD_ERROR_BUFFER_SIZE,
"DynamicLoader encountered error 0x%lX. "
"WideCharToMultiByte failed with error 0x%lX",
error, GetLastError());
2017-04-14 19:02:05 +02:00
}
return str;
}
} // namespace KWSYS_NAMESPACE
2017-04-14 19:02:05 +02:00
#elif defined(__BEOS__)
// Implementation for BeOS / Haiku
2018-08-09 18:06:22 +02:00
# include <string.h> // for strerror()
2018-08-09 18:06:22 +02:00
# include <be/kernel/image.h>
# include <be/support/Errors.h>
2017-04-14 19:02:05 +02:00
namespace KWSYS_NAMESPACE {
static image_id last_dynamic_err = B_OK;
2017-04-14 19:02:05 +02:00
DynamicLoader::LibraryHandle DynamicLoader::OpenLibrary(
2019-11-11 23:01:05 +01:00
const std::string& libname, int flags)
{
2019-11-11 23:01:05 +01:00
CHECK_OPEN_FLAGS(flags, 0, 0);
// image_id's are integers, errors are negative. Add one just in case we
// get a valid image_id of zero (is that even possible?).
2015-04-27 22:25:09 +02:00
image_id rc = load_add_on(libname.c_str());
2017-04-14 19:02:05 +02:00
if (rc < 0) {
last_dynamic_err = rc;
return 0;
2017-04-14 19:02:05 +02:00
}
2017-04-14 19:02:05 +02:00
return rc + 1;
}
int DynamicLoader::CloseLibrary(DynamicLoader::LibraryHandle lib)
{
2017-04-14 19:02:05 +02:00
if (!lib) {
last_dynamic_err = B_BAD_VALUE;
return 0;
} else {
// The function dlclose() returns 0 on success, and non-zero on error.
2017-04-14 19:02:05 +02:00
status_t rc = unload_add_on(lib - 1);
if (rc != B_OK) {
last_dynamic_err = rc;
return 0;
}
2017-04-14 19:02:05 +02:00
}
return 1;
}
DynamicLoader::SymbolPointer DynamicLoader::GetSymbolAddress(
2015-11-17 17:22:37 +01:00
DynamicLoader::LibraryHandle lib, const std::string& sym)
{
// Hack to cast pointer-to-data to pointer-to-function.
2017-04-14 19:02:05 +02:00
union
{
void* pvoid;
DynamicLoader::SymbolPointer psym;
} result;
2020-02-01 23:06:01 +01:00
result.psym = nullptr;
2017-04-14 19:02:05 +02:00
if (!lib) {
last_dynamic_err = B_BAD_VALUE;
} else {
// !!! FIXME: BeOS can do function-only lookups...does this ever
// !!! FIXME: actually _want_ a data symbol lookup, or was this union
// !!! FIXME: a leftover of dlsym()? (s/ANY/TEXT for functions only).
2017-04-14 19:02:05 +02:00
status_t rc =
get_image_symbol(lib - 1, sym.c_str(), B_SYMBOL_TYPE_ANY, &result.pvoid);
if (rc != B_OK) {
last_dynamic_err = rc;
2020-02-01 23:06:01 +01:00
result.psym = nullptr;
}
2017-04-14 19:02:05 +02:00
}
return result.psym;
}
const char* DynamicLoader::LastError()
{
2017-04-14 19:02:05 +02:00
const char* retval = strerror(last_dynamic_err);
last_dynamic_err = B_OK;
return retval;
}
} // namespace KWSYS_NAMESPACE
2017-04-14 19:02:05 +02:00
#elif defined(__MINT__)
// Implementation for FreeMiNT on Atari
2018-08-09 18:06:22 +02:00
# define _GNU_SOURCE /* for program_invocation_name */
# include <dld.h>
# include <errno.h>
# include <malloc.h>
# include <string.h>
2013-03-16 19:13:01 +02:00
2017-04-14 19:02:05 +02:00
namespace KWSYS_NAMESPACE {
2013-03-16 19:13:01 +02:00
2017-04-14 19:02:05 +02:00
DynamicLoader::LibraryHandle DynamicLoader::OpenLibrary(
2019-11-11 23:01:05 +01:00
const std::string& libname, int flags)
2013-03-16 19:13:01 +02:00
{
2020-02-01 23:06:01 +01:00
CHECK_OPEN_FLAGS(flags, 0, nullptr);
2019-11-11 23:01:05 +01:00
2017-04-14 19:02:05 +02:00
char* name = (char*)calloc(1, libname.size() + 1);
2013-03-16 19:13:01 +02:00
dld_init(program_invocation_name);
2015-04-27 22:25:09 +02:00
strncpy(name, libname.c_str(), libname.size());
dld_link(libname.c_str());
2017-04-14 19:02:05 +02:00
return (void*)name;
2013-03-16 19:13:01 +02:00
}
int DynamicLoader::CloseLibrary(DynamicLoader::LibraryHandle lib)
{
2017-04-14 19:02:05 +02:00
dld_unlink_by_file((char*)lib, 0);
2013-03-16 19:13:01 +02:00
free(lib);
return 0;
}
DynamicLoader::SymbolPointer DynamicLoader::GetSymbolAddress(
2015-11-17 17:22:37 +01:00
DynamicLoader::LibraryHandle lib, const std::string& sym)
2013-03-16 19:13:01 +02:00
{
// Hack to cast pointer-to-data to pointer-to-function.
union
{
void* pvoid;
DynamicLoader::SymbolPointer psym;
} result;
2015-04-27 22:25:09 +02:00
result.pvoid = dld_get_symbol(sym.c_str());
2013-03-16 19:13:01 +02:00
return result.psym;
}
const char* DynamicLoader::LastError()
{
return dld_strerror(dld_errno);
}
} // namespace KWSYS_NAMESPACE
2017-04-14 19:02:05 +02:00
#else
// Default implementation for *NIX systems (including Mac OS X 10.3 and
// later) which use dlopen
2018-08-09 18:06:22 +02:00
# include <dlfcn.h>
2017-04-14 19:02:05 +02:00
namespace KWSYS_NAMESPACE {
2017-04-14 19:02:05 +02:00
DynamicLoader::LibraryHandle DynamicLoader::OpenLibrary(
2019-11-11 23:01:05 +01:00
const std::string& libname, int flags)
{
2022-03-29 21:10:50 +02:00
CHECK_OPEN_FLAGS(flags, RTLDGlobal, nullptr);
int llFlags = RTLD_LAZY;
if (flags & RTLDGlobal) {
llFlags |= RTLD_GLOBAL;
}
2019-11-11 23:01:05 +01:00
2022-03-29 21:10:50 +02:00
return dlopen(libname.c_str(), llFlags);
}
int DynamicLoader::CloseLibrary(DynamicLoader::LibraryHandle lib)
{
2017-04-14 19:02:05 +02:00
if (lib) {
// The function dlclose() returns 0 on success, and non-zero on error.
return !dlclose(lib);
2017-04-14 19:02:05 +02:00
}
// else
return 0;
}
DynamicLoader::SymbolPointer DynamicLoader::GetSymbolAddress(
2015-11-17 17:22:37 +01:00
DynamicLoader::LibraryHandle lib, const std::string& sym)
{
// Hack to cast pointer-to-data to pointer-to-function.
2017-04-14 19:02:05 +02:00
union
{
void* pvoid;
DynamicLoader::SymbolPointer psym;
} result;
2015-04-27 22:25:09 +02:00
result.pvoid = dlsym(lib, sym.c_str());
return result.psym;
}
const char* DynamicLoader::LastError()
{
return dlerror();
}
} // namespace KWSYS_NAMESPACE
#endif