xref: /openbsd-src/gnu/llvm/clang/lib/CodeGen/SanitizerMetadata.cpp (revision 12c855180aad702bbcca06e0398d774beeafb155)
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