// The Construct // // Copyright (C) The Construct Developers, Authors & Contributors // Copyright (C) 2016-2020 Jason Volk // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice is present in all copies. The // full license for this software is available in the LICENSE file. #pragma once #define HAVE_IRCD_SPIRIT_PARSE_H // This file is not part of the IRCd standard include list (stdinc.h) because // it involves extremely expensive boost headers for creating formal spirit // grammars. This file is automatically included in the spirit.h group. namespace ircd { namespace spirit __attribute__((visibility("internal"))) { struct substring_view; template struct custom_parser; BOOST_SPIRIT_TERMINAL(custom0); BOOST_SPIRIT_TERMINAL(custom1); BOOST_SPIRIT_TERMINAL(custom2); template bool parse(const char *&start, const char *const &stop, rule&&, attr&&...); template bool parse(const char *&start, const char *const &stop, rule&&, attr&&...); template bool parse(std::nothrow_t, const char *&start, const char *const &stop, rule&&, attr&&...) noexcept; }} namespace boost { namespace spirit __attribute__((visibility("internal"))) { namespace qi { template struct make_primitive; template struct make_primitive; template struct make_primitive; } template<> struct use_terminal :mpl::true_ {}; template<> struct use_terminal :mpl::true_ {}; template<> struct use_terminal :mpl::true_ {}; }} template struct [[gnu::visibility("internal")]] ircd::spirit::custom_parser :qi::primitive_parser> { template struct attribute { using type = iterator; }; template boost::spirit::info what(context &) const { return boost::spirit::info("custom"); } template bool parse(iterator &, const iterator &, context &, const skipper &, attr &) const; }; template struct [[gnu::visibility("internal")]] boost::spirit::qi::make_primitive { using result_type = ircd::spirit::custom_parser<0>; result_type operator()(unused_type, unused_type) const { return result_type{}; } }; template struct [[gnu::visibility("internal")]] boost::spirit::qi::make_primitive { using result_type = ircd::spirit::custom_parser<1>; result_type operator()(unused_type, unused_type) const { return result_type{}; } }; template struct [[gnu::visibility("internal")]] boost::spirit::qi::make_primitive { using result_type = ircd::spirit::custom_parser<2>; result_type operator()(unused_type, unused_type) const { return result_type{}; } }; struct ircd::spirit::substring_view :ircd::string_view { using _iterator = const char *; using _iterator_range = boost::iterator_range<_iterator>; using _indirect_iterator = karma::detail::indirect_iterator<_iterator>; using _indirect_iterator_range = boost::iterator_range<_indirect_iterator>; explicit substring_view(const _iterator_range &range) :ircd::string_view(std::addressof(*range.begin()), std::addressof(*range.end())) {} explicit substring_view(const _indirect_iterator_range &range) :ircd::string_view(std::addressof(*range.begin()), std::addressof(*range.end())) {} using ircd::string_view::string_view; }; /// Execute the parse. The start pointer is advanced upon successful execution. /// Failures must not throw: If the grammar contains any epsilon expressions or /// callbacks which throw it is UB. This overload exists to force suppression /// of EH from the base of a complex/opaque rule tree. template [[using gnu: always_inline, gnu_inline, artificial]] extern inline bool ircd::spirit::parse(std::nothrow_t, const char *&start, const char *const &stop, rule&& r, attr&&... a) noexcept try { return ircd::spirit::parse(start, stop, std::forward(r), std::forward(a)...); } catch(...) { assert(false); __builtin_unreachable(); } /// Execute the parse. The start pointer is advanced upon successful execution. /// Failures may throw depending on the grammar. Boost's expectation_failure is /// translated into our expectation_failure describing the failure. template [[using gnu: always_inline, gnu_inline, artificial]] extern inline bool ircd::spirit::parse(const char *&start, const char *const &stop, rule&& r, attr&&... a) try { return ircd::spirit::parse(start, stop, std::forward(r), std::forward(a)...); } catch(const qi::expectation_failure &e) { throw expectation_failure { e, error_show_max }; } /// Low-level qi::parse entry point. Throws boost's qi::expectation_failure; /// Try not to allow this exception to escape the calling unit. template [[using gnu: always_inline, gnu_inline, artificial]] extern inline bool ircd::spirit::parse(const char *&start, const char *const &stop, rule&& r, attr&&... a) { return qi::parse(start, stop, std::forward(r), std::forward(a)...); }