2014-02-10 13:54:46 -05:00
|
|
|
// Copyright 2013 Dolphin Emulator Project
|
2015-05-18 01:08:10 +02:00
|
|
|
// Licensed under GPLv2+
|
2014-02-10 13:54:46 -05:00
|
|
|
// Refer to the license.txt file included.
|
2013-06-13 23:09:55 -04:00
|
|
|
|
|
|
|
#include <cassert>
|
2018-12-30 13:16:28 -06:00
|
|
|
#include <cmath>
|
2013-06-13 23:09:55 -04:00
|
|
|
#include <iostream>
|
2016-06-26 05:34:09 +02:00
|
|
|
#include <memory>
|
2018-12-30 12:38:02 -06:00
|
|
|
#include <regex>
|
2014-02-17 05:18:15 -05:00
|
|
|
#include <string>
|
2019-01-26 12:17:30 -06:00
|
|
|
#include <utility>
|
2013-06-13 23:09:55 -04:00
|
|
|
#include <vector>
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
#include "Common/Common.h"
|
2017-06-07 16:03:36 -07:00
|
|
|
#include "Common/StringUtil.h"
|
2019-01-26 12:17:30 -06:00
|
|
|
|
2016-10-11 17:48:38 -07:00
|
|
|
#include "InputCommon/ControlReference/ExpressionParser.h"
|
2019-01-26 12:17:30 -06:00
|
|
|
#include "InputCommon/ControlReference/FunctionExpression.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
|
2019-06-17 16:39:24 -04:00
|
|
|
namespace ciface::ExpressionParser
|
|
|
|
{
|
2013-06-13 23:09:55 -04:00
|
|
|
using namespace ciface::Core;
|
|
|
|
|
|
|
|
inline std::string OpName(TokenType op)
|
|
|
|
{
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case TOK_AND:
|
|
|
|
return "And";
|
|
|
|
case TOK_OR:
|
|
|
|
return "Or";
|
2019-01-06 09:08:35 -06:00
|
|
|
case TOK_FUNCTION:
|
|
|
|
return "Function";
|
2013-06-14 02:52:07 -04:00
|
|
|
case TOK_ADD:
|
|
|
|
return "Add";
|
2018-12-30 17:32:32 -06:00
|
|
|
case TOK_SUB:
|
|
|
|
return "Sub";
|
2018-12-30 11:51:12 -06:00
|
|
|
case TOK_MUL:
|
|
|
|
return "Mul";
|
|
|
|
case TOK_DIV:
|
|
|
|
return "Div";
|
2018-12-30 13:16:28 -06:00
|
|
|
case TOK_MOD:
|
|
|
|
return "Mod";
|
2018-12-30 16:06:29 -06:00
|
|
|
case TOK_ASSIGN:
|
|
|
|
return "Assign";
|
2018-12-30 16:11:42 -06:00
|
|
|
case TOK_LTHAN:
|
|
|
|
return "LThan";
|
|
|
|
case TOK_GTHAN:
|
|
|
|
return "GThan";
|
2018-12-30 17:32:32 -06:00
|
|
|
case TOK_COMMA:
|
|
|
|
return "Comma";
|
2018-12-30 16:06:29 -06:00
|
|
|
case TOK_VARIABLE:
|
|
|
|
return "Var";
|
2013-06-13 23:09:55 -04:00
|
|
|
default:
|
|
|
|
assert(false);
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
Token::Token(TokenType type_) : type(type_)
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
2019-03-02 10:10:26 -06:00
|
|
|
}
|
2013-06-13 23:09:55 -04:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
Token::Token(TokenType type_, std::string data_) : type(type_), data(std::move(data_))
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
2019-03-02 10:10:26 -06:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
bool Token::IsBinaryOperator() const
|
|
|
|
{
|
|
|
|
return type >= TOK_BINARY_OPS_BEGIN && type < TOK_BINARY_OPS_END;
|
|
|
|
}
|
2018-12-30 10:37:23 -06:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
Token::operator std::string() const
|
|
|
|
{
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case TOK_DISCARD:
|
|
|
|
return "Discard";
|
|
|
|
case TOK_EOF:
|
|
|
|
return "EOF";
|
|
|
|
case TOK_LPAREN:
|
|
|
|
return "(";
|
|
|
|
case TOK_RPAREN:
|
|
|
|
return ")";
|
|
|
|
case TOK_AND:
|
|
|
|
return "&";
|
|
|
|
case TOK_OR:
|
|
|
|
return "|";
|
|
|
|
case TOK_FUNCTION:
|
|
|
|
return '!' + data;
|
|
|
|
case TOK_ADD:
|
|
|
|
return "+";
|
|
|
|
case TOK_SUB:
|
|
|
|
return "-";
|
|
|
|
case TOK_MUL:
|
|
|
|
return "*";
|
|
|
|
case TOK_DIV:
|
|
|
|
return "/";
|
|
|
|
case TOK_MOD:
|
|
|
|
return "%";
|
|
|
|
case TOK_ASSIGN:
|
|
|
|
return "=";
|
|
|
|
case TOK_LTHAN:
|
|
|
|
return "<";
|
|
|
|
case TOK_GTHAN:
|
|
|
|
return ">";
|
|
|
|
case TOK_COMMA:
|
|
|
|
return ",";
|
|
|
|
case TOK_CONTROL:
|
|
|
|
return "Device(" + data + ')';
|
|
|
|
case TOK_LITERAL:
|
|
|
|
return '\'' + data + '\'';
|
|
|
|
case TOK_VARIABLE:
|
|
|
|
return '$' + data;
|
|
|
|
default:
|
|
|
|
break;
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
return "Invalid";
|
|
|
|
}
|
2018-12-30 12:38:02 -06:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
Lexer::Lexer(const std::string& expr_) : expr(expr_)
|
|
|
|
{
|
|
|
|
it = expr.begin();
|
|
|
|
}
|
2019-01-05 15:31:05 -06:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
std::string Lexer::FetchDelimString(char delim)
|
|
|
|
{
|
|
|
|
const std::string result = FetchCharsWhile([delim](char c) { return c != delim; });
|
|
|
|
if (it != expr.end())
|
|
|
|
++it;
|
|
|
|
return result;
|
|
|
|
}
|
2018-12-30 12:38:02 -06:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
std::string Lexer::FetchWordChars()
|
|
|
|
{
|
|
|
|
// Words must start with a letter or underscore.
|
|
|
|
if (expr.end() == it || (!std::isalpha(*it, std::locale::classic()) && ('_' != *it)))
|
|
|
|
return "";
|
2018-12-30 12:38:02 -06:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
// Valid word characters:
|
2019-03-02 14:47:26 -06:00
|
|
|
std::regex rx(R"([a-z\d_])", std::regex_constants::icase);
|
2018-12-30 16:06:29 -06:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
return FetchCharsWhile([&rx](char c) { return std::regex_match(std::string(1, c), rx); });
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
Token Lexer::GetFunction()
|
|
|
|
{
|
|
|
|
return Token(TOK_FUNCTION, FetchWordChars());
|
|
|
|
}
|
2018-12-30 16:06:29 -06:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
Token Lexer::GetDelimitedLiteral()
|
|
|
|
{
|
|
|
|
return Token(TOK_LITERAL, FetchDelimString('\''));
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
Token Lexer::GetVariable()
|
|
|
|
{
|
|
|
|
return Token(TOK_VARIABLE, FetchWordChars());
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
Token Lexer::GetFullyQualifiedControl()
|
|
|
|
{
|
|
|
|
return Token(TOK_CONTROL, FetchDelimString('`'));
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
Token Lexer::GetBarewordsControl(char c)
|
|
|
|
{
|
|
|
|
std::string name;
|
|
|
|
name += c;
|
|
|
|
name += FetchCharsWhile([](char c) { return std::isalpha(c, std::locale::classic()); });
|
2019-01-05 13:43:39 -06:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
ControlQualifier qualifier;
|
|
|
|
qualifier.control_name = name;
|
|
|
|
return Token(TOK_CONTROL, qualifier);
|
|
|
|
}
|
2019-01-05 13:43:39 -06:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
Token Lexer::GetRealLiteral(char c)
|
|
|
|
{
|
|
|
|
std::string value;
|
|
|
|
value += c;
|
|
|
|
value += FetchCharsWhile([](char c) { return isdigit(c, std::locale::classic()) || ('.' == c); });
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
if (std::regex_match(value, std::regex(R"(\d+(\.\d+)?)")))
|
|
|
|
return Token(TOK_LITERAL, value);
|
|
|
|
|
|
|
|
return Token(TOK_INVALID);
|
2019-03-02 10:10:26 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
Token Lexer::NextToken()
|
|
|
|
{
|
|
|
|
if (it == expr.end())
|
|
|
|
return Token(TOK_EOF);
|
|
|
|
|
|
|
|
char c = *it++;
|
|
|
|
switch (c)
|
|
|
|
{
|
|
|
|
case ' ':
|
|
|
|
case '\t':
|
|
|
|
case '\n':
|
|
|
|
case '\r':
|
|
|
|
return Token(TOK_DISCARD);
|
|
|
|
case '(':
|
|
|
|
return Token(TOK_LPAREN);
|
|
|
|
case ')':
|
|
|
|
return Token(TOK_RPAREN);
|
|
|
|
case '&':
|
|
|
|
return Token(TOK_AND);
|
|
|
|
case '|':
|
|
|
|
return Token(TOK_OR);
|
|
|
|
case '!':
|
|
|
|
return GetFunction();
|
|
|
|
case '+':
|
|
|
|
return Token(TOK_ADD);
|
|
|
|
case '-':
|
|
|
|
return Token(TOK_SUB);
|
|
|
|
case '*':
|
|
|
|
return Token(TOK_MUL);
|
|
|
|
case '/':
|
|
|
|
return Token(TOK_DIV);
|
|
|
|
case '%':
|
|
|
|
return Token(TOK_MOD);
|
|
|
|
case '=':
|
|
|
|
return Token(TOK_ASSIGN);
|
|
|
|
case '<':
|
|
|
|
return Token(TOK_LTHAN);
|
|
|
|
case '>':
|
|
|
|
return Token(TOK_GTHAN);
|
|
|
|
case ',':
|
|
|
|
return Token(TOK_COMMA);
|
|
|
|
case '\'':
|
|
|
|
return GetDelimitedLiteral();
|
|
|
|
case '$':
|
|
|
|
return GetVariable();
|
|
|
|
case '`':
|
|
|
|
return GetFullyQualifiedControl();
|
|
|
|
default:
|
|
|
|
if (isalpha(c, std::locale::classic()))
|
|
|
|
return GetBarewordsControl(c);
|
|
|
|
else if (isdigit(c, std::locale::classic()))
|
|
|
|
return GetRealLiteral(c);
|
|
|
|
else
|
|
|
|
return Token(TOK_INVALID);
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
2019-03-02 10:10:26 -06:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
ParseStatus Lexer::Tokenize(std::vector<Token>& tokens)
|
|
|
|
{
|
|
|
|
while (true)
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
2019-03-02 10:10:26 -06:00
|
|
|
const std::size_t string_position = it - expr.begin();
|
|
|
|
Token tok = NextToken();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
tok.string_position = string_position;
|
|
|
|
tok.string_length = it - expr.begin();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
if (tok.type == TOK_DISCARD)
|
|
|
|
continue;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
tokens.push_back(tok);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
if (tok.type == TOK_INVALID)
|
|
|
|
return ParseStatus::SyntaxError;
|
|
|
|
|
|
|
|
if (tok.type == TOK_EOF)
|
|
|
|
break;
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
2019-03-02 10:10:26 -06:00
|
|
|
return ParseStatus::Successful;
|
|
|
|
}
|
2013-06-13 23:09:55 -04:00
|
|
|
|
2017-06-07 17:30:07 -07:00
|
|
|
class ControlExpression : public Expression
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
|
|
|
public:
|
2019-03-02 14:47:26 -06:00
|
|
|
// Keep a shared_ptr to the device so the control pointer doesn't become invalid.
|
2017-06-07 15:54:59 -07:00
|
|
|
std::shared_ptr<Device> m_device;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-06-07 19:02:16 -07:00
|
|
|
explicit ControlExpression(ControlQualifier qualifier_) : qualifier(qualifier_) {}
|
2019-02-26 19:46:21 -06:00
|
|
|
ControlState GetValue() const override
|
|
|
|
{
|
2018-12-30 16:06:29 -06:00
|
|
|
if (!input)
|
2019-02-26 19:46:21 -06:00
|
|
|
return 0.0;
|
|
|
|
|
|
|
|
// Note: Inputs may return negative values in situations where opposing directions are
|
|
|
|
// activated. We clamp off the negative values here.
|
|
|
|
|
|
|
|
// FYI: Clamping values greater than 1.0 is purposely not done to support unbounded values in
|
|
|
|
// the future. (e.g. raw accelerometer/gyro data)
|
|
|
|
|
2018-12-30 16:06:29 -06:00
|
|
|
return std::max(0.0, input->GetState());
|
2019-02-26 19:46:21 -06:00
|
|
|
}
|
2017-06-07 15:56:49 -07:00
|
|
|
void SetValue(ControlState value) override
|
|
|
|
{
|
2018-12-30 16:06:29 -06:00
|
|
|
if (output)
|
|
|
|
output->SetState(value);
|
2017-06-07 15:56:49 -07:00
|
|
|
}
|
2018-12-30 16:06:29 -06:00
|
|
|
int CountNumControls() const override { return (input || output) ? 1 : 0; }
|
|
|
|
void UpdateReferences(ControlEnvironment& env) override
|
2017-06-07 18:48:17 -07:00
|
|
|
{
|
2018-12-30 16:06:29 -06:00
|
|
|
m_device = env.FindDevice(qualifier);
|
|
|
|
input = env.FindInput(qualifier);
|
|
|
|
output = env.FindOutput(qualifier);
|
2017-06-07 18:48:17 -07:00
|
|
|
}
|
2017-06-07 15:54:59 -07:00
|
|
|
operator std::string() const override { return "`" + static_cast<std::string>(qualifier) + "`"; }
|
2018-12-30 16:06:29 -06:00
|
|
|
|
|
|
|
private:
|
|
|
|
ControlQualifier qualifier;
|
|
|
|
Device::Input* input = nullptr;
|
|
|
|
Device::Output* output = nullptr;
|
2013-06-13 23:09:55 -04:00
|
|
|
};
|
|
|
|
|
2017-06-07 17:30:07 -07:00
|
|
|
class BinaryExpression : public Expression
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
TokenType op;
|
2017-06-07 17:30:07 -07:00
|
|
|
std::unique_ptr<Expression> lhs;
|
|
|
|
std::unique_ptr<Expression> rhs;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-06-07 17:30:07 -07:00
|
|
|
BinaryExpression(TokenType op_, std::unique_ptr<Expression>&& lhs_,
|
|
|
|
std::unique_ptr<Expression>&& rhs_)
|
2017-06-07 15:29:00 -07:00
|
|
|
: op(op_), lhs(std::move(lhs_)), rhs(std::move(rhs_))
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-02-26 02:04:16 -05:00
|
|
|
ControlState GetValue() const override
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case TOK_AND:
|
2018-12-30 16:29:48 -06:00
|
|
|
return std::min(lhs->GetValue(), rhs->GetValue());
|
2013-06-13 23:09:55 -04:00
|
|
|
case TOK_OR:
|
2018-12-30 16:29:48 -06:00
|
|
|
return std::max(lhs->GetValue(), rhs->GetValue());
|
2013-06-14 02:52:07 -04:00
|
|
|
case TOK_ADD:
|
2018-12-30 16:29:48 -06:00
|
|
|
return lhs->GetValue() + rhs->GetValue();
|
2018-12-30 17:32:32 -06:00
|
|
|
case TOK_SUB:
|
|
|
|
return lhs->GetValue() - rhs->GetValue();
|
2018-12-30 11:51:12 -06:00
|
|
|
case TOK_MUL:
|
2018-12-30 16:29:48 -06:00
|
|
|
return lhs->GetValue() * rhs->GetValue();
|
2018-12-30 11:51:12 -06:00
|
|
|
case TOK_DIV:
|
|
|
|
{
|
2018-12-30 16:29:48 -06:00
|
|
|
const ControlState result = lhs->GetValue() / rhs->GetValue();
|
2018-12-30 11:51:12 -06:00
|
|
|
return std::isinf(result) ? 0.0 : result;
|
|
|
|
}
|
2018-12-30 13:16:28 -06:00
|
|
|
case TOK_MOD:
|
|
|
|
{
|
2018-12-30 16:29:48 -06:00
|
|
|
const ControlState result = std::fmod(lhs->GetValue(), rhs->GetValue());
|
2018-12-30 13:16:28 -06:00
|
|
|
return std::isnan(result) ? 0.0 : result;
|
|
|
|
}
|
2018-12-30 16:06:29 -06:00
|
|
|
case TOK_ASSIGN:
|
|
|
|
{
|
2018-12-30 16:29:48 -06:00
|
|
|
lhs->SetValue(rhs->GetValue());
|
|
|
|
return lhs->GetValue();
|
2018-12-30 16:06:29 -06:00
|
|
|
}
|
2018-12-30 16:11:42 -06:00
|
|
|
case TOK_LTHAN:
|
2018-12-30 16:29:48 -06:00
|
|
|
return lhs->GetValue() < rhs->GetValue();
|
2018-12-30 16:11:42 -06:00
|
|
|
case TOK_GTHAN:
|
2018-12-30 16:29:48 -06:00
|
|
|
return lhs->GetValue() > rhs->GetValue();
|
2018-12-30 17:32:32 -06:00
|
|
|
case TOK_COMMA:
|
|
|
|
{
|
|
|
|
// Eval and discard lhs:
|
|
|
|
lhs->GetValue();
|
|
|
|
return rhs->GetValue();
|
|
|
|
}
|
2013-06-13 23:09:55 -04:00
|
|
|
default:
|
|
|
|
assert(false);
|
|
|
|
return 0;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
|
|
|
|
2015-07-30 06:47:02 -04:00
|
|
|
void SetValue(ControlState value) override
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
|
|
|
// Don't do anything special with the op we have.
|
|
|
|
// Treat "A & B" the same as "A | B".
|
|
|
|
lhs->SetValue(value);
|
|
|
|
rhs->SetValue(value);
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-02-26 02:04:16 -05:00
|
|
|
int CountNumControls() const override
|
|
|
|
{
|
|
|
|
return lhs->CountNumControls() + rhs->CountNumControls();
|
|
|
|
}
|
|
|
|
|
2018-12-30 16:06:29 -06:00
|
|
|
void UpdateReferences(ControlEnvironment& env) override
|
2017-06-07 18:48:17 -07:00
|
|
|
{
|
2018-12-30 16:06:29 -06:00
|
|
|
lhs->UpdateReferences(env);
|
|
|
|
rhs->UpdateReferences(env);
|
2017-06-07 18:48:17 -07:00
|
|
|
}
|
|
|
|
|
2017-02-26 02:04:16 -05:00
|
|
|
operator std::string() const override
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
2019-03-02 14:47:26 -06:00
|
|
|
return OpName(op) + "(" + std::string(*lhs) + ", " + std::string(*rhs) + ")";
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-12-30 10:37:23 -06:00
|
|
|
class LiteralExpression : public Expression
|
|
|
|
{
|
|
|
|
public:
|
2019-03-02 10:10:26 -06:00
|
|
|
void SetValue(ControlState) override
|
2018-12-30 10:37:23 -06:00
|
|
|
{
|
|
|
|
// Do nothing.
|
|
|
|
}
|
|
|
|
|
|
|
|
int CountNumControls() const override { return 1; }
|
|
|
|
|
2018-12-30 16:06:29 -06:00
|
|
|
void UpdateReferences(ControlEnvironment&) override
|
2018-12-30 10:37:23 -06:00
|
|
|
{
|
|
|
|
// Nothing needed.
|
|
|
|
}
|
|
|
|
|
2018-12-30 13:16:28 -06:00
|
|
|
operator std::string() const override { return '\'' + GetName() + '\''; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual std::string GetName() const = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
class LiteralReal : public LiteralExpression
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
LiteralReal(ControlState value) : m_value(value) {}
|
|
|
|
|
|
|
|
ControlState GetValue() const override { return m_value; }
|
|
|
|
|
|
|
|
std::string GetName() const override { return ValueToString(m_value); }
|
2018-12-30 10:37:23 -06:00
|
|
|
|
|
|
|
private:
|
2018-12-30 13:16:28 -06:00
|
|
|
const ControlState m_value{};
|
2018-12-30 10:37:23 -06:00
|
|
|
};
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
ParseResult MakeLiteralExpression(Token token)
|
2018-12-30 13:16:28 -06:00
|
|
|
{
|
2019-01-05 15:31:05 -06:00
|
|
|
ControlState val{};
|
2019-03-02 14:47:26 -06:00
|
|
|
if (TryParse(token.data, &val))
|
|
|
|
return ParseResult::MakeSuccessfulResult(std::make_unique<LiteralReal>(val));
|
|
|
|
else
|
|
|
|
return ParseResult::MakeErrorResult(token, _trans("Invalid literal."));
|
2018-12-30 13:16:28 -06:00
|
|
|
}
|
|
|
|
|
2018-12-30 16:06:29 -06:00
|
|
|
class VariableExpression : public Expression
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
VariableExpression(std::string name) : m_name(name) {}
|
|
|
|
|
|
|
|
ControlState GetValue() const override { return *m_value_ptr; }
|
|
|
|
|
|
|
|
void SetValue(ControlState value) override { *m_value_ptr = value; }
|
|
|
|
|
|
|
|
int CountNumControls() const override { return 1; }
|
|
|
|
|
|
|
|
void UpdateReferences(ControlEnvironment& env) override
|
|
|
|
{
|
|
|
|
m_value_ptr = env.GetVariablePtr(m_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
operator std::string() const override { return '$' + m_name; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
const std::string m_name;
|
|
|
|
ControlState* m_value_ptr{};
|
|
|
|
};
|
|
|
|
|
2017-06-07 18:29:02 -07:00
|
|
|
// This class proxies all methods to its either left-hand child if it has bound controls, or its
|
|
|
|
// right-hand child. Its intended use is for supporting old-style barewords expressions.
|
|
|
|
class CoalesceExpression : public Expression
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CoalesceExpression(std::unique_ptr<Expression>&& lhs, std::unique_ptr<Expression>&& rhs)
|
|
|
|
: m_lhs(std::move(lhs)), m_rhs(std::move(rhs))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
ControlState GetValue() const override { return GetActiveChild()->GetValue(); }
|
2018-12-22 11:17:05 -06:00
|
|
|
void SetValue(ControlState value) override { GetActiveChild()->SetValue(value); }
|
2017-06-07 18:29:02 -07:00
|
|
|
|
|
|
|
int CountNumControls() const override { return GetActiveChild()->CountNumControls(); }
|
|
|
|
operator std::string() const override
|
|
|
|
{
|
|
|
|
return "Coalesce(" + static_cast<std::string>(*m_lhs) + ", " +
|
|
|
|
static_cast<std::string>(*m_rhs) + ')';
|
|
|
|
}
|
|
|
|
|
2018-12-30 16:06:29 -06:00
|
|
|
void UpdateReferences(ControlEnvironment& env) override
|
2017-06-07 18:48:17 -07:00
|
|
|
{
|
2018-12-30 16:06:29 -06:00
|
|
|
m_lhs->UpdateReferences(env);
|
|
|
|
m_rhs->UpdateReferences(env);
|
2017-06-07 18:48:17 -07:00
|
|
|
}
|
|
|
|
|
2017-06-07 18:29:02 -07:00
|
|
|
private:
|
|
|
|
const std::unique_ptr<Expression>& GetActiveChild() const
|
|
|
|
{
|
|
|
|
return m_lhs->CountNumControls() > 0 ? m_lhs : m_rhs;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<Expression> m_lhs;
|
|
|
|
std::unique_ptr<Expression> m_rhs;
|
|
|
|
};
|
|
|
|
|
2018-12-30 16:06:29 -06:00
|
|
|
std::shared_ptr<Device> ControlEnvironment::FindDevice(ControlQualifier qualifier) const
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
|
|
|
if (qualifier.has_device)
|
|
|
|
return container.FindDevice(qualifier.device_qualifier);
|
|
|
|
else
|
|
|
|
return container.FindDevice(default_device);
|
|
|
|
}
|
|
|
|
|
2018-12-30 16:06:29 -06:00
|
|
|
Device::Input* ControlEnvironment::FindInput(ControlQualifier qualifier) const
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
2016-06-25 21:46:39 +02:00
|
|
|
const std::shared_ptr<Device> device = FindDevice(qualifier);
|
2013-06-26 16:54:48 -04:00
|
|
|
if (!device)
|
2014-03-09 21:14:26 +01:00
|
|
|
return nullptr;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-12-30 16:06:29 -06:00
|
|
|
return device->FindInput(qualifier.control_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
Device::Output* ControlEnvironment::FindOutput(ControlQualifier qualifier) const
|
|
|
|
{
|
|
|
|
const std::shared_ptr<Device> device = FindDevice(qualifier);
|
|
|
|
if (!device)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
return device->FindOutput(qualifier.control_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
ControlState* ControlEnvironment::GetVariablePtr(const std::string& name)
|
|
|
|
{
|
|
|
|
return &m_variables[name];
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
ParseResult ParseResult::MakeEmptyResult()
|
2017-06-07 15:08:55 -07:00
|
|
|
{
|
2019-03-02 14:47:26 -06:00
|
|
|
ParseResult result;
|
|
|
|
result.status = ParseStatus::EmptyExpression;
|
|
|
|
return result;
|
|
|
|
}
|
2017-06-07 15:29:00 -07:00
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
ParseResult ParseResult::MakeSuccessfulResult(std::unique_ptr<Expression>&& expr)
|
|
|
|
{
|
|
|
|
ParseResult result;
|
|
|
|
result.status = ParseStatus::Successful;
|
|
|
|
result.expr = std::move(expr);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
ParseResult ParseResult::MakeErrorResult(Token token, std::string description)
|
|
|
|
{
|
|
|
|
ParseResult result;
|
|
|
|
result.status = ParseStatus::SyntaxError;
|
|
|
|
result.token = std::move(token);
|
|
|
|
result.description = std::move(description);
|
|
|
|
return result;
|
|
|
|
}
|
2017-06-07 15:08:55 -07:00
|
|
|
|
2013-06-13 23:09:55 -04:00
|
|
|
class Parser
|
|
|
|
{
|
|
|
|
public:
|
2019-03-02 14:47:26 -06:00
|
|
|
explicit Parser(const std::vector<Token>& tokens_) : tokens(tokens_) { m_it = tokens.begin(); }
|
2019-01-08 18:36:58 -06:00
|
|
|
ParseResult Parse()
|
|
|
|
{
|
|
|
|
ParseResult result = ParseToplevel();
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
if (ParseStatus::Successful != result.status)
|
|
|
|
return result;
|
|
|
|
|
2019-01-08 18:36:58 -06:00
|
|
|
if (Peek().type == TOK_EOF)
|
|
|
|
return result;
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
return ParseResult::MakeErrorResult(Peek(), _trans("Expected EOF."));
|
2019-01-08 18:36:58 -06:00
|
|
|
}
|
2018-04-12 14:18:04 +02:00
|
|
|
|
2013-06-13 23:09:55 -04:00
|
|
|
private:
|
2019-01-08 18:26:36 -06:00
|
|
|
struct FunctionArguments
|
|
|
|
{
|
2019-03-02 14:47:26 -06:00
|
|
|
FunctionArguments(ParseResult&& result_, std::vector<std::unique_ptr<Expression>>&& args_ = {})
|
|
|
|
: result(std::move(result_)), args(std::move(args_))
|
2019-01-08 18:26:36 -06:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
// Note: expression member isn't being used.
|
|
|
|
ParseResult result;
|
|
|
|
|
2019-01-08 18:26:36 -06:00
|
|
|
std::vector<std::unique_ptr<Expression>> args;
|
|
|
|
};
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
const std::vector<Token>& tokens;
|
|
|
|
std::vector<Token>::const_iterator m_it;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-01-20 17:44:01 -06:00
|
|
|
Token Chew()
|
|
|
|
{
|
|
|
|
const Token tok = Peek();
|
|
|
|
if (TOK_EOF != tok.type)
|
|
|
|
++m_it;
|
|
|
|
return tok;
|
|
|
|
}
|
|
|
|
|
2013-06-13 23:09:55 -04:00
|
|
|
Token Peek() { return *m_it; }
|
2019-01-08 18:26:36 -06:00
|
|
|
|
2013-06-13 23:09:55 -04:00
|
|
|
bool Expects(TokenType type)
|
|
|
|
{
|
|
|
|
Token tok = Chew();
|
|
|
|
return tok.type == type;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-01-08 18:26:36 -06:00
|
|
|
FunctionArguments ParseFunctionArguments()
|
|
|
|
{
|
2019-01-20 17:44:01 -06:00
|
|
|
std::vector<std::unique_ptr<Expression>> args;
|
|
|
|
|
|
|
|
if (TOK_LPAREN != Peek().type)
|
|
|
|
{
|
|
|
|
// Single argument with no parens (useful for unary ! function)
|
|
|
|
auto arg = ParseAtom(Chew());
|
|
|
|
if (ParseStatus::Successful != arg.status)
|
2019-03-02 14:47:26 -06:00
|
|
|
return {std::move(arg)};
|
2019-01-20 17:44:01 -06:00
|
|
|
|
|
|
|
args.emplace_back(std::move(arg.expr));
|
2019-03-02 14:47:26 -06:00
|
|
|
return {ParseResult::MakeSuccessfulResult({}), std::move(args)};
|
2019-01-20 17:44:01 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Chew the L-Paren
|
|
|
|
Chew();
|
2019-01-08 18:26:36 -06:00
|
|
|
|
|
|
|
// Check for empty argument list:
|
|
|
|
if (TOK_RPAREN == Peek().type)
|
2019-01-20 17:44:01 -06:00
|
|
|
{
|
|
|
|
Chew();
|
2019-03-02 14:47:26 -06:00
|
|
|
return {ParseResult::MakeSuccessfulResult({})};
|
2019-01-20 17:44:01 -06:00
|
|
|
}
|
2019-01-08 18:26:36 -06:00
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
// Read one argument.
|
|
|
|
// Grab an expression, but stop at comma.
|
2019-01-08 18:36:58 -06:00
|
|
|
auto arg = ParseBinary(BinaryOperatorPrecedence(TOK_COMMA));
|
2019-01-08 18:26:36 -06:00
|
|
|
if (ParseStatus::Successful != arg.status)
|
2019-03-02 14:47:26 -06:00
|
|
|
return {std::move(arg)};
|
2019-01-08 18:26:36 -06:00
|
|
|
|
|
|
|
args.emplace_back(std::move(arg.expr));
|
|
|
|
|
|
|
|
// Right paren is the end of our arguments.
|
|
|
|
const Token tok = Chew();
|
|
|
|
if (TOK_RPAREN == tok.type)
|
2019-03-02 14:47:26 -06:00
|
|
|
return {ParseResult::MakeSuccessfulResult({}), std::move(args)};
|
2019-01-08 18:26:36 -06:00
|
|
|
|
|
|
|
// Comma before the next argument.
|
|
|
|
if (TOK_COMMA != tok.type)
|
2019-03-02 14:47:26 -06:00
|
|
|
return {ParseResult::MakeErrorResult(tok, _trans("Expected comma."))};
|
2019-01-08 18:26:36 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-08 18:36:58 -06:00
|
|
|
ParseResult ParseAtom(const Token& tok)
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
|
|
|
switch (tok.type)
|
|
|
|
{
|
2019-01-06 09:08:35 -06:00
|
|
|
case TOK_FUNCTION:
|
2019-01-05 15:31:05 -06:00
|
|
|
{
|
2019-01-06 09:08:35 -06:00
|
|
|
auto func = MakeFunctionExpression(tok.data);
|
2019-03-02 14:47:26 -06:00
|
|
|
|
|
|
|
if (!func)
|
|
|
|
return ParseResult::MakeErrorResult(tok, _trans("Unknown function."));
|
|
|
|
|
2019-01-08 18:26:36 -06:00
|
|
|
auto args = ParseFunctionArguments();
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
if (ParseStatus::Successful != args.result.status)
|
|
|
|
return std::move(args.result);
|
2019-01-08 18:26:36 -06:00
|
|
|
|
|
|
|
if (!func->SetArguments(std::move(args.args)))
|
2019-03-02 14:47:26 -06:00
|
|
|
{
|
|
|
|
// TODO: It would be nice to output how many arguments are expected.
|
|
|
|
return ParseResult::MakeErrorResult(tok, _trans("Wrong number of arguments."));
|
|
|
|
}
|
2019-01-06 09:08:35 -06:00
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
return ParseResult::MakeSuccessfulResult(std::move(func));
|
2019-01-05 15:31:05 -06:00
|
|
|
}
|
2013-06-13 23:09:55 -04:00
|
|
|
case TOK_CONTROL:
|
2018-12-30 10:37:23 -06:00
|
|
|
{
|
|
|
|
ControlQualifier cq;
|
|
|
|
cq.FromString(tok.data);
|
2019-03-02 14:47:26 -06:00
|
|
|
return ParseResult::MakeSuccessfulResult(std::make_unique<ControlExpression>(cq));
|
2018-12-30 10:37:23 -06:00
|
|
|
}
|
|
|
|
case TOK_LITERAL:
|
|
|
|
{
|
2019-03-02 14:47:26 -06:00
|
|
|
return MakeLiteralExpression(tok);
|
2018-12-30 10:37:23 -06:00
|
|
|
}
|
2018-12-30 16:06:29 -06:00
|
|
|
case TOK_VARIABLE:
|
|
|
|
{
|
2019-03-02 14:47:26 -06:00
|
|
|
return ParseResult::MakeSuccessfulResult(std::make_unique<VariableExpression>(tok.data));
|
2018-12-30 16:06:29 -06:00
|
|
|
}
|
2013-06-13 23:09:55 -04:00
|
|
|
case TOK_LPAREN:
|
2019-01-05 15:31:05 -06:00
|
|
|
{
|
2019-01-08 18:36:58 -06:00
|
|
|
return ParseParens();
|
2019-01-05 15:31:05 -06:00
|
|
|
}
|
2019-01-06 10:03:21 -06:00
|
|
|
case TOK_SUB:
|
|
|
|
{
|
|
|
|
// An atom was expected but we got a subtraction symbol.
|
|
|
|
// Interpret it as a unary minus function.
|
2019-03-02 14:47:26 -06:00
|
|
|
|
|
|
|
// Make sure to copy the existing string position values for proper error results.
|
|
|
|
Token func = tok;
|
|
|
|
func.type = TOK_FUNCTION;
|
|
|
|
func.data = "minus";
|
|
|
|
return ParseAtom(std::move(func));
|
2019-01-06 10:03:21 -06:00
|
|
|
}
|
2013-06-13 23:09:55 -04:00
|
|
|
default:
|
2019-03-02 14:47:26 -06:00
|
|
|
return ParseResult::MakeErrorResult(tok, _trans("Expected start of expression."));
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-12-30 19:50:20 -06:00
|
|
|
static int BinaryOperatorPrecedence(TokenType type)
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
2018-12-30 19:50:20 -06:00
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case TOK_MUL:
|
|
|
|
case TOK_DIV:
|
|
|
|
case TOK_MOD:
|
|
|
|
return 1;
|
|
|
|
case TOK_ADD:
|
|
|
|
case TOK_SUB:
|
|
|
|
return 2;
|
|
|
|
case TOK_GTHAN:
|
|
|
|
case TOK_LTHAN:
|
|
|
|
return 3;
|
|
|
|
case TOK_AND:
|
|
|
|
return 4;
|
|
|
|
case TOK_OR:
|
|
|
|
return 5;
|
|
|
|
case TOK_ASSIGN:
|
|
|
|
return 6;
|
|
|
|
case TOK_COMMA:
|
|
|
|
return 7;
|
|
|
|
default:
|
|
|
|
assert(false);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-08 18:36:58 -06:00
|
|
|
ParseResult ParseBinary(int precedence = 999)
|
2018-12-30 19:50:20 -06:00
|
|
|
{
|
2019-01-08 18:36:58 -06:00
|
|
|
ParseResult lhs = ParseAtom(Chew());
|
2018-12-30 19:50:20 -06:00
|
|
|
|
|
|
|
if (lhs.status == ParseStatus::SyntaxError)
|
|
|
|
return lhs;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-12-30 19:50:20 -06:00
|
|
|
std::unique_ptr<Expression> expr = std::move(lhs.expr);
|
|
|
|
|
|
|
|
// TODO: handle LTR/RTL associativity?
|
2019-03-02 10:10:26 -06:00
|
|
|
while (Peek().IsBinaryOperator() && BinaryOperatorPrecedence(Peek().type) < precedence)
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
2018-12-30 19:50:20 -06:00
|
|
|
const Token tok = Chew();
|
2019-01-08 18:36:58 -06:00
|
|
|
ParseResult rhs = ParseBinary(BinaryOperatorPrecedence(tok.type));
|
2018-12-30 19:50:20 -06:00
|
|
|
if (rhs.status == ParseStatus::SyntaxError)
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
2018-12-30 19:50:20 -06:00
|
|
|
return rhs;
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-12-30 19:50:20 -06:00
|
|
|
expr = std::make_unique<BinaryExpression>(tok.type, std::move(expr), std::move(rhs.expr));
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
return ParseResult::MakeSuccessfulResult(std::move(expr));
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-01-08 18:36:58 -06:00
|
|
|
ParseResult ParseParens()
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
|
|
|
// lparen already chewed
|
2019-01-08 18:36:58 -06:00
|
|
|
ParseResult result = ParseToplevel();
|
2017-06-07 15:08:55 -07:00
|
|
|
if (result.status != ParseStatus::Successful)
|
|
|
|
return result;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
const auto rparen = Chew();
|
|
|
|
if (rparen.type != TOK_RPAREN)
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
2019-03-02 14:47:26 -06:00
|
|
|
return ParseResult::MakeErrorResult(rparen, _trans("Expected closing paren."));
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-06-07 15:08:55 -07:00
|
|
|
return result;
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-01-08 18:36:58 -06:00
|
|
|
ParseResult ParseToplevel() { return ParseBinary(); }
|
2018-12-30 19:50:20 -06:00
|
|
|
}; // namespace ExpressionParser
|
2013-06-13 23:09:55 -04:00
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
ParseResult ParseTokens(const std::vector<Token>& tokens)
|
|
|
|
{
|
|
|
|
return Parser(tokens).Parse();
|
|
|
|
}
|
|
|
|
|
2017-06-07 18:48:17 -07:00
|
|
|
static ParseResult ParseComplexExpression(const std::string& str)
|
2013-06-13 23:09:55 -04:00
|
|
|
{
|
|
|
|
Lexer l(str);
|
|
|
|
std::vector<Token> tokens;
|
2019-03-02 14:47:26 -06:00
|
|
|
const ParseStatus tokenize_status = l.Tokenize(tokens);
|
2017-06-07 14:53:41 -07:00
|
|
|
if (tokenize_status != ParseStatus::Successful)
|
2019-03-02 14:47:26 -06:00
|
|
|
return ParseResult::MakeErrorResult(Token(TOK_INVALID), _trans("Tokenizing failed."));
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
return ParseTokens(tokens);
|
2013-06-13 23:09:55 -04:00
|
|
|
}
|
|
|
|
|
2017-06-07 18:48:17 -07:00
|
|
|
static std::unique_ptr<Expression> ParseBarewordExpression(const std::string& str)
|
2013-06-26 20:19:23 -04:00
|
|
|
{
|
|
|
|
ControlQualifier qualifier;
|
|
|
|
qualifier.control_name = str;
|
|
|
|
qualifier.has_device = false;
|
|
|
|
|
2017-06-07 18:48:17 -07:00
|
|
|
return std::make_unique<ControlExpression>(qualifier);
|
2017-06-07 18:29:02 -07:00
|
|
|
}
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
ParseResult ParseExpression(const std::string& str)
|
2017-06-07 18:29:02 -07:00
|
|
|
{
|
|
|
|
if (StripSpaces(str).empty())
|
2019-03-02 14:47:26 -06:00
|
|
|
return ParseResult::MakeEmptyResult();
|
2017-06-07 18:29:02 -07:00
|
|
|
|
2017-06-07 18:48:17 -07:00
|
|
|
auto bareword_expr = ParseBarewordExpression(str);
|
|
|
|
ParseResult complex_result = ParseComplexExpression(str);
|
2017-06-07 18:29:02 -07:00
|
|
|
|
|
|
|
if (complex_result.status != ParseStatus::Successful)
|
2014-08-30 16:44:28 -04:00
|
|
|
{
|
2019-03-02 14:47:26 -06:00
|
|
|
// This is a bit odd.
|
|
|
|
// Return the error status of the complex expression with the fallback barewords expression.
|
|
|
|
complex_result.expr = std::move(bareword_expr);
|
|
|
|
return complex_result;
|
2013-06-26 20:19:23 -04:00
|
|
|
}
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
complex_result.expr = std::make_unique<CoalesceExpression>(std::move(bareword_expr),
|
|
|
|
std::move(complex_result.expr));
|
|
|
|
return complex_result;
|
2013-06-26 20:19:23 -04:00
|
|
|
}
|
2019-06-17 16:39:24 -04:00
|
|
|
} // namespace ciface::ExpressionParser
|