blob: bb3546141aec693e8edc9d52ae3c29823104c62d [file] [log] [blame]
// Copyright (c) 2001-2011 Hartmut Kaiser
// Copyright (c) 2001-2011 Joel de Guzman
//
// 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_KARMA_DEBUG_HANDLER_APR_21_2010_0148PM)
#define BOOST_SPIRIT_KARMA_DEBUG_HANDLER_APR_21_2010_0148PM
#if defined(_MSC_VER)
#pragma once
#endif
#include <boost/spirit/home/support/unused.hpp>
#include <boost/spirit/home/karma/nonterminal/rule.hpp>
#include <boost/spirit/home/karma/nonterminal/debug_handler_state.hpp>
#include <boost/function.hpp>
#include <boost/fusion/include/at.hpp>
#include <boost/fusion/include/vector.hpp>
#include <boost/fusion/include/out.hpp>
#include <iostream>
namespace boost { namespace spirit { namespace karma
{
template <
typename OutputIterator, typename Context, typename Delimiter
, typename Properties, typename F>
struct debug_handler
{
typedef detail::output_iterator<OutputIterator, Properties>
output_iterator;
typedef detail::enable_buffering<output_iterator> buffer_type;
typedef function<bool(output_iterator&, Context&, Delimiter const&)>
function_type;
debug_handler(function_type subject, F f, std::string const& rule_name)
: subject(subject)
, f(f)
, rule_name(rule_name)
{}
bool operator()(output_iterator& sink, Context& context
, Delimiter const& delim) const
{
buffer_type buffer(sink);
bool r = false;
f (sink, context, pre_generate, rule_name, buffer);
{
detail::disable_counting<output_iterator> nocount(sink);
r = subject(sink, context, delim);
}
if (r)
{
f (sink, context, successful_generate, rule_name, buffer);
buffer.buffer_copy();
return true;
}
f (sink, context, failed_generate, rule_name, buffer);
return false;
}
function_type subject;
F f;
std::string rule_name;
};
template <typename OutputIterator
, typename T1, typename T2, typename T3, typename T4, typename F>
void debug(rule<OutputIterator, T1, T2, T3, T4>& r, F f)
{
typedef rule<OutputIterator, T1, T2, T3, T4> rule_type;
typedef
debug_handler<
OutputIterator
, typename rule_type::context_type
, typename rule_type::delimiter_type
, typename rule_type::properties
, F>
debug_handler;
r.f = debug_handler(r.f, f, r.name());
}
struct simple_trace;
namespace detail
{
// This class provides an extra level of indirection through a
// template to produce the simple_trace type. This way, the use
// of simple_trace below is hidden behind a dependent type, so
// that compilers eagerly type-checking template definitions
// won't complain that simple_trace is incomplete.
template<typename T>
struct get_simple_trace
{
typedef simple_trace type;
};
}
template <typename OutputIterator
, typename T1, typename T2, typename T3, typename T4>
void debug(rule<OutputIterator, T1, T2, T3, T4>& r)
{
typedef rule<OutputIterator, T1, T2, T3, T4> rule_type;
typedef
debug_handler<
OutputIterator
, typename rule_type::context_type
, typename rule_type::delimiter_type
, typename rule_type::properties
, simple_trace>
debug_handler;
typedef typename karma::detail::get_simple_trace<OutputIterator>::type
trace;
r.f = debug_handler(r.f, trace(), r.name());
}
}}}
///////////////////////////////////////////////////////////////////////////////
// Utility macro for easy enabling of rule and grammar debugging
#if !defined(BOOST_SPIRIT_DEBUG_NODE)
#if defined(BOOST_SPIRIT_KARMA_DEBUG)
#define BOOST_SPIRIT_DEBUG_NODE(r) r.name(#r); debug(r)
#else
#define BOOST_SPIRIT_DEBUG_NODE(r) r.name(#r);
#endif
#endif
#endif