geode/loader/src/platform/android/gdstdlib.cpp

112 lines
3.3 KiB
C++
Raw Normal View History

#include <Geode/c++stl/gdstdlib.hpp>
2023-12-23 07:44:29 -05:00
#include "../../c++stl/string-impl.hpp"
2023-12-23 10:02:13 -05:00
#ifdef GEODE_IS_ANDROID
#if defined(GEODE_IS_ANDROID32)
static auto constexpr NEW_SYM = "_Znwj";
static constexpr uintptr_t MENULAYER_SCENE = 0x309068 - 0x10000;
static constexpr uintptr_t STRING_EMPTY = 0xaa1c48 - 0x10000 - 0xc; // the internal struct size
2023-12-23 10:02:13 -05:00
#elif defined(GEODE_IS_ANDROID64)
static auto constexpr NEW_SYM = "_Znwm";
static constexpr uintptr_t MENULAYER_SCENE = 0x6a62ec - 0x100000;
static constexpr uintptr_t STRING_EMPTY = 0x12d8568 - 0x100000 - 0x18; // the internal struct size
2023-12-23 10:02:13 -05:00
#endif
static auto constexpr DELETE_SYM = "_ZdlPv";
2023-12-20 11:58:47 -05:00
// 2.2 addition
// zmx please fix this
namespace geode::base {
uintptr_t get() {
2023-12-23 10:02:13 -05:00
static uintptr_t base = (reinterpret_cast<uintptr_t>(&MenuLayer::scene) - MENULAYER_SCENE) & (~0x1);
// static uintptr_t base = reinterpret_cast<uintptr_t>(dlopen("libcocos2dcpp.so", RTLD_NOW));
return base;
}
}
static void* getLibHandle() {
static void* handle = dlopen("libcocos2dcpp.so", RTLD_LAZY | RTLD_NOLOAD);
return handle;
}
static void* gdOperatorNew(size_t size) {
static auto fnPtr = reinterpret_cast<void*(*)(size_t)>(dlsym(getLibHandle(), NEW_SYM));
return fnPtr(size);
}
static void gdOperatorDelete(void* ptr) {
static auto fnPtr = reinterpret_cast<void(*)(void*)>(dlsym(getLibHandle(), DELETE_SYM));
return fnPtr(ptr);
}
2023-12-22 16:09:58 -05:00
namespace geode::stl {
static inline auto emptyInternalString() {
return reinterpret_cast<StringData::Internal*>(
2023-12-23 10:02:13 -05:00
geode::base::get() + STRING_EMPTY + sizeof(StringData::Internal)
2023-12-22 16:09:58 -05:00
);
}
void StringImpl::setEmpty() {
data.m_data = emptyInternalString();
}
void StringImpl::free() {
if (data.m_data == nullptr || data.m_data == emptyInternalString()) return;
if (data.m_data[-1].m_refcount <= 0) {
gdOperatorDelete(&data.m_data[-1]);
data.m_data = nullptr;
} else {
--data.m_data[-1].m_refcount;
}
}
char* StringImpl::getStorage() {
return reinterpret_cast<char*>(data.m_data);
}
2023-12-23 09:10:36 -05:00
// TODO: add a copyFrom(string const&) to take advantage
// of gnustl refcounted strings
void StringImpl::setStorage(const std::string_view str) {
2023-12-22 16:09:58 -05:00
this->free();
if (str.size() == 0) {
this->setEmpty();
return;
}
StringData::Internal internal;
internal.m_size = str.size();
internal.m_capacity = str.size();
internal.m_refcount = 0;
// use char* so we can do easy pointer arithmetic with it
auto* buffer = static_cast<char*>(gdOperatorNew(str.size() + 1 + sizeof(internal)));
std::memcpy(buffer, &internal, sizeof(internal));
std::memcpy(buffer + sizeof(internal), str.data(), str.size());
data.m_data = reinterpret_cast<StringData::Internal*>(buffer + sizeof(internal));
this->getStorage()[str.size()] = 0;
}
size_t StringImpl::getSize() {
return data.m_data[-1].m_size;
}
void StringImpl::setSize(size_t size) {
2023-12-22 16:09:58 -05:00
// TODO: implement this, remember its copy-on-write...
}
size_t StringImpl::getCapacity() {
return data.m_data[-1].m_capacity;
}
void StringImpl::setCapacity(size_t cap) {
2023-12-22 16:09:58 -05:00
// TODO: implement this, remember its copy-on-write...
}
}
#endif