Implement ClassManager
Very similar to CallbackManager in its functioning.
This commit is contained in:
parent
74670f4683
commit
258237cbf3
11 changed files with 218 additions and 30 deletions
|
@ -18,6 +18,7 @@
|
||||||
#include "wrenpp/def_configuration.hpp"
|
#include "wrenpp/def_configuration.hpp"
|
||||||
#include "wrenpp/vm_fun.hpp"
|
#include "wrenpp/vm_fun.hpp"
|
||||||
#include "wrenpp/callback_manager.hpp"
|
#include "wrenpp/callback_manager.hpp"
|
||||||
|
#include "wrenpp/class_manager.hpp"
|
||||||
#include <string_view>
|
#include <string_view>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <ctime>
|
#include <ctime>
|
||||||
|
@ -107,19 +108,6 @@ System.print("You have %(cale.appointment_count()) appointment(s)")
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
wren::foreign_class_t foreign_class_fn(
|
|
||||||
wren::VM* vm,
|
|
||||||
std::string_view module,
|
|
||||||
std::string_view class_name
|
|
||||||
) {
|
|
||||||
if (module == "calendar" and class_name == "Calendar") {
|
|
||||||
return wren::make_foreign_class<Calendar>();
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
return {nullptr, nullptr};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
} //unnamed namespace
|
} //unnamed namespace
|
||||||
|
|
||||||
|
@ -131,6 +119,7 @@ int main() {
|
||||||
vm.callback_manager().add_callback(true, "calendar", "Calendar", "today()", &Calendar::today)
|
vm.callback_manager().add_callback(true, "calendar", "Calendar", "today()", &Calendar::today)
|
||||||
.add_callback(false, "calendar", "Calendar", "add_appointment(_)", wren::make_method_bindable<&Calendar::add_appointment>())
|
.add_callback(false, "calendar", "Calendar", "add_appointment(_)", wren::make_method_bindable<&Calendar::add_appointment>())
|
||||||
.add_callback(false, "calendar", "Calendar", "appointment_count()", wren::make_method_bindable<&Calendar::appointment_count>());
|
.add_callback(false, "calendar", "Calendar", "appointment_count()", wren::make_method_bindable<&Calendar::appointment_count>());
|
||||||
|
vm.class_manager().add_class_maker("calendar", "Calendar", &wren::make_foreign_class<Calendar>);
|
||||||
|
|
||||||
vm.interpret("main", g_test_script);
|
vm.interpret("main", g_test_script);
|
||||||
|
|
||||||
|
|
|
@ -22,6 +22,7 @@
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <string_view>
|
#include <string_view>
|
||||||
|
#include <type_traits>
|
||||||
|
|
||||||
namespace wren::detail {
|
namespace wren::detail {
|
||||||
class FullSignatureOwning;
|
class FullSignatureOwning;
|
||||||
|
|
88
include/wrenpp/class_manager.hpp
Normal file
88
include/wrenpp/class_manager.hpp
Normal file
|
@ -0,0 +1,88 @@
|
||||||
|
/* Copyright 2020-2022, Michele Santullo
|
||||||
|
* This file is part of wrenpp.
|
||||||
|
*
|
||||||
|
* Wrenpp is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* Wrenpp 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 General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with wrenpp. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "wren_types.hpp"
|
||||||
|
#include "strings_in_vector.hpp"
|
||||||
|
#include <unordered_map>
|
||||||
|
#include <string>
|
||||||
|
#include <type_traits>
|
||||||
|
|
||||||
|
namespace wren {
|
||||||
|
namespace detail {
|
||||||
|
class ClassNameBase {
|
||||||
|
public:
|
||||||
|
ClassNameBase (const std::string_view& module_name, const std::string_view& class_name);
|
||||||
|
ClassNameBase (const ClassNameBase& other) = default;
|
||||||
|
|
||||||
|
std::size_t module_hash() const noexcept { return m_module_hash; }
|
||||||
|
std::size_t class_hash() const noexcept { return m_class_name_hash; }
|
||||||
|
|
||||||
|
bool operator== (const ClassNameBase& other) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::size_t m_module_hash;
|
||||||
|
std::size_t m_class_name_hash;
|
||||||
|
};
|
||||||
|
|
||||||
|
class ClassNameOwning : StringsInVector<2>, public ClassNameBase {
|
||||||
|
public:
|
||||||
|
ClassNameOwning (const std::string_view& module_name, const std::string_view& class_name);
|
||||||
|
|
||||||
|
std::string_view module_name() const { return this->as_string_view<0>(); }
|
||||||
|
std::string_view class_name() const { return this->as_string_view<1>(); }
|
||||||
|
|
||||||
|
using StringsInVector<2>::operator<;
|
||||||
|
bool operator== (const ClassNameOwning& other) const;
|
||||||
|
};
|
||||||
|
|
||||||
|
class TempClassName : public ClassNameBase {
|
||||||
|
public:
|
||||||
|
TempClassName (std::string_view module_name, std::string_view class_name);
|
||||||
|
TempClassName (const ClassNameOwning& model);
|
||||||
|
bool operator== (const TempClassName& other) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::string_view m_module_name, m_class_name;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ClassNameEqual {
|
||||||
|
using is_transparent = std::true_type;
|
||||||
|
bool operator()(TempClassName left, TempClassName right) const;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ClassNameHash {
|
||||||
|
using is_transparent = std::true_type;
|
||||||
|
std::size_t operator()(TempClassName value) const;
|
||||||
|
};
|
||||||
|
} //namespace detail
|
||||||
|
|
||||||
|
class ClassManager {
|
||||||
|
typedef foreign_class_t(*make_foreign_class_t)();
|
||||||
|
typedef std::unordered_map<detail::ClassNameOwning, make_foreign_class_t, detail::ClassNameHash, detail::ClassNameEqual> ClassMap;
|
||||||
|
public:
|
||||||
|
ClassManager();
|
||||||
|
~ClassManager() noexcept;
|
||||||
|
|
||||||
|
void add_class_maker (std::string_view module_name, std::string_view class_name, make_foreign_class_t);
|
||||||
|
foreign_class_t make_class(std::string_view module_name, std::string_view class_name);
|
||||||
|
|
||||||
|
private:
|
||||||
|
ClassMap m_classes;
|
||||||
|
};
|
||||||
|
} //namespace wren
|
|
@ -25,8 +25,9 @@ namespace wren {
|
||||||
class DefConfiguration : public Configuration {
|
class DefConfiguration : public Configuration {
|
||||||
public:
|
public:
|
||||||
static void write_fn (VM*, wren_string_t text);
|
static void write_fn (VM*, wren_string_t text);
|
||||||
static void error_fn (VM*, ErrorType, wren_string_t module, int line, wren_string_t msg);
|
static void error_fn (VM*, ErrorType, wren_string_t module_name, int line, wren_string_t msg);
|
||||||
static void* reallocate_fn(void* ptr, std::size_t size);
|
static void* reallocate_fn(void* ptr, std::size_t size);
|
||||||
static foreign_method_t foreign_method_fn (VM* vm, wren_string_t module, wren_string_t class_name, bool is_static, wren_string_t signature);
|
static foreign_method_t foreign_method_fn (VM* vm, wren_string_t module_name, wren_string_t class_name, bool is_static, wren_string_t signature);
|
||||||
|
static foreign_class_t foreign_class_fn (wren::VM* vm, std::string_view module_name, std::string_view class_name);
|
||||||
};
|
};
|
||||||
} //namespace wren
|
} //namespace wren
|
||||||
|
|
|
@ -36,14 +36,7 @@ namespace wren {
|
||||||
class VM;
|
class VM;
|
||||||
class DynafuncMaker;
|
class DynafuncMaker;
|
||||||
class CallbackManager;
|
class CallbackManager;
|
||||||
|
class ClassManager;
|
||||||
typedef void(*foreign_method_t)(VM&);
|
|
||||||
typedef void(*finalizer_t)(void*);
|
|
||||||
|
|
||||||
struct foreign_class_t {
|
|
||||||
foreign_method_t allocate;
|
|
||||||
finalizer_t finalize;
|
|
||||||
};
|
|
||||||
|
|
||||||
enum class SlotType {
|
enum class SlotType {
|
||||||
Bool, Num, Foreign, List, Null, String, Unknown, Map
|
Bool, Num, Foreign, List, Null, String, Unknown, Map
|
||||||
|
@ -114,6 +107,7 @@ namespace wren {
|
||||||
void reset (Configuration* conf);
|
void reset (Configuration* conf);
|
||||||
|
|
||||||
CallbackManager& callback_manager();
|
CallbackManager& callback_manager();
|
||||||
|
ClassManager& class_manager();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
struct LocalData;
|
struct LocalData;
|
||||||
|
|
|
@ -15,13 +15,21 @@
|
||||||
* along with wrenpp. If not, see <http://www.gnu.org/licenses/>.
|
* along with wrenpp. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
#include <string_view>
|
#include <string_view>
|
||||||
#include <tuple>
|
#include <tuple>
|
||||||
|
|
||||||
namespace wren {
|
namespace wren {
|
||||||
class VM;
|
class VM;
|
||||||
|
|
||||||
typedef std::string_view wren_string_t;
|
|
||||||
typedef void(*foreign_method_t)(VM&);
|
|
||||||
typedef std::tuple<const char*, const char*> ModuleAndName;
|
typedef std::tuple<const char*, const char*> ModuleAndName;
|
||||||
|
typedef std::string_view wren_string_t;
|
||||||
|
typedef void(*finalizer_t)(void*);
|
||||||
|
typedef void(*foreign_method_t)(VM&);
|
||||||
|
|
||||||
|
struct foreign_class_t {
|
||||||
|
foreign_method_t allocate;
|
||||||
|
finalizer_t finalize;
|
||||||
|
};
|
||||||
} //namespace wren
|
} //namespace wren
|
||||||
|
|
|
@ -62,6 +62,7 @@ wrenpp = library(meson.project_name(),
|
||||||
'src/handle.cpp',
|
'src/handle.cpp',
|
||||||
'src/vm_fun.cpp',
|
'src/vm_fun.cpp',
|
||||||
'src/callback_manager.cpp',
|
'src/callback_manager.cpp',
|
||||||
|
'src/class_manager.cpp',
|
||||||
dependencies: [wren_dep],
|
dependencies: [wren_dep],
|
||||||
include_directories: public_incl,
|
include_directories: public_incl,
|
||||||
install: true,
|
install: true,
|
||||||
|
|
97
src/class_manager.cpp
Normal file
97
src/class_manager.cpp
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
/* Copyright 2020-2022, Michele Santullo
|
||||||
|
* This file is part of wrenpp.
|
||||||
|
*
|
||||||
|
* Wrenpp is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* Wrenpp 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 General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with wrenpp. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "wrenpp/class_manager.hpp"
|
||||||
|
|
||||||
|
namespace wren {
|
||||||
|
namespace detail {
|
||||||
|
ClassNameBase::ClassNameBase (const std::string_view& module_name, const std::string_view& class_name) :
|
||||||
|
m_module_hash(std::hash<std::string_view>{}(module_name)),
|
||||||
|
m_class_name_hash(std::hash<std::string_view>{}(class_name))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ClassNameBase::operator==(const ClassNameBase& other) const {
|
||||||
|
return m_module_hash == other.m_module_hash and
|
||||||
|
m_class_name_hash == other.m_class_name_hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
ClassNameOwning::ClassNameOwning(const std::string_view& module_name, const std::string_view& class_name) :
|
||||||
|
StringsInVector<2>(module_name, class_name),
|
||||||
|
ClassNameBase(module_name, class_name)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ClassNameOwning::operator== (const ClassNameOwning& other) const {
|
||||||
|
return
|
||||||
|
*static_cast<const ClassNameBase*>(this) == static_cast<const ClassNameBase&>(other) or
|
||||||
|
*static_cast<const StringsInVector<2>*>(this) == static_cast<const StringsInVector<2>&>(other);
|
||||||
|
}
|
||||||
|
|
||||||
|
TempClassName::TempClassName(std::string_view module_name, std::string_view class_name) :
|
||||||
|
ClassNameBase(module_name, class_name),
|
||||||
|
m_module_name(module_name),
|
||||||
|
m_class_name(class_name)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
TempClassName::TempClassName (const ClassNameOwning& other) :
|
||||||
|
ClassNameBase(other),
|
||||||
|
m_module_name(other.module_name()),
|
||||||
|
m_class_name(other.class_name())
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TempClassName::operator== (const TempClassName& other) const {
|
||||||
|
return *static_cast<const ClassNameBase*>(this) == static_cast<const ClassNameBase&>(other) and
|
||||||
|
m_module_name == other.m_module_name and
|
||||||
|
m_class_name == other.m_class_name;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ClassNameEqual::operator()(TempClassName left, TempClassName right) const {
|
||||||
|
return left == right;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t ClassNameHash::operator()(TempClassName value) const {
|
||||||
|
return value.module_hash() ^ (value.class_hash() << 1);
|
||||||
|
}
|
||||||
|
} //namespace detail
|
||||||
|
|
||||||
|
ClassManager::ClassManager() = default;
|
||||||
|
ClassManager::~ClassManager() noexcept = default;
|
||||||
|
|
||||||
|
void ClassManager::add_class_maker (std::string_view module_name, std::string_view class_name, make_foreign_class_t maker) {
|
||||||
|
using detail::TempClassName;
|
||||||
|
using detail::ClassNameOwning;
|
||||||
|
|
||||||
|
auto it_found = m_classes.find(TempClassName{module_name, class_name});
|
||||||
|
if (m_classes.cend() == it_found)
|
||||||
|
m_classes.insert(it_found, std::make_pair(ClassNameOwning{module_name, class_name}, maker));
|
||||||
|
else
|
||||||
|
it_found->second = maker;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreign_class_t ClassManager::make_class (std::string_view module_name, std::string_view class_name) {
|
||||||
|
using detail::TempClassName;
|
||||||
|
|
||||||
|
auto it_found = m_classes.find(TempClassName{module_name, class_name});
|
||||||
|
if (m_classes.cend() != it_found)
|
||||||
|
return (*it_found->second)();
|
||||||
|
else
|
||||||
|
return {nullptr, nullptr};
|
||||||
|
}
|
||||||
|
} //namespace wren
|
|
@ -17,6 +17,7 @@
|
||||||
|
|
||||||
#include "wrenpp/def_configuration.hpp"
|
#include "wrenpp/def_configuration.hpp"
|
||||||
#include "wrenpp/callback_manager.hpp"
|
#include "wrenpp/callback_manager.hpp"
|
||||||
|
#include "wrenpp/class_manager.hpp"
|
||||||
#include "wrenpp/vm.hpp"
|
#include "wrenpp/vm.hpp"
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
@ -68,7 +69,11 @@ namespace wren {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
foreign_method_t DefConfiguration::foreign_method_fn (VM* vm, wren_string_t module, wren_string_t class_name, bool is_static, wren_string_t signature) {
|
foreign_method_t DefConfiguration::foreign_method_fn (VM* vm, wren_string_t module_name, wren_string_t class_name, bool is_static, wren_string_t signature) {
|
||||||
return vm->callback_manager().callback(is_static, module, class_name, signature);
|
return vm->callback_manager().callback(is_static, module_name, class_name, signature);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreign_class_t DefConfiguration::foreign_class_fn (wren::VM* vm, std::string_view module_name, std::string_view class_name) {
|
||||||
|
return vm->class_manager().make_class(module_name, class_name);
|
||||||
}
|
}
|
||||||
} //namespace wren
|
} //namespace wren
|
||||||
|
|
|
@ -17,13 +17,11 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include "wrenpp/wren_types.hpp"
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
|
|
||||||
namespace wren {
|
namespace wren {
|
||||||
class VM;
|
|
||||||
typedef void(*foreign_method_t)(VM&);
|
|
||||||
|
|
||||||
class DynafuncMaker {
|
class DynafuncMaker {
|
||||||
public:
|
public:
|
||||||
DynafuncMaker();
|
DynafuncMaker();
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
#include "wrenpp/vm.hpp"
|
#include "wrenpp/vm.hpp"
|
||||||
#include "wrenpp/configuration.hpp"
|
#include "wrenpp/configuration.hpp"
|
||||||
#include "wrenpp/callback_manager.hpp"
|
#include "wrenpp/callback_manager.hpp"
|
||||||
|
#include "wrenpp/class_manager.hpp"
|
||||||
#include "dynafunc_maker.hpp"
|
#include "dynafunc_maker.hpp"
|
||||||
#include <wren.hpp>
|
#include <wren.hpp>
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
|
@ -154,6 +155,7 @@ namespace wren {
|
||||||
}
|
}
|
||||||
|
|
||||||
CallbackManager callback_manager;
|
CallbackManager callback_manager;
|
||||||
|
ClassManager class_manager;
|
||||||
detail::Callbacks callbacks;
|
detail::Callbacks callbacks;
|
||||||
DynafuncMaker dynafunc;
|
DynafuncMaker dynafunc;
|
||||||
WrenVM* wvm;
|
WrenVM* wvm;
|
||||||
|
@ -322,6 +324,10 @@ namespace wren {
|
||||||
return m_local->callback_manager;
|
return m_local->callback_manager;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ClassManager& VM::class_manager() {
|
||||||
|
return m_local->class_manager;
|
||||||
|
}
|
||||||
|
|
||||||
void VM::ensure_slots (int num_slots) {
|
void VM::ensure_slots (int num_slots) {
|
||||||
wrenEnsureSlots(m_local->wvm, num_slots);
|
wrenEnsureSlots(m_local->wvm, num_slots);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue