2019-03-15 16:51:23 +03:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** Copyright (C) 2019 Denis Shienkov <denis.shienkov@gmail.com>
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
|
**
|
|
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
|
|
|
**
|
|
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "iarewparser.h"
|
|
|
|
|
|
|
|
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
|
|
|
#include <projectexplorer/task.h>
|
|
|
|
|
|
|
|
|
|
#include <texteditor/fontsettings.h>
|
|
|
|
|
#include <texteditor/texteditorsettings.h>
|
|
|
|
|
|
|
|
|
|
#include <QRegularExpression>
|
|
|
|
|
|
|
|
|
|
using namespace ProjectExplorer;
|
2020-01-15 08:56:11 +01:00
|
|
|
using namespace Utils;
|
2019-03-15 16:51:23 +03:00
|
|
|
|
|
|
|
|
namespace BareMetal {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
2019-05-08 13:49:19 +03:00
|
|
|
// Helpers:
|
|
|
|
|
|
2019-03-15 16:51:23 +03:00
|
|
|
static Task::TaskType taskType(const QString &msgType)
|
|
|
|
|
{
|
|
|
|
|
if (msgType == "Warning")
|
|
|
|
|
return Task::TaskType::Warning;
|
|
|
|
|
else if (msgType == "Error" || msgType == "Fatal error")
|
|
|
|
|
return Task::TaskType::Error;
|
|
|
|
|
return Task::TaskType::Unknown;
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-08 13:49:19 +03:00
|
|
|
// IarParser
|
|
|
|
|
|
2019-03-15 16:51:23 +03:00
|
|
|
IarParser::IarParser()
|
|
|
|
|
{
|
|
|
|
|
setObjectName("IarParser");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Core::Id IarParser::id()
|
|
|
|
|
{
|
|
|
|
|
return "BareMetal.OutputParser.Iar";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IarParser::newTask(const Task &task)
|
|
|
|
|
{
|
|
|
|
|
doFlush();
|
|
|
|
|
m_lastTask = task;
|
|
|
|
|
m_lines = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IarParser::amendDescription()
|
|
|
|
|
{
|
|
|
|
|
while (!m_descriptionParts.isEmpty())
|
|
|
|
|
m_lastTask.description.append(m_descriptionParts.takeFirst());
|
|
|
|
|
|
|
|
|
|
while (!m_snippets.isEmpty()) {
|
|
|
|
|
const QString snippet = m_snippets.takeFirst();
|
|
|
|
|
const int start = m_lastTask.description.count() + 1;
|
2019-11-04 14:23:21 +03:00
|
|
|
m_lastTask.description.append('\n');
|
2019-03-15 16:51:23 +03:00
|
|
|
m_lastTask.description.append(snippet);
|
|
|
|
|
|
|
|
|
|
QTextLayout::FormatRange fr;
|
|
|
|
|
fr.start = start;
|
|
|
|
|
fr.length = m_lastTask.description.count() + 1;
|
|
|
|
|
fr.format.setFont(TextEditor::TextEditorSettings::fontSettings().font());
|
|
|
|
|
fr.format.setFontStyleHint(QFont::Monospace);
|
|
|
|
|
m_lastTask.formats.append(fr);
|
|
|
|
|
|
|
|
|
|
++m_lines;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IarParser::amendFilePath()
|
|
|
|
|
{
|
|
|
|
|
if (m_filePathParts.isEmpty())
|
|
|
|
|
return;
|
|
|
|
|
QString filePath;
|
|
|
|
|
while (!m_filePathParts.isEmpty())
|
|
|
|
|
filePath.append(m_filePathParts.takeFirst().trimmed());
|
2019-05-28 13:49:26 +02:00
|
|
|
m_lastTask.setFile(Utils::FilePath::fromUserInput(filePath));
|
2019-03-15 16:51:23 +03:00
|
|
|
m_expectFilePath = false;
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-25 12:00:39 +03:00
|
|
|
bool IarParser::parseErrorOrFatalErrorDetailsMessage1(const QString &lne)
|
|
|
|
|
{
|
|
|
|
|
const QRegularExpression re("^(Error|Fatal error)\\[(.+)\\]:\\s(.+)\\s\\[(.+)$");
|
|
|
|
|
const QRegularExpressionMatch match = re.match(lne);
|
|
|
|
|
if (!match.hasMatch())
|
|
|
|
|
return false;
|
|
|
|
|
enum CaptureIndex { MessageTypeIndex = 1, MessageCodeIndex,
|
|
|
|
|
DescriptionIndex, FilepathBeginIndex };
|
|
|
|
|
const Task::TaskType type = taskType(match.captured(MessageTypeIndex));
|
|
|
|
|
const QString descr = QString("[%1]: %2").arg(match.captured(MessageCodeIndex),
|
|
|
|
|
match.captured(DescriptionIndex));
|
|
|
|
|
// This task has a file path, but this patch are split on
|
|
|
|
|
// some lines, which will be received later.
|
2020-01-15 08:56:11 +01:00
|
|
|
newTask(CompileTask(type, descr));
|
2019-10-25 12:00:39 +03:00
|
|
|
// Prepare first part of a file path.
|
|
|
|
|
QString firstPart = match.captured(FilepathBeginIndex);
|
|
|
|
|
firstPart.remove("referenced from ");
|
|
|
|
|
m_filePathParts.push_back(firstPart);
|
|
|
|
|
m_expectFilePath = true;
|
|
|
|
|
m_expectSnippet = false;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool IarParser::parseErrorOrFatalErrorDetailsMessage2(const QString &lne)
|
|
|
|
|
{
|
|
|
|
|
const QRegularExpression re("^.*(Error|Fatal error)\\[(.+)\\]:\\s(.+)$");
|
|
|
|
|
const QRegularExpressionMatch match = re.match(lne);
|
|
|
|
|
if (!match.hasMatch())
|
|
|
|
|
return false;
|
|
|
|
|
enum CaptureIndex { MessageTypeIndex = 1, MessageCodeIndex,
|
|
|
|
|
DescriptionIndex };
|
|
|
|
|
const Task::TaskType type = taskType(match.captured(MessageTypeIndex));
|
|
|
|
|
const QString descr = QString("[%1]: %2").arg(match.captured(MessageCodeIndex),
|
|
|
|
|
match.captured(DescriptionIndex));
|
|
|
|
|
// This task has not a file path. The description details
|
|
|
|
|
// will be received later on the next lines.
|
2020-01-15 08:56:11 +01:00
|
|
|
newTask(CompileTask(type, descr));
|
2019-10-25 12:00:39 +03:00
|
|
|
m_expectSnippet = true;
|
|
|
|
|
m_expectFilePath = false;
|
|
|
|
|
m_expectDescription = false;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool IarParser::parseWarningOrErrorOrFatalErrorDetailsMessage1(const QString &lne)
|
|
|
|
|
{
|
|
|
|
|
const QRegularExpression re("^\"(.+)\",(\\d+)?\\s+(Warning|Error|Fatal error)\\[(.+)\\].+$");
|
|
|
|
|
const QRegularExpressionMatch match = re.match(lne);
|
|
|
|
|
if (!match.hasMatch())
|
|
|
|
|
return false;
|
|
|
|
|
enum CaptureIndex { FilePathIndex = 1, LineNumberIndex,
|
|
|
|
|
MessageTypeIndex, MessageCodeIndex };
|
|
|
|
|
const Utils::FilePath fileName = Utils::FilePath::fromUserInput(
|
|
|
|
|
match.captured(FilePathIndex));
|
|
|
|
|
const int lineno = match.captured(LineNumberIndex).toInt();
|
|
|
|
|
const Task::TaskType type = taskType(match.captured(MessageTypeIndex));
|
|
|
|
|
// A full description will be received later on next lines.
|
2020-04-07 13:49:34 +02:00
|
|
|
newTask(CompileTask(type, {}, absoluteFilePath(fileName), lineno));
|
2019-10-25 12:00:39 +03:00
|
|
|
const QString firstPart = QString("[%1]: ").arg(match.captured(MessageCodeIndex));
|
|
|
|
|
m_descriptionParts.append(firstPart);
|
|
|
|
|
m_expectDescription = true;
|
|
|
|
|
m_expectSnippet = false;
|
|
|
|
|
m_expectFilePath = false;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool IarParser::parseErrorInCommandLineMessage(const QString &lne)
|
|
|
|
|
{
|
|
|
|
|
if (!lne.startsWith("Error in command line"))
|
|
|
|
|
return false;
|
2020-01-15 08:56:11 +01:00
|
|
|
newTask(CompileTask(Task::TaskType::Error, lne.trimmed()));
|
2019-10-25 12:00:39 +03:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-25 12:20:38 +03:00
|
|
|
bool IarParser::parseErrorMessage1(const QString &lne)
|
|
|
|
|
{
|
|
|
|
|
const QRegularExpression re("^(Error)\\[(.+)\\]:\\s(.+)$");
|
|
|
|
|
const QRegularExpressionMatch match = re.match(lne);
|
|
|
|
|
if (!match.hasMatch())
|
|
|
|
|
return false;
|
|
|
|
|
enum CaptureIndex { MessageTypeIndex = 1, MessageCodeIndex, DescriptionIndex };
|
|
|
|
|
const Task::TaskType type = taskType(match.captured(MessageTypeIndex));
|
|
|
|
|
const QString descr = QString("[%1]: %2").arg(match.captured(MessageCodeIndex),
|
|
|
|
|
match.captured(DescriptionIndex));
|
|
|
|
|
// This task has not a file path and line number (as it is a linker message)
|
2020-01-15 08:56:11 +01:00
|
|
|
newTask(CompileTask(type, descr));
|
2019-10-25 12:20:38 +03:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-08 17:45:39 +02:00
|
|
|
IOutputParser::Status IarParser::doHandleLine(const QString &line, OutputFormat type)
|
2020-04-08 09:42:51 +02:00
|
|
|
{
|
2019-03-15 16:51:23 +03:00
|
|
|
const QString lne = rightTrimmed(line);
|
2020-04-08 17:45:39 +02:00
|
|
|
if (type == StdOutFormat) {
|
|
|
|
|
// The call sequence has the meaning!
|
|
|
|
|
const bool leastOneParsed = parseErrorInCommandLineMessage(lne)
|
|
|
|
|
|| parseErrorMessage1(lne);
|
|
|
|
|
if (!leastOneParsed) {
|
|
|
|
|
doFlush();
|
|
|
|
|
return Status::NotHandled;
|
|
|
|
|
}
|
|
|
|
|
return Status::InProgress;
|
|
|
|
|
}
|
2019-03-15 16:51:23 +03:00
|
|
|
|
2019-10-25 12:00:39 +03:00
|
|
|
if (parseErrorOrFatalErrorDetailsMessage1(lne))
|
2020-04-08 17:45:39 +02:00
|
|
|
return Status::InProgress;
|
2019-10-25 12:00:39 +03:00
|
|
|
if (parseErrorOrFatalErrorDetailsMessage2(lne))
|
2020-04-08 17:45:39 +02:00
|
|
|
return Status::InProgress;
|
2019-10-25 12:00:39 +03:00
|
|
|
if (parseWarningOrErrorOrFatalErrorDetailsMessage1(lne))
|
2020-04-08 17:45:39 +02:00
|
|
|
return Status::InProgress;
|
2019-03-15 16:51:23 +03:00
|
|
|
|
2020-04-08 17:45:39 +02:00
|
|
|
if (m_expectFilePath) {
|
2019-11-04 14:23:21 +03:00
|
|
|
if (lne.endsWith(']')) {
|
2019-03-15 16:51:23 +03:00
|
|
|
const QString lastPart = lne.left(lne.size() - 1);
|
|
|
|
|
m_filePathParts.push_back(lastPart);
|
2020-04-08 17:45:39 +02:00
|
|
|
doFlush();
|
|
|
|
|
return Status::Done;
|
2019-03-15 16:51:23 +03:00
|
|
|
} else {
|
|
|
|
|
m_filePathParts.push_back(lne);
|
2020-04-08 17:45:39 +02:00
|
|
|
return Status::InProgress;
|
2019-03-15 16:51:23 +03:00
|
|
|
}
|
2020-04-08 17:45:39 +02:00
|
|
|
}
|
|
|
|
|
if (m_expectSnippet && lne.startsWith(' ')) {
|
2019-03-15 16:51:23 +03:00
|
|
|
if (!lne.endsWith("Fatal error detected, aborting.")) {
|
|
|
|
|
m_snippets.push_back(lne);
|
2020-04-08 17:45:39 +02:00
|
|
|
return Status::InProgress;
|
2019-03-15 16:51:23 +03:00
|
|
|
}
|
|
|
|
|
} else if (m_expectDescription) {
|
|
|
|
|
if (!lne.startsWith(" ")) {
|
|
|
|
|
m_descriptionParts.push_back(lne.trimmed());
|
2020-04-08 17:45:39 +02:00
|
|
|
return Status::InProgress;
|
2019-03-15 16:51:23 +03:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-08 17:45:39 +02:00
|
|
|
if (!m_lastTask.isNull()) {
|
|
|
|
|
doFlush();
|
|
|
|
|
return Status::Done;
|
|
|
|
|
}
|
2019-03-15 16:51:23 +03:00
|
|
|
|
2020-04-08 17:45:39 +02:00
|
|
|
return Status::NotHandled;
|
2019-03-15 16:51:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void IarParser::doFlush()
|
|
|
|
|
{
|
|
|
|
|
if (m_lastTask.isNull())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
amendDescription();
|
|
|
|
|
amendFilePath();
|
|
|
|
|
|
|
|
|
|
m_expectSnippet = true;
|
|
|
|
|
m_expectFilePath = false;
|
|
|
|
|
m_expectDescription = false;
|
|
|
|
|
|
|
|
|
|
Task t = m_lastTask;
|
|
|
|
|
m_lastTask.clear();
|
|
|
|
|
emit addTask(t, m_lines, 1);
|
|
|
|
|
m_lines = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace BareMetal
|
|
|
|
|
|
|
|
|
|
// Unit tests:
|
|
|
|
|
|
|
|
|
|
#ifdef WITH_TESTS
|
|
|
|
|
#include "baremetalplugin.h"
|
|
|
|
|
#include <projectexplorer/outputparser_test.h>
|
|
|
|
|
#include <QTest>
|
|
|
|
|
|
|
|
|
|
namespace BareMetal {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
void BareMetalPlugin::testIarOutputParsers_data()
|
|
|
|
|
{
|
|
|
|
|
QTest::addColumn<QString>("input");
|
|
|
|
|
QTest::addColumn<OutputParserTester::Channel>("inputChannel");
|
|
|
|
|
QTest::addColumn<QString>("childStdOutLines");
|
|
|
|
|
QTest::addColumn<QString>("childStdErrLines");
|
2019-05-27 16:09:44 +02:00
|
|
|
QTest::addColumn<Tasks >("tasks");
|
2019-03-15 16:51:23 +03:00
|
|
|
QTest::addColumn<QString>("outputLines");
|
|
|
|
|
|
|
|
|
|
QTest::newRow("pass-through stdout")
|
|
|
|
|
<< "Sometext" << OutputParserTester::STDOUT
|
|
|
|
|
<< "Sometext\n" << QString()
|
2019-05-27 16:09:44 +02:00
|
|
|
<< Tasks()
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
QTest::newRow("pass-through stderr")
|
|
|
|
|
<< "Sometext" << OutputParserTester::STDERR
|
|
|
|
|
<< QString() << "Sometext\n"
|
2019-05-27 16:09:44 +02:00
|
|
|
<< Tasks()
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
|
|
|
|
// For std out.
|
|
|
|
|
QTest::newRow("Error in command line")
|
|
|
|
|
<< QString::fromLatin1("Error in command line: Some error")
|
|
|
|
|
<< OutputParserTester::STDOUT
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Error,
|
|
|
|
|
"Error in command line: Some error"))
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
2019-10-25 12:20:38 +03:00
|
|
|
QTest::newRow("Linker error")
|
|
|
|
|
<< QString::fromLatin1("Error[e46]: Some error")
|
|
|
|
|
<< OutputParserTester::STDOUT
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2019-10-25 12:20:38 +03:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Error,
|
|
|
|
|
"[e46]: Some error"))
|
2019-10-25 12:20:38 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
2019-03-15 16:51:23 +03:00
|
|
|
// For std error.
|
|
|
|
|
QTest::newRow("No details warning")
|
|
|
|
|
<< QString::fromLatin1("\"c:\\foo\\main.c\",63 Warning[Pe223]:\n"
|
|
|
|
|
" Some warning \"foo\" bar")
|
|
|
|
|
<< OutputParserTester::STDERR
|
|
|
|
|
<< QString()
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Warning,
|
|
|
|
|
"[Pe223]: Some warning \"foo\" bar",
|
|
|
|
|
Utils::FilePath::fromUserInput("c:\\foo\\main.c"),
|
|
|
|
|
63))
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
|
|
|
|
QTest::newRow("Details warning")
|
|
|
|
|
<< QString::fromLatin1(" some_detail;\n"
|
|
|
|
|
" ^\n"
|
|
|
|
|
"\"c:\\foo\\main.c\",63 Warning[Pe223]:\n"
|
|
|
|
|
" Some warning")
|
|
|
|
|
<< OutputParserTester::STDERR
|
|
|
|
|
<< QString()
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Warning,
|
|
|
|
|
"[Pe223]: Some warning\n"
|
|
|
|
|
" some_detail;\n"
|
|
|
|
|
" ^",
|
|
|
|
|
FilePath::fromUserInput("c:\\foo\\main.c"),
|
|
|
|
|
63))
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
|
|
|
|
QTest::newRow("No details split-description warning")
|
|
|
|
|
<< QString::fromLatin1("\"c:\\foo\\main.c\",63 Warning[Pe223]:\n"
|
|
|
|
|
" Some warning\n"
|
|
|
|
|
" , split")
|
|
|
|
|
<< OutputParserTester::STDERR
|
|
|
|
|
<< QString()
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Warning,
|
|
|
|
|
"[Pe223]: Some warning, split",
|
|
|
|
|
FilePath::fromUserInput("c:\\foo\\main.c"),
|
|
|
|
|
63))
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
|
|
|
|
QTest::newRow("No details error")
|
|
|
|
|
<< QString::fromLatin1("\"c:\\foo\\main.c\",63 Error[Pe223]:\n"
|
|
|
|
|
" Some error")
|
|
|
|
|
<< OutputParserTester::STDERR
|
|
|
|
|
<< QString()
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Error,
|
|
|
|
|
"[Pe223]: Some error",
|
|
|
|
|
FilePath::fromUserInput("c:\\foo\\main.c"),
|
|
|
|
|
63))
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
|
|
|
|
QTest::newRow("Details error")
|
|
|
|
|
<< QString::fromLatin1(" some_detail;\n"
|
|
|
|
|
" ^\n"
|
|
|
|
|
"\"c:\\foo\\main.c\",63 Error[Pe223]:\n"
|
|
|
|
|
" Some error")
|
|
|
|
|
<< OutputParserTester::STDERR
|
|
|
|
|
<< QString()
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Error,
|
|
|
|
|
"[Pe223]: Some error\n"
|
|
|
|
|
" some_detail;\n"
|
|
|
|
|
" ^",
|
|
|
|
|
FilePath::fromUserInput("c:\\foo\\main.c"),
|
|
|
|
|
63))
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
|
|
|
|
QTest::newRow("No details split-description error")
|
|
|
|
|
<< QString::fromLatin1("\"c:\\foo\\main.c\",63 Error[Pe223]:\n"
|
|
|
|
|
" Some error\n"
|
|
|
|
|
" , split")
|
|
|
|
|
<< OutputParserTester::STDERR
|
|
|
|
|
<< QString()
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Error,
|
|
|
|
|
"[Pe223]: Some error, split",
|
|
|
|
|
FilePath::fromUserInput("c:\\foo\\main.c"),
|
|
|
|
|
63))
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
|
|
|
|
QTest::newRow("No definition for")
|
|
|
|
|
<< QString::fromLatin1("Error[Li005]: Some error \"foo\" [referenced from c:\\fo\n"
|
|
|
|
|
" o\\bar\\mai\n"
|
|
|
|
|
" n.c.o\n"
|
|
|
|
|
"]")
|
|
|
|
|
<< OutputParserTester::STDERR
|
|
|
|
|
<< QString()
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Error,
|
|
|
|
|
"[Li005]: Some error \"foo\"",
|
|
|
|
|
FilePath::fromUserInput("c:\\foo\\bar\\main.c.o")))
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
|
|
|
|
QTest::newRow("More than one source file specified")
|
|
|
|
|
<< QString::fromLatin1("Fatal error[Su011]: Some error:\n"
|
|
|
|
|
" c:\\foo.c\n"
|
|
|
|
|
" c:\\bar.c\n"
|
|
|
|
|
"Fatal error detected, aborting.")
|
|
|
|
|
<< OutputParserTester::STDERR
|
|
|
|
|
<< QString()
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Error,
|
|
|
|
|
"[Su011]: Some error:\n"
|
|
|
|
|
" c:\\foo.c\n"
|
|
|
|
|
" c:\\bar.c"))
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
|
|
|
|
|
QTest::newRow("At end of source")
|
|
|
|
|
<< QString::fromLatin1("At end of source Error[Pe040]: Some error \";\"")
|
|
|
|
|
<< OutputParserTester::STDERR
|
|
|
|
|
<< QString()
|
2020-04-08 17:45:39 +02:00
|
|
|
<< QString()
|
2020-01-15 08:56:11 +01:00
|
|
|
<< (Tasks() << CompileTask(Task::Error,
|
|
|
|
|
"[Pe040]: Some error \";\""))
|
2019-03-15 16:51:23 +03:00
|
|
|
<< QString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void BareMetalPlugin::testIarOutputParsers()
|
|
|
|
|
{
|
|
|
|
|
OutputParserTester testbench;
|
2020-04-08 17:45:39 +02:00
|
|
|
testbench.addLineParser(new IarParser);
|
2019-03-15 16:51:23 +03:00
|
|
|
QFETCH(QString, input);
|
|
|
|
|
QFETCH(OutputParserTester::Channel, inputChannel);
|
2019-05-27 16:09:44 +02:00
|
|
|
QFETCH(Tasks, tasks);
|
2019-03-15 16:51:23 +03:00
|
|
|
QFETCH(QString, childStdOutLines);
|
|
|
|
|
QFETCH(QString, childStdErrLines);
|
|
|
|
|
QFETCH(QString, outputLines);
|
|
|
|
|
|
|
|
|
|
testbench.testParsing(input, inputChannel,
|
|
|
|
|
tasks, childStdOutLines, childStdErrLines,
|
|
|
|
|
outputLines);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace BareMetal
|
|
|
|
|
|
|
|
|
|
#endif // WITH_TESTS
|