| /*============================================================================= |
| Copyright (c) 2001-2014 Joel de Guzman |
| 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_X3_SIMPLE_TRACE_DECEMBER_06_2008_1102AM) |
| #define BOOST_SPIRIT_X3_SIMPLE_TRACE_DECEMBER_06_2008_1102AM |
| |
| #if defined(_MSC_VER) |
| #pragma once |
| #endif |
| |
| #include <boost/spirit/home/x3/support/unused.hpp> |
| #include <boost/spirit/home/x3/support/traits/print_token.hpp> |
| #include <boost/spirit/home/x3/support/traits/print_attribute.hpp> |
| #include <boost/spirit/home/x3/nonterminal/debug_handler_state.hpp> |
| #include <boost/fusion/include/out.hpp> |
| #include <boost/type_traits/is_same.hpp> |
| #include <ostream> |
| |
| // The stream to use for debug output |
| #if !defined(BOOST_SPIRIT_X3_DEBUG_OUT) |
| #define BOOST_SPIRIT_X3_DEBUG_OUT std::cerr |
| #endif |
| |
| // number of tokens to print while debugging |
| #if !defined(BOOST_SPIRIT_X3_DEBUG_PRINT_SOME) |
| #define BOOST_SPIRIT_X3_DEBUG_PRINT_SOME 20 |
| #endif |
| |
| // number of spaces to indent |
| #if !defined(BOOST_SPIRIT_X3_DEBUG_INDENT) |
| #define BOOST_SPIRIT_X3_DEBUG_INDENT 2 |
| #endif |
| |
| namespace boost { namespace spirit { namespace x3 |
| { |
| namespace detail |
| { |
| template <typename Char> |
| inline void token_printer(std::ostream& o, Char c) |
| { |
| // allow customization of the token printer routine |
| x3::traits::print_token(o, c); |
| } |
| } |
| |
| template <int IndentSpaces = 2, int CharsToPrint = 20> |
| struct simple_trace |
| { |
| simple_trace(std::ostream& out) |
| : out(out), indent(0) {} |
| |
| void print_indent(int n) const |
| { |
| n *= IndentSpaces; |
| for (int i = 0; i != n; ++i) |
| out << ' '; |
| } |
| |
| template <typename Iterator> |
| void print_some( |
| char const* tag |
| , Iterator first, Iterator const& last) const |
| { |
| print_indent(indent); |
| out << '<' << tag << '>'; |
| int const n = CharsToPrint; |
| for (int i = 0; first != last && i != n && *first; ++i, ++first) |
| detail::token_printer(out, *first); |
| out << "</" << tag << '>' << std::endl; |
| |
| // $$$ FIXME convert invalid xml characters (e.g. '<') to valid |
| // character entities. $$$ |
| } |
| |
| template <typename Iterator, typename Attribute, typename State> |
| void operator()( |
| Iterator const& first |
| , Iterator const& last |
| , Attribute const& attr |
| , State state |
| , std::string const& rule_name) const |
| { |
| switch (state) |
| { |
| case pre_parse: |
| print_indent(indent++); |
| out |
| << '<' << rule_name << '>' |
| << std::endl; |
| print_some("try", first, last); |
| break; |
| |
| case successful_parse: |
| print_some("success", first, last); |
| if (!is_same<Attribute, unused_type>::value) |
| { |
| print_indent(indent); |
| out |
| << "<attributes>"; |
| traits::print_attribute(out, attr); |
| out |
| << "</attributes>"; |
| out << std::endl; |
| } |
| //~ if (!fusion::empty(context.locals)) |
| //~ out |
| //~ << "<locals>" |
| //~ << context.locals |
| //~ << "</locals>"; |
| print_indent(--indent); |
| out |
| << "</" << rule_name << '>' |
| << std::endl; |
| break; |
| |
| case failed_parse: |
| print_indent(indent); |
| out << "<fail/>" << std::endl; |
| print_indent(--indent); |
| out |
| << "</" << rule_name << '>' |
| << std::endl; |
| break; |
| } |
| } |
| |
| std::ostream& out; |
| mutable int indent; |
| }; |
| |
| namespace detail |
| { |
| typedef simple_trace< |
| BOOST_SPIRIT_X3_DEBUG_INDENT, BOOST_SPIRIT_X3_DEBUG_PRINT_SOME> |
| simple_trace_type; |
| |
| inline simple_trace_type& |
| get_simple_trace() |
| { |
| static simple_trace_type tracer(BOOST_SPIRIT_X3_DEBUG_OUT); |
| return tracer; |
| } |
| } |
| }}} |
| |
| #endif |