mirror of
https://github.com/boostorg/python.git
synced 2026-01-24 06:02:14 +00:00
151 lines
4.6 KiB
C++
151 lines
4.6 KiB
C++
// Copyright David Abrahams 2001. Permission to copy, use,
|
|
// modify, sell and distribute this software is granted provided this
|
|
// copyright notice appears in all copies. This software is provided
|
|
// "as is" without express or implied warranty, and with no claim as
|
|
// to its suitability for any purpose.
|
|
#include <boost/python/converter/registry.hpp>
|
|
#include <boost/python/converter/registrations.hpp>
|
|
#include <boost/python/converter/builtin_converters.hpp>
|
|
#include <set>
|
|
#include <stdexcept>
|
|
|
|
#ifdef BOOST_PYTHON_TRACE_REGISTRY
|
|
# include <iostream>
|
|
#endif
|
|
|
|
namespace boost { namespace python { namespace converter {
|
|
|
|
namespace // <unnamed>
|
|
{
|
|
typedef registration entry;
|
|
|
|
typedef std::set<entry> registry_t;
|
|
|
|
registry_t& entries()
|
|
{
|
|
static registry_t registry;
|
|
|
|
# ifdef BOOST_PYTHON_DYNAMIC_LIB // this conditional should go away eventually.
|
|
static bool builtin_converters_initialized = false;
|
|
if (!builtin_converters_initialized)
|
|
{
|
|
// Make this true early because registering the builtin
|
|
// converters will cause recursion.
|
|
builtin_converters_initialized = true;
|
|
|
|
initialize_builtin_converters();
|
|
}
|
|
# ifdef BOOST_PYTHON_TRACE_REGISTRY
|
|
std::cout << "registry: ";
|
|
for (registry_t::iterator p = registry.begin(); p != registry.end(); ++p)
|
|
{
|
|
std::cout << p->target_type << "; ";
|
|
}
|
|
std::cout << '\n';
|
|
# endif
|
|
# endif
|
|
return registry;
|
|
}
|
|
|
|
entry* get(type_info type)
|
|
{
|
|
# ifdef BOOST_PYTHON_TRACE_REGISTRY
|
|
registry_t::iterator p = entries().find(entry(type));
|
|
|
|
std::cout << "looking up " << type
|
|
<< (p == entries().end() || p->target_type != type
|
|
? "...NOT found\n" : "...found\n");
|
|
# endif
|
|
return const_cast<entry*>(
|
|
&*entries().insert(entry(type)).first
|
|
);
|
|
}
|
|
} // namespace <unnamed>
|
|
|
|
namespace registry
|
|
{
|
|
void insert(to_python_function_t f, type_info source_t)
|
|
{
|
|
# ifdef BOOST_PYTHON_TRACE_REGISTRY
|
|
std::cout << "inserting to_python " << source_t << "\n";
|
|
# endif
|
|
to_python_function_t& slot = get(source_t)->to_python;
|
|
|
|
assert(slot == 0); // we have a problem otherwise
|
|
if (slot != 0)
|
|
{
|
|
throw std::runtime_error(
|
|
"trying to register to_python_converter for a type which already has a registered to_python_converter");
|
|
}
|
|
slot = f;
|
|
}
|
|
|
|
// Insert an lvalue from_python converter
|
|
void insert(convertible_function convert, type_info key)
|
|
{
|
|
# ifdef BOOST_PYTHON_TRACE_REGISTRY
|
|
std::cout << "inserting lvalue from_python " << key << "\n";
|
|
# endif
|
|
entry* found = get(key);
|
|
lvalue_from_python_chain *registration = new lvalue_from_python_chain;
|
|
registration->convert = convert;
|
|
registration->next = found->lvalue_chain;
|
|
found->lvalue_chain = registration;
|
|
|
|
insert(convert, 0, key);
|
|
}
|
|
|
|
// Insert an rvalue from_python converter
|
|
void insert(void* (*convertible)(PyObject*)
|
|
, constructor_function construct
|
|
, type_info key)
|
|
{
|
|
# ifdef BOOST_PYTHON_TRACE_REGISTRY
|
|
std::cout << "inserting rvalue from_python " << key << "\n";
|
|
# endif
|
|
entry* found = get(key);
|
|
rvalue_from_python_chain *registration = new rvalue_from_python_chain;
|
|
registration->convertible = convertible;
|
|
registration->construct = construct;
|
|
registration->next = found->rvalue_chain;
|
|
found->rvalue_chain = registration;
|
|
}
|
|
|
|
// Insert an rvalue from_python converter
|
|
void push_back(void* (*convertible)(PyObject*)
|
|
, constructor_function construct
|
|
, type_info key)
|
|
{
|
|
# ifdef BOOST_PYTHON_TRACE_REGISTRY
|
|
std::cout << "push_back rvalue from_python " << key << "\n";
|
|
# endif
|
|
rvalue_from_python_chain** found = &get(key)->rvalue_chain;
|
|
while (*found != 0)
|
|
found = &(*found)->next;
|
|
|
|
rvalue_from_python_chain *registration = new rvalue_from_python_chain;
|
|
registration->convertible = convertible;
|
|
registration->construct = construct;
|
|
registration->next = 0;
|
|
*found = registration;
|
|
}
|
|
|
|
registration const& lookup(type_info key)
|
|
{
|
|
return *get(key);
|
|
}
|
|
|
|
registration const* query(type_info type)
|
|
{
|
|
registry_t::iterator p = entries().find(entry(type));
|
|
# ifdef BOOST_PYTHON_TRACE_REGISTRY
|
|
std::cout << "querying " << type
|
|
<< (p == entries().end() || p->target_type != type
|
|
? "...NOT found\n" : "...found\n");
|
|
# endif
|
|
return (p == entries().end() || p->target_type != type) ? 0 : &*p;
|
|
}
|
|
} // namespace registry
|
|
|
|
}}} // namespace boost::python::converter
|