// // immer: immutable data structures for C++ // Copyright (C) 2016, 2017, 2018 Juan Pedro Bolivar Puente // // This software is distributed under the Boost Software License, Version 1.0. // See accompanying file LICENSE or copy at http://boost.org/LICENSE_1_0.txt // #include <boost/python.hpp> #include <boost/python/suite/indexing/vector_indexing_suite.hpp> #include <immer/vector.hpp> #include <immer/refcount/unsafe_refcount_policy.hpp> namespace { struct heap_t { template <typename ...Tags> static void* allocate(std::size_t size, Tags...) { return PyMem_Malloc(size); } template <typename ...Tags> static void deallocate(std::size_t, void* obj, Tags...) { PyMem_Free(obj); } }; using memory_t = immer::memory_policy< immer::unsafe_free_list_heap_policy<heap_t>, immer::unsafe_refcount_policy>; template <typename Vector> struct immer_vector_indexing_suite : boost::python::vector_indexing_suite< Vector, true, immer_vector_indexing_suite<Vector>> { using value_t = typename Vector::value_type; using index_t = typename Vector::size_type; using object_t = boost::python::object; static void forbidden() { throw std::runtime_error{"immutable!"}; } static void todo() { throw std::runtime_error{"TODO!"}; } static const value_t& get_item(const Vector& v, index_t i) { return v[i]; } static object_t get_slice(const Vector&, index_t, index_t) { todo(); } static void set_item(const Vector&, index_t, const value_t&) { forbidden(); } static void delete_item(const Vector&, index_t) { forbidden(); } static void set_slice(const Vector&, index_t, index_t, const value_t&) { forbidden(); } static void delete_slice(const Vector&, index_t, index_t) { forbidden(); } template <typename Iter> static void set_slice(const Vector&, index_t, index_t, Iter, Iter) { forbidden(); } template <class Iter> static void extend(const Vector& container, Iter, Iter) { forbidden(); } }; } // anonymous namespace BOOST_PYTHON_MODULE(immer_python_module) { using namespace boost::python; using vector_t = immer::vector<object, memory_t>; class_<vector_t>("Vector") .def(immer_vector_indexing_suite<vector_t>()) .def("append", +[] (const vector_t& v, object x) { return v.push_back(std::move(x)); }) .def("set", +[] (const vector_t& v, std::size_t i, object x) { return v.set(i, std::move(x)); }) ; }