mirror of
https://github.com/VCMP-SqMod/SqMod.git
synced 2025-01-19 03:57:14 +01:00
355 lines
13 KiB
C++
355 lines
13 KiB
C++
//
|
|
// SqratClassType: Type Translators
|
|
//
|
|
|
|
//
|
|
// Copyright (c) 2009 Brandon Jones
|
|
//
|
|
// This software is provided 'as-is', without any express or implied
|
|
// warranty. In no event will the authors be held liable for any damages
|
|
// arising from the use of this software.
|
|
//
|
|
// Permission is granted to anyone to use this software for any purpose,
|
|
// including commercial applications, and to alter it and redistribute it
|
|
// freely, subject to the following restrictions:
|
|
//
|
|
// 1. The origin of this software must not be misrepresented; you must not
|
|
// claim that you wrote the original software. If you use this software
|
|
// in a product, an acknowledgment in the product documentation would be
|
|
// appreciated but is not required.
|
|
//
|
|
// 2. Altered source versions must be plainly marked as such, and must not be
|
|
// misrepresented as being the original software.
|
|
//
|
|
// 3. This notice may not be removed or altered from any source
|
|
// distribution.
|
|
//
|
|
|
|
#pragma once
|
|
|
|
#include <squirrelex.h>
|
|
|
|
#include <typeinfo>
|
|
|
|
#include "sqratUtil.h"
|
|
#include "Core/VecMap.hpp"
|
|
|
|
namespace Sqrat
|
|
{
|
|
|
|
/// @cond DEV
|
|
|
|
// The copy function for a class
|
|
typedef SQInteger (*COPYFUNC)(HSQUIRRELVM, SQInteger, const void*);
|
|
|
|
//Obsolete: Every Squirrel class instance made by Sqrat has its type tag set to a AbstractStaticClassData object that is unique per C++ class
|
|
struct AbstractStaticClassData {
|
|
AbstractStaticClassData() = default;
|
|
virtual ~AbstractStaticClassData() = default;
|
|
virtual SQUserPointer Cast(SQUserPointer ptr, SQUserPointer classType) = 0;
|
|
AbstractStaticClassData* baseClass{nullptr};
|
|
string className{};
|
|
COPYFUNC copyFunc{};
|
|
};
|
|
|
|
// Optimization technique with certain risks and requirements. Read the warnings.
|
|
// Provides an application-wide unique pointer for types and is used as an identification method.
|
|
template<class T>
|
|
struct StaticClassTypeTag {
|
|
StaticClassTypeTag() noexcept = default;
|
|
/* @WARNING : this makes the assumption that one (and only one) VM exists at once. which in this application is actually true. */
|
|
static AbstractStaticClassData* TAG;
|
|
// We don't need these:
|
|
StaticClassTypeTag(const StaticClassTypeTag&) = delete;
|
|
StaticClassTypeTag(StaticClassTypeTag&&) = delete;
|
|
StaticClassTypeTag& operator=(const StaticClassTypeTag&) = delete;
|
|
StaticClassTypeTag& operator=(StaticClassTypeTag&&) = delete;
|
|
// Just so we don't have to access it directly.
|
|
static inline AbstractStaticClassData* Get() noexcept { return TAG; }
|
|
};
|
|
// We provide initial unique tag to prevent any collision with null.
|
|
template<class T> AbstractStaticClassData* StaticClassTypeTag<T>::TAG=nullptr;
|
|
|
|
// StaticClassData keeps track of the nearest base class B and the class associated with itself C in order to cast C++ pointers to the right base class
|
|
template<class C, class B>
|
|
struct StaticClassData : public AbstractStaticClassData, public StaticClassTypeTag<C> {
|
|
StaticClassData() : AbstractStaticClassData() { StaticClassTypeTag<C>::TAG = static_cast<AbstractStaticClassData*>(this); }
|
|
~StaticClassData() override { StaticClassTypeTag<C>::TAG = nullptr; }
|
|
SQUserPointer Cast(SQUserPointer ptr, SQUserPointer classType) override {
|
|
if (classType != this) {
|
|
ptr = baseClass->Cast(static_cast<B*>(static_cast<C*>(ptr)), classType);
|
|
}
|
|
return ptr;
|
|
}
|
|
};
|
|
|
|
// Base class for ClassData to provide a common interface
|
|
struct AbstractClassData {
|
|
HSQOBJECT classObj{};
|
|
HSQOBJECT getTable{};
|
|
HSQOBJECT setTable{};
|
|
AbstractClassData() = default;
|
|
virtual ~AbstractClassData() { Release(); }
|
|
virtual size_t InstanceCount() = 0;
|
|
// This should be called before closing the VM
|
|
void Release() {
|
|
// Should we release class object?
|
|
if (!sq_isnull(classObj)) {
|
|
sq_release(SqVM(), &classObj);
|
|
}
|
|
// Should we release get table object?
|
|
if (!sq_isnull(getTable)) {
|
|
sq_release(SqVM(), &getTable);
|
|
}
|
|
// Should we release set table object?
|
|
if (!sq_isnull(setTable)) {
|
|
sq_release(SqVM(), &setTable);
|
|
}
|
|
}
|
|
void Reset() {
|
|
sq_resetobject(&classObj);
|
|
sq_resetobject(&getTable);
|
|
sq_resetobject(&setTable);
|
|
}
|
|
};
|
|
|
|
// Every Squirrel class object created by Sqrat in every VM has its own unique ClassData object stored in the registry table of the VM
|
|
template<class C>
|
|
struct ClassData : public AbstractClassData {
|
|
SharedPtr<std::unordered_map<C*, HSQOBJECT>> instances{};
|
|
SharedPtr<AbstractStaticClassData> staticData{};
|
|
ClassData() = default;
|
|
~ClassData() override = default;
|
|
size_t InstanceCount() override { return instances->size(); }
|
|
};
|
|
|
|
// Inrternal structure used to keep track of created objects.
|
|
struct VMContext {
|
|
using ClsPtr = std::unique_ptr<AbstractClassData>;
|
|
VecMap<std::string, ClsPtr> classes{};
|
|
VMContext() = default;
|
|
~VMContext() = default;
|
|
};
|
|
// Retrieve the associated context from VM
|
|
inline VMContext* GetVMContext(HSQUIRRELVM vm) { return reinterpret_cast<VMContext*>(sq_getforeignptr(vm)); }
|
|
|
|
// Lookup static class data by type_info rather than a template because C++ cannot export generic templates
|
|
/*
|
|
class _ClassType_helper {
|
|
public:
|
|
#if defined(SCRAT_IMPORT)
|
|
static SQRAT_API WeakPtr<AbstractStaticClassData>& _getStaticClassData(const std::type_info* type);
|
|
#else
|
|
struct compare_type_info {
|
|
bool operator ()(const std::type_info* left, const std::type_info* right) const {
|
|
return left->before(*right) != 0;
|
|
}
|
|
};
|
|
static SQRAT_API WeakPtr<AbstractStaticClassData>& _getStaticClassData(const std::type_info* type) {
|
|
static std::map<const std::type_info*, WeakPtr<AbstractStaticClassData>, compare_type_info> data;
|
|
return data[type];
|
|
}
|
|
#endif
|
|
};
|
|
*/
|
|
|
|
// Internal helper class for managing classes
|
|
template<class C>
|
|
class ClassType {
|
|
public:
|
|
|
|
static inline ClassData<C>* getClassData(HSQUIRRELVM vm) {
|
|
auto* ctx = reinterpret_cast<VMContext*>(sq_getforeignptr(vm));
|
|
return static_cast<ClassData<C>*>(ctx->classes[ClassName()].get());
|
|
/*
|
|
sq_pushregistrytable(vm);
|
|
sq_pushstring(vm, "__classes", -1);
|
|
#ifndef NDEBUG
|
|
SQRESULT r = sq_rawget(vm, -2);
|
|
assert(SQ_SUCCEEDED(r)); // fails if getClassData is called when the data does not exist for the given VM yet (bind the class)
|
|
#else
|
|
sq_rawget(vm, -2);
|
|
#endif
|
|
sq_pushstring(vm, ClassName().c_str(), -1);
|
|
#ifndef NDEBUG
|
|
r = sq_rawget(vm, -2);
|
|
assert(SQ_SUCCEEDED(r)); // fails if getClassData is called when the data does not exist for the given VM yet (bind the class)
|
|
#else
|
|
sq_rawget(vm, -2);
|
|
#endif
|
|
ClassData<C>** ud;
|
|
sq_getuserdata(vm, -1, (SQUserPointer*)&ud, nullptr);
|
|
sq_pop(vm, 3);
|
|
return *ud;
|
|
*/
|
|
}
|
|
|
|
//static WeakPtr<AbstractStaticClassData>& getStaticClassData() {
|
|
// return _ClassType_helper::_getStaticClassData(&typeid(C));
|
|
//}
|
|
static AbstractStaticClassData* getStaticClassData() {
|
|
return StaticClassTypeTag<C>::Get();
|
|
}
|
|
|
|
static inline bool hasClassData(HSQUIRRELVM vm) {
|
|
//if (!getStaticClassData().Expired()) {
|
|
if (StaticClassTypeTag<C>::Get()!=nullptr) {
|
|
VMContext* ctx = reinterpret_cast<VMContext*>(sq_getforeignptr(vm));
|
|
return ctx->classes.exists(ClassName());
|
|
/*
|
|
sq_pushregistrytable(vm);
|
|
sq_pushstring(vm, "__classes", -1);
|
|
if (SQ_SUCCEEDED(sq_rawget(vm, -2))) {
|
|
sq_pushstring(vm, ClassName().c_str(), -1);
|
|
if (SQ_SUCCEEDED(sq_rawget(vm, -2))) {
|
|
sq_pop(vm, 3);
|
|
return true;
|
|
}
|
|
sq_pop(vm, 1);
|
|
}
|
|
sq_pop(vm, 1);
|
|
*/
|
|
}
|
|
return false;
|
|
}
|
|
|
|
//static inline AbstractStaticClassData*& BaseClass() {
|
|
// assert(!getStaticClassData().Expired()); // fails because called before a Sqrat::Class for this type exists
|
|
// return getStaticClassData().Lock()->baseClass;
|
|
//}
|
|
|
|
static inline AbstractStaticClassData*& BaseClass() {
|
|
assert(StaticClassTypeTag<C>::Get()); // fails because called before a Sqrat::Class for this type exists
|
|
return StaticClassTypeTag<C>::Get()->baseClass;
|
|
}
|
|
|
|
//static inline string& ClassName() {
|
|
// assert(!getStaticClassData().Expired()); // fails because called before a Sqrat::Class for this type exists
|
|
// return getStaticClassData().Lock()->className;
|
|
//}
|
|
|
|
static inline string& ClassName() {
|
|
assert(StaticClassTypeTag<C>::Get()); // fails because called before a Sqrat::Class for this type exists
|
|
return StaticClassTypeTag<C>::Get()->className;
|
|
}
|
|
|
|
//static inline COPYFUNC& CopyFunc() {
|
|
// assert(!getStaticClassData().Expired()); // fails because called before a Sqrat::Class for this type exists
|
|
// return getStaticClassData().Lock()->copyFunc;
|
|
//}
|
|
|
|
static inline COPYFUNC& CopyFunc() {
|
|
assert(StaticClassTypeTag<C>::Get()); // fails because called before a Sqrat::Class for this type exists
|
|
return StaticClassTypeTag<C>::Get()->copyFunc;
|
|
}
|
|
|
|
static SQInteger DeleteInstance(SQUserPointer ptr, SQInteger size) {
|
|
SQUNUSED(size);
|
|
auto* instance = reinterpret_cast<std::pair<C*, SharedPtr<std::unordered_map<C*, HSQOBJECT>> >*>(ptr);
|
|
instance->second->erase(instance->first);
|
|
delete instance;
|
|
return 0;
|
|
}
|
|
|
|
static void PushInstance(HSQUIRRELVM vm, C* ptr, std::nullptr_t) {
|
|
if (!ptr) {
|
|
sq_pushnull(vm);
|
|
return;
|
|
}
|
|
|
|
ClassData<C>* cd = getClassData(vm);
|
|
|
|
typename std::unordered_map<C*, HSQOBJECT>::iterator it = cd->instances->find(ptr);
|
|
if (it != cd->instances->end()) {
|
|
sq_pushobject(vm, it->second);
|
|
} else sq_pushnull(vm);
|
|
}
|
|
|
|
static void PushInstance(HSQUIRRELVM vm, C* ptr) {
|
|
if (!ptr) {
|
|
sq_pushnull(vm);
|
|
return;
|
|
}
|
|
|
|
ClassData<C>* cd = getClassData(vm);
|
|
|
|
typename std::unordered_map<C*, HSQOBJECT>::iterator it = cd->instances->find(ptr);
|
|
if (it != cd->instances->end()) {
|
|
sq_pushobject(vm, it->second);
|
|
return;
|
|
}
|
|
|
|
sq_pushobject(vm, cd->classObj);
|
|
sq_createinstance(vm, -1);
|
|
sq_remove(vm, -2);
|
|
sq_setinstanceup(vm, -1, new std::pair<C*, SharedPtr<std::unordered_map<C*, HSQOBJECT>> >(ptr, cd->instances));
|
|
sq_setreleasehook(vm, -1, &DeleteInstance);
|
|
sq_getstackobj(vm, -1, &((*cd->instances)[ptr]));
|
|
}
|
|
|
|
static void PushInstanceCopy(HSQUIRRELVM vm, const C& value) {
|
|
sq_pushobject(vm, getClassData(vm)->classObj);
|
|
sq_createinstance(vm, -1);
|
|
sq_remove(vm, -2);
|
|
#ifndef NDEBUG
|
|
SQRESULT result = CopyFunc()(vm, -1, &value);
|
|
assert(SQ_SUCCEEDED(result)); // fails when trying to copy an object defined as non-copyable
|
|
#else
|
|
CopyFunc()(vm, -1, &value);
|
|
#endif
|
|
}
|
|
|
|
static C* GetInstance(HSQUIRRELVM vm, SQInteger idx, bool nullAllowed = false) {
|
|
//AbstractStaticClassData* classType = nullptr;
|
|
AbstractStaticClassData* classType = StaticClassTypeTag<C>::Get();
|
|
std::pair<C*, SharedPtr<std::unordered_map<C*, HSQOBJECT>> >* instance = nullptr;
|
|
//if (hasClassData(vm)) /* type checking only done if the value has type data else it may be enum */
|
|
if (classType!=nullptr) /* type checking only done if the value has type data else it may be enum */
|
|
{
|
|
if (nullAllowed && sq_gettype(vm, idx) == OT_NULL) {
|
|
return nullptr;
|
|
}
|
|
|
|
//classType = getStaticClassData().Lock().Get();
|
|
|
|
#if !defined (SCRAT_NO_ERROR_CHECKING)
|
|
if (SQ_FAILED(sq_getinstanceup(vm, idx, (SQUserPointer*)&instance, classType))) {
|
|
SQTHROW(vm, FormatTypeError(vm, idx, ClassName()));
|
|
return nullptr;
|
|
}
|
|
|
|
if (instance == nullptr) {
|
|
SQTHROW(vm, _SC("got unconstructed native class (call base.constructor in the constructor of Squirrel classes that extend native classes)"));
|
|
return nullptr;
|
|
}
|
|
#else
|
|
sq_getinstanceup(vm, idx, (SQUserPointer*)&instance, 0);
|
|
#endif
|
|
}
|
|
else /* value is likely of integral type like enums, cannot return a pointer */
|
|
{
|
|
SQTHROW(vm, FormatTypeError(vm, idx, _SC("unknown")));
|
|
return nullptr;
|
|
}
|
|
AbstractStaticClassData* actualType;
|
|
sq_gettypetag(vm, idx, (SQUserPointer*)&actualType);
|
|
if (actualType == nullptr) {
|
|
SQInteger top = sq_gettop(vm);
|
|
sq_getclass(vm, idx);
|
|
while (actualType == nullptr) {
|
|
sq_getbase(vm, -1);
|
|
sq_gettypetag(vm, -1, (SQUserPointer*)&actualType);
|
|
}
|
|
sq_settop(vm, top);
|
|
}
|
|
if (classType != actualType) {
|
|
return static_cast<C*>(actualType->Cast(instance->first, classType));
|
|
}
|
|
return instance->first;
|
|
}
|
|
};
|
|
|
|
/// @endcond
|
|
|
|
}
|