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/vm_fun.hpp"
|
||||
#include "wrenpp/callback_manager.hpp"
|
||||
#include "wrenpp/class_manager.hpp"
|
||||
#include <string_view>
|
||||
#include <algorithm>
|
||||
#include <ctime>
|
||||
|
@ -107,19 +108,6 @@ System.print("You have %(cale.appointment_count()) appointment(s)")
|
|||
}
|
||||
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
|
||||
|
||||
|
@ -131,6 +119,7 @@ int main() {
|
|||
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", "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);
|
||||
|
||||
|
|
|
@ -22,6 +22,7 @@
|
|||
#include <unordered_map>
|
||||
#include <vector>
|
||||
#include <string_view>
|
||||
#include <type_traits>
|
||||
|
||||
namespace wren::detail {
|
||||
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 {
|
||||
public:
|
||||
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 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
|
||||
|
|
|
@ -36,14 +36,7 @@ namespace wren {
|
|||
class VM;
|
||||
class DynafuncMaker;
|
||||
class CallbackManager;
|
||||
|
||||
typedef void(*foreign_method_t)(VM&);
|
||||
typedef void(*finalizer_t)(void*);
|
||||
|
||||
struct foreign_class_t {
|
||||
foreign_method_t allocate;
|
||||
finalizer_t finalize;
|
||||
};
|
||||
class ClassManager;
|
||||
|
||||
enum class SlotType {
|
||||
Bool, Num, Foreign, List, Null, String, Unknown, Map
|
||||
|
@ -114,6 +107,7 @@ namespace wren {
|
|||
void reset (Configuration* conf);
|
||||
|
||||
CallbackManager& callback_manager();
|
||||
ClassManager& class_manager();
|
||||
|
||||
private:
|
||||
struct LocalData;
|
||||
|
|
|
@ -15,13 +15,21 @@
|
|||
* along with wrenpp. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <string_view>
|
||||
#include <tuple>
|
||||
|
||||
namespace wren {
|
||||
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::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
|
||||
|
|
|
@ -62,6 +62,7 @@ wrenpp = library(meson.project_name(),
|
|||
'src/handle.cpp',
|
||||
'src/vm_fun.cpp',
|
||||
'src/callback_manager.cpp',
|
||||
'src/class_manager.cpp',
|
||||
dependencies: [wren_dep],
|
||||
include_directories: public_incl,
|
||||
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/callback_manager.hpp"
|
||||
#include "wrenpp/class_manager.hpp"
|
||||
#include "wrenpp/vm.hpp"
|
||||
#include <iostream>
|
||||
#include <algorithm>
|
||||
|
@ -68,7 +69,11 @@ namespace wren {
|
|||
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) {
|
||||
return vm->callback_manager().callback(is_static, module, class_name, 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_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
|
||||
|
|
|
@ -17,13 +17,11 @@
|
|||
|
||||
#pragma once
|
||||
|
||||
#include "wrenpp/wren_types.hpp"
|
||||
#include <vector>
|
||||
#include <cstddef>
|
||||
|
||||
namespace wren {
|
||||
class VM;
|
||||
typedef void(*foreign_method_t)(VM&);
|
||||
|
||||
class DynafuncMaker {
|
||||
public:
|
||||
DynafuncMaker();
|
||||
|
|
|
@ -18,6 +18,7 @@
|
|||
#include "wrenpp/vm.hpp"
|
||||
#include "wrenpp/configuration.hpp"
|
||||
#include "wrenpp/callback_manager.hpp"
|
||||
#include "wrenpp/class_manager.hpp"
|
||||
#include "dynafunc_maker.hpp"
|
||||
#include <wren.hpp>
|
||||
#include <cassert>
|
||||
|
@ -154,6 +155,7 @@ namespace wren {
|
|||
}
|
||||
|
||||
CallbackManager callback_manager;
|
||||
ClassManager class_manager;
|
||||
detail::Callbacks callbacks;
|
||||
DynafuncMaker dynafunc;
|
||||
WrenVM* wvm;
|
||||
|
@ -322,6 +324,10 @@ namespace wren {
|
|||
return m_local->callback_manager;
|
||||
}
|
||||
|
||||
ClassManager& VM::class_manager() {
|
||||
return m_local->class_manager;
|
||||
}
|
||||
|
||||
void VM::ensure_slots (int num_slots) {
|
||||
wrenEnsureSlots(m_local->wvm, num_slots);
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue