RedPanda-CPP/RedPandaIDE/compiler/filecompiler.cpp

208 lines
7.6 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 "compilermanager.h"
#include "qsynedit/syntaxer/asm.h"
#include "../systemconsts.h"
#include <QFile>
#include <QFileInfo>
#include <QMessageBox>
FileCompiler::FileCompiler(const QString &filename, const QByteArray &encoding,
CppCompileType compileType, bool onlyCheckSyntax):
Compiler(filename, 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 = QStringList{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());
}
#ifdef ENABLE_SDCC
if (compilerSet()->compilerType()==CompilerType::SDCC) {
if (compilerSet()->executableSuffix()==SDCC_IHX_SUFFIX) {
}
}
#endif
mArguments += {"-o", 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 = tr("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 (mArguments.contains("-g3")) {
mArguments << "-D_DEBUG";
}
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));
QString command = escapeCommandForLog(mCompiler, mArguments);
log(tr("Command: %1").arg(command));
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;
}