summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJohn Wiegley <johnw@newartisans.com>2009-02-08 03:01:51 -0400
committerJohn Wiegley <johnw@newartisans.com>2009-02-08 03:01:51 -0400
commit75946395818ca22c9a14f2d62ec9ed02acfe766d (patch)
tree585da94e6be0f200827302caa47c7a441eee1c2c
parentaebfc92a4dc9b25183ba0aa637582e8ab8d24c93 (diff)
downloadfork-ledger-75946395818ca22c9a14f2d62ec9ed02acfe766d.tar.gz
fork-ledger-75946395818ca22c9a14f2d62ec9ed02acfe766d.tar.bz2
fork-ledger-75946395818ca22c9a14f2d62ec9ed02acfe766d.zip
Better semantics for the ?: ternary operator.
-rw-r--r--src/op.cc33
-rw-r--r--src/parser.cc14
2 files changed, 41 insertions, 6 deletions
diff --git a/src/op.cc b/src/op.cc
index ed5f8736..3aa2a8cd 100644
--- a/src/op.cc
+++ b/src/op.cc
@@ -201,6 +201,20 @@ value_t expr_t::op_t::calc(scope_t& scope, ptr_op_t * context)
result = right()->calc(scope, context);
break;
+ case O_QUERY:
+ assert(right());
+ assert(right()->kind == O_COLON);
+
+ if (value_t temp = left()->calc(scope, context))
+ result = right()->left()->calc(scope, context);
+ else
+ result = right()->right()->calc(scope, context);
+ break;
+
+ case O_COLON:
+ assert(! "We should never calculate an O_COLON operator");
+ break;
+
case O_COMMA: {
value_t temp(left()->calc(scope, context));
@@ -379,6 +393,22 @@ bool expr_t::op_t::print(std::ostream& out, const context_t& context) const
out << ")";
break;
+ case O_QUERY:
+ if (left() && left()->print(out, context))
+ found = true;
+ out << " ? ";
+ if (has_right() && right()->print(out, context))
+ found = true;
+ break;
+
+ case O_COLON:
+ if (left() && left()->print(out, context))
+ found = true;
+ out << " : ";
+ if (has_right() && right()->print(out, context))
+ found = true;
+ break;
+
case O_COMMA:
if (left() && left()->print(out, context))
found = true;
@@ -476,6 +506,9 @@ void expr_t::op_t::dump(std::ostream& out, const int depth) const
case O_AND: out << "O_AND"; break;
case O_OR: out << "O_OR"; break;
+ case O_QUERY: out << "O_QUERY"; break;
+ case O_COLON: out << "O_COLON"; break;
+
case O_COMMA: out << "O_COMMA"; break;
case LAST:
diff --git a/src/parser.cc b/src/parser.cc
index 0819e3d8..eeff59f0 100644
--- a/src/parser.cc
+++ b/src/parser.cc
@@ -340,7 +340,7 @@ expr_t::parser_t::parse_querycolon_expr(std::istream& in,
if (tok.kind == token_t::QUERY) {
ptr_op_t prev(node);
- node = new op_t(op_t::O_AND);
+ node = new op_t(op_t::O_QUERY);
node->set_left(prev);
node->set_right(parse_or_expr(in, tflags));
if (! node->right())
@@ -351,13 +351,15 @@ expr_t::parser_t::parse_querycolon_expr(std::istream& in,
if (next_tok.kind != token_t::COLON)
next_tok.expected(':');
- prev = node;
- node = new op_t(op_t::O_OR);
- node->set_left(prev);
- node->set_right(parse_or_expr(in, tflags));
- if (! node->right())
+ prev = node->right();
+ ptr_op_t subnode = new op_t(op_t::O_COLON);
+ subnode->set_left(prev);
+ subnode->set_right(parse_or_expr(in, tflags));
+ if (! subnode->right())
throw_(parse_error,
tok.symbol << " operator not followed by argument");
+
+ node->set_right(subnode);
} else {
push_token(tok);
}