2018-12-20 23:30:35 +00:00
|
|
|
#include "log.h"
|
2018-09-25 01:04:42 +01:00
|
|
|
#include "parseutil.h"
|
|
|
|
|
|
|
|
#include <QRegularExpression>
|
2019-05-06 17:42:09 +01:00
|
|
|
#include <QJsonDocument>
|
|
|
|
#include <QJsonObject>
|
2018-09-25 01:04:42 +01:00
|
|
|
#include <QStack>
|
|
|
|
|
2021-01-30 03:05:08 +00:00
|
|
|
const QRegularExpression ParseUtil::re_incScriptLabel("\\b(?<label>[\\w_][\\w\\d_]*):{1,2}");
|
|
|
|
const QRegularExpression ParseUtil::re_globalIncScriptLabel("\\b(?<label>[\\w_][\\w\\d_]*)::");
|
|
|
|
const QRegularExpression ParseUtil::re_poryScriptLabel("\\b(script)(\\((global|local)\\))?\\s*\\b(?<label>[\\w_][\\w\\d_]*)");
|
|
|
|
const QRegularExpression ParseUtil::re_globalPoryScriptLabel("\\b(script)(\\((global)\\))?\\s*\\b(?<label>[\\w_][\\w\\d_]*)");
|
|
|
|
const QRegularExpression ParseUtil::re_poryRawSection("\\b(raw)\\s*`(?<raw_script>[^`]*)");
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
void ParseUtil::set_root(const QString &dir) {
|
2019-05-06 17:42:09 +01:00
|
|
|
this->root = dir;
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
void ParseUtil::error(const QString &message, const QString &expression) {
|
2019-05-05 21:11:00 +01:00
|
|
|
QStringList lines = text.split(QRegularExpression("[\r\n]"));
|
2019-05-06 17:42:09 +01:00
|
|
|
int lineNum = 0, colNum = 0;
|
|
|
|
for (QString line : lines) {
|
|
|
|
lineNum++;
|
|
|
|
colNum = line.indexOf(expression) + 1;
|
|
|
|
if (colNum) break;
|
|
|
|
}
|
|
|
|
logError(QString("%1:%2:%3: %4").arg(file).arg(lineNum).arg(colNum).arg(message));
|
2018-09-25 01:04:42 +01:00
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
QString ParseUtil::readTextFile(const QString &path) {
|
2019-05-06 17:42:09 +01:00
|
|
|
QFile file(path);
|
|
|
|
if (!file.open(QIODevice::ReadOnly)) {
|
|
|
|
logError(QString("Could not open '%1': ").arg(path) + file.errorString());
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
QTextStream in(&file);
|
|
|
|
QString text = "";
|
|
|
|
while (!in.atEnd()) {
|
2021-02-03 15:24:59 +00:00
|
|
|
text += in.readLine() + '\n';
|
2019-05-06 17:42:09 +01:00
|
|
|
}
|
|
|
|
return text;
|
|
|
|
}
|
|
|
|
|
2021-02-03 15:24:59 +00:00
|
|
|
int ParseUtil::textFileLineCount(const QString &path) {
|
|
|
|
const QString text = readTextFile(path);
|
|
|
|
return text.split('\n').count() + 1;
|
|
|
|
}
|
2019-05-06 17:42:09 +01:00
|
|
|
|
2021-02-16 11:15:54 +00:00
|
|
|
QList<QStringList> ParseUtil::parseAsm(const QString &filename) {
|
|
|
|
QList<QStringList> parsed;
|
2019-05-06 17:42:09 +01:00
|
|
|
|
2021-02-16 11:15:54 +00:00
|
|
|
text = readTextFile(root + '/' + filename);
|
2021-02-16 12:15:47 +00:00
|
|
|
const QStringList lines = removeLineComments(text, "@").split('\n');
|
|
|
|
for (const auto &line : lines) {
|
|
|
|
const QString trimmedLine = line.trimmed();
|
|
|
|
if (trimmedLine.isEmpty()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (line.contains(':')) {
|
2021-02-16 11:15:54 +00:00
|
|
|
const QString label = line.left(line.indexOf(':'));
|
|
|
|
const QStringList list{ ".label", label }; // .label is not a real keyword. It's used only to make the output more regular.
|
|
|
|
parsed.append(list);
|
2018-09-25 01:04:42 +01:00
|
|
|
// There should not be anything else on the line.
|
|
|
|
// gas will raise a syntax error if there is.
|
|
|
|
} else {
|
2021-02-06 00:43:49 +00:00
|
|
|
int index = trimmedLine.indexOf(QRegularExpression("\\s+"));
|
2021-02-16 12:15:47 +00:00
|
|
|
const QString macro = trimmedLine.left(index);
|
2021-02-06 00:43:49 +00:00
|
|
|
QStringList params(trimmedLine.right(trimmedLine.length() - index).trimmed().split(QRegularExpression("\\s*,\\s*")));
|
2018-09-25 01:04:42 +01:00
|
|
|
params.prepend(macro);
|
2021-02-16 11:15:54 +00:00
|
|
|
parsed.append(params);
|
2018-09-25 01:04:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return parsed;
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
int ParseUtil::evaluateDefine(const QString &define, const QMap<QString, int> &knownDefines) {
|
2018-09-25 01:04:42 +01:00
|
|
|
QList<Token> tokens = tokenizeExpression(define, knownDefines);
|
|
|
|
QList<Token> postfixExpression = generatePostfix(tokens);
|
|
|
|
return evaluatePostfix(postfixExpression);
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
QList<Token> ParseUtil::tokenizeExpression(QString expression, const QMap<QString, int> &knownIdentifiers) {
|
2018-09-25 01:04:42 +01:00
|
|
|
QList<Token> tokens;
|
|
|
|
|
|
|
|
QStringList tokenTypes = (QStringList() << "hex" << "decimal" << "identifier" << "operator" << "leftparen" << "rightparen");
|
|
|
|
QRegularExpression re("^(?<hex>0x[0-9a-fA-F]+)|(?<decimal>[0-9]+)|(?<identifier>[a-zA-Z_0-9]+)|(?<operator>[+\\-*\\/<>|^%]+)|(?<leftparen>\\()|(?<rightparen>\\))");
|
|
|
|
|
|
|
|
expression = expression.trimmed();
|
|
|
|
while (!expression.isEmpty()) {
|
|
|
|
QRegularExpressionMatch match = re.match(expression);
|
|
|
|
if (!match.hasMatch()) {
|
2018-12-20 23:30:35 +00:00
|
|
|
logWarn(QString("Failed to tokenize expression: '%1'").arg(expression));
|
2018-09-25 01:04:42 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
for (QString tokenType : tokenTypes) {
|
|
|
|
QString token = match.captured(tokenType);
|
|
|
|
if (!token.isEmpty()) {
|
|
|
|
if (tokenType == "identifier") {
|
2021-02-16 12:15:47 +00:00
|
|
|
if (knownIdentifiers.contains(token)) {
|
|
|
|
QString actualToken = QString("%1").arg(knownIdentifiers.value(token));
|
2018-09-25 01:04:42 +01:00
|
|
|
expression = expression.replace(0, token.length(), actualToken);
|
|
|
|
token = actualToken;
|
|
|
|
tokenType = "decimal";
|
|
|
|
} else {
|
2019-05-05 21:11:00 +01:00
|
|
|
tokenType = "error";
|
2019-05-06 17:42:09 +01:00
|
|
|
QString message = QString("unknown token '%1' found in expression '%2'")
|
|
|
|
.arg(token).arg(expression);
|
|
|
|
error(message, expression);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (tokenType == "operator") {
|
|
|
|
if (!Token::precedenceMap.contains(token)) {
|
|
|
|
QString message = QString("unsupported postfix operator: '%1'")
|
|
|
|
.arg(token);
|
|
|
|
error(message, expression);
|
2018-09-25 01:04:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tokens.append(Token(token, tokenType));
|
|
|
|
expression = expression.remove(0, token.length()).trimmed();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return tokens;
|
|
|
|
}
|
|
|
|
|
|
|
|
QMap<QString, int> Token::precedenceMap = QMap<QString, int>(
|
|
|
|
{
|
|
|
|
{"*", 3},
|
|
|
|
{"/", 3},
|
2021-12-19 23:48:06 +00:00
|
|
|
{"%", 3},
|
2018-09-25 01:04:42 +01:00
|
|
|
{"+", 4},
|
|
|
|
{"-", 4},
|
|
|
|
{"<<", 5},
|
|
|
|
{">>", 5},
|
|
|
|
{"&", 8},
|
|
|
|
{"^", 9},
|
|
|
|
{"|", 10}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Shunting-yard algorithm for generating postfix notation.
|
|
|
|
// https://en.wikipedia.org/wiki/Shunting-yard_algorithm
|
2021-02-16 12:15:47 +00:00
|
|
|
QList<Token> ParseUtil::generatePostfix(const QList<Token> &tokens) {
|
2018-09-25 01:04:42 +01:00
|
|
|
QList<Token> output;
|
|
|
|
QStack<Token> operatorStack;
|
|
|
|
for (Token token : tokens) {
|
2019-09-21 22:06:04 +01:00
|
|
|
if (token.type == TokenClass::Number) {
|
2018-09-25 01:04:42 +01:00
|
|
|
output.append(token);
|
|
|
|
} else if (token.value == "(") {
|
|
|
|
operatorStack.push(token);
|
|
|
|
} else if (token.value == ")") {
|
|
|
|
while (!operatorStack.empty() && operatorStack.top().value != "(") {
|
|
|
|
output.append(operatorStack.pop());
|
|
|
|
}
|
|
|
|
if (!operatorStack.empty()) {
|
|
|
|
// pop the left parenthesis token
|
|
|
|
operatorStack.pop();
|
|
|
|
} else {
|
2018-12-20 23:30:35 +00:00
|
|
|
logError("Mismatched parentheses detected in expression!");
|
2018-09-25 01:04:42 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// token is an operator
|
|
|
|
while (!operatorStack.isEmpty()
|
|
|
|
&& operatorStack.top().operatorPrecedence <= token.operatorPrecedence
|
|
|
|
&& operatorStack.top().value != "(") {
|
|
|
|
output.append(operatorStack.pop());
|
|
|
|
}
|
|
|
|
operatorStack.push(token);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!operatorStack.isEmpty()) {
|
|
|
|
if (operatorStack.top().value == "(" || operatorStack.top().value == ")") {
|
2018-12-20 23:30:35 +00:00
|
|
|
logError("Mismatched parentheses detected in expression!");
|
2018-09-25 01:04:42 +01:00
|
|
|
} else {
|
|
|
|
output.append(operatorStack.pop());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Evaluate postfix expression.
|
|
|
|
// https://en.wikipedia.org/wiki/Reverse_Polish_notation#Postfix_evaluation_algorithm
|
2021-02-16 12:15:47 +00:00
|
|
|
int ParseUtil::evaluatePostfix(const QList<Token> &postfix) {
|
2018-09-25 01:04:42 +01:00
|
|
|
QStack<Token> stack;
|
|
|
|
for (Token token : postfix) {
|
2019-09-21 22:06:04 +01:00
|
|
|
if (token.type == TokenClass::Operator && stack.size() > 1) {
|
2018-09-25 01:04:42 +01:00
|
|
|
int op2 = stack.pop().value.toInt(nullptr, 0);
|
|
|
|
int op1 = stack.pop().value.toInt(nullptr, 0);
|
|
|
|
int result = 0;
|
|
|
|
if (token.value == "*") {
|
|
|
|
result = op1 * op2;
|
|
|
|
} else if (token.value == "/") {
|
|
|
|
result = op1 / op2;
|
2021-12-19 23:48:06 +00:00
|
|
|
} else if (token.value == "%") {
|
|
|
|
result = op1 % op2;
|
2018-09-25 01:04:42 +01:00
|
|
|
} else if (token.value == "+") {
|
|
|
|
result = op1 + op2;
|
|
|
|
} else if (token.value == "-") {
|
|
|
|
result = op1 - op2;
|
|
|
|
} else if (token.value == "<<") {
|
|
|
|
result = op1 << op2;
|
|
|
|
} else if (token.value == ">>") {
|
|
|
|
result = op1 >> op2;
|
|
|
|
} else if (token.value == "&") {
|
|
|
|
result = op1 & op2;
|
|
|
|
} else if (token.value == "^") {
|
|
|
|
result = op1 ^ op2;
|
|
|
|
} else if (token.value == "|") {
|
|
|
|
result = op1 | op2;
|
|
|
|
}
|
|
|
|
stack.push(Token(QString("%1").arg(result), "decimal"));
|
2019-09-21 22:06:04 +01:00
|
|
|
} else if (token.type != TokenClass::Error) {
|
2018-09-25 01:04:42 +01:00
|
|
|
stack.push(token);
|
2019-05-05 21:11:00 +01:00
|
|
|
} // else ignore errored tokens, we have already warned the user.
|
2018-09-25 01:04:42 +01:00
|
|
|
}
|
2020-06-04 18:28:51 +01:00
|
|
|
return stack.size() ? stack.pop().value.toInt(nullptr, 0) : 0;
|
2018-09-25 01:04:42 +01:00
|
|
|
}
|
2019-05-06 17:42:09 +01:00
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
QString ParseUtil::readCIncbin(const QString &filename, const QString &label) {
|
2019-05-06 17:42:09 +01:00
|
|
|
QString path;
|
|
|
|
|
|
|
|
if (label.isNull()) {
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
text = readTextFile(root + "/" + filename);
|
|
|
|
|
2021-02-06 00:43:49 +00:00
|
|
|
QRegularExpression re(QString(
|
2019-05-06 17:42:09 +01:00
|
|
|
"\\b%1\\b"
|
|
|
|
"\\s*\\[?\\s*\\]?\\s*=\\s*"
|
|
|
|
"INCBIN_[US][0-9][0-9]?"
|
2019-06-04 04:48:39 +01:00
|
|
|
"\\(\\s*\"([^\"]*)\"\\s*\\)").arg(label));
|
2019-05-06 17:42:09 +01:00
|
|
|
|
2021-02-06 00:43:49 +00:00
|
|
|
QRegularExpressionMatch match;
|
|
|
|
qsizetype pos = text.indexOf(re, 0, &match);
|
2019-05-06 17:42:09 +01:00
|
|
|
if (pos != -1) {
|
2021-02-06 00:43:49 +00:00
|
|
|
path = match.captured(1);
|
2019-05-06 17:42:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
QMap<QString, int> ParseUtil::readCDefines(const QString &filename,
|
|
|
|
const QStringList &prefixes,
|
|
|
|
QMap<QString, int> allDefines)
|
|
|
|
{
|
2019-05-06 17:42:09 +01:00
|
|
|
QMap<QString, int> filteredDefines;
|
|
|
|
|
|
|
|
file = filename;
|
|
|
|
|
|
|
|
if (file.isEmpty()) {
|
|
|
|
return filteredDefines;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString filepath = root + "/" + file;
|
|
|
|
text = readTextFile(filepath);
|
|
|
|
|
|
|
|
if (text.isNull()) {
|
|
|
|
logError(QString("Failed to read C defines file: '%1'").arg(filepath));
|
|
|
|
return filteredDefines;
|
|
|
|
}
|
|
|
|
|
|
|
|
text.replace(QRegularExpression("(//.*)|(\\/+\\*+[^*]*\\*+\\/+)"), "");
|
2019-06-04 02:29:09 +01:00
|
|
|
text.replace(QRegularExpression("(\\\\\\s+)"), "");
|
2020-06-04 18:28:51 +01:00
|
|
|
allDefines.insert("FALSE", 0);
|
|
|
|
allDefines.insert("TRUE", 1);
|
2019-05-06 17:42:09 +01:00
|
|
|
|
|
|
|
QRegularExpression re("#define\\s+(?<defineName>\\w+)[^\\S\\n]+(?<defineValue>.+)");
|
|
|
|
QRegularExpressionMatchIterator iter = re.globalMatch(text);
|
|
|
|
while (iter.hasNext()) {
|
|
|
|
QRegularExpressionMatch match = iter.next();
|
|
|
|
QString name = match.captured("defineName");
|
|
|
|
QString expression = match.captured("defineValue");
|
|
|
|
if (expression == " ") continue;
|
2021-02-16 12:15:47 +00:00
|
|
|
int value = evaluateDefine(expression, allDefines);
|
2019-05-06 17:42:09 +01:00
|
|
|
allDefines.insert(name, value);
|
|
|
|
for (QString prefix : prefixes) {
|
|
|
|
if (name.startsWith(prefix) || QRegularExpression(prefix).match(name).hasMatch()) {
|
|
|
|
filteredDefines.insert(name, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return filteredDefines;
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
QStringList ParseUtil::readCDefinesSorted(const QString &filename,
|
|
|
|
const QStringList &prefixes,
|
|
|
|
const QMap<QString, int> &knownDefines)
|
|
|
|
{
|
2020-07-13 21:36:43 +01:00
|
|
|
QMap<QString, int> defines = readCDefines(filename, prefixes, knownDefines);
|
2019-05-06 17:42:09 +01:00
|
|
|
|
|
|
|
// The defines should to be sorted by their underlying value, not alphabetically.
|
|
|
|
// Reverse the map and read out the resulting keys in order.
|
|
|
|
QMultiMap<int, QString> definesInverse;
|
|
|
|
for (QString defineName : defines.keys()) {
|
|
|
|
definesInverse.insert(defines[defineName], defineName);
|
|
|
|
}
|
2021-02-16 12:15:47 +00:00
|
|
|
return definesInverse.values();
|
2019-05-06 17:42:09 +01:00
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
QStringList ParseUtil::readCArray(const QString &filename, const QString &label) {
|
2019-05-06 17:42:09 +01:00
|
|
|
QStringList list;
|
|
|
|
|
|
|
|
if (label.isNull()) {
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
|
|
|
file = filename;
|
|
|
|
text = readTextFile(root + "/" + filename);
|
|
|
|
|
2020-02-12 22:38:07 +00:00
|
|
|
QRegularExpression re(QString(R"(\b%1\b\s*(\[?[^\]]*\])?\s*=\s*\{([^\}]*)\})").arg(label));
|
2019-05-06 17:42:09 +01:00
|
|
|
QRegularExpressionMatch match = re.match(text);
|
|
|
|
|
|
|
|
if (match.hasMatch()) {
|
2020-02-12 22:38:07 +00:00
|
|
|
QString body = match.captured(2);
|
2019-05-06 17:42:09 +01:00
|
|
|
QStringList split = body.split(',');
|
|
|
|
for (QString item : split) {
|
|
|
|
item = item.trimmed();
|
2019-06-04 04:48:39 +01:00
|
|
|
if (!item.contains(QRegularExpression("[^A-Za-z0-9_&()\\s]"))) list.append(item);
|
2019-05-06 17:42:09 +01:00
|
|
|
// do not print error info here because this is called dozens of times
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
QMap<QString, QString> ParseUtil::readNamedIndexCArray(const QString &filename, const QString &label) {
|
2019-05-06 17:42:09 +01:00
|
|
|
text = readTextFile(root + "/" + filename);
|
|
|
|
QMap<QString, QString> map;
|
|
|
|
|
2020-02-12 22:38:07 +00:00
|
|
|
QRegularExpression re_text(QString(R"(\b%1\b\s*(\[?[^\]]*\])?\s*=\s*\{([^\}]*)\})").arg(label));
|
|
|
|
QString body = re_text.match(text).captured(2).replace(QRegularExpression("\\s*"), "");
|
2021-02-16 11:15:54 +00:00
|
|
|
|
2020-12-13 23:05:28 +00:00
|
|
|
QRegularExpression re("\\[(?<index>[A-Za-z0-9_]*)\\]=(?<value>&?[A-Za-z0-9_]*)");
|
2019-05-06 17:42:09 +01:00
|
|
|
QRegularExpressionMatchIterator iter = re.globalMatch(body);
|
|
|
|
|
|
|
|
while (iter.hasNext()) {
|
|
|
|
QRegularExpressionMatch match = iter.next();
|
|
|
|
QString key = match.captured("index");
|
|
|
|
QString value = match.captured("value");
|
|
|
|
map.insert(key, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
2021-02-16 11:15:54 +00:00
|
|
|
QList<QStringList> ParseUtil::getLabelMacros(const QList<QStringList> &list, const QString &label) {
|
2019-05-06 17:42:09 +01:00
|
|
|
bool in_label = false;
|
2021-02-16 11:15:54 +00:00
|
|
|
QList<QStringList> new_list;
|
|
|
|
for (const auto ¶ms : list) {
|
|
|
|
const QString macro = params.value(0);
|
2019-05-06 17:42:09 +01:00
|
|
|
if (macro == ".label") {
|
|
|
|
if (params.value(1) == label) {
|
|
|
|
in_label = true;
|
|
|
|
} else if (in_label) {
|
|
|
|
// If nothing has been read yet, assume the label
|
|
|
|
// we're looking for is in a stack of labels.
|
2021-02-16 11:15:54 +00:00
|
|
|
if (new_list.length() > 0) {
|
2019-05-06 17:42:09 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (in_label) {
|
2021-02-16 11:15:54 +00:00
|
|
|
new_list.append(params);
|
2019-05-06 17:42:09 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return new_list;
|
|
|
|
}
|
|
|
|
|
|
|
|
// For if you don't care about filtering by macro,
|
|
|
|
// and just want all values associated with some label.
|
2021-02-16 11:15:54 +00:00
|
|
|
QStringList ParseUtil::getLabelValues(const QList<QStringList> &list, const QString &label) {
|
|
|
|
const QList<QStringList> labelMacros = getLabelMacros(list, label);
|
|
|
|
QStringList values;
|
|
|
|
for (const auto ¶ms : labelMacros) {
|
|
|
|
const QString macro = params.value(0);
|
2019-05-06 17:42:09 +01:00
|
|
|
if (macro == ".align" || macro == ".ifdef" || macro == ".ifndef") {
|
|
|
|
continue;
|
|
|
|
}
|
2021-02-16 11:15:54 +00:00
|
|
|
for (int i = 1; i < params.length(); i++) {
|
|
|
|
values.append(params.value(i));
|
2019-05-06 17:42:09 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return values;
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
bool ParseUtil::tryParseJsonFile(QJsonDocument *out, const QString &filepath) {
|
2019-05-06 17:42:09 +01:00
|
|
|
QFile file(filepath);
|
|
|
|
if (!file.open(QIODevice::ReadOnly)) {
|
|
|
|
logError(QString("Error: Could not open %1 for reading").arg(filepath));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
const QByteArray data = file.readAll();
|
2019-05-06 17:42:09 +01:00
|
|
|
QJsonParseError parseError;
|
2021-02-16 12:15:47 +00:00
|
|
|
const QJsonDocument jsonDoc = QJsonDocument::fromJson(data, &parseError);
|
2019-05-06 17:42:09 +01:00
|
|
|
file.close();
|
|
|
|
if (parseError.error != QJsonParseError::NoError) {
|
|
|
|
logError(QString("Error: Failed to parse json file %1: %2").arg(filepath).arg(parseError.errorString()));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out = jsonDoc;
|
|
|
|
return true;
|
|
|
|
}
|
2020-02-12 00:34:08 +00:00
|
|
|
|
2021-08-13 00:19:21 +01:00
|
|
|
bool ParseUtil::tryParseOrderedJsonFile(poryjson::Json::object *out, const QString &filepath) {
|
|
|
|
QString err;
|
|
|
|
QString jsonTxt = readTextFile(filepath);
|
|
|
|
*out = OrderedJson::parse(jsonTxt, err).object_items();
|
|
|
|
if (!err.isEmpty()) {
|
|
|
|
logError(QString("Error: Failed to parse json file %1: %2").arg(filepath).arg(err));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
bool ParseUtil::ensureFieldsExist(const QJsonObject &obj, const QList<QString> &fields) {
|
2020-02-12 00:34:08 +00:00
|
|
|
for (QString field : fields) {
|
|
|
|
if (!obj.contains(field)) {
|
|
|
|
logError(QString("JSON object is missing field '%1'.").arg(field));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2020-11-22 06:04:46 +00:00
|
|
|
|
2020-11-26 11:09:58 +00:00
|
|
|
int ParseUtil::getScriptLineNumber(const QString &filePath, const QString &scriptLabel) {
|
2020-12-02 08:39:08 +00:00
|
|
|
if (scriptLabel.isEmpty())
|
|
|
|
return 0;
|
|
|
|
|
2020-12-04 14:29:38 +00:00
|
|
|
if (filePath.endsWith(".inc") || filePath.endsWith(".s"))
|
2020-11-26 11:09:58 +00:00
|
|
|
return getRawScriptLineNumber(readTextFile(filePath), scriptLabel);
|
|
|
|
else if (filePath.endsWith(".pory"))
|
|
|
|
return getPoryScriptLineNumber(readTextFile(filePath), scriptLabel);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ParseUtil::getRawScriptLineNumber(QString text, const QString &scriptLabel) {
|
2021-02-16 12:15:47 +00:00
|
|
|
text = removeStringLiterals(text);
|
|
|
|
text = removeLineComments(text, "@");
|
2020-11-26 11:09:58 +00:00
|
|
|
|
|
|
|
QRegularExpressionMatchIterator it = re_incScriptLabel.globalMatch(text);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
const QRegularExpressionMatch match = it.next();
|
|
|
|
if (match.captured("label") == scriptLabel)
|
|
|
|
return text.left(match.capturedStart("label")).count('\n') + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2020-11-22 06:04:46 +00:00
|
|
|
|
2020-11-26 11:09:58 +00:00
|
|
|
int ParseUtil::getPoryScriptLineNumber(QString text, const QString &scriptLabel) {
|
2021-02-16 12:15:47 +00:00
|
|
|
text = removeStringLiterals(text);
|
|
|
|
text = removeLineComments(text, {"//", "#"});
|
2020-11-26 11:09:58 +00:00
|
|
|
|
|
|
|
QRegularExpressionMatchIterator it = re_poryScriptLabel.globalMatch(text);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
const QRegularExpressionMatch match = it.next();
|
|
|
|
if (match.captured("label") == scriptLabel)
|
|
|
|
return text.left(match.capturedStart("label")).count('\n') + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
QRegularExpressionMatchIterator raw_it = re_poryRawSection.globalMatch(text);
|
|
|
|
while (raw_it.hasNext()) {
|
|
|
|
const QRegularExpressionMatch match = raw_it.next();
|
2020-12-01 12:12:32 +00:00
|
|
|
const int relativelineNum = getRawScriptLineNumber(match.captured("raw_script"), scriptLabel);
|
|
|
|
if (relativelineNum)
|
|
|
|
return text.left(match.capturedStart("raw_script")).count('\n') + relativelineNum;
|
2020-11-22 06:04:46 +00:00
|
|
|
}
|
2020-11-26 11:09:58 +00:00
|
|
|
|
2020-11-22 06:04:46 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2020-11-26 11:09:58 +00:00
|
|
|
|
2021-01-30 03:05:08 +00:00
|
|
|
QStringList ParseUtil::getGlobalScriptLabels(const QString &filePath) {
|
|
|
|
if (filePath.endsWith(".inc") || filePath.endsWith(".s"))
|
|
|
|
return getGlobalRawScriptLabels(readTextFile(filePath));
|
|
|
|
else if (filePath.endsWith(".pory"))
|
|
|
|
return getGlobalPoryScriptLabels(readTextFile(filePath));
|
2021-04-16 14:04:38 +01:00
|
|
|
else
|
|
|
|
return { };
|
2021-01-30 03:05:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QStringList ParseUtil::getGlobalRawScriptLabels(QString text) {
|
|
|
|
removeStringLiterals(text);
|
|
|
|
removeLineComments(text, "@");
|
|
|
|
|
2021-04-16 14:04:38 +01:00
|
|
|
QStringList rawScriptLabels;
|
2021-01-30 03:05:08 +00:00
|
|
|
|
|
|
|
QRegularExpressionMatchIterator it = re_globalIncScriptLabel.globalMatch(text);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
const QRegularExpressionMatch match = it.next();
|
|
|
|
rawScriptLabels << match.captured("label");
|
|
|
|
}
|
|
|
|
|
|
|
|
return rawScriptLabels;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList ParseUtil::getGlobalPoryScriptLabels(QString text) {
|
|
|
|
removeStringLiterals(text);
|
|
|
|
removeLineComments(text, {"//", "#"});
|
|
|
|
|
2021-04-16 14:04:38 +01:00
|
|
|
QStringList poryScriptLabels;
|
2021-01-30 03:05:08 +00:00
|
|
|
|
|
|
|
QRegularExpressionMatchIterator it = re_globalPoryScriptLabel.globalMatch(text);
|
|
|
|
while (it.hasNext())
|
|
|
|
poryScriptLabels << it.next().captured("label");
|
|
|
|
|
|
|
|
QRegularExpressionMatchIterator raw_it = re_poryRawSection.globalMatch(text);
|
|
|
|
while (raw_it.hasNext())
|
|
|
|
poryScriptLabels << getGlobalRawScriptLabels(raw_it.next().captured("raw_script"));
|
|
|
|
|
|
|
|
return poryScriptLabels;
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
QString ParseUtil::removeStringLiterals(QString text) {
|
2020-11-26 11:09:58 +00:00
|
|
|
static const QRegularExpression re_string("\".*\"");
|
|
|
|
return text.remove(re_string);
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
QString ParseUtil::removeLineComments(QString text, const QString &commentSymbol) {
|
2020-11-26 11:09:58 +00:00
|
|
|
const QRegularExpression re_lineComment(commentSymbol + "+.*");
|
|
|
|
return text.remove(re_lineComment);
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:15:47 +00:00
|
|
|
QString ParseUtil::removeLineComments(QString text, const QStringList &commentSymbols) {
|
2020-11-26 11:09:58 +00:00
|
|
|
for (const auto &commentSymbol : commentSymbols)
|
2021-02-16 12:15:47 +00:00
|
|
|
text = removeLineComments(text, commentSymbol);
|
2020-11-26 11:09:58 +00:00
|
|
|
return text;
|
|
|
|
}
|
2020-12-13 09:00:00 +00:00
|
|
|
|
|
|
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 15, 0))
|
2021-04-16 14:04:38 +01:00
|
|
|
|
2020-12-13 09:00:00 +00:00
|
|
|
#include <QProcess>
|
|
|
|
|
|
|
|
QStringList ParseUtil::splitShellCommand(QStringView command) {
|
|
|
|
return QProcess::splitCommand(command);
|
|
|
|
}
|
2021-04-16 14:04:38 +01:00
|
|
|
|
2020-12-13 09:00:00 +00:00
|
|
|
#else
|
2021-04-16 14:04:38 +01:00
|
|
|
|
2020-12-13 09:00:00 +00:00
|
|
|
// The source for QProcess::splitCommand() as of Qt 5.15.2
|
|
|
|
QStringList ParseUtil::splitShellCommand(QStringView command) {
|
|
|
|
QStringList args;
|
|
|
|
QString tmp;
|
|
|
|
int quoteCount = 0;
|
|
|
|
bool inQuote = false;
|
|
|
|
|
|
|
|
// handle quoting. tokens can be surrounded by double quotes
|
|
|
|
// "hello world". three consecutive double quotes represent
|
|
|
|
// the quote character itself.
|
|
|
|
for (int i = 0; i < command.size(); ++i) {
|
|
|
|
if (command.at(i) == QLatin1Char('"')) {
|
|
|
|
++quoteCount;
|
|
|
|
if (quoteCount == 3) {
|
|
|
|
// third consecutive quote
|
|
|
|
quoteCount = 0;
|
|
|
|
tmp += command.at(i);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (quoteCount) {
|
|
|
|
if (quoteCount == 1)
|
|
|
|
inQuote = !inQuote;
|
|
|
|
quoteCount = 0;
|
|
|
|
}
|
|
|
|
if (!inQuote && command.at(i).isSpace()) {
|
|
|
|
if (!tmp.isEmpty()) {
|
|
|
|
args += tmp;
|
|
|
|
tmp.clear();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
tmp += command.at(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!tmp.isEmpty())
|
|
|
|
args += tmp;
|
|
|
|
|
|
|
|
return args;
|
|
|
|
}
|
2021-04-16 14:04:38 +01:00
|
|
|
|
2020-12-13 09:00:00 +00:00
|
|
|
#endif
|