2012-05-26 16:00:43 +02:00
|
|
|
#include "parser_internal.h"
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
|
const HParser *p1;
|
|
|
|
|
const HParser *p2;
|
|
|
|
|
} HTwoParsers;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static HParseResult* parse_xor(void *env, HParseState *state) {
|
|
|
|
|
HTwoParsers *parsers = (HTwoParsers*)env;
|
|
|
|
|
// cache the initial state of the input stream
|
|
|
|
|
HInputStream start_state = state->input_stream;
|
|
|
|
|
HParseResult *r1 = h_do_parse(parsers->p1, state);
|
|
|
|
|
HInputStream after_p1_state = state->input_stream;
|
|
|
|
|
// reset input stream, parse again
|
|
|
|
|
state->input_stream = start_state;
|
|
|
|
|
HParseResult *r2 = h_do_parse(parsers->p2, state);
|
|
|
|
|
if (NULL == r1) {
|
|
|
|
|
if (NULL != r2) {
|
|
|
|
|
return r2;
|
|
|
|
|
} else {
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (NULL == r2) {
|
|
|
|
|
state->input_stream = after_p1_state;
|
|
|
|
|
return r1;
|
|
|
|
|
} else {
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-02-03 02:18:19 -05:00
|
|
|
static HCFChoice* desugar_xor(HAllocator *mm__, void *env) {
|
2013-02-20 18:58:15 -05:00
|
|
|
assert_message(0, "'h_xor' is not context-free, can't be desugared");
|
|
|
|
|
return NULL;
|
2013-02-03 02:18:19 -05:00
|
|
|
}
|
|
|
|
|
|
2012-05-26 16:00:43 +02:00
|
|
|
static const HParserVtable xor_vt = {
|
|
|
|
|
.parse = parse_xor,
|
2012-12-18 18:10:40 -05:00
|
|
|
.isValidRegular = h_false,
|
2013-02-20 18:58:15 -05:00
|
|
|
.isValidCF = h_false,
|
2013-02-03 02:18:19 -05:00
|
|
|
.desugar = desugar_xor,
|
2012-05-26 16:00:43 +02:00
|
|
|
};
|
|
|
|
|
|
2012-10-10 15:58:03 +02:00
|
|
|
const HParser* h_xor(const HParser* p1, const HParser* p2) {
|
|
|
|
|
return h_xor__m(&system_allocator, p1, p2);
|
|
|
|
|
}
|
|
|
|
|
const HParser* h_xor__m(HAllocator* mm__, const HParser* p1, const HParser* p2) {
|
|
|
|
|
HTwoParsers *env = h_new(HTwoParsers, 1);
|
2012-05-26 16:00:43 +02:00
|
|
|
env->p1 = p1; env->p2 = p2;
|
2012-10-10 15:58:03 +02:00
|
|
|
HParser *ret = h_new(HParser, 1);
|
2012-05-26 16:00:43 +02:00
|
|
|
ret->vtable = &xor_vt; ret->env = (void*)env;
|
|
|
|
|
return ret;
|
|
|
|
|
}
|