/************************************************************************************ * * * Copyright (c) 2014 - 2018 Axel Menzel * * * * This file is part of RTTR (Run Time Type Reflection) * * License: MIT License * * * * 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. * * * *************************************************************************************/ #ifndef RTTR_ITERATOR_WRAPPER_H_ #define RTTR_ITERATOR_WRAPPER_H_ #include "rttr/detail/base/core_prerequisites.h" #include "rttr/variant.h" #include #include namespace rttr { namespace detail { using iterator_data = std::aligned_storage::type; template::value <= std::alignment_of::value)> using can_place_in_iterator_data = std::integral_constant; ///////////////////////////////////////////////////////////////////////////////////////// template struct iterator_wrapper_small { using type = Itr; using iterator = Itr; static const Itr& get_iterator(const iterator_data& data) { return reinterpret_cast(data); } static Itr& get_iterator(iterator_data& data) { return reinterpret_cast(data); } static void create(iterator_data& itr_tgt, const iterator& itr_src) { new (&itr_tgt) iterator(itr_src); } static void create(iterator_data& itr_tgt, const iterator_data& itr_src) { new (&itr_tgt) iterator(get_iterator(itr_src)); } static void destroy(iterator_data& itr) { get_iterator(itr).~Itr(); } }; ///////////////////////////////////////////////////////////////////////////////////////// template struct iterator_wrapper_big { using type = Itr; using iterator = Itr; static const Itr& get_iterator(const iterator_data& data) { return *reinterpret_cast(data); } static Itr& get_iterator(iterator_data& data) { return *reinterpret_cast(data); } static void create(iterator_data& itr_tgt, const iterator& itr_src) { reinterpret_cast(itr_tgt) = new iterator(itr_src); } static void create(iterator_data& itr_tgt, const iterator_data& itr_src) { reinterpret_cast(itr_tgt) = new iterator(*reinterpret_cast(itr_src)); } static void destroy(iterator_data& itr) { auto& it = get_iterator(itr); delete ⁢ } }; ///////////////////////////////////////////////////////////////////////////////////////// template::value, iterator_wrapper_small, iterator_wrapper_big>> struct iterator_wrapper_base : Tp { static void advance(iterator_data& itr, std::ptrdiff_t idx) { std::advance(Tp::get_iterator(itr), idx); } static bool equal(const iterator_data& lhs_itr, const iterator_data& rhs_itr) RTTR_NOEXCEPT { return (Tp::get_iterator(lhs_itr) == Tp::get_iterator(rhs_itr)); } }; ///////////////////////////////////////////////////////////////////////////////////////// template struct iterator_wrapper_associative_container : iterator_wrapper_base { static variant get_key(const iterator_data& itr) { auto& it = iterator_wrapper_base::get_iterator(itr); return variant(std::ref(it->first)); } static variant get_value(const iterator_data& itr) { auto& it = iterator_wrapper_base::get_iterator(itr); return variant(std::ref(it->second)); } }; } // end namespace detail } // end namespace rttr #endif // RTTR_ITERATOR_WRAPPER_H_