2021-12-26 23:18:28 +08:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2020-2022 Roy Qu (royqh1979@gmail.com)
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2021-04-20 22:24:33 +08:00
|
|
|
#include "compiler.h"
|
|
|
|
#include "utils.h"
|
2024-02-28 19:41:05 +08:00
|
|
|
#include "utils/escape.h"
|
|
|
|
#include "utils/parsearg.h"
|
2021-06-21 11:21:26 +08:00
|
|
|
#include "compilermanager.h"
|
2021-09-14 12:10:43 +08:00
|
|
|
#include "../systemconsts.h"
|
2021-04-20 22:24:33 +08:00
|
|
|
|
2023-03-13 21:57:15 +08:00
|
|
|
#include <cmath>
|
2021-04-20 22:24:33 +08:00
|
|
|
#include <QFileInfo>
|
|
|
|
#include <QProcess>
|
|
|
|
#include <QString>
|
|
|
|
#include <QTextCodec>
|
|
|
|
#include <QTime>
|
2021-09-04 11:37:04 +08:00
|
|
|
#include <QApplication>
|
|
|
|
#include "../editor.h"
|
|
|
|
#include "../mainwindow.h"
|
|
|
|
#include "../editorlist.h"
|
|
|
|
#include "../parser/cppparser.h"
|
|
|
|
#include "../autolinkmanager.h"
|
2022-09-26 12:01:45 +08:00
|
|
|
#include "qt_utils/charsetinfo.h"
|
2021-09-12 22:45:00 +08:00
|
|
|
#include "../project.h"
|
2021-04-20 22:24:33 +08:00
|
|
|
|
2021-06-24 20:43:09 +08:00
|
|
|
#define COMPILE_PROCESS_END "---//END//----"
|
|
|
|
|
2023-08-15 10:57:03 +08:00
|
|
|
Compiler::Compiler(const QString &filename, bool onlyCheckSyntax):
|
2024-02-20 21:47:12 +08:00
|
|
|
QThread{},
|
|
|
|
mOnlyCheckSyntax{onlyCheckSyntax},
|
|
|
|
mFilename{filename},
|
|
|
|
mRebuild{false},
|
2024-04-03 16:50:19 +08:00
|
|
|
mParserForFile{},
|
|
|
|
mForceEnglishOutput{false}
|
2021-04-20 22:24:33 +08:00
|
|
|
{
|
2023-08-13 14:46:53 +08:00
|
|
|
getParserForFile(filename);
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Compiler::run()
|
|
|
|
{
|
|
|
|
emit compileStarted();
|
2021-09-13 07:49:36 +08:00
|
|
|
auto action = finally([this]{
|
2022-12-25 12:00:09 +08:00
|
|
|
emit compileFinished(mFilename);
|
2021-09-13 07:49:36 +08:00
|
|
|
});
|
2021-06-21 11:21:26 +08:00
|
|
|
try {
|
2021-09-13 07:49:36 +08:00
|
|
|
if (!prepareForCompile()){
|
|
|
|
return;
|
|
|
|
}
|
2021-06-25 12:40:11 +08:00
|
|
|
if (mRebuild && !prepareForRebuild()) {
|
|
|
|
throw CompileError(tr("Clean before rebuild failed."));
|
|
|
|
}
|
2021-09-13 07:49:36 +08:00
|
|
|
mErrorCount = 0;
|
|
|
|
mWarningCount = 0;
|
|
|
|
QElapsedTimer timer;
|
|
|
|
timer.start();
|
2021-09-13 22:45:50 +08:00
|
|
|
runCommand(mCompiler, mArguments, mDirectory, pipedText());
|
2023-03-02 18:15:31 +08:00
|
|
|
for(int i=0;i<mExtraArgumentsList.count();i++) {
|
2023-08-14 12:22:24 +08:00
|
|
|
if (!beforeRunExtraCommand(i))
|
|
|
|
break;
|
2024-02-28 19:41:05 +08:00
|
|
|
QString command = escapeCommandForLog(mExtraCompilersList[i], mExtraArgumentsList[i]);
|
2023-08-13 18:53:48 +08:00
|
|
|
if (mExtraOutputFilesList[i].isEmpty()) {
|
2024-02-28 19:41:05 +08:00
|
|
|
log(tr(" - Command: %1").arg(command));
|
2023-08-13 18:53:48 +08:00
|
|
|
} else {
|
2024-02-28 19:41:05 +08:00
|
|
|
log(tr(" - Command: %1 > %2").arg(command, escapeArgumentForPlatformShell(mExtraOutputFilesList[i], false)));
|
2023-08-13 18:53:48 +08:00
|
|
|
}
|
|
|
|
runCommand(mExtraCompilersList[i],mExtraArgumentsList[i],mDirectory, pipedText(),mExtraOutputFilesList[i]);
|
2023-03-02 18:15:31 +08:00
|
|
|
}
|
2021-09-13 07:49:36 +08:00
|
|
|
log("");
|
|
|
|
log(tr("Compile Result:"));
|
|
|
|
log("------------------");
|
|
|
|
log(tr("- Errors: %1").arg(mErrorCount));
|
|
|
|
log(tr("- Warnings: %1").arg(mWarningCount));
|
|
|
|
if (!mOutputFile.isEmpty()) {
|
|
|
|
log(tr("- Output Filename: %1").arg(mOutputFile));
|
|
|
|
QLocale locale = QLocale::system();
|
|
|
|
log(tr("- Output Size: %1").arg(locale.formattedDataSize(QFileInfo(mOutputFile).size())));
|
2021-04-21 23:06:55 +08:00
|
|
|
}
|
2021-09-13 07:49:36 +08:00
|
|
|
log(tr("- Compilation Time: %1 secs").arg(timer.elapsed() / 1000.0));
|
2021-06-21 11:21:26 +08:00
|
|
|
} catch (CompileError e) {
|
|
|
|
emit compileErrorOccured(e.reason());
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
2021-09-13 07:49:36 +08:00
|
|
|
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
|
2021-04-24 15:57:45 +08:00
|
|
|
QString Compiler::getFileNameFromOutputLine(QString &line) {
|
|
|
|
QString temp;
|
|
|
|
line = line.trimmed();
|
|
|
|
while (true) {
|
|
|
|
int pos;
|
|
|
|
if (line.length() > 2 && line[1]==':') { // full file path at start, ignore this ':'
|
|
|
|
pos = line.indexOf(':',2);
|
|
|
|
} else {
|
|
|
|
pos = line.indexOf(':');
|
|
|
|
}
|
|
|
|
if ( pos < 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
temp = line.mid(0,pos);
|
|
|
|
line.remove(0,pos+1);
|
2021-05-27 01:05:49 +08:00
|
|
|
line=line.trimmed();
|
2021-06-24 20:43:09 +08:00
|
|
|
if (temp.compare("<stdin>", Qt::CaseInsensitive)==0 ) {
|
|
|
|
temp = mFilename;
|
2022-10-26 08:12:00 +08:00
|
|
|
return temp;
|
2023-02-27 08:53:39 +08:00
|
|
|
} else if (temp.compare("{standard input}", Qt::CaseInsensitive)==0 ) {
|
|
|
|
temp = mFilename;
|
|
|
|
return temp;
|
2021-06-24 20:43:09 +08:00
|
|
|
}
|
2021-04-24 15:57:45 +08:00
|
|
|
|
2022-10-23 10:40:00 +08:00
|
|
|
QFileInfo fileInfo(temp);
|
|
|
|
if (fileInfo.fileName() == QLatin1String("ld.exe")) { // skip ld.exe
|
|
|
|
continue;
|
|
|
|
} else if (fileInfo.fileName() == QLatin1String("make")) { // skip make.exe
|
|
|
|
continue;
|
|
|
|
} else if (fileInfo.fileName() == QLatin1String("mingw32-make")) { // skip mingw32-make.exe
|
|
|
|
continue;
|
|
|
|
} else if (fileInfo.suffix()=="o") { // skip obj file
|
2021-04-24 15:57:45 +08:00
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2022-10-26 08:12:00 +08:00
|
|
|
if (!mDirectory.isEmpty()) {
|
|
|
|
QFileInfo info(temp);
|
2022-11-16 09:24:42 +08:00
|
|
|
return info.isRelative()?generateAbsolutePath(mDirectory,temp):cleanPath(temp);
|
2022-10-26 08:12:00 +08:00
|
|
|
}
|
|
|
|
return temp;
|
2021-04-24 15:57:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int Compiler::getLineNumberFromOutputLine(QString &line)
|
|
|
|
{
|
|
|
|
line = line.trimmed();
|
|
|
|
int pos = line.indexOf(':');
|
|
|
|
int result=0;
|
|
|
|
if (pos < 0) {
|
|
|
|
pos = line.indexOf(',');
|
|
|
|
}
|
|
|
|
if (pos>=0) {
|
2024-05-14 20:26:32 +08:00
|
|
|
result = QStringView(line.constData(), pos).toInt();
|
2021-05-27 01:05:49 +08:00
|
|
|
if (result > 0)
|
|
|
|
line.remove(0,pos+1);
|
2022-04-27 19:13:38 +08:00
|
|
|
} else {
|
|
|
|
result = line.toInt();
|
|
|
|
if (result > 0)
|
|
|
|
line="";
|
2021-04-24 15:57:45 +08:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Compiler::getColunmnFromOutputLine(QString &line)
|
|
|
|
{
|
|
|
|
line = line.trimmed();
|
|
|
|
int pos = line.indexOf(':');
|
|
|
|
int result=0;
|
|
|
|
if (pos < 0) {
|
|
|
|
pos = line.indexOf(',');
|
|
|
|
}
|
|
|
|
if (pos>=0) {
|
2024-05-14 20:26:32 +08:00
|
|
|
result = QStringView(line.constData(), pos).toInt();
|
2021-06-24 16:05:19 +08:00
|
|
|
if (result > 0)
|
|
|
|
line.remove(0,pos+1);
|
2021-04-24 15:57:45 +08:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
CompileIssueType Compiler::getIssueTypeFromOutputLine(QString &line)
|
|
|
|
{
|
|
|
|
CompileIssueType result = CompileIssueType::Other;
|
|
|
|
line = line.trimmed();
|
2024-03-07 20:33:16 +08:00
|
|
|
if (line.startsWith(tr("error:"))) {
|
|
|
|
mErrorCount += 1;
|
|
|
|
line = tr("[Error] ")+line.mid(tr("error:").length());
|
|
|
|
result = CompileIssueType::Error;
|
|
|
|
} else if (line.startsWith(tr("warning:"))) {
|
|
|
|
mWarningCount += 1;
|
|
|
|
line = tr("[Warning] ")+line.mid(tr("warning:").length());
|
|
|
|
result = CompileIssueType::Warning;
|
|
|
|
} else {
|
|
|
|
int pos = line.indexOf(':');
|
|
|
|
if (pos>=0) {
|
|
|
|
QString s=line.mid(0,pos);
|
|
|
|
if (s == "error" || s == "fatal error"
|
|
|
|
|| s == "syntax error") {
|
|
|
|
mErrorCount += 1;
|
|
|
|
line = tr("[Error] ")+line.mid(pos+1);
|
|
|
|
result = CompileIssueType::Error;
|
|
|
|
} else if (s.startsWith("warning")
|
|
|
|
|| s.startsWith(tr("warning"))) {
|
|
|
|
mWarningCount += 1;
|
|
|
|
line = tr("[Warning] ")+line.mid(pos+1);
|
|
|
|
result = CompileIssueType::Warning;
|
|
|
|
} else if (s == "info"
|
|
|
|
|| s == tr("info")) {
|
|
|
|
mWarningCount += 1;
|
|
|
|
line = tr("[Info] ")+line.mid(pos+1);
|
|
|
|
result = CompileIssueType::Info;
|
|
|
|
} else if (s == "note"
|
|
|
|
|| s == tr("note")) {
|
|
|
|
mWarningCount += 1;
|
|
|
|
line = tr("[Note] ")+line.mid(pos+1);
|
|
|
|
result = CompileIssueType::Note;
|
|
|
|
}
|
2021-04-24 15:57:45 +08:00
|
|
|
}
|
|
|
|
}
|
2024-03-07 20:33:16 +08:00
|
|
|
|
2021-04-24 15:57:45 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-09-13 07:49:36 +08:00
|
|
|
Settings::PCompilerSet Compiler::compilerSet()
|
|
|
|
{
|
2022-01-10 10:53:16 +08:00
|
|
|
if (mProject) {
|
|
|
|
int index = mProject->options().compilerSet;
|
|
|
|
Settings::PCompilerSet set = pSettings->compilerSets().getSet(index);
|
|
|
|
if (set)
|
|
|
|
return set;
|
|
|
|
}
|
2021-09-13 07:49:36 +08:00
|
|
|
return pSettings->compilerSets().defaultSet();
|
|
|
|
}
|
|
|
|
|
2022-03-31 09:56:49 +08:00
|
|
|
QByteArray Compiler::pipedText()
|
|
|
|
{
|
|
|
|
return QByteArray();
|
|
|
|
}
|
|
|
|
|
2023-10-22 13:19:03 +08:00
|
|
|
bool Compiler::beforeRunExtraCommand(int /* idx */)
|
2023-08-14 12:22:24 +08:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-24 15:57:45 +08:00
|
|
|
void Compiler::processOutput(QString &line)
|
|
|
|
{
|
2021-06-24 20:43:09 +08:00
|
|
|
if (line == COMPILE_PROCESS_END) {
|
|
|
|
if (mLastIssue) {
|
|
|
|
emit compileIssue(mLastIssue);
|
|
|
|
mLastIssue.reset();
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2022-04-27 19:13:38 +08:00
|
|
|
if (line.startsWith(">>>"))
|
|
|
|
line.remove(0,3);
|
|
|
|
QString referencePrefix = QString(" referenced by ");
|
|
|
|
if(mLastIssue && line.startsWith(referencePrefix)) {
|
|
|
|
line.remove(0,referencePrefix.length());
|
|
|
|
mLastIssue->filename = getFileNameFromOutputLine(line);
|
2022-10-10 07:11:59 +08:00
|
|
|
//qDebug()<<line;
|
2022-04-27 19:13:38 +08:00
|
|
|
mLastIssue->line = getLineNumberFromOutputLine(line);
|
|
|
|
emit compileIssue(mLastIssue);
|
|
|
|
mLastIssue.reset();
|
|
|
|
return;
|
|
|
|
}
|
2021-04-24 15:57:45 +08:00
|
|
|
QString inFilePrefix = QString("In file included from ");
|
|
|
|
QString fromPrefix = QString("from ");
|
|
|
|
PCompileIssue issue = std::make_shared<CompileIssue>();
|
|
|
|
issue->type = CompileIssueType::Other;
|
2021-06-24 16:05:19 +08:00
|
|
|
issue->endColumn = -1;
|
2021-04-24 15:57:45 +08:00
|
|
|
if (line.startsWith(inFilePrefix)) {
|
|
|
|
line.remove(0,inFilePrefix.length());
|
|
|
|
issue->filename = getFileNameFromOutputLine(line);
|
|
|
|
issue->line = getLineNumberFromOutputLine(line);
|
2021-05-27 01:05:49 +08:00
|
|
|
if (issue->line > 0)
|
|
|
|
issue->column = getColunmnFromOutputLine(line);
|
2021-04-24 15:57:45 +08:00
|
|
|
issue->type = getIssueTypeFromOutputLine(line);
|
|
|
|
issue->description = inFilePrefix + issue->filename;
|
|
|
|
emit compileIssue(issue);
|
|
|
|
return;
|
|
|
|
} else if(line.startsWith(fromPrefix)) {
|
|
|
|
line.remove(0,fromPrefix.length());
|
|
|
|
issue->filename = getFileNameFromOutputLine(line);
|
|
|
|
issue->line = getLineNumberFromOutputLine(line);
|
2021-05-27 01:05:49 +08:00
|
|
|
if (issue->line > 0)
|
|
|
|
issue->column = getColunmnFromOutputLine(line);
|
2021-04-24 15:57:45 +08:00
|
|
|
issue->type = getIssueTypeFromOutputLine(line);
|
|
|
|
issue->description = " from " + issue->filename;
|
|
|
|
emit compileIssue(issue);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ignore code snippets that GCC produces
|
|
|
|
// they always start with a space
|
|
|
|
if (line.length()>0 && line[0] == ' ') {
|
2021-06-24 16:05:19 +08:00
|
|
|
if (!mLastIssue)
|
|
|
|
return;
|
|
|
|
QString s = line.trimmed();
|
|
|
|
if (s.startsWith('|') && s.indexOf('^')) {
|
|
|
|
int pos = 0;
|
|
|
|
while (pos < s.length()) {
|
|
|
|
if (s[pos]=='^')
|
|
|
|
break;
|
|
|
|
pos++;
|
|
|
|
}
|
|
|
|
if (pos<s.length()) {
|
|
|
|
int i=pos+1;
|
|
|
|
while (i<s.length()) {
|
|
|
|
if (s[i]!='~' && s[i]!='^')
|
|
|
|
break;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
mLastIssue->endColumn = mLastIssue->column+i-pos;
|
|
|
|
emit compileIssue(mLastIssue);
|
|
|
|
mLastIssue.reset();
|
|
|
|
}
|
|
|
|
}
|
2021-04-24 15:57:45 +08:00
|
|
|
return;
|
|
|
|
}
|
2021-06-24 16:05:19 +08:00
|
|
|
|
|
|
|
if (mLastIssue) {
|
|
|
|
emit compileIssue(mLastIssue);
|
|
|
|
mLastIssue.reset();
|
|
|
|
}
|
|
|
|
|
2021-04-24 15:57:45 +08:00
|
|
|
// assume regular main.cpp:line:col: message
|
|
|
|
issue->filename = getFileNameFromOutputLine(line);
|
|
|
|
issue->line = getLineNumberFromOutputLine(line);
|
2021-06-24 16:05:19 +08:00
|
|
|
if (issue->line > 0) {
|
2021-05-27 01:05:49 +08:00
|
|
|
issue->column = getColunmnFromOutputLine(line);
|
2021-06-24 22:33:57 +08:00
|
|
|
issue->type = getIssueTypeFromOutputLine(line);
|
|
|
|
if (issue->column<=0 && issue->type == CompileIssueType::Other) {
|
2021-06-24 16:05:19 +08:00
|
|
|
issue->type = CompileIssueType::Error; //linkage error
|
2021-06-24 22:33:57 +08:00
|
|
|
mErrorCount += 1;
|
|
|
|
}
|
2021-06-24 16:05:19 +08:00
|
|
|
} else {
|
|
|
|
issue->column = -1;
|
|
|
|
issue->type = getIssueTypeFromOutputLine(line);
|
|
|
|
}
|
2021-04-24 15:57:45 +08:00
|
|
|
issue->description = line.trimmed();
|
2022-04-27 19:13:38 +08:00
|
|
|
if (issue->line<=0 && (issue->filename=="ld" || issue->filename=="lld")) {
|
|
|
|
mLastIssue = issue;
|
|
|
|
} else if (issue->line<=0) {
|
2021-06-24 16:05:19 +08:00
|
|
|
emit compileIssue(issue);
|
|
|
|
} else
|
|
|
|
mLastIssue = issue;
|
2021-04-24 15:57:45 +08:00
|
|
|
}
|
|
|
|
|
2021-04-20 22:24:33 +08:00
|
|
|
void Compiler::stopCompile()
|
|
|
|
{
|
|
|
|
mStop = true;
|
|
|
|
}
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList Compiler::getCharsetArgument(const QByteArray& encoding,FileType fileType, bool checkSyntax)
|
2021-04-20 22:24:33 +08:00
|
|
|
{
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList result;
|
2022-06-23 19:07:48 +08:00
|
|
|
bool forceExecUTF8=false;
|
|
|
|
// test if force utf8 from autolink infos
|
|
|
|
if ((fileType == FileType::CSource ||
|
2023-08-13 14:46:53 +08:00
|
|
|
fileType == FileType::CppSource) && pSettings->editor().enableAutolink()
|
2023-08-14 12:22:24 +08:00
|
|
|
&& mParserForFile){
|
2023-08-13 14:46:53 +08:00
|
|
|
int waitCount = 0;
|
|
|
|
//wait parsing ends, at most 1 second
|
2023-08-14 12:22:24 +08:00
|
|
|
while(mParserForFile->parsing()) {
|
2023-08-13 14:46:53 +08:00
|
|
|
if (waitCount>10)
|
|
|
|
break;
|
|
|
|
waitCount++;
|
|
|
|
QThread::msleep(100);
|
|
|
|
QApplication *app=dynamic_cast<QApplication*>(
|
|
|
|
QApplication::instance());
|
|
|
|
app->processEvents();
|
|
|
|
}
|
|
|
|
if (waitCount<=10) {
|
|
|
|
QSet<QString> parsedFiles;
|
|
|
|
forceExecUTF8 = parseForceUTF8ForAutolink(
|
|
|
|
mFilename,
|
|
|
|
parsedFiles);
|
2022-06-23 19:07:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((forceExecUTF8 || compilerSet()->autoAddCharsetParams()) && encoding != ENCODING_ASCII
|
2022-10-30 11:58:42 +08:00
|
|
|
&& compilerSet()->compilerType()!=CompilerType::Clang) {
|
2021-04-20 22:24:33 +08:00
|
|
|
QString encodingName;
|
2022-01-24 11:23:30 +08:00
|
|
|
QString execEncodingName;
|
|
|
|
QString compilerSetExecCharset = compilerSet()->execCharset();
|
2021-09-28 17:17:33 +08:00
|
|
|
QString systemEncodingName=pCharsetInfoManager->getDefaultSystemEncoding();
|
2021-04-20 22:24:33 +08:00
|
|
|
if (encoding == ENCODING_SYSTEM_DEFAULT) {
|
|
|
|
encodingName = systemEncodingName;
|
|
|
|
} else if (encoding == ENCODING_UTF8_BOM) {
|
|
|
|
encodingName = "UTF-8";
|
2023-01-14 21:51:55 +08:00
|
|
|
} else if (encoding == ENCODING_UTF16_BOM) {
|
|
|
|
encodingName = "UTF-16";
|
|
|
|
} else if (encoding == ENCODING_UTF32_BOM) {
|
|
|
|
encodingName = "UTF-32";
|
2021-04-20 22:24:33 +08:00
|
|
|
} else {
|
|
|
|
encodingName = encoding;
|
|
|
|
}
|
2022-06-23 19:07:48 +08:00
|
|
|
if (forceExecUTF8) {
|
|
|
|
execEncodingName = "UTF-8";
|
|
|
|
} else if (compilerSetExecCharset == ENCODING_SYSTEM_DEFAULT || compilerSetExecCharset.isEmpty()) {
|
2022-01-24 11:23:30 +08:00
|
|
|
execEncodingName = systemEncodingName;
|
|
|
|
} else {
|
|
|
|
execEncodingName = compilerSetExecCharset;
|
|
|
|
}
|
2022-10-10 07:11:59 +08:00
|
|
|
//qDebug()<<encodingName<<execEncodingName;
|
2022-03-31 09:56:49 +08:00
|
|
|
if (checkSyntax) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-finput-charset=" + encodingName;
|
2024-04-11 07:00:32 +08:00
|
|
|
} else if (QString::compare(encodingName, execEncodingName, Qt::CaseInsensitive) != 0) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result += {
|
|
|
|
"-finput-charset=" + encodingName,
|
|
|
|
"-fexec-charset=" + execEncodingName,
|
|
|
|
};
|
2022-03-31 09:56:49 +08:00
|
|
|
}
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList Compiler::getCCompileArguments(bool checkSyntax)
|
2021-04-20 22:24:33 +08:00
|
|
|
{
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList result;
|
2021-04-20 22:24:33 +08:00
|
|
|
if (checkSyntax) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-fsyntax-only";
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
|
2022-05-12 15:28:08 +08:00
|
|
|
QMap<QString, QString> compileOptions;
|
2022-05-14 11:21:59 +08:00
|
|
|
if (mProject && !mProject->options().compilerOptions.isEmpty()) {
|
2022-05-12 15:28:08 +08:00
|
|
|
compileOptions = mProject->options().compilerOptions;
|
|
|
|
} else {
|
|
|
|
compileOptions = compilerSet()->compileOptions();
|
|
|
|
}
|
2022-05-14 11:21:59 +08:00
|
|
|
foreach (const QString& key, compileOptions.keys()) {
|
|
|
|
if (compileOptions[key]=="")
|
|
|
|
continue;
|
|
|
|
PCompilerOption pOption = CompilerInfoManager::getCompilerOption(compilerSet()->compilerType(), key);
|
2022-05-12 15:28:08 +08:00
|
|
|
if (pOption && pOption->isC && !pOption->isLinker) {
|
2023-08-21 11:48:23 +08:00
|
|
|
if (pOption->type == CompilerOptionType::Checkbox)
|
2024-02-28 19:41:05 +08:00
|
|
|
result << pOption->setting;
|
2023-08-21 11:48:23 +08:00
|
|
|
else if (pOption->type == CompilerOptionType::Input)
|
2024-02-28 19:41:05 +08:00
|
|
|
result += {pOption->setting, compileOptions[key]};
|
2024-05-04 21:03:40 +08:00
|
|
|
else if (pOption->type == CompilerOptionType::Number) {
|
|
|
|
bool ok;
|
|
|
|
int val = compileOptions[key].toInt(&ok);
|
2024-05-17 18:15:12 +08:00
|
|
|
if (ok && val>0) {
|
2024-05-04 21:03:40 +08:00
|
|
|
val = pOption->scale * val;
|
|
|
|
result += QString("%1%2").arg(pOption->setting).arg(val);
|
|
|
|
}
|
|
|
|
} else {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << pOption->setting + compileOptions[key];
|
2022-10-24 19:23:43 +08:00
|
|
|
}
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
QMap<QString, QString> macros = devCppMacroVariables();
|
|
|
|
|
2021-04-20 22:24:33 +08:00
|
|
|
if (compilerSet()->useCustomCompileParams() && !compilerSet()->customCompileParams().isEmpty()) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << parseArguments(compilerSet()->customCompileParams(), macros, true);
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
2021-09-13 19:09:15 +08:00
|
|
|
|
|
|
|
if (mProject) {
|
|
|
|
QString s = mProject->options().compilerCmd;
|
|
|
|
if (!s.isEmpty()) {
|
|
|
|
s.replace("_@@_", " ");
|
2024-02-28 19:41:05 +08:00
|
|
|
result << parseArguments(s, macros, true);
|
2021-09-13 19:09:15 +08:00
|
|
|
}
|
|
|
|
}
|
2024-03-26 16:33:22 +08:00
|
|
|
|
|
|
|
if (result.contains("-g3")) {
|
|
|
|
result << "-D_DEBUG";
|
|
|
|
}
|
2021-04-20 22:24:33 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList Compiler::getCppCompileArguments(bool checkSyntax)
|
2021-04-20 22:24:33 +08:00
|
|
|
{
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList result;
|
2021-04-20 22:24:33 +08:00
|
|
|
if (checkSyntax) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-fsyntax-only";
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
2022-05-12 15:28:08 +08:00
|
|
|
QMap<QString, QString> compileOptions;
|
2022-05-14 11:21:59 +08:00
|
|
|
if (mProject && !mProject->options().compilerOptions.isEmpty()) {
|
2022-05-12 15:28:08 +08:00
|
|
|
compileOptions = mProject->options().compilerOptions;
|
|
|
|
} else {
|
|
|
|
compileOptions = compilerSet()->compileOptions();
|
|
|
|
}
|
2022-05-14 11:21:59 +08:00
|
|
|
foreach (const QString& key, compileOptions.keys()) {
|
|
|
|
if (compileOptions[key]=="")
|
|
|
|
continue;
|
|
|
|
PCompilerOption pOption = CompilerInfoManager::getCompilerOption(compilerSet()->compilerType(), key);
|
2022-05-12 15:28:08 +08:00
|
|
|
if (pOption && pOption->isCpp && !pOption->isLinker) {
|
2023-08-21 11:48:23 +08:00
|
|
|
if (pOption->type == CompilerOptionType::Checkbox)
|
2024-02-28 19:41:05 +08:00
|
|
|
result << pOption->setting;
|
2023-08-21 11:48:23 +08:00
|
|
|
else if (pOption->type == CompilerOptionType::Input)
|
2024-02-28 19:41:05 +08:00
|
|
|
result += {pOption->setting, compileOptions[key]};
|
2024-05-04 21:03:40 +08:00
|
|
|
else if (pOption->type == CompilerOptionType::Number) {
|
|
|
|
bool ok;
|
|
|
|
int val = compileOptions[key].toInt(&ok);
|
2024-05-17 18:15:12 +08:00
|
|
|
if (ok && val>0) {
|
2024-05-04 21:03:40 +08:00
|
|
|
val = pOption->scale * val;
|
|
|
|
result += QString("%1%2").arg(pOption->setting).arg(val);
|
|
|
|
}
|
|
|
|
} else {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << pOption->setting + compileOptions[key];
|
2023-08-21 11:48:23 +08:00
|
|
|
}
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
}
|
2024-02-28 19:41:05 +08:00
|
|
|
|
|
|
|
QMap<QString, QString> macros = devCppMacroVariables();
|
2021-04-20 22:24:33 +08:00
|
|
|
if (compilerSet()->useCustomCompileParams() && !compilerSet()->customCompileParams().isEmpty()) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << parseArguments(compilerSet()->customCompileParams(), macros, true);
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
2021-09-13 19:09:15 +08:00
|
|
|
if (mProject) {
|
|
|
|
QString s = mProject->options().cppCompilerCmd;
|
|
|
|
if (!s.isEmpty()) {
|
|
|
|
s.replace("_@@_", " ");
|
2024-02-28 19:41:05 +08:00
|
|
|
result << parseArguments(s, macros, true);
|
2021-09-13 19:09:15 +08:00
|
|
|
}
|
|
|
|
}
|
2024-03-26 16:33:22 +08:00
|
|
|
|
|
|
|
if (result.contains("-g3")) {
|
|
|
|
result << "-D_DEBUG";
|
|
|
|
}
|
2021-04-20 22:24:33 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList Compiler::getCIncludeArguments()
|
2021-04-20 22:24:33 +08:00
|
|
|
{
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList result;
|
2021-08-29 10:14:07 +08:00
|
|
|
foreach (const QString& folder,compilerSet()->CIncludeDirs()) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-I" + folder;
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList Compiler::getProjectIncludeArguments()
|
2021-09-12 22:45:00 +08:00
|
|
|
{
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList result;
|
2021-09-12 22:45:00 +08:00
|
|
|
if (mProject) {
|
2022-06-16 21:34:31 +08:00
|
|
|
foreach (const QString& folder,mProject->options().includeDirs) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-I" + folder;
|
2021-09-12 22:45:00 +08:00
|
|
|
}
|
2021-09-13 22:45:50 +08:00
|
|
|
// result += QString(" -I\"%1\"").arg(extractFilePath(mProject->filename()));
|
2021-09-12 22:45:00 +08:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList Compiler::getCppIncludeArguments()
|
2021-04-20 22:24:33 +08:00
|
|
|
{
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList result;
|
2021-08-29 10:14:07 +08:00
|
|
|
foreach (const QString& folder,compilerSet()->CppIncludeDirs()) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-I" + folder;
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList Compiler::getLibraryArguments(FileType fileType)
|
2021-04-20 22:24:33 +08:00
|
|
|
{
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList result;
|
2021-04-20 22:24:33 +08:00
|
|
|
|
2021-09-13 19:09:15 +08:00
|
|
|
//Add libraries
|
2021-08-29 10:14:07 +08:00
|
|
|
foreach (const QString& folder, compilerSet()->libDirs()) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-L" + folder;
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
|
2021-09-13 19:09:15 +08:00
|
|
|
//add libs added via project
|
|
|
|
if (mProject) {
|
2022-06-16 21:34:31 +08:00
|
|
|
foreach (const QString& folder, mProject->options().libDirs){
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-L" + folder;
|
2021-09-13 19:09:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-04 11:37:04 +08:00
|
|
|
//Add auto links
|
|
|
|
// is file and auto link enabled
|
2021-09-04 19:27:44 +08:00
|
|
|
if (pSettings->editor().enableAutolink() && (fileType == FileType::CSource ||
|
2023-08-13 14:46:53 +08:00
|
|
|
fileType == FileType::CppSource)
|
2023-08-14 12:22:24 +08:00
|
|
|
&& mParserForFile){
|
2023-08-13 14:46:53 +08:00
|
|
|
int waitCount = 0;
|
|
|
|
//wait parsing ends, at most 1 second
|
2023-08-14 12:22:24 +08:00
|
|
|
while(mParserForFile->parsing()) {
|
2023-08-13 14:46:53 +08:00
|
|
|
if (waitCount>10)
|
|
|
|
break;
|
|
|
|
waitCount++;
|
|
|
|
QThread::msleep(100);
|
|
|
|
QApplication *app=dynamic_cast<QApplication*>(
|
|
|
|
QApplication::instance());
|
|
|
|
app->processEvents();
|
|
|
|
}
|
|
|
|
if (waitCount<=10) {
|
|
|
|
QSet<QString> parsedFiles;
|
2024-02-28 19:41:05 +08:00
|
|
|
result += parseFileIncludesForAutolink(mFilename, parsedFiles);
|
2021-09-04 11:37:04 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-13 19:09:15 +08:00
|
|
|
//add compiler set link options
|
|
|
|
//options like "-static" must be added after "-lxxx"
|
2022-05-12 15:28:08 +08:00
|
|
|
QMap<QString, QString> compileOptions;
|
2022-05-14 11:21:59 +08:00
|
|
|
if (mProject && !mProject->options().compilerOptions.isEmpty()) {
|
2022-05-12 15:28:08 +08:00
|
|
|
compileOptions = mProject->options().compilerOptions;
|
|
|
|
} else {
|
|
|
|
compileOptions = compilerSet()->compileOptions();
|
|
|
|
}
|
2022-05-14 11:21:59 +08:00
|
|
|
foreach (const QString& key, compileOptions.keys()) {
|
|
|
|
if (compileOptions[key]=="")
|
|
|
|
continue;
|
|
|
|
PCompilerOption pOption = CompilerInfoManager::getCompilerOption(compilerSet()->compilerType(), key);
|
|
|
|
if (pOption && pOption->isLinker) {
|
2023-08-21 11:48:23 +08:00
|
|
|
if (pOption->type == CompilerOptionType::Checkbox)
|
2024-02-28 19:41:05 +08:00
|
|
|
result << pOption->setting;
|
2023-08-21 11:48:23 +08:00
|
|
|
else if (pOption->type == CompilerOptionType::Input)
|
2024-02-28 19:41:05 +08:00
|
|
|
result += {pOption->setting, compileOptions[key]};
|
2024-05-04 21:03:40 +08:00
|
|
|
else if (pOption->type == CompilerOptionType::Number) {
|
|
|
|
bool ok;
|
|
|
|
int val = compileOptions[key].toInt(&ok);
|
2024-05-17 18:15:12 +08:00
|
|
|
if (ok && val>0) {
|
2024-05-04 21:03:40 +08:00
|
|
|
val = pOption->scale * val;
|
|
|
|
result += QString("%1%2").arg(pOption->setting).arg(val);
|
|
|
|
}
|
|
|
|
} else {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << pOption->setting + compileOptions[key];
|
2023-08-21 11:48:23 +08:00
|
|
|
}
|
2021-09-13 19:09:15 +08:00
|
|
|
}
|
|
|
|
}
|
2021-09-04 11:37:04 +08:00
|
|
|
|
2021-04-20 22:24:33 +08:00
|
|
|
// Add global compiler linker extras
|
|
|
|
if (compilerSet()->useCustomLinkParams() && !compilerSet()->customLinkParams().isEmpty()) {
|
2024-03-19 20:09:47 +08:00
|
|
|
QMap<QString, QString> macros = devCppMacroVariables();
|
|
|
|
QStringList params = parseArguments(compilerSet()->customLinkParams(), macros, true);
|
|
|
|
if (!params.isEmpty()) {
|
2022-06-22 17:08:35 +08:00
|
|
|
foreach(const QString& param, params)
|
2024-02-28 19:41:05 +08:00
|
|
|
result << param;
|
2024-03-19 20:09:47 +08:00
|
|
|
}
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
|
2021-09-13 19:09:15 +08:00
|
|
|
if (mProject) {
|
|
|
|
if (mProject->options().type == ProjectType::GUI) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-mwindows";
|
2021-09-13 19:09:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!mProject->options().linkerCmd.isEmpty()) {
|
2021-09-13 22:45:50 +08:00
|
|
|
QString s = mProject->options().linkerCmd;
|
|
|
|
if (!s.isEmpty()) {
|
2024-02-28 19:41:05 +08:00
|
|
|
s.replace("_@@_", " "); // historical reason
|
|
|
|
result += parseArguments(s, {}, true);
|
2021-09-13 22:45:50 +08:00
|
|
|
}
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
2021-09-13 19:09:15 +08:00
|
|
|
if (mProject->options().staticLink)
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-static";
|
2023-03-13 21:57:15 +08:00
|
|
|
} else {
|
|
|
|
if (compilerSet()->staticLink()) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result << "-static";
|
2023-03-13 21:57:15 +08:00
|
|
|
}
|
2021-09-02 21:01:23 +08:00
|
|
|
}
|
2021-04-20 22:24:33 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList Compiler::parseFileIncludesForAutolink(
|
2021-09-04 11:37:04 +08:00
|
|
|
const QString &filename,
|
2023-08-13 14:46:53 +08:00
|
|
|
QSet<QString>& parsedFiles)
|
2021-09-04 11:37:04 +08:00
|
|
|
{
|
2024-02-28 19:41:05 +08:00
|
|
|
QStringList result;
|
2021-09-04 11:37:04 +08:00
|
|
|
if (parsedFiles.contains(filename))
|
|
|
|
return result;
|
|
|
|
parsedFiles.insert(filename);
|
2022-01-24 20:59:02 +08:00
|
|
|
PAutolink autolink = pAutolinkManager->getLink(filename);
|
2021-09-04 11:37:04 +08:00
|
|
|
if (autolink) {
|
2024-02-28 19:41:05 +08:00
|
|
|
result += parseArgumentsWithoutVariables(autolink->linkOption);
|
2021-09-04 11:37:04 +08:00
|
|
|
}
|
2023-08-14 12:22:24 +08:00
|
|
|
QStringList includedFiles = mParserForFile->getFileDirectIncludes(filename);
|
2022-06-04 14:44:50 +08:00
|
|
|
// log(QString("File %1 included:").arg(filename));
|
|
|
|
// for (int i=includedFiles.size()-1;i>=0;i--) {
|
|
|
|
// QString includeFilename = includedFiles[i];
|
|
|
|
// log(includeFilename);
|
|
|
|
// }
|
|
|
|
|
|
|
|
for (int i=includedFiles.size()-1;i>=0;i--) {
|
|
|
|
QString includeFilename = includedFiles[i];
|
2024-02-28 19:41:05 +08:00
|
|
|
result += parseFileIncludesForAutolink(includeFilename, parsedFiles);
|
2021-09-04 11:37:04 +08:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-08-13 14:46:53 +08:00
|
|
|
bool Compiler::parseForceUTF8ForAutolink(const QString &filename, QSet<QString> &parsedFiles)
|
2022-06-23 19:07:48 +08:00
|
|
|
{
|
|
|
|
bool result;
|
|
|
|
if (parsedFiles.contains(filename))
|
|
|
|
return false;
|
|
|
|
parsedFiles.insert(filename);
|
|
|
|
PAutolink autolink = pAutolinkManager->getLink(filename);
|
|
|
|
if (autolink && autolink->execUseUTF8) {
|
|
|
|
return true;
|
|
|
|
}
|
2023-08-14 12:22:24 +08:00
|
|
|
QStringList includedFiles = mParserForFile->getFileDirectIncludes(filename);
|
2022-06-23 19:07:48 +08:00
|
|
|
// log(QString("File %1 included:").arg(filename));
|
|
|
|
// for (int i=includedFiles.size()-1;i>=0;i--) {
|
|
|
|
// QString includeFilename = includedFiles[i];
|
|
|
|
// log(includeFilename);
|
|
|
|
// }
|
|
|
|
|
|
|
|
for (int i=includedFiles.size()-1;i>=0;i--) {
|
|
|
|
QString includeFilename = includedFiles[i];
|
|
|
|
result = parseForceUTF8ForAutolink(
|
|
|
|
includeFilename,
|
2023-08-13 14:46:53 +08:00
|
|
|
parsedFiles);
|
2022-06-23 19:07:48 +08:00
|
|
|
if (result)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
void Compiler::runCommand(const QString &cmd, const QStringList &arguments, const QString &workingDir, const QByteArray& inputText, const QString& outputFile)
|
2021-04-20 22:24:33 +08:00
|
|
|
{
|
|
|
|
QProcess process;
|
|
|
|
mStop = false;
|
2021-06-21 11:21:26 +08:00
|
|
|
bool errorOccurred = false;
|
2021-04-20 22:24:33 +08:00
|
|
|
process.setProgram(cmd);
|
2021-09-13 22:45:50 +08:00
|
|
|
QString cmdDir = extractFileDir(cmd);
|
2023-08-08 12:44:46 +08:00
|
|
|
bool compilerErrorUTF8=compilerSet()->isCompilerInfoUsingUTF8();
|
|
|
|
bool outputUTF8=compilerSet()->forceUTF8();
|
2021-12-16 17:56:42 +08:00
|
|
|
QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
|
2021-09-13 22:45:50 +08:00
|
|
|
if (!cmdDir.isEmpty()) {
|
|
|
|
QString path = env.value("PATH");
|
2021-09-14 12:10:43 +08:00
|
|
|
if (path.isEmpty()) {
|
|
|
|
path = cmdDir;
|
|
|
|
} else {
|
|
|
|
path = cmdDir + PATH_SEPARATOR + path;
|
|
|
|
}
|
2021-09-13 22:45:50 +08:00
|
|
|
env.insert("PATH",path);
|
|
|
|
}
|
2024-02-20 21:47:12 +08:00
|
|
|
if (compilerSet() && compilerSet()->forceEnglishOutput())
|
|
|
|
env.insert("LANG","en");
|
2024-05-05 10:24:36 +08:00
|
|
|
//env.insert("LDFLAGS","-Wl,--stack,12582912");
|
|
|
|
env.insert("LDFLAGS","");
|
2022-02-11 14:04:18 +08:00
|
|
|
env.insert("CFLAGS","");
|
|
|
|
env.insert("CXXFLAGS","");
|
2021-12-16 17:56:42 +08:00
|
|
|
process.setProcessEnvironment(env);
|
2024-02-28 19:41:05 +08:00
|
|
|
process.setArguments(arguments);
|
2021-04-20 22:24:33 +08:00
|
|
|
process.setWorkingDirectory(workingDir);
|
2023-08-13 18:53:48 +08:00
|
|
|
QFile output;
|
|
|
|
if (!outputFile.isEmpty()) {
|
|
|
|
output.setFileName(outputFile);
|
|
|
|
if (!output.open(QFile::WriteOnly | QFile::Truncate)) {
|
|
|
|
this->error(tr("Can't open file \"%1\" for write!"));
|
|
|
|
return;
|
|
|
|
};
|
|
|
|
}
|
2021-06-21 11:21:26 +08:00
|
|
|
process.connect(&process, &QProcess::errorOccurred,
|
|
|
|
[&](){
|
|
|
|
errorOccurred= true;
|
|
|
|
});
|
2023-08-08 12:44:46 +08:00
|
|
|
process.connect(&process, &QProcess::readyReadStandardError,[&process,this,compilerErrorUTF8](){
|
|
|
|
if (compilerErrorUTF8)
|
2022-04-27 15:26:13 +08:00
|
|
|
this->error(QString::fromUtf8(process.readAllStandardError()));
|
|
|
|
else
|
|
|
|
this->error(QString::fromLocal8Bit( process.readAllStandardError()));
|
2021-04-20 22:24:33 +08:00
|
|
|
});
|
2023-08-13 18:53:48 +08:00
|
|
|
process.connect(&process, &QProcess::readyReadStandardOutput,[&process,this,outputUTF8,&outputFile,&output](){
|
|
|
|
if (!outputFile.isEmpty()) {
|
|
|
|
output.write(process.readAllStandardOutput());
|
|
|
|
} else {
|
|
|
|
if (outputUTF8)
|
|
|
|
this->log(QString::fromUtf8(process.readAllStandardOutput()));
|
|
|
|
else
|
|
|
|
this->log(QString::fromLocal8Bit( process.readAllStandardOutput()));
|
|
|
|
}
|
2021-04-20 22:24:33 +08:00
|
|
|
});
|
2021-08-29 10:14:07 +08:00
|
|
|
process.connect(&process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished),[this](){
|
2021-06-24 20:43:09 +08:00
|
|
|
this->error(COMPILE_PROCESS_END);
|
|
|
|
});
|
2021-04-20 22:24:33 +08:00
|
|
|
process.start();
|
2021-06-24 20:43:09 +08:00
|
|
|
process.waitForStarted(5000);
|
2022-03-31 09:56:49 +08:00
|
|
|
if (!inputText.isEmpty()) {
|
|
|
|
process.write(inputText);
|
|
|
|
process.waitForFinished(0);
|
|
|
|
}
|
|
|
|
bool writeChannelClosed = false;
|
2021-04-20 22:24:33 +08:00
|
|
|
while (true) {
|
2022-03-31 09:56:49 +08:00
|
|
|
if (process.bytesToWrite()==0 && !writeChannelClosed ) {
|
|
|
|
writeChannelClosed=true;
|
|
|
|
process.closeWriteChannel();
|
|
|
|
}
|
|
|
|
process.waitForFinished(100);
|
2021-04-20 22:24:33 +08:00
|
|
|
if (process.state()!=QProcess::Running) {
|
|
|
|
break;
|
|
|
|
}
|
2021-06-25 12:40:11 +08:00
|
|
|
if (mStop) {
|
|
|
|
process.terminate();
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
2021-06-25 12:40:11 +08:00
|
|
|
if (errorOccurred)
|
|
|
|
break;
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
2021-06-21 11:21:26 +08:00
|
|
|
if (errorOccurred) {
|
|
|
|
switch (process.error()) {
|
|
|
|
case QProcess::FailedToStart:
|
2021-09-04 14:06:48 +08:00
|
|
|
throw CompileError(tr("The compiler process for '%1' failed to start.").arg(mFilename));
|
2021-06-21 11:21:26 +08:00
|
|
|
break;
|
|
|
|
case QProcess::Crashed:
|
2021-06-25 12:40:11 +08:00
|
|
|
if (!mStop)
|
|
|
|
throw CompileError(tr("The compiler process crashed after starting successfully."));
|
2021-06-21 11:21:26 +08:00
|
|
|
break;
|
|
|
|
case QProcess::Timedout:
|
|
|
|
throw CompileError(tr("The last waitFor...() function timed out."));
|
|
|
|
break;
|
|
|
|
case QProcess::WriteError:
|
|
|
|
throw CompileError(tr("An error occurred when attempting to write to the compiler process."));
|
|
|
|
break;
|
|
|
|
case QProcess::ReadError:
|
|
|
|
throw CompileError(tr("An error occurred when attempting to read from the compiler process."));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw CompileError(tr("An unknown error occurred."));
|
|
|
|
}
|
|
|
|
}
|
2023-08-13 18:53:48 +08:00
|
|
|
if (!outputFile.isEmpty())
|
|
|
|
output.close();
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|
|
|
|
|
2024-02-28 19:41:05 +08:00
|
|
|
QString Compiler::escapeCommandForLog(const QString &cmd, const QStringList &arguments)
|
|
|
|
{
|
|
|
|
return escapeCommandForPlatformShell(extractFileName(cmd), arguments);
|
|
|
|
}
|
|
|
|
|
2023-08-13 14:46:53 +08:00
|
|
|
PCppParser Compiler::parser() const
|
|
|
|
{
|
2023-08-14 12:22:24 +08:00
|
|
|
return mParserForFile;
|
2023-08-13 14:46:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Compiler::getParserForFile(const QString &filename)
|
|
|
|
{
|
|
|
|
FileType fileType = getFileType(filename);
|
|
|
|
if (fileType == FileType::CSource ||
|
|
|
|
fileType == FileType::CppSource){
|
|
|
|
Editor* editor = pMainWindow->editorList()->getOpenedEditorByFilename(filename);
|
|
|
|
if (editor && editor->parser()) {
|
2023-08-14 12:22:24 +08:00
|
|
|
mParserForFile=editor->parser();
|
2023-08-13 14:46:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-12 22:45:00 +08:00
|
|
|
const std::shared_ptr<Project> &Compiler::project() const
|
|
|
|
{
|
|
|
|
return mProject;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Compiler::setProject(const std::shared_ptr<Project> &newProject)
|
|
|
|
{
|
|
|
|
mProject = newProject;
|
|
|
|
}
|
|
|
|
|
2021-06-25 12:40:11 +08:00
|
|
|
bool Compiler::isRebuild() const
|
|
|
|
{
|
|
|
|
return mRebuild;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Compiler::setRebuild(bool isRebuild)
|
|
|
|
{
|
|
|
|
mRebuild = isRebuild;
|
|
|
|
}
|
|
|
|
|
2021-04-20 22:24:33 +08:00
|
|
|
void Compiler::log(const QString &msg)
|
|
|
|
{
|
|
|
|
emit compileOutput(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Compiler::error(const QString &msg)
|
|
|
|
{
|
2021-06-24 20:43:09 +08:00
|
|
|
if (msg != COMPILE_PROCESS_END)
|
|
|
|
emit compileOutput(msg);
|
2021-04-24 15:57:45 +08:00
|
|
|
for (QString& s:msg.split("\n")) {
|
|
|
|
if (!s.isEmpty())
|
|
|
|
processOutput(s);
|
|
|
|
}
|
2021-04-20 22:24:33 +08:00
|
|
|
}
|