Assuming that the grammar rules are ready, and the actions and states prepared, performing a parsing run consists of two steps:
- Constructing an input class that represents the to-be-parsed data.
- Calling a PEGTL parse function with the input (and any states).
using namespace tao::pegtl;
struct my_grammar : ...;
template< typename Rule >
struct my_actions {};
// Specialisations of my_actions as required...
bool my_parse( const std::string& filename, my_state& state )
{
file_input in( filename );
return parse< my_grammar, my_actions >( in, state );
}
In the context of PEGTL input classes and positions there is usually an additional (i.e. beyond indicating or supplying the to-be-parsed data) string parameter source
that identifies where the to-be-parsed data comes from.
For example when parsing a file with one of the appropriate included input classes, the filename is automatically used as source
so that it will appear in exceptions and error messages.
In other cases the source
parameter needs to be explicitly passed to the input's constructor.
All classes and functions on this page are in namespace tao::pegtl
.
- Tracking Mode
- Line Ending
- Source
- File Input
- Memory Input
- String Input
- Stream Inputs
- Argument Input
- Parse Function
- Nested Parsing
- Incremental Input
- Error Reporting
- Deduction Guides
Some input classes allow a choice of tracking mode, or whether the byte
, line
and column
counters are continuously updated during a parsing run with tracking_mode::eager
, or only calculated on-demand in position()
by scanning the complete input again with tracking_mode::lazy
.
Lazy tracking is recommended when the position is used very infrequently, for example only in the case of throwing a parse_error
.
Eager tracking is recommended when the position is used frequently and/or in non-exceptional cases, for example when annotating every AST node with the line number.
All input classes allow the choice of which line endings should be recognised by the eol
and eolf
rules, and used for line counting.
The supported line endings are cr
, a single carriage-return/"\r"
/0x0d
character as used on classic Mac OS, lf
, a single line-feed/"\n"
/0x0a
as used on Unix, Linux, Mac OS X and macOS, and crlf
, a sequence of both as used on MS-DOS and Windows.
The default template parameter for all input classes is eol::lf_crlf
which recognises both Unix and MS-DOS line endings.
The supplied alternatives are eol::cr
, eol::lf
, eol::crlf
and eol::cr_crlf
.
Some input classes allow a choice of how to store the source parameter, with the default being a std::string
.
When creating many instances of an input class, it can be changed to a non-owning const char*
to optimise away the memory allocation performed by std::string
.
The classes file_input<>
, read_input<>
and, on supported platforms, mmap_input<>
, can be used to parse the contents of a file.
read_input<>
uses C "stdio" facilities to read the file.mmap_input<>
usesmmap(2)
on POSIX compliant systems orMapViewOfFile()
on Windows.file_input<>
is derived frommmap_input<>
when available, andread_input<>
otherwise, inheriting the respective constructors.
They immediately make available the complete contents of the file; read_input<>
reads the entire file upon construction.
The constructors that take a FILE*
argument take ownership of the file pointer, i.e. they fclose()
it in the destructor.
template< tracking_mode P = tracking_mode::eager, typename Eol = eol::lf_crlf >
struct read_input
{
explicit read_input( const std::filesystem::path& path );
read_input( const std::filesystem::path& path, const std::string& source );
read_input( FILE* file, const std::filesystem::path& path );
read_input( FILE* file, const std::filesystem::path& path, const std::string& source );
};
template< tracking_mode P = tracking_mode::eager, typename Eol = eol::lf_crlf >
struct mmap_input
{
explicit mmap_input( const std::filesystem::path& path );
mmap_input( const std::filesystem::path& path, const std::string& source );
};
template< tracking_mode P = tracking_mode::eager, typename Eol = eol::lf_crlf >
using file_input = mmap_input< P, Eol >; // Or read_input when no mmap_input available.
The class memory_input<>
can be used to parse existing contiguous blocks of memory like the contents of a std::string
.
The input neither copies the data nor takes ownership, it only keeps pointers.
The various constructors accept the to-be-parsed data in different formats.
The source
parameter is required for all constructors to disambiguate the different overloads.
If you don't want to specify a source just use the empty string (""
).
The constructors that only takes a const char* begin
for the data uses std::strlen()
to determine the length.
It will therefore only work correctly with data that is terminated with a 0-byte (and does not contain embedded 0-bytes, which are otherwise fine).
The constructors that take additional byte
, line
and column
arguments initialise the internal counters with the supplied values, rather than the defaults of 0
, 1
and 1
.
template< tracking_mode P = tracking_mode::eager, typename Eol = eol::lf_crlf, typename Source = std::string >
class memory_input
{
template< typename T >
memory_input( const internal::iterator& iter, const char* end, T&& source ) noexcept(...);
template< typename T >
memory_input( const char* begin, const char* end, T&& source ) noexcept(...);
template< typename T >
memory_input( const char* begin, const std::size_t size, T&& source ) noexcept(...);
template< typename T >
memory_input( const std::string& string, T&& source ) noexcept(...);
template< typename T >
memory_input( const char* begin, T&& source ) noexcept(...);
template< typename T >
memory_input( const char* begin, const char* end, T&& source,
const std::size_t byte, const std::size_t line, const std::size_t column ) noexcept(...);
};
memory_input in1( "this is the input to parse", "" );
Construct a memory_input
with default tracking mode, default end-of-line mode (accepting Unix and MS-DOS line endings), and default source storage.
As there are only two parameters, the 5th overload from above is choosen.
The data to parse is given directly as a string literal which is not copied.
As no pointer to the end or the size of the input is given, the length of the data to be parsed will be determined by calling strlen
on the pointer passed as the first parameter.
The source is the empty string.
struct packet
{
// ...
const std::array< char >& buffer() const noexcept;
std::string identifier() const;
// ...
};
packet p = ...; // some UDP packet class
memory_input< tracking_mode::lazy, eol::crlf > in2( p.buffer().begin(), p.buffer().end(), p.identifier() );
Consider a UDP packet that was received and should be parsed.
Construct a memory_input
with lazy tracking mode, MS-DOS end-of-line mode (accepting only MS-DOS line endings), and default source storage.
This example chooses the second overload from above.
The data to parse is given as two const char*
pointers (as the data is not null-terminated) and is, of course, not copied.
Consider the source to be an identifier for the packet that was received, e.g. a string constructed from the timestamp, the source IP/port, the interface it was received on, a sequence number, or similar information.
Note that this example shows why the source parameter is necessary to disambiguate the overloads.
If the source would be optional (defaulted), the signature of this overload would also match the first example and therefore be ambiguous.
Note that noexcept(...)
is a conditional noexcept-specification, depending on whether the construction of the source stored in the class can throw given the perfectly-forwarded parameter source
. Technically, it is implemented as noexcept( std::is_nothrow_constructible< Source, T&& >::value )
.
With the default Source
type of std::string
, the source
parameter to the constructors is usually a const char*
or (any reference to) a std::string
, but anything that can be used to construct a std::string
will work. When Source
is set to const char*
then only a const char *
(or something that can implicitly be converted to one) will work.
The implementation of the constructors is different than shown. They should be used "as if" this was the actual signature.
The class string_input<>
can also be used to parse a std::string
.
Unlike class memory_input<>
, this class stores a copied (or moved) version of the data for which it takes ownership.
template< tracking_mode P = tracking_mode::eager, typename Eol = eol::lf_crlf, typename Source = std::string >
class string_input
{
template< typename V, typename T >
string_input( V&& data, T&& source ) noexcept(...);
template< typename V, typename T >
string_input( V&& data, T&& source,
const std::size_t byte, const std::size_t line, const std::size_t column ) noexcept(...);
};
std::string content(); // returns the content
string_input in1( content(), "from_content" );
Construct a string_input
with default tracking mode, default end-of-line mode (accepting Unix and MS-DOS line endings), and default source storage.
The data returned from calling content()
is copied into the input.
The source is from_content
.
Note that the implementation of the constructors is different than shown. They should be used "as if" this was the actual signature.
The classes cstream_input<>
and istream_input<>
can be used to parse data from C-streams (std::FILE*
) and C++-streams (std::istream
), respectively.
Unlike the file inputs above, they internally use buffer_input<>
and therefore do not read the complete stream upon construction.
They all have a single constructor that takes a stream, the maximum buffer size, and the name of the source.
Note that these classes only keep a pointer/reference to the stream and do not take ownership; in particular cstream_input<>
does not call std::close()
.
See Incremental Input for details on the maximum
argument, and how to use the mandatory discard facilities.
template< typename Eol = eol::lf_crlf >
struct cstream_input
{
cstream_input( std::FILE* stream, const std::size_t maximum, const char* source );
cstream_input( std::FILE* stream, const std::size_t maximum, const std::string& source );
};
template< typename Eol = eol::lf_crlf >
struct istream_input
{
istream_input( std::istream& stream, const std::size_t maximum, const char* source );
istream_input( std::istream& stream, const std::size_t maximum, const std::string& source );
};
Note that the implementation of the constructors is different than shown. They should be used "as if" this was the actual signature.
The class argv_input<>
can be used to parse a string passed from the command line.
template< tracking_mode P = tracking_mode::eager, typename Eol = eol::lf_crlf >
class argv_input
{
argv_input( char** argv, const std::size_t n );
argv_input( char** argv, const std::size_t n, const char* source );
argv_input( char** argv, const std::size_t n, const std::string& source );
};
If no source
is given, the source is set to "argv[N]"
where N is the string representation of n
.
Note that the implementation of the constructors is different than shown. They should be used "as if" this was the actual signature.
The parse functions accept the following template parameters and arguments:
- The
Rule
class represents the top-level parsing rule of the grammar and is mandatory. - The
Action<>
class template is required to actually do something during a parsing run. - The
Control<>
class template is only required for grammar debugging or some advanced uses. - The
States
are the types of the objects that are passed to all actions and control hooks.
Additionally, two enumeration values can be used to control the behaviour:
- The
apply_mode
which can also be set tonothing
in order to disable action invocations, just like thedisable<>
rule does. - The
rewind_mode
which can also be set todontcare
in order to not require rewinding of the input on local failure, a micro optimisation.
The result of a parsing run, i.e. an invocation of tao::pegtl::parse()
, can be either
- success, a return value of
true
, - local failure, a return value of
false
, - global failure, an exception of type
tao::pegtl::parse_error
, or - any other exception thrown by the input class or an action function.
template< typename Rule,
template< typename... > class Action = nothing,
template< typename... > class Control = normal,
apply_mode A = apply_mode::action,
rewind_mode M = rewind_mode::required,
typename ParseInput,
typename... States >
bool parse( ParseInput& in,
States&&... st );
Nested parsing refers to an (inner) parsing run that is performed "in the middle of" another (outer) parsing run, for example when one file "includes" another file.
The difference to the regular tao::pegtl::parse()
function is that tao::pegtl::parse_nested()
takes care of adding to the std::vector
of tao::pegtl::position
objects in the exception class tao::pegtl::parse_error
.
This allows generating error messages of the form "error in file F1 line L1 included from file F2 line L2...".
Calling parse_nested()
requires one additional argument compared to parse()
, the input from the outer parsing run as first argument.
Everything else remains the same.
template< typename Rule,
template< typename... > class Action = nothing,
template< typename... > class Control = normal,
apply_mode A = apply_mode::action,
rewind_mode M = rewind_mode::required,
typename OuterInput,
typename ParseInput,
typename... States >
bool parse_nested( const OuterInput& oi,
ParseInput& in,
States&&... st );
The PEGTL is designed and optimised for parsing single contiguous blocks of memory like a memory-mapped file or the contents of a std::string
.
In cases where the data does not fit into memory, or other reasons prevent parsing the data as single memory block, an incremental input can be used.
This allows parsing with only (small) portions of the input in a memory buffer at any single time. The buffer is filled automatically, however the discard facilities must be used to regularly flush the buffer and make space for a new portion of input data.
The stream inputs are ready-to-use input classes for C++-style and C-style streams. Apart from having to use the discard facilities, and some extra care when implementing custom rules, they can be used just like any other input class.
The stream inputs, and all other inputs based on buffer_input<>
, contain a buffer that is allocated in the constructor.
The buffer capacity is the sum of a maximum value and a chunk size.
The maximum value is passed to the constructor as function argument, the chunk size is a (rarely changed) template parameter. The required buffer capacity depends on the grammar, the actions, and the input data.
The buffer must be able to hold
- any and all data for look-ahead performed by the grammar,
- any and all data for back-tracking performed by the grammar,
- any and all data for actions'
apply()
(notapply0()
).
For example consider an excerpt from the JSON grammar from include/tao/pegtl/contrib/json.hpp
.
struct xdigit : abnf::HEXDIG {};
struct unicode : list< seq< one< 'u' >, rep< 4, must< xdigit > > >, one< '\\' > > {};
struct escaped_char : one< '"', '\\', '/', 'b', 'f', 'n', 'r', 't' > {};
struct escaped : sor< escaped_char, unicode > {};
struct unescaped : utf8::range< 0x20, 0x10FFFF > {};
struct char_ : if_then_else< one< '\\' >, must< escaped >, unescaped > {};
struct string_content : until< at< one< '"' > >, must< char_ > > {};
struct string : seq< one< '"' >, must< string_content >, any >
{
using content = string_content;
};
The rule string_content
matches JSON strings as they occur in a JSON document.
If an action with apply()
(rather than apply0()
) is attached to the string_content
rule, the buffer capacity is an upper bound on the length of the JSON strings that can be processed.
If the actions are only attached to say unescaped
, escaped_char
and rep< 4, must< xdigit > >
, the latter because it, too, occurs in an (implicit, inside of list
) unbounded loop, then the JSON strings are processed unescaped-character-by-unescaped-character and escape-sequence-by-escape-sequence.
As long as the buffer is discarded frequently, like after every unescaped character and every single escape sequence, a buffer capacity as small as 8 or 12 should suffice for parsing arbitrarily long JSON strings.
Note that the eof
rule requires at least one byte of free buffer space when there is no unconsumed data in the buffer.
To prevent the buffer from overflowing, the discard()
member function of class buffer_input<>
must be called regularly.
Discarding invalidates all pointers to the input's data and MUST NOT be used where backtracking to before the discard might occur AND/OR nested within a rule for which an action with input can be called.
Calling discard()
on a non-buffered input is an empty method and will be optimised away completely.
Usually you don't call discard()
manually. Instead, one of the two following methods might be used.
The discard
rule behaves just like the success
rule but calls the discard function on the input before returning true
.
The tao::pegtl::discard_input
, tao::pegtl::discard_input_on_success
and tao::pegtl::discard_input_on_failure
actions can be used to discard input non-intrusively, i.e. without changing the grammar like with the discard
rule.
These actions are used in the usual way, by deriving a custom action class template specialisation from them.
In the case of discard_input
, the input is discarded unconditionally after every match attempt of the rule that the action is attached to.
As discard_input
is based on the match()
method, it is unaffected by enabling or disabling actions (which only applies to the apply
/apply0
-methods).
The other two variants behave as implied by their respective names, keeping in mind that "failure" is to be understood as "local failure" (false), no discard is performed on global failure (exception). Similarly "unconditional" is wrt. success or local failure, not global failure.
template<>
struct my_action< R >
: tao::pegtl::discard_input
{
// It is safe to implement apply() here if appropriate:
// discard() will be called by discard_input's match()
// only _after_ calling this action's apply().
};
In practice, since the "must"-rules like must<>
and if_must<>
inhibit backtracking, they can be good indicators of where to perform a discard.
For example consider again this rule from the JSON grammar from include/tao/pegtl/contrib/json.hpp
.
struct unicode : list< seq< one< 'u' >, rep< 4, must< xdigit > > >, one< '\\' > > {};
The xdigit
rule is within a must
, wherefore we know that no backtracking is possible, so we could discard after xdigit
or must< xdigit >
.
However then we can't attach an action with apply()
to the rep< 4, ... >
since we would be discarding after every single digit.
This is not ideal, it would be more efficient to process all four xdigits in a single action invocation.
Looking close we can see that backtracking to before the rep<>
is actually impossible because once the list<>
has successfully matched seq< one< 'u' >, rep< 4, must< xdigit > > >
it will never go back.
It will attempt to match another backslash, the list item separator, and if successful loop to the seq<>
, but once the next character is a 'u'
, the must<>
in the rep
seals the deal, there is no way to not complete the next list entry.
Therefore we can safely attach an action to the rep<>
that processes the four xdigits and then discards the input.
template<>
struct my_action< rep< 4, must< xdigit > >
: tao::pegtl::discard_input
{
template< typename ActionInput >
static void apply( const ActionInput& in, /* the states */ )
{
assert( in.size() == 4 );
// process the 4 xdigits
}
};
Another good candidate in the JSON grammar to discard after is the tao::pegtl::json::value
rule...
All incremental inputs included with the library and documented here are based on buffer_input<>
.
A custom rule that is compatible with incremental inputs needs to pay attention to the amount
argument in the input's interface.
Unlike the inputs based on memory_input<>
, the size( amount )
and end( amount )
member functions do not ignore the amount
argument, and the require( amount )
member function is not a complete dummy.
template< ... >
class buffer_input
{
bool empty();
std::size_t size( const std::size_t amount );
const char* end( const std::size_t amount );
void require( const std::size_t amount );
...
};
The require( amount )
member function tells the input to make available at least amount
unconsumed bytes of input data.
It is not normally called directly unless there is good reason to prefetch some data.
The empty()
, size( amount )
and end( amount )
member functions call require( amount )
, or, in the case of empty()
, require( 1 )
.
The amount
parameter should be understood as a parsing rule wishing to inspect and consume up to amount
bytes of input.
A custom rule must make sure to use appropriate values of amount
.
For examples of how the amount
is set by parsing rules please search for in.size
in include/tao/pegtl/internal/
.
An incremental input consists of buffer_input<>
together with a reader, a class or function that is used by the buffer input to fill the buffer.
The buffer input is a class template with multiple template parameters.
template< typename Reader,
typename Eol = eol::lf_crlf,
typename Source = std::string,
std::size_t Chunk = 64 >
class buffer_input;
The Eol
and Source
parameters are like for the other input classes.
The Chunk
parameter is explained below in detail.
The Reader
can be anything that can be called like the following wrapper.
std::function< std::size_t( char* buffer, const std::size_t length ) >
The arguments and return value are similar to other read()
-style functions, a request to read length
bytes into the memory pointed to by buffer
that returns the number of bytes actually read.
Reaching the end of the input MUST be the only reason for the reader to return zero.
The reader might be called again after returning zero, with the expectation of returning zero again.
Note that buffer_input<>
consumes the first two arguments to its constructor for the source and maximum, and uses perfect forwarding to pass everything else to the constructor of the embedded instance of Reader
.
For examples of how to implement readers please look at istream_reader.hpp
and cstream_reader.hpp
in include/tao/pegtl/internal/
.
The buffer input's Chunk
template parameter is actually used in multiple places.
- The
maximum
buffer capacity passed by the user is incremented byChunk
. - A discard does nothing when there are less than
Chunk
bytes of consumed buffered data. - The buffer input requests at least
Chunk
bytes from the reader if there is enough space.
Note that the first and second point go hand-in-hand, in order to optimise away some discards, the buffer must be extended in order to guarantee that at least maximum
bytes can be buffered after a call to discard, even when it does nothing. The third point is simply an optimisation to call the reader less frequently.
When reporting an error, one often wants to print the complete line from the input where the error occurred and a marker at the position where the error is found within that line.
To support this, the memory_input<>
class has member functions at( p )
, begin_of_line( p )
, end_of_line( p )
and line_at( p )
which take a tao::pegtl::position
as parameter.
The first three functions return a const char*
to position p
, the begin-of-line before p
, or the end-of-line after p
(or the end of the input if the input is not terminated by an end-of-line), respectively.
For convenience, line_at( p )
returns a std::string_view
with the complete line around p
.
Example usage:
// create input 'in' here...
try {
// call parse on the input 'in' here...
}
catch( const parse_error& e ) {
const auto p = e.positions().front();
std::cerr << e.what() << std::endl
<< in.line_at( p ) << '\n'
<< std::setw( p.column ) << '^' << std::endl;
}
All input classes based on memory_input<>
support the above, while all classes based on buffer_input<>
are unable to supply the same functionality as previous input might have been discarded already.
Trying to call any of those functions on buffer_input<>
-based instances will lead to a compile error.
All input classes support deduction guides, e.g. instead of file_input<> in( "filename.txt" )
one can use file_input in( "filename.txt" )
.
Copyright (c) 2014-2020 Dr. Colin Hirsch and Daniel Frey