diff --git a/flang/include/flang/Frontend/CodeGenOptions.h b/flang/include/flang/Frontend/CodeGenOptions.h --- a/flang/include/flang/Frontend/CodeGenOptions.h +++ b/flang/include/flang/Frontend/CodeGenOptions.h @@ -69,6 +69,57 @@ /// The format used for serializing remarks (default: YAML) std::string OptRecordFormat; + // The RemarkKind enum class and OptRemark struct are identical to what Clang + // has + // TODO: Share with clang instead of re-implementing here + enum class RemarkKind { + RK_Missing, // Remark argument not present on the command line. + RK_Enabled, // Remark enabled via '-Rgroup', i.e. -Rpass, -Rpass-missed, + // -Rpass-analysis + RK_Disabled, // Remark disabled via '-Rno-group', i.e. -Rno-pass, + // -Rno-pass-missed, -Rno-pass-analysis. + }; + + /// Optimization remark with an optional regular expression pattern. + struct OptRemark { + RemarkKind Kind = RemarkKind::RK_Missing; + std::string Pattern; + std::shared_ptr Regex; + + /// By default, optimization remark is missing. + OptRemark() = default; + + /// Returns true iff the optimization remark holds a valid regular + /// expression. + bool hasValidPattern() const { return Regex != nullptr; } + + /// Matches the given string against the regex, if there is some. + bool patternMatches(llvm::StringRef string) const { + return hasValidPattern() && Regex->match(string); + } + }; + + // The OptRemark fields provided here are identical to Clang. + + /// Selected optimizations for which we should enable optimization remarks. + /// Transformation passes whose name matches the contained (optional) regular + /// expression (and support this feature), will emit a diagnostic whenever + /// they perform a transformation. + OptRemark OptimizationRemark; + + /// Selected optimizations for which we should enable missed optimization + /// remarks. Transformation passes whose name matches the contained (optional) + /// regular expression (and support this feature), will emit a diagnostic + /// whenever they tried but failed to perform a transformation. + OptRemark OptimizationRemarkMissed; + + /// Selected optimizations for which we should enable optimization analyses. + /// Transformation passes whose name matches the contained (optional) regular + /// expression (and support this feature), will emit a diagnostic whenever + /// they want to explain why they decided to apply or not apply a given + /// transformation. + OptRemark OptimizationRemarkAnalysis; + // Define accessors/mutators for code generation options of enumeration type. #define CODEGENOPT(Name, Bits, Default) #define ENUM_CODEGENOPT(Name, Type, Bits, Default) \ diff --git a/flang/lib/Frontend/CompilerInvocation.cpp b/flang/lib/Frontend/CompilerInvocation.cpp --- a/flang/lib/Frontend/CompilerInvocation.cpp +++ b/flang/lib/Frontend/CompilerInvocation.cpp @@ -153,6 +153,36 @@ return true; } +static CodeGenOptions::OptRemark +parseOptimizationRemark(clang::DiagnosticsEngine &diags, + llvm::opt::ArgList &args, llvm::opt::OptSpecifier optEq, + llvm::StringRef name) { + CodeGenOptions::OptRemark result; + + for (llvm::opt::Arg *a : args) { + if (a->getOption().matches(clang::driver::options::OPT_R_Joined)) { + llvm::StringRef value = a->getValue(); + + if (value == name) + result.Kind = CodeGenOptions::RemarkKind::RK_Enabled; + else if (value.split('-') == std::make_pair(llvm::StringRef("no"), name)) + result.Kind = CodeGenOptions::RemarkKind::RK_Disabled; + else + continue; + + if (result.Kind == CodeGenOptions::RemarkKind::RK_Disabled) { + result.Pattern = ""; + result.Regex = nullptr; + } else { + result.Pattern = ".*"; + result.Regex = std::make_shared(result.Pattern); + } + } + } + + return result; +} + static void parseCodeGenArgs(Fortran::frontend::CodeGenOptions &opts, llvm::opt::ArgList &args, clang::DiagnosticsEngine &diags) { @@ -194,13 +224,55 @@ args.getLastArg(clang::driver::options::OPT_opt_record_file)) opts.OptRecordFile = a->getValue(); + // Specifies, using a regex, which successful optimization passes(middle and + // backend), to include in the final optimization record file generated. If + // not provided -fsave-optimization-record will include all passes. + if (const llvm::opt::Arg *a = + args.getLastArg(clang::driver::options::OPT_opt_record_passes)) + opts.OptRecordPasses = a->getValue(); + + // Optimization file format. Defaults to yaml if (const llvm::opt::Arg *a = args.getLastArg(clang::driver::options::OPT_opt_record_format)) opts.OptRecordFormat = a->getValue(); - if (const llvm::opt::Arg *a = - args.getLastArg(clang::driver::options::OPT_opt_record_passes)) - opts.OptRecordPasses = a->getValue(); + // OptimizationRemark, OptimizationRemarkMissed and OptimizationRemarkAnalysis + // contain regex values which are used in optimizationRemarkHandler in + // FrontendActions.cpp to determine which remarks generated should be outputed + // to console. They all have the type OptRemark + + // Get -Rpass option regex. If empty, "".*"" is used. From all successful + // optimization passes applied, the regex will return only pass names that + // match it. + opts.OptimizationRemark = parseOptimizationRemark( + diags, args, clang::driver::options::OPT_Rpass_EQ, "pass"); + + // Get -Rpass-missed option regex. If empty, "".*"" is used. From all + // optimization passes that failed to be applied, the regex will return only + // pass names that match it. + opts.OptimizationRemarkMissed = parseOptimizationRemark( + diags, args, clang::driver::options::OPT_Rpass_missed_EQ, "pass-missed"); + + // Specify which passes, with additional information, + // should be reported using a regex. + opts.OptimizationRemarkAnalysis = parseOptimizationRemark( + diags, args, clang::driver::options::OPT_Rpass_analysis_EQ, + "pass-analysis"); + + if (opts.getDebugInfo() == llvm::codegenoptions::NoDebugInfo) { + // If the user requested a flag that requires source locations available in + // the backend, make sure that the backend tracks source location + // information. + bool needLocTracking = !opts.OptRecordFile.empty() || + !opts.OptRecordPasses.empty() || + !opts.OptRecordFormat.empty() || + opts.OptimizationRemark.hasValidPattern() || + opts.OptimizationRemarkMissed.hasValidPattern() || + opts.OptimizationRemarkAnalysis.hasValidPattern(); + + if (needLocTracking) + opts.setDebugInfo(llvm::codegenoptions::LocTrackingOnly); + } if (auto *a = args.getLastArg(clang::driver::options::OPT_save_temps_EQ)) opts.SaveTempsDir = a->getValue(); @@ -959,6 +1031,12 @@ res.loweringOpts.setNoPPCNativeVecElemOrder(true); } + // Preserve all the remark options requested, i.e. -Rpass, -Rpass-missed or + // -Rpass-analysis. This will be used later when processing and outputting the + // remarks generated by LLVM in ExecuteCompilerInvocation.cpp. + for (auto *a : args.filtered(clang::driver::options::OPT_R_Group)) + res.getDiagnosticOpts().Remarks.push_back(a->getValue()); + success &= parseFrontendArgs(res.getFrontendOpts(), args, diags); parseTargetArgs(res.getTargetOpts(), args); parsePreprocessorArgs(res.getPreprocessorOpts(), args); diff --git a/flang/lib/Frontend/FrontendActions.cpp b/flang/lib/Frontend/FrontendActions.cpp --- a/flang/lib/Frontend/FrontendActions.cpp +++ b/flang/lib/Frontend/FrontendActions.cpp @@ -48,6 +48,8 @@ #include "llvm/Analysis/TargetLibraryInfo.h" #include "llvm/Analysis/TargetTransformInfo.h" #include "llvm/Bitcode/BitcodeWriterPass.h" +#include "llvm/CodeGen/MachineOptimizationRemarkEmitter.h" +#include "llvm/IR/DiagnosticPrinter.h" #include "llvm/IR/LLVMRemarkStreamer.h" #include "llvm/IR/LegacyPassManager.h" #include "llvm/IR/Verifier.h" @@ -919,6 +921,133 @@ mpm.run(*llvmModule, mam); } +// This class handles optimization remark messages requested if +// any of -Rpass, -Rpass-analysis or -Rpass-missed flags were provided +class BackendRemarkConsumer : public llvm::DiagnosticHandler { + + const CodeGenOptions &codeGenOpts; + clang::DiagnosticsEngine &diags; + +public: + BackendRemarkConsumer(clang::DiagnosticsEngine &diags, + const CodeGenOptions &codeGenOpts) + : codeGenOpts(codeGenOpts), diags(diags) {} + + bool isAnalysisRemarkEnabled(llvm::StringRef passName) const override { + return codeGenOpts.OptimizationRemarkAnalysis.patternMatches(passName); + } + bool isMissedOptRemarkEnabled(llvm::StringRef passName) const override { + return codeGenOpts.OptimizationRemarkMissed.patternMatches(passName); + } + bool isPassedOptRemarkEnabled(llvm::StringRef passName) const override { + return codeGenOpts.OptimizationRemark.patternMatches(passName); + } + + bool isAnyRemarkEnabled() const override { + return codeGenOpts.OptimizationRemarkAnalysis.hasValidPattern() || + codeGenOpts.OptimizationRemarkMissed.hasValidPattern() || + codeGenOpts.OptimizationRemark.hasValidPattern(); + } + + void + emitOptimizationMessage(const llvm::DiagnosticInfoOptimizationBase &diagInfo, + unsigned diagID) { + // We only support warnings and remarks. + assert(diagInfo.getSeverity() == llvm::DS_Remark || + diagInfo.getSeverity() == llvm::DS_Warning); + + std::string msg; + llvm::raw_string_ostream msgStream(msg); + + if (diagInfo.isLocationAvailable()) { + // Since sourceMgr isn't available, send file name and absolute path + // through msgStream, to use for printing + msgStream << diagInfo.getLocationStr() << ";;" + << diagInfo.getAbsolutePath() << ";;"; + msgStream << diagInfo.getMsg(); + } else { + msgStream << diagInfo.getMsg(); + llvm::DiagnosticPrinterRawOStream dp(msgStream); + diagInfo.print(dp); + } + + // Emit message. + diags.Report(diagID) << clang::AddFlagValue(diagInfo.getPassName()) + << msgStream.str(); + } + + void optimizationRemarkHandler( + const llvm::DiagnosticInfoOptimizationBase &diagInfo) { + auto passName = diagInfo.getPassName(); + if (diagInfo.isPassed()) { + if (codeGenOpts.OptimizationRemark.patternMatches(passName)) + // Optimization remarks are active only if the -Rpass flag has a regular + // expression that matches the name of the pass name in \p d. + emitOptimizationMessage( + diagInfo, clang::diag::remark_fe_backend_optimization_remark); + + return; + } + + if (diagInfo.isMissed()) { + if (codeGenOpts.OptimizationRemarkMissed.patternMatches(passName)) + // Missed optimization remarks are active only if the -Rpass-missed + // flag has a regular expression that matches the name of the pass + // name in \p d. + emitOptimizationMessage( + diagInfo, + clang::diag::remark_fe_backend_optimization_remark_missed); + + return; + } + + assert(diagInfo.isAnalysis() && "Unknown remark type"); + + bool shouldAlwaysPrint = false; + auto *ora = llvm::dyn_cast(&diagInfo); + if (ora) + shouldAlwaysPrint = ora->shouldAlwaysPrint(); + + if (shouldAlwaysPrint || + codeGenOpts.OptimizationRemarkAnalysis.patternMatches(passName)) + emitOptimizationMessage( + diagInfo, + clang::diag::remark_fe_backend_optimization_remark_analysis); + } + + bool handleDiagnostics(const llvm::DiagnosticInfo &di) override { + // Handle both middle and backend passes, other passes handled by clang + // can be implemented here later + switch (di.getKind()) { + case llvm::DK_OptimizationRemark: + optimizationRemarkHandler(llvm::cast(di)); + break; + case llvm::DK_OptimizationRemarkMissed: + optimizationRemarkHandler(llvm::cast(di)); + break; + case llvm::DK_OptimizationRemarkAnalysis: + optimizationRemarkHandler( + llvm::cast(di)); + break; + case llvm::DK_MachineOptimizationRemark: + optimizationRemarkHandler( + llvm::cast(di)); + break; + case llvm::DK_MachineOptimizationRemarkMissed: + optimizationRemarkHandler( + llvm::cast(di)); + break; + case llvm::DK_MachineOptimizationRemarkAnalysis: + optimizationRemarkHandler( + llvm::cast(di)); + break; + default: + break; + } + return true; + } +}; + void CodeGenAction::embedOffloadObjects() { CompilerInstance &ci = this->getInstance(); const auto &cgOpts = ci.getInvocation().getCodeGenOpts(); @@ -1029,6 +1158,11 @@ if (!codeGenOpts.OffloadObjects.empty()) embedOffloadObjects(); + BackendRemarkConsumer remarkConsumer(diags, codeGenOpts); + + llvmModule->getContext().setDiagnosticHandler( + std::make_unique(remarkConsumer)); + // write optimization-record llvm::Expected> optRecordFileOrErr = setupLLVMOptimizationRemarks( diff --git a/flang/lib/Frontend/TextDiagnosticPrinter.cpp b/flang/lib/Frontend/TextDiagnosticPrinter.cpp --- a/flang/lib/Frontend/TextDiagnosticPrinter.cpp +++ b/flang/lib/Frontend/TextDiagnosticPrinter.cpp @@ -16,9 +16,14 @@ #include "flang/Frontend/TextDiagnosticPrinter.h" #include "flang/Frontend/TextDiagnostic.h" +#include "string" #include "clang/Basic/DiagnosticOptions.h" #include "llvm/ADT/SmallString.h" +#include "llvm/ADT/SmallVector.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/Support/Automaton.h" #include "llvm/Support/ErrorHandling.h" +#include "llvm/Support/Path.h" #include "llvm/Support/raw_ostream.h" using namespace Fortran::frontend; @@ -29,6 +34,25 @@ TextDiagnosticPrinter::~TextDiagnosticPrinter() {} +// For remarks only, print the remark option and pass name that was used to a +// raw_ostream. +static void printRemarkOption(llvm::raw_ostream &os, + clang::DiagnosticsEngine::Level level, + const clang::Diagnostic &info) { + llvm::StringRef opt = + clang::DiagnosticIDs::getWarningOptionForDiag(info.getID()); + if (!opt.empty()) { + // We still need to check if the level is a Remark since, an unknown option + // warning could be printed i.e. [-Wunknown-warning-option] + os << " [" << (level == clang::DiagnosticsEngine::Remark ? "-R" : "-W") + << opt; + llvm::StringRef optValue = info.getDiags()->getFlagValue(); + if (!optValue.empty()) + os << "=" << optValue; + os << ']'; + } +} + void TextDiagnosticPrinter::HandleDiagnostic( clang::DiagnosticsEngine::Level level, const clang::Diagnostic &info) { // Default implementation (Warnings/errors count). @@ -40,6 +64,7 @@ info.FormatDiagnostic(outStr); llvm::raw_svector_ostream diagMessageStream(outStr); + printRemarkOption(diagMessageStream, level, info); if (!prefix.empty()) os << prefix << ": "; @@ -48,12 +73,43 @@ assert(!info.getLocation().isValid() && "Diagnostics with valid source location are not supported"); + // split incoming string to get the absolute path and filename in the + // case we are receiving optimization remarks from BackendRemarkConsumer + std::string diagMsg = std::string(diagMessageStream.str()); + std::string delimiter = ";;"; + + size_t pos = 0; + llvm::SmallVector tokens; + while ((pos = diagMsg.find(delimiter)) != std::string::npos) { + tokens.push_back(diagMsg.substr(0, pos)); + diagMsg.erase(0, pos + delimiter.length()); + } + + // Tokens will always be of size 2 in the case of optimization + // remark message received, in this format; + // [file location with line and column];;[path to file];;[the remark message] + if (tokens.size() == 2) { + // Extract relative path + llvm::SmallString<128> absPath = llvm::sys::path::relative_path(tokens[1]); + llvm::sys::path::remove_filename(absPath); + // Add the last separator before the file name + llvm::sys::path::append(absPath, llvm::sys::path::get_separator()); + llvm::sys::path::make_preferred(absPath); + + // Used for changing only the bold attribute + if (diagOpts->ShowColors) + os.changeColor(llvm::raw_ostream::SAVEDCOLOR, true); + + // Print path, file name, line and column + os << absPath << tokens[0] << ": "; + } + Fortran::frontend::TextDiagnostic::printDiagnosticLevel(os, level, diagOpts->ShowColors); Fortran::frontend::TextDiagnostic::printDiagnosticMessage( os, - /*IsSupplemental=*/level == clang::DiagnosticsEngine::Note, - diagMessageStream.str(), diagOpts->ShowColors); + /*IsSupplemental=*/level == clang::DiagnosticsEngine::Note, diagMsg, + diagOpts->ShowColors); os.flush(); } diff --git a/flang/lib/FrontendTool/ExecuteCompilerInvocation.cpp b/flang/lib/FrontendTool/ExecuteCompilerInvocation.cpp --- a/flang/lib/FrontendTool/ExecuteCompilerInvocation.cpp +++ b/flang/lib/FrontendTool/ExecuteCompilerInvocation.cpp @@ -22,6 +22,7 @@ #include "mlir/IR/AsmState.h" #include "mlir/IR/MLIRContext.h" #include "mlir/Pass/PassManager.h" +#include "clang/Basic/DiagnosticFrontend.h" #include "clang/Driver/Options.h" #include "llvm/Option/OptTable.h" #include "llvm/Option/Option.h" @@ -100,6 +101,54 @@ llvm_unreachable("Invalid program action!"); } +// Emit a warning and typo hint for unknown warning opts +static void emitUnknownDiagWarning(clang::DiagnosticsEngine &diags, + clang::diag::Flavor flavor, + llvm::StringRef prefix, + llvm::StringRef opt) { + llvm::StringRef suggestion = + clang::DiagnosticIDs::getNearestOption(flavor, opt); + diags.Report(clang::diag::warn_unknown_diag_option) + << (flavor == clang::diag::Flavor::WarningOrError ? 0 : 1) + << (prefix.str() += std::string(opt)) << !suggestion.empty() + << (prefix.str() += std::string(suggestion)); +} + +// Remarks are ignored by default in Diagnostic.td, hence, we have to +// enable them here before execution. Clang follows same idea using +// ProcessWarningOptions in Warnings.cpp +// This function is also responsible for emitting early warnings for +// invalid -R options. +static void +updateDiagEngineForOptRemarks(clang::DiagnosticsEngine &diagsEng, + const clang::DiagnosticOptions &opts) { + llvm::SmallVector diags; + const llvm::IntrusiveRefCntPtr diagIDs = + diagsEng.getDiagnosticIDs(); + + for (unsigned i = 0; i < opts.Remarks.size(); i++) { + llvm::StringRef remarkOpt = opts.Remarks[i]; + const auto flavor = clang::diag::Flavor::Remark; + + // Check to see if this opt starts with "no-", if so, this is a + // negative form of the option. + bool isPositive = !remarkOpt.startswith("no-"); + if (!isPositive) + remarkOpt = remarkOpt.substr(3); + + // Just issue a warning when encountering an unrecognised remark option. + if (diagIDs->getDiagnosticsInGroup(flavor, remarkOpt, diags)) { + emitUnknownDiagWarning(diagsEng, flavor, isPositive ? "-R" : "-Rno-", + remarkOpt); + continue; + } + + diagsEng.setSeverityForGroup(flavor, remarkOpt, + isPositive ? clang::diag::Severity::Remark + : clang::diag::Severity::Ignored); + } +} + bool executeCompilerInvocation(CompilerInstance *flang) { // Honor -help. if (flang->getFrontendOpts().showHelp) { @@ -166,6 +215,9 @@ // Honor color diagnostics. flang->getDiagnosticOpts().ShowColors = flang->getFrontendOpts().showColors; + updateDiagEngineForOptRemarks(flang->getDiagnostics(), + flang->getDiagnosticOpts()); + // Create and execute the frontend action. std::unique_ptr act(createFrontendAction(*flang)); if (!act) diff --git a/flang/test/Driver/optimization-remark.f90 b/flang/test/Driver/optimization-remark.f90 new file mode 100644 --- /dev/null +++ b/flang/test/Driver/optimization-remark.f90 @@ -0,0 +1,58 @@ +! This file tests the -Rpass family of flags (-Rpass, -Rpass-missed +! and -Rpass-analysis) +! loop-delete isn't enabled at O0 so we use at least O1 + +! Check that we can override -Rpass= with -Rno-pass. +! RUN: %flang_fc1 %s -O1 -Rpass -emit-llvm -o - 2>&1 | FileCheck %s --check-prefix=CHECK-REMARKS +! RUN: %flang_fc1 %s -O1 -Rpass -Rno-pass -emit-llvm -o - 2>&1 | FileCheck %s --check-prefix=CHECK-NO-REMARKS + +! Check "unknown remark option" warning +! RUN: %flang %s -O1 -R 2>&1 | FileCheck %s --check-prefix=CHECK-REMARKS-WARN + +! Check "unknown remark option" warning with suggestion +! RUN: %flang %s -O1 -Rpas 2>&1 | FileCheck %s --check-prefix=CHECK-WARN-SUGGEST + +! Check -Rno-pass, -Rno-pass-analysis, -Rno-pass-missed nothing emitted +! RUN: %flang %s -O1 -Rno-pass 2>&1 | FileCheck %s --allow-empty --check-prefix=CHECK-NO-REMARKS +! RUN: %flang %s -O1 -Rno-pass-missed 2>&1 | FileCheck %s --allow-empty --check-prefix=CHECK-NO-REMARKS +! RUN: %flang %s -O1 -Rno-pass-analysis 2>&1 | FileCheck %s --allow-empty --check-prefix=CHECK-NO-REMARKS + +! Check full -Rpass message is emitted +! RUN: %flang %s -O1 -Rpass 2>&1 | FileCheck %s + +! Check full -Rpass-missed message is emitted +! RUN: %flang %s -O1 -Rpass-missed 2>&1 | FileCheck %s --check-prefix=CHECK-REMARKS-MISSED + +! Check full -Rpass-analysis message is emitted +! RUN: %flang %s -O1 -Rpass-analysis 2>&1 | FileCheck %s --check-prefix=CHECK-REMARKS-ANALYSIS + +! Check a backend pass can be generated +! RUN: %flang %s -O1 -Rpass-analysis 2>&1 | FileCheck %s --check-prefix=BACKEND + + +! CHECK: optimization-remark.f90:54:5: remark: Loop deleted because it is invariant [-Rpass=loop-delete] +! CHECK-REMARKS-MISSED: optimization-remark.f90:49:5: remark: loop not vectorized [-Rpass-missed=loop-vectorize] +! CHECK-REMARKS-ANALYSIS: optimization-remark.f90:49:5: remark: loop not vectorized: instruction cannot be vectorized [-Rpass-analysis=loop-vectorize] +! CHECK-REMARKS: remark: +! CHECK-NO-REMARKS-NOT: remark: + +! CHECK-REMARKS-WARN: warning: unknown remark option '-R' [-Wunknown-warning-option] +! CHECK-WARN-SUGGEST: warning: unknown remark option '-Rpas'; did you mean '-Rpass'? [-Wunknown-warning-option] + +! BACKEND: optimization-remark.f90:1:0: remark: 25 instructions in function [-Rpass-analysis=asm-printer] + +program forttest + implicit none + real, dimension(1:50) :: aR1 + integer :: n + + do n = 1,50 + aR1(n) = n * 1.34 + print *, "hello" + end do + + do n = 1,50 + aR1(n) = n * 1.34 + end do + +end program forttest