mirror of
https://github.com/vcmi/vcmi.git
synced 2025-04-02 22:05:43 +02:00
tak czy siak lua/src jest wykorzystywane w include :( nie umiem zmusić mojego programu svn do współpracy, zwłaszcz z VC
422 lines
12 KiB
C++
422 lines
12 KiB
C++
// Copyright (c) 2004 Daniel Wallin and Arvid Norberg
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the "Software"),
|
|
// to deal in the Software without restriction, including without limitation
|
|
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
// and/or sell copies of the Software, and to permit persons to whom the
|
|
// Software is furnished to do so, subject to the following conditions:
|
|
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF
|
|
// ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
|
|
// TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
|
|
// PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT
|
|
// SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR
|
|
// ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
// ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE
|
|
// OR OTHER DEALINGS IN THE SOFTWARE.
|
|
#include "stdafx.h"
|
|
#include <luabind/lua_include.hpp>
|
|
|
|
#include <luabind/config.hpp>
|
|
#include <luabind/class.hpp>
|
|
|
|
#include <cstring>
|
|
#include <iostream>
|
|
|
|
namespace luabind { namespace detail {
|
|
|
|
struct method_name
|
|
{
|
|
method_name(char const* n): name(n) {}
|
|
bool operator()(method_rep const& o) const
|
|
{ return std::strcmp(o.name, name) == 0; }
|
|
char const* name;
|
|
};
|
|
|
|
struct class_registration : registration
|
|
{
|
|
class_registration(char const* name);
|
|
|
|
void register_(lua_State* L) const;
|
|
|
|
const char* m_name;
|
|
|
|
mutable std::list<detail::method_rep> m_methods;
|
|
|
|
// datamembers, some members may be readonly, and
|
|
// only have a getter function
|
|
mutable std::map<const char*, detail::class_rep::callback, detail::ltstr> m_getters;
|
|
mutable std::map<const char*, detail::class_rep::callback, detail::ltstr> m_setters;
|
|
|
|
// the operators in lua
|
|
mutable std::vector<detail::class_rep::operator_callback> m_operators[detail::number_of_operators];
|
|
mutable std::map<const char*, int, detail::ltstr> m_static_constants;
|
|
|
|
mutable std::vector<class_base::base_desc> m_bases;
|
|
mutable detail::construct_rep m_constructor;
|
|
|
|
void(*m_destructor)(void*);
|
|
void(*m_const_holder_destructor)(void*);
|
|
|
|
void*(*m_extractor)(void*);
|
|
const void*(*m_const_extractor)(void*);
|
|
|
|
void(*m_const_converter)(void*,void*);
|
|
|
|
void(*m_construct_holder)(void*, void*);
|
|
void(*m_construct_const_holder)(void*, void*);
|
|
|
|
void(*m_default_construct_holder)(void*);
|
|
void(*m_default_construct_const_holder)(void*);
|
|
|
|
void(*m_adopt_fun)(void*);
|
|
|
|
int m_holder_size;
|
|
int m_holder_alignment;
|
|
|
|
LUABIND_TYPE_INFO m_type;
|
|
LUABIND_TYPE_INFO m_holder_type;
|
|
LUABIND_TYPE_INFO m_const_holder_type;
|
|
|
|
scope m_scope;
|
|
};
|
|
|
|
class_registration::class_registration(char const* name)
|
|
{
|
|
m_name = name;
|
|
}
|
|
|
|
void class_registration::register_(lua_State* L) const
|
|
{
|
|
LUABIND_CHECK_STACK(L);
|
|
|
|
assert(lua_type(L, -1) == LUA_TTABLE);
|
|
|
|
lua_pushstring(L, m_name);
|
|
|
|
detail::class_rep* crep;
|
|
|
|
detail::class_registry* r = detail::class_registry::get_registry(L);
|
|
// create a class_rep structure for this class.
|
|
// allocate it within lua to let lua collect it on
|
|
// lua_close(). This is better than allocating it
|
|
// as a static, since it will then be destructed
|
|
// when the program exits instead.
|
|
// warning: we assume that lua will not
|
|
// move the userdata memory.
|
|
lua_newuserdata(L, sizeof(detail::class_rep));
|
|
crep = reinterpret_cast<detail::class_rep*>(lua_touserdata(L, -1));
|
|
|
|
new(crep) detail::class_rep(
|
|
m_type
|
|
, m_name
|
|
, L
|
|
, m_destructor
|
|
, m_const_holder_destructor
|
|
, m_holder_type
|
|
, m_const_holder_type
|
|
, m_extractor
|
|
, m_const_extractor
|
|
, m_const_converter
|
|
, m_construct_holder
|
|
, m_construct_const_holder
|
|
, m_default_construct_holder
|
|
, m_default_construct_const_holder
|
|
, m_adopt_fun
|
|
, m_holder_size
|
|
, m_holder_alignment);
|
|
|
|
// register this new type in the class registry
|
|
r->add_class(m_type, crep);
|
|
if (!(LUABIND_TYPE_INFO_EQUAL(m_holder_type, LUABIND_INVALID_TYPE_INFO)))
|
|
{
|
|
// if we have a held type
|
|
// we have to register it in the class-table
|
|
// but only for the base class, if it already
|
|
// exists, we don't have to register it
|
|
detail::class_rep* c = r->find_class(m_holder_type);
|
|
if (c == 0)
|
|
{
|
|
r->add_class(m_holder_type, crep);
|
|
r->add_class(m_const_holder_type, crep);
|
|
}
|
|
}
|
|
|
|
// constructors
|
|
m_constructor.swap(crep->m_constructor);
|
|
|
|
crep->m_getters.swap(m_getters);
|
|
crep->m_setters.swap(m_setters);
|
|
|
|
for (int i = 0; i < detail::number_of_operators; ++i)
|
|
crep->m_operators[i].swap(m_operators[i]);
|
|
|
|
crep->m_static_constants.swap(m_static_constants);
|
|
|
|
typedef std::list<detail::method_rep> methods_t;
|
|
|
|
detail::class_registry* registry = detail::class_registry::get_registry(L);
|
|
|
|
for (std::vector<class_base::base_desc>::iterator i = m_bases.begin();
|
|
i != m_bases.end(); ++i)
|
|
{
|
|
LUABIND_CHECK_STACK(L);
|
|
|
|
// the baseclass' class_rep structure
|
|
detail::class_rep* bcrep = registry->find_class(i->type);
|
|
|
|
detail::class_rep::base_info base;
|
|
base.pointer_offset = i->ptr_offset;
|
|
base.base = bcrep;
|
|
|
|
crep->add_base_class(base);
|
|
|
|
// copy base class table
|
|
crep->get_table(L);
|
|
bcrep->get_table(L);
|
|
lua_pushnil(L);
|
|
|
|
while (lua_next(L, -2))
|
|
{
|
|
lua_pushvalue(L, -2); // copy key
|
|
lua_insert(L, -2);
|
|
lua_settable(L, -5);
|
|
}
|
|
lua_pop(L, 2);
|
|
|
|
// copy base class detaults table
|
|
crep->get_default_table(L);
|
|
bcrep->get_default_table(L);
|
|
lua_pushnil(L);
|
|
|
|
while (lua_next(L, -2))
|
|
{
|
|
lua_pushvalue(L, -2); // copy key
|
|
lua_insert(L, -2);
|
|
lua_settable(L, -5);
|
|
}
|
|
lua_pop(L, 2);
|
|
|
|
}
|
|
|
|
// add methods
|
|
for (std::list<detail::method_rep>::iterator i
|
|
= m_methods.begin(); i != m_methods.end(); ++i)
|
|
{
|
|
LUABIND_CHECK_STACK(L);
|
|
crep->add_method(*i);
|
|
}
|
|
|
|
crep->register_methods(L);
|
|
|
|
m_methods.clear();
|
|
|
|
crep->get_default_table(L);
|
|
m_scope.register_(L);
|
|
lua_pop(L, 1);
|
|
|
|
lua_settable(L, -3);
|
|
}
|
|
|
|
// -- interface ---------------------------------------------------------
|
|
|
|
class_base::class_base(char const* name)
|
|
: scope(std::auto_ptr<registration>(
|
|
m_registration = new class_registration(name))
|
|
)
|
|
{
|
|
}
|
|
|
|
void class_base::init(
|
|
LUABIND_TYPE_INFO type_id
|
|
, LUABIND_TYPE_INFO holder_type
|
|
, LUABIND_TYPE_INFO const_holder_type
|
|
, void*(*extractor)(void*)
|
|
, const void*(*const_extractor)(void*)
|
|
, void(*const_converter_)(void*,void*)
|
|
, void(*holder_constructor_)(void*,void*)
|
|
, void(*const_holder_constructor_)(void*,void*)
|
|
, void(*holder_default_constructor_)(void*)
|
|
, void(*const_holder_default_constructor_)(void*)
|
|
, void(*adopt_fun)(void*)
|
|
, void(*destructor)(void*)
|
|
, void(*const_holder_destructor)(void*)
|
|
, int holder_size
|
|
, int holder_alignment)
|
|
{
|
|
m_registration->m_type = type_id;
|
|
m_registration->m_holder_type = holder_type;
|
|
m_registration->m_const_holder_type = const_holder_type;
|
|
m_registration->m_extractor = extractor;
|
|
m_registration->m_const_extractor = const_extractor;
|
|
m_registration->m_const_converter = const_converter_;
|
|
m_registration->m_construct_holder = holder_constructor_;
|
|
m_registration->m_construct_const_holder = const_holder_constructor_;
|
|
m_registration->m_default_construct_holder = holder_default_constructor_;
|
|
m_registration->m_default_construct_const_holder = const_holder_default_constructor_;
|
|
m_registration->m_destructor = destructor;
|
|
m_registration->m_const_holder_destructor = const_holder_destructor;
|
|
m_registration->m_adopt_fun = adopt_fun;
|
|
m_registration->m_holder_size = holder_size;
|
|
m_registration->m_holder_alignment = holder_alignment;
|
|
}
|
|
|
|
void class_base::add_getter(
|
|
const char* name, const boost::function2<int, lua_State*, int>& g)
|
|
{
|
|
detail::class_rep::callback c;
|
|
c.func = g;
|
|
c.pointer_offset = 0;
|
|
|
|
const char* key = name;
|
|
m_registration->m_getters[key] = c;
|
|
}
|
|
|
|
#ifdef LUABIND_NO_ERROR_CHECKING
|
|
void class_base::add_setter(
|
|
const char* name
|
|
, const boost::function2<int, lua_State*, int>& s)
|
|
#else
|
|
void class_base::add_setter(
|
|
const char* name
|
|
, const boost::function2<int, lua_State*, int>& s
|
|
, int (*match)(lua_State*, int)
|
|
, void (*get_sig_ptr)(lua_State*, std::string&))
|
|
#endif
|
|
{
|
|
detail::class_rep::callback c;
|
|
c.func = s;
|
|
c.pointer_offset = 0;
|
|
|
|
#ifndef LUABIND_NO_ERROR_CHECKING
|
|
c.match = match;
|
|
c.sig = get_sig_ptr;
|
|
#endif
|
|
|
|
|
|
const char* key = name;
|
|
m_registration->m_setters[key] = c;
|
|
}
|
|
|
|
void class_base::add_base(const base_desc& b)
|
|
{
|
|
m_registration->m_bases.push_back(b);
|
|
}
|
|
|
|
void class_base::add_constructor(const detail::construct_rep::overload_t& o)
|
|
{
|
|
m_registration->m_constructor.overloads.push_back(o);
|
|
}
|
|
|
|
void class_base::add_method(const char* name, const detail::overload_rep& o)
|
|
{
|
|
typedef std::list<detail::method_rep> methods_t;
|
|
|
|
methods_t::iterator m = std::find_if(
|
|
m_registration->m_methods.begin()
|
|
, m_registration->m_methods.end()
|
|
, method_name(name));
|
|
if (m == m_registration->m_methods.end())
|
|
{
|
|
m_registration->m_methods.push_back(method_rep());
|
|
m = m_registration->m_methods.end();
|
|
std::advance(m, -1);
|
|
m->name = name;
|
|
}
|
|
|
|
m->add_overload(o);
|
|
m->crep = 0;
|
|
}
|
|
|
|
#ifndef LUABIND_NO_ERROR_CHECKING
|
|
void class_base::add_operator(
|
|
int op_id, int(*func)(lua_State*), int(*matcher)(lua_State*)
|
|
, void(*sig)(lua_State*, std::string&), int arity)
|
|
#else
|
|
void class_base::add_operator(
|
|
int op_id, int(*func)(lua_State*)
|
|
, int(*matcher)(lua_State*), int arity)
|
|
#endif
|
|
{
|
|
detail::class_rep::operator_callback o;
|
|
o.set_fun(func);
|
|
o.set_match_fun(matcher);
|
|
o.set_arity(arity);
|
|
|
|
#ifndef LUABIND_NO_ERROR_CHECKING
|
|
|
|
o.set_sig_fun(sig);
|
|
|
|
#endif
|
|
m_registration->m_operators[op_id].push_back(o);
|
|
}
|
|
|
|
const char* class_base::name() const
|
|
{
|
|
return m_registration->m_name;
|
|
}
|
|
|
|
void class_base::add_static_constant(const char* name, int val)
|
|
{
|
|
m_registration->m_static_constants[name] = val;
|
|
}
|
|
|
|
void class_base::add_inner_scope(scope& s)
|
|
{
|
|
m_registration->m_scope.operator,(s);
|
|
}
|
|
|
|
template<class T>
|
|
void add_custom_name(T i, std::string& s) {}
|
|
|
|
void add_custom_name(std::type_info const* i, std::string& s)
|
|
{
|
|
s += " [";
|
|
s += i->name();
|
|
s += "]";
|
|
}
|
|
|
|
std::string get_class_name(lua_State* L, LUABIND_TYPE_INFO i)
|
|
{
|
|
std::string ret;
|
|
|
|
assert(L);
|
|
|
|
class_registry* r = class_registry::get_registry(L);
|
|
class_rep* crep = r->find_class(i);
|
|
|
|
if (crep == 0)
|
|
{
|
|
ret = "custom";
|
|
add_custom_name(i, ret);
|
|
}
|
|
else
|
|
{
|
|
if (LUABIND_TYPE_INFO_EQUAL(i, crep->holder_type()))
|
|
{
|
|
ret += "smart_ptr<";
|
|
ret += crep->name();
|
|
ret += ">";
|
|
}
|
|
else if (LUABIND_TYPE_INFO_EQUAL(i, crep->const_holder_type()))
|
|
{
|
|
ret += "smart_ptr<const ";
|
|
ret += crep->name();
|
|
ret += ">";
|
|
}
|
|
else
|
|
{
|
|
ret += crep->name();
|
|
}
|
|
}
|
|
return ret;
|
|
};
|
|
|
|
}} // namespace luabind::detail
|
|
|