Boost C++ Libraries

...one of the most highly regarded and expertly designed C++ library projects in the world. Herb Sutter and Andrei Alexandrescu, C++ Coding Standards

This is the documentation for a snapshot of the develop branch, built from commit b5267c6e8a.

boost/spirit/home/x3/core/skip_over.hpp

/*=============================================================================
    Copyright (c) 2001-2014 Joel de Guzman
    Copyright (c) 2017 wanghan02
    Copyright (c) 2024 Nana Sakisaka

    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_SKIP_APRIL_16_2006_0625PM)
#define BOOST_SPIRIT_X3_SKIP_APRIL_16_2006_0625PM

#include <boost/spirit/home/x3/support/expectation.hpp>
#include <boost/spirit/home/x3/support/unused.hpp>
#include <boost/spirit/home/x3/support/context.hpp>
#include <boost/spirit/home/x3/support/traits/attribute_category.hpp>
#include <boost/mpl/bool.hpp>
#include <boost/mpl/not.hpp>
#include <boost/type_traits/remove_cv.hpp>
#include <boost/type_traits/remove_reference.hpp>
#include <boost/utility/declval.hpp>
#include <boost/core/ignore_unused.hpp>

namespace boost { namespace spirit { namespace x3
{
    ///////////////////////////////////////////////////////////////////////////
    // Move the /first/ iterator to the first non-matching position
    // given a skip-parser. The function is a no-op if unused_type or
    // unused_skipper is passed as the skip-parser.
    ///////////////////////////////////////////////////////////////////////////
    template <typename Skipper>
    struct unused_skipper : unused_type
    {
        unused_skipper(Skipper const& skipper)
          : skipper(skipper) {}
        Skipper const& skipper;
    };

    namespace detail
    {
        template <typename Skipper>
        struct is_unused_skipper
          : mpl::false_ {};

        template <typename Skipper>
        struct is_unused_skipper<unused_skipper<Skipper>>
          : mpl::true_ {};

        template <>
        struct is_unused_skipper<unused_type>
          : mpl::true_ {};

        template <typename Skipper>
        inline Skipper const&
        get_unused_skipper(Skipper const& skipper)
        {
            return skipper;
        }
        template <typename Skipper>
        inline Skipper const&
        get_unused_skipper(unused_skipper<Skipper> const& unused_skipper)
        {
            return unused_skipper.skipper;
        }

        template <typename Iterator, typename Context, typename Skipper>
        inline void skip_over(
            Iterator& first, Iterator const& last, Context& context, Skipper const& skipper)
        {
        #if BOOST_SPIRIT_X3_THROW_EXPECTATION_FAILURE
            boost::ignore_unused(context);
            while (skipper.parse(first, last, unused, unused, unused))
                /* loop */;
        #else
            if constexpr (std::is_same_v<expectation_failure_t<Context>, unused_type>)
            {
                // The context given by parent was truly `unused_type`.
                // There exists only one such case in core; that is
                // `x3::phrase_parse(...)` which creates a fresh context
                // for the (post)skipper.
                //
                // In that case, it is perfectly fine to pass `unused`
                // because the skipper should have been wrapped
                // like `x3::with<x3::expectation_failure_tag>(failure)[skipper]`.
                // (Note that we have plenty of static_asserts in other
                // locations to detect the absence of the context.)
                //
                // If we encounter this branch in any other situations,
                // that should be a BUG of `expectation_failure` logic.

                while (skipper.parse(first, last, unused, unused, unused))
                    /* loop */;
            }
            else
            {
                // In order to cut the template instantiation chain,
                // we must *forget* the original context at least once
                // during the (recursive) invocation of skippers.
                //
                // Traditionally, implementation detail of `skip_over`
                // was disposing the context because we can clearly assume
                // that any 'context,' including those provided by users,
                // is semantically meaningless as long as we're just
                // *skipping* iterators. As you can see in the other branch,
                // `unused` was passed for that purpose.
                //
                // However, we need to do a quite different thing when the
                // non-throwing expectation_failure mode is enabled.
                //
                // Since the reference bound to `x3::expectation_failure_tag` is
                // provided by the user in the first place, if we do forget it
                // then it will be impossible to resurrect the value afterwards.
                // It will also be problematic for `skip_over` itself because the
                // underlying skipper may (or may not) raise an expectation failure.
                // In traditional mode, the error was thrown by a C++ exception.
                // But how can we propagate that error without throwing?
                //
                // For this reason we're going to cherry-pick the reference
                // and repack it into a brand new context.

                auto const local_ctx = make_context<expectation_failure_tag>(
                    x3::get<expectation_failure_tag>(context));

                while (skipper.parse(first, last, local_ctx, unused, unused))
                    /* loop */;
            }
        #endif
        }

        template <typename Iterator, typename Context>
        inline void skip_over(Iterator&, Iterator const&, Context&, unused_type)
        {
        }

        template <typename Iterator, typename Context, typename Skipper>
        inline void skip_over(
            Iterator&, Iterator const&, Context&, unused_skipper<Skipper> const&)
        {
        }
    }

    // this tag is used to find the skipper from the context
    struct skipper_tag;

    template <typename Context>
    struct has_skipper
      : mpl::not_<detail::is_unused_skipper<
            typename remove_cv<typename remove_reference<
                decltype(x3::get<skipper_tag>(boost::declval<Context>()))
            >::type>::type
        >> {};

    template <typename Iterator, typename Context>
    inline void skip_over(
        Iterator& first, Iterator const& last, Context& context)
    {
        detail::skip_over(first, last, context, x3::get<skipper_tag>(context));
    }
}}}

#endif