2023-08-11 15:59:52 +00:00
|
|
|
#include <libsmtutil/SMTLibParser.h>
|
|
|
|
|
|
|
|
#include <liblangutil/Common.h>
|
|
|
|
|
|
|
|
#include <libsolutil/Visitor.h>
|
|
|
|
#include <libsolutil/StringUtils.h>
|
|
|
|
|
|
|
|
|
|
|
|
using namespace solidity::langutil;
|
|
|
|
using namespace solidity::smtutil;
|
|
|
|
|
|
|
|
std::string SMTLib2Expression::toString() const {
|
|
|
|
return std::visit(solidity::util::GenericVisitor{
|
|
|
|
[](std::string const& _sv) { return _sv; },
|
|
|
|
[](std::vector<SMTLib2Expression> const& _subExpr) {
|
|
|
|
std::vector<std::string> formatted;
|
|
|
|
for (auto const& item: _subExpr)
|
|
|
|
formatted.emplace_back(item.toString());
|
|
|
|
return "(" + solidity::util::joinHumanReadable(formatted, " ") + ")";
|
|
|
|
}
|
|
|
|
}, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
SMTLib2Expression SMTLib2Parser::parseExpression() {
|
|
|
|
skipWhitespace();
|
|
|
|
if (token() == '(')
|
|
|
|
{
|
|
|
|
advance();
|
|
|
|
skipWhitespace();
|
|
|
|
std::vector<SMTLib2Expression> subExpressions;
|
|
|
|
while (token() != 0 && token() != ')')
|
|
|
|
{
|
|
|
|
subExpressions.emplace_back(parseExpression());
|
|
|
|
skipWhitespace();
|
|
|
|
}
|
|
|
|
solAssert(token() == ')');
|
|
|
|
// simulate whitespace because we do not want to read the next token
|
|
|
|
// since it might block.
|
|
|
|
m_token = ' ';
|
|
|
|
return {std::move(subExpressions)};
|
|
|
|
} else
|
|
|
|
return {parseToken()};
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string SMTLib2Parser::parseToken() {
|
|
|
|
std::string result;
|
|
|
|
|
|
|
|
skipWhitespace();
|
|
|
|
bool isPipe = token() == '|';
|
|
|
|
if (isPipe)
|
|
|
|
advance();
|
|
|
|
while (token() != 0)
|
|
|
|
{
|
|
|
|
char c = token();
|
|
|
|
if (isPipe && c == '|')
|
|
|
|
{
|
|
|
|
advance();
|
|
|
|
break;
|
|
|
|
} else if (!isPipe && (isWhiteSpace(c) || c == '(' || c == ')'))
|
|
|
|
break;
|
|
|
|
result.push_back(c);
|
|
|
|
advance();
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMTLib2Parser::advance() {
|
2023-08-12 17:12:33 +00:00
|
|
|
if (!m_input.good())
|
|
|
|
throw ParsingException{};
|
2023-08-11 15:59:52 +00:00
|
|
|
m_token = static_cast<char>(m_input.get());
|
|
|
|
if (token() == ';')
|
|
|
|
while (token() != '\n' && token() != 0)
|
|
|
|
m_token = static_cast<char>(m_input.get());
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMTLib2Parser::skipWhitespace() {
|
|
|
|
while (isWhiteSpace(token()))
|
|
|
|
advance();
|
|
|
|
}
|