2024-04-05 11:20:57 +00:00
|
|
|
// SPDX-FileCopyrightText: 2002-2024 PCSX2 Dev Team
|
2023-12-22 11:57:49 +00:00
|
|
|
// SPDX-License-Identifier: LGPL-3.0+
|
2022-12-18 13:04:48 +00:00
|
|
|
|
|
|
|
#include "common/DynamicLibrary.h"
|
|
|
|
#include "common/Assertions.h"
|
|
|
|
#include "common/Console.h"
|
2024-04-05 11:20:57 +00:00
|
|
|
#include "common/Error.h"
|
2024-05-14 21:55:58 +00:00
|
|
|
#include "common/FileSystem.h"
|
2024-04-05 11:20:57 +00:00
|
|
|
#include "common/SmallString.h"
|
2024-05-14 21:55:58 +00:00
|
|
|
#include "common/Path.h"
|
2022-12-18 13:04:48 +00:00
|
|
|
#include "common/StringUtil.h"
|
|
|
|
|
|
|
|
#include <cstring>
|
|
|
|
#include "fmt/format.h"
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
#include "common/RedtapeWindows.h"
|
|
|
|
#else
|
|
|
|
#include <dlfcn.h>
|
2024-05-14 21:55:58 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#include "common/CocoaTools.h"
|
|
|
|
#endif
|
2022-12-18 13:04:48 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
DynamicLibrary::DynamicLibrary() = default;
|
|
|
|
|
|
|
|
DynamicLibrary::DynamicLibrary(const char* filename)
|
|
|
|
{
|
2024-04-05 11:20:57 +00:00
|
|
|
Error error;
|
|
|
|
if (!Open(filename, &error))
|
|
|
|
Console.ErrorFmt("DynamicLibrary open failed: {}", error.GetDescription());
|
2022-12-18 13:04:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DynamicLibrary::DynamicLibrary(DynamicLibrary&& move)
|
|
|
|
: m_handle(move.m_handle)
|
|
|
|
{
|
|
|
|
move.m_handle = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
DynamicLibrary::~DynamicLibrary()
|
|
|
|
{
|
|
|
|
Close();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string DynamicLibrary::GetUnprefixedFilename(const char* filename)
|
|
|
|
{
|
|
|
|
#if defined(_WIN32)
|
|
|
|
return std::string(filename) + ".dll";
|
|
|
|
#elif defined(__APPLE__)
|
|
|
|
return std::string(filename) + ".dylib";
|
|
|
|
#else
|
|
|
|
return std::string(filename) + ".so";
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string DynamicLibrary::GetVersionedFilename(const char* libname, int major, int minor)
|
|
|
|
{
|
|
|
|
#if defined(_WIN32)
|
|
|
|
if (major >= 0 && minor >= 0)
|
|
|
|
return fmt::format("{}-{}-{}.dll", libname, major, minor);
|
|
|
|
else if (major >= 0)
|
|
|
|
return fmt::format("{}-{}.dll", libname, major);
|
|
|
|
else
|
|
|
|
return fmt::format("{}.dll", libname);
|
|
|
|
#elif defined(__APPLE__)
|
|
|
|
const char* prefix = std::strncmp(libname, "lib", 3) ? "lib" : "";
|
|
|
|
if (major >= 0 && minor >= 0)
|
|
|
|
return fmt::format("{}{}.{}.{}.dylib", prefix, libname, major, minor);
|
|
|
|
else if (major >= 0)
|
|
|
|
return fmt::format("{}{}.{}.dylib", prefix, libname, major);
|
|
|
|
else
|
|
|
|
return fmt::format("{}{}.dylib", prefix, libname);
|
|
|
|
#else
|
|
|
|
const char* prefix = std::strncmp(libname, "lib", 3) ? "lib" : "";
|
|
|
|
if (major >= 0 && minor >= 0)
|
|
|
|
return fmt::format("{}{}.so.{}.{}", prefix, libname, major, minor);
|
|
|
|
else if (major >= 0)
|
|
|
|
return fmt::format("{}{}.so.{}", prefix, libname, major);
|
|
|
|
else
|
|
|
|
return fmt::format("{}{}.so", prefix, libname);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2024-04-05 11:20:57 +00:00
|
|
|
bool DynamicLibrary::Open(const char* filename, Error* error)
|
2022-12-18 13:04:48 +00:00
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
|
|
|
m_handle = reinterpret_cast<void*>(LoadLibraryW(StringUtil::UTF8StringToWideString(filename).c_str()));
|
|
|
|
if (!m_handle)
|
|
|
|
{
|
2024-04-12 13:24:20 +00:00
|
|
|
Error::SetWin32(error, TinyString::from_format("Loading {} failed: ", filename), GetLastError());
|
2022-12-18 13:04:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
#else
|
|
|
|
m_handle = dlopen(filename, RTLD_NOW);
|
|
|
|
if (!m_handle)
|
|
|
|
{
|
2024-05-14 21:55:58 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
// On MacOS, try searching in Frameworks.
|
|
|
|
if (!Path::IsAbsolute(filename))
|
|
|
|
{
|
|
|
|
std::optional<std::string> bundle_path = CocoaTools::GetBundlePath();
|
|
|
|
if (bundle_path.has_value())
|
|
|
|
{
|
|
|
|
std::string frameworks_path = fmt::format("{}/Contents/Frameworks/{}", bundle_path.value(), filename);
|
|
|
|
if (FileSystem::FileExists(frameworks_path.c_str()))
|
|
|
|
{
|
|
|
|
m_handle = dlopen(frameworks_path.c_str(), RTLD_NOW);
|
|
|
|
if (m_handle)
|
|
|
|
{
|
|
|
|
Error::Clear(error);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2022-12-18 13:04:48 +00:00
|
|
|
const char* err = dlerror();
|
2024-04-05 11:20:57 +00:00
|
|
|
Error::SetStringFmt(error, "Loading {} failed: {}", filename, err ? err : "<UNKNOWN>");
|
2022-12-18 13:04:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2024-05-06 15:42:59 +00:00
|
|
|
void DynamicLibrary::Adopt(void* handle)
|
|
|
|
{
|
|
|
|
pxAssertRel(handle, "Handle is valid");
|
|
|
|
|
|
|
|
Close();
|
|
|
|
|
|
|
|
m_handle = handle;
|
|
|
|
}
|
|
|
|
|
2022-12-18 13:04:48 +00:00
|
|
|
void DynamicLibrary::Close()
|
|
|
|
{
|
|
|
|
if (!IsOpen())
|
|
|
|
return;
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
FreeLibrary(reinterpret_cast<HMODULE>(m_handle));
|
|
|
|
#else
|
|
|
|
dlclose(m_handle);
|
|
|
|
#endif
|
|
|
|
m_handle = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void* DynamicLibrary::GetSymbolAddress(const char* name) const
|
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
|
|
|
return reinterpret_cast<void*>(GetProcAddress(reinterpret_cast<HMODULE>(m_handle), name));
|
|
|
|
#else
|
|
|
|
return reinterpret_cast<void*>(dlsym(m_handle, name));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
DynamicLibrary& DynamicLibrary::operator=(DynamicLibrary&& move)
|
|
|
|
{
|
|
|
|
Close();
|
|
|
|
m_handle = move.m_handle;
|
|
|
|
move.m_handle = nullptr;
|
|
|
|
return *this;
|
|
|
|
}
|