/* * 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 . */ #include "filecompiler.h" #include "utils.h" #include "compilermanager.h" #include "qsynedit/syntaxer/asm.h" #include "../systemconsts.h" #include #include #include 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 = 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()); } #ifdef ENABLE_SDCC if (compilerSet()->compilerType()==CompilerType::SDCC) { if (compilerSet()->executableSuffix()==SDCC_IHX_SUFFIX) { } } #endif 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 = 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 (!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;itokenType() != 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) +"
" +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; }