Does Spirit provide any capabilities for working with non-blocking IO?
To provide a more concrete example: I’d like to use Boost’s Spirit parsing framework to parse data coming in from a network socket that’s been placed in non-blocking mode. If the data is not completely available, I’d like to be able to use that thread to perform other work instead of blocking.
The trivial answer is to simply read all the data before invoking Spirit, but potentially gigabytes of data would need to be received and parsed from the socket.
It seems like that in order to support non-blocking I/O while parsing, Spirit would need some ability to partially parse the data and be able to pause and save its parse state when no more data is available. Additionally, it would need to be able to resume parsing from the saved parse state when data does become available. Or maybe I’m making this too complicated?
TODO Will post a example for a simple single-threaded ‘event-based’ parsing model. This is largely trivial but might just be what you need.
For anything less trivial, please heed to following considerations/hints/tips:
How would you be consuming the result? You wouldn’t have the synthesized attributes any earlier anyway, or are you intending to use semantic actions on the fly?
That doesn’t usually work well due to backtracking. The caveats could be worked around by careful and judicious use of qi::hold, qi::locals and putting semantic actions with side-effects only at stations that will never be backtracked. In other words:
Now, everything can be forced, of course, but in general, experienced programmers should have learned to avoid swimming upstream.
Now, if you still want to do this:
You should be able to get spirit library thread safe / reentrant by defining BOOST_SPIRIT_THREADSAFE and linking to libboost_thread. Note this makes the gobals used by Spirit threadsafe (at the cost of fine grained locking) but not your parsers: you can’t share your own parsers/rules/sub grammars/expressions across threads. In fact, you can only share you own (Phoenix/Fusion) functors iff they are threadsafe, and any other extensions defined outside the core Spirit library should be audited for thread-safety.
If you manage the above, I think by far the best approach would seem to
boost::spirit::istreambuf_iteratorusing theboost::spirit::multi_pass<>template class) to consume the input. Note that depending on your grammar, quite a bit of memory could be used for buffering and the performance is suboptimalSome more loose pointers:
qi::locals<>to coordinate state across rules in ‘pure functional fashion’.