//===- 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 #include 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 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); } // 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(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(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 &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 &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(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(modeStr) .Case("always", llvm::AsanDetectStackUseAfterReturnMode::Always) .Case("runtime", llvm::AsanDetectStackUseAfterReturnMode::Runtime) .Case("never", llvm::AsanDetectStackUseAfterReturnMode::Never) .Default(llvm::AsanDetectStackUseAfterReturnMode::Invalid); } } // namespace clang