1a9ac8606Spatrick //===--- SanitizerMetadata.cpp - Ignored entities for sanitizers ----------===//
2e5dd7070Spatrick //
3e5dd7070Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e5dd7070Spatrick // See https://llvm.org/LICENSE.txt for license information.
5e5dd7070Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e5dd7070Spatrick //
7e5dd7070Spatrick //===----------------------------------------------------------------------===//
8e5dd7070Spatrick //
9e5dd7070Spatrick // Class which emits metadata consumed by sanitizer instrumentation passes.
10e5dd7070Spatrick //
11e5dd7070Spatrick //===----------------------------------------------------------------------===//
12e5dd7070Spatrick #include "SanitizerMetadata.h"
13e5dd7070Spatrick #include "CodeGenModule.h"
14e5dd7070Spatrick #include "clang/AST/Attr.h"
15e5dd7070Spatrick #include "clang/AST/Type.h"
16ec727ea7Spatrick #include "clang/Basic/SourceManager.h"
17e5dd7070Spatrick #include "llvm/ADT/StringRef.h"
18e5dd7070Spatrick #include "llvm/IR/Constants.h"
19e5dd7070Spatrick
20e5dd7070Spatrick using namespace clang;
21e5dd7070Spatrick using namespace CodeGen;
22e5dd7070Spatrick
SanitizerMetadata(CodeGenModule & CGM)23e5dd7070Spatrick SanitizerMetadata::SanitizerMetadata(CodeGenModule &CGM) : CGM(CGM) {}
24e5dd7070Spatrick
isAsanHwasanOrMemTag(const SanitizerSet & SS)25e5dd7070Spatrick static bool isAsanHwasanOrMemTag(const SanitizerSet &SS) {
26e5dd7070Spatrick return SS.hasOneOf(SanitizerKind::Address | SanitizerKind::KernelAddress |
27*12c85518Srobert SanitizerKind::HWAddress | SanitizerKind::MemTag);
28e5dd7070Spatrick }
29e5dd7070Spatrick
expandKernelSanitizerMasks(SanitizerMask Mask)30*12c85518Srobert SanitizerMask expandKernelSanitizerMasks(SanitizerMask Mask) {
31*12c85518Srobert if (Mask & (SanitizerKind::Address | SanitizerKind::KernelAddress))
32*12c85518Srobert Mask |= SanitizerKind::Address | SanitizerKind::KernelAddress;
33*12c85518Srobert // Note: KHWASan doesn't support globals.
34*12c85518Srobert return Mask;
35*12c85518Srobert }
36*12c85518Srobert
reportGlobal(llvm::GlobalVariable * GV,SourceLocation Loc,StringRef Name,QualType Ty,SanitizerMask NoSanitizeAttrMask,bool IsDynInit)37*12c85518Srobert void SanitizerMetadata::reportGlobal(llvm::GlobalVariable *GV,
38e5dd7070Spatrick SourceLocation Loc, StringRef Name,
39*12c85518Srobert QualType Ty,
40*12c85518Srobert SanitizerMask NoSanitizeAttrMask,
41*12c85518Srobert bool IsDynInit) {
42*12c85518Srobert SanitizerSet FsanitizeArgument = CGM.getLangOpts().Sanitize;
43*12c85518Srobert if (!isAsanHwasanOrMemTag(FsanitizeArgument))
44e5dd7070Spatrick return;
45e5dd7070Spatrick
46*12c85518Srobert FsanitizeArgument.Mask = expandKernelSanitizerMasks(FsanitizeArgument.Mask);
47*12c85518Srobert NoSanitizeAttrMask = expandKernelSanitizerMasks(NoSanitizeAttrMask);
48*12c85518Srobert SanitizerSet NoSanitizeAttrSet = {NoSanitizeAttrMask &
49*12c85518Srobert FsanitizeArgument.Mask};
50*12c85518Srobert
51*12c85518Srobert llvm::GlobalVariable::SanitizerMetadata Meta;
52*12c85518Srobert if (GV->hasSanitizerMetadata())
53*12c85518Srobert Meta = GV->getSanitizerMetadata();
54*12c85518Srobert
55*12c85518Srobert Meta.NoAddress |= NoSanitizeAttrSet.hasOneOf(SanitizerKind::Address);
56*12c85518Srobert Meta.NoAddress |= CGM.isInNoSanitizeList(
57*12c85518Srobert FsanitizeArgument.Mask & SanitizerKind::Address, GV, Loc, Ty);
58*12c85518Srobert
59*12c85518Srobert Meta.NoHWAddress |= NoSanitizeAttrSet.hasOneOf(SanitizerKind::HWAddress);
60*12c85518Srobert Meta.NoHWAddress |= CGM.isInNoSanitizeList(
61*12c85518Srobert FsanitizeArgument.Mask & SanitizerKind::HWAddress, GV, Loc, Ty);
62*12c85518Srobert
63*12c85518Srobert Meta.Memtag |=
64*12c85518Srobert static_cast<bool>(FsanitizeArgument.Mask & SanitizerKind::MemtagGlobals);
65*12c85518Srobert Meta.Memtag &= !NoSanitizeAttrSet.hasOneOf(SanitizerKind::MemTag);
66*12c85518Srobert Meta.Memtag &= !CGM.isInNoSanitizeList(
67*12c85518Srobert FsanitizeArgument.Mask & SanitizerKind::MemTag, GV, Loc, Ty);
68*12c85518Srobert
69*12c85518Srobert Meta.IsDynInit = IsDynInit && !Meta.NoAddress &&
70*12c85518Srobert FsanitizeArgument.has(SanitizerKind::Address) &&
71*12c85518Srobert !CGM.isInNoSanitizeList(SanitizerKind::Address |
72*12c85518Srobert SanitizerKind::KernelAddress,
73*12c85518Srobert GV, Loc, Ty, "init");
74*12c85518Srobert
75*12c85518Srobert GV->setSanitizerMetadata(Meta);
76e5dd7070Spatrick }
77e5dd7070Spatrick
reportGlobal(llvm::GlobalVariable * GV,const VarDecl & D,bool IsDynInit)78*12c85518Srobert void SanitizerMetadata::reportGlobal(llvm::GlobalVariable *GV, const VarDecl &D,
79*12c85518Srobert bool IsDynInit) {
80e5dd7070Spatrick if (!isAsanHwasanOrMemTag(CGM.getLangOpts().Sanitize))
81e5dd7070Spatrick return;
82e5dd7070Spatrick std::string QualName;
83e5dd7070Spatrick llvm::raw_string_ostream OS(QualName);
84e5dd7070Spatrick D.printQualifiedName(OS);
85e5dd7070Spatrick
86*12c85518Srobert auto getNoSanitizeMask = [](const VarDecl &D) {
87*12c85518Srobert if (D.hasAttr<DisableSanitizerInstrumentationAttr>())
88*12c85518Srobert return SanitizerKind::All;
89*12c85518Srobert
90*12c85518Srobert SanitizerMask NoSanitizeMask;
91*12c85518Srobert for (auto *Attr : D.specific_attrs<NoSanitizeAttr>())
92*12c85518Srobert NoSanitizeMask |= Attr->getMask();
93*12c85518Srobert
94*12c85518Srobert return NoSanitizeMask;
95*12c85518Srobert };
96*12c85518Srobert
97*12c85518Srobert reportGlobal(GV, D.getLocation(), OS.str(), D.getType(), getNoSanitizeMask(D),
98*12c85518Srobert IsDynInit);
99e5dd7070Spatrick }
100e5dd7070Spatrick
disableSanitizerForGlobal(llvm::GlobalVariable * GV)101e5dd7070Spatrick void SanitizerMetadata::disableSanitizerForGlobal(llvm::GlobalVariable *GV) {
102*12c85518Srobert reportGlobal(GV, SourceLocation(), "", QualType(), SanitizerKind::All);
103e5dd7070Spatrick }
104e5dd7070Spatrick
disableSanitizerForInstruction(llvm::Instruction * I)105e5dd7070Spatrick void SanitizerMetadata::disableSanitizerForInstruction(llvm::Instruction *I) {
106*12c85518Srobert I->setMetadata(llvm::LLVMContext::MD_nosanitize,
107*12c85518Srobert llvm::MDNode::get(CGM.getLLVMContext(), std::nullopt));
108e5dd7070Spatrick }
109