1257b2971SCaroline Concatto //===- CompilerInvocation.cpp ---------------------------------------------===// 2257b2971SCaroline Concatto // 3257b2971SCaroline Concatto // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4257b2971SCaroline Concatto // See https://llvm.org/LICENSE.txt for license information. 5257b2971SCaroline Concatto // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6257b2971SCaroline Concatto // 7257b2971SCaroline Concatto //===----------------------------------------------------------------------===// 8257b2971SCaroline Concatto 9257b2971SCaroline Concatto #include "flang/Frontend/CompilerInvocation.h" 106d48a1a5SFaris Rehman #include "flang/Common/Fortran-features.h" 117809fa20SFaris Rehman #include "flang/Frontend/PreprocessorOptions.h" 126d48a1a5SFaris Rehman #include "flang/Semantics/semantics.h" 13197d9a55SFaris Rehman #include "flang/Version.inc" 14257b2971SCaroline Concatto #include "clang/Basic/AllDiagnostics.h" 15257b2971SCaroline Concatto #include "clang/Basic/DiagnosticDriver.h" 16257b2971SCaroline Concatto #include "clang/Basic/DiagnosticOptions.h" 17257b2971SCaroline Concatto #include "clang/Driver/DriverDiagnostic.h" 18257b2971SCaroline Concatto #include "clang/Driver/Options.h" 19257b2971SCaroline Concatto #include "llvm/ADT/StringRef.h" 20257b2971SCaroline Concatto #include "llvm/ADT/StringSwitch.h" 21257b2971SCaroline Concatto #include "llvm/Option/Arg.h" 22257b2971SCaroline Concatto #include "llvm/Option/ArgList.h" 23257b2971SCaroline Concatto #include "llvm/Option/OptTable.h" 24cd4abc52SArnamoy Bhattacharyya #include "llvm/Support/FileSystem.h" 25cd4abc52SArnamoy Bhattacharyya #include "llvm/Support/FileUtilities.h" 26*5597ec2dSserge-sans-paille #include "llvm/Support/Path.h" 278d51d37eSAndrzej Warzynski #include "llvm/Support/Process.h" 28257b2971SCaroline Concatto #include "llvm/Support/raw_ostream.h" 296d48a1a5SFaris Rehman #include <memory> 30257b2971SCaroline Concatto 31257b2971SCaroline Concatto using namespace Fortran::frontend; 32257b2971SCaroline Concatto 33257b2971SCaroline Concatto //===----------------------------------------------------------------------===// 34257b2971SCaroline Concatto // Initialization. 35257b2971SCaroline Concatto //===----------------------------------------------------------------------===// 36257b2971SCaroline Concatto CompilerInvocationBase::CompilerInvocationBase() 377809fa20SFaris Rehman : diagnosticOpts_(new clang::DiagnosticOptions()), 387809fa20SFaris Rehman preprocessorOpts_(new PreprocessorOptions()) {} 39257b2971SCaroline Concatto 40257b2971SCaroline Concatto CompilerInvocationBase::CompilerInvocationBase(const CompilerInvocationBase &x) 417809fa20SFaris Rehman : diagnosticOpts_(new clang::DiagnosticOptions(x.GetDiagnosticOpts())), 427809fa20SFaris Rehman preprocessorOpts_(new PreprocessorOptions(x.preprocessorOpts())) {} 43257b2971SCaroline Concatto 44257b2971SCaroline Concatto CompilerInvocationBase::~CompilerInvocationBase() = default; 45257b2971SCaroline Concatto 46257b2971SCaroline Concatto //===----------------------------------------------------------------------===// 47257b2971SCaroline Concatto // Deserialization (from args) 48257b2971SCaroline Concatto //===----------------------------------------------------------------------===// 498d51d37eSAndrzej Warzynski static bool parseShowColorsArgs( 508d51d37eSAndrzej Warzynski const llvm::opt::ArgList &args, bool defaultColor) { 518d51d37eSAndrzej Warzynski // Color diagnostics default to auto ("on" if terminal supports) in the driver 528d51d37eSAndrzej Warzynski // but default to off in cc1, needing an explicit OPT_fdiagnostics_color. 538d51d37eSAndrzej Warzynski // Support both clang's -f[no-]color-diagnostics and gcc's 548d51d37eSAndrzej Warzynski // -f[no-]diagnostics-colors[=never|always|auto]. 558d51d37eSAndrzej Warzynski enum { 568d51d37eSAndrzej Warzynski Colors_On, 578d51d37eSAndrzej Warzynski Colors_Off, 588d51d37eSAndrzej Warzynski Colors_Auto 598d51d37eSAndrzej Warzynski } ShowColors = defaultColor ? Colors_Auto : Colors_Off; 608d51d37eSAndrzej Warzynski 618d51d37eSAndrzej Warzynski for (auto *a : args) { 628d51d37eSAndrzej Warzynski const llvm::opt::Option &O = a->getOption(); 638d51d37eSAndrzej Warzynski if (O.matches(clang::driver::options::OPT_fcolor_diagnostics) || 648d51d37eSAndrzej Warzynski O.matches(clang::driver::options::OPT_fdiagnostics_color)) { 658d51d37eSAndrzej Warzynski ShowColors = Colors_On; 668d51d37eSAndrzej Warzynski } else if (O.matches(clang::driver::options::OPT_fno_color_diagnostics) || 678d51d37eSAndrzej Warzynski O.matches(clang::driver::options::OPT_fno_diagnostics_color)) { 688d51d37eSAndrzej Warzynski ShowColors = Colors_Off; 698d51d37eSAndrzej Warzynski } else if (O.matches(clang::driver::options::OPT_fdiagnostics_color_EQ)) { 708d51d37eSAndrzej Warzynski llvm::StringRef value(a->getValue()); 718d51d37eSAndrzej Warzynski if (value == "always") 728d51d37eSAndrzej Warzynski ShowColors = Colors_On; 738d51d37eSAndrzej Warzynski else if (value == "never") 748d51d37eSAndrzej Warzynski ShowColors = Colors_Off; 758d51d37eSAndrzej Warzynski else if (value == "auto") 768d51d37eSAndrzej Warzynski ShowColors = Colors_Auto; 778d51d37eSAndrzej Warzynski } 788d51d37eSAndrzej Warzynski } 798d51d37eSAndrzej Warzynski 808d51d37eSAndrzej Warzynski return ShowColors == Colors_On || 818d51d37eSAndrzej Warzynski (ShowColors == Colors_Auto && llvm::sys::Process::StandardErrHasColors()); 828d51d37eSAndrzej Warzynski } 838d51d37eSAndrzej Warzynski 848d51d37eSAndrzej Warzynski bool Fortran::frontend::ParseDiagnosticArgs(clang::DiagnosticOptions &opts, 858d51d37eSAndrzej Warzynski llvm::opt::ArgList &args, bool defaultDiagColor) { 868d51d37eSAndrzej Warzynski opts.ShowColors = parseShowColorsArgs(args, defaultDiagColor); 878d51d37eSAndrzej Warzynski 888d51d37eSAndrzej Warzynski return true; 898d51d37eSAndrzej Warzynski } 908d51d37eSAndrzej Warzynski 91523d7bc6SAndrzej Warzynski // Tweak the frontend configuration based on the frontend action 92523d7bc6SAndrzej Warzynski static void setUpFrontendBasedOnAction(FrontendOptions &opts) { 9323d4c4f3SAndrzej Warzynski assert(opts.programAction != Fortran::frontend::InvalidAction && 94523d7bc6SAndrzej Warzynski "Fortran frontend action not set!"); 95523d7bc6SAndrzej Warzynski 9623d4c4f3SAndrzej Warzynski if (opts.programAction == DebugDumpParsingLog) 9723d4c4f3SAndrzej Warzynski opts.instrumentedParse = true; 9885b86c6fSAndrzej Warzynski 9923d4c4f3SAndrzej Warzynski if (opts.programAction == DebugDumpProvenance || 10023d4c4f3SAndrzej Warzynski opts.programAction == Fortran::frontend::GetDefinition) 10123d4c4f3SAndrzej Warzynski opts.needProvenanceRangeToCharBlockMappings = true; 102523d7bc6SAndrzej Warzynski } 103523d7bc6SAndrzej Warzynski 104dc256a44SAndrzej Warzynski static bool ParseFrontendArgs(FrontendOptions &opts, llvm::opt::ArgList &args, 1056d0fef48SAndrzej Warzynski clang::DiagnosticsEngine &diags) { 106dc256a44SAndrzej Warzynski unsigned numErrorsBefore = diags.getNumErrors(); 107760e6c4cSAndrzej Warzynski 108760e6c4cSAndrzej Warzynski // By default the frontend driver creates a ParseSyntaxOnly action. 10923d4c4f3SAndrzej Warzynski opts.programAction = ParseSyntaxOnly; 110760e6c4cSAndrzej Warzynski 111d18a9aeaSAndrzej Warzynski // Treat multiple action options as an invocation error. Note that `clang 112d18a9aeaSAndrzej Warzynski // -cc1` does accept multiple action options, but will only consider the 113d18a9aeaSAndrzej Warzynski // rightmost one. 114d18a9aeaSAndrzej Warzynski if (args.hasMultipleArgs(clang::driver::options::OPT_Action_Group)) { 115d18a9aeaSAndrzej Warzynski const unsigned diagID = diags.getCustomDiagID( 116d18a9aeaSAndrzej Warzynski clang::DiagnosticsEngine::Error, "Only one action option is allowed"); 117d18a9aeaSAndrzej Warzynski diags.Report(diagID); 118d18a9aeaSAndrzej Warzynski return false; 119d18a9aeaSAndrzej Warzynski } 120d18a9aeaSAndrzej Warzynski 121257b2971SCaroline Concatto // Identify the action (i.e. opts.ProgramAction) 122257b2971SCaroline Concatto if (const llvm::opt::Arg *a = 123257b2971SCaroline Concatto args.getLastArg(clang::driver::options::OPT_Action_Group)) { 124257b2971SCaroline Concatto switch (a->getOption().getID()) { 125257b2971SCaroline Concatto default: { 126257b2971SCaroline Concatto llvm_unreachable("Invalid option in group!"); 127257b2971SCaroline Concatto } 1284c5906cfSCaroline Concatto case clang::driver::options::OPT_test_io: 12923d4c4f3SAndrzej Warzynski opts.programAction = InputOutputTest; 1304c5906cfSCaroline Concatto break; 131d28de0d7SCaroline Concatto case clang::driver::options::OPT_E: 13223d4c4f3SAndrzej Warzynski opts.programAction = PrintPreprocessedInput; 133d28de0d7SCaroline Concatto break; 1347d246cb1SAndrzej Warzynski case clang::driver::options::OPT_fsyntax_only: 13523d4c4f3SAndrzej Warzynski opts.programAction = ParseSyntaxOnly; 1367d246cb1SAndrzej Warzynski break; 137e5cdb6c5SAndrzej Warzynski case clang::driver::options::OPT_emit_obj: 13823d4c4f3SAndrzej Warzynski opts.programAction = EmitObj; 139e5cdb6c5SAndrzej Warzynski break; 14096d229c9SAndrzej Warzynski case clang::driver::options::OPT_fdebug_unparse: 14123d4c4f3SAndrzej Warzynski opts.programAction = DebugUnparse; 14296d229c9SAndrzej Warzynski break; 143e81b3401SAndrzej Warzynski case clang::driver::options::OPT_fdebug_unparse_no_sema: 14423d4c4f3SAndrzej Warzynski opts.programAction = DebugUnparseNoSema; 145e81b3401SAndrzej Warzynski break; 14696d229c9SAndrzej Warzynski case clang::driver::options::OPT_fdebug_unparse_with_symbols: 14723d4c4f3SAndrzej Warzynski opts.programAction = DebugUnparseWithSymbols; 14896d229c9SAndrzej Warzynski break; 1494bd08dabSFaris Rehman case clang::driver::options::OPT_fdebug_dump_symbols: 15023d4c4f3SAndrzej Warzynski opts.programAction = DebugDumpSymbols; 1514bd08dabSFaris Rehman break; 1524bd08dabSFaris Rehman case clang::driver::options::OPT_fdebug_dump_parse_tree: 15323d4c4f3SAndrzej Warzynski opts.programAction = DebugDumpParseTree; 1544bd08dabSFaris Rehman break; 155a6be6e31SAndrzej Warzynski case clang::driver::options::OPT_fdebug_dump_all: 15623d4c4f3SAndrzej Warzynski opts.programAction = DebugDumpAll; 157a6be6e31SAndrzej Warzynski break; 158e81b3401SAndrzej Warzynski case clang::driver::options::OPT_fdebug_dump_parse_tree_no_sema: 15923d4c4f3SAndrzej Warzynski opts.programAction = DebugDumpParseTreeNoSema; 160e81b3401SAndrzej Warzynski break; 1614bd08dabSFaris Rehman case clang::driver::options::OPT_fdebug_dump_provenance: 16223d4c4f3SAndrzej Warzynski opts.programAction = DebugDumpProvenance; 1634bd08dabSFaris Rehman break; 164523d7bc6SAndrzej Warzynski case clang::driver::options::OPT_fdebug_dump_parsing_log: 16523d4c4f3SAndrzej Warzynski opts.programAction = DebugDumpParsingLog; 166523d7bc6SAndrzej Warzynski break; 167529f7181SFaris Rehman case clang::driver::options::OPT_fdebug_measure_parse_tree: 16823d4c4f3SAndrzej Warzynski opts.programAction = DebugMeasureParseTree; 169529f7181SFaris Rehman break; 170529f7181SFaris Rehman case clang::driver::options::OPT_fdebug_pre_fir_tree: 17123d4c4f3SAndrzej Warzynski opts.programAction = DebugPreFIRTree; 172529f7181SFaris Rehman break; 173eefda605SAndrzej Warzynski case clang::driver::options::OPT_fget_symbols_sources: 17423d4c4f3SAndrzej Warzynski opts.programAction = GetSymbolsSources; 175eefda605SAndrzej Warzynski break; 176dc256a44SAndrzej Warzynski case clang::driver::options::OPT_fget_definition: 17723d4c4f3SAndrzej Warzynski opts.programAction = GetDefinition; 178dc256a44SAndrzej Warzynski break; 179e1da3297SStuart Ellis case clang::driver::options::OPT_init_only: 18023d4c4f3SAndrzej Warzynski opts.programAction = InitOnly; 181e1da3297SStuart Ellis break; 182d28de0d7SCaroline Concatto 183257b2971SCaroline Concatto // TODO: 184e1da3297SStuart Ellis // case clang::driver::options::OPT_emit_llvm: 185257b2971SCaroline Concatto // case clang::driver::options::OPT_emit_llvm_only: 186257b2971SCaroline Concatto // case clang::driver::options::OPT_emit_codegen_only: 187257b2971SCaroline Concatto // case clang::driver::options::OPT_emit_module: 188257b2971SCaroline Concatto // (...) 189257b2971SCaroline Concatto } 190dc256a44SAndrzej Warzynski 191dc256a44SAndrzej Warzynski // Parse the values provided with `-fget-definition` (there should be 3 192dc256a44SAndrzej Warzynski // integers) 193dc256a44SAndrzej Warzynski if (llvm::opt::OptSpecifier(a->getOption().getID()) == 194dc256a44SAndrzej Warzynski clang::driver::options::OPT_fget_definition) { 195dc256a44SAndrzej Warzynski unsigned optVals[3] = {0, 0, 0}; 196dc256a44SAndrzej Warzynski 197dc256a44SAndrzej Warzynski for (unsigned i = 0; i < 3; i++) { 198dc256a44SAndrzej Warzynski llvm::StringRef val = a->getValue(i); 199dc256a44SAndrzej Warzynski 200dc256a44SAndrzej Warzynski if (val.getAsInteger(10, optVals[i])) { 201dc256a44SAndrzej Warzynski // A non-integer was encountered - that's an error. 202dc256a44SAndrzej Warzynski diags.Report(clang::diag::err_drv_invalid_value) 203dc256a44SAndrzej Warzynski << a->getOption().getName() << val; 204dc256a44SAndrzej Warzynski break; 205dc256a44SAndrzej Warzynski } 206dc256a44SAndrzej Warzynski } 20723d4c4f3SAndrzej Warzynski opts.getDefVals.line = optVals[0]; 20823d4c4f3SAndrzej Warzynski opts.getDefVals.startColumn = optVals[1]; 20923d4c4f3SAndrzej Warzynski opts.getDefVals.endColumn = optVals[2]; 210dc256a44SAndrzej Warzynski } 211257b2971SCaroline Concatto } 212257b2971SCaroline Concatto 213f52fc591SStuart Ellis // Parsing -load <dsopath> option and storing shared object path 214f52fc591SStuart Ellis if (llvm::opt::Arg *a = args.getLastArg(clang::driver::options::OPT_load)) { 215f52fc591SStuart Ellis opts.plugins.push_back(a->getValue()); 216f52fc591SStuart Ellis } 217f52fc591SStuart Ellis 218f52fc591SStuart Ellis // Parsing -plugin <name> option and storing plugin name and setting action 219f52fc591SStuart Ellis if (const llvm::opt::Arg *a = 220f52fc591SStuart Ellis args.getLastArg(clang::driver::options::OPT_plugin)) { 221f52fc591SStuart Ellis opts.programAction = PluginAction; 222f52fc591SStuart Ellis opts.ActionName = a->getValue(); 223f52fc591SStuart Ellis } 224f52fc591SStuart Ellis 22523d4c4f3SAndrzej Warzynski opts.outputFile = args.getLastArgValue(clang::driver::options::OPT_o); 22623d4c4f3SAndrzej Warzynski opts.showHelp = args.hasArg(clang::driver::options::OPT_help); 22723d4c4f3SAndrzej Warzynski opts.showVersion = args.hasArg(clang::driver::options::OPT_version); 228257b2971SCaroline Concatto 229257b2971SCaroline Concatto // Get the input kind (from the value passed via `-x`) 230257b2971SCaroline Concatto InputKind dashX(Language::Unknown); 231257b2971SCaroline Concatto if (const llvm::opt::Arg *a = 232257b2971SCaroline Concatto args.getLastArg(clang::driver::options::OPT_x)) { 233257b2971SCaroline Concatto llvm::StringRef XValue = a->getValue(); 234257b2971SCaroline Concatto // Principal languages. 235257b2971SCaroline Concatto dashX = llvm::StringSwitch<InputKind>(XValue) 236257b2971SCaroline Concatto .Case("f90", Language::Fortran) 237257b2971SCaroline Concatto .Default(Language::Unknown); 238257b2971SCaroline Concatto 239257b2971SCaroline Concatto // Some special cases cannot be combined with suffixes. 240257b2971SCaroline Concatto if (dashX.IsUnknown()) 241257b2971SCaroline Concatto dashX = llvm::StringSwitch<InputKind>(XValue) 242257b2971SCaroline Concatto .Case("ir", Language::LLVM_IR) 243257b2971SCaroline Concatto .Default(Language::Unknown); 244257b2971SCaroline Concatto 245257b2971SCaroline Concatto if (dashX.IsUnknown()) 246257b2971SCaroline Concatto diags.Report(clang::diag::err_drv_invalid_value) 247257b2971SCaroline Concatto << a->getAsString(args) << a->getValue(); 248257b2971SCaroline Concatto } 249257b2971SCaroline Concatto 2504c5906cfSCaroline Concatto // Collect the input files and save them in our instance of FrontendOptions. 2514c5906cfSCaroline Concatto std::vector<std::string> inputs = 2524c5906cfSCaroline Concatto args.getAllArgValues(clang::driver::options::OPT_INPUT); 25323d4c4f3SAndrzej Warzynski opts.inputs.clear(); 2544c5906cfSCaroline Concatto if (inputs.empty()) 2554c5906cfSCaroline Concatto // '-' is the default input if none is given. 2564c5906cfSCaroline Concatto inputs.push_back("-"); 2574c5906cfSCaroline Concatto for (unsigned i = 0, e = inputs.size(); i != e; ++i) { 2584c5906cfSCaroline Concatto InputKind ik = dashX; 2594c5906cfSCaroline Concatto if (ik.IsUnknown()) { 2604c5906cfSCaroline Concatto ik = FrontendOptions::GetInputKindForExtension( 2614c5906cfSCaroline Concatto llvm::StringRef(inputs[i]).rsplit('.').second); 2624c5906cfSCaroline Concatto if (ik.IsUnknown()) 2634c5906cfSCaroline Concatto ik = Language::Unknown; 2644c5906cfSCaroline Concatto if (i == 0) 2654c5906cfSCaroline Concatto dashX = ik; 2664c5906cfSCaroline Concatto } 2674c5906cfSCaroline Concatto 26823d4c4f3SAndrzej Warzynski opts.inputs.emplace_back(std::move(inputs[i]), ik); 2694c5906cfSCaroline Concatto } 2703a1513c1SFaris Rehman 27123d4c4f3SAndrzej Warzynski // Set fortranForm based on options -ffree-form and -ffixed-form. 2723a1513c1SFaris Rehman if (const auto *arg = args.getLastArg(clang::driver::options::OPT_ffixed_form, 2733a1513c1SFaris Rehman clang::driver::options::OPT_ffree_form)) { 27423d4c4f3SAndrzej Warzynski opts.fortranForm = 2753a1513c1SFaris Rehman arg->getOption().matches(clang::driver::options::OPT_ffixed_form) 2763a1513c1SFaris Rehman ? FortranForm::FixedForm 2773a1513c1SFaris Rehman : FortranForm::FreeForm; 2783a1513c1SFaris Rehman } 2793a1513c1SFaris Rehman 28023d4c4f3SAndrzej Warzynski // Set fixedFormColumns based on -ffixed-line-length=<value> 2813a1513c1SFaris Rehman if (const auto *arg = 2823a1513c1SFaris Rehman args.getLastArg(clang::driver::options::OPT_ffixed_line_length_EQ)) { 2833a1513c1SFaris Rehman llvm::StringRef argValue = llvm::StringRef(arg->getValue()); 2843a1513c1SFaris Rehman std::int64_t columns = -1; 2853a1513c1SFaris Rehman if (argValue == "none") { 2863a1513c1SFaris Rehman columns = 0; 2873a1513c1SFaris Rehman } else if (argValue.getAsInteger(/*Radix=*/10, columns)) { 2883a1513c1SFaris Rehman columns = -1; 2893a1513c1SFaris Rehman } 2903a1513c1SFaris Rehman if (columns < 0) { 291aefbfbcbSNick Desaulniers diags.Report(clang::diag::err_drv_negative_columns) 292aefbfbcbSNick Desaulniers << arg->getOption().getName() << arg->getValue(); 2933a1513c1SFaris Rehman } else if (columns == 0) { 29423d4c4f3SAndrzej Warzynski opts.fixedFormColumns = 1000000; 2953a1513c1SFaris Rehman } else if (columns < 7) { 296aefbfbcbSNick Desaulniers diags.Report(clang::diag::err_drv_small_columns) 297aefbfbcbSNick Desaulniers << arg->getOption().getName() << arg->getValue() << "7"; 2983a1513c1SFaris Rehman } else { 29923d4c4f3SAndrzej Warzynski opts.fixedFormColumns = columns; 3003a1513c1SFaris Rehman } 3013a1513c1SFaris Rehman } 3026d48a1a5SFaris Rehman 30355a96155SAndrzej Warzynski // -f{no-}implicit-none 30423d4c4f3SAndrzej Warzynski opts.features.Enable( 30510826ea7SFaris Rehman Fortran::common::LanguageFeature::ImplicitNoneTypeAlways, 30655a96155SAndrzej Warzynski args.hasFlag(clang::driver::options::OPT_fimplicit_none, 30755a96155SAndrzej Warzynski clang::driver::options::OPT_fno_implicit_none, false)); 30855a96155SAndrzej Warzynski 30955a96155SAndrzej Warzynski // -f{no-}backslash 31023d4c4f3SAndrzej Warzynski opts.features.Enable(Fortran::common::LanguageFeature::BackslashEscapes, 31155a96155SAndrzej Warzynski args.hasFlag(clang::driver::options::OPT_fbackslash, 31255a96155SAndrzej Warzynski clang::driver::options::OPT_fno_backslash, false)); 31355a96155SAndrzej Warzynski 31455a96155SAndrzej Warzynski // -f{no-}logical-abbreviations 31523d4c4f3SAndrzej Warzynski opts.features.Enable(Fortran::common::LanguageFeature::LogicalAbbreviations, 31655a96155SAndrzej Warzynski args.hasFlag(clang::driver::options::OPT_flogical_abbreviations, 31755a96155SAndrzej Warzynski clang::driver::options::OPT_fno_logical_abbreviations, false)); 31855a96155SAndrzej Warzynski 31955a96155SAndrzej Warzynski // -f{no-}xor-operator 32023d4c4f3SAndrzej Warzynski opts.features.Enable(Fortran::common::LanguageFeature::XOROperator, 32155a96155SAndrzej Warzynski args.hasFlag(clang::driver::options::OPT_fxor_operator, 32255a96155SAndrzej Warzynski clang::driver::options::OPT_fno_xor_operator, false)); 32355a96155SAndrzej Warzynski 324996ef895SPeter Klausler // -fno-automatic 325996ef895SPeter Klausler if (args.hasArg(clang::driver::options::OPT_fno_automatic)) { 326996ef895SPeter Klausler opts.features.Enable(Fortran::common::LanguageFeature::DefaultSave); 327996ef895SPeter Klausler } 328996ef895SPeter Klausler 32910826ea7SFaris Rehman if (args.hasArg( 33010826ea7SFaris Rehman clang::driver::options::OPT_falternative_parameter_statement)) { 33123d4c4f3SAndrzej Warzynski opts.features.Enable(Fortran::common::LanguageFeature::OldStyleParameter); 33210826ea7SFaris Rehman } 33310826ea7SFaris Rehman if (const llvm::opt::Arg *arg = 33410826ea7SFaris Rehman args.getLastArg(clang::driver::options::OPT_finput_charset_EQ)) { 33510826ea7SFaris Rehman llvm::StringRef argValue = arg->getValue(); 33610826ea7SFaris Rehman if (argValue == "utf-8") { 33723d4c4f3SAndrzej Warzynski opts.encoding = Fortran::parser::Encoding::UTF_8; 33810826ea7SFaris Rehman } else if (argValue == "latin-1") { 33923d4c4f3SAndrzej Warzynski opts.encoding = Fortran::parser::Encoding::LATIN_1; 34010826ea7SFaris Rehman } else { 34110826ea7SFaris Rehman diags.Report(clang::diag::err_drv_invalid_value) 34210826ea7SFaris Rehman << arg->getAsString(args) << argValue; 34310826ea7SFaris Rehman } 34410826ea7SFaris Rehman } 345523d7bc6SAndrzej Warzynski 346523d7bc6SAndrzej Warzynski setUpFrontendBasedOnAction(opts); 34723d4c4f3SAndrzej Warzynski opts.dashX = dashX; 348dc256a44SAndrzej Warzynski 349dc256a44SAndrzej Warzynski return diags.getNumErrors() == numErrorsBefore; 350257b2971SCaroline Concatto } 351257b2971SCaroline Concatto 352cd4abc52SArnamoy Bhattacharyya // Generate the path to look for intrinsic modules 353cd4abc52SArnamoy Bhattacharyya static std::string getIntrinsicDir() { 354cd4abc52SArnamoy Bhattacharyya // TODO: Find a system independent API 355cd4abc52SArnamoy Bhattacharyya llvm::SmallString<128> driverPath; 356cd4abc52SArnamoy Bhattacharyya driverPath.assign(llvm::sys::fs::getMainExecutable(nullptr, nullptr)); 357cd4abc52SArnamoy Bhattacharyya llvm::sys::path::remove_filename(driverPath); 358cd4abc52SArnamoy Bhattacharyya driverPath.append("/../include/flang/"); 359cd4abc52SArnamoy Bhattacharyya return std::string(driverPath); 360cd4abc52SArnamoy Bhattacharyya } 361cd4abc52SArnamoy Bhattacharyya 3627809fa20SFaris Rehman /// Parses all preprocessor input arguments and populates the preprocessor 3637809fa20SFaris Rehman /// options accordingly. 3647809fa20SFaris Rehman /// 3657809fa20SFaris Rehman /// \param [in] opts The preprocessor options instance 3667809fa20SFaris Rehman /// \param [out] args The list of input arguments 3677809fa20SFaris Rehman static void parsePreprocessorArgs( 3687809fa20SFaris Rehman Fortran::frontend::PreprocessorOptions &opts, llvm::opt::ArgList &args) { 3697809fa20SFaris Rehman // Add macros from the command line. 3707809fa20SFaris Rehman for (const auto *currentArg : args.filtered( 3717809fa20SFaris Rehman clang::driver::options::OPT_D, clang::driver::options::OPT_U)) { 3727809fa20SFaris Rehman if (currentArg->getOption().matches(clang::driver::options::OPT_D)) { 3737809fa20SFaris Rehman opts.addMacroDef(currentArg->getValue()); 3747809fa20SFaris Rehman } else { 3757809fa20SFaris Rehman opts.addMacroUndef(currentArg->getValue()); 3767809fa20SFaris Rehman } 3777809fa20SFaris Rehman } 37887dfd5e0SFaris Rehman 37987dfd5e0SFaris Rehman // Add the ordered list of -I's. 38087dfd5e0SFaris Rehman for (const auto *currentArg : args.filtered(clang::driver::options::OPT_I)) 38187dfd5e0SFaris Rehman opts.searchDirectoriesFromDashI.emplace_back(currentArg->getValue()); 382cd4abc52SArnamoy Bhattacharyya 383cd4abc52SArnamoy Bhattacharyya // Prepend the ordered list of -intrinsic-modules-path 384cd4abc52SArnamoy Bhattacharyya // to the default location to search. 385cd4abc52SArnamoy Bhattacharyya for (const auto *currentArg : 386cd4abc52SArnamoy Bhattacharyya args.filtered(clang::driver::options::OPT_fintrinsic_modules_path)) 387cd4abc52SArnamoy Bhattacharyya opts.searchDirectoriesFromIntrModPath.emplace_back(currentArg->getValue()); 388b83a4450SAndrzej Warzynski 389b83a4450SAndrzej Warzynski // -cpp/-nocpp 390b83a4450SAndrzej Warzynski if (const auto *currentArg = args.getLastArg( 391b83a4450SAndrzej Warzynski clang::driver::options::OPT_cpp, clang::driver::options::OPT_nocpp)) 39223d4c4f3SAndrzej Warzynski opts.macrosFlag = 393b83a4450SAndrzej Warzynski (currentArg->getOption().matches(clang::driver::options::OPT_cpp)) 394b83a4450SAndrzej Warzynski ? PPMacrosFlag::Include 395b83a4450SAndrzej Warzynski : PPMacrosFlag::Exclude; 3963338ef93Speter klausler 3973338ef93Speter klausler opts.noReformat = args.hasArg(clang::driver::options::OPT_fno_reformat); 3983338ef93Speter klausler opts.noLineDirectives = args.hasArg(clang::driver::options::OPT_P); 3997809fa20SFaris Rehman } 4007809fa20SFaris Rehman 401985a42fdSArnamoy Bhattacharyya /// Parses all semantic related arguments and populates the variables 4026d0fef48SAndrzej Warzynski /// options accordingly. Returns false if new errors are generated. 4036d0fef48SAndrzej Warzynski static bool parseSemaArgs(CompilerInvocation &res, llvm::opt::ArgList &args, 404985a42fdSArnamoy Bhattacharyya clang::DiagnosticsEngine &diags) { 4056d0fef48SAndrzej Warzynski unsigned numErrorsBefore = diags.getNumErrors(); 406985a42fdSArnamoy Bhattacharyya 4071fd4beecSArnamoy Bhattacharyya // -J/module-dir option 408985a42fdSArnamoy Bhattacharyya auto moduleDirList = 409985a42fdSArnamoy Bhattacharyya args.getAllArgValues(clang::driver::options::OPT_module_dir); 410985a42fdSArnamoy Bhattacharyya // User can only specify -J/-module-dir once 411985a42fdSArnamoy Bhattacharyya // https://gcc.gnu.org/onlinedocs/gfortran/Directory-Options.html 412985a42fdSArnamoy Bhattacharyya if (moduleDirList.size() > 1) { 413985a42fdSArnamoy Bhattacharyya const unsigned diagID = 414985a42fdSArnamoy Bhattacharyya diags.getCustomDiagID(clang::DiagnosticsEngine::Error, 415985a42fdSArnamoy Bhattacharyya "Only one '-module-dir/-J' option allowed"); 416985a42fdSArnamoy Bhattacharyya diags.Report(diagID); 417985a42fdSArnamoy Bhattacharyya } 418985a42fdSArnamoy Bhattacharyya if (moduleDirList.size() == 1) 4191fd4beecSArnamoy Bhattacharyya res.SetModuleDir(moduleDirList[0]); 4201fd4beecSArnamoy Bhattacharyya 4211fd4beecSArnamoy Bhattacharyya // -fdebug-module-writer option 4221fd4beecSArnamoy Bhattacharyya if (args.hasArg(clang::driver::options::OPT_fdebug_module_writer)) { 4231fd4beecSArnamoy Bhattacharyya res.SetDebugModuleDir(true); 4241fd4beecSArnamoy Bhattacharyya } 4256d0fef48SAndrzej Warzynski 42620bd2142SAndrzej Warzynski // -module-suffix 42720bd2142SAndrzej Warzynski if (const auto *moduleSuffix = 42820bd2142SAndrzej Warzynski args.getLastArg(clang::driver::options::OPT_module_suffix)) { 42920bd2142SAndrzej Warzynski res.SetModuleFileSuffix(moduleSuffix->getValue()); 43020bd2142SAndrzej Warzynski } 43120bd2142SAndrzej Warzynski 43255a96155SAndrzej Warzynski // -f{no-}analyzed-objects-for-unparse 43355a96155SAndrzej Warzynski res.SetUseAnalyzedObjectsForUnparse( 43455a96155SAndrzej Warzynski args.hasFlag(clang::driver::options::OPT_fanalyzed_objects_for_unparse, 43555a96155SAndrzej Warzynski clang::driver::options::OPT_fno_analyzed_objects_for_unparse, true)); 4362a7bb849SAndrzej Warzynski 4376d0fef48SAndrzej Warzynski return diags.getNumErrors() == numErrorsBefore; 438985a42fdSArnamoy Bhattacharyya } 439985a42fdSArnamoy Bhattacharyya 4407416e8a8SArnamoy Bhattacharyya /// Parses all diagnostics related arguments and populates the variables 4416d0fef48SAndrzej Warzynski /// options accordingly. Returns false if new errors are generated. 4426d0fef48SAndrzej Warzynski static bool parseDiagArgs(CompilerInvocation &res, llvm::opt::ArgList &args, 4437416e8a8SArnamoy Bhattacharyya clang::DiagnosticsEngine &diags) { 4446d0fef48SAndrzej Warzynski unsigned numErrorsBefore = diags.getNumErrors(); 4456d0fef48SAndrzej Warzynski 4467416e8a8SArnamoy Bhattacharyya // -Werror option 4477416e8a8SArnamoy Bhattacharyya // TODO: Currently throws a Diagnostic for anything other than -W<error>, 4487416e8a8SArnamoy Bhattacharyya // this has to change when other -W<opt>'s are supported. 4497416e8a8SArnamoy Bhattacharyya if (args.hasArg(clang::driver::options::OPT_W_Joined)) { 4507416e8a8SArnamoy Bhattacharyya if (args.getLastArgValue(clang::driver::options::OPT_W_Joined) 4517416e8a8SArnamoy Bhattacharyya .equals("error")) { 4527416e8a8SArnamoy Bhattacharyya res.SetWarnAsErr(true); 4537416e8a8SArnamoy Bhattacharyya } else { 4547416e8a8SArnamoy Bhattacharyya const unsigned diagID = 4557416e8a8SArnamoy Bhattacharyya diags.getCustomDiagID(clang::DiagnosticsEngine::Error, 4567416e8a8SArnamoy Bhattacharyya "Only `-Werror` is supported currently."); 4577416e8a8SArnamoy Bhattacharyya diags.Report(diagID); 4587416e8a8SArnamoy Bhattacharyya } 4597416e8a8SArnamoy Bhattacharyya } 4606d0fef48SAndrzej Warzynski 4616d0fef48SAndrzej Warzynski return diags.getNumErrors() == numErrorsBefore; 4627416e8a8SArnamoy Bhattacharyya } 4637416e8a8SArnamoy Bhattacharyya 464ab971c29SArnamoy Bhattacharyya /// Parses all Dialect related arguments and populates the variables 4656d0fef48SAndrzej Warzynski /// options accordingly. Returns false if new errors are generated. 4666d0fef48SAndrzej Warzynski static bool parseDialectArgs(CompilerInvocation &res, llvm::opt::ArgList &args, 467ab971c29SArnamoy Bhattacharyya clang::DiagnosticsEngine &diags) { 4686d0fef48SAndrzej Warzynski unsigned numErrorsBefore = diags.getNumErrors(); 469ab971c29SArnamoy Bhattacharyya 470ab971c29SArnamoy Bhattacharyya // -fdefault* family 471ab971c29SArnamoy Bhattacharyya if (args.hasArg(clang::driver::options::OPT_fdefault_real_8)) { 472ab971c29SArnamoy Bhattacharyya res.defaultKinds().set_defaultRealKind(8); 473ab971c29SArnamoy Bhattacharyya res.defaultKinds().set_doublePrecisionKind(16); 474ab971c29SArnamoy Bhattacharyya } 475ab971c29SArnamoy Bhattacharyya if (args.hasArg(clang::driver::options::OPT_fdefault_integer_8)) { 476ab971c29SArnamoy Bhattacharyya res.defaultKinds().set_defaultIntegerKind(8); 477ab971c29SArnamoy Bhattacharyya res.defaultKinds().set_subscriptIntegerKind(8); 478ab971c29SArnamoy Bhattacharyya res.defaultKinds().set_sizeIntegerKind(8); 479ab971c29SArnamoy Bhattacharyya } 480ab971c29SArnamoy Bhattacharyya if (args.hasArg(clang::driver::options::OPT_fdefault_double_8)) { 481ab971c29SArnamoy Bhattacharyya if (!args.hasArg(clang::driver::options::OPT_fdefault_real_8)) { 482ab971c29SArnamoy Bhattacharyya // -fdefault-double-8 has to be used with -fdefault-real-8 483ab971c29SArnamoy Bhattacharyya // to be compatible with gfortran 484ab971c29SArnamoy Bhattacharyya const unsigned diagID = 485ab971c29SArnamoy Bhattacharyya diags.getCustomDiagID(clang::DiagnosticsEngine::Error, 486ab971c29SArnamoy Bhattacharyya "Use of `-fdefault-double-8` requires `-fdefault-real-8`"); 487ab971c29SArnamoy Bhattacharyya diags.Report(diagID); 488ab971c29SArnamoy Bhattacharyya } 489ab971c29SArnamoy Bhattacharyya // https://gcc.gnu.org/onlinedocs/gfortran/Fortran-Dialect-Options.html 490ab971c29SArnamoy Bhattacharyya res.defaultKinds().set_doublePrecisionKind(8); 491ab971c29SArnamoy Bhattacharyya } 492ab971c29SArnamoy Bhattacharyya if (args.hasArg(clang::driver::options::OPT_flarge_sizes)) 493ab971c29SArnamoy Bhattacharyya res.defaultKinds().set_sizeIntegerKind(8); 494ab971c29SArnamoy Bhattacharyya 495ab971c29SArnamoy Bhattacharyya // -fopenmp and -fopenacc 496ab971c29SArnamoy Bhattacharyya if (args.hasArg(clang::driver::options::OPT_fopenacc)) { 49723d4c4f3SAndrzej Warzynski res.frontendOpts().features.Enable( 498ab971c29SArnamoy Bhattacharyya Fortran::common::LanguageFeature::OpenACC); 499ab971c29SArnamoy Bhattacharyya } 500ab971c29SArnamoy Bhattacharyya if (args.hasArg(clang::driver::options::OPT_fopenmp)) { 50123d4c4f3SAndrzej Warzynski res.frontendOpts().features.Enable( 502ab971c29SArnamoy Bhattacharyya Fortran::common::LanguageFeature::OpenMP); 503ab971c29SArnamoy Bhattacharyya } 5044c7ebf79SArnamoy Bhattacharyya 505fcf629d7SAndrzej Warzynski // -pedantic 5064c7ebf79SArnamoy Bhattacharyya if (args.hasArg(clang::driver::options::OPT_pedantic)) { 5074c7ebf79SArnamoy Bhattacharyya res.set_EnableConformanceChecks(); 5084c7ebf79SArnamoy Bhattacharyya } 509fcf629d7SAndrzej Warzynski // -std=f2018 (currently this implies -pedantic) 5104c7ebf79SArnamoy Bhattacharyya // TODO: Set proper options when more fortran standards 5114c7ebf79SArnamoy Bhattacharyya // are supported. 5124c7ebf79SArnamoy Bhattacharyya if (args.hasArg(clang::driver::options::OPT_std_EQ)) { 5134c7ebf79SArnamoy Bhattacharyya auto standard = args.getLastArgValue(clang::driver::options::OPT_std_EQ); 5144c7ebf79SArnamoy Bhattacharyya // We only allow f2018 as the given standard 5154c7ebf79SArnamoy Bhattacharyya if (standard.equals("f2018")) { 5164c7ebf79SArnamoy Bhattacharyya res.set_EnableConformanceChecks(); 5174c7ebf79SArnamoy Bhattacharyya } else { 5184c7ebf79SArnamoy Bhattacharyya const unsigned diagID = 5194c7ebf79SArnamoy Bhattacharyya diags.getCustomDiagID(clang::DiagnosticsEngine::Error, 5204c7ebf79SArnamoy Bhattacharyya "Only -std=f2018 is allowed currently."); 5214c7ebf79SArnamoy Bhattacharyya diags.Report(diagID); 5224c7ebf79SArnamoy Bhattacharyya } 5234c7ebf79SArnamoy Bhattacharyya } 5246d0fef48SAndrzej Warzynski return diags.getNumErrors() == numErrorsBefore; 525ab971c29SArnamoy Bhattacharyya } 526ab971c29SArnamoy Bhattacharyya 527257b2971SCaroline Concatto bool CompilerInvocation::CreateFromArgs(CompilerInvocation &res, 528257b2971SCaroline Concatto llvm::ArrayRef<const char *> commandLineArgs, 529257b2971SCaroline Concatto clang::DiagnosticsEngine &diags) { 530257b2971SCaroline Concatto 531257b2971SCaroline Concatto bool success = true; 532257b2971SCaroline Concatto 533257b2971SCaroline Concatto // Parse the arguments 534257b2971SCaroline Concatto const llvm::opt::OptTable &opts = clang::driver::getDriverOptTable(); 5358e3adda8SFaris Rehman const unsigned includedFlagsBitmask = clang::driver::options::FC1Option; 536257b2971SCaroline Concatto unsigned missingArgIndex, missingArgCount; 537257b2971SCaroline Concatto llvm::opt::InputArgList args = opts.ParseArgs( 538257b2971SCaroline Concatto commandLineArgs, missingArgIndex, missingArgCount, includedFlagsBitmask); 539257b2971SCaroline Concatto 5402b4c9bc4SAndrzej Warzynski // Check for missing argument error. 5412b4c9bc4SAndrzej Warzynski if (missingArgCount) { 5422b4c9bc4SAndrzej Warzynski diags.Report(clang::diag::err_drv_missing_argument) 5432b4c9bc4SAndrzej Warzynski << args.getArgString(missingArgIndex) << missingArgCount; 5442b4c9bc4SAndrzej Warzynski success = false; 5452b4c9bc4SAndrzej Warzynski } 5462b4c9bc4SAndrzej Warzynski 547257b2971SCaroline Concatto // Issue errors on unknown arguments 548257b2971SCaroline Concatto for (const auto *a : args.filtered(clang::driver::options::OPT_UNKNOWN)) { 549257b2971SCaroline Concatto auto argString = a->getAsString(args); 550257b2971SCaroline Concatto std::string nearest; 551257b2971SCaroline Concatto if (opts.findNearest(argString, nearest, includedFlagsBitmask) > 1) 552257b2971SCaroline Concatto diags.Report(clang::diag::err_drv_unknown_argument) << argString; 553257b2971SCaroline Concatto else 554257b2971SCaroline Concatto diags.Report(clang::diag::err_drv_unknown_argument_with_suggestion) 555257b2971SCaroline Concatto << argString << nearest; 556257b2971SCaroline Concatto success = false; 557257b2971SCaroline Concatto } 558257b2971SCaroline Concatto 559dc256a44SAndrzej Warzynski success &= ParseFrontendArgs(res.frontendOpts(), args, diags); 5607809fa20SFaris Rehman parsePreprocessorArgs(res.preprocessorOpts(), args); 5616d0fef48SAndrzej Warzynski success &= parseSemaArgs(res, args, diags); 5626d0fef48SAndrzej Warzynski success &= parseDialectArgs(res, args, diags); 5636d0fef48SAndrzej Warzynski success &= parseDiagArgs(res, args, diags); 564257b2971SCaroline Concatto 565257b2971SCaroline Concatto return success; 566257b2971SCaroline Concatto } 567d28de0d7SCaroline Concatto 568b83a4450SAndrzej Warzynski void CompilerInvocation::collectMacroDefinitions() { 569b83a4450SAndrzej Warzynski auto &ppOpts = this->preprocessorOpts(); 570b83a4450SAndrzej Warzynski 5717809fa20SFaris Rehman for (unsigned i = 0, n = ppOpts.macros.size(); i != n; ++i) { 5727809fa20SFaris Rehman llvm::StringRef macro = ppOpts.macros[i].first; 5737809fa20SFaris Rehman bool isUndef = ppOpts.macros[i].second; 5747809fa20SFaris Rehman 5757809fa20SFaris Rehman std::pair<llvm::StringRef, llvm::StringRef> macroPair = macro.split('='); 5767809fa20SFaris Rehman llvm::StringRef macroName = macroPair.first; 5777809fa20SFaris Rehman llvm::StringRef macroBody = macroPair.second; 5787809fa20SFaris Rehman 5797809fa20SFaris Rehman // For an #undef'd macro, we only care about the name. 5807809fa20SFaris Rehman if (isUndef) { 581b83a4450SAndrzej Warzynski parserOpts_.predefinitions.emplace_back( 5827809fa20SFaris Rehman macroName.str(), std::optional<std::string>{}); 5837809fa20SFaris Rehman continue; 5847809fa20SFaris Rehman } 5857809fa20SFaris Rehman 5867809fa20SFaris Rehman // For a #define'd macro, figure out the actual definition. 5877809fa20SFaris Rehman if (macroName.size() == macro.size()) 5887809fa20SFaris Rehman macroBody = "1"; 5897809fa20SFaris Rehman else { 5907809fa20SFaris Rehman // Note: GCC drops anything following an end-of-line character. 5917809fa20SFaris Rehman llvm::StringRef::size_type End = macroBody.find_first_of("\n\r"); 5927809fa20SFaris Rehman macroBody = macroBody.substr(0, End); 5937809fa20SFaris Rehman } 594b83a4450SAndrzej Warzynski parserOpts_.predefinitions.emplace_back( 5957809fa20SFaris Rehman macroName, std::optional<std::string>(macroBody.str())); 5967809fa20SFaris Rehman } 5977809fa20SFaris Rehman } 5987809fa20SFaris Rehman 599d28de0d7SCaroline Concatto void CompilerInvocation::SetDefaultFortranOpts() { 6006bbbe4a5SAndrzej Warzynski auto &fortranOptions = fortranOpts(); 601d28de0d7SCaroline Concatto 602d28de0d7SCaroline Concatto std::vector<std::string> searchDirectories{"."s}; 603d28de0d7SCaroline Concatto fortranOptions.searchDirectories = searchDirectories; 604d28de0d7SCaroline Concatto fortranOptions.isFixedForm = false; 6050feff71eSAndrzej Warzynski } 6060feff71eSAndrzej Warzynski 6070feff71eSAndrzej Warzynski // TODO: When expanding this method, consider creating a dedicated API for 6080feff71eSAndrzej Warzynski // this. Also at some point we will need to differentiate between different 6090feff71eSAndrzej Warzynski // targets and add dedicated predefines for each. 6100feff71eSAndrzej Warzynski void CompilerInvocation::setDefaultPredefinitions() { 6110feff71eSAndrzej Warzynski auto &fortranOptions = fortranOpts(); 6120feff71eSAndrzej Warzynski const auto &frontendOptions = frontendOpts(); 613197d9a55SFaris Rehman 614197d9a55SFaris Rehman // Populate the macro list with version numbers and other predefinitions. 615197d9a55SFaris Rehman fortranOptions.predefinitions.emplace_back("__flang__", "1"); 616197d9a55SFaris Rehman fortranOptions.predefinitions.emplace_back( 617197d9a55SFaris Rehman "__flang_major__", FLANG_VERSION_MAJOR_STRING); 618197d9a55SFaris Rehman fortranOptions.predefinitions.emplace_back( 619197d9a55SFaris Rehman "__flang_minor__", FLANG_VERSION_MINOR_STRING); 620197d9a55SFaris Rehman fortranOptions.predefinitions.emplace_back( 621197d9a55SFaris Rehman "__flang_patchlevel__", FLANG_VERSION_PATCHLEVEL_STRING); 6226d48a1a5SFaris Rehman 6236d48a1a5SFaris Rehman // Add predefinitions based on extensions enabled 62423d4c4f3SAndrzej Warzynski if (frontendOptions.features.IsEnabled( 6256d48a1a5SFaris Rehman Fortran::common::LanguageFeature::OpenACC)) { 6266d48a1a5SFaris Rehman fortranOptions.predefinitions.emplace_back("_OPENACC", "202011"); 6276d48a1a5SFaris Rehman } 62823d4c4f3SAndrzej Warzynski if (frontendOptions.features.IsEnabled( 6296d48a1a5SFaris Rehman Fortran::common::LanguageFeature::OpenMP)) { 6306d48a1a5SFaris Rehman fortranOptions.predefinitions.emplace_back("_OPENMP", "201511"); 6316d48a1a5SFaris Rehman } 6326d48a1a5SFaris Rehman } 6336d48a1a5SFaris Rehman 6347809fa20SFaris Rehman void CompilerInvocation::setFortranOpts() { 6357809fa20SFaris Rehman auto &fortranOptions = fortranOpts(); 6363a1513c1SFaris Rehman const auto &frontendOptions = frontendOpts(); 6377809fa20SFaris Rehman const auto &preprocessorOptions = preprocessorOpts(); 638985a42fdSArnamoy Bhattacharyya auto &moduleDirJ = moduleDir(); 6397809fa20SFaris Rehman 64023d4c4f3SAndrzej Warzynski if (frontendOptions.fortranForm != FortranForm::Unknown) { 6413a1513c1SFaris Rehman fortranOptions.isFixedForm = 64223d4c4f3SAndrzej Warzynski frontendOptions.fortranForm == FortranForm::FixedForm; 6433a1513c1SFaris Rehman } 64423d4c4f3SAndrzej Warzynski fortranOptions.fixedFormColumns = frontendOptions.fixedFormColumns; 6453a1513c1SFaris Rehman 64623d4c4f3SAndrzej Warzynski fortranOptions.features = frontendOptions.features; 64723d4c4f3SAndrzej Warzynski fortranOptions.encoding = frontendOptions.encoding; 6486d48a1a5SFaris Rehman 649cd4abc52SArnamoy Bhattacharyya // Adding search directories specified by -I 65087dfd5e0SFaris Rehman fortranOptions.searchDirectories.insert( 65187dfd5e0SFaris Rehman fortranOptions.searchDirectories.end(), 65287dfd5e0SFaris Rehman preprocessorOptions.searchDirectoriesFromDashI.begin(), 65387dfd5e0SFaris Rehman preprocessorOptions.searchDirectoriesFromDashI.end()); 654985a42fdSArnamoy Bhattacharyya 655cd4abc52SArnamoy Bhattacharyya // Add the ordered list of -intrinsic-modules-path 656cd4abc52SArnamoy Bhattacharyya fortranOptions.searchDirectories.insert( 657cd4abc52SArnamoy Bhattacharyya fortranOptions.searchDirectories.end(), 658cd4abc52SArnamoy Bhattacharyya preprocessorOptions.searchDirectoriesFromIntrModPath.begin(), 659cd4abc52SArnamoy Bhattacharyya preprocessorOptions.searchDirectoriesFromIntrModPath.end()); 660cd4abc52SArnamoy Bhattacharyya 661cd4abc52SArnamoy Bhattacharyya // Add the default intrinsic module directory at the end 662cd4abc52SArnamoy Bhattacharyya fortranOptions.searchDirectories.emplace_back(getIntrinsicDir()); 663cd4abc52SArnamoy Bhattacharyya 664985a42fdSArnamoy Bhattacharyya // Add the directory supplied through -J/-module-dir to the list of search 665985a42fdSArnamoy Bhattacharyya // directories 666985a42fdSArnamoy Bhattacharyya if (moduleDirJ.compare(".") != 0) 667985a42fdSArnamoy Bhattacharyya fortranOptions.searchDirectories.emplace_back(moduleDirJ); 668523d7bc6SAndrzej Warzynski 66923d4c4f3SAndrzej Warzynski if (frontendOptions.instrumentedParse) 670523d7bc6SAndrzej Warzynski fortranOptions.instrumentedParse = true; 6714c7ebf79SArnamoy Bhattacharyya 67223d4c4f3SAndrzej Warzynski if (frontendOptions.needProvenanceRangeToCharBlockMappings) 67385b86c6fSAndrzej Warzynski fortranOptions.needProvenanceRangeToCharBlockMappings = true; 67485b86c6fSAndrzej Warzynski 6754c7ebf79SArnamoy Bhattacharyya if (enableConformanceChecks()) { 6764c7ebf79SArnamoy Bhattacharyya fortranOptions.features.WarnOnAllNonstandard(); 6774c7ebf79SArnamoy Bhattacharyya } 678985a42fdSArnamoy Bhattacharyya } 679985a42fdSArnamoy Bhattacharyya 680985a42fdSArnamoy Bhattacharyya void CompilerInvocation::setSemanticsOpts( 6816d48a1a5SFaris Rehman Fortran::parser::AllCookedSources &allCookedSources) { 6826d48a1a5SFaris Rehman const auto &fortranOptions = fortranOpts(); 6836d48a1a5SFaris Rehman 6846d48a1a5SFaris Rehman semanticsContext_ = std::make_unique<semantics::SemanticsContext>( 685ab971c29SArnamoy Bhattacharyya defaultKinds(), fortranOptions.features, allCookedSources); 6866d48a1a5SFaris Rehman 6871fd4beecSArnamoy Bhattacharyya semanticsContext_->set_moduleDirectory(moduleDir()) 6884c7ebf79SArnamoy Bhattacharyya .set_searchDirectories(fortranOptions.searchDirectories) 6897416e8a8SArnamoy Bhattacharyya .set_warnOnNonstandardUsage(enableConformanceChecks()) 69020bd2142SAndrzej Warzynski .set_warningsAreErrors(warnAsErr()) 69120bd2142SAndrzej Warzynski .set_moduleFileSuffix(moduleFileSuffix()); 6927809fa20SFaris Rehman } 693