2018-03-26 21:45:04 +00:00
|
|
|
//===- CompilerInvocation.cpp ---------------------------------------------===//
|
2009-11-17 06:02:29 +00:00
|
|
|
//
|
2019-01-19 08:50:56 +00:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2009-11-17 06:02:29 +00:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2016-07-18 19:02:11 +00:00
|
|
|
#include "clang/Frontend/CompilerInvocation.h"
|
2015-11-03 18:33:07 +00:00
|
|
|
#include "TestModuleFileExtension.h"
|
2016-01-06 14:35:46 +00:00
|
|
|
#include "clang/Basic/Builtins.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "clang/Basic/CharInfo.h"
|
2018-12-11 03:18:39 +00:00
|
|
|
#include "clang/Basic/CodeGenOptions.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "clang/Basic/CommentOptions.h"
|
|
|
|
#include "clang/Basic/DebugInfoOptions.h"
|
|
|
|
#include "clang/Basic/Diagnostic.h"
|
2020-11-17 10:43:02 -08:00
|
|
|
#include "clang/Basic/DiagnosticDriver.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "clang/Basic/DiagnosticOptions.h"
|
|
|
|
#include "clang/Basic/FileSystemOptions.h"
|
|
|
|
#include "clang/Basic/LLVM.h"
|
|
|
|
#include "clang/Basic/LangOptions.h"
|
2019-08-05 13:59:26 +00:00
|
|
|
#include "clang/Basic/LangStandard.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "clang/Basic/ObjCRuntime.h"
|
|
|
|
#include "clang/Basic/Sanitizers.h"
|
|
|
|
#include "clang/Basic/SourceLocation.h"
|
|
|
|
#include "clang/Basic/TargetOptions.h"
|
2012-12-04 09:13:33 +00:00
|
|
|
#include "clang/Basic/Version.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "clang/Basic/Visibility.h"
|
2018-04-13 02:31:58 +00:00
|
|
|
#include "clang/Basic/XRayInstr.h"
|
2014-12-29 12:09:08 +00:00
|
|
|
#include "clang/Config/config.h"
|
2019-01-31 22:15:32 +00:00
|
|
|
#include "clang/Driver/Driver.h"
|
2009-12-01 03:16:53 +00:00
|
|
|
#include "clang/Driver/DriverDiagnostic.h"
|
2012-12-04 09:13:33 +00:00
|
|
|
#include "clang/Driver/Options.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "clang/Frontend/CommandLineSourceLoc.h"
|
|
|
|
#include "clang/Frontend/DependencyOutputOptions.h"
|
2014-04-15 18:16:25 +00:00
|
|
|
#include "clang/Frontend/FrontendDiagnostic.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "clang/Frontend/FrontendOptions.h"
|
2019-01-03 18:26:06 +00:00
|
|
|
#include "clang/Frontend/FrontendPluginRegistry.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "clang/Frontend/MigratorOptions.h"
|
|
|
|
#include "clang/Frontend/PreprocessorOutputOptions.h"
|
2013-06-14 17:17:23 +00:00
|
|
|
#include "clang/Frontend/Utils.h"
|
2012-12-04 09:13:33 +00:00
|
|
|
#include "clang/Lex/HeaderSearchOptions.h"
|
2016-07-18 19:02:11 +00:00
|
|
|
#include "clang/Lex/PreprocessorOptions.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "clang/Sema/CodeCompleteOptions.h"
|
Make AST reading work better with LLVM_APPEND_VC_REV=NO
With LLVM_APPEND_VC_REV=NO, Modules/merge-lifetime-extended-temporary.cpp
would fail if it ran before a0f50d731639350c7a7 (which changed
the serialization format) and then after, for these reasons:
1. With LLVM_APPEND_VC_REV=NO, the module hash before and after the
change was the same.
2. Modules/merge-lifetime-extended-temporary.cpp is the only test
we have that uses -fmodule-cache-path=%t that
a) actually writes to the cache path
b) doesn't do `rm -rf %t` at the top of the test
So the old run would write a module file, and then the new run would
try to load it, but the serialized format changed.
Do several things to fix this:
1. Include clang::serialization::VERSION_MAJOR/VERSION_MINOR in
the module hash, so that when the AST format changes (...and
we remember to bump these), we use a different module cache dir.
2. Bump VERSION_MAJOR, since a0f50d731639350c7a7 changed the
on-disk format in a way that a gch file written before that change
can't be read after that change.
3. Add `rm -rf %t` to all tests that pass -fmodule-cache-path=%t.
This is unnecessary from a correctness PoV after 1 and 2,
but makes it so that we don't amass many cache dirs over time.
(Arguably, it also makes it so that the test suite doesn't catch
when we change the serialization format but don't bump
clang::serialization::VERSION_MAJOR/VERSION_MINOR; oh well.)
Differential Revision: https://reviews.llvm.org/D73202
2020-01-22 10:34:34 -05:00
|
|
|
#include "clang/Serialization/ASTBitCodes.h"
|
2015-11-03 18:33:07 +00:00
|
|
|
#include "clang/Serialization/ModuleFileExtension.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "clang/StaticAnalyzer/Core/AnalyzerOptions.h"
|
|
|
|
#include "llvm/ADT/APInt.h"
|
|
|
|
#include "llvm/ADT/ArrayRef.h"
|
|
|
|
#include "llvm/ADT/CachedHashString.h"
|
2020-03-26 14:51:09 +07:00
|
|
|
#include "llvm/ADT/FloatingPointMode.h"
|
2012-11-05 19:45:09 +00:00
|
|
|
#include "llvm/ADT/Hashing.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "llvm/ADT/None.h"
|
|
|
|
#include "llvm/ADT/Optional.h"
|
|
|
|
#include "llvm/ADT/SmallString.h"
|
2015-01-14 11:29:14 +00:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "llvm/ADT/StringRef.h"
|
2009-12-01 03:16:53 +00:00
|
|
|
#include "llvm/ADT/StringSwitch.h"
|
2010-08-30 09:42:39 +00:00
|
|
|
#include "llvm/ADT/Triple.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "llvm/ADT/Twine.h"
|
2020-11-24 20:40:47 -08:00
|
|
|
#include "llvm/Config/llvm-config.h"
|
2018-08-20 20:14:08 +00:00
|
|
|
#include "llvm/IR/DebugInfoMetadata.h"
|
2015-10-27 17:56:59 +00:00
|
|
|
#include "llvm/Linker/Linker.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "llvm/MC/MCTargetOptions.h"
|
2013-06-14 17:17:23 +00:00
|
|
|
#include "llvm/Option/Arg.h"
|
|
|
|
#include "llvm/Option/ArgList.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "llvm/Option/OptSpecifier.h"
|
2013-06-14 17:17:23 +00:00
|
|
|
#include "llvm/Option/OptTable.h"
|
|
|
|
#include "llvm/Option/Option.h"
|
2016-03-02 20:59:36 +00:00
|
|
|
#include "llvm/ProfileData/InstrProfReader.h"
|
2020-11-17 10:43:02 -08:00
|
|
|
#include "llvm/Remarks/HotnessThresholdParser.h"
|
2014-05-08 02:28:32 +00:00
|
|
|
#include "llvm/Support/CodeGen.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "llvm/Support/Compiler.h"
|
|
|
|
#include "llvm/Support/Error.h"
|
2009-11-17 06:02:29 +00:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "llvm/Support/ErrorOr.h"
|
2013-06-26 05:03:40 +00:00
|
|
|
#include "llvm/Support/FileSystem.h"
|
2010-11-29 18:12:39 +00:00
|
|
|
#include "llvm/Support/Host.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "llvm/Support/MathExtras.h"
|
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
2010-11-29 18:12:39 +00:00
|
|
|
#include "llvm/Support/Path.h"
|
2013-09-11 00:38:02 +00:00
|
|
|
#include "llvm/Support/Process.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "llvm/Support/Regex.h"
|
2018-06-11 10:28:04 +00:00
|
|
|
#include "llvm/Support/VersionTuple.h"
|
2018-10-10 13:27:25 +00:00
|
|
|
#include "llvm/Support/VirtualFileSystem.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2015-11-09 12:40:41 +00:00
|
|
|
#include "llvm/Target/TargetOptions.h"
|
2018-03-26 21:45:04 +00:00
|
|
|
#include <algorithm>
|
2014-03-02 17:08:31 +00:00
|
|
|
#include <atomic>
|
2018-03-26 21:45:04 +00:00
|
|
|
#include <cassert>
|
|
|
|
#include <cstddef>
|
|
|
|
#include <cstring>
|
2014-03-09 11:36:40 +00:00
|
|
|
#include <memory>
|
2018-03-26 21:45:04 +00:00
|
|
|
#include <string>
|
|
|
|
#include <tuple>
|
2020-12-16 13:14:50 +01:00
|
|
|
#include <type_traits>
|
2018-03-26 21:45:04 +00:00
|
|
|
#include <utility>
|
|
|
|
#include <vector>
|
|
|
|
|
2009-11-17 06:02:29 +00:00
|
|
|
using namespace clang;
|
2018-03-26 21:45:04 +00:00
|
|
|
using namespace driver;
|
|
|
|
using namespace options;
|
|
|
|
using namespace llvm::opt;
|
2009-11-17 06:02:29 +00:00
|
|
|
|
2011-11-17 23:01:24 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Initialization.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-11-18 04:32:13 +00:00
|
|
|
CompilerInvocationBase::CompilerInvocationBase()
|
2018-03-26 21:45:04 +00:00
|
|
|
: LangOpts(new LangOptions()), TargetOpts(new TargetOptions()),
|
|
|
|
DiagnosticOpts(new DiagnosticOptions()),
|
|
|
|
HeaderSearchOpts(new HeaderSearchOptions()),
|
|
|
|
PreprocessorOpts(new PreprocessorOptions()) {}
|
2011-11-17 23:01:24 +00:00
|
|
|
|
2011-11-18 04:32:13 +00:00
|
|
|
CompilerInvocationBase::CompilerInvocationBase(const CompilerInvocationBase &X)
|
2017-01-06 19:49:01 +00:00
|
|
|
: LangOpts(new LangOptions(*X.getLangOpts())),
|
|
|
|
TargetOpts(new TargetOptions(X.getTargetOpts())),
|
|
|
|
DiagnosticOpts(new DiagnosticOptions(X.getDiagnosticOpts())),
|
|
|
|
HeaderSearchOpts(new HeaderSearchOptions(X.getHeaderSearchOpts())),
|
|
|
|
PreprocessorOpts(new PreprocessorOptions(X.getPreprocessorOpts())) {}
|
2011-11-17 23:01:24 +00:00
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
CompilerInvocationBase::~CompilerInvocationBase() = default;
|
2014-05-11 22:10:52 +00:00
|
|
|
|
2020-05-11 11:42:38 +01:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Normalizers
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#define SIMPLE_ENUM_VALUE_TABLE
|
|
|
|
#include "clang/Driver/Options.inc"
|
|
|
|
#undef SIMPLE_ENUM_VALUE_TABLE
|
|
|
|
|
2020-12-22 15:00:31 +01:00
|
|
|
static llvm::Optional<bool>
|
|
|
|
normalizeSimpleFlag(OptSpecifier Opt, unsigned TableIndex, const ArgList &Args,
|
|
|
|
DiagnosticsEngine &Diags, bool &Success) {
|
2020-11-11 11:05:24 +01:00
|
|
|
if (Args.hasArg(Opt))
|
|
|
|
return true;
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2020-11-20 12:49:51 +01:00
|
|
|
static Optional<bool> normalizeSimpleNegativeFlag(OptSpecifier Opt, unsigned,
|
|
|
|
const ArgList &Args,
|
2020-12-22 15:00:31 +01:00
|
|
|
DiagnosticsEngine &,
|
|
|
|
bool &Success) {
|
2020-11-20 12:49:51 +01:00
|
|
|
if (Args.hasArg(Opt))
|
|
|
|
return false;
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2020-11-20 12:49:51 +01:00
|
|
|
/// The tblgen-erated code passes in a fifth parameter of an arbitrary type, but
|
|
|
|
/// denormalizeSimpleFlags never looks at it. Avoid bloating compile-time with
|
|
|
|
/// unnecessary template instantiations and just ignore it with a variadic
|
|
|
|
/// argument.
|
|
|
|
static void denormalizeSimpleFlag(SmallVectorImpl<const char *> &Args,
|
|
|
|
const char *Spelling,
|
2021-01-05 17:00:46 +01:00
|
|
|
CompilerInvocation::StringAllocator,
|
|
|
|
Option::OptionClass, unsigned, /*T*/...) {
|
2020-11-13 14:17:54 +01:00
|
|
|
Args.push_back(Spelling);
|
|
|
|
}
|
|
|
|
|
2020-12-21 13:50:09 +01:00
|
|
|
template <typename T> static constexpr bool is_uint64_t_convertible() {
|
|
|
|
return !std::is_same<T, uint64_t>::value &&
|
|
|
|
llvm::is_integral_or_enum<T>::value;
|
|
|
|
}
|
2020-11-20 12:49:51 +01:00
|
|
|
|
2020-12-21 13:50:09 +01:00
|
|
|
template <typename T,
|
|
|
|
std::enable_if_t<!is_uint64_t_convertible<T>(), bool> = false>
|
|
|
|
static auto makeFlagToValueNormalizer(T Value) {
|
|
|
|
return [Value](OptSpecifier Opt, unsigned, const ArgList &Args,
|
2020-12-22 15:00:31 +01:00
|
|
|
DiagnosticsEngine &, bool &Success) -> Optional<T> {
|
2020-11-20 12:49:51 +01:00
|
|
|
if (Args.hasArg(Opt))
|
|
|
|
return Value;
|
|
|
|
return None;
|
2020-12-21 13:50:09 +01:00
|
|
|
};
|
2020-11-20 12:49:51 +01:00
|
|
|
}
|
|
|
|
|
2020-12-21 13:50:09 +01:00
|
|
|
template <typename T,
|
|
|
|
std::enable_if_t<is_uint64_t_convertible<T>(), bool> = false>
|
|
|
|
static auto makeFlagToValueNormalizer(T Value) {
|
|
|
|
return makeFlagToValueNormalizer(uint64_t(Value));
|
2020-11-11 11:05:24 +01:00
|
|
|
}
|
|
|
|
|
2020-12-08 18:15:21 +01:00
|
|
|
static auto makeBooleanOptionNormalizer(bool Value, bool OtherValue,
|
|
|
|
OptSpecifier OtherOpt) {
|
|
|
|
return [Value, OtherValue, OtherOpt](OptSpecifier Opt, unsigned,
|
2020-12-22 15:00:31 +01:00
|
|
|
const ArgList &Args, DiagnosticsEngine &,
|
|
|
|
bool &Success) -> Optional<bool> {
|
2020-12-08 18:15:21 +01:00
|
|
|
if (const Arg *A = Args.getLastArg(Opt, OtherOpt)) {
|
|
|
|
return A->getOption().matches(Opt) ? Value : OtherValue;
|
|
|
|
}
|
2020-12-07 14:22:23 +01:00
|
|
|
return None;
|
|
|
|
};
|
2020-11-13 14:17:54 +01:00
|
|
|
}
|
|
|
|
|
2020-12-09 13:26:39 +01:00
|
|
|
static auto makeBooleanOptionDenormalizer(bool Value) {
|
|
|
|
return [Value](SmallVectorImpl<const char *> &Args, const char *Spelling,
|
2021-01-05 17:00:46 +01:00
|
|
|
CompilerInvocation::StringAllocator, Option::OptionClass,
|
|
|
|
unsigned, bool KeyPath) {
|
2020-12-09 13:26:39 +01:00
|
|
|
if (KeyPath == Value)
|
|
|
|
Args.push_back(Spelling);
|
2020-12-07 14:22:23 +01:00
|
|
|
};
|
2020-11-13 14:17:54 +01:00
|
|
|
}
|
|
|
|
|
2021-01-05 17:00:46 +01:00
|
|
|
static void denormalizeStringImpl(SmallVectorImpl<const char *> &Args,
|
|
|
|
const char *Spelling,
|
|
|
|
CompilerInvocation::StringAllocator SA,
|
|
|
|
Option::OptionClass OptClass, unsigned,
|
|
|
|
Twine Value) {
|
|
|
|
switch (OptClass) {
|
|
|
|
case Option::SeparateClass:
|
|
|
|
case Option::JoinedOrSeparateClass:
|
|
|
|
Args.push_back(Spelling);
|
|
|
|
Args.push_back(SA(Value));
|
|
|
|
break;
|
|
|
|
case Option::JoinedClass:
|
|
|
|
Args.push_back(SA(Twine(Spelling) + Value));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
llvm_unreachable("Cannot denormalize an option with option class "
|
|
|
|
"incompatible with string denormalization.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
static void
|
|
|
|
denormalizeString(SmallVectorImpl<const char *> &Args, const char *Spelling,
|
|
|
|
CompilerInvocation::StringAllocator SA,
|
|
|
|
Option::OptionClass OptClass, unsigned TableIndex, T Value) {
|
|
|
|
denormalizeStringImpl(Args, Spelling, SA, OptClass, TableIndex, Twine(Value));
|
|
|
|
}
|
|
|
|
|
2020-11-16 14:30:54 +01:00
|
|
|
static Optional<SimpleEnumValue>
|
|
|
|
findValueTableByName(const SimpleEnumValueTable &Table, StringRef Name) {
|
|
|
|
for (int I = 0, E = Table.Size; I != E; ++I)
|
|
|
|
if (Name == Table.Table[I].Name)
|
|
|
|
return Table.Table[I];
|
|
|
|
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Optional<SimpleEnumValue>
|
|
|
|
findValueTableByValue(const SimpleEnumValueTable &Table, unsigned Value) {
|
|
|
|
for (int I = 0, E = Table.Size; I != E; ++I)
|
|
|
|
if (Value == Table.Table[I].Value)
|
|
|
|
return Table.Table[I];
|
|
|
|
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2020-12-22 15:00:31 +01:00
|
|
|
static llvm::Optional<unsigned>
|
|
|
|
normalizeSimpleEnum(OptSpecifier Opt, unsigned TableIndex, const ArgList &Args,
|
|
|
|
DiagnosticsEngine &Diags, bool &Success) {
|
2020-05-11 11:42:38 +01:00
|
|
|
assert(TableIndex < SimpleEnumValueTablesSize);
|
|
|
|
const SimpleEnumValueTable &Table = SimpleEnumValueTables[TableIndex];
|
|
|
|
|
|
|
|
auto *Arg = Args.getLastArg(Opt);
|
|
|
|
if (!Arg)
|
|
|
|
return None;
|
|
|
|
|
|
|
|
StringRef ArgValue = Arg->getValue();
|
2020-11-16 14:30:54 +01:00
|
|
|
if (auto MaybeEnumVal = findValueTableByName(Table, ArgValue))
|
|
|
|
return MaybeEnumVal->Value;
|
2020-05-11 11:42:38 +01:00
|
|
|
|
2020-12-22 15:00:31 +01:00
|
|
|
Success = false;
|
2020-05-11 11:42:38 +01:00
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< Arg->getAsString(Args) << ArgValue;
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2020-12-16 13:17:24 +01:00
|
|
|
static void denormalizeSimpleEnumImpl(SmallVectorImpl<const char *> &Args,
|
|
|
|
const char *Spelling,
|
|
|
|
CompilerInvocation::StringAllocator SA,
|
2021-01-05 17:00:46 +01:00
|
|
|
Option::OptionClass OptClass,
|
2020-12-16 13:17:24 +01:00
|
|
|
unsigned TableIndex, unsigned Value) {
|
2020-05-11 11:42:38 +01:00
|
|
|
assert(TableIndex < SimpleEnumValueTablesSize);
|
|
|
|
const SimpleEnumValueTable &Table = SimpleEnumValueTables[TableIndex];
|
2020-11-16 14:30:54 +01:00
|
|
|
if (auto MaybeEnumVal = findValueTableByValue(Table, Value)) {
|
2021-01-05 17:00:46 +01:00
|
|
|
denormalizeString(Args, Spelling, SA, OptClass, TableIndex,
|
|
|
|
MaybeEnumVal->Name);
|
2020-11-16 14:30:54 +01:00
|
|
|
} else {
|
|
|
|
llvm_unreachable("The simple enum value was not correctly defined in "
|
|
|
|
"the tablegen option description");
|
2020-11-11 11:05:24 +01:00
|
|
|
}
|
2020-11-16 14:30:54 +01:00
|
|
|
}
|
2020-05-11 11:42:38 +01:00
|
|
|
|
2020-12-16 13:17:24 +01:00
|
|
|
template <typename T>
|
|
|
|
static void denormalizeSimpleEnum(SmallVectorImpl<const char *> &Args,
|
|
|
|
const char *Spelling,
|
|
|
|
CompilerInvocation::StringAllocator SA,
|
2021-01-05 17:00:46 +01:00
|
|
|
Option::OptionClass OptClass,
|
2020-12-16 13:17:24 +01:00
|
|
|
unsigned TableIndex, T Value) {
|
2021-01-05 17:00:46 +01:00
|
|
|
return denormalizeSimpleEnumImpl(Args, Spelling, SA, OptClass, TableIndex,
|
2020-12-16 13:17:24 +01:00
|
|
|
static_cast<unsigned>(Value));
|
|
|
|
}
|
|
|
|
|
2020-12-14 09:49:55 +01:00
|
|
|
static Optional<std::string> normalizeString(OptSpecifier Opt, int TableIndex,
|
|
|
|
const ArgList &Args,
|
2020-12-22 15:00:31 +01:00
|
|
|
DiagnosticsEngine &Diags,
|
|
|
|
bool &Success) {
|
2020-12-14 09:49:55 +01:00
|
|
|
auto *Arg = Args.getLastArg(Opt);
|
|
|
|
if (!Arg)
|
|
|
|
return None;
|
|
|
|
return std::string(Arg->getValue());
|
|
|
|
}
|
|
|
|
|
2020-12-16 13:14:50 +01:00
|
|
|
template <typename IntTy>
|
2020-12-22 15:00:31 +01:00
|
|
|
static Optional<IntTy>
|
|
|
|
normalizeStringIntegral(OptSpecifier Opt, int, const ArgList &Args,
|
|
|
|
DiagnosticsEngine &Diags, bool &Success) {
|
2020-12-16 13:14:50 +01:00
|
|
|
auto *Arg = Args.getLastArg(Opt);
|
|
|
|
if (!Arg)
|
|
|
|
return None;
|
|
|
|
IntTy Res;
|
|
|
|
if (StringRef(Arg->getValue()).getAsInteger(0, Res)) {
|
2020-12-22 15:00:31 +01:00
|
|
|
Success = false;
|
2020-12-16 13:14:50 +01:00
|
|
|
Diags.Report(diag::err_drv_invalid_int_value)
|
|
|
|
<< Arg->getAsString(Args) << Arg->getValue();
|
|
|
|
}
|
|
|
|
return Res;
|
|
|
|
}
|
|
|
|
|
2021-01-05 17:00:46 +01:00
|
|
|
static Optional<std::vector<std::string>>
|
|
|
|
normalizeStringVector(OptSpecifier Opt, int, const ArgList &Args,
|
2020-12-22 15:00:31 +01:00
|
|
|
DiagnosticsEngine &, bool &Success) {
|
2021-01-05 17:00:46 +01:00
|
|
|
return Args.getAllArgValues(Opt);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void denormalizeStringVector(SmallVectorImpl<const char *> &Args,
|
|
|
|
const char *Spelling,
|
|
|
|
CompilerInvocation::StringAllocator SA,
|
|
|
|
Option::OptionClass OptClass,
|
|
|
|
unsigned TableIndex,
|
|
|
|
const std::vector<std::string> &Values) {
|
2020-12-22 10:01:51 +01:00
|
|
|
switch (OptClass) {
|
|
|
|
case Option::CommaJoinedClass: {
|
2020-12-21 16:35:32 +01:00
|
|
|
std::string CommaJoinedValue;
|
|
|
|
if (!Values.empty()) {
|
|
|
|
CommaJoinedValue.append(Values.front());
|
|
|
|
for (const std::string &Value : llvm::drop_begin(Values, 1)) {
|
|
|
|
CommaJoinedValue.append(",");
|
|
|
|
CommaJoinedValue.append(Value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
denormalizeString(Args, Spelling, SA, Option::OptionClass::JoinedClass,
|
|
|
|
TableIndex, CommaJoinedValue);
|
2020-12-22 10:01:51 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Option::JoinedClass:
|
|
|
|
case Option::SeparateClass:
|
|
|
|
case Option::JoinedOrSeparateClass:
|
2020-12-21 16:35:32 +01:00
|
|
|
for (const std::string &Value : Values)
|
|
|
|
denormalizeString(Args, Spelling, SA, OptClass, TableIndex, Value);
|
2020-12-22 10:01:51 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
llvm_unreachable("Cannot denormalize an option with option class "
|
|
|
|
"incompatible with string vector denormalization.");
|
2021-01-05 17:00:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-11 11:42:38 +01:00
|
|
|
static Optional<std::string> normalizeTriple(OptSpecifier Opt, int TableIndex,
|
|
|
|
const ArgList &Args,
|
2020-12-22 15:00:31 +01:00
|
|
|
DiagnosticsEngine &Diags,
|
|
|
|
bool &Success) {
|
2020-05-11 11:42:38 +01:00
|
|
|
auto *Arg = Args.getLastArg(Opt);
|
|
|
|
if (!Arg)
|
|
|
|
return None;
|
|
|
|
return llvm::Triple::normalize(Arg->getValue());
|
|
|
|
}
|
|
|
|
|
2020-11-11 11:05:24 +01:00
|
|
|
template <typename T, typename U>
|
|
|
|
static T mergeForwardValue(T KeyPath, U Value) {
|
2020-12-15 09:59:19 +01:00
|
|
|
return static_cast<T>(Value);
|
2020-11-11 11:05:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename U> static T mergeMaskValue(T KeyPath, U Value) {
|
|
|
|
return KeyPath | Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T> static T extractForwardValue(T KeyPath) {
|
|
|
|
return KeyPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename U, U Value>
|
|
|
|
static T extractMaskValue(T KeyPath) {
|
|
|
|
return KeyPath & Value;
|
|
|
|
}
|
|
|
|
|
2020-12-15 09:59:19 +01:00
|
|
|
static void FixupInvocation(CompilerInvocation &Invocation,
|
2021-01-05 17:00:46 +01:00
|
|
|
DiagnosticsEngine &Diags,
|
|
|
|
const InputArgList &Args) {
|
2020-11-16 12:17:29 +01:00
|
|
|
LangOptions &LangOpts = *Invocation.getLangOpts();
|
|
|
|
DiagnosticOptions &DiagOpts = Invocation.getDiagnosticOpts();
|
|
|
|
CodeGenOptions &CodeGenOpts = Invocation.getCodeGenOpts();
|
2020-12-15 09:59:19 +01:00
|
|
|
TargetOptions &TargetOpts = Invocation.getTargetOpts();
|
2020-11-20 12:49:51 +01:00
|
|
|
FrontendOptions &FrontendOpts = Invocation.getFrontendOpts();
|
2020-11-16 12:17:29 +01:00
|
|
|
CodeGenOpts.XRayInstrumentFunctions = LangOpts.XRayInstrument;
|
|
|
|
CodeGenOpts.XRayAlwaysEmitCustomEvents = LangOpts.XRayAlwaysEmitCustomEvents;
|
|
|
|
CodeGenOpts.XRayAlwaysEmitTypedEvents = LangOpts.XRayAlwaysEmitTypedEvents;
|
2021-01-05 17:00:46 +01:00
|
|
|
CodeGenOpts.DisableFree = FrontendOpts.DisableFree;
|
2020-11-20 12:49:51 +01:00
|
|
|
FrontendOpts.GenerateGlobalModuleIndex = FrontendOpts.UseGlobalModuleIndex;
|
2020-11-16 12:17:29 +01:00
|
|
|
|
2021-01-05 17:00:46 +01:00
|
|
|
LangOpts.ForceEmitVTables = CodeGenOpts.ForceEmitVTables;
|
|
|
|
LangOpts.SpeculativeLoadHardening = CodeGenOpts.SpeculativeLoadHardening;
|
|
|
|
LangOpts.CurrentModule = LangOpts.ModuleName;
|
|
|
|
|
2020-11-16 12:17:29 +01:00
|
|
|
llvm::sys::Process::UseANSIEscapeCodes(DiagOpts.UseANSIEscapeCodes);
|
2020-12-15 09:59:19 +01:00
|
|
|
|
|
|
|
llvm::Triple T(TargetOpts.Triple);
|
2021-01-05 17:00:46 +01:00
|
|
|
llvm::Triple::ArchType Arch = T.getArch();
|
2020-12-15 09:59:19 +01:00
|
|
|
|
|
|
|
if (LangOpts.getExceptionHandling() != llvm::ExceptionHandling::None &&
|
|
|
|
T.isWindowsMSVCEnvironment())
|
|
|
|
Diags.Report(diag::err_fe_invalid_exception_model)
|
|
|
|
<< static_cast<unsigned>(LangOpts.getExceptionHandling()) << T.str();
|
2021-01-05 17:00:46 +01:00
|
|
|
|
|
|
|
if (LangOpts.AppleKext && !LangOpts.CPlusPlus)
|
|
|
|
Diags.Report(diag::warn_c_kext);
|
|
|
|
|
|
|
|
if (LangOpts.NewAlignOverride &&
|
|
|
|
!llvm::isPowerOf2_32(LangOpts.NewAlignOverride)) {
|
|
|
|
Arg *A = Args.getLastArg(OPT_fnew_alignment_EQ);
|
|
|
|
Diags.Report(diag::err_fe_invalid_alignment)
|
|
|
|
<< A->getAsString(Args) << A->getValue();
|
|
|
|
LangOpts.NewAlignOverride = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Arg *A = Args.getLastArg(OPT_fdefault_calling_conv_EQ)) {
|
|
|
|
auto DefaultCC = LangOpts.getDefaultCallingConv();
|
|
|
|
|
|
|
|
bool emitError = (DefaultCC == LangOptions::DCC_FastCall ||
|
|
|
|
DefaultCC == LangOptions::DCC_StdCall) &&
|
|
|
|
Arch != llvm::Triple::x86;
|
|
|
|
emitError |= (DefaultCC == LangOptions::DCC_VectorCall ||
|
|
|
|
DefaultCC == LangOptions::DCC_RegCall) &&
|
|
|
|
!T.isX86();
|
|
|
|
if (emitError)
|
|
|
|
Diags.Report(diag::err_drv_argument_not_allowed_with)
|
|
|
|
<< A->getSpelling() << T.getTriple();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!CodeGenOpts.ProfileRemappingFile.empty() && CodeGenOpts.LegacyPassManager)
|
|
|
|
Diags.Report(diag::err_drv_argument_only_allowed_with)
|
|
|
|
<< Args.getLastArg(OPT_fprofile_remapping_file_EQ)->getAsString(Args)
|
|
|
|
<< "-fno-legacy-pass-manager";
|
2020-11-16 12:17:29 +01:00
|
|
|
}
|
|
|
|
|
2011-11-17 23:01:24 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
2012-11-01 03:48:49 +00:00
|
|
|
// Deserialization (from args)
|
2009-12-01 03:16:53 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-12-04 01:50:36 +00:00
|
|
|
static unsigned getOptimizationLevel(ArgList &Args, InputKind IK,
|
2011-09-25 23:23:43 +00:00
|
|
|
DiagnosticsEngine &Diags) {
|
2018-11-26 17:26:49 +00:00
|
|
|
unsigned DefaultOpt = llvm::CodeGenOpt::None;
|
2019-08-05 13:59:26 +00:00
|
|
|
if (IK.getLanguage() == Language::OpenCL && !Args.hasArg(OPT_cl_opt_disable))
|
2018-11-26 17:26:49 +00:00
|
|
|
DefaultOpt = llvm::CodeGenOpt::Default;
|
2012-05-01 14:57:16 +00:00
|
|
|
|
|
|
|
if (Arg *A = Args.getLastArg(options::OPT_O_Group)) {
|
|
|
|
if (A->getOption().matches(options::OPT_O0))
|
2018-11-26 17:26:49 +00:00
|
|
|
return llvm::CodeGenOpt::None;
|
2012-05-01 14:57:16 +00:00
|
|
|
|
2013-04-10 21:26:02 +00:00
|
|
|
if (A->getOption().matches(options::OPT_Ofast))
|
2018-11-26 17:26:49 +00:00
|
|
|
return llvm::CodeGenOpt::Aggressive;
|
2013-04-10 21:26:02 +00:00
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
assert(A->getOption().matches(options::OPT_O));
|
2012-05-01 14:57:16 +00:00
|
|
|
|
2013-01-12 19:30:44 +00:00
|
|
|
StringRef S(A->getValue());
|
2020-05-13 17:25:04 -07:00
|
|
|
if (S == "s" || S == "z")
|
2018-11-26 17:26:49 +00:00
|
|
|
return llvm::CodeGenOpt::Default;
|
2012-05-01 14:57:16 +00:00
|
|
|
|
Add a new optimization option -Og
Summary:
Just like gcc, we should have the -Og option as more and more software are using it:
https://llvm.org/bugs/show_bug.cgi?id=20765
Reviewers: echristo, dberlin, dblaikie, keith.walker.arm, rengolin
Subscribers: aprantl, friss, mehdi_amini, RKSimon, probinson, majnemer, cfe-commits
Differential Revision: https://reviews.llvm.org/D24998
llvm-svn: 286602
2016-11-11 17:29:56 +00:00
|
|
|
if (S == "g")
|
2018-11-26 17:26:49 +00:00
|
|
|
return llvm::CodeGenOpt::Less;
|
Add a new optimization option -Og
Summary:
Just like gcc, we should have the -Og option as more and more software are using it:
https://llvm.org/bugs/show_bug.cgi?id=20765
Reviewers: echristo, dberlin, dblaikie, keith.walker.arm, rengolin
Subscribers: aprantl, friss, mehdi_amini, RKSimon, probinson, majnemer, cfe-commits
Differential Revision: https://reviews.llvm.org/D24998
llvm-svn: 286602
2016-11-11 17:29:56 +00:00
|
|
|
|
2013-06-14 17:17:23 +00:00
|
|
|
return getLastArgIntValue(Args, OPT_O, DefaultOpt, Diags);
|
2012-05-01 14:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return DefaultOpt;
|
|
|
|
}
|
|
|
|
|
2013-04-10 21:30:03 +00:00
|
|
|
static unsigned getOptimizationLevelSize(ArgList &Args) {
|
2012-05-01 14:57:16 +00:00
|
|
|
if (Arg *A = Args.getLastArg(options::OPT_O_Group)) {
|
|
|
|
if (A->getOption().matches(options::OPT_O)) {
|
2012-11-01 04:30:05 +00:00
|
|
|
switch (A->getValue()[0]) {
|
2012-05-01 14:57:16 +00:00
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
case 's':
|
|
|
|
return 1;
|
|
|
|
case 'z':
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
Use -Rblah, not -Wblah, to control remark diagnostics. This was always the
intent when we added remark support, but was never implemented in the general
case, because the first -R flags didn't need it. (-Rpass= had special handling
to accomodate its argument.)
-Rno-foo, -Reverything, and -Rno-everything can be used to turn off a remark,
or to turn on or off all remarks. Per discussion on cfe-commits, -Weverything
does not affect remarks, and -Reverything does not affect warnings or errors.
The only "real" -R flag we have right now is -Rmodule-build; that flag is
effectively renamed from -Wmodule-build to -Rmodule-build by this change.
-Wpass and -Wno-pass (and their friends) are also renamed to -Rpass and
-Rno-pass by this change; it's not completely clear whether we intended to have
a -Rpass (with no =pattern), but that is unchanged by this commit, other than
the flag name. The default pattern is effectively one which matches no passes.
In future, we may want to make the default pattern be .*, so that -Reverything
works for -Rpass properly.
llvm-svn: 215046
2014-08-07 00:24:21 +00:00
|
|
|
static void addDiagnosticArgs(ArgList &Args, OptSpecifier Group,
|
|
|
|
OptSpecifier GroupWithValue,
|
|
|
|
std::vector<std::string> &Diagnostics) {
|
2018-03-26 21:45:04 +00:00
|
|
|
for (auto *A : Args.filtered(Group)) {
|
2012-05-01 14:57:16 +00:00
|
|
|
if (A->getOption().getKind() == Option::FlagClass) {
|
Use -Rblah, not -Wblah, to control remark diagnostics. This was always the
intent when we added remark support, but was never implemented in the general
case, because the first -R flags didn't need it. (-Rpass= had special handling
to accomodate its argument.)
-Rno-foo, -Reverything, and -Rno-everything can be used to turn off a remark,
or to turn on or off all remarks. Per discussion on cfe-commits, -Weverything
does not affect remarks, and -Reverything does not affect warnings or errors.
The only "real" -R flag we have right now is -Rmodule-build; that flag is
effectively renamed from -Wmodule-build to -Rmodule-build by this change.
-Wpass and -Wno-pass (and their friends) are also renamed to -Rpass and
-Rno-pass by this change; it's not completely clear whether we intended to have
a -Rpass (with no =pattern), but that is unchanged by this commit, other than
the flag name. The default pattern is effectively one which matches no passes.
In future, we may want to make the default pattern be .*, so that -Reverything
works for -Rpass properly.
llvm-svn: 215046
2014-08-07 00:24:21 +00:00
|
|
|
// The argument is a pure flag (such as OPT_Wall or OPT_Wdeprecated). Add
|
|
|
|
// its name (minus the "W" or "R" at the beginning) to the warning list.
|
2020-01-28 20:23:46 +01:00
|
|
|
Diagnostics.push_back(
|
|
|
|
std::string(A->getOption().getName().drop_front(1)));
|
Use -Rblah, not -Wblah, to control remark diagnostics. This was always the
intent when we added remark support, but was never implemented in the general
case, because the first -R flags didn't need it. (-Rpass= had special handling
to accomodate its argument.)
-Rno-foo, -Reverything, and -Rno-everything can be used to turn off a remark,
or to turn on or off all remarks. Per discussion on cfe-commits, -Weverything
does not affect remarks, and -Reverything does not affect warnings or errors.
The only "real" -R flag we have right now is -Rmodule-build; that flag is
effectively renamed from -Wmodule-build to -Rmodule-build by this change.
-Wpass and -Wno-pass (and their friends) are also renamed to -Rpass and
-Rno-pass by this change; it's not completely clear whether we intended to have
a -Rpass (with no =pattern), but that is unchanged by this commit, other than
the flag name. The default pattern is effectively one which matches no passes.
In future, we may want to make the default pattern be .*, so that -Reverything
works for -Rpass properly.
llvm-svn: 215046
2014-08-07 00:24:21 +00:00
|
|
|
} else if (A->getOption().matches(GroupWithValue)) {
|
|
|
|
// This is -Wfoo= or -Rfoo=, where foo is the name of the diagnostic group.
|
2020-01-28 20:23:46 +01:00
|
|
|
Diagnostics.push_back(
|
|
|
|
std::string(A->getOption().getName().drop_front(1).rtrim("=-")));
|
2012-05-01 14:57:16 +00:00
|
|
|
} else {
|
Use -Rblah, not -Wblah, to control remark diagnostics. This was always the
intent when we added remark support, but was never implemented in the general
case, because the first -R flags didn't need it. (-Rpass= had special handling
to accomodate its argument.)
-Rno-foo, -Reverything, and -Rno-everything can be used to turn off a remark,
or to turn on or off all remarks. Per discussion on cfe-commits, -Weverything
does not affect remarks, and -Reverything does not affect warnings or errors.
The only "real" -R flag we have right now is -Rmodule-build; that flag is
effectively renamed from -Wmodule-build to -Rmodule-build by this change.
-Wpass and -Wno-pass (and their friends) are also renamed to -Rpass and
-Rno-pass by this change; it's not completely clear whether we intended to have
a -Rpass (with no =pattern), but that is unchanged by this commit, other than
the flag name. The default pattern is effectively one which matches no passes.
In future, we may want to make the default pattern be .*, so that -Reverything
works for -Rpass properly.
llvm-svn: 215046
2014-08-07 00:24:21 +00:00
|
|
|
// Otherwise, add its value (for OPT_W_Joined and similar).
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *Arg : A->getValues())
|
2015-05-29 19:42:19 +00:00
|
|
|
Diagnostics.emplace_back(Arg);
|
2012-05-01 14:57:16 +00:00
|
|
|
}
|
|
|
|
}
|
2010-12-04 01:50:36 +00:00
|
|
|
}
|
|
|
|
|
2018-11-30 21:24:31 +00:00
|
|
|
// Parse the Static Analyzer configuration. If \p Diags is set to nullptr,
|
|
|
|
// it won't verify the input.
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
static void parseAnalyzerConfigs(AnalyzerOptions &AnOpts,
|
2018-11-30 21:24:31 +00:00
|
|
|
DiagnosticsEngine *Diags);
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
|
2016-01-06 14:35:46 +00:00
|
|
|
static void getAllNoBuiltinFuncValues(ArgList &Args,
|
|
|
|
std::vector<std::string> &Funcs) {
|
|
|
|
SmallVector<const char *, 8> Values;
|
|
|
|
for (const auto &Arg : Args) {
|
|
|
|
const Option &O = Arg->getOption();
|
|
|
|
if (O.matches(options::OPT_fno_builtin_)) {
|
|
|
|
const char *FuncName = Arg->getValue();
|
|
|
|
if (Builtin::Context::isBuiltinFunc(FuncName))
|
|
|
|
Values.push_back(FuncName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Funcs.insert(Funcs.end(), Values.begin(), Values.end());
|
|
|
|
}
|
|
|
|
|
2011-12-23 03:05:38 +00:00
|
|
|
static bool ParseAnalyzerArgs(AnalyzerOptions &Opts, ArgList &Args,
|
2011-09-25 23:23:43 +00:00
|
|
|
DiagnosticsEngine &Diags) {
|
2011-12-23 03:05:38 +00:00
|
|
|
bool Success = true;
|
2009-12-01 03:16:53 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_analyzer_store)) {
|
2012-11-01 04:30:05 +00:00
|
|
|
StringRef Name = A->getValue();
|
2009-12-01 03:16:53 +00:00
|
|
|
AnalysisStores Value = llvm::StringSwitch<AnalysisStores>(Name)
|
|
|
|
#define ANALYSIS_STORE(NAME, CMDFLAG, DESC, CREATFN) \
|
|
|
|
.Case(CMDFLAG, NAME##Model)
|
2012-08-31 04:35:58 +00:00
|
|
|
#include "clang/StaticAnalyzer/Core/Analyses.def"
|
2009-12-01 03:16:53 +00:00
|
|
|
.Default(NumStores);
|
2011-12-23 03:05:38 +00:00
|
|
|
if (Value == NumStores) {
|
2009-12-01 03:16:53 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
2010-06-09 22:30:54 +00:00
|
|
|
<< A->getAsString(Args) << Name;
|
2011-12-23 03:05:38 +00:00
|
|
|
Success = false;
|
|
|
|
} else {
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.AnalysisStoreOpt = Value;
|
2011-12-23 03:05:38 +00:00
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Arg *A = Args.getLastArg(OPT_analyzer_constraints)) {
|
2012-11-01 04:30:05 +00:00
|
|
|
StringRef Name = A->getValue();
|
2009-12-01 03:16:53 +00:00
|
|
|
AnalysisConstraints Value = llvm::StringSwitch<AnalysisConstraints>(Name)
|
|
|
|
#define ANALYSIS_CONSTRAINTS(NAME, CMDFLAG, DESC, CREATFN) \
|
|
|
|
.Case(CMDFLAG, NAME##Model)
|
2012-08-31 04:35:58 +00:00
|
|
|
#include "clang/StaticAnalyzer/Core/Analyses.def"
|
2009-12-01 03:16:53 +00:00
|
|
|
.Default(NumConstraints);
|
2011-12-23 03:05:38 +00:00
|
|
|
if (Value == NumConstraints) {
|
2009-12-01 03:16:53 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
2010-06-09 22:30:54 +00:00
|
|
|
<< A->getAsString(Args) << Name;
|
2011-12-23 03:05:38 +00:00
|
|
|
Success = false;
|
|
|
|
} else {
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.AnalysisConstraintsOpt = Value;
|
2011-12-23 03:05:38 +00:00
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Arg *A = Args.getLastArg(OPT_analyzer_output)) {
|
2012-11-01 04:30:05 +00:00
|
|
|
StringRef Name = A->getValue();
|
2009-12-01 03:16:53 +00:00
|
|
|
AnalysisDiagClients Value = llvm::StringSwitch<AnalysisDiagClients>(Name)
|
2013-08-16 01:06:30 +00:00
|
|
|
#define ANALYSIS_DIAGNOSTICS(NAME, CMDFLAG, DESC, CREATFN) \
|
2009-12-01 03:16:53 +00:00
|
|
|
.Case(CMDFLAG, PD_##NAME)
|
2021-01-06 06:35:09 -08:00
|
|
|
#include "clang/Analysis/PathDiagnosticConsumers.def"
|
2009-12-01 03:16:53 +00:00
|
|
|
.Default(NUM_ANALYSIS_DIAG_CLIENTS);
|
2011-12-23 03:05:38 +00:00
|
|
|
if (Value == NUM_ANALYSIS_DIAG_CLIENTS) {
|
2009-12-01 03:16:53 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
2010-06-09 22:30:54 +00:00
|
|
|
<< A->getAsString(Args) << Name;
|
2011-12-23 03:05:38 +00:00
|
|
|
Success = false;
|
|
|
|
} else {
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.AnalysisDiagOpt = Value;
|
2011-12-23 03:05:38 +00:00
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
2011-09-30 02:03:00 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_analyzer_purge)) {
|
2012-11-01 04:30:05 +00:00
|
|
|
StringRef Name = A->getValue();
|
2011-09-30 02:03:00 +00:00
|
|
|
AnalysisPurgeMode Value = llvm::StringSwitch<AnalysisPurgeMode>(Name)
|
|
|
|
#define ANALYSIS_PURGE(NAME, CMDFLAG, DESC) \
|
|
|
|
.Case(CMDFLAG, NAME)
|
2012-08-31 04:35:58 +00:00
|
|
|
#include "clang/StaticAnalyzer/Core/Analyses.def"
|
2011-09-30 02:03:00 +00:00
|
|
|
.Default(NumPurgeModes);
|
2011-12-23 03:05:38 +00:00
|
|
|
if (Value == NumPurgeModes) {
|
2011-09-30 02:03:00 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< A->getAsString(Args) << Name;
|
2011-12-23 03:05:38 +00:00
|
|
|
Success = false;
|
|
|
|
} else {
|
2011-09-30 02:03:00 +00:00
|
|
|
Opts.AnalysisPurgeOpt = Value;
|
2011-12-23 03:05:38 +00:00
|
|
|
}
|
2011-09-30 02:03:00 +00:00
|
|
|
}
|
|
|
|
|
2012-03-08 23:16:35 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_analyzer_inlining_mode)) {
|
2012-11-01 04:30:05 +00:00
|
|
|
StringRef Name = A->getValue();
|
2012-03-08 23:16:35 +00:00
|
|
|
AnalysisInliningMode Value = llvm::StringSwitch<AnalysisInliningMode>(Name)
|
|
|
|
#define ANALYSIS_INLINING_MODE(NAME, CMDFLAG, DESC) \
|
|
|
|
.Case(CMDFLAG, NAME)
|
2012-08-31 04:35:58 +00:00
|
|
|
#include "clang/StaticAnalyzer/Core/Analyses.def"
|
2012-03-08 23:16:35 +00:00
|
|
|
.Default(NumInliningModes);
|
|
|
|
if (Value == NumInliningModes) {
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< A->getAsString(Args) << Name;
|
|
|
|
Success = false;
|
|
|
|
} else {
|
|
|
|
Opts.InliningMode = Value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-30 21:24:31 +00:00
|
|
|
Opts.ShouldEmitErrorsOnInvalidConfigValue =
|
|
|
|
/* negated */!llvm::StringSwitch<bool>(
|
|
|
|
Args.getLastArgValue(OPT_analyzer_config_compatibility_mode))
|
|
|
|
.Case("true", true)
|
|
|
|
.Case("false", false)
|
|
|
|
.Default(false);
|
2014-08-29 20:01:38 +00:00
|
|
|
|
2019-08-16 01:53:14 +00:00
|
|
|
Opts.CheckersAndPackages.clear();
|
2015-06-09 01:57:17 +00:00
|
|
|
for (const Arg *A :
|
|
|
|
Args.filtered(OPT_analyzer_checker, OPT_analyzer_disable_checker)) {
|
2011-02-14 18:13:31 +00:00
|
|
|
A->claim();
|
2019-08-16 01:53:14 +00:00
|
|
|
bool IsEnabled = A->getOption().getID() == OPT_analyzer_checker;
|
2011-02-24 08:42:20 +00:00
|
|
|
// We can have a list of comma separated checker names, e.g:
|
|
|
|
// '-analyzer-checker=cocoa,unix'
|
2019-08-16 01:53:14 +00:00
|
|
|
StringRef CheckerAndPackageList = A->getValue();
|
|
|
|
SmallVector<StringRef, 16> CheckersAndPackages;
|
|
|
|
CheckerAndPackageList.split(CheckersAndPackages, ",");
|
2019-12-17 21:52:47 +01:00
|
|
|
for (const StringRef &CheckerOrPackage : CheckersAndPackages)
|
2020-01-29 02:48:15 +01:00
|
|
|
Opts.CheckersAndPackages.emplace_back(std::string(CheckerOrPackage),
|
|
|
|
IsEnabled);
|
2011-02-14 18:13:31 +00:00
|
|
|
}
|
2015-02-24 21:45:33 +00:00
|
|
|
|
2012-08-29 05:55:00 +00:00
|
|
|
// Go through the analyzer configuration options.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_analyzer_config)) {
|
2018-11-30 21:24:31 +00:00
|
|
|
|
2012-08-29 05:55:00 +00:00
|
|
|
// We can have a list of comma separated config names, e.g:
|
2012-08-30 05:49:16 +00:00
|
|
|
// '-analyzer-config key1=val1,key2=val2'
|
2012-11-01 04:30:05 +00:00
|
|
|
StringRef configList = A->getValue();
|
2012-08-29 05:55:00 +00:00
|
|
|
SmallVector<StringRef, 4> configVals;
|
|
|
|
configList.split(configVals, ",");
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto &configVal : configVals) {
|
2012-08-29 05:55:00 +00:00
|
|
|
StringRef key, val;
|
2018-03-26 21:45:04 +00:00
|
|
|
std::tie(key, val) = configVal.split("=");
|
2012-08-29 05:55:00 +00:00
|
|
|
if (val.empty()) {
|
|
|
|
Diags.Report(SourceLocation(),
|
2018-03-26 21:45:04 +00:00
|
|
|
diag::err_analyzer_config_no_value) << configVal;
|
2012-08-29 05:55:00 +00:00
|
|
|
Success = false;
|
|
|
|
break;
|
|
|
|
}
|
2012-08-30 05:49:16 +00:00
|
|
|
if (val.find('=') != StringRef::npos) {
|
2012-08-29 05:55:00 +00:00
|
|
|
Diags.Report(SourceLocation(),
|
|
|
|
diag::err_analyzer_config_multiple_values)
|
2018-03-26 21:45:04 +00:00
|
|
|
<< configVal;
|
2012-08-29 05:55:00 +00:00
|
|
|
Success = false;
|
|
|
|
break;
|
|
|
|
}
|
2018-11-30 21:24:31 +00:00
|
|
|
|
|
|
|
// TODO: Check checker options too, possibly in CheckerRegistry.
|
|
|
|
// Leave unknown non-checker configs unclaimed.
|
|
|
|
if (!key.contains(":") && Opts.isUnknownAnalyzerConfig(key)) {
|
|
|
|
if (Opts.ShouldEmitErrorsOnInvalidConfigValue)
|
|
|
|
Diags.Report(diag::err_analyzer_config_unknown) << key;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
A->claim();
|
2020-01-28 20:23:46 +01:00
|
|
|
Opts.Config[key] = std::string(val);
|
2012-08-29 05:55:00 +00:00
|
|
|
}
|
|
|
|
}
|
2011-12-23 03:05:38 +00:00
|
|
|
|
2018-11-30 21:24:31 +00:00
|
|
|
if (Opts.ShouldEmitErrorsOnInvalidConfigValue)
|
|
|
|
parseAnalyzerConfigs(Opts, &Diags);
|
|
|
|
else
|
|
|
|
parseAnalyzerConfigs(Opts, nullptr);
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
|
2018-01-23 19:28:52 +00:00
|
|
|
llvm::raw_string_ostream os(Opts.FullCompilerInvocation);
|
2018-03-26 21:45:04 +00:00
|
|
|
for (unsigned i = 0; i < Args.getNumInputArgStrings(); ++i) {
|
2018-01-23 19:28:52 +00:00
|
|
|
if (i != 0)
|
|
|
|
os << " ";
|
|
|
|
os << Args.getArgString(i);
|
|
|
|
}
|
|
|
|
os.flush();
|
|
|
|
|
2011-12-23 03:05:38 +00:00
|
|
|
return Success;
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
static StringRef getStringOption(AnalyzerOptions::ConfigTable &Config,
|
|
|
|
StringRef OptionName, StringRef DefaultVal) {
|
2020-01-28 20:23:46 +01:00
|
|
|
return Config.insert({OptionName, std::string(DefaultVal)}).first->second;
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void initOption(AnalyzerOptions::ConfigTable &Config,
|
2018-11-30 21:24:31 +00:00
|
|
|
DiagnosticsEngine *Diags,
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
StringRef &OptionField, StringRef Name,
|
|
|
|
StringRef DefaultVal) {
|
2018-11-30 21:24:31 +00:00
|
|
|
// String options may be known to invalid (e.g. if the expected string is a
|
|
|
|
// file name, but the file does not exist), those will have to be checked in
|
|
|
|
// parseConfigs.
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
OptionField = getStringOption(Config, Name, DefaultVal);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void initOption(AnalyzerOptions::ConfigTable &Config,
|
2018-11-30 21:24:31 +00:00
|
|
|
DiagnosticsEngine *Diags,
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
bool &OptionField, StringRef Name, bool DefaultVal) {
|
2018-11-30 21:24:31 +00:00
|
|
|
auto PossiblyInvalidVal = llvm::StringSwitch<Optional<bool>>(
|
|
|
|
getStringOption(Config, Name, (DefaultVal ? "true" : "false")))
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
.Case("true", true)
|
|
|
|
.Case("false", false)
|
2018-11-30 21:24:31 +00:00
|
|
|
.Default(None);
|
|
|
|
|
|
|
|
if (!PossiblyInvalidVal) {
|
|
|
|
if (Diags)
|
|
|
|
Diags->Report(diag::err_analyzer_config_invalid_input)
|
|
|
|
<< Name << "a boolean";
|
|
|
|
else
|
|
|
|
OptionField = DefaultVal;
|
|
|
|
} else
|
|
|
|
OptionField = PossiblyInvalidVal.getValue();
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void initOption(AnalyzerOptions::ConfigTable &Config,
|
2018-11-30 21:24:31 +00:00
|
|
|
DiagnosticsEngine *Diags,
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
unsigned &OptionField, StringRef Name,
|
|
|
|
unsigned DefaultVal) {
|
2018-11-30 21:24:31 +00:00
|
|
|
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
OptionField = DefaultVal;
|
|
|
|
bool HasFailed = getStringOption(Config, Name, std::to_string(DefaultVal))
|
[analyzer][NFC] Reimplement checker options
TL;DR:
* Add checker and package options to the TableGen files
* Added a new class called CmdLineOption, and both Package and Checker recieved
a list<CmdLineOption> field.
* Added every existing checker and package option to Checkers.td.
* The CheckerRegistry class
* Received some comments to most of it's inline classes
* Received the CmdLineOption and PackageInfo inline classes, a list of
CmdLineOption was added to CheckerInfo and PackageInfo
* Added addCheckerOption and addPackageOption
* Added a new field called Packages, used in addPackageOptions, filled up in
addPackage
Detailed description:
In the last couple months, a lot of effort was put into tightening the
analyzer's command line interface. The main issue is that it's spectacularly
easy to mess up a lenghty enough invocation of the analyzer, and the user was
given no warnings or errors at all in that case.
We can divide the effort of resolving this into several chapters:
* Non-checker analyzer configurations:
Gather every analyzer configuration into a dedicated file. Emit errors for
non-existent configurations or incorrect values. Be able to list these
configurations. Tighten AnalyzerOptions interface to disallow making such
a mistake in the future.
* Fix the "Checker Naming Bug" by reimplementing checker dependencies:
When cplusplus.InnerPointer was enabled, it implicitly registered
unix.Malloc, which implicitly registered some sort of a modeling checker
from the CStringChecker family. This resulted in all of these checker
objects recieving the name "cplusplus.InnerPointer", making AnalyzerOptions
asking for the wrong checker options from the command line:
cplusplus.InnerPointer:Optimisic
istead of
unix.Malloc:Optimistic.
This was resolved by making CheckerRegistry responsible for checker
dependency handling, instead of checkers themselves.
* Checker options: (this patch included!)
Same as the first item, but for checkers.
(+ minor fixes here and there, and everything else that is yet to come)
There were several issues regarding checker options, that non-checker
configurations didn't suffer from: checker plugins are loaded runtime, and they
could add new checkers and new options, meaning that unlike for non-checker
configurations, we can't collect every checker option purely by generating code.
Also, as seen from the "Checker Naming Bug" issue raised above, they are very
rarely used in practice, and all sorts of skeletons fell out of the closet while
working on this project.
They were extremely problematic for users as well, purely because of how long
they were. Consider the following monster of a checker option:
alpha.cplusplus.UninitializedObject:CheckPointeeInitialization=false
While we were able to verify whether the checker itself (the part before the
colon) existed, any errors past that point were unreported, easily resulting
in 7+ hours of analyses going to waste.
This patch, similarly to how dependencies were reimplemented, uses TableGen to
register checker options into Checkers.td, so that Checkers.inc now contains
entries for both checker and package options. Using the preprocessor,
Checkers.inc is converted into code in CheckerRegistry, adding every builtin
(checkers and packages that have an entry in the Checkers.td file) checker and
package option to the registry. The new addPackageOption and addCheckerOption
functions expose the same functionality to statically-linked non-builtin and
plugin checkers and packages as well.
Emitting errors for incorrect user input, being able to list these options, and
some other functionalies will land in later patches.
Differential Revision: https://reviews.llvm.org/D57855
llvm-svn: 358752
2019-04-19 12:32:10 +00:00
|
|
|
.getAsInteger(0, OptionField);
|
2018-11-30 21:24:31 +00:00
|
|
|
if (Diags && HasFailed)
|
|
|
|
Diags->Report(diag::err_analyzer_config_invalid_input)
|
|
|
|
<< Name << "an unsigned";
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void parseAnalyzerConfigs(AnalyzerOptions &AnOpts,
|
2018-11-30 21:24:31 +00:00
|
|
|
DiagnosticsEngine *Diags) {
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
// TODO: There's no need to store the entire configtable, it'd be plenty
|
|
|
|
// enough tostore checker options.
|
|
|
|
|
|
|
|
#define ANALYZER_OPTION(TYPE, NAME, CMDFLAG, DESC, DEFAULT_VAL) \
|
2018-11-30 21:24:31 +00:00
|
|
|
initOption(AnOpts.Config, Diags, AnOpts.NAME, CMDFLAG, DEFAULT_VAL);
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
|
|
|
|
#define ANALYZER_OPTION_DEPENDS_ON_USER_MODE(TYPE, NAME, CMDFLAG, DESC, \
|
|
|
|
SHALLOW_VAL, DEEP_VAL) \
|
|
|
|
switch (AnOpts.getUserMode()) { \
|
|
|
|
case UMK_Shallow: \
|
2018-11-30 21:24:31 +00:00
|
|
|
initOption(AnOpts.Config, Diags, AnOpts.NAME, CMDFLAG, SHALLOW_VAL); \
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
break; \
|
|
|
|
case UMK_Deep: \
|
2018-11-30 21:24:31 +00:00
|
|
|
initOption(AnOpts.Config, Diags, AnOpts.NAME, CMDFLAG, DEEP_VAL); \
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
break; \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#include "clang/StaticAnalyzer/Core/AnalyzerOptions.def"
|
|
|
|
#undef ANALYZER_OPTION
|
|
|
|
#undef ANALYZER_OPTION_DEPENDS_ON_USER_MODE
|
2018-11-30 21:24:31 +00:00
|
|
|
|
|
|
|
// At this point, AnalyzerOptions is configured. Let's validate some options.
|
|
|
|
|
2019-08-16 01:53:14 +00:00
|
|
|
// FIXME: Here we try to validate the silenced checkers or packages are valid.
|
|
|
|
// The current approach only validates the registered checkers which does not
|
|
|
|
// contain the runtime enabled checkers and optimally we would validate both.
|
|
|
|
if (!AnOpts.RawSilencedCheckersAndPackages.empty()) {
|
|
|
|
std::vector<StringRef> Checkers =
|
|
|
|
AnOpts.getRegisteredCheckers(/*IncludeExperimental=*/true);
|
|
|
|
std::vector<StringRef> Packages =
|
|
|
|
AnOpts.getRegisteredPackages(/*IncludeExperimental=*/true);
|
|
|
|
|
|
|
|
SmallVector<StringRef, 16> CheckersAndPackages;
|
|
|
|
AnOpts.RawSilencedCheckersAndPackages.split(CheckersAndPackages, ";");
|
|
|
|
|
2019-12-17 21:52:47 +01:00
|
|
|
for (const StringRef &CheckerOrPackage : CheckersAndPackages) {
|
2019-08-24 12:17:49 +00:00
|
|
|
if (Diags) {
|
|
|
|
bool IsChecker = CheckerOrPackage.contains('.');
|
|
|
|
bool IsValidName =
|
|
|
|
IsChecker
|
|
|
|
? llvm::find(Checkers, CheckerOrPackage) != Checkers.end()
|
|
|
|
: llvm::find(Packages, CheckerOrPackage) != Packages.end();
|
|
|
|
|
|
|
|
if (!IsValidName)
|
|
|
|
Diags->Report(diag::err_unknown_analyzer_checker_or_package)
|
|
|
|
<< CheckerOrPackage;
|
|
|
|
}
|
2019-08-16 01:53:14 +00:00
|
|
|
|
|
|
|
AnOpts.SilencedCheckersAndPackages.emplace_back(CheckerOrPackage);
|
|
|
|
}
|
|
|
|
}
|
2019-08-24 12:17:49 +00:00
|
|
|
|
|
|
|
if (!Diags)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (AnOpts.ShouldTrackConditionsDebug && !AnOpts.ShouldTrackConditions)
|
|
|
|
Diags->Report(diag::err_analyzer_config_invalid_input)
|
|
|
|
<< "track-conditions-debug" << "'track-conditions' to also be enabled";
|
|
|
|
|
|
|
|
if (!AnOpts.CTUDir.empty() && !llvm::sys::fs::is_directory(AnOpts.CTUDir))
|
|
|
|
Diags->Report(diag::err_analyzer_config_invalid_input) << "ctu-dir"
|
|
|
|
<< "a filename";
|
|
|
|
|
|
|
|
if (!AnOpts.ModelPath.empty() &&
|
|
|
|
!llvm::sys::fs::is_directory(AnOpts.ModelPath))
|
|
|
|
Diags->Report(diag::err_analyzer_config_invalid_input) << "model-path"
|
|
|
|
<< "a filename";
|
[analyzer] Evaluate all non-checker config options before analysis
In earlier patches regarding AnalyzerOptions, a lot of effort went into
gathering all config options, and changing the interface so that potential
misuse can be eliminited.
Up until this point, AnalyzerOptions only evaluated an option when it was
querried. For example, if we had a "-no-false-positives" flag, AnalyzerOptions
would store an Optional field for it that would be None up until somewhere in
the code until the flag's getter function is called.
However, now that we're confident that we've gathered all configs, we can
evaluate off of them before analysis, so we can emit a error on invalid input
even if that prticular flag will not matter in that particular run of the
analyzer. Another very big benefit of this is that debug.ConfigDumper will now
show the value of all configs every single time.
Also, almost all options related class have a similar interface, so uniformity
is also a benefit.
The implementation for errors on invalid input will be commited shorty.
Differential Revision: https://reviews.llvm.org/D53692
llvm-svn: 348031
2018-11-30 20:44:00 +00:00
|
|
|
}
|
|
|
|
|
2013-02-22 14:21:27 +00:00
|
|
|
static void ParseCommentArgs(CommentOptions &Opts, ArgList &Args) {
|
2013-04-10 15:35:17 +00:00
|
|
|
Opts.ParseAllComments = Args.hasArg(OPT_fparse_all_comments);
|
2013-02-22 14:21:27 +00:00
|
|
|
}
|
|
|
|
|
2018-05-09 01:00:01 +00:00
|
|
|
/// Create a new Regex instance out of the string value in \p RpassArg.
|
2014-05-29 19:55:06 +00:00
|
|
|
/// It returns a pointer to the newly generated Regex instance.
|
|
|
|
static std::shared_ptr<llvm::Regex>
|
|
|
|
GenerateOptimizationRemarkRegex(DiagnosticsEngine &Diags, ArgList &Args,
|
|
|
|
Arg *RpassArg) {
|
|
|
|
StringRef Val = RpassArg->getValue();
|
|
|
|
std::string RegexError;
|
|
|
|
std::shared_ptr<llvm::Regex> Pattern = std::make_shared<llvm::Regex>(Val);
|
|
|
|
if (!Pattern->isValid(RegexError)) {
|
|
|
|
Diags.Report(diag::err_drv_optimization_remark_pattern)
|
|
|
|
<< RegexError << RpassArg->getAsString(Args);
|
|
|
|
Pattern.reset();
|
|
|
|
}
|
|
|
|
return Pattern;
|
|
|
|
}
|
|
|
|
|
2015-06-13 07:11:40 +00:00
|
|
|
static bool parseDiagnosticLevelMask(StringRef FlagName,
|
|
|
|
const std::vector<std::string> &Levels,
|
|
|
|
DiagnosticsEngine *Diags,
|
|
|
|
DiagnosticLevelMask &M) {
|
|
|
|
bool Success = true;
|
|
|
|
for (const auto &Level : Levels) {
|
|
|
|
DiagnosticLevelMask const PM =
|
|
|
|
llvm::StringSwitch<DiagnosticLevelMask>(Level)
|
|
|
|
.Case("note", DiagnosticLevelMask::Note)
|
|
|
|
.Case("remark", DiagnosticLevelMask::Remark)
|
|
|
|
.Case("warning", DiagnosticLevelMask::Warning)
|
|
|
|
.Case("error", DiagnosticLevelMask::Error)
|
|
|
|
.Default(DiagnosticLevelMask::None);
|
|
|
|
if (PM == DiagnosticLevelMask::None) {
|
|
|
|
Success = false;
|
|
|
|
if (Diags)
|
|
|
|
Diags->Report(diag::err_drv_invalid_value) << FlagName << Level;
|
|
|
|
}
|
|
|
|
M = M | PM;
|
|
|
|
}
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
Reimplement -fsanitize-recover family of flags.
Introduce the following -fsanitize-recover flags:
- -fsanitize-recover=<list>: Enable recovery for selected checks or
group of checks. It is forbidden to explicitly list unrecoverable
sanitizers here (that is, "address", "unreachable", "return").
- -fno-sanitize-recover=<list>: Disable recovery for selected checks or
group of checks.
- -f(no-)?sanitize-recover is now a synonym for
-f(no-)?sanitize-recover=undefined,integer and will soon be deprecated.
These flags are parsed left to right, and mask of "recoverable"
sanitizer is updated accordingly, much like what we do for -fsanitize= flags.
-fsanitize= and -fsanitize-recover= flag families are independent.
CodeGen change: If there is a single UBSan handler function, responsible
for implementing multiple checks, which have different recoverable setting,
then we emit two handler calls instead of one:
the first one for the set of "unrecoverable" checks, another one - for
set of "recoverable" checks. If all checks implemented by a handler have the
same recoverability setting, then the generated code will be the same.
llvm-svn: 225719
2015-01-12 22:39:12 +00:00
|
|
|
static void parseSanitizerKinds(StringRef FlagName,
|
|
|
|
const std::vector<std::string> &Sanitizers,
|
|
|
|
DiagnosticsEngine &Diags, SanitizerSet &S) {
|
|
|
|
for (const auto &Sanitizer : Sanitizers) {
|
2015-05-11 21:39:20 +00:00
|
|
|
SanitizerMask K = parseSanitizerValue(Sanitizer, /*AllowGroups=*/false);
|
2019-03-01 10:05:15 +00:00
|
|
|
if (K == SanitizerMask())
|
Reimplement -fsanitize-recover family of flags.
Introduce the following -fsanitize-recover flags:
- -fsanitize-recover=<list>: Enable recovery for selected checks or
group of checks. It is forbidden to explicitly list unrecoverable
sanitizers here (that is, "address", "unreachable", "return").
- -fno-sanitize-recover=<list>: Disable recovery for selected checks or
group of checks.
- -f(no-)?sanitize-recover is now a synonym for
-f(no-)?sanitize-recover=undefined,integer and will soon be deprecated.
These flags are parsed left to right, and mask of "recoverable"
sanitizer is updated accordingly, much like what we do for -fsanitize= flags.
-fsanitize= and -fsanitize-recover= flag families are independent.
CodeGen change: If there is a single UBSan handler function, responsible
for implementing multiple checks, which have different recoverable setting,
then we emit two handler calls instead of one:
the first one for the set of "unrecoverable" checks, another one - for
set of "recoverable" checks. If all checks implemented by a handler have the
same recoverability setting, then the generated code will be the same.
llvm-svn: 225719
2015-01-12 22:39:12 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_value) << FlagName << Sanitizer;
|
|
|
|
else
|
|
|
|
S.set(K, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-13 02:31:58 +00:00
|
|
|
static void parseXRayInstrumentationBundle(StringRef FlagName, StringRef Bundle,
|
|
|
|
ArgList &Args, DiagnosticsEngine &D,
|
|
|
|
XRayInstrSet &S) {
|
|
|
|
llvm::SmallVector<StringRef, 2> BundleParts;
|
|
|
|
llvm::SplitString(Bundle, BundleParts, ",");
|
2019-12-17 21:52:47 +01:00
|
|
|
for (const auto &B : BundleParts) {
|
2018-04-13 02:31:58 +00:00
|
|
|
auto Mask = parseXRayInstrValue(B);
|
|
|
|
if (Mask == XRayInstrKind::None)
|
|
|
|
if (B != "none")
|
|
|
|
D.Report(diag::err_drv_invalid_value) << FlagName << Bundle;
|
|
|
|
else
|
|
|
|
S.Mask = Mask;
|
|
|
|
else if (Mask == XRayInstrKind::All)
|
|
|
|
S.Mask = Mask;
|
|
|
|
else
|
|
|
|
S.set(Mask, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-02 20:59:36 +00:00
|
|
|
// Set the profile kind using fprofile-instrument-use-path.
|
|
|
|
static void setPGOUseInstrumentor(CodeGenOptions &Opts,
|
2016-05-29 11:04:56 +00:00
|
|
|
const Twine &ProfileName) {
|
2016-03-02 20:59:36 +00:00
|
|
|
auto ReaderOrErr = llvm::IndexedInstrProfReader::create(ProfileName);
|
|
|
|
// In error, return silently and let Clang PGOUse report the error message.
|
2016-05-19 03:54:54 +00:00
|
|
|
if (auto E = ReaderOrErr.takeError()) {
|
|
|
|
llvm::consumeError(std::move(E));
|
2016-03-02 20:59:36 +00:00
|
|
|
Opts.setProfileUse(CodeGenOptions::ProfileClangInstr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
std::unique_ptr<llvm::IndexedInstrProfReader> PGOReader =
|
|
|
|
std::move(ReaderOrErr.get());
|
2019-03-04 20:21:31 +00:00
|
|
|
if (PGOReader->isIRLevelProfile()) {
|
|
|
|
if (PGOReader->hasCSIRLevelProfile())
|
|
|
|
Opts.setProfileUse(CodeGenOptions::ProfileCSIRInstr);
|
|
|
|
else
|
|
|
|
Opts.setProfileUse(CodeGenOptions::ProfileIRInstr);
|
|
|
|
} else
|
2016-03-02 20:59:36 +00:00
|
|
|
Opts.setProfileUse(CodeGenOptions::ProfileClangInstr);
|
|
|
|
}
|
|
|
|
|
2011-12-23 03:05:38 +00:00
|
|
|
static bool ParseCodeGenArgs(CodeGenOptions &Opts, ArgList &Args, InputKind IK,
|
2014-01-07 01:19:08 +00:00
|
|
|
DiagnosticsEngine &Diags,
|
2018-04-17 16:39:25 +00:00
|
|
|
const TargetOptions &TargetOpts,
|
|
|
|
const FrontendOptions &FrontendOpts) {
|
2011-12-23 03:05:38 +00:00
|
|
|
bool Success = true;
|
2015-12-10 18:52:50 +00:00
|
|
|
llvm::Triple Triple = llvm::Triple(TargetOpts.Triple);
|
2010-12-04 01:50:36 +00:00
|
|
|
|
2014-12-16 21:57:03 +00:00
|
|
|
unsigned OptimizationLevel = getOptimizationLevel(Args, IK, Diags);
|
2013-11-18 13:23:07 +00:00
|
|
|
// TODO: This could be done in Driver
|
|
|
|
unsigned MaxOptLevel = 3;
|
2014-12-16 21:57:03 +00:00
|
|
|
if (OptimizationLevel > MaxOptLevel) {
|
|
|
|
// If the optimization level is not supported, fall back on the default
|
|
|
|
// optimization
|
2013-11-18 13:23:07 +00:00
|
|
|
Diags.Report(diag::warn_drv_optimization_value)
|
|
|
|
<< Args.getLastArg(OPT_O)->getAsString(Args) << "-O" << MaxOptLevel;
|
2014-12-16 21:57:03 +00:00
|
|
|
OptimizationLevel = MaxOptLevel;
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
2014-12-16 21:57:03 +00:00
|
|
|
Opts.OptimizationLevel = OptimizationLevel;
|
2009-12-01 03:16:53 +00:00
|
|
|
|
Cleanup the handling of noinline function attributes, -fno-inline,
-fno-inline-functions, -O0, and optnone.
These were really, really tangled together:
- We used the noinline LLVM attribute for -fno-inline
- But not for -fno-inline-functions (breaking LTO)
- But we did use it for -finline-hint-functions (yay, LTO is happy!)
- But we didn't for -O0 (LTO is sad yet again...)
- We had weird structuring of CodeGenOpts with both an inlining
enumeration and a boolean. They interacted in weird ways and
needlessly.
- A *lot* of set smashing went on with setting these, and then got worse
when we considered optnone and other inlining-effecting attributes.
- A bunch of inline affecting attributes were managed in a completely
different place from -fno-inline.
- Even with -fno-inline we failed to put the LLVM noinline attribute
onto many generated function definitions because they didn't show up
as AST-level functions.
- If you passed -O0 but -finline-functions we would run the normal
inliner pass in LLVM despite it being in the O0 pipeline, which really
doesn't make much sense.
- Lastly, we used things like '-fno-inline' to manipulate the pass
pipeline which forced the pass pipeline to be much more
parameterizable than it really needs to be. Instead we can *just* use
the optimization level to select a pipeline and control the rest via
attributes.
Sadly, this causes a bunch of churn in tests because we don't run the
optimizer in the tests and check the contents of attribute sets. It
would be awesome if attribute sets were a bit more FileCheck friendly,
but oh well.
I think this is a significant improvement and should remove the semantic
need to change what inliner pass we run in order to comply with the
requested inlining semantics by relying completely on attributes. It
also cleans up tho optnone and related handling a bit.
One unfortunate aspect of this is that for generating alwaysinline
routines like those in OpenMP we end up removing noinline and then
adding alwaysinline. I tried a bunch of other approaches, but because we
recompute function attributes from scratch and don't have a declaration
here I couldn't find anything substantially cleaner than this.
Differential Revision: https://reviews.llvm.org/D28053
llvm-svn: 290398
2016-12-23 01:24:49 +00:00
|
|
|
// At O0 we want to fully disable inlining outside of cases marked with
|
|
|
|
// 'alwaysinline' that are required for correctness.
|
|
|
|
Opts.setInlining((Opts.OptimizationLevel == 0)
|
|
|
|
? CodeGenOptions::OnlyAlwaysInlining
|
|
|
|
: CodeGenOptions::NormalInlining);
|
|
|
|
// Explicit inlining flags can disable some or all inlining even at
|
|
|
|
// optimization levels above zero.
|
|
|
|
if (Arg *InlineArg = Args.getLastArg(
|
|
|
|
options::OPT_finline_functions, options::OPT_finline_hint_functions,
|
|
|
|
options::OPT_fno_inline_functions, options::OPT_fno_inline)) {
|
|
|
|
if (Opts.OptimizationLevel > 0) {
|
|
|
|
const Option &InlineOpt = InlineArg->getOption();
|
|
|
|
if (InlineOpt.matches(options::OPT_finline_functions))
|
|
|
|
Opts.setInlining(CodeGenOptions::NormalInlining);
|
|
|
|
else if (InlineOpt.matches(options::OPT_finline_hint_functions))
|
|
|
|
Opts.setInlining(CodeGenOptions::OnlyHintInlining);
|
|
|
|
else
|
|
|
|
Opts.setInlining(CodeGenOptions::OnlyAlwaysInlining);
|
|
|
|
}
|
2016-05-24 20:40:51 +00:00
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
|
2020-08-13 15:48:55 -07:00
|
|
|
// If -fuse-ctor-homing is set and limited debug info is already on, then use
|
|
|
|
// constructor homing.
|
2020-08-16 21:49:13 -07:00
|
|
|
if (Args.getLastArg(OPT_fuse_ctor_homing))
|
2020-08-13 15:48:55 -07:00
|
|
|
if (Opts.getDebugInfo() == codegenoptions::LimitedDebugInfo)
|
|
|
|
Opts.setDebugInfo(codegenoptions::DebugInfoConstructor);
|
|
|
|
|
2020-01-29 00:42:56 +01:00
|
|
|
for (const auto &Arg : Args.getAllArgValues(OPT_fdebug_prefix_map_EQ)) {
|
|
|
|
auto Split = StringRef(Arg).split('=');
|
|
|
|
Opts.DebugPrefixMap.insert(
|
|
|
|
{std::string(Split.first), std::string(Split.second)});
|
|
|
|
}
|
2015-10-12 20:21:08 +00:00
|
|
|
|
2020-02-20 13:43:01 +01:00
|
|
|
const llvm::Triple::ArchType DebugEntryValueArchs[] = {
|
|
|
|
llvm::Triple::x86, llvm::Triple::x86_64, llvm::Triple::aarch64,
|
2020-05-15 09:24:02 +02:00
|
|
|
llvm::Triple::arm, llvm::Triple::armeb, llvm::Triple::mips,
|
|
|
|
llvm::Triple::mipsel, llvm::Triple::mips64, llvm::Triple::mips64el};
|
2020-02-20 13:43:01 +01:00
|
|
|
|
|
|
|
llvm::Triple T(TargetOpts.Triple);
|
|
|
|
if (Opts.OptimizationLevel > 0 && Opts.hasReducedDebugInfo() &&
|
2020-03-19 12:13:18 +01:00
|
|
|
llvm::is_contained(DebugEntryValueArchs, T.getArch()))
|
2020-03-09 11:02:35 +01:00
|
|
|
Opts.EmitCallSiteInfo = true;
|
2020-02-20 13:43:01 +01:00
|
|
|
|
2017-11-20 11:16:16 +00:00
|
|
|
Opts.NewStructPathTBAA = !Args.hasArg(OPT_no_struct_path_tbaa) &&
|
|
|
|
Args.hasArg(OPT_new_struct_path_tbaa);
|
2013-04-10 21:30:03 +00:00
|
|
|
Opts.OptimizeSize = getOptimizationLevelSize(Args);
|
2010-06-07 23:19:17 +00:00
|
|
|
Opts.SimplifyLibCalls = !(Args.hasArg(OPT_fno_builtin) ||
|
|
|
|
Args.hasArg(OPT_ffreestanding));
|
2016-01-06 14:35:46 +00:00
|
|
|
if (Opts.SimplifyLibCalls)
|
|
|
|
getAllNoBuiltinFuncValues(Args, Opts.NoBuiltinFuncs);
|
2013-08-08 08:34:35 +00:00
|
|
|
Opts.UnrollLoops =
|
|
|
|
Args.hasFlag(OPT_funroll_loops, OPT_fno_unroll_loops,
|
2016-05-04 15:26:28 +00:00
|
|
|
(Opts.OptimizationLevel > 1));
|
2021-01-05 17:00:46 +01:00
|
|
|
|
2018-08-20 20:14:08 +00:00
|
|
|
Opts.DebugNameTable = static_cast<unsigned>(
|
|
|
|
Args.hasArg(OPT_ggnu_pubnames)
|
|
|
|
? llvm::DICompileUnit::DebugNameTableKind::GNU
|
|
|
|
: Args.hasArg(OPT_gpubnames)
|
|
|
|
? llvm::DICompileUnit::DebugNameTableKind::Default
|
|
|
|
: llvm::DICompileUnit::DebugNameTableKind::None);
|
2021-01-05 17:00:46 +01:00
|
|
|
|
2016-03-02 20:59:36 +00:00
|
|
|
if (!Opts.ProfileInstrumentUsePath.empty())
|
|
|
|
setPGOUseInstrumentor(Opts, Opts.ProfileInstrumentUsePath);
|
2016-02-04 18:39:09 +00:00
|
|
|
|
2018-10-08 22:25:20 +00:00
|
|
|
Opts.CodeModel = TargetOpts.CodeModel;
|
2019-07-20 22:50:50 +00:00
|
|
|
|
2020-12-02 10:18:18 -08:00
|
|
|
if (const Arg *A = Args.getLastArg(OPT_ftime_report, OPT_ftime_report_EQ)) {
|
|
|
|
Opts.TimePasses = true;
|
|
|
|
|
|
|
|
// -ftime-report= is only for new pass manager.
|
|
|
|
if (A->getOption().getID() == OPT_ftime_report_EQ) {
|
2020-12-09 16:57:36 -08:00
|
|
|
if (Opts.LegacyPassManager)
|
2020-12-02 10:18:18 -08:00
|
|
|
Diags.Report(diag::err_drv_argument_only_allowed_with)
|
2020-12-09 16:57:36 -08:00
|
|
|
<< A->getAsString(Args) << "-fno-legacy-pass-manager";
|
2020-12-02 10:18:18 -08:00
|
|
|
|
|
|
|
StringRef Val = A->getValue();
|
|
|
|
if (Val == "per-pass")
|
|
|
|
Opts.TimePassesPerRun = false;
|
|
|
|
else if (Val == "per-pass-run")
|
|
|
|
Opts.TimePassesPerRun = true;
|
|
|
|
else
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< A->getAsString(Args) << A->getValue();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-01 23:17:29 -07:00
|
|
|
// Basic Block Sections implies Function Sections.
|
|
|
|
Opts.FunctionSections =
|
|
|
|
Args.hasArg(OPT_ffunction_sections) ||
|
|
|
|
(Opts.BBSections != "none" && Opts.BBSections != "labels");
|
|
|
|
|
2015-10-15 20:35:53 +00:00
|
|
|
Opts.PrepareForLTO = Args.hasArg(OPT_flto, OPT_flto_EQ);
|
2018-06-22 20:23:21 +00:00
|
|
|
Opts.PrepareForThinLTO = false;
|
2017-06-14 15:37:11 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_flto_EQ)) {
|
|
|
|
StringRef S = A->getValue();
|
|
|
|
if (S == "thin")
|
2018-06-22 20:23:21 +00:00
|
|
|
Opts.PrepareForThinLTO = true;
|
2017-06-14 15:37:11 +00:00
|
|
|
else if (S != "full")
|
|
|
|
Diags.Report(diag::err_drv_invalid_value) << A->getAsString(Args) << S;
|
|
|
|
}
|
2015-12-07 19:21:34 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_fthinlto_index_EQ)) {
|
2019-08-05 13:59:26 +00:00
|
|
|
if (IK.getLanguage() != Language::LLVM_IR)
|
2015-12-07 19:21:34 +00:00
|
|
|
Diags.Report(diag::err_drv_argument_only_allowed_with)
|
|
|
|
<< A->getAsString(Args) << "-x ir";
|
2020-01-28 20:23:46 +01:00
|
|
|
Opts.ThinLTOIndexFile =
|
|
|
|
std::string(Args.getLastArgValue(OPT_fthinlto_index_EQ));
|
2015-12-07 19:21:34 +00:00
|
|
|
}
|
2018-04-17 16:39:25 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_save_temps_EQ))
|
|
|
|
Opts.SaveTempsFilePrefix =
|
|
|
|
llvm::StringSwitch<std::string>(A->getValue())
|
|
|
|
.Case("obj", FrontendOpts.OutputFile)
|
|
|
|
.Default(llvm::sys::path::filename(FrontendOpts.OutputFile).str());
|
|
|
|
|
2020-09-29 15:53:41 -07:00
|
|
|
// The memory profile runtime appends the pid to make this name more unique.
|
|
|
|
const char *MemProfileBasename = "memprof.profraw";
|
|
|
|
if (Args.hasArg(OPT_fmemory_profile_EQ)) {
|
|
|
|
SmallString<128> Path(
|
|
|
|
std::string(Args.getLastArgValue(OPT_fmemory_profile_EQ)));
|
|
|
|
llvm::sys::path::append(Path, MemProfileBasename);
|
|
|
|
Opts.MemoryProfileOutput = std::string(Path);
|
|
|
|
} else if (Args.hasArg(OPT_fmemory_profile))
|
|
|
|
Opts.MemoryProfileOutput = MemProfileBasename;
|
2020-08-13 16:29:38 -07:00
|
|
|
|
2013-03-07 08:28:53 +00:00
|
|
|
if (Opts.EmitGcovArcs || Opts.EmitGcovNotes) {
|
2020-01-28 20:23:46 +01:00
|
|
|
Opts.CoverageDataFile =
|
|
|
|
std::string(Args.getLastArgValue(OPT_coverage_data_file));
|
|
|
|
Opts.CoverageNotesFile =
|
|
|
|
std::string(Args.getLastArgValue(OPT_coverage_notes_file));
|
2018-11-17 19:41:39 +00:00
|
|
|
Opts.ProfileFilterFiles =
|
2020-01-28 20:23:46 +01:00
|
|
|
std::string(Args.getLastArgValue(OPT_fprofile_filter_files_EQ));
|
2018-11-17 19:41:39 +00:00
|
|
|
Opts.ProfileExcludeFiles =
|
2020-01-28 20:23:46 +01:00
|
|
|
std::string(Args.getLastArgValue(OPT_fprofile_exclude_files_EQ));
|
2013-03-07 08:28:53 +00:00
|
|
|
if (Args.hasArg(OPT_coverage_version_EQ)) {
|
|
|
|
StringRef CoverageVersion = Args.getLastArgValue(OPT_coverage_version_EQ);
|
|
|
|
if (CoverageVersion.size() != 4) {
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< Args.getLastArg(OPT_coverage_version_EQ)->getAsString(Args)
|
|
|
|
<< CoverageVersion;
|
|
|
|
} else {
|
2013-03-14 05:14:01 +00:00
|
|
|
memcpy(Opts.CoverageVersion, CoverageVersion.data(), 4);
|
2013-03-07 08:28:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-05-11 16:26:03 +00:00
|
|
|
// FIXME: For backend options that are not yet recorded as function
|
|
|
|
// attributes in the IR, keep track of them so we can embed them in a
|
|
|
|
// separate data section and use them when building the bitcode.
|
2020-10-28 20:16:51 -07:00
|
|
|
for (const auto &A : Args) {
|
|
|
|
// Do not encode output and input.
|
|
|
|
if (A->getOption().getID() == options::OPT_o ||
|
|
|
|
A->getOption().getID() == options::OPT_INPUT ||
|
|
|
|
A->getOption().getID() == options::OPT_x ||
|
|
|
|
A->getOption().getID() == options::OPT_fembed_bitcode ||
|
|
|
|
A->getOption().matches(options::OPT_W_Group))
|
|
|
|
continue;
|
|
|
|
ArgStringList ASL;
|
|
|
|
A->render(Args, ASL);
|
|
|
|
for (const auto &arg : ASL) {
|
|
|
|
StringRef ArgStr(arg);
|
|
|
|
Opts.CmdArgs.insert(Opts.CmdArgs.end(), ArgStr.begin(), ArgStr.end());
|
|
|
|
// using \00 to separate each commandline options.
|
|
|
|
Opts.CmdArgs.push_back('\0');
|
2016-05-11 16:26:03 +00:00
|
|
|
}
|
|
|
|
}
|
2013-03-07 08:28:53 +00:00
|
|
|
|
2020-09-18 14:45:51 -04:00
|
|
|
Opts.XRayTotalFunctionGroups =
|
|
|
|
getLastArgIntValue(Args, OPT_fxray_function_groups, 1, Diags);
|
|
|
|
Opts.XRaySelectedFunctionGroup =
|
|
|
|
getLastArgIntValue(Args, OPT_fxray_selected_function_group, 0, Diags);
|
2018-04-13 02:31:58 +00:00
|
|
|
|
|
|
|
auto XRayInstrBundles =
|
|
|
|
Args.getAllArgValues(OPT_fxray_instrumentation_bundle);
|
|
|
|
if (XRayInstrBundles.empty())
|
|
|
|
Opts.XRayInstrumentationBundle.Mask = XRayInstrKind::All;
|
|
|
|
else
|
|
|
|
for (const auto &A : XRayInstrBundles)
|
|
|
|
parseXRayInstrumentationBundle("-fxray-instrumentation-bundle=", A, Args,
|
|
|
|
Diags, Opts.XRayInstrumentationBundle);
|
|
|
|
|
2018-01-09 08:53:59 +00:00
|
|
|
if (const Arg *A = Args.getLastArg(OPT_fcf_protection_EQ)) {
|
|
|
|
StringRef Name = A->getValue();
|
|
|
|
if (Name == "full") {
|
|
|
|
Opts.CFProtectionReturn = 1;
|
|
|
|
Opts.CFProtectionBranch = 1;
|
|
|
|
} else if (Name == "return")
|
|
|
|
Opts.CFProtectionReturn = 1;
|
|
|
|
else if (Name == "branch")
|
|
|
|
Opts.CFProtectionBranch = 1;
|
|
|
|
else if (Name != "none") {
|
|
|
|
Diags.Report(diag::err_drv_invalid_value) << A->getAsString(Args) << Name;
|
|
|
|
Success = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-19 23:05:26 -07:00
|
|
|
if (const Arg *A = Args.getLastArg(OPT_compress_debug_sections_EQ)) {
|
|
|
|
auto DCT = llvm::StringSwitch<llvm::DebugCompressionType>(A->getValue())
|
|
|
|
.Case("none", llvm::DebugCompressionType::None)
|
|
|
|
.Case("zlib", llvm::DebugCompressionType::Z)
|
|
|
|
.Case("zlib-gnu", llvm::DebugCompressionType::GNU)
|
|
|
|
.Default(llvm::DebugCompressionType::None);
|
|
|
|
Opts.setCompressDebugSections(DCT);
|
2017-06-23 15:34:16 +00:00
|
|
|
}
|
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
for (auto *A :
|
2018-08-20 18:16:48 +00:00
|
|
|
Args.filtered(OPT_mlink_bitcode_file, OPT_mlink_builtin_bitcode)) {
|
[CodeGen] [CUDA] Add the ability set default attrs on functions in linked modules.
Summary:
Now when you ask clang to link in a bitcode module, you can tell it to
set attributes on that module's functions to match what we would have
set if we'd emitted those functions ourselves.
This is particularly important for fast-math attributes in CUDA
compilations.
Each CUDA compilation links in libdevice, a bitcode library provided by
nvidia as part of the CUDA distribution. Without this patch, if we have
a user-function F that is compiled with -ffast-math that calls a
function G from libdevice, F will have the unsafe-fp-math=true (etc.)
attributes, but G will have no attributes.
Since F calls G, the inliner will merge G's attributes into F's. It
considers the lack of an unsafe-fp-math=true attribute on G to be
tantamount to unsafe-fp-math=false, so it "merges" these by setting
unsafe-fp-math=false on F.
This then continues up the call graph, until every function that
(transitively) calls something in libdevice gets unsafe-fp-math=false
set, thus disabling fastmath in almost all CUDA code.
Reviewers: echristo
Subscribers: hfinkel, llvm-commits, mehdi_amini
Differential Revision: https://reviews.llvm.org/D28538
llvm-svn: 293097
2017-01-25 21:29:48 +00:00
|
|
|
CodeGenOptions::BitcodeFileToLink F;
|
|
|
|
F.Filename = A->getValue();
|
2018-08-20 18:16:48 +00:00
|
|
|
if (A->getOption().matches(OPT_mlink_builtin_bitcode)) {
|
2017-03-13 18:08:11 +00:00
|
|
|
F.LinkFlags = llvm::Linker::Flags::LinkOnlyNeeded;
|
[CodeGen] [CUDA] Add the ability set default attrs on functions in linked modules.
Summary:
Now when you ask clang to link in a bitcode module, you can tell it to
set attributes on that module's functions to match what we would have
set if we'd emitted those functions ourselves.
This is particularly important for fast-math attributes in CUDA
compilations.
Each CUDA compilation links in libdevice, a bitcode library provided by
nvidia as part of the CUDA distribution. Without this patch, if we have
a user-function F that is compiled with -ffast-math that calls a
function G from libdevice, F will have the unsafe-fp-math=true (etc.)
attributes, but G will have no attributes.
Since F calls G, the inliner will merge G's attributes into F's. It
considers the lack of an unsafe-fp-math=true attribute on G to be
tantamount to unsafe-fp-math=false, so it "merges" these by setting
unsafe-fp-math=false on F.
This then continues up the call graph, until every function that
(transitively) calls something in libdevice gets unsafe-fp-math=false
set, thus disabling fastmath in almost all CUDA code.
Reviewers: echristo
Subscribers: hfinkel, llvm-commits, mehdi_amini
Differential Revision: https://reviews.llvm.org/D28538
llvm-svn: 293097
2017-01-25 21:29:48 +00:00
|
|
|
// When linking CUDA bitcode, propagate function attributes so that
|
|
|
|
// e.g. libdevice gets fast-math attrs if we're building with fast-math.
|
|
|
|
F.PropagateAttrs = true;
|
2017-03-13 18:08:11 +00:00
|
|
|
F.Internalize = true;
|
[CodeGen] [CUDA] Add the ability set default attrs on functions in linked modules.
Summary:
Now when you ask clang to link in a bitcode module, you can tell it to
set attributes on that module's functions to match what we would have
set if we'd emitted those functions ourselves.
This is particularly important for fast-math attributes in CUDA
compilations.
Each CUDA compilation links in libdevice, a bitcode library provided by
nvidia as part of the CUDA distribution. Without this patch, if we have
a user-function F that is compiled with -ffast-math that calls a
function G from libdevice, F will have the unsafe-fp-math=true (etc.)
attributes, but G will have no attributes.
Since F calls G, the inliner will merge G's attributes into F's. It
considers the lack of an unsafe-fp-math=true attribute on G to be
tantamount to unsafe-fp-math=false, so it "merges" these by setting
unsafe-fp-math=false on F.
This then continues up the call graph, until every function that
(transitively) calls something in libdevice gets unsafe-fp-math=false
set, thus disabling fastmath in almost all CUDA code.
Reviewers: echristo
Subscribers: hfinkel, llvm-commits, mehdi_amini
Differential Revision: https://reviews.llvm.org/D28538
llvm-svn: 293097
2017-01-25 21:29:48 +00:00
|
|
|
}
|
|
|
|
Opts.LinkBitcodeFiles.push_back(F);
|
2015-10-27 17:56:59 +00:00
|
|
|
}
|
2012-08-21 16:16:06 +00:00
|
|
|
Opts.SSPBufferSize =
|
2013-06-14 17:17:23 +00:00
|
|
|
getLastArgIntValue(Args, OPT_stack_protector_buffer_size, 8, Diags);
|
2020-10-22 09:46:42 +08:00
|
|
|
|
|
|
|
Opts.StackProtectorGuard =
|
|
|
|
std::string(Args.getLastArgValue(OPT_mstack_protector_guard_EQ));
|
|
|
|
|
|
|
|
if (Arg *A = Args.getLastArg(OPT_mstack_protector_guard_offset_EQ)) {
|
|
|
|
StringRef Val = A->getValue();
|
|
|
|
unsigned Offset = Opts.StackProtectorGuardOffset;
|
|
|
|
Val.getAsInteger(10, Offset);
|
|
|
|
Opts.StackProtectorGuardOffset = Offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
Opts.StackProtectorGuardReg =
|
|
|
|
std::string(Args.getLastArgValue(OPT_mstack_protector_guard_reg_EQ,
|
|
|
|
"none"));
|
|
|
|
|
2018-12-08 05:13:50 +00:00
|
|
|
|
2018-03-01 22:26:19 +00:00
|
|
|
if (Args.getLastArg(OPT_femulated_tls) ||
|
|
|
|
Args.getLastArg(OPT_fno_emulated_tls)) {
|
|
|
|
Opts.ExplicitEmulatedTLS = true;
|
|
|
|
Opts.EmulatedTLS =
|
|
|
|
Args.hasFlag(OPT_femulated_tls, OPT_fno_emulated_tls, false);
|
|
|
|
}
|
2015-07-28 16:27:56 +00:00
|
|
|
|
2016-08-30 08:09:45 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_fdenormal_fp_math_EQ)) {
|
|
|
|
StringRef Val = A->getValue();
|
2019-10-29 16:16:05 -07:00
|
|
|
Opts.FPDenormalMode = llvm::parseDenormalFPAttribute(Val);
|
2019-11-06 17:10:52 -08:00
|
|
|
if (!Opts.FPDenormalMode.isValid())
|
2016-08-30 08:09:45 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_value) << A->getAsString(Args) << Val;
|
|
|
|
}
|
|
|
|
|
Consolidate internal denormal flushing controls
Currently there are 4 different mechanisms for controlling denormal
flushing behavior, and about as many equivalent frontend controls.
- AMDGPU uses the fp32-denormals and fp64-f16-denormals subtarget features
- NVPTX uses the nvptx-f32ftz attribute
- ARM directly uses the denormal-fp-math attribute
- Other targets indirectly use denormal-fp-math in one DAGCombine
- cl-denorms-are-zero has a corresponding denorms-are-zero attribute
AMDGPU wants a distinct control for f32 flushing from f16/f64, and as
far as I can tell the same is true for NVPTX (based on the attribute
name).
Work on consolidating these into the denormal-fp-math attribute, and a
new type specific denormal-fp-math-f32 variant. Only ARM seems to
support the two different flush modes, so this is overkill for the
other use cases. Ideally we would error on the unsupported
positive-zero mode on other targets from somewhere.
Move the logic for selecting the flush mode into the compiler driver,
instead of handling it in cc1. denormal-fp-math/denormal-fp-math-f32
are now both cc1 flags, but denormal-fp-math-f32 is not yet exposed as
a user flag.
-cl-denorms-are-zero, -fcuda-flush-denormals-to-zero and
-fno-cuda-flush-denormals-to-zero will be mapped to
-fp-denormal-math-f32=ieee or preserve-sign rather than the old
attributes.
Stop emitting the denorms-are-zero attribute for the OpenCL flag. It
has no in-tree users. The meaning would also be target dependent, such
as the AMDGPU choice to treat this as only meaning allow flushing of
f32 and not f16 or f64. The naming is also potentially confusing,
since DAZ in other contexts refers to instructions implicitly treating
input denormals as zero, not necessarily flushing output denormals to
zero.
This also does not attempt to change the behavior for the current
attribute. The LangRef now states that the default is ieee behavior,
but this is inaccurate for the current implementation. The clang
handling is slightly hacky to avoid touching the existing
denormal-fp-math uses. Fixing this will be left for a future patch.
AMDGPU is still using the subtarget feature to control the denormal
mode, but the new attribute are now emitted. A future change will
switch this and remove the subtarget features.
2019-11-01 17:57:29 -07:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_fdenormal_fp_math_f32_EQ)) {
|
|
|
|
StringRef Val = A->getValue();
|
|
|
|
Opts.FP32DenormalMode = llvm::parseDenormalFPAttribute(Val);
|
2019-11-06 17:10:52 -08:00
|
|
|
if (!Opts.FP32DenormalMode.isValid())
|
Consolidate internal denormal flushing controls
Currently there are 4 different mechanisms for controlling denormal
flushing behavior, and about as many equivalent frontend controls.
- AMDGPU uses the fp32-denormals and fp64-f16-denormals subtarget features
- NVPTX uses the nvptx-f32ftz attribute
- ARM directly uses the denormal-fp-math attribute
- Other targets indirectly use denormal-fp-math in one DAGCombine
- cl-denorms-are-zero has a corresponding denorms-are-zero attribute
AMDGPU wants a distinct control for f32 flushing from f16/f64, and as
far as I can tell the same is true for NVPTX (based on the attribute
name).
Work on consolidating these into the denormal-fp-math attribute, and a
new type specific denormal-fp-math-f32 variant. Only ARM seems to
support the two different flush modes, so this is overkill for the
other use cases. Ideally we would error on the unsupported
positive-zero mode on other targets from somewhere.
Move the logic for selecting the flush mode into the compiler driver,
instead of handling it in cc1. denormal-fp-math/denormal-fp-math-f32
are now both cc1 flags, but denormal-fp-math-f32 is not yet exposed as
a user flag.
-cl-denorms-are-zero, -fcuda-flush-denormals-to-zero and
-fno-cuda-flush-denormals-to-zero will be mapped to
-fp-denormal-math-f32=ieee or preserve-sign rather than the old
attributes.
Stop emitting the denorms-are-zero attribute for the OpenCL flag. It
has no in-tree users. The meaning would also be target dependent, such
as the AMDGPU choice to treat this as only meaning allow flushing of
f32 and not f16 or f64. The naming is also potentially confusing,
since DAZ in other contexts refers to instructions implicitly treating
input denormals as zero, not necessarily flushing output denormals to
zero.
This also does not attempt to change the behavior for the current
attribute. The LangRef now states that the default is ieee behavior,
but this is inaccurate for the current implementation. The clang
handling is slightly hacky to avoid touching the existing
denormal-fp-math uses. Fixing this will be left for a future patch.
AMDGPU is still using the subtarget feature to control the denormal
mode, but the new attribute are now emitted. A future change will
switch this and remove the subtarget features.
2019-11-01 17:57:29 -07:00
|
|
|
Diags.Report(diag::err_drv_invalid_value) << A->getAsString(Args) << Val;
|
|
|
|
}
|
|
|
|
|
2020-04-21 15:36:08 -05:00
|
|
|
// X86_32 has -fppc-struct-return and -freg-struct-return.
|
|
|
|
// PPC32 has -maix-struct-return and -msvr4-struct-return.
|
|
|
|
if (Arg *A =
|
|
|
|
Args.getLastArg(OPT_fpcc_struct_return, OPT_freg_struct_return,
|
|
|
|
OPT_maix_struct_return, OPT_msvr4_struct_return)) {
|
2020-05-19 14:56:56 +00:00
|
|
|
// TODO: We might want to consider enabling these options on AIX in the
|
|
|
|
// future.
|
|
|
|
if (T.isOSAIX())
|
|
|
|
Diags.Report(diag::err_drv_unsupported_opt_for_target)
|
|
|
|
<< A->getSpelling() << T.str();
|
|
|
|
|
2020-04-21 15:36:08 -05:00
|
|
|
const Option &O = A->getOption();
|
|
|
|
if (O.matches(OPT_fpcc_struct_return) ||
|
|
|
|
O.matches(OPT_maix_struct_return)) {
|
2013-06-18 02:46:29 +00:00
|
|
|
Opts.setStructReturnConvention(CodeGenOptions::SRCK_OnStack);
|
|
|
|
} else {
|
2020-04-21 15:36:08 -05:00
|
|
|
assert(O.matches(OPT_freg_struct_return) ||
|
|
|
|
O.matches(OPT_msvr4_struct_return));
|
2013-06-18 02:46:29 +00:00
|
|
|
Opts.setStructReturnConvention(CodeGenOptions::SRCK_InRegs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-08 09:34:58 -04:00
|
|
|
if (T.isOSAIX() && (Args.hasArg(OPT_mignore_xcoff_visibility) ||
|
|
|
|
!Args.hasArg(OPT_fvisibility)))
|
|
|
|
Opts.IgnoreXCOFFVisibility = 1;
|
|
|
|
|
2020-11-24 18:11:46 -05:00
|
|
|
if (Arg *A =
|
|
|
|
Args.getLastArg(OPT_mabi_EQ_vec_default, OPT_mabi_EQ_vec_extabi)) {
|
|
|
|
if (!T.isOSAIX())
|
|
|
|
Diags.Report(diag::err_drv_unsupported_opt_for_target)
|
|
|
|
<< A->getSpelling() << T.str();
|
|
|
|
|
|
|
|
const Option &O = A->getOption();
|
|
|
|
if (O.matches(OPT_mabi_EQ_vec_default))
|
|
|
|
Diags.Report(diag::err_aix_default_altivec_abi)
|
|
|
|
<< A->getSpelling() << T.str();
|
|
|
|
else {
|
|
|
|
assert(O.matches(OPT_mabi_EQ_vec_extabi));
|
|
|
|
Opts.EnableAIXExtendedAltivecABI = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-24 17:02:17 +00:00
|
|
|
bool NeedLocTracking = false;
|
2013-08-08 00:17:41 +00:00
|
|
|
|
2016-10-11 00:26:09 +00:00
|
|
|
if (!Opts.OptRecordFile.empty())
|
|
|
|
NeedLocTracking = true;
|
|
|
|
|
2019-03-12 21:22:27 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_opt_record_passes)) {
|
|
|
|
Opts.OptRecordPasses = A->getValue();
|
|
|
|
NeedLocTracking = true;
|
|
|
|
}
|
|
|
|
|
2019-06-17 16:06:00 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_opt_record_format)) {
|
|
|
|
Opts.OptRecordFormat = A->getValue();
|
|
|
|
NeedLocTracking = true;
|
|
|
|
}
|
|
|
|
|
2014-06-24 17:02:17 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_Rpass_EQ)) {
|
2014-05-29 19:55:06 +00:00
|
|
|
Opts.OptimizationRemarkPattern =
|
|
|
|
GenerateOptimizationRemarkRegex(Diags, Args, A);
|
2014-06-24 17:02:17 +00:00
|
|
|
NeedLocTracking = true;
|
|
|
|
}
|
2014-05-29 19:55:06 +00:00
|
|
|
|
2014-06-24 17:02:17 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_Rpass_missed_EQ)) {
|
2014-05-29 19:55:06 +00:00
|
|
|
Opts.OptimizationRemarkMissedPattern =
|
|
|
|
GenerateOptimizationRemarkRegex(Diags, Args, A);
|
2014-06-24 17:02:17 +00:00
|
|
|
NeedLocTracking = true;
|
|
|
|
}
|
2014-05-29 19:55:06 +00:00
|
|
|
|
2014-06-24 17:02:17 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_Rpass_analysis_EQ)) {
|
2014-05-29 19:55:06 +00:00
|
|
|
Opts.OptimizationRemarkAnalysisPattern =
|
|
|
|
GenerateOptimizationRemarkRegex(Diags, Args, A);
|
2014-06-24 17:02:17 +00:00
|
|
|
NeedLocTracking = true;
|
|
|
|
}
|
|
|
|
|
2017-06-23 02:38:45 +00:00
|
|
|
bool UsingSampleProfile = !Opts.SampleProfileFile.empty();
|
2017-07-01 05:45:26 +00:00
|
|
|
bool UsingProfile = UsingSampleProfile ||
|
|
|
|
(Opts.getProfileUse() != CodeGenOptions::ProfileNone);
|
|
|
|
|
2018-05-05 14:37:29 +00:00
|
|
|
if (Opts.DiagnosticsWithHotness && !UsingProfile &&
|
|
|
|
// An IR file will contain PGO as metadata
|
2019-08-05 13:59:26 +00:00
|
|
|
IK.getLanguage() != Language::LLVM_IR)
|
2017-07-01 05:45:26 +00:00
|
|
|
Diags.Report(diag::warn_drv_diagnostics_hotness_requires_pgo)
|
|
|
|
<< "-fdiagnostics-show-hotness";
|
|
|
|
|
2020-11-17 10:43:02 -08:00
|
|
|
// Parse remarks hotness threshold. Valid value is either integer or 'auto'.
|
|
|
|
if (auto *arg =
|
|
|
|
Args.getLastArg(options::OPT_fdiagnostics_hotness_threshold_EQ)) {
|
|
|
|
auto ResultOrErr =
|
|
|
|
llvm::remarks::parseHotnessThresholdOption(arg->getValue());
|
|
|
|
|
|
|
|
if (!ResultOrErr) {
|
|
|
|
Diags.Report(diag::err_drv_invalid_diagnotics_hotness_threshold)
|
|
|
|
<< "-fdiagnostics-hotness-threshold=";
|
|
|
|
} else {
|
|
|
|
Opts.DiagnosticsHotnessThreshold = *ResultOrErr;
|
|
|
|
if ((!Opts.DiagnosticsHotnessThreshold.hasValue() ||
|
|
|
|
Opts.DiagnosticsHotnessThreshold.getValue() > 0) &&
|
|
|
|
!UsingProfile)
|
|
|
|
Diags.Report(diag::warn_drv_diagnostics_hotness_requires_pgo)
|
|
|
|
<< "-fdiagnostics-hotness-threshold=";
|
|
|
|
}
|
|
|
|
}
|
2016-09-13 04:32:40 +00:00
|
|
|
|
2014-10-22 13:00:05 +00:00
|
|
|
// If the user requested to use a sample profile for PGO, then the
|
|
|
|
// backend will need to track source location information so the profile
|
|
|
|
// can be incorporated into the IR.
|
2017-06-23 02:38:45 +00:00
|
|
|
if (UsingSampleProfile)
|
2014-10-22 13:00:05 +00:00
|
|
|
NeedLocTracking = true;
|
|
|
|
|
|
|
|
// If the user requested a flag that requires source locations available in
|
|
|
|
// the backend, make sure that the backend tracks source location information.
|
2016-02-02 11:06:51 +00:00
|
|
|
if (NeedLocTracking && Opts.getDebugInfo() == codegenoptions::NoDebugInfo)
|
|
|
|
Opts.setDebugInfo(codegenoptions::LocTrackingOnly);
|
2014-05-29 19:55:06 +00:00
|
|
|
|
Reimplement -fsanitize-recover family of flags.
Introduce the following -fsanitize-recover flags:
- -fsanitize-recover=<list>: Enable recovery for selected checks or
group of checks. It is forbidden to explicitly list unrecoverable
sanitizers here (that is, "address", "unreachable", "return").
- -fno-sanitize-recover=<list>: Disable recovery for selected checks or
group of checks.
- -f(no-)?sanitize-recover is now a synonym for
-f(no-)?sanitize-recover=undefined,integer and will soon be deprecated.
These flags are parsed left to right, and mask of "recoverable"
sanitizer is updated accordingly, much like what we do for -fsanitize= flags.
-fsanitize= and -fsanitize-recover= flag families are independent.
CodeGen change: If there is a single UBSan handler function, responsible
for implementing multiple checks, which have different recoverable setting,
then we emit two handler calls instead of one:
the first one for the set of "unrecoverable" checks, another one - for
set of "recoverable" checks. If all checks implemented by a handler have the
same recoverability setting, then the generated code will be the same.
llvm-svn: 225719
2015-01-12 22:39:12 +00:00
|
|
|
// Parse -fsanitize-recover= arguments.
|
|
|
|
// FIXME: Report unrecoverable sanitizers incorrectly specified here.
|
|
|
|
parseSanitizerKinds("-fsanitize-recover=",
|
|
|
|
Args.getAllArgValues(OPT_fsanitize_recover_EQ), Diags,
|
|
|
|
Opts.SanitizeRecover);
|
2015-06-18 23:59:22 +00:00
|
|
|
parseSanitizerKinds("-fsanitize-trap=",
|
|
|
|
Args.getAllArgValues(OPT_fsanitize_trap_EQ), Diags,
|
|
|
|
Opts.SanitizeTrap);
|
Reimplement -fsanitize-recover family of flags.
Introduce the following -fsanitize-recover flags:
- -fsanitize-recover=<list>: Enable recovery for selected checks or
group of checks. It is forbidden to explicitly list unrecoverable
sanitizers here (that is, "address", "unreachable", "return").
- -fno-sanitize-recover=<list>: Disable recovery for selected checks or
group of checks.
- -f(no-)?sanitize-recover is now a synonym for
-f(no-)?sanitize-recover=undefined,integer and will soon be deprecated.
These flags are parsed left to right, and mask of "recoverable"
sanitizer is updated accordingly, much like what we do for -fsanitize= flags.
-fsanitize= and -fsanitize-recover= flag families are independent.
CodeGen change: If there is a single UBSan handler function, responsible
for implementing multiple checks, which have different recoverable setting,
then we emit two handler calls instead of one:
the first one for the set of "unrecoverable" checks, another one - for
set of "recoverable" checks. If all checks implemented by a handler have the
same recoverability setting, then the generated code will be the same.
llvm-svn: 225719
2015-01-12 22:39:12 +00:00
|
|
|
|
2018-04-23 10:08:46 +00:00
|
|
|
Opts.EmitVersionIdentMetadata = Args.hasFlag(OPT_Qy, OPT_Qn, true);
|
|
|
|
|
2011-12-23 03:05:38 +00:00
|
|
|
return Success;
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ParseDependencyOutputArgs(DependencyOutputOptions &Opts,
|
|
|
|
ArgList &Args) {
|
2018-05-04 15:58:31 +00:00
|
|
|
if (Args.hasArg(OPT_show_includes)) {
|
|
|
|
// Writing both /showIncludes and preprocessor output to stdout
|
|
|
|
// would produce interleaved output, so use stderr for /showIncludes.
|
|
|
|
// This behaves the same as cl.exe, when /E, /EP or /P are passed.
|
|
|
|
if (Args.hasArg(options::OPT_E) || Args.hasArg(options::OPT_P))
|
|
|
|
Opts.ShowIncludesDest = ShowIncludesDestination::Stderr;
|
|
|
|
else
|
|
|
|
Opts.ShowIncludesDest = ShowIncludesDestination::Stdout;
|
|
|
|
} else {
|
|
|
|
Opts.ShowIncludesDest = ShowIncludesDestination::None;
|
|
|
|
}
|
2015-08-13 04:04:37 +00:00
|
|
|
// Add sanitizer blacklists as extra dependencies.
|
|
|
|
// They won't be discovered by the regular preprocessor, so
|
|
|
|
// we let make / ninja to know about this implicit dependency.
|
2019-11-08 14:16:15 -08:00
|
|
|
if (!Args.hasArg(OPT_fno_sanitize_blacklist)) {
|
|
|
|
for (const auto *A : Args.filtered(OPT_fsanitize_blacklist)) {
|
|
|
|
StringRef Val = A->getValue();
|
|
|
|
if (Val.find('=') == StringRef::npos)
|
2020-01-28 20:23:46 +01:00
|
|
|
Opts.ExtraDeps.push_back(std::string(Val));
|
2019-11-08 14:16:15 -08:00
|
|
|
}
|
|
|
|
if (Opts.IncludeSystemHeaders) {
|
|
|
|
for (const auto *A : Args.filtered(OPT_fsanitize_system_blacklist)) {
|
|
|
|
StringRef Val = A->getValue();
|
|
|
|
if (Val.find('=') == StringRef::npos)
|
2020-01-28 20:23:46 +01:00
|
|
|
Opts.ExtraDeps.push_back(std::string(Val));
|
2019-11-08 14:16:15 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Propagate the extra dependencies.
|
|
|
|
for (const auto *A : Args.filtered(OPT_fdepfile_entry)) {
|
|
|
|
Opts.ExtraDeps.push_back(A->getValue());
|
|
|
|
}
|
|
|
|
|
2017-08-31 06:26:43 +00:00
|
|
|
// Only the -fmodule-file=<file> form.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_fmodule_file)) {
|
2017-08-31 06:26:43 +00:00
|
|
|
StringRef Val = A->getValue();
|
|
|
|
if (Val.find('=') == StringRef::npos)
|
2020-01-28 20:23:46 +01:00
|
|
|
Opts.ExtraDeps.push_back(std::string(Val));
|
2017-08-31 06:26:43 +00:00
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
2016-05-27 20:43:00 +00:00
|
|
|
static bool parseShowColorsArgs(const ArgList &Args, bool DefaultColor) {
|
|
|
|
// Color diagnostics default to auto ("on" if terminal supports) in the driver
|
|
|
|
// but default to off in cc1, needing an explicit OPT_fdiagnostics_color.
|
|
|
|
// Support both clang's -f[no-]color-diagnostics and gcc's
|
|
|
|
// -f[no-]diagnostics-colors[=never|always|auto].
|
|
|
|
enum {
|
|
|
|
Colors_On,
|
|
|
|
Colors_Off,
|
|
|
|
Colors_Auto
|
|
|
|
} ShowColors = DefaultColor ? Colors_Auto : Colors_Off;
|
2018-03-26 21:45:04 +00:00
|
|
|
for (auto *A : Args) {
|
2016-05-27 20:43:00 +00:00
|
|
|
const Option &O = A->getOption();
|
|
|
|
if (O.matches(options::OPT_fcolor_diagnostics) ||
|
|
|
|
O.matches(options::OPT_fdiagnostics_color)) {
|
|
|
|
ShowColors = Colors_On;
|
|
|
|
} else if (O.matches(options::OPT_fno_color_diagnostics) ||
|
|
|
|
O.matches(options::OPT_fno_diagnostics_color)) {
|
|
|
|
ShowColors = Colors_Off;
|
2016-12-10 14:55:14 +00:00
|
|
|
} else if (O.matches(options::OPT_fdiagnostics_color_EQ)) {
|
2016-05-27 20:43:00 +00:00
|
|
|
StringRef Value(A->getValue());
|
|
|
|
if (Value == "always")
|
|
|
|
ShowColors = Colors_On;
|
|
|
|
else if (Value == "never")
|
|
|
|
ShowColors = Colors_Off;
|
|
|
|
else if (Value == "auto")
|
|
|
|
ShowColors = Colors_Auto;
|
|
|
|
}
|
|
|
|
}
|
2016-12-10 14:55:14 +00:00
|
|
|
return ShowColors == Colors_On ||
|
|
|
|
(ShowColors == Colors_Auto &&
|
|
|
|
llvm::sys::Process::StandardErrHasColors());
|
2016-05-27 20:43:00 +00:00
|
|
|
}
|
|
|
|
|
2017-12-16 02:23:22 +00:00
|
|
|
static bool checkVerifyPrefixes(const std::vector<std::string> &VerifyPrefixes,
|
|
|
|
DiagnosticsEngine *Diags) {
|
|
|
|
bool Success = true;
|
|
|
|
for (const auto &Prefix : VerifyPrefixes) {
|
|
|
|
// Every prefix must start with a letter and contain only alphanumeric
|
|
|
|
// characters, hyphens, and underscores.
|
2019-03-31 08:48:19 +00:00
|
|
|
auto BadChar = llvm::find_if(Prefix, [](char C) {
|
|
|
|
return !isAlphanumeric(C) && C != '-' && C != '_';
|
|
|
|
});
|
2017-12-16 02:23:22 +00:00
|
|
|
if (BadChar != Prefix.end() || !isLetter(Prefix[0])) {
|
|
|
|
Success = false;
|
|
|
|
if (Diags) {
|
|
|
|
Diags->Report(diag::err_drv_invalid_value) << "-verify=" << Prefix;
|
|
|
|
Diags->Report(diag::note_drv_verify_prefix_spelling);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
2012-03-13 20:09:56 +00:00
|
|
|
bool clang::ParseDiagnosticArgs(DiagnosticOptions &Opts, ArgList &Args,
|
2016-05-27 20:43:00 +00:00
|
|
|
DiagnosticsEngine *Diags,
|
2020-03-31 21:30:14 -07:00
|
|
|
bool DefaultDiagColor) {
|
2011-12-23 03:05:38 +00:00
|
|
|
bool Success = true;
|
|
|
|
|
2020-01-28 20:23:46 +01:00
|
|
|
Opts.DiagnosticLogFile =
|
|
|
|
std::string(Args.getLastArgValue(OPT_diagnostic_log_file));
|
2014-10-23 22:20:11 +00:00
|
|
|
if (Arg *A =
|
|
|
|
Args.getLastArg(OPT_diagnostic_serialized_file, OPT__serialize_diags))
|
|
|
|
Opts.DiagnosticSerializationFile = A->getValue();
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.IgnoreWarnings = Args.hasArg(OPT_w);
|
|
|
|
Opts.NoRewriteMacros = Args.hasArg(OPT_Wno_rewrite_macros);
|
|
|
|
Opts.Pedantic = Args.hasArg(OPT_pedantic);
|
|
|
|
Opts.PedanticErrors = Args.hasArg(OPT_pedantic_errors);
|
|
|
|
Opts.ShowCarets = !Args.hasArg(OPT_fno_caret_diagnostics);
|
2016-05-27 20:43:00 +00:00
|
|
|
Opts.ShowColors = parseShowColorsArgs(Args, DefaultDiagColor);
|
2020-01-23 21:29:09 -08:00
|
|
|
Opts.ShowColumn = !Args.hasArg(OPT_fno_show_column);
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.ShowFixits = !Args.hasArg(OPT_fno_diagnostics_fixit_info);
|
|
|
|
Opts.ShowLocation = !Args.hasArg(OPT_fno_show_source_location);
|
2016-08-26 15:45:36 +00:00
|
|
|
Opts.AbsolutePath = Args.hasArg(OPT_fdiagnostics_absolute_paths);
|
2020-03-31 21:30:14 -07:00
|
|
|
Opts.ShowOptionNames = !Args.hasArg(OPT_fno_diagnostics_show_option);
|
2010-06-11 05:57:47 +00:00
|
|
|
|
2011-03-27 20:00:08 +00:00
|
|
|
// Default behavior is to not to show note include stacks.
|
|
|
|
Opts.ShowNoteIncludeStack = false;
|
2011-03-27 01:50:55 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_fdiagnostics_show_note_include_stack,
|
|
|
|
OPT_fno_diagnostics_show_note_include_stack))
|
2011-03-27 20:00:08 +00:00
|
|
|
if (A->getOption().matches(OPT_fdiagnostics_show_note_include_stack))
|
|
|
|
Opts.ShowNoteIncludeStack = true;
|
2011-03-27 01:50:55 +00:00
|
|
|
|
2011-07-23 10:55:15 +00:00
|
|
|
StringRef ShowOverloads =
|
2010-06-11 05:57:47 +00:00
|
|
|
Args.getLastArgValue(OPT_fshow_overloads_EQ, "all");
|
|
|
|
if (ShowOverloads == "best")
|
2012-10-23 23:11:23 +00:00
|
|
|
Opts.setShowOverloads(Ovl_Best);
|
2010-06-11 05:57:47 +00:00
|
|
|
else if (ShowOverloads == "all")
|
2012-10-23 23:11:23 +00:00
|
|
|
Opts.setShowOverloads(Ovl_All);
|
2011-12-23 03:05:38 +00:00
|
|
|
else {
|
2012-03-13 20:09:56 +00:00
|
|
|
Success = false;
|
|
|
|
if (Diags)
|
|
|
|
Diags->Report(diag::err_drv_invalid_value)
|
2010-06-11 05:57:47 +00:00
|
|
|
<< Args.getLastArg(OPT_fshow_overloads_EQ)->getAsString(Args)
|
|
|
|
<< ShowOverloads;
|
2011-12-23 03:05:38 +00:00
|
|
|
}
|
2010-06-11 05:57:47 +00:00
|
|
|
|
2011-07-23 10:55:15 +00:00
|
|
|
StringRef ShowCategory =
|
2010-05-20 16:54:55 +00:00
|
|
|
Args.getLastArgValue(OPT_fdiagnostics_show_category, "none");
|
2010-05-04 21:55:25 +00:00
|
|
|
if (ShowCategory == "none")
|
|
|
|
Opts.ShowCategories = 0;
|
|
|
|
else if (ShowCategory == "id")
|
|
|
|
Opts.ShowCategories = 1;
|
|
|
|
else if (ShowCategory == "name")
|
|
|
|
Opts.ShowCategories = 2;
|
2011-12-23 03:05:38 +00:00
|
|
|
else {
|
2012-03-13 20:09:56 +00:00
|
|
|
Success = false;
|
|
|
|
if (Diags)
|
|
|
|
Diags->Report(diag::err_drv_invalid_value)
|
2010-05-04 21:55:25 +00:00
|
|
|
<< Args.getLastArg(OPT_fdiagnostics_show_category)->getAsString(Args)
|
|
|
|
<< ShowCategory;
|
2011-12-23 03:05:38 +00:00
|
|
|
}
|
2010-10-21 03:16:25 +00:00
|
|
|
|
2011-07-23 10:55:15 +00:00
|
|
|
StringRef Format =
|
2011-05-21 17:07:29 +00:00
|
|
|
Args.getLastArgValue(OPT_fdiagnostics_format, "clang");
|
|
|
|
if (Format == "clang")
|
2012-10-23 23:11:23 +00:00
|
|
|
Opts.setFormat(DiagnosticOptions::Clang);
|
2011-12-23 03:05:38 +00:00
|
|
|
else if (Format == "msvc")
|
2015-02-02 19:30:52 +00:00
|
|
|
Opts.setFormat(DiagnosticOptions::MSVC);
|
2013-09-24 00:08:55 +00:00
|
|
|
else if (Format == "msvc-fallback") {
|
2015-02-02 19:30:52 +00:00
|
|
|
Opts.setFormat(DiagnosticOptions::MSVC);
|
2013-09-24 00:08:55 +00:00
|
|
|
Opts.CLFallbackMode = true;
|
|
|
|
} else if (Format == "vi")
|
2012-10-23 23:11:23 +00:00
|
|
|
Opts.setFormat(DiagnosticOptions::Vi);
|
2011-12-23 03:05:38 +00:00
|
|
|
else {
|
2012-03-13 20:09:56 +00:00
|
|
|
Success = false;
|
|
|
|
if (Diags)
|
|
|
|
Diags->Report(diag::err_drv_invalid_value)
|
2011-05-21 17:07:29 +00:00
|
|
|
<< Args.getLastArg(OPT_fdiagnostics_format)->getAsString(Args)
|
|
|
|
<< Format;
|
2011-12-23 03:05:38 +00:00
|
|
|
}
|
2015-06-13 07:11:40 +00:00
|
|
|
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.ShowSourceRanges = Args.hasArg(OPT_fdiagnostics_print_source_range_info);
|
2010-08-19 20:24:43 +00:00
|
|
|
Opts.ShowParseableFixits = Args.hasArg(OPT_fdiagnostics_parseable_fixits);
|
2012-11-14 23:55:25 +00:00
|
|
|
Opts.ShowPresumedLoc = !Args.hasArg(OPT_fno_diagnostics_use_presumed_location);
|
2017-12-16 02:23:22 +00:00
|
|
|
Opts.VerifyDiagnostics = Args.hasArg(OPT_verify) || Args.hasArg(OPT_verify_EQ);
|
|
|
|
if (Args.hasArg(OPT_verify))
|
|
|
|
Opts.VerifyPrefixes.push_back("expected");
|
|
|
|
// Keep VerifyPrefixes in its original order for the sake of diagnostics, and
|
|
|
|
// then sort it to prepare for fast lookup using std::binary_search.
|
|
|
|
if (!checkVerifyPrefixes(Opts.VerifyPrefixes, Diags)) {
|
|
|
|
Opts.VerifyDiagnostics = false;
|
|
|
|
Success = false;
|
|
|
|
}
|
|
|
|
else
|
2018-09-30 21:41:11 +00:00
|
|
|
llvm::sort(Opts.VerifyPrefixes);
|
2015-06-13 07:11:40 +00:00
|
|
|
DiagnosticLevelMask DiagMask = DiagnosticLevelMask::None;
|
|
|
|
Success &= parseDiagnosticLevelMask("-verify-ignore-unexpected=",
|
|
|
|
Args.getAllArgValues(OPT_verify_ignore_unexpected_EQ),
|
|
|
|
Diags, DiagMask);
|
|
|
|
if (Args.hasArg(OPT_verify_ignore_unexpected))
|
|
|
|
DiagMask = DiagnosticLevelMask::All;
|
|
|
|
Opts.setVerifyIgnoreUnexpected(DiagMask);
|
2012-06-26 18:18:47 +00:00
|
|
|
Opts.ElideType = !Args.hasArg(OPT_fno_elide_type);
|
|
|
|
Opts.ShowTemplateTree = Args.hasArg(OPT_fdiagnostics_show_template_tree);
|
2013-06-14 17:17:23 +00:00
|
|
|
Opts.ErrorLimit = getLastArgIntValue(Args, OPT_ferror_limit, 0, Diags);
|
|
|
|
Opts.MacroBacktraceLimit =
|
|
|
|
getLastArgIntValue(Args, OPT_fmacro_backtrace_limit,
|
2010-05-04 17:13:42 +00:00
|
|
|
DiagnosticOptions::DefaultMacroBacktraceLimit, Diags);
|
2013-06-14 17:17:23 +00:00
|
|
|
Opts.TemplateBacktraceLimit = getLastArgIntValue(
|
|
|
|
Args, OPT_ftemplate_backtrace_limit,
|
|
|
|
DiagnosticOptions::DefaultTemplateBacktraceLimit, Diags);
|
|
|
|
Opts.ConstexprBacktraceLimit = getLastArgIntValue(
|
|
|
|
Args, OPT_fconstexpr_backtrace_limit,
|
|
|
|
DiagnosticOptions::DefaultConstexprBacktraceLimit, Diags);
|
2014-12-16 21:39:02 +00:00
|
|
|
Opts.SpellCheckingLimit = getLastArgIntValue(
|
|
|
|
Args, OPT_fspell_checking_limit,
|
|
|
|
DiagnosticOptions::DefaultSpellCheckingLimit, Diags);
|
2017-05-22 23:51:40 +00:00
|
|
|
Opts.SnippetLineLimit = getLastArgIntValue(
|
|
|
|
Args, OPT_fcaret_diagnostics_max_lines,
|
|
|
|
DiagnosticOptions::DefaultSnippetLineLimit, Diags);
|
2013-06-14 17:17:23 +00:00
|
|
|
Opts.TabStop = getLastArgIntValue(Args, OPT_ftabstop,
|
2010-01-13 03:06:50 +00:00
|
|
|
DiagnosticOptions::DefaultTabStop, Diags);
|
|
|
|
if (Opts.TabStop == 0 || Opts.TabStop > DiagnosticOptions::MaxTabStop) {
|
|
|
|
Opts.TabStop = DiagnosticOptions::DefaultTabStop;
|
2012-03-13 20:09:56 +00:00
|
|
|
if (Diags)
|
|
|
|
Diags->Report(diag::warn_ignoring_ftabstop_value)
|
|
|
|
<< Opts.TabStop << DiagnosticOptions::DefaultTabStop;
|
2010-01-13 03:06:50 +00:00
|
|
|
}
|
2020-03-31 16:46:51 -07:00
|
|
|
Opts.MessageLength =
|
|
|
|
getLastArgIntValue(Args, OPT_fmessage_length_EQ, 0, Diags);
|
[clang][diagnostics] Add '-Wundef-prefix' warning option
Summary:
Add an `-Wundef-prefix=<arg1>,<arg2>...` option, which is similar to `-Wundef`, but only give warnings for undefined macros with the given prefixes.
Reviewers: ributzka, steven_wu, cishida, bruno, arphaman, rsmith
Reviewed By: ributzka, arphaman
Subscribers: riccibruno, dexonsmith, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D80751
This patch was authored by Zixu Wang <zixu_wang@apple.com>
2020-06-30 09:28:06 -07:00
|
|
|
|
Use -Rblah, not -Wblah, to control remark diagnostics. This was always the
intent when we added remark support, but was never implemented in the general
case, because the first -R flags didn't need it. (-Rpass= had special handling
to accomodate its argument.)
-Rno-foo, -Reverything, and -Rno-everything can be used to turn off a remark,
or to turn on or off all remarks. Per discussion on cfe-commits, -Weverything
does not affect remarks, and -Reverything does not affect warnings or errors.
The only "real" -R flag we have right now is -Rmodule-build; that flag is
effectively renamed from -Wmodule-build to -Rmodule-build by this change.
-Wpass and -Wno-pass (and their friends) are also renamed to -Rpass and
-Rno-pass by this change; it's not completely clear whether we intended to have
a -Rpass (with no =pattern), but that is unchanged by this commit, other than
the flag name. The default pattern is effectively one which matches no passes.
In future, we may want to make the default pattern be .*, so that -Reverything
works for -Rpass properly.
llvm-svn: 215046
2014-08-07 00:24:21 +00:00
|
|
|
addDiagnosticArgs(Args, OPT_W_Group, OPT_W_value_Group, Opts.Warnings);
|
|
|
|
addDiagnosticArgs(Args, OPT_R_Group, OPT_R_value_Group, Opts.Remarks);
|
2011-12-23 03:05:38 +00:00
|
|
|
|
|
|
|
return Success;
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
2015-11-03 18:33:07 +00:00
|
|
|
/// Parse the argument to the -ftest-module-file-extension
|
|
|
|
/// command-line argument.
|
|
|
|
///
|
|
|
|
/// \returns true on error, false on success.
|
|
|
|
static bool parseTestModuleFileExtensionArg(StringRef Arg,
|
|
|
|
std::string &BlockName,
|
|
|
|
unsigned &MajorVersion,
|
|
|
|
unsigned &MinorVersion,
|
|
|
|
bool &Hashed,
|
|
|
|
std::string &UserInfo) {
|
|
|
|
SmallVector<StringRef, 5> Args;
|
|
|
|
Arg.split(Args, ':', 5);
|
|
|
|
if (Args.size() < 5)
|
|
|
|
return true;
|
|
|
|
|
2020-01-28 20:23:46 +01:00
|
|
|
BlockName = std::string(Args[0]);
|
2015-11-03 18:33:07 +00:00
|
|
|
if (Args[1].getAsInteger(10, MajorVersion)) return true;
|
|
|
|
if (Args[2].getAsInteger(10, MinorVersion)) return true;
|
|
|
|
if (Args[3].getAsInteger(2, Hashed)) return true;
|
|
|
|
if (Args.size() > 4)
|
2020-01-28 20:23:46 +01:00
|
|
|
UserInfo = std::string(Args[4]);
|
2015-11-03 18:33:07 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-06-07 23:22:09 +00:00
|
|
|
static InputKind ParseFrontendArgs(FrontendOptions &Opts, ArgList &Args,
|
2016-10-27 14:17:10 +00:00
|
|
|
DiagnosticsEngine &Diags,
|
|
|
|
bool &IsHeaderFile) {
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.ProgramAction = frontend::ParseSyntaxOnly;
|
|
|
|
if (const Arg *A = Args.getLastArg(OPT_Action_Group)) {
|
|
|
|
switch (A->getOption().getID()) {
|
|
|
|
default:
|
2011-09-23 05:06:16 +00:00
|
|
|
llvm_unreachable("Invalid option in group!");
|
2012-07-31 09:37:40 +00:00
|
|
|
case OPT_ast_list:
|
|
|
|
Opts.ProgramAction = frontend::ASTDeclList; break;
|
2019-05-13 21:39:55 +00:00
|
|
|
case OPT_ast_dump_all_EQ:
|
|
|
|
case OPT_ast_dump_EQ: {
|
|
|
|
unsigned Val = llvm::StringSwitch<unsigned>(A->getValue())
|
|
|
|
.CaseLower("default", ADOF_Default)
|
|
|
|
.CaseLower("json", ADOF_JSON)
|
|
|
|
.Default(std::numeric_limits<unsigned>::max());
|
|
|
|
|
|
|
|
if (Val != std::numeric_limits<unsigned>::max())
|
|
|
|
Opts.ASTDumpFormat = static_cast<ASTDumpOutputFormat>(Val);
|
|
|
|
else {
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< A->getAsString(Args) << A->getValue();
|
|
|
|
Opts.ASTDumpFormat = ADOF_Default;
|
|
|
|
}
|
|
|
|
LLVM_FALLTHROUGH;
|
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
case OPT_ast_dump:
|
2017-03-09 22:00:01 +00:00
|
|
|
case OPT_ast_dump_all:
|
2014-08-11 22:11:07 +00:00
|
|
|
case OPT_ast_dump_lookups:
|
2020-06-22 15:55:59 -07:00
|
|
|
case OPT_ast_dump_decl_types:
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.ProgramAction = frontend::ASTDump; break;
|
|
|
|
case OPT_ast_print:
|
|
|
|
Opts.ProgramAction = frontend::ASTPrint; break;
|
|
|
|
case OPT_ast_view:
|
|
|
|
Opts.ProgramAction = frontend::ASTView; break;
|
2018-05-31 13:57:09 +00:00
|
|
|
case OPT_compiler_options_dump:
|
|
|
|
Opts.ProgramAction = frontend::DumpCompilerOptions; break;
|
2009-12-01 03:16:53 +00:00
|
|
|
case OPT_dump_raw_tokens:
|
|
|
|
Opts.ProgramAction = frontend::DumpRawTokens; break;
|
|
|
|
case OPT_dump_tokens:
|
|
|
|
Opts.ProgramAction = frontend::DumpTokens; break;
|
|
|
|
case OPT_S:
|
|
|
|
Opts.ProgramAction = frontend::EmitAssembly; break;
|
|
|
|
case OPT_emit_llvm_bc:
|
|
|
|
Opts.ProgramAction = frontend::EmitBC; break;
|
|
|
|
case OPT_emit_html:
|
|
|
|
Opts.ProgramAction = frontend::EmitHTML; break;
|
|
|
|
case OPT_emit_llvm:
|
|
|
|
Opts.ProgramAction = frontend::EmitLLVM; break;
|
|
|
|
case OPT_emit_llvm_only:
|
|
|
|
Opts.ProgramAction = frontend::EmitLLVMOnly; break;
|
2010-05-25 18:41:01 +00:00
|
|
|
case OPT_emit_codegen_only:
|
|
|
|
Opts.ProgramAction = frontend::EmitCodeGenOnly; break;
|
2010-02-03 01:18:43 +00:00
|
|
|
case OPT_emit_obj:
|
|
|
|
Opts.ProgramAction = frontend::EmitObj; break;
|
2010-04-24 01:30:46 +00:00
|
|
|
case OPT_fixit_EQ:
|
2012-11-01 04:30:05 +00:00
|
|
|
Opts.FixItSuffix = A->getValue();
|
Fix clang -Wimplicit-fallthrough warnings across llvm, NFC
This patch should not introduce any behavior changes. It consists of
mostly one of two changes:
1. Replacing fall through comments with the LLVM_FALLTHROUGH macro
2. Inserting 'break' before falling through into a case block consisting
of only 'break'.
We were already using this warning with GCC, but its warning behaves
slightly differently. In this patch, the following differences are
relevant:
1. GCC recognizes comments that say "fall through" as annotations, clang
doesn't
2. GCC doesn't warn on "case N: foo(); default: break;", clang does
3. GCC doesn't warn when the case contains a switch, but falls through
the outer case.
I will enable the warning separately in a follow-up patch so that it can
be cleanly reverted if necessary.
Reviewers: alexfh, rsmith, lattner, rtrieu, EricWF, bollu
Differential Revision: https://reviews.llvm.org/D53950
llvm-svn: 345882
2018-11-01 19:54:45 +00:00
|
|
|
LLVM_FALLTHROUGH;
|
2009-12-01 03:16:53 +00:00
|
|
|
case OPT_fixit:
|
|
|
|
Opts.ProgramAction = frontend::FixIt; break;
|
2011-08-25 22:30:56 +00:00
|
|
|
case OPT_emit_module:
|
|
|
|
Opts.ProgramAction = frontend::GenerateModule; break;
|
2016-08-26 00:14:38 +00:00
|
|
|
case OPT_emit_module_interface:
|
|
|
|
Opts.ProgramAction = frontend::GenerateModuleInterface; break;
|
2018-09-15 01:21:15 +00:00
|
|
|
case OPT_emit_header_module:
|
|
|
|
Opts.ProgramAction = frontend::GenerateHeaderModule; break;
|
2009-12-01 03:16:53 +00:00
|
|
|
case OPT_emit_pch:
|
|
|
|
Opts.ProgramAction = frontend::GeneratePCH; break;
|
2019-10-12 06:25:07 +00:00
|
|
|
case OPT_emit_interface_stubs: {
|
2019-08-22 23:44:34 +00:00
|
|
|
StringRef ArgStr =
|
2019-10-12 06:25:07 +00:00
|
|
|
Args.hasArg(OPT_interface_stub_version_EQ)
|
|
|
|
? Args.getLastArgValue(OPT_interface_stub_version_EQ)
|
2020-03-28 04:08:27 -04:00
|
|
|
: "experimental-ifs-v2";
|
2019-10-08 15:23:14 +00:00
|
|
|
if (ArgStr == "experimental-yaml-elf-v1" ||
|
2020-03-28 04:08:27 -04:00
|
|
|
ArgStr == "experimental-ifs-v1" ||
|
2019-10-08 15:23:14 +00:00
|
|
|
ArgStr == "experimental-tapi-elf-v1") {
|
2019-08-22 23:44:34 +00:00
|
|
|
std::string ErrorMessage =
|
|
|
|
"Invalid interface stub format: " + ArgStr.str() +
|
2019-10-08 15:23:14 +00:00
|
|
|
" is deprecated.";
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< "Must specify a valid interface stub format type, ie: "
|
2020-03-28 04:08:27 -04:00
|
|
|
"-interface-stub-version=experimental-ifs-v2"
|
2019-10-08 15:23:14 +00:00
|
|
|
<< ErrorMessage;
|
2020-03-28 04:08:27 -04:00
|
|
|
} else if (!ArgStr.startswith("experimental-ifs-")) {
|
2019-10-08 15:23:14 +00:00
|
|
|
std::string ErrorMessage =
|
|
|
|
"Invalid interface stub format: " + ArgStr.str() + ".";
|
2019-06-20 16:59:48 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
2019-08-22 23:44:34 +00:00
|
|
|
<< "Must specify a valid interface stub format type, ie: "
|
2020-03-28 04:08:27 -04:00
|
|
|
"-interface-stub-version=experimental-ifs-v2"
|
2019-08-22 23:44:34 +00:00
|
|
|
<< ErrorMessage;
|
|
|
|
} else {
|
2020-03-28 04:08:27 -04:00
|
|
|
Opts.ProgramAction = frontend::GenerateInterfaceStubs;
|
2019-08-22 23:44:34 +00:00
|
|
|
}
|
2019-06-20 16:59:48 +00:00
|
|
|
break;
|
|
|
|
}
|
2010-03-19 19:44:04 +00:00
|
|
|
case OPT_init_only:
|
|
|
|
Opts.ProgramAction = frontend::InitOnly; break;
|
2009-12-01 03:16:53 +00:00
|
|
|
case OPT_fsyntax_only:
|
|
|
|
Opts.ProgramAction = frontend::ParseSyntaxOnly; break;
|
2013-03-27 16:47:18 +00:00
|
|
|
case OPT_module_file_info:
|
|
|
|
Opts.ProgramAction = frontend::ModuleFileInfo; break;
|
2014-02-05 22:21:15 +00:00
|
|
|
case OPT_verify_pch:
|
|
|
|
Opts.ProgramAction = frontend::VerifyPCH; break;
|
2010-07-20 20:18:03 +00:00
|
|
|
case OPT_print_preamble:
|
|
|
|
Opts.ProgramAction = frontend::PrintPreamble; break;
|
2009-12-01 03:16:53 +00:00
|
|
|
case OPT_E:
|
|
|
|
Opts.ProgramAction = frontend::PrintPreprocessedInput; break;
|
2018-02-10 14:04:45 +00:00
|
|
|
case OPT_templight_dump:
|
|
|
|
Opts.ProgramAction = frontend::TemplightDump; break;
|
2009-12-01 03:16:53 +00:00
|
|
|
case OPT_rewrite_macros:
|
|
|
|
Opts.ProgramAction = frontend::RewriteMacros; break;
|
|
|
|
case OPT_rewrite_objc:
|
|
|
|
Opts.ProgramAction = frontend::RewriteObjC; break;
|
|
|
|
case OPT_rewrite_test:
|
|
|
|
Opts.ProgramAction = frontend::RewriteTest; break;
|
|
|
|
case OPT_analyze:
|
|
|
|
Opts.ProgramAction = frontend::RunAnalysis; break;
|
2012-03-06 20:06:33 +00:00
|
|
|
case OPT_migrate:
|
|
|
|
Opts.ProgramAction = frontend::MigrateSource; break;
|
2009-12-01 03:16:53 +00:00
|
|
|
case OPT_Eonly:
|
|
|
|
Opts.ProgramAction = frontend::RunPreprocessorOnly; break;
|
2019-06-03 22:59:17 +00:00
|
|
|
case OPT_print_dependency_directives_minimized_source:
|
|
|
|
Opts.ProgramAction =
|
|
|
|
frontend::PrintDependencyDirectivesSourceMinimizerOutput;
|
|
|
|
break;
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
}
|
2010-06-16 16:59:23 +00:00
|
|
|
|
|
|
|
if (const Arg* A = Args.getLastArg(OPT_plugin)) {
|
2015-05-29 19:42:19 +00:00
|
|
|
Opts.Plugins.emplace_back(A->getValue(0));
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.ProgramAction = frontend::PluginAction;
|
2012-11-01 04:30:05 +00:00
|
|
|
Opts.ActionName = A->getValue();
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
2019-01-05 01:10:20 +00:00
|
|
|
Opts.AddPluginActions = Args.getAllArgValues(OPT_add_plugin);
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *AA : Args.filtered(OPT_plugin_arg))
|
2016-03-15 12:51:40 +00:00
|
|
|
Opts.PluginArgs[AA->getValue(0)].emplace_back(AA->getValue(1));
|
2011-01-25 20:34:14 +00:00
|
|
|
|
2015-11-03 18:33:07 +00:00
|
|
|
for (const std::string &Arg :
|
|
|
|
Args.getAllArgValues(OPT_ftest_module_file_extension_EQ)) {
|
|
|
|
std::string BlockName;
|
|
|
|
unsigned MajorVersion;
|
|
|
|
unsigned MinorVersion;
|
|
|
|
bool Hashed;
|
|
|
|
std::string UserInfo;
|
|
|
|
if (parseTestModuleFileExtensionArg(Arg, BlockName, MajorVersion,
|
|
|
|
MinorVersion, Hashed, UserInfo)) {
|
|
|
|
Diags.Report(diag::err_test_module_file_extension_format) << Arg;
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the testing module file extension.
|
|
|
|
Opts.ModuleFileExtensions.push_back(
|
2017-01-05 18:23:18 +00:00
|
|
|
std::make_shared<TestModuleFileExtension>(
|
|
|
|
BlockName, MajorVersion, MinorVersion, Hashed, UserInfo));
|
2015-11-03 18:33:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-01 03:16:53 +00:00
|
|
|
if (const Arg *A = Args.getLastArg(OPT_code_completion_at)) {
|
|
|
|
Opts.CodeCompletionAt =
|
2012-11-01 04:30:05 +00:00
|
|
|
ParsedSourceLocation::FromString(A->getValue());
|
2009-12-01 03:16:53 +00:00
|
|
|
if (Opts.CodeCompletionAt.FileName.empty())
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
2012-11-01 04:30:05 +00:00
|
|
|
<< A->getAsString(Args) << A->getValue();
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
2010-05-20 16:54:55 +00:00
|
|
|
Opts.Plugins = Args.getAllArgValues(OPT_load);
|
2019-05-13 21:39:55 +00:00
|
|
|
Opts.ASTDumpDecls = Args.hasArg(OPT_ast_dump, OPT_ast_dump_EQ);
|
|
|
|
Opts.ASTDumpAll = Args.hasArg(OPT_ast_dump_all, OPT_ast_dump_all_EQ);
|
2017-08-31 06:26:43 +00:00
|
|
|
// Only the -fmodule-file=<file> form.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_fmodule_file)) {
|
2017-08-31 06:26:43 +00:00
|
|
|
StringRef Val = A->getValue();
|
|
|
|
if (Val.find('=') == StringRef::npos)
|
2020-01-28 20:23:46 +01:00
|
|
|
Opts.ModuleFiles.push_back(std::string(Val));
|
2017-08-31 06:26:43 +00:00
|
|
|
}
|
2020-11-17 17:25:50 -08:00
|
|
|
Opts.AllowPCMWithCompilerErrors = Args.hasArg(OPT_fallow_pcm_with_errors);
|
2020-02-28 17:31:52 -08:00
|
|
|
|
|
|
|
if (Opts.ProgramAction != frontend::GenerateModule && Opts.IsSystemModule)
|
|
|
|
Diags.Report(diag::err_drv_argument_only_allowed_with) << "-fsystem-module"
|
|
|
|
<< "-emit-module";
|
2014-10-22 02:05:46 +00:00
|
|
|
|
[hip] Properly populate macros based on host processor.
Summary:
- The device compilation needs to have a consistent source code compared
to the corresponding host compilation. If macros based on the
host-specific target processor is not properly populated, the device
compilation may fail due to the inconsistent source after the
preprocessor. So far, only the host triple is used to build the
macros. If a detailed host CPU target or certain features are
specified, macros derived from them won't be populated properly, e.g.
`__SSE3__` won't be added unless `+sse3` feature is present. On
Windows compilation compatible with MSVC, that missing macros result
in that intrinsics are not included and cause device compilation
failure on the host-side source.
- This patch addresses this issue by introducing two `cc1` options,
i.e., `-aux-target-cpu` and `-aux-target-feature`. If a specific host
CPU target or certain features are specified, the compiler driver will
append them during the construction of the offline compilation
actions. Then, the toolchain in `cc1` phase will populate macros
accordingly.
- An internal option `--gpu-use-aux-triple-only` is added to fall back
the original behavior to help diagnosing potential issues from the new
behavior.
Reviewers: tra, yaxunl
Subscribers: cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D73942
2020-02-03 15:35:18 -05:00
|
|
|
if (Args.hasArg(OPT_aux_target_cpu))
|
|
|
|
Opts.AuxTargetCPU = std::string(Args.getLastArgValue(OPT_aux_target_cpu));
|
|
|
|
if (Args.hasArg(OPT_aux_target_feature))
|
|
|
|
Opts.AuxTargetFeatures = Args.getAllArgValues(OPT_aux_target_feature);
|
2013-11-14 16:33:29 +00:00
|
|
|
|
2012-03-06 20:06:33 +00:00
|
|
|
if (Opts.ARCMTAction != FrontendOptions::ARCMT_None &&
|
|
|
|
Opts.ObjCMTAction != FrontendOptions::ObjCMT_None) {
|
|
|
|
Diags.Report(diag::err_drv_argument_not_allowed_with)
|
|
|
|
<< "ARC migration" << "ObjC migration";
|
|
|
|
}
|
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
InputKind DashX(Language::Unknown);
|
2009-12-01 03:16:53 +00:00
|
|
|
if (const Arg *A = Args.getLastArg(OPT_x)) {
|
2017-04-26 18:57:40 +00:00
|
|
|
StringRef XValue = A->getValue();
|
2017-04-28 01:49:42 +00:00
|
|
|
|
|
|
|
// Parse suffixes: '<lang>(-header|[-module-map][-cpp-output])'.
|
|
|
|
// FIXME: Supporting '<lang>-header-cpp-output' would be useful.
|
|
|
|
bool Preprocessed = XValue.consume_back("-cpp-output");
|
|
|
|
bool ModuleMap = XValue.consume_back("-module-map");
|
2020-07-13 22:24:44 +02:00
|
|
|
IsHeaderFile = !Preprocessed && !ModuleMap &&
|
|
|
|
XValue != "precompiled-header" &&
|
|
|
|
XValue.consume_back("-header");
|
2017-04-28 01:49:42 +00:00
|
|
|
|
|
|
|
// Principal languages.
|
2017-04-26 18:57:40 +00:00
|
|
|
DashX = llvm::StringSwitch<InputKind>(XValue)
|
2019-08-05 13:59:26 +00:00
|
|
|
.Case("c", Language::C)
|
|
|
|
.Case("cl", Language::OpenCL)
|
|
|
|
.Case("cuda", Language::CUDA)
|
|
|
|
.Case("hip", Language::HIP)
|
|
|
|
.Case("c++", Language::CXX)
|
|
|
|
.Case("objective-c", Language::ObjC)
|
|
|
|
.Case("objective-c++", Language::ObjCXX)
|
|
|
|
.Case("renderscript", Language::RenderScript)
|
|
|
|
.Default(Language::Unknown);
|
2017-04-28 01:49:42 +00:00
|
|
|
|
|
|
|
// "objc[++]-cpp-output" is an acceptable synonym for
|
|
|
|
// "objective-c[++]-cpp-output".
|
|
|
|
if (DashX.isUnknown() && Preprocessed && !IsHeaderFile && !ModuleMap)
|
|
|
|
DashX = llvm::StringSwitch<InputKind>(XValue)
|
2019-08-05 13:59:26 +00:00
|
|
|
.Case("objc", Language::ObjC)
|
|
|
|
.Case("objc++", Language::ObjCXX)
|
|
|
|
.Default(Language::Unknown);
|
2017-04-28 01:49:42 +00:00
|
|
|
|
|
|
|
// Some special cases cannot be combined with suffixes.
|
|
|
|
if (DashX.isUnknown() && !Preprocessed && !ModuleMap && !IsHeaderFile)
|
|
|
|
DashX = llvm::StringSwitch<InputKind>(XValue)
|
2019-08-05 13:59:26 +00:00
|
|
|
.Case("cpp-output", InputKind(Language::C).getPreprocessed())
|
|
|
|
.Case("assembler-with-cpp", Language::Asm)
|
2020-07-13 22:24:44 +02:00
|
|
|
.Cases("ast", "pcm", "precompiled-header",
|
2019-08-05 13:59:26 +00:00
|
|
|
InputKind(Language::Unknown, InputKind::Precompiled))
|
|
|
|
.Case("ir", Language::LLVM_IR)
|
|
|
|
.Default(Language::Unknown);
|
2017-04-28 01:49:42 +00:00
|
|
|
|
2017-04-26 18:57:40 +00:00
|
|
|
if (DashX.isUnknown())
|
2009-12-01 03:16:53 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
2012-11-01 04:30:05 +00:00
|
|
|
<< A->getAsString(Args) << A->getValue();
|
2017-04-26 18:57:40 +00:00
|
|
|
|
2017-04-28 01:49:42 +00:00
|
|
|
if (Preprocessed)
|
2017-04-26 18:57:40 +00:00
|
|
|
DashX = DashX.getPreprocessed();
|
2017-04-28 01:49:42 +00:00
|
|
|
if (ModuleMap)
|
|
|
|
DashX = DashX.withFormat(InputKind::ModuleMap);
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// '-' is the default input if none is given.
|
2010-05-20 16:54:55 +00:00
|
|
|
std::vector<std::string> Inputs = Args.getAllArgValues(OPT_INPUT);
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.Inputs.clear();
|
|
|
|
if (Inputs.empty())
|
|
|
|
Inputs.push_back("-");
|
|
|
|
for (unsigned i = 0, e = Inputs.size(); i != e; ++i) {
|
2010-06-07 23:22:09 +00:00
|
|
|
InputKind IK = DashX;
|
2017-04-26 18:57:40 +00:00
|
|
|
if (IK.isUnknown()) {
|
2009-12-01 03:16:53 +00:00
|
|
|
IK = FrontendOptions::getInputKindForExtension(
|
2011-07-23 10:55:15 +00:00
|
|
|
StringRef(Inputs[i]).rsplit('.').second);
|
2017-04-26 18:57:40 +00:00
|
|
|
// FIXME: Warn on this?
|
|
|
|
if (IK.isUnknown())
|
2019-08-05 13:59:26 +00:00
|
|
|
IK = Language::C;
|
2009-12-01 03:16:53 +00:00
|
|
|
// FIXME: Remove this hack.
|
|
|
|
if (i == 0)
|
|
|
|
DashX = IK;
|
|
|
|
}
|
2017-04-28 01:49:42 +00:00
|
|
|
|
2020-02-28 17:31:52 -08:00
|
|
|
bool IsSystem = false;
|
|
|
|
|
2017-04-28 01:49:42 +00:00
|
|
|
// The -emit-module action implicitly takes a module map.
|
|
|
|
if (Opts.ProgramAction == frontend::GenerateModule &&
|
2020-02-28 17:31:52 -08:00
|
|
|
IK.getFormat() == InputKind::Source) {
|
2017-04-28 01:49:42 +00:00
|
|
|
IK = IK.withFormat(InputKind::ModuleMap);
|
2020-02-28 17:31:52 -08:00
|
|
|
IsSystem = Opts.IsSystemModule;
|
|
|
|
}
|
2017-04-28 01:49:42 +00:00
|
|
|
|
2020-02-28 17:31:52 -08:00
|
|
|
Opts.Inputs.emplace_back(std::move(Inputs[i]), IK, IsSystem);
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return DashX;
|
|
|
|
}
|
|
|
|
|
2009-12-15 00:06:45 +00:00
|
|
|
std::string CompilerInvocation::GetResourcesPath(const char *Argv0,
|
|
|
|
void *MainAddr) {
|
2014-12-29 21:28:15 +00:00
|
|
|
std::string ClangExecutable =
|
|
|
|
llvm::sys::fs::getMainExecutable(Argv0, MainAddr);
|
2019-01-31 22:15:32 +00:00
|
|
|
return Driver::GetResourcesPath(ClangExecutable, CLANG_RESOURCE_DIR);
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
2017-03-14 23:07:49 +00:00
|
|
|
static void ParseHeaderSearchArgs(HeaderSearchOptions &Opts, ArgList &Args,
|
|
|
|
const std::string &WorkingDir) {
|
2011-06-21 21:12:29 +00:00
|
|
|
if (const Arg *A = Args.getLastArg(OPT_stdlib_EQ))
|
2012-11-01 04:30:05 +00:00
|
|
|
Opts.UseLibcxx = (strcmp(A->getValue(), "libc++") == 0);
|
2017-03-14 23:07:49 +00:00
|
|
|
|
|
|
|
// Canonicalize -fmodules-cache-path before storing it.
|
|
|
|
SmallString<128> P(Args.getLastArgValue(OPT_fmodules_cache_path));
|
|
|
|
if (!(P.empty() || llvm::sys::path::is_absolute(P))) {
|
|
|
|
if (WorkingDir.empty())
|
|
|
|
llvm::sys::fs::make_absolute(P);
|
|
|
|
else
|
|
|
|
llvm::sys::fs::make_absolute(WorkingDir, P);
|
|
|
|
}
|
|
|
|
llvm::sys::path::remove_dots(P);
|
2020-01-28 20:23:46 +01:00
|
|
|
Opts.ModuleCachePath = std::string(P.str());
|
2017-03-14 23:07:49 +00:00
|
|
|
|
2017-08-31 06:26:43 +00:00
|
|
|
// Only the -fmodule-file=<name>=<file> form.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_fmodule_file)) {
|
2017-08-31 06:26:43 +00:00
|
|
|
StringRef Val = A->getValue();
|
2020-01-29 00:42:56 +01:00
|
|
|
if (Val.find('=') != StringRef::npos){
|
|
|
|
auto Split = Val.split('=');
|
|
|
|
Opts.PrebuiltModuleFiles.insert(
|
|
|
|
{std::string(Split.first), std::string(Split.second)});
|
|
|
|
}
|
2017-08-31 06:26:43 +00:00
|
|
|
}
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_fprebuilt_module_path))
|
2016-08-18 17:42:15 +00:00
|
|
|
Opts.AddPrebuiltModulePath(A->getValue());
|
2014-03-12 00:06:17 +00:00
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_fmodules_ignore_macro)) {
|
2015-06-09 01:57:17 +00:00
|
|
|
StringRef MacroDef = A->getValue();
|
2016-10-21 21:45:01 +00:00
|
|
|
Opts.ModulesIgnoreMacros.insert(
|
|
|
|
llvm::CachedHashString(MacroDef.split('=').first));
|
2013-02-07 00:21:12 +00:00
|
|
|
}
|
|
|
|
|
2011-07-28 04:45:53 +00:00
|
|
|
// Add -I..., -F..., and -index-header-map options in order.
|
|
|
|
bool IsIndexHeaderMap = false;
|
2016-05-06 19:13:55 +00:00
|
|
|
bool IsSysrootSpecified =
|
|
|
|
Args.hasArg(OPT__sysroot_EQ) || Args.hasArg(OPT_isysroot);
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_I, OPT_F, OPT_index_header_map)) {
|
2015-06-09 01:57:17 +00:00
|
|
|
if (A->getOption().matches(OPT_index_header_map)) {
|
2011-07-28 04:45:53 +00:00
|
|
|
// -index-header-map applies to the next -I or -F.
|
|
|
|
IsIndexHeaderMap = true;
|
|
|
|
continue;
|
|
|
|
}
|
2015-06-09 01:57:17 +00:00
|
|
|
|
|
|
|
frontend::IncludeDirGroup Group =
|
|
|
|
IsIndexHeaderMap ? frontend::IndexHeaderMap : frontend::Angled;
|
|
|
|
|
2016-05-06 19:13:55 +00:00
|
|
|
bool IsFramework = A->getOption().matches(OPT_F);
|
|
|
|
std::string Path = A->getValue();
|
|
|
|
|
|
|
|
if (IsSysrootSpecified && !IsFramework && A->getValue()[0] == '=') {
|
|
|
|
SmallString<32> Buffer;
|
|
|
|
llvm::sys::path::append(Buffer, Opts.Sysroot,
|
|
|
|
llvm::StringRef(A->getValue()).substr(1));
|
2020-01-28 20:23:46 +01:00
|
|
|
Path = std::string(Buffer.str());
|
2016-05-06 19:13:55 +00:00
|
|
|
}
|
|
|
|
|
2016-11-02 10:39:27 +00:00
|
|
|
Opts.AddPath(Path, Group, IsFramework,
|
2016-05-06 19:13:55 +00:00
|
|
|
/*IgnoreSysroot*/ true);
|
2011-07-28 04:45:53 +00:00
|
|
|
IsIndexHeaderMap = false;
|
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
|
2013-01-25 01:50:34 +00:00
|
|
|
// Add -iprefix/-iwithprefix/-iwithprefixbefore options.
|
2011-07-23 10:55:15 +00:00
|
|
|
StringRef Prefix = ""; // FIXME: This isn't the correct default prefix.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A :
|
2015-06-09 01:57:17 +00:00
|
|
|
Args.filtered(OPT_iprefix, OPT_iwithprefix, OPT_iwithprefixbefore)) {
|
2010-06-11 22:00:13 +00:00
|
|
|
if (A->getOption().matches(OPT_iprefix))
|
2012-11-01 04:30:05 +00:00
|
|
|
Prefix = A->getValue();
|
2010-06-11 22:00:13 +00:00
|
|
|
else if (A->getOption().matches(OPT_iwithprefix))
|
2015-06-09 01:57:17 +00:00
|
|
|
Opts.AddPath(Prefix.str() + A->getValue(), frontend::After, false, true);
|
2009-12-01 03:16:53 +00:00
|
|
|
else
|
2015-06-09 01:57:17 +00:00
|
|
|
Opts.AddPath(Prefix.str() + A->getValue(), frontend::Angled, false, true);
|
|
|
|
}
|
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_idirafter))
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AddPath(A->getValue(), frontend::After, false, true);
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_iquote))
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AddPath(A->getValue(), frontend::Quoted, false, true);
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_isystem, OPT_iwithsysroot))
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AddPath(A->getValue(), frontend::System, false,
|
|
|
|
!A->getOption().matches(OPT_iwithsysroot));
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_iframework))
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AddPath(A->getValue(), frontend::System, true, true);
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_iframeworkwithsysroot))
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AddPath(A->getValue(), frontend::System, /*IsFramework=*/true,
|
|
|
|
/*IgnoreSysRoot=*/false);
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
// Add the paths for the various language specific isystem flags.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_c_isystem))
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AddPath(A->getValue(), frontend::CSystem, false, true);
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_cxx_isystem))
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AddPath(A->getValue(), frontend::CXXSystem, false, true);
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_objc_isystem))
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AddPath(A->getValue(), frontend::ObjCSystem, false,true);
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_objcxx_isystem))
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AddPath(A->getValue(), frontend::ObjCXXSystem, false, true);
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
// Add the internal paths from a driver that detects standard include paths.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A :
|
2017-12-07 21:46:26 +00:00
|
|
|
Args.filtered(OPT_internal_isystem, OPT_internal_externc_isystem)) {
|
|
|
|
frontend::IncludeDirGroup Group = frontend::System;
|
|
|
|
if (A->getOption().matches(OPT_internal_externc_isystem))
|
|
|
|
Group = frontend::ExternCSystem;
|
|
|
|
Opts.AddPath(A->getValue(), Group, false, true);
|
|
|
|
}
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
// Add the path prefixes which are implicitly treated as being system headers.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A :
|
2017-12-07 21:46:26 +00:00
|
|
|
Args.filtered(OPT_system_header_prefix, OPT_no_system_header_prefix))
|
|
|
|
Opts.AddSystemHeaderPrefix(
|
|
|
|
A->getValue(), A->getOption().matches(OPT_system_header_prefix));
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_ivfsoverlay))
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AddVFSOverlayFile(A->getValue());
|
|
|
|
}
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
void CompilerInvocation::setLangDefaults(LangOptions &Opts, InputKind IK,
|
|
|
|
const llvm::Triple &T,
|
|
|
|
PreprocessorOptions &PPOpts,
|
|
|
|
LangStandard::Kind LangStd) {
|
|
|
|
// Set some properties which depend solely on the input kind; it would be nice
|
|
|
|
// to move these to the language standard, and have the driver resolve the
|
|
|
|
// input kind + language standard.
|
|
|
|
//
|
|
|
|
// FIXME: Perhaps a better model would be for a single source file to have
|
|
|
|
// multiple language standards (C / C++ std, ObjC std, OpenCL std, OpenMP std)
|
|
|
|
// simultaneously active?
|
2019-08-05 13:59:26 +00:00
|
|
|
if (IK.getLanguage() == Language::Asm) {
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AsmPreprocessor = 1;
|
|
|
|
} else if (IK.isObjectiveC()) {
|
2018-10-30 20:31:30 +00:00
|
|
|
Opts.ObjC = 1;
|
2017-12-07 21:46:26 +00:00
|
|
|
}
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
if (LangStd == LangStandard::lang_unspecified) {
|
|
|
|
// Based on the base language, pick one.
|
|
|
|
switch (IK.getLanguage()) {
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::Unknown:
|
|
|
|
case Language::LLVM_IR:
|
2017-12-07 21:46:26 +00:00
|
|
|
llvm_unreachable("Invalid input kind!");
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::OpenCL:
|
2017-12-07 21:46:26 +00:00
|
|
|
LangStd = LangStandard::lang_opencl10;
|
|
|
|
break;
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::CUDA:
|
2017-12-07 21:46:26 +00:00
|
|
|
LangStd = LangStandard::lang_cuda;
|
|
|
|
break;
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::Asm:
|
|
|
|
case Language::C:
|
2018-03-06 21:26:28 +00:00
|
|
|
#if defined(CLANG_DEFAULT_STD_C)
|
|
|
|
LangStd = CLANG_DEFAULT_STD_C;
|
|
|
|
#else
|
2017-12-07 21:46:26 +00:00
|
|
|
// The PS4 uses C99 as the default C standard.
|
|
|
|
if (T.isPS4())
|
|
|
|
LangStd = LangStandard::lang_gnu99;
|
|
|
|
else
|
2020-03-02 09:33:14 -08:00
|
|
|
LangStd = LangStandard::lang_gnu17;
|
2018-03-06 21:26:28 +00:00
|
|
|
#endif
|
2017-12-07 21:46:26 +00:00
|
|
|
break;
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::ObjC:
|
2018-03-06 21:26:28 +00:00
|
|
|
#if defined(CLANG_DEFAULT_STD_C)
|
|
|
|
LangStd = CLANG_DEFAULT_STD_C;
|
|
|
|
#else
|
2017-12-07 21:46:26 +00:00
|
|
|
LangStd = LangStandard::lang_gnu11;
|
2018-03-06 21:26:28 +00:00
|
|
|
#endif
|
2017-12-07 21:46:26 +00:00
|
|
|
break;
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::CXX:
|
|
|
|
case Language::ObjCXX:
|
2018-03-06 21:26:28 +00:00
|
|
|
#if defined(CLANG_DEFAULT_STD_CXX)
|
|
|
|
LangStd = CLANG_DEFAULT_STD_CXX;
|
|
|
|
#else
|
2017-12-09 12:09:54 +00:00
|
|
|
LangStd = LangStandard::lang_gnucxx14;
|
2018-03-06 21:26:28 +00:00
|
|
|
#endif
|
2017-12-07 21:46:26 +00:00
|
|
|
break;
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::RenderScript:
|
2017-12-07 21:46:26 +00:00
|
|
|
LangStd = LangStandard::lang_c99;
|
|
|
|
break;
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::HIP:
|
2018-04-25 01:10:37 +00:00
|
|
|
LangStd = LangStandard::lang_hip;
|
|
|
|
break;
|
2017-12-07 21:46:26 +00:00
|
|
|
}
|
|
|
|
}
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
const LangStandard &Std = LangStandard::getLangStandardForKind(LangStd);
|
|
|
|
Opts.LineComment = Std.hasLineComments();
|
|
|
|
Opts.C99 = Std.isC99();
|
|
|
|
Opts.C11 = Std.isC11();
|
|
|
|
Opts.C17 = Std.isC17();
|
2019-05-14 12:09:55 +00:00
|
|
|
Opts.C2x = Std.isC2x();
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.CPlusPlus = Std.isCPlusPlus();
|
|
|
|
Opts.CPlusPlus11 = Std.isCPlusPlus11();
|
|
|
|
Opts.CPlusPlus14 = Std.isCPlusPlus14();
|
|
|
|
Opts.CPlusPlus17 = Std.isCPlusPlus17();
|
2020-04-21 15:37:19 -04:00
|
|
|
Opts.CPlusPlus20 = Std.isCPlusPlus20();
|
2020-12-03 10:27:09 +01:00
|
|
|
Opts.CPlusPlus2b = Std.isCPlusPlus2b();
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.Digraphs = Std.hasDigraphs();
|
|
|
|
Opts.GNUMode = Std.isGNUMode();
|
|
|
|
Opts.GNUInline = !Opts.C99 && !Opts.CPlusPlus;
|
Add -fgnuc-version= to control __GNUC__ and other GCC macros
I noticed that compiling on Windows with -fno-ms-compatibility had the
side effect of defining __GNUC__, along with __GNUG__, __GXX_RTTI__, and
a number of other macros for GCC compatibility. This is undesirable and
causes Chromium to do things like mix __attribute__ and __declspec,
which doesn't work. We should have a positive language option to enable
GCC compatibility features so that we can experiment with
-fno-ms-compatibility on Windows. This change adds -fgnuc-version= to be
that option.
My issue aside, users have, for a long time, reported that __GNUC__
doesn't match their expectations in one way or another. We have
encouraged users to migrate code away from this macro, but new code
continues to be written assuming a GCC-only environment. There's really
nothing we can do to stop that. By adding this flag, we can allow them
to choose their own adventure with __GNUC__.
This overlaps a bit with the "GNUMode" language option from -std=gnu*.
The gnu language mode tends to enable non-conforming behaviors that we'd
rather not enable by default, but the we want to set things like
__GXX_RTTI__ by default, so I've kept these separate.
Helps address PR42817
Reviewed By: hans, nickdesaulniers, MaskRay
Differential Revision: https://reviews.llvm.org/D68055
llvm-svn: 374449
2019-10-10 21:04:25 +00:00
|
|
|
Opts.GNUCVersion = 0;
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.HexFloats = Std.hasHexFloats();
|
|
|
|
Opts.ImplicitInt = Std.hasImplicitInt();
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
// Set OpenCL Version.
|
|
|
|
Opts.OpenCL = Std.isOpenCL();
|
|
|
|
if (LangStd == LangStandard::lang_opencl10)
|
|
|
|
Opts.OpenCLVersion = 100;
|
|
|
|
else if (LangStd == LangStandard::lang_opencl11)
|
|
|
|
Opts.OpenCLVersion = 110;
|
|
|
|
else if (LangStd == LangStandard::lang_opencl12)
|
|
|
|
Opts.OpenCLVersion = 120;
|
|
|
|
else if (LangStd == LangStandard::lang_opencl20)
|
|
|
|
Opts.OpenCLVersion = 200;
|
2020-10-09 15:13:39 +01:00
|
|
|
else if (LangStd == LangStandard::lang_opencl30)
|
|
|
|
Opts.OpenCLVersion = 300;
|
2018-04-12 14:17:04 +00:00
|
|
|
else if (LangStd == LangStandard::lang_openclcpp)
|
|
|
|
Opts.OpenCLCPlusPlusVersion = 100;
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
// OpenCL has some additional defaults.
|
|
|
|
if (Opts.OpenCL) {
|
|
|
|
Opts.AltiVec = 0;
|
|
|
|
Opts.ZVector = 0;
|
2019-09-13 06:02:15 +00:00
|
|
|
Opts.setLaxVectorConversions(LangOptions::LaxVectorConversionKind::None);
|
2020-05-04 10:48:12 -07:00
|
|
|
Opts.setDefaultFPContractMode(LangOptions::FPM_On);
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.NativeHalfType = 1;
|
|
|
|
Opts.NativeHalfArgsAndReturns = 1;
|
2018-04-23 11:23:47 +00:00
|
|
|
Opts.OpenCLCPlusPlus = Opts.CPlusPlus;
|
2019-06-19 12:48:22 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
// Include default header file for OpenCL.
|
2019-06-19 12:48:22 +00:00
|
|
|
if (Opts.IncludeDefaultHeader) {
|
|
|
|
if (Opts.DeclareOpenCLBuiltins) {
|
|
|
|
// Only include base header file for builtin types and constants.
|
|
|
|
PPOpts.Includes.push_back("opencl-c-base.h");
|
|
|
|
} else {
|
|
|
|
PPOpts.Includes.push_back("opencl-c.h");
|
|
|
|
}
|
2017-12-07 21:46:26 +00:00
|
|
|
}
|
|
|
|
}
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
Opts.HIP = IK.getLanguage() == Language::HIP;
|
|
|
|
Opts.CUDA = IK.getLanguage() == Language::CUDA || Opts.HIP;
|
[HIP] Fix regressions due to fp contract change
Recently HIP toolchain made a change to use clang instead of opt/llc to do compilation
(https://reviews.llvm.org/D81861). The intention is to make HIP toolchain canonical like
other toolchains.
However, this change introduced an unintentional change regarding backend fp fuse
option, which caused regressions in some HIP applications.
Basically before the change, HIP toolchain used clang to generate bitcode, then use
opt/llc to optimize bitcode and generate ISA. As such, the amdgpu backend takes
the default fp fuse mode which is 'Standard'. This mode respect contract flag of
fmul/fadd instructions and do not fuse fmul/fadd instructions without contract flag.
However, after the change, HIP toolchain now use clang to generate IR, do optimization,
and generate ISA as one process. Now amdgpu backend fp fuse option is determined
by -ffp-contract option, which is 'fast' by default. And this -ffp-contract=fast language option
is translated to 'Fast' fp fuse option in backend. Suddenly backend starts to fuse fmul/fadd
instructions without contract flag.
This causes wrong result for some device library functions, e.g. tan(-1e20), which should
return 0.8446, now returns -0.933. What is worse is that since backend with 'Fast' fp fuse
option does not respect contract flag, there is no way to use #pragma clang fp contract
directive to enforce fp contract requirements.
This patch fixes the regression by introducing a new value 'fast-honor-pragmas' for -ffp-contract
and use it for HIP by default. 'fast-honor-pragmas' is equivalent to 'fast' in frontend but
let the backend to use 'Standard' fp fuse option. 'fast-honor-pragmas' is useful since 'Fast'
fp fuse option in backend does not honor contract flag, it is of little use to HIP
applications since all code with #pragma STDC FP_CONTRACT or any IR from a
source compiled with -ffp-contract=on is broken.
Differential Revision: https://reviews.llvm.org/D90174
2020-10-23 16:24:48 -04:00
|
|
|
if (Opts.HIP) {
|
|
|
|
// HIP toolchain does not support 'Fast' FPOpFusion in backends since it
|
|
|
|
// fuses multiplication/addition instructions without contract flag from
|
|
|
|
// device library functions in LLVM bitcode, which causes accuracy loss in
|
|
|
|
// certain math functions, e.g. tan(-1e20) becomes -0.933 instead of 0.8446.
|
|
|
|
// For device library functions in bitcode to work, 'Strict' or 'Standard'
|
|
|
|
// FPOpFusion options in backends is needed. Therefore 'fast-honor-pragmas'
|
|
|
|
// FP contract option is used to allow fuse across statements in frontend
|
|
|
|
// whereas respecting contract flag in backend.
|
|
|
|
Opts.setDefaultFPContractMode(LangOptions::FPM_FastHonorPragmas);
|
|
|
|
} else if (Opts.CUDA) {
|
|
|
|
// Allow fuse across statements disregarding pragmas.
|
2020-05-04 10:48:12 -07:00
|
|
|
Opts.setDefaultFPContractMode(LangOptions::FPM_Fast);
|
[HIP] Fix regressions due to fp contract change
Recently HIP toolchain made a change to use clang instead of opt/llc to do compilation
(https://reviews.llvm.org/D81861). The intention is to make HIP toolchain canonical like
other toolchains.
However, this change introduced an unintentional change regarding backend fp fuse
option, which caused regressions in some HIP applications.
Basically before the change, HIP toolchain used clang to generate bitcode, then use
opt/llc to optimize bitcode and generate ISA. As such, the amdgpu backend takes
the default fp fuse mode which is 'Standard'. This mode respect contract flag of
fmul/fadd instructions and do not fuse fmul/fadd instructions without contract flag.
However, after the change, HIP toolchain now use clang to generate IR, do optimization,
and generate ISA as one process. Now amdgpu backend fp fuse option is determined
by -ffp-contract option, which is 'fast' by default. And this -ffp-contract=fast language option
is translated to 'Fast' fp fuse option in backend. Suddenly backend starts to fuse fmul/fadd
instructions without contract flag.
This causes wrong result for some device library functions, e.g. tan(-1e20), which should
return 0.8446, now returns -0.933. What is worse is that since backend with 'Fast' fp fuse
option does not respect contract flag, there is no way to use #pragma clang fp contract
directive to enforce fp contract requirements.
This patch fixes the regression by introducing a new value 'fast-honor-pragmas' for -ffp-contract
and use it for HIP by default. 'fast-honor-pragmas' is equivalent to 'fast' in frontend but
let the backend to use 'Standard' fp fuse option. 'fast-honor-pragmas' is useful since 'Fast'
fp fuse option in backend does not honor contract flag, it is of little use to HIP
applications since all code with #pragma STDC FP_CONTRACT or any IR from a
source compiled with -ffp-contract=on is broken.
Differential Revision: https://reviews.llvm.org/D90174
2020-10-23 16:24:48 -04:00
|
|
|
}
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
Opts.RenderScript = IK.getLanguage() == Language::RenderScript;
|
2017-12-07 21:46:26 +00:00
|
|
|
if (Opts.RenderScript) {
|
|
|
|
Opts.NativeHalfType = 1;
|
|
|
|
Opts.NativeHalfArgsAndReturns = 1;
|
|
|
|
}
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
// OpenCL and C++ both have bool, true, false keywords.
|
|
|
|
Opts.Bool = Opts.OpenCL || Opts.CPlusPlus;
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
// OpenCL has half keyword
|
|
|
|
Opts.Half = Opts.OpenCL;
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
// C++ has wchar_t keyword.
|
|
|
|
Opts.WChar = Opts.CPlusPlus;
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.GNUKeywords = Opts.GNUMode;
|
|
|
|
Opts.CXXOperatorNames = Opts.CPlusPlus;
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.AlignedAllocation = Opts.CPlusPlus17;
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
Opts.DollarIdents = !Opts.AsmPreprocessor;
|
2019-05-14 12:09:55 +00:00
|
|
|
|
|
|
|
// Enable [[]] attributes in C++11 and C2x by default.
|
|
|
|
Opts.DoubleSquareBracketAttributes = Opts.CPlusPlus11 || Opts.C2x;
|
2017-12-07 21:46:26 +00:00
|
|
|
}
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
/// Attempt to parse a visibility value out of the given argument.
|
|
|
|
static Visibility parseVisibility(Arg *arg, ArgList &args,
|
|
|
|
DiagnosticsEngine &diags) {
|
|
|
|
StringRef value = arg->getValue();
|
|
|
|
if (value == "default") {
|
|
|
|
return DefaultVisibility;
|
|
|
|
} else if (value == "hidden" || value == "internal") {
|
|
|
|
return HiddenVisibility;
|
|
|
|
} else if (value == "protected") {
|
|
|
|
// FIXME: diagnose if target does not support protected visibility
|
|
|
|
return ProtectedVisibility;
|
|
|
|
}
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< arg->getAsString(args) << value;
|
|
|
|
return DefaultVisibility;
|
|
|
|
}
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2017-12-07 21:46:26 +00:00
|
|
|
/// Check if input file kind and language standard are compatible.
|
|
|
|
static bool IsInputCompatibleWithStandard(InputKind IK,
|
|
|
|
const LangStandard &S) {
|
|
|
|
switch (IK.getLanguage()) {
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::Unknown:
|
|
|
|
case Language::LLVM_IR:
|
2017-12-07 21:46:26 +00:00
|
|
|
llvm_unreachable("should not parse language flags for this input");
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::C:
|
|
|
|
case Language::ObjC:
|
|
|
|
case Language::RenderScript:
|
|
|
|
return S.getLanguage() == Language::C;
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::OpenCL:
|
|
|
|
return S.getLanguage() == Language::OpenCL;
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::CXX:
|
|
|
|
case Language::ObjCXX:
|
|
|
|
return S.getLanguage() == Language::CXX;
|
2017-04-26 18:57:40 +00:00
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::CUDA:
|
2017-04-26 23:44:33 +00:00
|
|
|
// FIXME: What -std= values should be permitted for CUDA compilations?
|
2019-08-05 13:59:26 +00:00
|
|
|
return S.getLanguage() == Language::CUDA ||
|
|
|
|
S.getLanguage() == Language::CXX;
|
2017-04-26 18:57:40 +00:00
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::HIP:
|
|
|
|
return S.getLanguage() == Language::CXX || S.getLanguage() == Language::HIP;
|
2018-04-25 01:10:37 +00:00
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::Asm:
|
2017-04-26 18:57:40 +00:00
|
|
|
// Accept (and ignore) all -std= values.
|
|
|
|
// FIXME: The -std= value is not ignored; it affects the tokenization
|
|
|
|
// and preprocessing rules if we're preprocessing this asm input.
|
2017-02-14 23:41:38 +00:00
|
|
|
return true;
|
2017-02-14 22:44:20 +00:00
|
|
|
}
|
2017-04-26 18:57:40 +00:00
|
|
|
|
|
|
|
llvm_unreachable("unexpected input language");
|
2017-02-14 22:44:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Get language name for given input kind.
|
|
|
|
static const StringRef GetInputKindName(InputKind IK) {
|
2017-04-26 18:57:40 +00:00
|
|
|
switch (IK.getLanguage()) {
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::C:
|
2017-04-26 23:44:33 +00:00
|
|
|
return "C";
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::ObjC:
|
2017-04-26 23:44:33 +00:00
|
|
|
return "Objective-C";
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::CXX:
|
2017-04-26 23:44:33 +00:00
|
|
|
return "C++";
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::ObjCXX:
|
2017-04-26 23:44:33 +00:00
|
|
|
return "Objective-C++";
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::OpenCL:
|
2017-02-14 22:44:20 +00:00
|
|
|
return "OpenCL";
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::CUDA:
|
2017-02-14 22:44:20 +00:00
|
|
|
return "CUDA";
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::RenderScript:
|
2017-04-26 23:44:33 +00:00
|
|
|
return "RenderScript";
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::HIP:
|
2018-04-25 01:10:37 +00:00
|
|
|
return "HIP";
|
2017-04-26 23:44:33 +00:00
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::Asm:
|
2017-04-26 23:44:33 +00:00
|
|
|
return "Asm";
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::LLVM_IR:
|
2017-04-26 23:44:33 +00:00
|
|
|
return "LLVM IR";
|
|
|
|
|
2019-08-05 13:59:26 +00:00
|
|
|
case Language::Unknown:
|
2017-04-26 23:44:33 +00:00
|
|
|
break;
|
2017-02-14 22:44:20 +00:00
|
|
|
}
|
2017-04-26 23:44:33 +00:00
|
|
|
llvm_unreachable("unknown input language");
|
2017-02-14 22:44:20 +00:00
|
|
|
}
|
|
|
|
|
2010-12-04 01:50:27 +00:00
|
|
|
static void ParseLangArgs(LangOptions &Opts, ArgList &Args, InputKind IK,
|
2016-02-08 15:59:20 +00:00
|
|
|
const TargetOptions &TargetOpts,
|
2020-05-11 12:29:52 -07:00
|
|
|
PreprocessorOptions &PPOpts,
|
2011-09-25 23:23:43 +00:00
|
|
|
DiagnosticsEngine &Diags) {
|
2010-12-04 01:50:27 +00:00
|
|
|
// FIXME: Cleanup per-file based stuff.
|
|
|
|
LangStandard::Kind LangStd = LangStandard::lang_unspecified;
|
|
|
|
if (const Arg *A = Args.getLastArg(OPT_std_EQ)) {
|
2019-08-05 13:59:26 +00:00
|
|
|
LangStd = LangStandard::getLangKind(A->getValue());
|
2017-02-14 22:44:20 +00:00
|
|
|
if (LangStd == LangStandard::lang_unspecified) {
|
2010-12-04 01:50:27 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
2012-11-01 04:30:05 +00:00
|
|
|
<< A->getAsString(Args) << A->getValue();
|
2017-02-14 22:44:20 +00:00
|
|
|
// Report supported standards with short description.
|
|
|
|
for (unsigned KindValue = 0;
|
|
|
|
KindValue != LangStandard::lang_unspecified;
|
|
|
|
++KindValue) {
|
|
|
|
const LangStandard &Std = LangStandard::getLangStandardForKind(
|
|
|
|
static_cast<LangStandard::Kind>(KindValue));
|
|
|
|
if (IsInputCompatibleWithStandard(IK, Std)) {
|
2017-04-27 01:17:05 +00:00
|
|
|
auto Diag = Diags.Report(diag::note_drv_use_standard);
|
|
|
|
Diag << Std.getName() << Std.getDescription();
|
|
|
|
unsigned NumAliases = 0;
|
|
|
|
#define LANGSTANDARD(id, name, lang, desc, features)
|
|
|
|
#define LANGSTANDARD_ALIAS(id, alias) \
|
|
|
|
if (KindValue == LangStandard::lang_##id) ++NumAliases;
|
|
|
|
#define LANGSTANDARD_ALIAS_DEPR(id, alias)
|
2019-08-05 13:59:26 +00:00
|
|
|
#include "clang/Basic/LangStandards.def"
|
2017-04-27 01:17:05 +00:00
|
|
|
Diag << NumAliases;
|
|
|
|
#define LANGSTANDARD(id, name, lang, desc, features)
|
|
|
|
#define LANGSTANDARD_ALIAS(id, alias) \
|
|
|
|
if (KindValue == LangStandard::lang_##id) Diag << alias;
|
|
|
|
#define LANGSTANDARD_ALIAS_DEPR(id, alias)
|
2019-08-05 13:59:26 +00:00
|
|
|
#include "clang/Basic/LangStandards.def"
|
2017-02-14 22:44:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2013-12-05 16:25:25 +00:00
|
|
|
// Valid standard, check to make sure language and standard are
|
|
|
|
// compatible.
|
2011-05-02 19:24:53 +00:00
|
|
|
const LangStandard &Std = LangStandard::getLangStandardForKind(LangStd);
|
2017-02-14 22:44:20 +00:00
|
|
|
if (!IsInputCompatibleWithStandard(IK, Std)) {
|
|
|
|
Diags.Report(diag::err_drv_argument_not_allowed_with)
|
|
|
|
<< A->getAsString(Args) << GetInputKindName(IK);
|
2011-05-02 19:24:53 +00:00
|
|
|
}
|
|
|
|
}
|
2010-12-04 01:50:27 +00:00
|
|
|
}
|
|
|
|
|
2018-03-17 13:31:35 +00:00
|
|
|
if (const Arg *A = Args.getLastArg(OPT_fcf_protection_EQ)) {
|
|
|
|
StringRef Name = A->getValue();
|
|
|
|
if (Name == "full" || Name == "branch") {
|
|
|
|
Opts.CFProtectionBranch = 1;
|
|
|
|
}
|
|
|
|
}
|
2012-06-19 23:09:52 +00:00
|
|
|
// -cl-std only applies for OpenCL language standards.
|
|
|
|
// Override the -std option in this case.
|
2010-12-04 01:51:40 +00:00
|
|
|
if (const Arg *A = Args.getLastArg(OPT_cl_std_EQ)) {
|
2012-06-19 23:09:52 +00:00
|
|
|
LangStandard::Kind OpenCLLangStd
|
2017-02-14 22:44:20 +00:00
|
|
|
= llvm::StringSwitch<LangStandard::Kind>(A->getValue())
|
2017-04-27 01:17:05 +00:00
|
|
|
.Cases("cl", "CL", LangStandard::lang_opencl10)
|
2020-11-11 16:45:58 +03:00
|
|
|
.Cases("cl1.0", "CL1.0", LangStandard::lang_opencl10)
|
2017-02-14 22:44:20 +00:00
|
|
|
.Cases("cl1.1", "CL1.1", LangStandard::lang_opencl11)
|
|
|
|
.Cases("cl1.2", "CL1.2", LangStandard::lang_opencl12)
|
|
|
|
.Cases("cl2.0", "CL2.0", LangStandard::lang_opencl20)
|
2020-10-09 15:13:39 +01:00
|
|
|
.Cases("cl3.0", "CL3.0", LangStandard::lang_opencl30)
|
2019-07-25 11:04:29 +00:00
|
|
|
.Cases("clc++", "CLC++", LangStandard::lang_openclcpp)
|
2017-02-14 22:44:20 +00:00
|
|
|
.Default(LangStandard::lang_unspecified);
|
2015-08-05 15:08:53 +00:00
|
|
|
|
2012-06-19 23:09:52 +00:00
|
|
|
if (OpenCLLangStd == LangStandard::lang_unspecified) {
|
2010-12-04 01:51:40 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
2017-02-14 22:44:20 +00:00
|
|
|
<< A->getAsString(Args) << A->getValue();
|
2010-12-04 01:51:40 +00:00
|
|
|
}
|
2012-06-19 23:09:52 +00:00
|
|
|
else
|
|
|
|
LangStd = OpenCLLangStd;
|
2010-12-04 01:51:40 +00:00
|
|
|
}
|
2015-08-05 15:08:53 +00:00
|
|
|
|
2019-07-26 16:21:01 +01:00
|
|
|
Opts.SYCLIsDevice = Opts.SYCL && Args.hasArg(options::OPT_fsycl_is_device);
|
|
|
|
|
2016-04-27 19:53:03 +00:00
|
|
|
llvm::Triple T(TargetOpts.Triple);
|
2016-06-20 19:26:00 +00:00
|
|
|
CompilerInvocation::setLangDefaults(Opts, IK, T, PPOpts, LangStd);
|
2010-12-04 01:50:27 +00:00
|
|
|
|
2016-07-13 21:21:05 +00:00
|
|
|
// -cl-strict-aliasing needs to emit diagnostic in the case where CL > 1.0.
|
|
|
|
// This option should be deprecated for CL > 1.0 because
|
|
|
|
// this option was added for compatibility with OpenCL 1.0.
|
|
|
|
if (Args.getLastArg(OPT_cl_strict_aliasing)
|
|
|
|
&& Opts.OpenCLVersion > 100) {
|
|
|
|
Diags.Report(diag::warn_option_invalid_ocl_version)
|
2018-05-08 13:47:43 +00:00
|
|
|
<< Opts.getOpenCLVersionTuple().getAsString()
|
|
|
|
<< Args.getLastArg(OPT_cl_strict_aliasing)->getAsString(Args);
|
2016-07-13 21:21:05 +00:00
|
|
|
}
|
|
|
|
|
2010-04-17 20:17:31 +00:00
|
|
|
// We abuse '-f[no-]gnu-keywords' to force overriding all GNU-extension
|
|
|
|
// keywords. This behavior is provided by GCC's poorly named '-fasm' flag,
|
|
|
|
// while a subset (the non-C++ GNU keywords) is provided by GCC's
|
|
|
|
// '-fgnu-keywords'. Clang conflates the two for simplicity under the single
|
|
|
|
// name, as it doesn't seem a useful distinction.
|
|
|
|
Opts.GNUKeywords = Args.hasFlag(OPT_fgnu_keywords, OPT_fno_gnu_keywords,
|
2010-12-04 01:50:27 +00:00
|
|
|
Opts.GNUKeywords);
|
2010-04-17 20:17:31 +00:00
|
|
|
|
2018-07-17 04:56:22 +00:00
|
|
|
Opts.Digraphs = Args.hasFlag(OPT_fdigraphs, OPT_fno_digraphs, Opts.Digraphs);
|
|
|
|
|
2017-05-25 16:24:49 +00:00
|
|
|
if (Args.hasArg(OPT_fno_operator_names))
|
2010-12-04 01:50:27 +00:00
|
|
|
Opts.CXXOperatorNames = 0;
|
2009-12-01 03:16:53 +00:00
|
|
|
|
2016-05-23 20:19:56 +00:00
|
|
|
if (Opts.CUDAIsDevice && Args.hasArg(OPT_fcuda_approx_transcendentals))
|
|
|
|
Opts.CUDADeviceApproxTranscendentals = 1;
|
|
|
|
|
2019-10-22 13:41:25 -04:00
|
|
|
if (Args.hasArg(OPT_fgpu_allow_device_init)) {
|
|
|
|
if (Opts.HIP)
|
|
|
|
Opts.GPUAllowDeviceInit = 1;
|
|
|
|
else
|
2019-10-22 16:15:04 -04:00
|
|
|
Diags.Report(diag::warn_ignored_hip_only_option)
|
2019-10-22 13:41:25 -04:00
|
|
|
<< Args.getLastArg(OPT_fgpu_allow_device_init)->getAsString(Args);
|
|
|
|
}
|
2019-12-09 14:55:34 -05:00
|
|
|
if (Opts.HIP)
|
|
|
|
Opts.GPUMaxThreadsPerBlock = getLastArgIntValue(
|
|
|
|
Args, OPT_gpu_max_threads_per_block_EQ, Opts.GPUMaxThreadsPerBlock);
|
|
|
|
else if (Args.hasArg(OPT_gpu_max_threads_per_block_EQ))
|
|
|
|
Diags.Report(diag::warn_ignored_hip_only_option)
|
|
|
|
<< Args.getLastArg(OPT_gpu_max_threads_per_block_EQ)->getAsString(Args);
|
2018-02-12 10:46:45 +00:00
|
|
|
|
2018-10-30 20:31:30 +00:00
|
|
|
if (Opts.ObjC) {
|
2012-06-20 06:18:46 +00:00
|
|
|
if (Arg *arg = Args.getLastArg(OPT_fobjc_runtime_EQ)) {
|
2012-11-01 04:30:05 +00:00
|
|
|
StringRef value = arg->getValue();
|
2012-06-20 06:18:46 +00:00
|
|
|
if (Opts.ObjCRuntime.tryParse(value))
|
|
|
|
Diags.Report(diag::err_drv_unknown_objc_runtime) << value;
|
|
|
|
}
|
|
|
|
|
2011-06-15 23:02:42 +00:00
|
|
|
if (Args.hasArg(OPT_fobjc_gc_only))
|
2011-09-13 17:21:33 +00:00
|
|
|
Opts.setGC(LangOptions::GCOnly);
|
2011-06-15 23:02:42 +00:00
|
|
|
else if (Args.hasArg(OPT_fobjc_gc))
|
2011-09-13 17:21:33 +00:00
|
|
|
Opts.setGC(LangOptions::HybridGC);
|
2011-06-15 23:02:42 +00:00
|
|
|
else if (Args.hasArg(OPT_fobjc_arc)) {
|
|
|
|
Opts.ObjCAutoRefCount = 1;
|
2012-08-21 02:47:43 +00:00
|
|
|
if (!Opts.ObjCRuntime.allowsARC())
|
|
|
|
Diags.Report(diag::err_arc_unsupported_on_runtime);
|
Define weak and __weak to mean ARC-style weak references, even in MRC.
Previously, __weak was silently accepted and ignored in MRC mode.
That makes this a potentially source-breaking change that we have to
roll out cautiously. Accordingly, for the time being, actual support
for __weak references in MRC is experimental, and the compiler will
reject attempts to actually form such references. The intent is to
eventually enable the feature by default in all non-GC modes.
(It is, of course, incompatible with ObjC GC's interpretation of
__weak.)
If you like, you can enable this feature with
-Xclang -fobjc-weak
but like any -Xclang option, this option may be removed at any point,
e.g. if/when it is eventually enabled by default.
This patch also enables the use of the ARC __unsafe_unretained qualifier
in MRC. Unlike __weak, this is being enabled immediately. Since
variables are essentially __unsafe_unretained by default in MRC,
the only practical uses are (1) communication and (2) changing the
default behavior of by-value block capture.
As an implementation matter, this means that the ObjC ownership
qualifiers may appear in any ObjC language mode, and so this patch
removes a number of checks for getLangOpts().ObjCAutoRefCount
that were guarding the processing of these qualifiers. I don't
expect this to be a significant drain on performance; it may even
be faster to just check for these qualifiers directly on a type
(since it's probably in a register anyway) than to do N dependent
loads to grab the LangOptions.
rdar://9674298
llvm-svn: 251041
2015-10-22 18:38:17 +00:00
|
|
|
}
|
2012-08-21 02:47:43 +00:00
|
|
|
|
Define weak and __weak to mean ARC-style weak references, even in MRC.
Previously, __weak was silently accepted and ignored in MRC mode.
That makes this a potentially source-breaking change that we have to
roll out cautiously. Accordingly, for the time being, actual support
for __weak references in MRC is experimental, and the compiler will
reject attempts to actually form such references. The intent is to
eventually enable the feature by default in all non-GC modes.
(It is, of course, incompatible with ObjC GC's interpretation of
__weak.)
If you like, you can enable this feature with
-Xclang -fobjc-weak
but like any -Xclang option, this option may be removed at any point,
e.g. if/when it is eventually enabled by default.
This patch also enables the use of the ARC __unsafe_unretained qualifier
in MRC. Unlike __weak, this is being enabled immediately. Since
variables are essentially __unsafe_unretained by default in MRC,
the only practical uses are (1) communication and (2) changing the
default behavior of by-value block capture.
As an implementation matter, this means that the ObjC ownership
qualifiers may appear in any ObjC language mode, and so this patch
removes a number of checks for getLangOpts().ObjCAutoRefCount
that were guarding the processing of these qualifiers. I don't
expect this to be a significant drain on performance; it may even
be faster to just check for these qualifiers directly on a type
(since it's probably in a register anyway) than to do N dependent
loads to grab the LangOptions.
rdar://9674298
llvm-svn: 251041
2015-10-22 18:38:17 +00:00
|
|
|
// ObjCWeakRuntime tracks whether the runtime supports __weak, not
|
|
|
|
// whether the feature is actually enabled. This is predominantly
|
|
|
|
// determined by -fobjc-runtime, but we allow it to be overridden
|
|
|
|
// from the command line for testing purposes.
|
|
|
|
if (Args.hasArg(OPT_fobjc_runtime_has_weak))
|
|
|
|
Opts.ObjCWeakRuntime = 1;
|
|
|
|
else
|
|
|
|
Opts.ObjCWeakRuntime = Opts.ObjCRuntime.allowsWeak();
|
|
|
|
|
|
|
|
// ObjCWeak determines whether __weak is actually enabled.
|
2015-11-05 19:19:56 +00:00
|
|
|
// Note that we allow -fno-objc-weak to disable this even in ARC mode.
|
|
|
|
if (auto weakArg = Args.getLastArg(OPT_fobjc_weak, OPT_fno_objc_weak)) {
|
|
|
|
if (!weakArg->getOption().matches(OPT_fobjc_weak)) {
|
|
|
|
assert(!Opts.ObjCWeak);
|
|
|
|
} else if (Opts.getGC() != LangOptions::NonGC) {
|
Define weak and __weak to mean ARC-style weak references, even in MRC.
Previously, __weak was silently accepted and ignored in MRC mode.
That makes this a potentially source-breaking change that we have to
roll out cautiously. Accordingly, for the time being, actual support
for __weak references in MRC is experimental, and the compiler will
reject attempts to actually form such references. The intent is to
eventually enable the feature by default in all non-GC modes.
(It is, of course, incompatible with ObjC GC's interpretation of
__weak.)
If you like, you can enable this feature with
-Xclang -fobjc-weak
but like any -Xclang option, this option may be removed at any point,
e.g. if/when it is eventually enabled by default.
This patch also enables the use of the ARC __unsafe_unretained qualifier
in MRC. Unlike __weak, this is being enabled immediately. Since
variables are essentially __unsafe_unretained by default in MRC,
the only practical uses are (1) communication and (2) changing the
default behavior of by-value block capture.
As an implementation matter, this means that the ObjC ownership
qualifiers may appear in any ObjC language mode, and so this patch
removes a number of checks for getLangOpts().ObjCAutoRefCount
that were guarding the processing of these qualifiers. I don't
expect this to be a significant drain on performance; it may even
be faster to just check for these qualifiers directly on a type
(since it's probably in a register anyway) than to do N dependent
loads to grab the LangOptions.
rdar://9674298
llvm-svn: 251041
2015-10-22 18:38:17 +00:00
|
|
|
Diags.Report(diag::err_objc_weak_with_gc);
|
2015-11-05 19:19:56 +00:00
|
|
|
} else if (!Opts.ObjCWeakRuntime) {
|
Define weak and __weak to mean ARC-style weak references, even in MRC.
Previously, __weak was silently accepted and ignored in MRC mode.
That makes this a potentially source-breaking change that we have to
roll out cautiously. Accordingly, for the time being, actual support
for __weak references in MRC is experimental, and the compiler will
reject attempts to actually form such references. The intent is to
eventually enable the feature by default in all non-GC modes.
(It is, of course, incompatible with ObjC GC's interpretation of
__weak.)
If you like, you can enable this feature with
-Xclang -fobjc-weak
but like any -Xclang option, this option may be removed at any point,
e.g. if/when it is eventually enabled by default.
This patch also enables the use of the ARC __unsafe_unretained qualifier
in MRC. Unlike __weak, this is being enabled immediately. Since
variables are essentially __unsafe_unretained by default in MRC,
the only practical uses are (1) communication and (2) changing the
default behavior of by-value block capture.
As an implementation matter, this means that the ObjC ownership
qualifiers may appear in any ObjC language mode, and so this patch
removes a number of checks for getLangOpts().ObjCAutoRefCount
that were guarding the processing of these qualifiers. I don't
expect this to be a significant drain on performance; it may even
be faster to just check for these qualifiers directly on a type
(since it's probably in a register anyway) than to do N dependent
loads to grab the LangOptions.
rdar://9674298
llvm-svn: 251041
2015-10-22 18:38:17 +00:00
|
|
|
Diags.Report(diag::err_objc_weak_unsupported);
|
2015-11-05 19:19:56 +00:00
|
|
|
} else {
|
|
|
|
Opts.ObjCWeak = 1;
|
Define weak and __weak to mean ARC-style weak references, even in MRC.
Previously, __weak was silently accepted and ignored in MRC mode.
That makes this a potentially source-breaking change that we have to
roll out cautiously. Accordingly, for the time being, actual support
for __weak references in MRC is experimental, and the compiler will
reject attempts to actually form such references. The intent is to
eventually enable the feature by default in all non-GC modes.
(It is, of course, incompatible with ObjC GC's interpretation of
__weak.)
If you like, you can enable this feature with
-Xclang -fobjc-weak
but like any -Xclang option, this option may be removed at any point,
e.g. if/when it is eventually enabled by default.
This patch also enables the use of the ARC __unsafe_unretained qualifier
in MRC. Unlike __weak, this is being enabled immediately. Since
variables are essentially __unsafe_unretained by default in MRC,
the only practical uses are (1) communication and (2) changing the
default behavior of by-value block capture.
As an implementation matter, this means that the ObjC ownership
qualifiers may appear in any ObjC language mode, and so this patch
removes a number of checks for getLangOpts().ObjCAutoRefCount
that were guarding the processing of these qualifiers. I don't
expect this to be a significant drain on performance; it may even
be faster to just check for these qualifiers directly on a type
(since it's probably in a register anyway) than to do N dependent
loads to grab the LangOptions.
rdar://9674298
llvm-svn: 251041
2015-10-22 18:38:17 +00:00
|
|
|
}
|
2015-11-05 19:19:56 +00:00
|
|
|
} else if (Opts.ObjCAutoRefCount) {
|
|
|
|
Opts.ObjCWeak = Opts.ObjCWeakRuntime;
|
2011-06-15 23:02:42 +00:00
|
|
|
}
|
|
|
|
|
2013-11-01 21:58:17 +00:00
|
|
|
if (Args.hasArg(OPT_fobjc_subscripting_legacy_runtime))
|
|
|
|
Opts.ObjCSubscriptingLegacyRuntime =
|
|
|
|
(Opts.ObjCRuntime.getKind() == ObjCRuntime::FragileMacOSX);
|
2011-06-15 23:02:42 +00:00
|
|
|
}
|
2015-08-05 15:08:53 +00:00
|
|
|
|
Add -fgnuc-version= to control __GNUC__ and other GCC macros
I noticed that compiling on Windows with -fno-ms-compatibility had the
side effect of defining __GNUC__, along with __GNUG__, __GXX_RTTI__, and
a number of other macros for GCC compatibility. This is undesirable and
causes Chromium to do things like mix __attribute__ and __declspec,
which doesn't work. We should have a positive language option to enable
GCC compatibility features so that we can experiment with
-fno-ms-compatibility on Windows. This change adds -fgnuc-version= to be
that option.
My issue aside, users have, for a long time, reported that __GNUC__
doesn't match their expectations in one way or another. We have
encouraged users to migrate code away from this macro, but new code
continues to be written assuming a GCC-only environment. There's really
nothing we can do to stop that. By adding this flag, we can allow them
to choose their own adventure with __GNUC__.
This overlaps a bit with the "GNUMode" language option from -std=gnu*.
The gnu language mode tends to enable non-conforming behaviors that we'd
rather not enable by default, but the we want to set things like
__GXX_RTTI__ by default, so I've kept these separate.
Helps address PR42817
Reviewed By: hans, nickdesaulniers, MaskRay
Differential Revision: https://reviews.llvm.org/D68055
llvm-svn: 374449
2019-10-10 21:04:25 +00:00
|
|
|
if (Arg *A = Args.getLastArg(options::OPT_fgnuc_version_EQ)) {
|
|
|
|
// Check that the version has 1 to 3 components and the minor and patch
|
|
|
|
// versions fit in two decimal digits.
|
|
|
|
VersionTuple GNUCVer;
|
|
|
|
bool Invalid = GNUCVer.tryParse(A->getValue());
|
|
|
|
unsigned Major = GNUCVer.getMajor();
|
|
|
|
unsigned Minor = GNUCVer.getMinor().getValueOr(0);
|
|
|
|
unsigned Patch = GNUCVer.getSubminor().getValueOr(0);
|
|
|
|
if (Invalid || GNUCVer.getBuild() || Minor >= 100 || Patch >= 100) {
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< A->getAsString(Args) << A->getValue();
|
|
|
|
}
|
|
|
|
Opts.GNUCVersion = Major * 100 * 100 + Minor * 100 + Patch;
|
|
|
|
}
|
|
|
|
|
2015-05-13 22:07:22 +00:00
|
|
|
if (Args.hasArg(OPT_fgnu89_inline)) {
|
|
|
|
if (Opts.CPlusPlus)
|
2017-02-14 22:44:20 +00:00
|
|
|
Diags.Report(diag::err_drv_argument_not_allowed_with)
|
|
|
|
<< "-fgnu89-inline" << GetInputKindName(IK);
|
2015-05-13 22:07:22 +00:00
|
|
|
else
|
|
|
|
Opts.GNUInline = 1;
|
|
|
|
}
|
2011-06-02 16:13:27 +00:00
|
|
|
|
2013-02-19 01:57:35 +00:00
|
|
|
// The type-visibility mode defaults to the value-visibility mode.
|
|
|
|
if (Arg *typeVisOpt = Args.getLastArg(OPT_ftype_visibility)) {
|
|
|
|
Opts.setTypeVisibilityMode(parseVisibility(typeVisOpt, Args, Diags));
|
|
|
|
} else {
|
|
|
|
Opts.setTypeVisibilityMode(Opts.getValueVisibilityMode());
|
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
|
2020-11-02 23:24:04 +00:00
|
|
|
if (Args.hasArg(OPT_fvisibility_from_dllstorageclass)) {
|
|
|
|
Opts.VisibilityFromDLLStorageClass = 1;
|
|
|
|
|
|
|
|
// Translate dllexport defintions to default visibility, by default.
|
|
|
|
if (Arg *O = Args.getLastArg(OPT_fvisibility_dllexport_EQ))
|
|
|
|
Opts.setDLLExportVisibility(parseVisibility(O, Args, Diags));
|
|
|
|
else
|
|
|
|
Opts.setDLLExportVisibility(DefaultVisibility);
|
|
|
|
|
|
|
|
// Translate defintions without an explict DLL storage class to hidden
|
|
|
|
// visibility, by default.
|
|
|
|
if (Arg *O = Args.getLastArg(OPT_fvisibility_nodllstorageclass_EQ))
|
|
|
|
Opts.setNoDLLStorageClassVisibility(parseVisibility(O, Args, Diags));
|
|
|
|
else
|
|
|
|
Opts.setNoDLLStorageClassVisibility(HiddenVisibility);
|
|
|
|
|
|
|
|
// Translate dllimport external declarations to default visibility, by
|
|
|
|
// default.
|
|
|
|
if (Arg *O = Args.getLastArg(OPT_fvisibility_externs_dllimport_EQ))
|
|
|
|
Opts.setExternDeclDLLImportVisibility(parseVisibility(O, Args, Diags));
|
|
|
|
else
|
|
|
|
Opts.setExternDeclDLLImportVisibility(DefaultVisibility);
|
|
|
|
|
|
|
|
// Translate external declarations without an explicit DLL storage class
|
|
|
|
// to hidden visibility, by default.
|
|
|
|
if (Arg *O = Args.getLastArg(OPT_fvisibility_externs_nodllstorageclass_EQ))
|
|
|
|
Opts.setExternDeclNoDLLStorageClassVisibility(
|
|
|
|
parseVisibility(O, Args, Diags));
|
|
|
|
else
|
|
|
|
Opts.setExternDeclNoDLLStorageClassVisibility(HiddenVisibility);
|
|
|
|
}
|
|
|
|
|
2010-09-17 18:29:54 +00:00
|
|
|
if (Args.hasArg(OPT_ftrapv)) {
|
2010-10-21 03:16:25 +00:00
|
|
|
Opts.setSignedOverflowBehavior(LangOptions::SOB_Trapping);
|
2010-09-17 18:29:54 +00:00
|
|
|
// Set the handler, if one is specified.
|
|
|
|
Opts.OverflowHandler =
|
2020-01-28 20:23:46 +01:00
|
|
|
std::string(Args.getLastArgValue(OPT_ftrapv_handler));
|
2010-09-17 18:29:54 +00:00
|
|
|
}
|
2010-06-26 21:25:03 +00:00
|
|
|
else if (Args.hasArg(OPT_fwrapv))
|
2010-10-21 03:16:25 +00:00
|
|
|
Opts.setSignedOverflowBehavior(LangOptions::SOB_Defined);
|
2009-12-01 03:16:53 +00:00
|
|
|
|
2014-12-22 18:35:03 +00:00
|
|
|
Opts.MicrosoftExt = Opts.MSVCCompat || Args.hasArg(OPT_fms_extensions);
|
|
|
|
Opts.AsmBlocks = Args.hasArg(OPT_fasm_blocks) || Opts.MicrosoftExt;
|
2015-03-22 08:39:22 +00:00
|
|
|
Opts.MSCompatibilityVersion = 0;
|
|
|
|
if (const Arg *A = Args.getLastArg(OPT_fms_compatibility_version)) {
|
|
|
|
VersionTuple VT;
|
|
|
|
if (VT.tryParse(A->getValue()))
|
|
|
|
Diags.Report(diag::err_drv_invalid_value) << A->getAsString(Args)
|
|
|
|
<< A->getValue();
|
|
|
|
Opts.MSCompatibilityVersion = VT.getMajor() * 10000000 +
|
|
|
|
VT.getMinor().getValueOr(0) * 100000 +
|
|
|
|
VT.getSubminor().getValueOr(0);
|
|
|
|
}
|
2014-12-22 18:35:03 +00:00
|
|
|
|
Misc typos fixes in ./lib folder
Summary: Found via `codespell -q 3 -I ../clang-whitelist.txt -L uint,importd,crasher,gonna,cant,ue,ons,orign,ned`
Reviewers: teemperor
Reviewed By: teemperor
Subscribers: teemperor, jholewinski, jvesely, nhaehnle, whisperity, jfb, cfe-commits
Differential Revision: https://reviews.llvm.org/D55475
llvm-svn: 348755
2018-12-10 12:37:46 +00:00
|
|
|
// Mimicking gcc's behavior, trigraphs are only enabled if -trigraphs
|
2014-12-22 18:35:03 +00:00
|
|
|
// is specified, or -std is set to a conforming mode.
|
|
|
|
// Trigraphs are disabled by default in c++1z onwards.
|
2020-08-13 14:33:04 -04:00
|
|
|
// For z/OS, trigraphs are enabled by default (without regard to the above).
|
|
|
|
Opts.Trigraphs =
|
|
|
|
(!Opts.GNUMode && !Opts.MSVCCompat && !Opts.CPlusPlus17) || T.isOSzOS();
|
2014-12-23 22:32:37 +00:00
|
|
|
Opts.Trigraphs =
|
|
|
|
Args.hasFlag(OPT_ftrigraphs, OPT_fno_trigraphs, Opts.Trigraphs);
|
2009-12-01 03:16:53 +00:00
|
|
|
|
2009-12-16 20:10:18 +00:00
|
|
|
Opts.DollarIdents = Args.hasFlag(OPT_fdollars_in_identifiers,
|
|
|
|
OPT_fno_dollars_in_identifiers,
|
2010-12-04 01:50:27 +00:00
|
|
|
Opts.DollarIdents);
|
2017-11-29 07:25:12 +00:00
|
|
|
|
2018-06-04 16:07:52 +00:00
|
|
|
// -ffixed-point
|
|
|
|
Opts.FixedPoint =
|
|
|
|
Args.hasFlag(OPT_ffixed_point, OPT_fno_fixed_point, /*Default=*/false) &&
|
|
|
|
!Opts.CPlusPlus;
|
2018-06-29 17:08:19 +00:00
|
|
|
Opts.PaddingOnUnsignedFixedPoint =
|
|
|
|
Args.hasFlag(OPT_fpadding_on_unsigned_fixed_point,
|
|
|
|
OPT_fno_padding_on_unsigned_fixed_point,
|
2018-06-20 17:19:40 +00:00
|
|
|
/*Default=*/false) &&
|
|
|
|
Opts.FixedPoint;
|
2018-06-04 16:07:52 +00:00
|
|
|
|
2016-01-26 20:15:02 +00:00
|
|
|
Opts.RTTI = Opts.CPlusPlus && !Args.hasArg(OPT_fno_rtti);
|
2014-07-01 22:24:56 +00:00
|
|
|
Opts.RTTIData = Opts.RTTI && !Args.hasArg(OPT_fno_rtti_data);
|
2016-06-14 21:43:01 +00:00
|
|
|
Opts.Blocks = Args.hasArg(OPT_fblocks) || (Opts.OpenCL
|
2018-04-12 14:17:04 +00:00
|
|
|
&& Opts.OpenCLVersion == 200);
|
2020-04-21 15:37:19 -04:00
|
|
|
Opts.Coroutines = Opts.CPlusPlus20 || Args.hasArg(OPT_fcoroutines_ts);
|
2017-12-07 23:04:11 +00:00
|
|
|
|
2019-10-26 17:15:10 -07:00
|
|
|
Opts.ConvergentFunctions = Opts.OpenCL || (Opts.CUDA && Opts.CUDAIsDevice) ||
|
2020-08-25 17:05:19 +03:00
|
|
|
Opts.SYCLIsDevice ||
|
|
|
|
Args.hasArg(OPT_fconvergent_functions);
|
2019-10-26 17:15:10 -07:00
|
|
|
|
2017-10-15 15:01:42 +00:00
|
|
|
Opts.DoubleSquareBracketAttributes =
|
|
|
|
Args.hasFlag(OPT_fdouble_square_bracket_attributes,
|
2019-05-14 12:09:55 +00:00
|
|
|
OPT_fno_double_square_bracket_attributes,
|
|
|
|
Opts.DoubleSquareBracketAttributes);
|
2017-10-15 15:01:42 +00:00
|
|
|
|
2020-04-21 15:37:19 -04:00
|
|
|
Opts.CPlusPlusModules = Opts.CPlusPlus20;
|
2019-04-11 21:18:23 +00:00
|
|
|
Opts.Modules =
|
|
|
|
Args.hasArg(OPT_fmodules) || Opts.ModulesTS || Opts.CPlusPlusModules;
|
2014-04-11 11:47:45 +00:00
|
|
|
Opts.ModulesDeclUse =
|
|
|
|
Args.hasArg(OPT_fmodules_decluse) || Opts.ModulesStrictDeclUse;
|
2019-04-11 21:18:23 +00:00
|
|
|
// FIXME: We only need this in C++ modules / Modules TS if we might textually
|
|
|
|
// enter a different module (eg, when building a header unit).
|
2015-05-01 21:22:17 +00:00
|
|
|
Opts.ModulesLocalVisibility =
|
2019-04-11 21:18:23 +00:00
|
|
|
Args.hasArg(OPT_fmodules_local_submodule_visibility) || Opts.ModulesTS ||
|
|
|
|
Opts.CPlusPlusModules;
|
2014-04-23 12:57:01 +00:00
|
|
|
Opts.ModulesSearchAll = Opts.Modules &&
|
|
|
|
!Args.hasArg(OPT_fno_modules_search_all) &&
|
|
|
|
Args.hasArg(OPT_fmodules_search_all);
|
2013-09-09 09:17:24 +00:00
|
|
|
Opts.CharIsSigned = Opts.OpenCL || !Args.hasArg(OPT_fno_signed_char);
|
2012-09-05 17:30:57 +00:00
|
|
|
Opts.WChar = Opts.CPlusPlus && !Args.hasArg(OPT_fno_wchar);
|
2020-04-21 15:37:19 -04:00
|
|
|
Opts.Char8 = Args.hasFlag(OPT_fchar8__t, OPT_fno_char8__t, Opts.CPlusPlus20);
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.NoBuiltin = Args.hasArg(OPT_fno_builtin) || Opts.Freestanding;
|
2016-01-06 14:35:46 +00:00
|
|
|
if (!Opts.NoBuiltin)
|
|
|
|
getAllNoBuiltinFuncValues(Args, Opts.NoBuiltinFuncs);
|
2016-09-30 22:41:36 +00:00
|
|
|
Opts.AlignedAllocation =
|
|
|
|
Args.hasFlag(OPT_faligned_allocation, OPT_fno_aligned_allocation,
|
|
|
|
Opts.AlignedAllocation);
|
2017-06-29 18:48:40 +00:00
|
|
|
Opts.AlignedAllocationUnavailable =
|
|
|
|
Opts.AlignedAllocation && Args.hasArg(OPT_aligned_alloc_unavailable);
|
2020-01-24 00:43:22 +02:00
|
|
|
if (Args.hasArg(OPT_fconcepts_ts))
|
|
|
|
Diags.Report(diag::warn_fe_concepts_ts_flag);
|
2013-09-09 09:17:24 +00:00
|
|
|
Opts.MathErrno = !Opts.OpenCL && Args.hasArg(OPT_fmath_errno);
|
2019-07-12 02:32:15 +00:00
|
|
|
Opts.LongDoubleSize = Args.hasArg(OPT_mlong_double_128)
|
|
|
|
? 128
|
|
|
|
: Args.hasArg(OPT_mlong_double_64) ? 64 : 0;
|
2020-11-24 18:11:46 -05:00
|
|
|
Opts.EnableAIXExtendedAltivecABI = Args.hasArg(OPT_mabi_EQ_vec_extabi);
|
2013-06-14 17:17:23 +00:00
|
|
|
Opts.PICLevel = getLastArgIntValue(Args, OPT_pic_level, 0, Diags);
|
2015-08-05 15:08:53 +00:00
|
|
|
Opts.DumpRecordLayouts = Opts.DumpRecordLayoutsSimple
|
Extend the ExternalASTSource interface to allow the AST source to
provide the layout of records, rather than letting Clang compute
the layout itself. LLDB provides the motivation for this feature:
because various layout-altering attributes (packed, aligned, etc.)
don't get reliably get placed into DWARF, the record layouts computed
by LLDB from the reconstructed records differ from the actual layouts,
and badness occurs. This interface lets the DWARF data drive layout,
so we don't need the attributes preserved to get the answer write.
The testing methodology for this change is fun. I've introduced a
variant of -fdump-record-layouts called -fdump-record-layouts-simple
that always has the simple C format and provides size/alignment/field
offsets. There is also a -cc1 option -foverride-record-layout=<file>
to take the output of -fdump-record-layouts-simple and parse it to
produce a set of overridden layouts, which is introduced into the AST
via a testing-only ExternalASTSource (called
LayoutOverrideSource). Each test contains a number of records to lay
out, which use various layout-changing attributes, and then dumps the
layouts. We then run the test again, using the preprocessor to
eliminate the layout-changing attributes entirely (which would give us
different layouts for the records), but supplying the
previously-computed record layouts. Finally, we diff the layouts
produced from the two runs to be sure that they are identical.
Note that this code makes the assumption that we don't *have* to
provide the offsets of bases or virtual bases to get the layout right,
because the alignment attributes don't affect it. I believe this
assumption holds, but if it does not, we can extend
LayoutOverrideSource to also provide base offset information.
Fixes the Clang side of <rdar://problem/10169539>.
llvm-svn: 149055
2012-01-26 07:55:45 +00:00
|
|
|
|| Args.hasArg(OPT_fdump_record_layouts);
|
2020-05-15 07:30:49 -07:00
|
|
|
if (Opts.FastRelaxedMath)
|
|
|
|
Opts.setDefaultFPContractMode(LangOptions::FPM_Fast);
|
2018-09-30 21:41:11 +00:00
|
|
|
llvm::sort(Opts.ModuleFeatures);
|
2015-05-14 23:44:18 +00:00
|
|
|
Opts.NativeHalfType |= Args.hasArg(OPT_fnative_half_type);
|
2016-03-18 16:58:36 +00:00
|
|
|
Opts.NativeHalfArgsAndReturns |= Args.hasArg(OPT_fnative_half_arguments_and_returns);
|
|
|
|
// Enable HalfArgsAndReturns if present in Args or if NativeHalfArgsAndReturns
|
|
|
|
// is enabled.
|
|
|
|
Opts.HalfArgsAndReturns = Args.hasArg(OPT_fallow_half_arguments_and_returns)
|
|
|
|
| Opts.NativeHalfArgsAndReturns;
|
2014-07-23 15:30:23 +00:00
|
|
|
|
[Sema][AArch64] Add parsing support for arm_sve_vector_bits attribute
Summary:
This patch implements parsing support for the 'arm_sve_vector_bits' type
attribute, defined by the Arm C Language Extensions (ACLE, version 00bet5,
section 3.7.3) for SVE [1].
The purpose of this attribute is to define fixed-length (VLST) versions
of existing sizeless types (VLAT). For example:
#if __ARM_FEATURE_SVE_BITS==512
typedef svint32_t fixed_svint32_t __attribute__((arm_sve_vector_bits(512)));
#endif
Creates a type 'fixed_svint32_t' that is a fixed-length version of
'svint32_t' that is normal-sized (rather than sizeless) and contains
exactly 512 bits. Unlike 'svint32_t', this type can be used in places
such as structs and arrays where sizeless types can't.
Implemented in this patch is the following:
* Defined and tested attribute taking single argument.
* Checks the argument is an integer constant expression.
* Attribute can only be attached to a single SVE vector or predicate
type, excluding tuple types such as svint32x4_t.
* Added the `-msve-vector-bits=<bits>` flag. When specified the
`__ARM_FEATURE_SVE_BITS__EXPERIMENTAL` macro is defined.
* Added a language option to store the vector size specified by the
`-msve-vector-bits=<bits>` flag. This is used to validate `N ==
__ARM_FEATURE_SVE_BITS`, where N is the number of bits passed to the
attribute and `__ARM_FEATURE_SVE_BITS` is the feature macro defined under
the same flag.
The `__ARM_FEATURE_SVE_BITS` macro will be made non-experimental in the final
patch of the series.
[1] https://developer.arm.com/documentation/100987/latest
This is patch 1/4 of a patch series.
Reviewers: sdesmalen, rsandifo-arm, efriedma, ctetreau, cameron.mcinally, rengolin, aaron.ballman
Reviewed By: sdesmalen, aaron.ballman
Differential Revision: https://reviews.llvm.org/D83550
2020-05-29 09:17:37 +00:00
|
|
|
Opts.ArmSveVectorBits =
|
|
|
|
getLastArgIntValue(Args, options::OPT_msve_vector_bits_EQ, 0, Diags);
|
|
|
|
|
2015-10-04 17:51:05 +00:00
|
|
|
// __declspec is enabled by default for the PS4 by the driver, and also
|
|
|
|
// enabled for Microsoft Extensions or Borland Extensions, here.
|
|
|
|
//
|
|
|
|
// FIXME: __declspec is also currently enabled for CUDA, but isn't really a
|
2016-10-08 22:16:08 +00:00
|
|
|
// CUDA extension. However, it is required for supporting
|
|
|
|
// __clang_cuda_builtin_vars.h, which uses __declspec(property). Once that has
|
|
|
|
// been rewritten in terms of something more generic, remove the Opts.CUDA
|
|
|
|
// term here.
|
2015-10-04 17:51:05 +00:00
|
|
|
Opts.DeclSpecKeyword =
|
|
|
|
Args.hasFlag(OPT_fdeclspec, OPT_fno_declspec,
|
|
|
|
(Opts.MicrosoftExt || Opts.Borland || Opts.CUDA));
|
|
|
|
|
2016-05-18 09:06:38 +00:00
|
|
|
// -mrtd option
|
|
|
|
if (Arg *A = Args.getLastArg(OPT_mrtd)) {
|
|
|
|
if (Opts.getDefaultCallingConv() != LangOptions::DCC_None)
|
|
|
|
Diags.Report(diag::err_drv_argument_not_allowed_with)
|
|
|
|
<< A->getSpelling() << "-fdefault-calling-conv";
|
|
|
|
else {
|
|
|
|
llvm::Triple T(TargetOpts.Triple);
|
|
|
|
if (T.getArch() != llvm::Triple::x86)
|
|
|
|
Diags.Report(diag::err_drv_argument_not_allowed_with)
|
|
|
|
<< A->getSpelling() << T.getTriple();
|
|
|
|
else
|
|
|
|
Opts.setDefaultCallingConv(LangOptions::DCC_StdCall);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-29 17:36:15 +00:00
|
|
|
// Check if -fopenmp-simd is specified.
|
2018-05-21 16:40:32 +00:00
|
|
|
bool IsSimdSpecified =
|
|
|
|
Args.hasFlag(options::OPT_fopenmp_simd, options::OPT_fno_openmp_simd,
|
|
|
|
/*Default=*/false);
|
|
|
|
Opts.OpenMPSimd = !Opts.OpenMP && IsSimdSpecified;
|
2015-07-13 22:54:53 +00:00
|
|
|
Opts.OpenMPUseTLS =
|
|
|
|
Opts.OpenMP && !Args.hasArg(options::OPT_fnoopenmp_use_tls);
|
2016-01-06 13:42:12 +00:00
|
|
|
Opts.OpenMPIsDevice =
|
|
|
|
Opts.OpenMP && Args.hasArg(options::OPT_fopenmp_is_device);
|
[OpenMP] Use the OpenMP-IR-Builder
This is a follow up patch to use the OpenMP-IR-Builder, as discussed on
the mailing list ([1] and later) and at the US Dev Meeting'19.
[1] http://lists.flang-compiler.org/pipermail/flang-dev_lists.flang-compiler.org/2019-May/000197.html
Reviewers: kiranchandramohan, ABataev, RaviNarayanaswamy, gtbercea, grokos, sdmitriev, JonChesterfield, hfinkel, fghanim
Subscribers: ppenzin, penzn, llvm-commits, cfe-commits, jfb, guansong, bollu, hiraditya, mgorny
Tags: #clang
Differential Revision: https://reviews.llvm.org/D69922
2019-11-04 23:00:36 -06:00
|
|
|
Opts.OpenMPIRBuilder =
|
|
|
|
Opts.OpenMP && Args.hasArg(options::OPT_fopenmp_enable_irbuilder);
|
2018-07-26 15:17:38 +00:00
|
|
|
bool IsTargetSpecified =
|
|
|
|
Opts.OpenMPIsDevice || Args.hasArg(options::OPT_fopenmp_targets_EQ);
|
2016-01-06 13:42:12 +00:00
|
|
|
|
2017-12-29 17:36:15 +00:00
|
|
|
if (Opts.OpenMP || Opts.OpenMPSimd) {
|
2018-07-26 15:17:38 +00:00
|
|
|
if (int Version = getLastArgIntValue(
|
|
|
|
Args, OPT_fopenmp_version_EQ,
|
2020-06-25 07:01:15 +00:00
|
|
|
(IsSimdSpecified || IsTargetSpecified) ? 50 : Opts.OpenMP, Diags))
|
2016-05-26 11:10:11 +00:00
|
|
|
Opts.OpenMP = Version;
|
|
|
|
// Provide diagnostic when a given target is not expected to be an OpenMP
|
|
|
|
// device or host.
|
|
|
|
if (!Opts.OpenMPIsDevice) {
|
|
|
|
switch (T.getArch()) {
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
// Add unsupported host targets here:
|
|
|
|
case llvm::Triple::nvptx:
|
|
|
|
case llvm::Triple::nvptx64:
|
2018-03-26 21:45:04 +00:00
|
|
|
Diags.Report(diag::err_drv_omp_host_target_not_supported)
|
2016-05-26 11:10:11 +00:00
|
|
|
<< TargetOpts.Triple;
|
|
|
|
break;
|
|
|
|
}
|
2016-02-08 15:59:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[OpenMP] Prevent emission of exception handling code when using OpenMP to offload to NVIDIA devices.
Summary: For the OpenMP toolchain which offloads to NVIDIA GPUs make sure that no exception handling code is emitted.
Reviewers: arpith-jacob, sfantao, caomhin, carlo.bertolli, ABataev, Hahnfeld, hfinkel, tstellar
Reviewed By: ABataev, Hahnfeld
Subscribers: rengolin, Hahnfeld, cfe-commits
Differential Revision: https://reviews.llvm.org/D29904
llvm-svn: 310306
2017-08-07 20:57:59 +00:00
|
|
|
// Set the flag to prevent the implementation from emitting device exception
|
|
|
|
// handling code for those requiring so.
|
[OpenMP][AMDGCN] Support OpenMP offloading for AMDGCN architecture - Part 1
Summary:
Allow AMDGCN as a GPU offloading target for OpenMP during compiler
invocation and allow setting CUDAMode for it.
Originally authored by Greg Rodgers (@gregrodgers).
Reviewers: ronlieb, yaxunl, b-sumner, scchan, JonChesterfield, jdoerfert, sameerds, msearles, hliao, arsenm
Reviewed By: sameerds
Subscribers: sstefan1, jvesely, wdng, arsenm, guansong, dexonsmith, cfe-commits, llvm-commits, gregrodgers
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D79754
2020-05-14 06:09:04 +00:00
|
|
|
if ((Opts.OpenMPIsDevice && (T.isNVPTX() || T.isAMDGCN())) ||
|
|
|
|
Opts.OpenCLCPlusPlus) {
|
[OpenMP] Prevent emission of exception handling code when using OpenMP to offload to NVIDIA devices.
Summary: For the OpenMP toolchain which offloads to NVIDIA GPUs make sure that no exception handling code is emitted.
Reviewers: arpith-jacob, sfantao, caomhin, carlo.bertolli, ABataev, Hahnfeld, hfinkel, tstellar
Reviewed By: ABataev, Hahnfeld
Subscribers: rengolin, Hahnfeld, cfe-commits
Differential Revision: https://reviews.llvm.org/D29904
llvm-svn: 310306
2017-08-07 20:57:59 +00:00
|
|
|
Opts.Exceptions = 0;
|
|
|
|
Opts.CXXExceptions = 0;
|
|
|
|
}
|
2018-11-02 14:54:07 +00:00
|
|
|
if (Opts.OpenMPIsDevice && T.isNVPTX()) {
|
|
|
|
Opts.OpenMPCUDANumSMs =
|
|
|
|
getLastArgIntValue(Args, options::OPT_fopenmp_cuda_number_of_sm_EQ,
|
|
|
|
Opts.OpenMPCUDANumSMs, Diags);
|
|
|
|
Opts.OpenMPCUDABlocksPerSM =
|
|
|
|
getLastArgIntValue(Args, options::OPT_fopenmp_cuda_blocks_per_sm_EQ,
|
|
|
|
Opts.OpenMPCUDABlocksPerSM, Diags);
|
2019-02-20 16:36:22 +00:00
|
|
|
Opts.OpenMPCUDAReductionBufNum = getLastArgIntValue(
|
|
|
|
Args, options::OPT_fopenmp_cuda_teams_reduction_recs_num_EQ,
|
|
|
|
Opts.OpenMPCUDAReductionBufNum, Diags);
|
2018-11-02 14:54:07 +00:00
|
|
|
}
|
[OpenMP] Prevent emission of exception handling code when using OpenMP to offload to NVIDIA devices.
Summary: For the OpenMP toolchain which offloads to NVIDIA GPUs make sure that no exception handling code is emitted.
Reviewers: arpith-jacob, sfantao, caomhin, carlo.bertolli, ABataev, Hahnfeld, hfinkel, tstellar
Reviewed By: ABataev, Hahnfeld
Subscribers: rengolin, Hahnfeld, cfe-commits
Differential Revision: https://reviews.llvm.org/D29904
llvm-svn: 310306
2017-08-07 20:57:59 +00:00
|
|
|
|
2016-01-06 13:42:12 +00:00
|
|
|
// Get the OpenMP target triples if any.
|
2016-06-30 21:22:08 +00:00
|
|
|
if (Arg *A = Args.getLastArg(options::OPT_fopenmp_targets_EQ)) {
|
2020-10-07 14:13:14 -04:00
|
|
|
enum ArchPtrSize { Arch16Bit, Arch32Bit, Arch64Bit };
|
|
|
|
auto getArchPtrSize = [](const llvm::Triple &T) {
|
|
|
|
if (T.isArch16Bit())
|
|
|
|
return Arch16Bit;
|
|
|
|
if (T.isArch32Bit())
|
|
|
|
return Arch32Bit;
|
|
|
|
assert(T.isArch64Bit() && "Expected 64-bit architecture");
|
|
|
|
return Arch64Bit;
|
|
|
|
};
|
2016-01-06 13:42:12 +00:00
|
|
|
|
|
|
|
for (unsigned i = 0; i < A->getNumValues(); ++i) {
|
|
|
|
llvm::Triple TT(A->getValue(i));
|
|
|
|
|
2017-08-07 21:11:10 +00:00
|
|
|
if (TT.getArch() == llvm::Triple::UnknownArch ||
|
2021-01-02 12:17:58 -06:00
|
|
|
!(TT.getArch() == llvm::Triple::aarch64 || TT.isPPC() ||
|
2017-08-07 21:11:10 +00:00
|
|
|
TT.getArch() == llvm::Triple::nvptx ||
|
|
|
|
TT.getArch() == llvm::Triple::nvptx64 ||
|
[OpenMP][AMDGCN] Support OpenMP offloading for AMDGCN architecture - Part 1
Summary:
Allow AMDGCN as a GPU offloading target for OpenMP during compiler
invocation and allow setting CUDAMode for it.
Originally authored by Greg Rodgers (@gregrodgers).
Reviewers: ronlieb, yaxunl, b-sumner, scchan, JonChesterfield, jdoerfert, sameerds, msearles, hliao, arsenm
Reviewed By: sameerds
Subscribers: sstefan1, jvesely, wdng, arsenm, guansong, dexonsmith, cfe-commits, llvm-commits, gregrodgers
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D79754
2020-05-14 06:09:04 +00:00
|
|
|
TT.getArch() == llvm::Triple::amdgcn ||
|
2017-08-07 21:11:10 +00:00
|
|
|
TT.getArch() == llvm::Triple::x86 ||
|
|
|
|
TT.getArch() == llvm::Triple::x86_64))
|
2018-03-26 21:45:04 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_omp_target) << A->getValue(i);
|
2020-10-07 14:13:14 -04:00
|
|
|
else if (getArchPtrSize(T) != getArchPtrSize(TT))
|
|
|
|
Diags.Report(diag::err_drv_incompatible_omp_arch)
|
|
|
|
<< A->getValue(i) << T.str();
|
2016-01-06 13:42:12 +00:00
|
|
|
else
|
|
|
|
Opts.OMPTargetTriples.push_back(TT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get OpenMP host file path if any and report if a non existent file is
|
|
|
|
// found
|
2016-06-30 21:22:08 +00:00
|
|
|
if (Arg *A = Args.getLastArg(options::OPT_fopenmp_host_ir_file_path)) {
|
2016-01-06 13:42:12 +00:00
|
|
|
Opts.OMPHostIRFile = A->getValue();
|
|
|
|
if (!llvm::sys::fs::exists(Opts.OMPHostIRFile))
|
2018-03-26 21:45:04 +00:00
|
|
|
Diags.Report(diag::err_drv_omp_host_ir_file_not_found)
|
2016-01-06 13:42:12 +00:00
|
|
|
<< Opts.OMPHostIRFile;
|
|
|
|
}
|
2013-01-15 06:45:29 +00:00
|
|
|
|
[OpenMP][AMDGCN] Support OpenMP offloading for AMDGCN architecture - Part 1
Summary:
Allow AMDGCN as a GPU offloading target for OpenMP during compiler
invocation and allow setting CUDAMode for it.
Originally authored by Greg Rodgers (@gregrodgers).
Reviewers: ronlieb, yaxunl, b-sumner, scchan, JonChesterfield, jdoerfert, sameerds, msearles, hliao, arsenm
Reviewed By: sameerds
Subscribers: sstefan1, jvesely, wdng, arsenm, guansong, dexonsmith, cfe-commits, llvm-commits, gregrodgers
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D79754
2020-05-14 06:09:04 +00:00
|
|
|
// Set CUDA mode for OpenMP target NVPTX/AMDGCN if specified in options
|
|
|
|
Opts.OpenMPCUDAMode = Opts.OpenMPIsDevice && (T.isNVPTX() || T.isAMDGCN()) &&
|
2018-02-28 20:48:35 +00:00
|
|
|
Args.hasArg(options::OPT_fopenmp_cuda_mode);
|
|
|
|
|
2020-06-22 15:04:58 -04:00
|
|
|
// Set CUDA support for parallel execution of target regions for OpenMP target
|
|
|
|
// NVPTX/AMDGCN if specified in options.
|
|
|
|
Opts.OpenMPCUDATargetParallel =
|
|
|
|
Opts.OpenMPIsDevice && (T.isNVPTX() || T.isAMDGCN()) &&
|
|
|
|
Args.hasArg(options::OPT_fopenmp_cuda_parallel_target_regions);
|
|
|
|
|
[OpenMP][AMDGCN] Support OpenMP offloading for AMDGCN architecture - Part 1
Summary:
Allow AMDGCN as a GPU offloading target for OpenMP during compiler
invocation and allow setting CUDAMode for it.
Originally authored by Greg Rodgers (@gregrodgers).
Reviewers: ronlieb, yaxunl, b-sumner, scchan, JonChesterfield, jdoerfert, sameerds, msearles, hliao, arsenm
Reviewed By: sameerds
Subscribers: sstefan1, jvesely, wdng, arsenm, guansong, dexonsmith, cfe-commits, llvm-commits, gregrodgers
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D79754
2020-05-14 06:09:04 +00:00
|
|
|
// Set CUDA mode for OpenMP target NVPTX/AMDGCN if specified in options
|
2018-08-30 14:45:24 +00:00
|
|
|
Opts.OpenMPCUDAForceFullRuntime =
|
[OpenMP][AMDGCN] Support OpenMP offloading for AMDGCN architecture - Part 1
Summary:
Allow AMDGCN as a GPU offloading target for OpenMP during compiler
invocation and allow setting CUDAMode for it.
Originally authored by Greg Rodgers (@gregrodgers).
Reviewers: ronlieb, yaxunl, b-sumner, scchan, JonChesterfield, jdoerfert, sameerds, msearles, hliao, arsenm
Reviewed By: sameerds
Subscribers: sstefan1, jvesely, wdng, arsenm, guansong, dexonsmith, cfe-commits, llvm-commits, gregrodgers
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D79754
2020-05-14 06:09:04 +00:00
|
|
|
Opts.OpenMPIsDevice && (T.isNVPTX() || T.isAMDGCN()) &&
|
2018-08-30 14:45:24 +00:00
|
|
|
Args.hasArg(options::OPT_fopenmp_cuda_force_full_runtime);
|
|
|
|
|
2011-04-23 09:27:53 +00:00
|
|
|
// Record whether the __DEPRECATED define was requested.
|
|
|
|
Opts.Deprecated = Args.hasFlag(OPT_fdeprecated_macro,
|
|
|
|
OPT_fno_deprecated_macro,
|
|
|
|
Opts.Deprecated);
|
|
|
|
|
2009-12-01 03:16:53 +00:00
|
|
|
// FIXME: Eliminate this dependency.
|
2012-08-08 16:09:15 +00:00
|
|
|
unsigned Opt = getOptimizationLevel(Args, IK, Diags),
|
2013-04-10 21:30:03 +00:00
|
|
|
OptSize = getOptimizationLevelSize(Args);
|
2009-12-01 03:16:53 +00:00
|
|
|
Opts.Optimize = Opt != 0;
|
2012-08-08 16:09:15 +00:00
|
|
|
Opts.OptimizeSize = OptSize != 0;
|
2009-12-01 03:16:53 +00:00
|
|
|
|
|
|
|
// This is the __NO_INLINE__ define, which just depends on things like the
|
|
|
|
// optimization level and -fno-inline, not actually whether the backend has
|
|
|
|
// inlining enabled.
|
Cleanup the handling of noinline function attributes, -fno-inline,
-fno-inline-functions, -O0, and optnone.
These were really, really tangled together:
- We used the noinline LLVM attribute for -fno-inline
- But not for -fno-inline-functions (breaking LTO)
- But we did use it for -finline-hint-functions (yay, LTO is happy!)
- But we didn't for -O0 (LTO is sad yet again...)
- We had weird structuring of CodeGenOpts with both an inlining
enumeration and a boolean. They interacted in weird ways and
needlessly.
- A *lot* of set smashing went on with setting these, and then got worse
when we considered optnone and other inlining-effecting attributes.
- A bunch of inline affecting attributes were managed in a completely
different place from -fno-inline.
- Even with -fno-inline we failed to put the LLVM noinline attribute
onto many generated function definitions because they didn't show up
as AST-level functions.
- If you passed -O0 but -finline-functions we would run the normal
inliner pass in LLVM despite it being in the O0 pipeline, which really
doesn't make much sense.
- Lastly, we used things like '-fno-inline' to manipulate the pass
pipeline which forced the pass pipeline to be much more
parameterizable than it really needs to be. Instead we can *just* use
the optimization level to select a pipeline and control the rest via
attributes.
Sadly, this causes a bunch of churn in tests because we don't run the
optimizer in the tests and check the contents of attribute sets. It
would be awesome if attribute sets were a bit more FileCheck friendly,
but oh well.
I think this is a significant improvement and should remove the semantic
need to change what inliner pass we run in order to comply with the
requested inlining semantics by relying completely on attributes. It
also cleans up tho optnone and related handling a bit.
One unfortunate aspect of this is that for generating alwaysinline
routines like those in OpenMP we end up removing noinline and then
adding alwaysinline. I tried a bunch of other approaches, but because we
recompute function attributes from scratch and don't have a declaration
here I couldn't find anything substantially cleaner than this.
Differential Revision: https://reviews.llvm.org/D28053
llvm-svn: 290398
2016-12-23 01:24:49 +00:00
|
|
|
Opts.NoInlineDefine = !Opts.Optimize;
|
|
|
|
if (Arg *InlineArg = Args.getLastArg(
|
|
|
|
options::OPT_finline_functions, options::OPT_finline_hint_functions,
|
|
|
|
options::OPT_fno_inline_functions, options::OPT_fno_inline))
|
|
|
|
if (InlineArg->getOption().matches(options::OPT_fno_inline))
|
|
|
|
Opts.NoInlineDefine = true;
|
2009-12-01 03:16:53 +00:00
|
|
|
|
2017-03-29 21:54:24 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_ffp_contract)) {
|
|
|
|
StringRef Val = A->getValue();
|
|
|
|
if (Val == "fast")
|
2020-05-04 10:48:12 -07:00
|
|
|
Opts.setDefaultFPContractMode(LangOptions::FPM_Fast);
|
2017-03-29 21:54:24 +00:00
|
|
|
else if (Val == "on")
|
2020-05-04 10:48:12 -07:00
|
|
|
Opts.setDefaultFPContractMode(LangOptions::FPM_On);
|
2017-03-29 21:54:24 +00:00
|
|
|
else if (Val == "off")
|
2020-05-04 10:48:12 -07:00
|
|
|
Opts.setDefaultFPContractMode(LangOptions::FPM_Off);
|
[HIP] Fix regressions due to fp contract change
Recently HIP toolchain made a change to use clang instead of opt/llc to do compilation
(https://reviews.llvm.org/D81861). The intention is to make HIP toolchain canonical like
other toolchains.
However, this change introduced an unintentional change regarding backend fp fuse
option, which caused regressions in some HIP applications.
Basically before the change, HIP toolchain used clang to generate bitcode, then use
opt/llc to optimize bitcode and generate ISA. As such, the amdgpu backend takes
the default fp fuse mode which is 'Standard'. This mode respect contract flag of
fmul/fadd instructions and do not fuse fmul/fadd instructions without contract flag.
However, after the change, HIP toolchain now use clang to generate IR, do optimization,
and generate ISA as one process. Now amdgpu backend fp fuse option is determined
by -ffp-contract option, which is 'fast' by default. And this -ffp-contract=fast language option
is translated to 'Fast' fp fuse option in backend. Suddenly backend starts to fuse fmul/fadd
instructions without contract flag.
This causes wrong result for some device library functions, e.g. tan(-1e20), which should
return 0.8446, now returns -0.933. What is worse is that since backend with 'Fast' fp fuse
option does not respect contract flag, there is no way to use #pragma clang fp contract
directive to enforce fp contract requirements.
This patch fixes the regression by introducing a new value 'fast-honor-pragmas' for -ffp-contract
and use it for HIP by default. 'fast-honor-pragmas' is equivalent to 'fast' in frontend but
let the backend to use 'Standard' fp fuse option. 'fast-honor-pragmas' is useful since 'Fast'
fp fuse option in backend does not honor contract flag, it is of little use to HIP
applications since all code with #pragma STDC FP_CONTRACT or any IR from a
source compiled with -ffp-contract=on is broken.
Differential Revision: https://reviews.llvm.org/D90174
2020-10-23 16:24:48 -04:00
|
|
|
else if (Val == "fast-honor-pragmas")
|
|
|
|
Opts.setDefaultFPContractMode(LangOptions::FPM_FastHonorPragmas);
|
2017-03-29 21:54:24 +00:00
|
|
|
else
|
|
|
|
Diags.Report(diag::err_drv_invalid_value) << A->getAsString(Args) << Val;
|
|
|
|
}
|
|
|
|
|
2019-12-04 12:23:46 -08:00
|
|
|
if (Args.hasArg(OPT_ftrapping_math)) {
|
|
|
|
Opts.setFPExceptionMode(LangOptions::FPE_Strict);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Args.hasArg(OPT_fno_trapping_math)) {
|
|
|
|
Opts.setFPExceptionMode(LangOptions::FPE_Ignore);
|
|
|
|
}
|
|
|
|
|
|
|
|
LangOptions::FPExceptionModeKind FPEB = LangOptions::FPE_Ignore;
|
|
|
|
if (Arg *A = Args.getLastArg(OPT_ffp_exception_behavior_EQ)) {
|
|
|
|
StringRef Val = A->getValue();
|
|
|
|
if (Val.equals("ignore"))
|
|
|
|
FPEB = LangOptions::FPE_Ignore;
|
|
|
|
else if (Val.equals("maytrap"))
|
|
|
|
FPEB = LangOptions::FPE_MayTrap;
|
|
|
|
else if (Val.equals("strict"))
|
|
|
|
FPEB = LangOptions::FPE_Strict;
|
|
|
|
else
|
|
|
|
Diags.Report(diag::err_drv_invalid_value) << A->getAsString(Args) << Val;
|
|
|
|
}
|
|
|
|
Opts.setFPExceptionMode(FPEB);
|
|
|
|
|
2012-11-05 22:04:41 +00:00
|
|
|
// Parse -fsanitize= arguments.
|
Reimplement -fsanitize-recover family of flags.
Introduce the following -fsanitize-recover flags:
- -fsanitize-recover=<list>: Enable recovery for selected checks or
group of checks. It is forbidden to explicitly list unrecoverable
sanitizers here (that is, "address", "unreachable", "return").
- -fno-sanitize-recover=<list>: Disable recovery for selected checks or
group of checks.
- -f(no-)?sanitize-recover is now a synonym for
-f(no-)?sanitize-recover=undefined,integer and will soon be deprecated.
These flags are parsed left to right, and mask of "recoverable"
sanitizer is updated accordingly, much like what we do for -fsanitize= flags.
-fsanitize= and -fsanitize-recover= flag families are independent.
CodeGen change: If there is a single UBSan handler function, responsible
for implementing multiple checks, which have different recoverable setting,
then we emit two handler calls instead of one:
the first one for the set of "unrecoverable" checks, another one - for
set of "recoverable" checks. If all checks implemented by a handler have the
same recoverability setting, then the generated code will be the same.
llvm-svn: 225719
2015-01-12 22:39:12 +00:00
|
|
|
parseSanitizerKinds("-fsanitize=", Args.getAllArgValues(OPT_fsanitize_EQ),
|
|
|
|
Diags, Opts.Sanitize);
|
2019-11-08 13:56:37 -08:00
|
|
|
std::vector<std::string> systemBlacklists =
|
|
|
|
Args.getAllArgValues(OPT_fsanitize_system_blacklist);
|
|
|
|
Opts.SanitizerBlacklistFiles.insert(Opts.SanitizerBlacklistFiles.end(),
|
|
|
|
systemBlacklists.begin(),
|
|
|
|
systemBlacklists.end());
|
2017-03-30 00:29:36 +00:00
|
|
|
|
2018-03-28 21:13:14 +00:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_fclang_abi_compat_EQ)) {
|
|
|
|
Opts.setClangABICompat(LangOptions::ClangABI::Latest);
|
|
|
|
|
|
|
|
StringRef Ver = A->getValue();
|
|
|
|
std::pair<StringRef, StringRef> VerParts = Ver.split('.');
|
|
|
|
unsigned Major, Minor = 0;
|
|
|
|
|
|
|
|
// Check the version number is valid: either 3.x (0 <= x <= 9) or
|
|
|
|
// y or y.0 (4 <= y <= current version).
|
|
|
|
if (!VerParts.first.startswith("0") &&
|
|
|
|
!VerParts.first.getAsInteger(10, Major) &&
|
|
|
|
3 <= Major && Major <= CLANG_VERSION_MAJOR &&
|
|
|
|
(Major == 3 ? VerParts.second.size() == 1 &&
|
|
|
|
!VerParts.second.getAsInteger(10, Minor)
|
|
|
|
: VerParts.first.size() == Ver.size() ||
|
|
|
|
VerParts.second == "0")) {
|
|
|
|
// Got a valid version number.
|
|
|
|
if (Major == 3 && Minor <= 8)
|
|
|
|
Opts.setClangABICompat(LangOptions::ClangABI::Ver3_8);
|
|
|
|
else if (Major <= 4)
|
|
|
|
Opts.setClangABICompat(LangOptions::ClangABI::Ver4);
|
2018-04-02 18:29:44 +00:00
|
|
|
else if (Major <= 6)
|
|
|
|
Opts.setClangABICompat(LangOptions::ClangABI::Ver6);
|
2018-10-26 19:26:45 +00:00
|
|
|
else if (Major <= 7)
|
|
|
|
Opts.setClangABICompat(LangOptions::ClangABI::Ver7);
|
2019-09-06 06:02:13 +00:00
|
|
|
else if (Major <= 9)
|
|
|
|
Opts.setClangABICompat(LangOptions::ClangABI::Ver9);
|
2020-10-19 18:17:34 -07:00
|
|
|
else if (Major <= 11)
|
|
|
|
Opts.setClangABICompat(LangOptions::ClangABI::Ver11);
|
2018-03-28 21:13:14 +00:00
|
|
|
} else if (Ver != "latest") {
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< A->getAsString(Args) << A->getValue();
|
|
|
|
}
|
|
|
|
}
|
2018-05-30 03:40:04 +00:00
|
|
|
|
2020-04-02 10:11:01 +02:00
|
|
|
if (Arg *A = Args.getLastArg(OPT_msign_return_address_EQ)) {
|
|
|
|
StringRef SignScope = A->getValue();
|
|
|
|
|
|
|
|
if (SignScope.equals_lower("none"))
|
|
|
|
Opts.setSignReturnAddressScope(
|
|
|
|
LangOptions::SignReturnAddressScopeKind::None);
|
|
|
|
else if (SignScope.equals_lower("all"))
|
|
|
|
Opts.setSignReturnAddressScope(
|
|
|
|
LangOptions::SignReturnAddressScopeKind::All);
|
|
|
|
else if (SignScope.equals_lower("non-leaf"))
|
|
|
|
Opts.setSignReturnAddressScope(
|
|
|
|
LangOptions::SignReturnAddressScopeKind::NonLeaf);
|
|
|
|
else
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< A->getAsString(Args) << SignScope;
|
|
|
|
|
|
|
|
if (Arg *A = Args.getLastArg(OPT_msign_return_address_key_EQ)) {
|
|
|
|
StringRef SignKey = A->getValue();
|
|
|
|
if (!SignScope.empty() && !SignKey.empty()) {
|
|
|
|
if (SignKey.equals_lower("a_key"))
|
|
|
|
Opts.setSignReturnAddressKey(
|
|
|
|
LangOptions::SignReturnAddressKeyKind::AKey);
|
|
|
|
else if (SignKey.equals_lower("b_key"))
|
|
|
|
Opts.setSignReturnAddressKey(
|
|
|
|
LangOptions::SignReturnAddressKeyKind::BKey);
|
|
|
|
else
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< A->getAsString(Args) << SignKey;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-18 15:26:21 -08:00
|
|
|
std::string ThreadModel =
|
|
|
|
std::string(Args.getLastArgValue(OPT_mthread_model, "posix"));
|
|
|
|
if (ThreadModel != "posix" && ThreadModel != "single")
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< Args.getLastArg(OPT_mthread_model)->getAsString(Args) << ThreadModel;
|
|
|
|
Opts.setThreadModel(
|
|
|
|
llvm::StringSwitch<LangOptions::ThreadModelKind>(ThreadModel)
|
|
|
|
.Case("posix", LangOptions::ThreadModelKind::POSIX)
|
|
|
|
.Case("single", LangOptions::ThreadModelKind::Single));
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
2017-06-16 20:13:39 +00:00
|
|
|
static bool isStrictlyPreprocessorAction(frontend::ActionKind Action) {
|
|
|
|
switch (Action) {
|
|
|
|
case frontend::ASTDeclList:
|
|
|
|
case frontend::ASTDump:
|
|
|
|
case frontend::ASTPrint:
|
|
|
|
case frontend::ASTView:
|
|
|
|
case frontend::EmitAssembly:
|
|
|
|
case frontend::EmitBC:
|
|
|
|
case frontend::EmitHTML:
|
|
|
|
case frontend::EmitLLVM:
|
|
|
|
case frontend::EmitLLVMOnly:
|
|
|
|
case frontend::EmitCodeGenOnly:
|
|
|
|
case frontend::EmitObj:
|
|
|
|
case frontend::FixIt:
|
|
|
|
case frontend::GenerateModule:
|
|
|
|
case frontend::GenerateModuleInterface:
|
2018-09-15 01:21:15 +00:00
|
|
|
case frontend::GenerateHeaderModule:
|
2017-06-16 20:13:39 +00:00
|
|
|
case frontend::GeneratePCH:
|
2020-03-28 04:08:27 -04:00
|
|
|
case frontend::GenerateInterfaceStubs:
|
2017-06-16 20:13:39 +00:00
|
|
|
case frontend::ParseSyntaxOnly:
|
|
|
|
case frontend::ModuleFileInfo:
|
|
|
|
case frontend::VerifyPCH:
|
|
|
|
case frontend::PluginAction:
|
|
|
|
case frontend::RewriteObjC:
|
|
|
|
case frontend::RewriteTest:
|
|
|
|
case frontend::RunAnalysis:
|
2018-02-10 14:04:45 +00:00
|
|
|
case frontend::TemplightDump:
|
2017-06-16 20:13:39 +00:00
|
|
|
case frontend::MigrateSource:
|
|
|
|
return false;
|
|
|
|
|
2018-05-31 13:57:09 +00:00
|
|
|
case frontend::DumpCompilerOptions:
|
2017-06-16 20:13:39 +00:00
|
|
|
case frontend::DumpRawTokens:
|
|
|
|
case frontend::DumpTokens:
|
|
|
|
case frontend::InitOnly:
|
|
|
|
case frontend::PrintPreamble:
|
|
|
|
case frontend::PrintPreprocessedInput:
|
|
|
|
case frontend::RewriteMacros:
|
|
|
|
case frontend::RunPreprocessorOnly:
|
2019-06-03 22:59:17 +00:00
|
|
|
case frontend::PrintDependencyDirectivesSourceMinimizerOutput:
|
2017-06-16 20:13:39 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
llvm_unreachable("invalid frontend action");
|
|
|
|
}
|
|
|
|
|
2009-12-03 05:11:16 +00:00
|
|
|
static void ParsePreprocessorArgs(PreprocessorOptions &Opts, ArgList &Args,
|
2017-06-16 20:13:39 +00:00
|
|
|
DiagnosticsEngine &Diags,
|
|
|
|
frontend::ActionKind Action) {
|
2018-09-11 17:10:44 +00:00
|
|
|
Opts.PCHWithHdrStop = Args.hasArg(OPT_pch_through_hdrstop_create) ||
|
|
|
|
Args.hasArg(OPT_pch_through_hdrstop_use);
|
2020-11-17 17:25:50 -08:00
|
|
|
Opts.AllowPCHWithCompilerErrors =
|
|
|
|
Args.hasArg(OPT_fallow_pch_with_errors, OPT_fallow_pcm_with_errors);
|
2010-10-14 20:14:25 +00:00
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_error_on_deserialized_pch_decl))
|
2012-11-01 04:30:05 +00:00
|
|
|
Opts.DeserializedPCHDeclsToErrorOn.insert(A->getValue());
|
2010-07-27 00:27:13 +00:00
|
|
|
|
2020-01-29 00:42:56 +01:00
|
|
|
for (const auto &A : Args.getAllArgValues(OPT_fmacro_prefix_map_EQ)) {
|
|
|
|
auto Split = StringRef(A).split('=');
|
|
|
|
Opts.MacroPrefixMap.insert(
|
|
|
|
{std::string(Split.first), std::string(Split.second)});
|
|
|
|
}
|
2019-11-26 14:23:07 -08:00
|
|
|
|
Introduce basic support for loading a precompiled preamble while
reparsing an ASTUnit. When saving a preamble, create a buffer larger
than the actual file we're working with but fill everything from the
end of the preamble to the end of the file with spaces (so the lexer
will quickly skip them). When we load the file, create a buffer of the
same size, filling it with the file and then spaces. Then, instruct
the lexer to start lexing after the preamble, therefore continuing the
parse from the spot where the preamble left off.
It's now possible to perform a simple preamble build + parse (+
reparse) with ASTUnit. However, one has to disable a bunch of checking
in the PCH reader to do so. That part isn't committed; it will likely
be handled with some other kind of flag (e.g., -fno-validate-pch).
As part of this, fix some issues with null termination of the memory
buffers created for the preamble; we were trying to explicitly
NULL-terminate them, even though they were also getting implicitly
NULL terminated, leading to excess warnings about NULL characters in
source files.
llvm-svn: 109445
2010-07-26 21:36:20 +00:00
|
|
|
if (const Arg *A = Args.getLastArg(OPT_preamble_bytes_EQ)) {
|
2012-11-01 04:30:05 +00:00
|
|
|
StringRef Value(A->getValue());
|
Introduce basic support for loading a precompiled preamble while
reparsing an ASTUnit. When saving a preamble, create a buffer larger
than the actual file we're working with but fill everything from the
end of the preamble to the end of the file with spaces (so the lexer
will quickly skip them). When we load the file, create a buffer of the
same size, filling it with the file and then spaces. Then, instruct
the lexer to start lexing after the preamble, therefore continuing the
parse from the spot where the preamble left off.
It's now possible to perform a simple preamble build + parse (+
reparse) with ASTUnit. However, one has to disable a bunch of checking
in the PCH reader to do so. That part isn't committed; it will likely
be handled with some other kind of flag (e.g., -fno-validate-pch).
As part of this, fix some issues with null termination of the memory
buffers created for the preamble; we were trying to explicitly
NULL-terminate them, even though they were also getting implicitly
NULL terminated, leading to excess warnings about NULL characters in
source files.
llvm-svn: 109445
2010-07-26 21:36:20 +00:00
|
|
|
size_t Comma = Value.find(',');
|
|
|
|
unsigned Bytes = 0;
|
|
|
|
unsigned EndOfLine = 0;
|
2010-10-21 03:16:25 +00:00
|
|
|
|
2011-07-23 10:55:15 +00:00
|
|
|
if (Comma == StringRef::npos ||
|
Introduce basic support for loading a precompiled preamble while
reparsing an ASTUnit. When saving a preamble, create a buffer larger
than the actual file we're working with but fill everything from the
end of the preamble to the end of the file with spaces (so the lexer
will quickly skip them). When we load the file, create a buffer of the
same size, filling it with the file and then spaces. Then, instruct
the lexer to start lexing after the preamble, therefore continuing the
parse from the spot where the preamble left off.
It's now possible to perform a simple preamble build + parse (+
reparse) with ASTUnit. However, one has to disable a bunch of checking
in the PCH reader to do so. That part isn't committed; it will likely
be handled with some other kind of flag (e.g., -fno-validate-pch).
As part of this, fix some issues with null termination of the memory
buffers created for the preamble; we were trying to explicitly
NULL-terminate them, even though they were also getting implicitly
NULL terminated, leading to excess warnings about NULL characters in
source files.
llvm-svn: 109445
2010-07-26 21:36:20 +00:00
|
|
|
Value.substr(0, Comma).getAsInteger(10, Bytes) ||
|
|
|
|
Value.substr(Comma + 1).getAsInteger(10, EndOfLine))
|
|
|
|
Diags.Report(diag::err_drv_preamble_format);
|
|
|
|
else {
|
|
|
|
Opts.PrecompiledPreambleBytes.first = Bytes;
|
|
|
|
Opts.PrecompiledPreambleBytes.second = (EndOfLine != 0);
|
|
|
|
}
|
|
|
|
}
|
2010-10-21 03:16:25 +00:00
|
|
|
|
2018-05-18 11:56:21 +00:00
|
|
|
// Add the __CET__ macro if a CFProtection option is set.
|
|
|
|
if (const Arg *A = Args.getLastArg(OPT_fcf_protection_EQ)) {
|
|
|
|
StringRef Name = A->getValue();
|
|
|
|
if (Name == "branch")
|
|
|
|
Opts.addMacroDef("__CET__=1");
|
|
|
|
else if (Name == "return")
|
|
|
|
Opts.addMacroDef("__CET__=2");
|
|
|
|
else if (Name == "full")
|
|
|
|
Opts.addMacroDef("__CET__=3");
|
|
|
|
}
|
|
|
|
|
2009-12-01 03:16:53 +00:00
|
|
|
// Add macros from the command line.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_D, OPT_U)) {
|
2015-06-09 01:57:17 +00:00
|
|
|
if (A->getOption().matches(OPT_D))
|
|
|
|
Opts.addMacroDef(A->getValue());
|
2009-12-01 03:16:53 +00:00
|
|
|
else
|
2015-06-09 01:57:17 +00:00
|
|
|
Opts.addMacroUndef(A->getValue());
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add the ordered list of -includes.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_include))
|
2015-05-29 19:42:19 +00:00
|
|
|
Opts.Includes.emplace_back(A->getValue());
|
2009-12-03 05:11:16 +00:00
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_chain_include))
|
2015-05-29 19:42:19 +00:00
|
|
|
Opts.ChainedIncludes.emplace_back(A->getValue());
|
2011-03-09 17:21:42 +00:00
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_remap_file)) {
|
2015-06-09 01:57:17 +00:00
|
|
|
std::pair<StringRef, StringRef> Split = StringRef(A->getValue()).split(';');
|
2009-12-03 05:11:16 +00:00
|
|
|
|
|
|
|
if (Split.second.empty()) {
|
2010-06-11 22:00:13 +00:00
|
|
|
Diags.Report(diag::err_drv_invalid_remap_file) << A->getAsString(Args);
|
2009-12-03 05:11:16 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
Opts.addRemappedFile(Split.first, Split.second);
|
|
|
|
}
|
2015-06-09 01:57:17 +00:00
|
|
|
|
2017-06-16 20:13:39 +00:00
|
|
|
// Always avoid lexing editor placeholders when we're just running the
|
|
|
|
// preprocessor as we never want to emit the
|
|
|
|
// "editor placeholder in source file" error in PP only mode.
|
|
|
|
if (isStrictlyPreprocessorAction(Action))
|
|
|
|
Opts.LexEditorPlaceholders = false;
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ParsePreprocessorOutputArgs(PreprocessorOutputOptions &Opts,
|
2013-01-30 01:52:57 +00:00
|
|
|
ArgList &Args,
|
|
|
|
frontend::ActionKind Action) {
|
2017-06-16 20:13:39 +00:00
|
|
|
if (isStrictlyPreprocessorAction(Action))
|
2013-01-30 01:52:57 +00:00
|
|
|
Opts.ShowCPP = !Args.hasArg(OPT_dM);
|
2017-06-16 20:13:39 +00:00
|
|
|
else
|
|
|
|
Opts.ShowCPP = 0;
|
2013-01-30 01:52:57 +00:00
|
|
|
|
2010-08-24 22:44:13 +00:00
|
|
|
Opts.ShowMacros = Args.hasArg(OPT_dM) || Args.hasArg(OPT_dD);
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
|
|
|
|
2016-04-08 16:52:05 +00:00
|
|
|
static void ParseTargetArgs(TargetOptions &Opts, ArgList &Args,
|
|
|
|
DiagnosticsEngine &Diags) {
|
2020-11-16 10:58:48 -05:00
|
|
|
Opts.AllowAMDGPUUnsafeFPAtomics =
|
|
|
|
Args.hasFlag(options::OPT_munsafe_fp_atomics,
|
|
|
|
options::OPT_mno_unsafe_fp_atomics, false);
|
2018-12-17 19:19:15 +00:00
|
|
|
if (Arg *A = Args.getLastArg(options::OPT_target_sdk_version_EQ)) {
|
|
|
|
llvm::VersionTuple Version;
|
|
|
|
if (Version.tryParse(A->getValue()))
|
|
|
|
Diags.Report(diag::err_drv_invalid_value)
|
|
|
|
<< A->getAsString(Args) << A->getValue();
|
|
|
|
else
|
|
|
|
Opts.SDKVersion = Version;
|
|
|
|
}
|
2014-01-13 19:48:18 +00:00
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
|
2020-05-11 11:42:38 +01:00
|
|
|
bool CompilerInvocation::parseSimpleArgs(const ArgList &Args,
|
|
|
|
DiagnosticsEngine &Diags) {
|
2020-12-22 15:00:31 +01:00
|
|
|
bool Success = true;
|
|
|
|
|
2020-11-11 11:05:24 +01:00
|
|
|
#define OPTION_WITH_MARSHALLING( \
|
2020-05-11 11:42:38 +01:00
|
|
|
PREFIX_TYPE, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
|
2021-01-07 10:14:48 +01:00
|
|
|
HELPTEXT, METAVAR, VALUES, SPELLING, SHOULD_PARSE, ALWAYS_EMIT, KEYPATH, \
|
|
|
|
DEFAULT_VALUE, IMPLIED_CHECK, IMPLIED_VALUE, NORMALIZER, DENORMALIZER, \
|
|
|
|
MERGER, EXTRACTOR, TABLE_INDEX) \
|
[clang][cli] Do not marshall only CC1Option flags in BoolOption
We cannot be sure whether a flag is CC1Option inside the definition of `BoolOption`. Take the example below:
```
let Flags = [CC1Option] in {
defm xxx : BoolOption<...>;
}
```
where TableGen applies `Flags = [CC1Option]` to the `xxx` and `no_xxx` records **after** they have been is fully declared by `BoolOption`.
For the refactored `-f[no-]debug-pass-manager` flags (see the diff), this means `BoolOption` never adds any marshalling info, as it doesn't see either of the flags as `CC1Option`.
For that reason, we should defensively append the marshalling information to both flags inside `BoolOption`. Now the check for `CC1Option` needs to happen later, in the parsing macro, when all TableGen logic has been resolved.
However, for some flags defined through `BoolOption`, we can run into issues:
```
// Options.td
def fenable_xxx : /* ... */;
// Both flags are CC1Option, the first is implied.
defm xxx : BoolOption<"xxx,
"Opts.Xxx", DefaultsToFalse,
ChangedBy<PosFlag, [CC1Option], "", [fenable_xxx]>,
ResetBy<NegFlag, [CC1Option]>>;
```
When parsing `clang -cc1 -fenable-xxx`:
* we run parsing for `PosFlag`:
* set `Opts.Xxx` to default `false`,
* discover `PosFlag` is implied by `-fenable-xxx`: set `Opts.Xxx` to `true`,
* don't see `-fxxx` on command line: do nothing,
* we run parsing for `NegFlag`:
* set `Opts.Xxx` to default `false`,
* discover `NegFlag` cannot be implied: do nothing,
* don't see `-fno-xxx` on command line: do nothing.
Now we ended up with `Opts.Xxx` set to `false` instead of `true`. For this reason, we need to ensure to append the same `ImpliedByAnyOf` instance to both flags.
This means both parsing runs now behave identically (they set the same default value, run the same "implied by" check, and call `makeBooleanOptionNormalizer` that already has information on both flags, so it returns the same value in both calls).
The solution works well, but what might be confusing is this: you have defined a flag **A** that is not `CC1Option`, but can be implied by another flag **B** that is `CC1Option`:
* if **A** is defined manually, it will never get implied, as the code never runs
```
def no_signed_zeros : Flag<["-"], "fno-signed-zeros">, Group<f_Group>, Flags<[]>,
MarshallingInfoFlag<"LangOpts->NoSignedZero">, ImpliedByAnyOf<[menable_unsafe_fp_math]>;
```
* if **A** is defined by `BoolOption`, it could get implied, as the code is run by its `CC1Option` counterpart:
```
defm signed_zeros : BoolOption<"signed-zeros",
"LangOpts->NoSignedZero", DefaultsToFalse,
ChangedBy<NegFlag, [], "Allow optimizations that ignore the sign of floating point zeros",
[cl_no_signed_zeros, menable_unsafe_fp_math]>,
ResetBy<PosFlag, [CC1Option]>, "f">, Group<f_Group>;
```
This is a surprising inconsistency.
One solution might be to somehow propagate the final `Flags` of the implied flag in `ImpliedByAnyOf` and check whether it has `CC1Option` in the parsing macro. However, I think it doesn't make sense to spend time thinking about a corner case that won't come up in real code.
An observation: it is unfortunate that the marshalling information is a part of the flag definition. If we represented it in a separate structure, we could avoid the "double parsing" problem by having a single source of truth. This would require a lot of additional work though.
Note: the original patch missed the `CC1Option` check in the parsing macro, making my reasoning here incomplete. Moreover, it contained a change to denormalization that wasn't necessarily related to these changes, so I've extracted that to a follow-up patch: D93094.
Reviewed By: dexonsmith, Bigcheese
Differential Revision: https://reviews.llvm.org/D93008
2020-12-08 18:15:21 +01:00
|
|
|
if ((FLAGS)&options::CC1Option) { \
|
2020-11-11 11:05:24 +01:00
|
|
|
this->KEYPATH = MERGER(this->KEYPATH, DEFAULT_VALUE); \
|
2020-11-20 12:49:51 +01:00
|
|
|
if (IMPLIED_CHECK) \
|
|
|
|
this->KEYPATH = MERGER(this->KEYPATH, IMPLIED_VALUE); \
|
2021-01-07 10:14:48 +01:00
|
|
|
if (SHOULD_PARSE) \
|
2020-12-22 15:00:31 +01:00
|
|
|
if (auto MaybeValue = \
|
|
|
|
NORMALIZER(OPT_##ID, TABLE_INDEX, Args, Diags, Success)) \
|
2021-01-07 10:14:48 +01:00
|
|
|
this->KEYPATH = MERGER( \
|
|
|
|
this->KEYPATH, static_cast<decltype(this->KEYPATH)>(*MaybeValue)); \
|
2020-05-11 11:42:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#include "clang/Driver/Options.inc"
|
2020-11-11 11:05:24 +01:00
|
|
|
#undef OPTION_WITH_MARSHALLING
|
2020-12-22 15:00:31 +01:00
|
|
|
|
|
|
|
return Success;
|
2020-05-11 11:42:38 +01:00
|
|
|
}
|
|
|
|
|
2011-12-23 03:05:38 +00:00
|
|
|
bool CompilerInvocation::CreateFromArgs(CompilerInvocation &Res,
|
2019-08-27 22:13:31 +00:00
|
|
|
ArrayRef<const char *> CommandLineArgs,
|
2020-06-18 08:57:50 -04:00
|
|
|
DiagnosticsEngine &Diags,
|
|
|
|
const char *Argv0) {
|
2011-12-23 03:05:38 +00:00
|
|
|
bool Success = true;
|
|
|
|
|
2009-12-01 03:16:53 +00:00
|
|
|
// Parse the arguments.
|
2019-09-04 14:26:28 +00:00
|
|
|
const OptTable &Opts = getDriverOptTable();
|
2013-08-02 20:16:22 +00:00
|
|
|
const unsigned IncludedFlagsBitmask = options::CC1Option;
|
2009-12-01 03:16:53 +00:00
|
|
|
unsigned MissingArgIndex, MissingArgCount;
|
2019-09-04 14:26:28 +00:00
|
|
|
InputArgList Args = Opts.ParseArgs(CommandLineArgs, MissingArgIndex,
|
|
|
|
MissingArgCount, IncludedFlagsBitmask);
|
2016-04-08 17:42:32 +00:00
|
|
|
LangOptions &LangOpts = *Res.getLangOpts();
|
2009-12-01 03:16:53 +00:00
|
|
|
|
|
|
|
// Check for missing argument error.
|
2011-12-23 03:05:38 +00:00
|
|
|
if (MissingArgCount) {
|
2009-12-01 03:16:53 +00:00
|
|
|
Diags.Report(diag::err_drv_missing_argument)
|
2015-06-22 22:07:27 +00:00
|
|
|
<< Args.getArgString(MissingArgIndex) << MissingArgCount;
|
2011-12-23 03:05:38 +00:00
|
|
|
Success = false;
|
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
|
|
|
|
// Issue errors on unknown arguments.
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto *A : Args.filtered(OPT_UNKNOWN)) {
|
[Driver] Suggest correctly spelled driver options
Summary:
Depends on https://reviews.llvm.org/D41732.
Utilities such as `opt`, when invoked with arguments that are very
nearly spelled correctly, suggest the correctly spelled options:
```
bin/opt -hel
opt: Unknown command line argument '-hel'. Try: 'bin/opt -help'
opt: Did you mean '-help'?
```
Clang, on the other hand, prior to this commit, does not:
```
bin/clang -hel
clang-6.0: error: unknown argument: '-hel'
```
This commit makes use of the new libLLVMOption API from
https://reviews.llvm.org/D41732 in order to provide correct suggestions:
```
bin/clang -hel
clang-6.0: error: unknown argument: '-hel', did you mean '-help'?
```
Test Plan: `check-clang`
Reviewers: yamaguchi, v.g.vassilev, teemperor, ruiu, bruno
Reviewed By: bruno
Subscribers: bruno, jroelofs, cfe-commits
Differential Revision: https://reviews.llvm.org/D41733
llvm-svn: 321917
2018-01-06 00:25:40 +00:00
|
|
|
auto ArgString = A->getAsString(Args);
|
|
|
|
std::string Nearest;
|
2019-09-04 14:26:28 +00:00
|
|
|
if (Opts.findNearest(ArgString, Nearest, IncludedFlagsBitmask) > 1)
|
[Driver] Suggest correctly spelled driver options
Summary:
Depends on https://reviews.llvm.org/D41732.
Utilities such as `opt`, when invoked with arguments that are very
nearly spelled correctly, suggest the correctly spelled options:
```
bin/opt -hel
opt: Unknown command line argument '-hel'. Try: 'bin/opt -help'
opt: Did you mean '-help'?
```
Clang, on the other hand, prior to this commit, does not:
```
bin/clang -hel
clang-6.0: error: unknown argument: '-hel'
```
This commit makes use of the new libLLVMOption API from
https://reviews.llvm.org/D41732 in order to provide correct suggestions:
```
bin/clang -hel
clang-6.0: error: unknown argument: '-hel', did you mean '-help'?
```
Test Plan: `check-clang`
Reviewers: yamaguchi, v.g.vassilev, teemperor, ruiu, bruno
Reviewed By: bruno
Subscribers: bruno, jroelofs, cfe-commits
Differential Revision: https://reviews.llvm.org/D41733
llvm-svn: 321917
2018-01-06 00:25:40 +00:00
|
|
|
Diags.Report(diag::err_drv_unknown_argument) << ArgString;
|
|
|
|
else
|
|
|
|
Diags.Report(diag::err_drv_unknown_argument_with_suggestion)
|
|
|
|
<< ArgString << Nearest;
|
2011-12-23 03:05:38 +00:00
|
|
|
Success = false;
|
|
|
|
}
|
2009-12-01 03:16:53 +00:00
|
|
|
|
2020-05-11 11:42:38 +01:00
|
|
|
Success &= Res.parseSimpleArgs(Args, Diags);
|
2020-06-30 14:25:23 +01:00
|
|
|
|
2015-06-22 22:07:27 +00:00
|
|
|
Success &= ParseAnalyzerArgs(*Res.getAnalyzerOpts(), Args, Diags);
|
|
|
|
ParseDependencyOutputArgs(Res.getDependencyOutputOpts(), Args);
|
2019-09-07 00:59:13 +00:00
|
|
|
if (!Res.getDependencyOutputOpts().OutputFile.empty() &&
|
|
|
|
Res.getDependencyOutputOpts().Targets.empty()) {
|
|
|
|
Diags.Report(diag::err_fe_dependency_file_requires_MT);
|
|
|
|
Success = false;
|
|
|
|
}
|
2020-03-31 21:30:14 -07:00
|
|
|
Success &= ParseDiagnosticArgs(Res.getDiagnosticOpts(), Args, &Diags,
|
|
|
|
/*DefaultDiagColor=*/false);
|
2016-04-08 17:42:32 +00:00
|
|
|
ParseCommentArgs(LangOpts.CommentOpts, Args);
|
2010-12-04 01:50:36 +00:00
|
|
|
// FIXME: We shouldn't have to pass the DashX option around here
|
2016-10-27 14:17:10 +00:00
|
|
|
InputKind DashX = ParseFrontendArgs(Res.getFrontendOpts(), Args, Diags,
|
|
|
|
LangOpts.IsHeaderFile);
|
2016-04-08 16:52:05 +00:00
|
|
|
ParseTargetArgs(Res.getTargetOpts(), Args, Diags);
|
2015-06-22 22:07:27 +00:00
|
|
|
Success &= ParseCodeGenArgs(Res.getCodeGenOpts(), Args, DashX, Diags,
|
2018-04-17 16:39:25 +00:00
|
|
|
Res.getTargetOpts(), Res.getFrontendOpts());
|
2017-03-14 23:07:49 +00:00
|
|
|
ParseHeaderSearchArgs(Res.getHeaderSearchOpts(), Args,
|
|
|
|
Res.getFileSystemOpts().WorkingDir);
|
2018-12-06 22:45:58 +00:00
|
|
|
llvm::Triple T(Res.getTargetOpts().Triple);
|
2017-04-26 18:57:40 +00:00
|
|
|
if (DashX.getFormat() == InputKind::Precompiled ||
|
2019-08-05 13:59:26 +00:00
|
|
|
DashX.getLanguage() == Language::LLVM_IR) {
|
2015-07-17 20:09:56 +00:00
|
|
|
// ObjCAAutoRefCount and Sanitize LangOpts are used to setup the
|
|
|
|
// PassManager in BackendUtil.cpp. They need to be initializd no matter
|
|
|
|
// what the input type is.
|
|
|
|
if (Args.hasArg(OPT_fobjc_arc))
|
2016-04-08 17:42:32 +00:00
|
|
|
LangOpts.ObjCAutoRefCount = 1;
|
|
|
|
// PIClevel and PIELevel are needed during code generation and this should be
|
|
|
|
// set regardless of the input type.
|
|
|
|
LangOpts.PICLevel = getLastArgIntValue(Args, OPT_pic_level, 0, Diags);
|
2015-07-17 20:09:56 +00:00
|
|
|
parseSanitizerKinds("-fsanitize=", Args.getAllArgValues(OPT_fsanitize_EQ),
|
2016-04-08 17:42:32 +00:00
|
|
|
Diags, LangOpts.Sanitize);
|
2015-07-17 20:09:56 +00:00
|
|
|
} else {
|
Misc typos fixes in ./lib folder
Summary: Found via `codespell -q 3 -I ../clang-whitelist.txt -L uint,importd,crasher,gonna,cant,ue,ons,orign,ned`
Reviewers: teemperor
Reviewed By: teemperor
Subscribers: teemperor, jholewinski, jvesely, nhaehnle, whisperity, jfb, cfe-commits
Differential Revision: https://reviews.llvm.org/D55475
llvm-svn: 348755
2018-12-10 12:37:46 +00:00
|
|
|
// Other LangOpts are only initialized when the input is not AST or LLVM IR.
|
2019-08-05 13:59:26 +00:00
|
|
|
// FIXME: Should we really be calling this for an Language::Asm input?
|
2016-06-20 19:26:00 +00:00
|
|
|
ParseLangArgs(LangOpts, Args, DashX, Res.getTargetOpts(),
|
2020-05-11 12:29:52 -07:00
|
|
|
Res.getPreprocessorOpts(), Diags);
|
2011-02-25 17:24:55 +00:00
|
|
|
if (Res.getFrontendOpts().ProgramAction == frontend::RewriteObjC)
|
2016-04-08 17:42:32 +00:00
|
|
|
LangOpts.ObjCExceptions = 1;
|
2018-12-06 22:45:58 +00:00
|
|
|
if (T.isOSDarwin() && DashX.isPreprocessed()) {
|
|
|
|
// Supress the darwin-specific 'stdlibcxx-not-found' diagnostic for
|
|
|
|
// preprocessed input as we don't expect it to be used with -std=libc++
|
|
|
|
// anyway.
|
|
|
|
Res.getDiagnosticOpts().Warnings.push_back("no-stdlibcxx-not-found");
|
|
|
|
}
|
2011-02-25 17:24:55 +00:00
|
|
|
}
|
2016-03-14 13:23:58 +00:00
|
|
|
|
2016-05-19 18:44:45 +00:00
|
|
|
if (LangOpts.CUDA) {
|
|
|
|
// During CUDA device-side compilation, the aux triple is the
|
|
|
|
// triple used for host compilation.
|
|
|
|
if (LangOpts.CUDAIsDevice)
|
|
|
|
Res.getTargetOpts().HostTriple = Res.getFrontendOpts().AuxTriple;
|
2016-04-29 23:05:19 +00:00
|
|
|
}
|
|
|
|
|
[OpenMP] Add support for auxiliary triple specification
Summary: Device offloading requires the specification of an additional flag containing the triple of the //other// architecture the code is being compiled on if such an architecture exists. If compiling for the host, the auxiliary triple flag will contain the triple describing the device and vice versa.
Reviewers: arpith-jacob, sfantao, caomhin, carlo.bertolli, ABataev, Hahnfeld, jlebar, hfinkel, tstellar
Reviewed By: Hahnfeld
Subscribers: rengolin, cfe-commits
Differential Revision: https://reviews.llvm.org/D29339
llvm-svn: 306689
2017-06-29 15:49:03 +00:00
|
|
|
// Set the triple of the host for OpenMP device compile.
|
|
|
|
if (LangOpts.OpenMPIsDevice)
|
|
|
|
Res.getTargetOpts().HostTriple = Res.getFrontendOpts().AuxTriple;
|
|
|
|
|
2016-03-15 09:41:39 +00:00
|
|
|
// FIXME: Override value name discarding when asan or msan is used because the
|
2016-03-14 13:23:58 +00:00
|
|
|
// backend passes depend on the name of the alloca in order to print out
|
|
|
|
// names.
|
2016-03-15 09:41:39 +00:00
|
|
|
Res.getCodeGenOpts().DiscardValueNames &=
|
2016-04-08 17:42:32 +00:00
|
|
|
!LangOpts.Sanitize.has(SanitizerKind::Address) &&
|
2018-09-07 09:21:09 +00:00
|
|
|
!LangOpts.Sanitize.has(SanitizerKind::KernelAddress) &&
|
|
|
|
!LangOpts.Sanitize.has(SanitizerKind::Memory) &&
|
|
|
|
!LangOpts.Sanitize.has(SanitizerKind::KernelMemory);
|
2016-03-14 13:23:58 +00:00
|
|
|
|
2018-01-09 21:26:47 +00:00
|
|
|
ParsePreprocessorArgs(Res.getPreprocessorOpts(), Args, Diags,
|
2017-06-16 20:13:39 +00:00
|
|
|
Res.getFrontendOpts().ProgramAction);
|
2015-06-22 22:07:27 +00:00
|
|
|
ParsePreprocessorOutputArgs(Res.getPreprocessorOutputOpts(), Args,
|
2013-01-30 01:52:57 +00:00
|
|
|
Res.getFrontendOpts().ProgramAction);
|
2016-07-28 19:26:30 +00:00
|
|
|
|
|
|
|
// Turn on -Wspir-compat for SPIR target.
|
2019-05-09 10:25:45 +00:00
|
|
|
if (T.isSPIR())
|
2016-07-28 19:26:30 +00:00
|
|
|
Res.getDiagnosticOpts().Warnings.push_back("spir-compat");
|
2017-10-16 16:50:27 +00:00
|
|
|
|
|
|
|
// If sanitizer is enabled, disable OPT_ffine_grained_bitfield_accesses.
|
|
|
|
if (Res.getCodeGenOpts().FineGrainedBitfieldAccesses &&
|
|
|
|
!Res.getLangOpts()->Sanitize.empty()) {
|
|
|
|
Res.getCodeGenOpts().FineGrainedBitfieldAccesses = false;
|
|
|
|
Diags.Report(diag::warn_drv_fine_grained_bitfield_accesses_ignored);
|
|
|
|
}
|
2020-06-18 08:57:50 -04:00
|
|
|
|
|
|
|
// Store the command-line for using in the CodeView backend.
|
|
|
|
Res.getCodeGenOpts().Argv0 = Argv0;
|
|
|
|
Res.getCodeGenOpts().CommandLineArgs = CommandLineArgs;
|
|
|
|
|
2021-01-05 17:00:46 +01:00
|
|
|
FixupInvocation(Res, Diags, Args);
|
2020-12-15 09:59:19 +01:00
|
|
|
|
2011-12-23 03:05:38 +00:00
|
|
|
return Success;
|
2009-12-01 03:16:53 +00:00
|
|
|
}
|
2011-09-13 23:15:45 +00:00
|
|
|
|
|
|
|
std::string CompilerInvocation::getModuleHash() const {
|
2013-03-27 16:47:18 +00:00
|
|
|
// Note: For QoI reasons, the things we use as a hash here should all be
|
|
|
|
// dumped via the -module-info flag.
|
2012-11-05 19:45:09 +00:00
|
|
|
using llvm::hash_code;
|
|
|
|
using llvm::hash_value;
|
|
|
|
using llvm::hash_combine;
|
2019-10-21 22:51:13 +00:00
|
|
|
using llvm::hash_combine_range;
|
2012-11-05 19:45:09 +00:00
|
|
|
|
2011-09-13 23:15:45 +00:00
|
|
|
// Start the signature with the compiler version.
|
2012-11-05 23:30:26 +00:00
|
|
|
// FIXME: We'd rather use something more cryptographically sound than
|
2012-11-05 19:45:09 +00:00
|
|
|
// CityHash, but this will do for now.
|
|
|
|
hash_code code = hash_value(getClangFullRepositoryVersion());
|
|
|
|
|
Make AST reading work better with LLVM_APPEND_VC_REV=NO
With LLVM_APPEND_VC_REV=NO, Modules/merge-lifetime-extended-temporary.cpp
would fail if it ran before a0f50d731639350c7a7 (which changed
the serialization format) and then after, for these reasons:
1. With LLVM_APPEND_VC_REV=NO, the module hash before and after the
change was the same.
2. Modules/merge-lifetime-extended-temporary.cpp is the only test
we have that uses -fmodule-cache-path=%t that
a) actually writes to the cache path
b) doesn't do `rm -rf %t` at the top of the test
So the old run would write a module file, and then the new run would
try to load it, but the serialized format changed.
Do several things to fix this:
1. Include clang::serialization::VERSION_MAJOR/VERSION_MINOR in
the module hash, so that when the AST format changes (...and
we remember to bump these), we use a different module cache dir.
2. Bump VERSION_MAJOR, since a0f50d731639350c7a7 changed the
on-disk format in a way that a gch file written before that change
can't be read after that change.
3. Add `rm -rf %t` to all tests that pass -fmodule-cache-path=%t.
This is unnecessary from a correctness PoV after 1 and 2,
but makes it so that we don't amass many cache dirs over time.
(Arguably, it also makes it so that the test suite doesn't catch
when we change the serialization format but don't bump
clang::serialization::VERSION_MAJOR/VERSION_MINOR; oh well.)
Differential Revision: https://reviews.llvm.org/D73202
2020-01-22 10:34:34 -05:00
|
|
|
// Also include the serialization version, in case LLVM_APPEND_VC_REV is off
|
|
|
|
// and getClangFullRepositoryVersion() doesn't include git revision.
|
|
|
|
code = hash_combine(code, serialization::VERSION_MAJOR,
|
|
|
|
serialization::VERSION_MINOR);
|
|
|
|
|
2011-09-13 23:15:45 +00:00
|
|
|
// Extend the signature with the language options
|
|
|
|
#define LANGOPT(Name, Bits, Default, Description) \
|
2012-11-05 19:45:09 +00:00
|
|
|
code = hash_combine(code, LangOpts->Name);
|
2011-09-13 23:15:45 +00:00
|
|
|
#define ENUM_LANGOPT(Name, Type, Bits, Default, Description) \
|
2012-11-05 19:45:09 +00:00
|
|
|
code = hash_combine(code, static_cast<unsigned>(LangOpts->get##Name()));
|
2011-09-13 23:15:45 +00:00
|
|
|
#define BENIGN_LANGOPT(Name, Bits, Default, Description)
|
|
|
|
#define BENIGN_ENUM_LANGOPT(Name, Type, Bits, Default, Description)
|
|
|
|
#include "clang/Basic/LangOptions.def"
|
2015-06-23 18:20:18 +00:00
|
|
|
|
|
|
|
for (StringRef Feature : LangOpts->ModuleFeatures)
|
|
|
|
code = hash_combine(code, Feature);
|
2015-08-05 15:08:53 +00:00
|
|
|
|
2020-07-07 17:13:02 -06:00
|
|
|
code = hash_combine(code, LangOpts->ObjCRuntime);
|
|
|
|
const auto &BCN = LangOpts->CommentOpts.BlockCommandNames;
|
|
|
|
code = hash_combine(code, hash_combine_range(BCN.begin(), BCN.end()));
|
|
|
|
|
2012-11-05 19:45:09 +00:00
|
|
|
// Extend the signature with the target options.
|
|
|
|
code = hash_combine(code, TargetOpts->Triple, TargetOpts->CPU,
|
2020-08-26 14:44:25 -07:00
|
|
|
TargetOpts->TuneCPU, TargetOpts->ABI);
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto &FeatureAsWritten : TargetOpts->FeaturesAsWritten)
|
|
|
|
code = hash_combine(code, FeatureAsWritten);
|
2011-09-13 23:15:45 +00:00
|
|
|
|
2011-09-14 15:55:12 +00:00
|
|
|
// Extend the signature with preprocessor options.
|
2012-11-05 19:45:09 +00:00
|
|
|
const PreprocessorOptions &ppOpts = getPreprocessorOpts();
|
2013-02-07 00:21:12 +00:00
|
|
|
const HeaderSearchOptions &hsOpts = getHeaderSearchOpts();
|
2012-11-05 19:45:09 +00:00
|
|
|
code = hash_combine(code, ppOpts.UsePredefines, ppOpts.DetailedRecord);
|
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto &I : getPreprocessorOpts().Macros) {
|
2013-02-07 00:21:12 +00:00
|
|
|
// If we're supposed to ignore this macro for the purposes of modules,
|
|
|
|
// don't put it into the hash.
|
|
|
|
if (!hsOpts.ModulesIgnoreMacros.empty()) {
|
|
|
|
// Check whether we're ignoring this macro.
|
2018-03-26 21:45:04 +00:00
|
|
|
StringRef MacroDef = I.first;
|
2016-10-21 21:45:01 +00:00
|
|
|
if (hsOpts.ModulesIgnoreMacros.count(
|
|
|
|
llvm::CachedHashString(MacroDef.split('=').first)))
|
2013-02-07 00:21:12 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-03-26 21:45:04 +00:00
|
|
|
code = hash_combine(code, I.first, I.second);
|
2011-10-17 14:55:37 +00:00
|
|
|
}
|
2012-11-05 19:45:09 +00:00
|
|
|
|
2016-01-12 21:01:56 +00:00
|
|
|
// Extend the signature with the sysroot and other header search options.
|
|
|
|
code = hash_combine(code, hsOpts.Sysroot,
|
|
|
|
hsOpts.ModuleFormat,
|
|
|
|
hsOpts.UseDebugInfo,
|
|
|
|
hsOpts.UseBuiltinIncludes,
|
2012-11-05 19:45:09 +00:00
|
|
|
hsOpts.UseStandardSystemIncludes,
|
|
|
|
hsOpts.UseStandardCXXIncludes,
|
2016-07-26 17:12:17 +00:00
|
|
|
hsOpts.UseLibcxx,
|
|
|
|
hsOpts.ModulesValidateDiagnosticOptions);
|
2014-05-04 05:27:24 +00:00
|
|
|
code = hash_combine(code, hsOpts.ResourceDir);
|
2012-11-05 19:45:09 +00:00
|
|
|
|
2019-10-21 22:51:13 +00:00
|
|
|
if (hsOpts.ModulesStrictContextHash) {
|
|
|
|
hash_code SHPC = hash_combine_range(hsOpts.SystemHeaderPrefixes.begin(),
|
|
|
|
hsOpts.SystemHeaderPrefixes.end());
|
|
|
|
hash_code UEC = hash_combine_range(hsOpts.UserEntries.begin(),
|
|
|
|
hsOpts.UserEntries.end());
|
|
|
|
code = hash_combine(code, hsOpts.SystemHeaderPrefixes.size(), SHPC,
|
|
|
|
hsOpts.UserEntries.size(), UEC);
|
|
|
|
|
|
|
|
const DiagnosticOptions &diagOpts = getDiagnosticOpts();
|
|
|
|
#define DIAGOPT(Name, Bits, Default) \
|
|
|
|
code = hash_combine(code, diagOpts.Name);
|
|
|
|
#define ENUM_DIAGOPT(Name, Type, Bits, Default) \
|
|
|
|
code = hash_combine(code, diagOpts.get##Name());
|
|
|
|
#include "clang/Basic/DiagnosticOptions.def"
|
|
|
|
#undef DIAGOPT
|
|
|
|
#undef ENUM_DIAGOPT
|
|
|
|
}
|
|
|
|
|
2014-03-03 08:12:05 +00:00
|
|
|
// Extend the signature with the user build path.
|
|
|
|
code = hash_combine(code, hsOpts.ModuleUserBuildPath);
|
|
|
|
|
2015-11-03 18:33:07 +00:00
|
|
|
// Extend the signature with the module file extensions.
|
|
|
|
const FrontendOptions &frontendOpts = getFrontendOpts();
|
2016-05-27 13:36:58 +00:00
|
|
|
for (const auto &ext : frontendOpts.ModuleFileExtensions) {
|
2015-11-03 18:33:07 +00:00
|
|
|
code = ext->hashExtension(code);
|
|
|
|
}
|
|
|
|
|
2018-11-29 22:33:09 +00:00
|
|
|
// When compiling with -gmodules, also hash -fdebug-prefix-map as it
|
|
|
|
// affects the debug info in the PCM.
|
|
|
|
if (getCodeGenOpts().DebugTypeExtRefs)
|
|
|
|
for (const auto &KeyValue : getCodeGenOpts().DebugPrefixMap)
|
|
|
|
code = hash_combine(code, KeyValue.first, KeyValue.second);
|
|
|
|
|
2017-06-01 20:01:01 +00:00
|
|
|
// Extend the signature with the enabled sanitizers, if at least one is
|
|
|
|
// enabled. Sanitizers which cannot affect AST generation aren't hashed.
|
|
|
|
SanitizerSet SanHash = LangOpts->Sanitize;
|
|
|
|
SanHash.clear(getPPTransparentSanitizers());
|
|
|
|
if (!SanHash.empty())
|
|
|
|
code = hash_combine(code, SanHash.Mask);
|
|
|
|
|
2012-11-05 19:45:09 +00:00
|
|
|
return llvm::APInt(64, code).toString(36, /*Signed=*/false);
|
2011-09-13 23:15:45 +00:00
|
|
|
}
|
2013-06-14 17:17:23 +00:00
|
|
|
|
2020-05-11 11:42:38 +01:00
|
|
|
void CompilerInvocation::generateCC1CommandLine(
|
|
|
|
SmallVectorImpl<const char *> &Args, StringAllocator SA) const {
|
2020-11-20 10:26:07 +01:00
|
|
|
// Capture the extracted value as a lambda argument to avoid potential issues
|
|
|
|
// with lifetime extension of the reference.
|
2020-11-11 11:05:24 +01:00
|
|
|
#define OPTION_WITH_MARSHALLING( \
|
2020-05-11 11:42:38 +01:00
|
|
|
PREFIX_TYPE, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
|
2021-01-07 10:14:48 +01:00
|
|
|
HELPTEXT, METAVAR, VALUES, SPELLING, SHOULD_PARSE, ALWAYS_EMIT, KEYPATH, \
|
|
|
|
DEFAULT_VALUE, IMPLIED_CHECK, IMPLIED_VALUE, NORMALIZER, DENORMALIZER, \
|
|
|
|
MERGER, EXTRACTOR, TABLE_INDEX) \
|
2020-11-20 10:26:07 +01:00
|
|
|
if ((FLAGS)&options::CC1Option) { \
|
|
|
|
[&](const auto &Extracted) { \
|
2020-12-15 09:59:19 +01:00
|
|
|
if (ALWAYS_EMIT || \
|
|
|
|
(Extracted != \
|
|
|
|
static_cast<decltype(this->KEYPATH)>( \
|
|
|
|
(IMPLIED_CHECK) ? (IMPLIED_VALUE) : (DEFAULT_VALUE)))) \
|
2021-01-05 17:00:46 +01:00
|
|
|
DENORMALIZER(Args, SPELLING, SA, Option::KIND##Class, TABLE_INDEX, \
|
|
|
|
Extracted); \
|
2020-11-20 10:26:07 +01:00
|
|
|
}(EXTRACTOR(this->KEYPATH)); \
|
2020-11-13 14:17:54 +01:00
|
|
|
}
|
|
|
|
|
2020-05-11 11:42:38 +01:00
|
|
|
#include "clang/Driver/Options.inc"
|
2020-11-11 11:05:24 +01:00
|
|
|
#undef OPTION_WITH_MARSHALLING
|
2020-05-11 11:42:38 +01:00
|
|
|
}
|
|
|
|
|
2018-10-10 13:27:25 +00:00
|
|
|
IntrusiveRefCntPtr<llvm::vfs::FileSystem>
|
2020-11-06 12:40:43 -05:00
|
|
|
clang::createVFSFromCompilerInvocation(const CompilerInvocation &CI,
|
|
|
|
DiagnosticsEngine &Diags) {
|
2018-10-10 13:27:25 +00:00
|
|
|
return createVFSFromCompilerInvocation(CI, Diags,
|
|
|
|
llvm::vfs::getRealFileSystem());
|
2017-05-23 11:37:52 +00:00
|
|
|
}
|
|
|
|
|
2020-11-06 12:40:43 -05:00
|
|
|
IntrusiveRefCntPtr<llvm::vfs::FileSystem>
|
|
|
|
clang::createVFSFromCompilerInvocation(
|
2018-10-10 13:27:25 +00:00
|
|
|
const CompilerInvocation &CI, DiagnosticsEngine &Diags,
|
|
|
|
IntrusiveRefCntPtr<llvm::vfs::FileSystem> BaseFS) {
|
2014-04-15 18:16:25 +00:00
|
|
|
if (CI.getHeaderSearchOpts().VFSOverlayFiles.empty())
|
2017-05-23 11:37:52 +00:00
|
|
|
return BaseFS;
|
2014-04-15 18:16:25 +00:00
|
|
|
|
2018-10-26 22:16:24 +00:00
|
|
|
IntrusiveRefCntPtr<llvm::vfs::FileSystem> Result = BaseFS;
|
2014-04-15 18:16:25 +00:00
|
|
|
// earlier vfs files are on the bottom
|
2018-03-26 21:45:04 +00:00
|
|
|
for (const auto &File : CI.getHeaderSearchOpts().VFSOverlayFiles) {
|
2014-07-06 17:43:24 +00:00
|
|
|
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> Buffer =
|
2018-10-26 22:16:24 +00:00
|
|
|
Result->getBufferForFile(File);
|
2014-07-06 17:43:24 +00:00
|
|
|
if (!Buffer) {
|
2014-04-15 18:16:25 +00:00
|
|
|
Diags.Report(diag::err_missing_vfs_overlay_file) << File;
|
2018-03-23 17:37:27 +00:00
|
|
|
continue;
|
2014-04-15 18:16:25 +00:00
|
|
|
}
|
|
|
|
|
2018-10-10 13:27:25 +00:00
|
|
|
IntrusiveRefCntPtr<llvm::vfs::FileSystem> FS = llvm::vfs::getVFSFromYAML(
|
2018-10-26 22:16:24 +00:00
|
|
|
std::move(Buffer.get()), /*DiagHandler*/ nullptr, File,
|
|
|
|
/*DiagContext*/ nullptr, Result);
|
|
|
|
if (!FS) {
|
2014-04-15 18:16:25 +00:00
|
|
|
Diags.Report(diag::err_invalid_vfs_overlay) << File;
|
2018-10-26 22:16:24 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
Result = FS;
|
2014-04-15 18:16:25 +00:00
|
|
|
}
|
2018-10-26 22:16:24 +00:00
|
|
|
return Result;
|
2013-06-14 17:17:23 +00:00
|
|
|
}
|