patrick
/
plasp
Archived
1
0
Fork 0
This repository has been archived on 2023-07-19. You can view files and clone it, but cannot push or open issues or pull requests.
plasp/src/plasp/pddl/Expression.cpp

337 lines
11 KiB
C++
Raw Normal View History

#include <plasp/pddl/Expression.h>
#include <plasp/output/TranslatorException.h>
#include <plasp/pddl/Context.h>
#include <plasp/pddl/Domain.h>
#include <plasp/pddl/ExpressionContext.h>
#include <plasp/pddl/IO.h>
2016-06-03 13:33:05 +02:00
#include <plasp/pddl/expressions/And.h>
#include <plasp/pddl/expressions/Exists.h>
#include <plasp/pddl/expressions/ForAll.h>
2016-06-04 17:20:25 +02:00
#include <plasp/pddl/expressions/Imply.h>
2016-06-03 13:33:05 +02:00
#include <plasp/pddl/expressions/Not.h>
#include <plasp/pddl/expressions/Or.h>
#include <plasp/pddl/expressions/Predicate.h>
2016-06-03 17:12:39 +02:00
#include <plasp/pddl/expressions/PredicateDeclaration.h>
#include <plasp/pddl/expressions/Unsupported.h>
#include <plasp/pddl/expressions/When.h>
2017-05-12 14:17:57 +02:00
#include <tokenize/TokenizerException.h>
namespace plasp
{
namespace pddl
{
////////////////////////////////////////////////////////////////////////////////////////////////////
//
// Expression
//
////////////////////////////////////////////////////////////////////////////////////////////////////
2016-09-08 03:42:32 +02:00
ExpressionPointer Expression::copy()
{
return this;
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer Expression::normalized()
{
return reduced()->simplified()->existentiallyQuantified()->simplified();
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer Expression::reduced()
{
return this;
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer Expression::existentiallyQuantified()
{
return this;
}
2016-09-07 00:34:26 +02:00
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer Expression::simplified()
{
return this;
}
////////////////////////////////////////////////////////////////////////////////////////////////////
2017-05-09 13:00:44 +02:00
ExpressionPointer Expression::decomposed(expressions::DerivedPredicates &)
{
throw output::TranslatorException("expression cannot be decomposed (not normalized)");
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer Expression::negated()
{
if (is<expressions::Not>())
2016-12-07 02:29:48 +01:00
return as<expressions::Not>().argument();
auto notExpression = expressions::NotPointer(new expressions::Not);
notExpression->setArgument(this);
return notExpression;
}
////////////////////////////////////////////////////////////////////////////////////////////////////
// TODO: implement better (visitor pattern?)
void Expression::collectParameters(std::set<expressions::VariablePointer> &)
{
throw output::TranslatorException("expression parameters could not be collected (expression not normalized)");
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseEffectBodyExpression(Context &context, ExpressionContext &expressionContext);
ExpressionPointer parsePredicate(Context &context, ExpressionContext &expressionContext);
2016-06-02 21:53:10 +02:00
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parsePreconditionExpression(Context &context,
ExpressionContext &expressionContext)
{
2017-05-12 14:17:57 +02:00
auto &tokenizer = context.tokenizer;
2017-05-12 14:17:57 +02:00
tokenizer.skipWhiteSpace();
ExpressionPointer expression;
if ((expression = expressions::And::parse(context, expressionContext, parsePreconditionExpression))
|| (expression = expressions::ForAll::parse(context, expressionContext, parsePreconditionExpression)))
{
return expression;
}
2017-05-12 14:17:57 +02:00
const auto position = tokenizer.position();
2017-05-12 14:17:57 +02:00
tokenizer.expect<std::string>("(");
2017-05-12 14:17:57 +02:00
const auto expressionIdentifierPosition = tokenizer.position();
2017-05-12 14:17:57 +02:00
if (tokenizer.testIdentifierAndSkip("preference"))
2016-06-02 21:53:10 +02:00
{
// TODO: refactor
2017-05-12 14:17:57 +02:00
tokenizer.seek(expressionIdentifierPosition);
const auto expressionIdentifier = tokenizer.getIdentifier();
2017-05-12 14:17:57 +02:00
tokenizer.seek(position);
return expressions::Unsupported::parse(context);
2016-06-02 21:53:10 +02:00
}
2016-06-02 18:31:47 +02:00
2017-05-12 14:17:57 +02:00
tokenizer.seek(position);
return parseExpression(context, expressionContext);
2016-06-02 18:31:47 +02:00
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseExpression(Context &context, ExpressionContext &expressionContext)
2016-06-02 18:31:47 +02:00
{
2017-05-12 14:17:57 +02:00
auto &tokenizer = context.tokenizer;
2016-06-02 18:31:47 +02:00
2017-05-12 14:17:57 +02:00
tokenizer.skipWhiteSpace();
2016-06-02 18:31:47 +02:00
ExpressionPointer expression;
if ((expression = expressions::And::parse(context, expressionContext, parseExpression))
|| (expression = expressions::Or::parse(context, expressionContext, parseExpression))
|| (expression = expressions::Exists::parse(context, expressionContext, parseExpression))
|| (expression = expressions::ForAll::parse(context, expressionContext, parseExpression))
|| (expression = expressions::Not::parse(context, expressionContext, parseExpression))
|| (expression = expressions::Imply::parse(context, expressionContext, parseExpression))
|| (expression = expressions::Predicate::parse(context, expressionContext)))
{
return expression;
}
2017-05-12 14:17:57 +02:00
const auto position = tokenizer.position();
2017-05-12 14:17:57 +02:00
tokenizer.expect<std::string>("(");
2017-05-12 14:17:57 +02:00
const auto expressionIdentifierPosition = tokenizer.position();
2017-05-12 14:17:57 +02:00
if (tokenizer.testIdentifierAndSkip("-")
|| tokenizer.testIdentifierAndSkip("=")
|| tokenizer.testIdentifierAndSkip("*")
|| tokenizer.testIdentifierAndSkip("+")
|| tokenizer.testIdentifierAndSkip("-")
|| tokenizer.testIdentifierAndSkip("/")
|| tokenizer.testIdentifierAndSkip(">")
|| tokenizer.testIdentifierAndSkip("<")
|| tokenizer.testIdentifierAndSkip("=")
|| tokenizer.testIdentifierAndSkip(">=")
|| tokenizer.testIdentifierAndSkip("<="))
2016-06-02 18:31:47 +02:00
{
2017-05-12 14:17:57 +02:00
tokenizer.seek(expressionIdentifierPosition);
const auto expressionIdentifier = tokenizer.getIdentifier();
2017-05-12 14:17:57 +02:00
tokenizer.seek(position);
return expressions::Unsupported::parse(context);
2016-06-02 21:53:10 +02:00
}
2017-05-12 14:17:57 +02:00
tokenizer.seek(expressionIdentifierPosition);
const auto expressionIdentifier = tokenizer.getIdentifier();
2016-06-04 18:12:41 +02:00
2017-05-12 14:17:57 +02:00
tokenizer.seek(position);
throw tokenize::TokenizerException(tokenizer.location(), "expression type “" + expressionIdentifier + "” unknown or not allowed in this context");
2016-06-02 21:53:10 +02:00
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseEffectExpression(Context &context, ExpressionContext &expressionContext)
2016-06-02 21:53:10 +02:00
{
2017-05-12 14:17:57 +02:00
auto &tokenizer = context.tokenizer;
2016-06-02 21:53:10 +02:00
ExpressionPointer expression;
2016-06-02 21:53:10 +02:00
if ((expression = expressions::And::parse(context, expressionContext, parseEffectExpression))
|| (expression = expressions::ForAll::parse(context, expressionContext, parseEffectExpression))
|| (expression = expressions::When::parse(context, expressionContext, parseExpression, parseConditionalEffectExpression)))
{
return expression;
}
2017-05-12 14:17:57 +02:00
const auto position = tokenizer.position();
2017-05-12 14:17:57 +02:00
tokenizer.expect<std::string>("(");
2017-05-12 14:17:57 +02:00
const auto expressionIdentifierPosition = tokenizer.position();
2017-05-12 14:17:57 +02:00
if (tokenizer.testIdentifierAndSkip("when"))
2016-06-02 21:53:10 +02:00
{
2017-05-12 14:17:57 +02:00
tokenizer.seek(expressionIdentifierPosition);
const auto expressionIdentifier = tokenizer.getIdentifier();
2017-05-12 14:17:57 +02:00
tokenizer.seek(position);
return expressions::Unsupported::parse(context);
2016-06-02 21:53:10 +02:00
}
2017-05-12 14:17:57 +02:00
tokenizer.seek(position);
return parseEffectBodyExpression(context, expressionContext);
2016-06-02 21:53:10 +02:00
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseEffectBodyExpression(Context &context, ExpressionContext &expressionContext)
2016-06-02 21:53:10 +02:00
{
2017-05-12 14:17:57 +02:00
auto &tokenizer = context.tokenizer;
ExpressionPointer expression;
2016-06-02 21:53:10 +02:00
if ((expression = expressions::Not::parse(context, expressionContext, parsePredicate))
|| (expression = expressions::Predicate::parse(context, expressionContext)))
{
return expression;
}
2017-05-12 14:17:57 +02:00
const auto position = tokenizer.position();
2017-05-12 14:17:57 +02:00
tokenizer.expect<std::string>("(");
2016-06-04 18:12:41 +02:00
2017-05-12 14:17:57 +02:00
const auto expressionIdentifierPosition = tokenizer.position();
2017-05-12 14:17:57 +02:00
if (tokenizer.testIdentifierAndSkip("=")
|| tokenizer.testIdentifierAndSkip("assign")
|| tokenizer.testIdentifierAndSkip("scale-up")
|| tokenizer.testIdentifierAndSkip("scale-down")
|| tokenizer.testIdentifierAndSkip("increase")
|| tokenizer.testIdentifierAndSkip("decrease"))
2016-06-02 21:53:10 +02:00
{
2017-05-12 14:17:57 +02:00
tokenizer.seek(expressionIdentifierPosition);
const auto expressionIdentifier = tokenizer.getIdentifier();
2017-05-12 14:17:57 +02:00
tokenizer.seek(position);
return expressions::Unsupported::parse(context);
2016-06-02 18:31:47 +02:00
}
2017-05-12 14:17:57 +02:00
tokenizer.seek(expressionIdentifierPosition);
const auto expressionIdentifier = tokenizer.getIdentifier();
2017-05-12 14:17:57 +02:00
tokenizer.seek(position);
throw tokenize::TokenizerException(tokenizer.location(), "expression type “" + expressionIdentifier + "” unknown or not allowed in this context");
2016-06-02 18:31:47 +02:00
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseConditionalEffectExpression(Context &context, ExpressionContext &expressionContext)
{
ExpressionPointer expression;
if ((expression = expressions::And::parse(context, expressionContext, parseEffectBodyExpression)))
return expression;
return parseEffectBodyExpression(context, expressionContext);
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parsePredicate(Context &context, ExpressionContext &expressionContext)
{
ExpressionPointer expression;
if ((expression = expressions::Predicate::parse(context, expressionContext)))
return expression;
2017-05-12 14:17:57 +02:00
throw tokenize::TokenizerException(context.tokenizer.location(), "expected predicate");
}
////////////////////////////////////////////////////////////////////////////////////////////////////
2016-06-13 19:02:15 +02:00
ExpressionPointer parseLiteral(Context &context, ExpressionContext &expressionContext)
{
ExpressionPointer expression;
if ((expression = parseAtomicFormula(context, expressionContext))
|| (expression = expressions::Not::parse(context, expressionContext, parseAtomicFormula)))
{
return expression;
}
return nullptr;
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseAtomicFormula(Context &context, ExpressionContext &expressionContext)
{
2017-05-12 14:17:57 +02:00
auto &tokenizer = context.tokenizer;
2016-06-13 19:02:15 +02:00
ExpressionPointer expression;
if ((expression = expressions::Predicate::parse(context, expressionContext)))
return expression;
2017-05-12 14:17:57 +02:00
const auto position = tokenizer.position();
2016-06-13 19:02:15 +02:00
2017-05-12 14:17:57 +02:00
if (!tokenizer.testAndSkip<std::string>("("))
return nullptr;
2016-06-13 19:02:15 +02:00
2017-05-12 14:17:57 +02:00
const auto expressionIdentifierPosition = tokenizer.position();
2016-06-13 19:02:15 +02:00
2017-05-12 14:17:57 +02:00
if (tokenizer.testIdentifierAndSkip("="))
2016-06-13 19:02:15 +02:00
{
2017-05-12 14:17:57 +02:00
tokenizer.seek(expressionIdentifierPosition);
const auto expressionIdentifier = tokenizer.getIdentifier();
2016-06-13 19:02:15 +02:00
2017-05-12 14:17:57 +02:00
tokenizer.seek(position);
return expressions::Unsupported::parse(context);
2016-06-13 19:02:15 +02:00
}
2017-05-12 14:17:57 +02:00
tokenizer.seek(position);
2016-06-13 19:02:15 +02:00
return nullptr;
}
////////////////////////////////////////////////////////////////////////////////////////////////////
}
}