mirror of
https://github.com/llvm/llvm-project.git
synced 2025-05-02 10:16:06 +00:00

This reverts commit 928cad49beec0120686478f502899222e836b545 i.e., relands dccd27112722109d2e2f03e8da9ce8690f06e11b, with a fix to avoid use-after-scope by changing the lambda to capture by value.
198 lines
6.9 KiB
C++
198 lines
6.9 KiB
C++
//===- Sanitizers.cpp - C Language Family Language Options ----------------===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file defines the classes from Sanitizers.h
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "clang/Basic/Sanitizers.h"
|
|
#include "llvm/ADT/Hashing.h"
|
|
#include "llvm/ADT/SmallVector.h"
|
|
#include "llvm/ADT/StringSwitch.h"
|
|
#include "llvm/Support/Format.h"
|
|
#include "llvm/Support/MathExtras.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include <algorithm>
|
|
#include <cmath>
|
|
#include <optional>
|
|
|
|
using namespace clang;
|
|
|
|
static const double SanitizerMaskCutoffsEps = 0.000000001f;
|
|
|
|
void SanitizerMaskCutoffs::set(SanitizerMask K, double V) {
|
|
if (V < SanitizerMaskCutoffsEps && Cutoffs.empty())
|
|
return;
|
|
for (unsigned int i = 0; i < SanitizerKind::SO_Count; ++i)
|
|
if (K & SanitizerMask::bitPosToMask(i)) {
|
|
Cutoffs.resize(SanitizerKind::SO_Count);
|
|
Cutoffs[i] = V;
|
|
}
|
|
}
|
|
|
|
std::optional<double> SanitizerMaskCutoffs::operator[](unsigned Kind) const {
|
|
if (Cutoffs.empty() || Cutoffs[Kind] < SanitizerMaskCutoffsEps)
|
|
return std::nullopt;
|
|
|
|
return Cutoffs[Kind];
|
|
}
|
|
|
|
void SanitizerMaskCutoffs::clear(SanitizerMask K) { set(K, 0); }
|
|
|
|
std::optional<std::vector<unsigned>>
|
|
SanitizerMaskCutoffs::getAllScaled(unsigned ScalingFactor) const {
|
|
std::vector<unsigned> ScaledCutoffs;
|
|
|
|
bool AnyCutoff = false;
|
|
for (unsigned int i = 0; i < SanitizerKind::SO_Count; ++i) {
|
|
auto C = (*this)[i];
|
|
if (C.has_value()) {
|
|
ScaledCutoffs.push_back(lround(std::clamp(*C, 0.0, 1.0) * ScalingFactor));
|
|
AnyCutoff = true;
|
|
} else {
|
|
ScaledCutoffs.push_back(0);
|
|
}
|
|
}
|
|
|
|
if (AnyCutoff)
|
|
return ScaledCutoffs;
|
|
|
|
return std::nullopt;
|
|
}
|
|
|
|
// Once LLVM switches to C++17, the constexpr variables can be inline and we
|
|
// won't need this.
|
|
#define SANITIZER(NAME, ID) constexpr SanitizerMask SanitizerKind::ID;
|
|
#define SANITIZER_GROUP(NAME, ID, ALIAS) \
|
|
constexpr SanitizerMask SanitizerKind::ID; \
|
|
constexpr SanitizerMask SanitizerKind::ID##Group;
|
|
#include "clang/Basic/Sanitizers.def"
|
|
|
|
SanitizerMask clang::parseSanitizerValue(StringRef Value, bool AllowGroups) {
|
|
SanitizerMask ParsedKind = llvm::StringSwitch<SanitizerMask>(Value)
|
|
#define SANITIZER(NAME, ID) .Case(NAME, SanitizerKind::ID)
|
|
#define SANITIZER_GROUP(NAME, ID, ALIAS) \
|
|
.Case(NAME, AllowGroups ? SanitizerKind::ID##Group : SanitizerMask())
|
|
#include "clang/Basic/Sanitizers.def"
|
|
.Default(SanitizerMask());
|
|
return ParsedKind;
|
|
}
|
|
|
|
bool clang::parseSanitizerWeightedValue(StringRef Value, bool AllowGroups,
|
|
SanitizerMaskCutoffs &Cutoffs) {
|
|
SanitizerMask ParsedKind = llvm::StringSwitch<SanitizerMask>(Value)
|
|
#define SANITIZER(NAME, ID) .StartsWith(NAME "=", SanitizerKind::ID)
|
|
#define SANITIZER_GROUP(NAME, ID, ALIAS) \
|
|
.StartsWith(NAME "=", \
|
|
AllowGroups ? SanitizerKind::ID##Group : SanitizerMask())
|
|
#include "clang/Basic/Sanitizers.def"
|
|
.Default(SanitizerMask());
|
|
|
|
if (!ParsedKind)
|
|
return false;
|
|
auto [N, W] = Value.split('=');
|
|
double A;
|
|
if (W.getAsDouble(A))
|
|
return false;
|
|
A = std::clamp(A, 0.0, 1.0);
|
|
// AllowGroups is already taken into account for ParsedKind,
|
|
// hence we unconditionally expandSanitizerGroups.
|
|
Cutoffs.set(expandSanitizerGroups(ParsedKind), A);
|
|
return true;
|
|
}
|
|
|
|
void clang::serializeSanitizerSet(SanitizerSet Set,
|
|
SmallVectorImpl<StringRef> &Values) {
|
|
#define SANITIZER(NAME, ID) \
|
|
if (Set.has(SanitizerKind::ID)) \
|
|
Values.push_back(NAME);
|
|
#include "clang/Basic/Sanitizers.def"
|
|
}
|
|
|
|
void clang::serializeSanitizerMaskCutoffs(
|
|
const SanitizerMaskCutoffs &Cutoffs, SmallVectorImpl<std::string> &Values) {
|
|
#define SANITIZER(NAME, ID) \
|
|
if (auto C = Cutoffs[SanitizerKind::SO_##ID]) { \
|
|
std::string Str; \
|
|
llvm::raw_string_ostream OS(Str); \
|
|
OS << NAME "=" << llvm::format("%.8f", *C); \
|
|
Values.emplace_back(StringRef(Str).rtrim('0')); \
|
|
}
|
|
#include "clang/Basic/Sanitizers.def"
|
|
}
|
|
|
|
SanitizerMask clang::expandSanitizerGroups(SanitizerMask Kinds) {
|
|
#define SANITIZER(NAME, ID)
|
|
#define SANITIZER_GROUP(NAME, ID, ALIAS) \
|
|
if (Kinds & SanitizerKind::ID##Group) \
|
|
Kinds |= SanitizerKind::ID;
|
|
#include "clang/Basic/Sanitizers.def"
|
|
return Kinds;
|
|
}
|
|
|
|
llvm::hash_code SanitizerMask::hash_value() const {
|
|
return llvm::hash_combine_range(&maskLoToHigh[0], &maskLoToHigh[kNumElem]);
|
|
}
|
|
|
|
namespace clang {
|
|
unsigned SanitizerMask::countPopulation() const {
|
|
unsigned total = 0;
|
|
for (const auto &Val : maskLoToHigh)
|
|
total += llvm::popcount(Val);
|
|
return total;
|
|
}
|
|
|
|
llvm::hash_code hash_value(const clang::SanitizerMask &Arg) {
|
|
return Arg.hash_value();
|
|
}
|
|
|
|
StringRef AsanDtorKindToString(llvm::AsanDtorKind kind) {
|
|
switch (kind) {
|
|
case llvm::AsanDtorKind::None:
|
|
return "none";
|
|
case llvm::AsanDtorKind::Global:
|
|
return "global";
|
|
case llvm::AsanDtorKind::Invalid:
|
|
return "invalid";
|
|
}
|
|
return "invalid";
|
|
}
|
|
|
|
llvm::AsanDtorKind AsanDtorKindFromString(StringRef kindStr) {
|
|
return llvm::StringSwitch<llvm::AsanDtorKind>(kindStr)
|
|
.Case("none", llvm::AsanDtorKind::None)
|
|
.Case("global", llvm::AsanDtorKind::Global)
|
|
.Default(llvm::AsanDtorKind::Invalid);
|
|
}
|
|
|
|
StringRef AsanDetectStackUseAfterReturnModeToString(
|
|
llvm::AsanDetectStackUseAfterReturnMode mode) {
|
|
switch (mode) {
|
|
case llvm::AsanDetectStackUseAfterReturnMode::Always:
|
|
return "always";
|
|
case llvm::AsanDetectStackUseAfterReturnMode::Runtime:
|
|
return "runtime";
|
|
case llvm::AsanDetectStackUseAfterReturnMode::Never:
|
|
return "never";
|
|
case llvm::AsanDetectStackUseAfterReturnMode::Invalid:
|
|
return "invalid";
|
|
}
|
|
return "invalid";
|
|
}
|
|
|
|
llvm::AsanDetectStackUseAfterReturnMode
|
|
AsanDetectStackUseAfterReturnModeFromString(StringRef modeStr) {
|
|
return llvm::StringSwitch<llvm::AsanDetectStackUseAfterReturnMode>(modeStr)
|
|
.Case("always", llvm::AsanDetectStackUseAfterReturnMode::Always)
|
|
.Case("runtime", llvm::AsanDetectStackUseAfterReturnMode::Runtime)
|
|
.Case("never", llvm::AsanDetectStackUseAfterReturnMode::Never)
|
|
.Default(llvm::AsanDetectStackUseAfterReturnMode::Invalid);
|
|
}
|
|
|
|
} // namespace clang
|