Moved 'choice' property to SemanticValues.

This commit is contained in:
yhirose 2015-02-26 22:40:00 -05:00
parent 8016f4f0bb
commit d93a007ef0
3 changed files with 22 additions and 14 deletions

View File

@ -12,6 +12,8 @@ The PEG syntax is well described on page 2 in the [document](http://pdos.csail.m
* `~` (Ignore operator) * `~` (Ignore operator)
* `\x??` (Hex number char) * `\x??` (Hex number char)
This library is a [*Packrat*](http://pdos.csail.mit.edu/~baford/packrat/thesis/thesis.pdf) parser which supports the linear-time parsing.
How to use How to use
---------- ----------
@ -48,12 +50,13 @@ int main(void) {
[](const SemanticValues& sv) { return sv[0]; } // 2nd choice [](const SemanticValues& sv) { return sv[0]; } // 2nd choice
}; };
parser["Multitive"] = { parser["Multitive"] = [](const SemanticValues& sv) {
nullptr, // Default action switch (sv.choice) {
[](const SemanticValues& sv) { case 0: // 1st choice
return sv[0].val.get<int>() * sv[1].val.get<int>(); // 1st choice return sv[0].val.get<int>() * sv[1].val.get<int>();
}, default: // 2nd choice
[](const SemanticValues& sv) { return sv[0]; } // 2nd choice return sv[0].val.get<int>();
}
}; };
/* This action is not necessary. /* This action is not necessary.
@ -97,6 +100,7 @@ struct SemanticValues : protected std::vector<SemanticValue>
{ {
const char* s; // Token start const char* s; // Token start
size_t l; // Token length size_t l; // Token length
size_t choice; // Choice number (0 based index)
} }
``` ```
@ -292,7 +296,6 @@ Tested Compilers
TODO TODO
---- ----
* Linear-time parsing (Packrat parsing)
* Optimization of grammars * Optimization of grammars
* Unicode support * Unicode support

View File

@ -186,8 +186,9 @@ struct SemanticValues : protected std::vector<SemanticValue>
{ {
const char* s; const char* s;
size_t l; size_t l;
size_t choice;
SemanticValues() : s(nullptr), l(0) {} SemanticValues() : s(nullptr), l(0), choice(0) {}
typedef SemanticValue T; typedef SemanticValue T;
using std::vector<T>::iterator; using std::vector<T>::iterator;
@ -429,7 +430,6 @@ struct Context
const char* s; const char* s;
size_t l; size_t l;
size_t choice;
const char* error_ptr; const char* error_ptr;
const char* msg; // TODO: should be `int`. const char* msg; // TODO: should be `int`.
@ -604,8 +604,8 @@ public:
} }
sv.s = chldsv.s; sv.s = chldsv.s;
sv.l = chldsv.l; sv.l = chldsv.l;
sv.choice = id;
c.pop(); c.pop();
c.choice = id;
return len; return len;
} }
id++; id++;
@ -1190,7 +1190,7 @@ inline int Holder::parse(const char* s, size_t l, SemanticValues& sv, Context& c
if (success(len) && !outer_->ignore) { if (success(len) && !outer_->ignore) {
assert(!outer_->actions.empty()); assert(!outer_->actions.empty());
auto i = c.choice + 1; // Index 0 is for the default action auto i = chldsv.choice + 1; // Index 0 is for the default action
const auto& action = (i < outer_->actions.size() && outer_->actions[i]) const auto& action = (i < outer_->actions.size() && outer_->actions[i])
? outer_->actions[i] ? outer_->actions[i]
: outer_->actions[0]; : outer_->actions[0];

View File

@ -252,7 +252,12 @@ TEST_CASE("Simple calculator test", "[general]")
}; };
parser["Multitive"] = [](const SemanticValues& sv) { parser["Multitive"] = [](const SemanticValues& sv) {
return sv.size() == 1 ? sv[0].val.get<int>() : sv[0].val.get<int>() * sv[1].val.get<int>(); switch (sv.choice) {
case 0: // Action for the first choice
return sv[0].val.get<int>() * sv[1].val.get<int>();
default: // Action for the second choice
return sv[0].val.get<int>();
}
}; };
parser["Number"] = [](const char* s, size_t l) { parser["Number"] = [](const char* s, size_t l) {