// Copyright (c) 2001-2011 Hartmut Kaiser // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) #if !defined(BOOST_SPIRIT_ITERATOR_SPLIT_FUNCTOR_INPUT_POLICY_JAN_17_2008_0103PM) #define BOOST_SPIRIT_ITERATOR_SPLIT_FUNCTOR_INPUT_POLICY_JAN_17_2008_0103PM #include #include #include #include namespace boost { namespace spirit { namespace iterator_policies { namespace split_functor_input_is_valid_test_ { template inline bool token_is_valid(Token const&) { return true; } } /////////////////////////////////////////////////////////////////////////// // class split_functor_input // Implementation of the InputPolicy used by multi_pass // split_functor_input gets tokens from a functor // // This policy should be used when the functor holds two parts of data: a // unique part (unique for each instance of the iterator) and a shared // part (to be shared between the different copies of the same iterator). // Using this policy allows to merge the shared part of the functor with // the shared part of the iterator data, saving one pointer and one // allocation per iterator instance. // // The Functor template parameter of this policy is expected to be a // std::pair, where 'unique' and 'shared' represent the // respective parts of the functor itself. // // Note: the unique part of the functor must have a typedef for result_type // It also must have a static variable of type result_type defined // to represent EOF that is called eof. // /////////////////////////////////////////////////////////////////////////// struct split_functor_input { /////////////////////////////////////////////////////////////////////// template ::value> class unique; // the unique part of the functor is empty, do not include the functor // as a member at all to avoid unnecessary padding bytes to be included // into the generated structure template class unique // : public detail::default_input_policy { protected: typedef typename Functor::first_type functor_type; typedef typename functor_type::result_type result_type; public: typedef result_type value_type; typedef std::ptrdiff_t difference_type; typedef std::ptrdiff_t distance_type; typedef result_type const* pointer; typedef result_type const& reference; protected: unique() {} explicit unique(Functor const&) {} public: void swap(unique&) {} // get the next token template static typename MultiPass::reference get_input(MultiPass& mp) { value_type& curtok = mp.shared()->curtok; using namespace split_functor_input_is_valid_test_; if (!token_is_valid(curtok)) functor_type::get_next(mp, curtok); return curtok; } template static void advance_input(MultiPass& mp) { functor_type::get_next(mp, mp.shared()->curtok); } // test, whether we reached the end of the underlying stream template static bool input_at_eof(MultiPass const& mp) { return mp.shared()->curtok == functor_type::eof; } template static bool input_is_valid(MultiPass const&, value_type const& t) { using namespace split_functor_input_is_valid_test_; return token_is_valid(t); } template static void destroy(MultiPass& mp) { functor_type::destroy(mp); } }; // the unique part of the functor is non-empty template class unique : public unique { protected: typedef typename Functor::first_type functor_type; typedef typename functor_type::result_type result_type; protected: unique() {} explicit unique(Functor const& x) : ftor(x.first) {} void swap(unique& x) { boost::swap(ftor, x.ftor); } public: typedef result_type value_type; typedef std::ptrdiff_t difference_type; typedef std::ptrdiff_t distance_type; typedef result_type const* pointer; typedef result_type const& reference; public: // get the next token template static typename MultiPass::reference get_input(MultiPass& mp) { value_type& curtok = mp.shared()->curtok; using namespace split_functor_input_is_valid_test_; if (!token_is_valid(curtok)) functor_type::get_next(mp, curtok); return curtok; } template static void advance_input(MultiPass& mp) { mp.ftor.get_next(mp, mp.shared()->curtok); } template static bool input_is_valid(MultiPass const&, value_type const& t) { using namespace split_functor_input_is_valid_test_; return token_is_valid(t); } // test, whether we reached the end of the underlying stream template static bool input_at_eof(MultiPass const& mp) { return mp.shared()->curtok == mp.ftor.eof; } typename Functor::first_type& get_functor() const { return ftor; } mutable functor_type ftor; }; /////////////////////////////////////////////////////////////////////// template struct shared { protected: typedef typename Functor::first_type functor_type; typedef typename functor_type::result_type result_type; public: explicit shared(Functor const& x) : ftor(x.second), curtok(0) {} mutable typename Functor::second_type ftor; result_type curtok; private: // silence MSVC warning C4512: assignment operator could not be generated shared& operator= (shared const&); }; }; }}} #endif