RedPanda-CPP/RedPandaIDE/compiler/filecompiler.cpp

199 lines
7.4 KiB
C++

/*
* 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/>.
*/
#include "filecompiler.h"
#include "utils.h"
#include "../mainwindow.h"
#include "compilermanager.h"
#include "qsynedit/syntaxer/asm.h"
#include <QFile>
#include <QFileInfo>
#include <QMessageBox>
FileCompiler::FileCompiler(const QString &filename, const QByteArray &encoding,
CppCompileType compileType,bool silent,bool onlyCheckSyntax):
Compiler(filename, silent,onlyCheckSyntax),
mEncoding(encoding),
mCompileType(compileType)
{
}
bool FileCompiler::prepareForCompile()
{
Settings::CompilerSet::CompilationStage oldStage = compilerSet()->compilationStage();
QString oldDebugOptionValue = compilerSet()->getCompileOptionValue(CC_CMD_OPT_DEBUG_INFO);
auto action = finally([this,oldStage,oldDebugOptionValue]{
compilerSet()->setCompilationStage(oldStage);
compilerSet()->setCompileOption(CC_CMD_OPT_DEBUG_INFO,oldDebugOptionValue);
});
Settings::CompilerSet::CompilationStage stage = oldStage;
switch(mCompileType) {
case CppCompileType::PreprocessOnly:
stage = Settings::CompilerSet::CompilationStage::PreprocessingOnly;
break;
case CppCompileType::GenerateAssemblyOnly:
stage = Settings::CompilerSet::CompilationStage::CompilationProperOnly;
if (pSettings->languages().noDebugDirectivesWhenGenerateASM())
compilerSet()->setCompileOption(CC_CMD_OPT_DEBUG_INFO,COMPILER_OPTION_OFF);
break;
default:
stage = oldStage;
}
compilerSet()->setCompilationStage(stage);
if (mOnlyCheckSyntax) {
log(tr("Checking single file..."));
} else {
log(tr("Compiling single file..."));
}
log("------------------");
log(tr("- Filename: %1").arg(mFilename));
log(tr("- Compiler Set Name: %1").arg(compilerSet()->name()));
log("");
FileType fileType = getFileType(mFilename);
mArguments = QString(" \"%1\"").arg(mFilename);
if (!mOnlyCheckSyntax) {
switch(compilerSet()->compilationStage()) {
case Settings::CompilerSet::CompilationStage::PreprocessingOnly:
mOutputFile=changeFileExt(mFilename,compilerSet()->preprocessingSuffix());
mArguments+=" -E";
break;
case Settings::CompilerSet::CompilationStage::CompilationProperOnly:
mOutputFile=changeFileExt(mFilename,compilerSet()->compilationProperSuffix());
mArguments+=" -S -fverbose-asm";
break;
case Settings::CompilerSet::CompilationStage::AssemblingOnly:
mOutputFile=changeFileExt(mFilename,compilerSet()->assemblingSuffix());
mArguments+=" -c";
break;
case Settings::CompilerSet::CompilationStage::GenerateExecutable:
mOutputFile = changeFileExt(mFilename,compilerSet()->executableSuffix());
}
mArguments+=QString(" -o \"%1\"").arg(mOutputFile);
#if defined(ARCH_X86_64) || defined(ARCH_X86)
if (mCompileType == CppCompileType::GenerateAssemblyOnly) {
if (pSettings->languages().noSEHDirectivesWhenGenerateASM())
mArguments+=" -fno-asynchronous-unwind-tables";
if (pSettings->languages().x86DialectOfASMGenerated()==Settings::Languages::X86ASMDialect::Intel)
mArguments+=" -masm=intel";
}
#endif
//remove the old file if it exists
QFile outputFile(mOutputFile);
if (outputFile.exists()) {
if (!outputFile.remove()) {
error(tr("Can't delete the old executable file \"%1\".\n").arg(mOutputFile));
return false;
}
}
}
mArguments += getCharsetArgument(mEncoding, fileType, mOnlyCheckSyntax);
QString strFileType;
switch(fileType) {
case FileType::GAS:
mArguments += getCCompileArguments(mOnlyCheckSyntax);
mArguments += getCIncludeArguments();
mArguments += getProjectIncludeArguments();
strFileType = "GNU Assembler";
mCompiler = compilerSet()->CCompiler();
break;
case FileType::CSource:
mArguments += getCCompileArguments(mOnlyCheckSyntax);
mArguments += getCIncludeArguments();
mArguments += getProjectIncludeArguments();
strFileType = "C";
mCompiler = compilerSet()->CCompiler();
break;
case FileType::CppSource:
mArguments += getCppCompileArguments(mOnlyCheckSyntax);
mArguments += getCppIncludeArguments();
mArguments += getProjectIncludeArguments();
strFileType = "C++";
mCompiler = compilerSet()->cppCompiler();
break;
default:
throw CompileError(tr("Can't find the compiler for file %1").arg(mFilename));
}
if (!mOnlyCheckSyntax)
mArguments += getLibraryArguments(fileType);
if (fileType==FileType::GAS) {
bool hasStart=false;
QStringList lines=readFileToLines(mFilename);
QSynedit::ASMSyntaxer syntaxer;
syntaxer.resetState();
QString lastToken;
QString token;
QSynedit::PTokenAttribute attr;
for (int i=0;i<lines.count();i++) {
QString line=lines[i];
syntaxer.setLine(line,i+1);
lastToken="";
while(!syntaxer.eol()) {
token=syntaxer.getToken();
if (token==":" && lastToken=="_start") {
hasStart=true;
break;
}
attr = syntaxer.getTokenAttribute();
if (attr->tokenType() != QSynedit::TokenType::Space
&& attr->tokenType()!=QSynedit::TokenType::String
&& attr->tokenType()!=QSynedit::TokenType::Character)
lastToken=token;
syntaxer.next();
}
if (hasStart)
break;
}
if (hasStart) {
mArguments+=" -nostartfiles";
}
}
if (!fileExists(mCompiler)) {
throw CompileError(
tr("The Compiler '%1' doesn't exists!").arg(mCompiler)
+"<br />"
+tr("Please check the \"program\" page of compiler settings."));
}
log(tr("Processing %1 source file:").arg(strFileType));
log("------------------");
log(tr("%1 Compiler: %2").arg(strFileType).arg(mCompiler));
log(tr("Command: %1 %2").arg(extractFileName(mCompiler)).arg(mArguments));
mDirectory = extractFileDir(mFilename);
return true;
}
bool FileCompiler::prepareForRebuild()
{
QString exeName=compilerSet()->getOutputFilename(mFilename);
QFile file(exeName);
if (file.exists() && !file.remove()) {
QFileInfo info(exeName);
throw CompileError(tr("Can't delete the old executable file \"%1\".\n").arg(info.absoluteFilePath()));
}
return true;
}