10b57cec5SDimitry Andric //===--- SanitizerMetadata.cpp - Blacklist for sanitizers -----------------===// 20b57cec5SDimitry Andric // 30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 60b57cec5SDimitry Andric // 70b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 80b57cec5SDimitry Andric // 90b57cec5SDimitry Andric // Class which emits metadata consumed by sanitizer instrumentation passes. 100b57cec5SDimitry Andric // 110b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 120b57cec5SDimitry Andric #include "SanitizerMetadata.h" 130b57cec5SDimitry Andric #include "CodeGenModule.h" 14480093f4SDimitry Andric #include "clang/AST/Attr.h" 150b57cec5SDimitry Andric #include "clang/AST/Type.h" 16*5ffd83dbSDimitry Andric #include "clang/Basic/SourceManager.h" 170b57cec5SDimitry Andric #include "llvm/ADT/StringRef.h" 180b57cec5SDimitry Andric #include "llvm/IR/Constants.h" 190b57cec5SDimitry Andric 200b57cec5SDimitry Andric using namespace clang; 210b57cec5SDimitry Andric using namespace CodeGen; 220b57cec5SDimitry Andric 230b57cec5SDimitry Andric SanitizerMetadata::SanitizerMetadata(CodeGenModule &CGM) : CGM(CGM) {} 240b57cec5SDimitry Andric 250b57cec5SDimitry Andric static bool isAsanHwasanOrMemTag(const SanitizerSet& SS) { 260b57cec5SDimitry Andric return SS.hasOneOf(SanitizerKind::Address | SanitizerKind::KernelAddress | 270b57cec5SDimitry Andric SanitizerKind::HWAddress | SanitizerKind::KernelHWAddress | 280b57cec5SDimitry Andric SanitizerKind::MemTag); 290b57cec5SDimitry Andric } 300b57cec5SDimitry Andric 310b57cec5SDimitry Andric void SanitizerMetadata::reportGlobalToASan(llvm::GlobalVariable *GV, 320b57cec5SDimitry Andric SourceLocation Loc, StringRef Name, 330b57cec5SDimitry Andric QualType Ty, bool IsDynInit, 34*5ffd83dbSDimitry Andric bool IsExcluded) { 350b57cec5SDimitry Andric if (!isAsanHwasanOrMemTag(CGM.getLangOpts().Sanitize)) 360b57cec5SDimitry Andric return; 370b57cec5SDimitry Andric IsDynInit &= !CGM.isInSanitizerBlacklist(GV, Loc, Ty, "init"); 38*5ffd83dbSDimitry Andric IsExcluded |= CGM.isInSanitizerBlacklist(GV, Loc, Ty); 390b57cec5SDimitry Andric 400b57cec5SDimitry Andric llvm::Metadata *LocDescr = nullptr; 410b57cec5SDimitry Andric llvm::Metadata *GlobalName = nullptr; 420b57cec5SDimitry Andric llvm::LLVMContext &VMContext = CGM.getLLVMContext(); 43*5ffd83dbSDimitry Andric if (!IsExcluded) { 440b57cec5SDimitry Andric // Don't generate source location and global name if it is blacklisted - 450b57cec5SDimitry Andric // it won't be instrumented anyway. 460b57cec5SDimitry Andric LocDescr = getLocationMetadata(Loc); 470b57cec5SDimitry Andric if (!Name.empty()) 480b57cec5SDimitry Andric GlobalName = llvm::MDString::get(VMContext, Name); 490b57cec5SDimitry Andric } 500b57cec5SDimitry Andric 510b57cec5SDimitry Andric llvm::Metadata *GlobalMetadata[] = { 520b57cec5SDimitry Andric llvm::ConstantAsMetadata::get(GV), LocDescr, GlobalName, 530b57cec5SDimitry Andric llvm::ConstantAsMetadata::get( 540b57cec5SDimitry Andric llvm::ConstantInt::get(llvm::Type::getInt1Ty(VMContext), IsDynInit)), 550b57cec5SDimitry Andric llvm::ConstantAsMetadata::get(llvm::ConstantInt::get( 56*5ffd83dbSDimitry Andric llvm::Type::getInt1Ty(VMContext), IsExcluded))}; 570b57cec5SDimitry Andric 580b57cec5SDimitry Andric llvm::MDNode *ThisGlobal = llvm::MDNode::get(VMContext, GlobalMetadata); 590b57cec5SDimitry Andric llvm::NamedMDNode *AsanGlobals = 600b57cec5SDimitry Andric CGM.getModule().getOrInsertNamedMetadata("llvm.asan.globals"); 610b57cec5SDimitry Andric AsanGlobals->addOperand(ThisGlobal); 620b57cec5SDimitry Andric } 630b57cec5SDimitry Andric 640b57cec5SDimitry Andric void SanitizerMetadata::reportGlobalToASan(llvm::GlobalVariable *GV, 650b57cec5SDimitry Andric const VarDecl &D, bool IsDynInit) { 660b57cec5SDimitry Andric if (!isAsanHwasanOrMemTag(CGM.getLangOpts().Sanitize)) 670b57cec5SDimitry Andric return; 680b57cec5SDimitry Andric std::string QualName; 690b57cec5SDimitry Andric llvm::raw_string_ostream OS(QualName); 700b57cec5SDimitry Andric D.printQualifiedName(OS); 710b57cec5SDimitry Andric 72*5ffd83dbSDimitry Andric bool IsExcluded = false; 730b57cec5SDimitry Andric for (auto Attr : D.specific_attrs<NoSanitizeAttr>()) 740b57cec5SDimitry Andric if (Attr->getMask() & SanitizerKind::Address) 75*5ffd83dbSDimitry Andric IsExcluded = true; 760b57cec5SDimitry Andric reportGlobalToASan(GV, D.getLocation(), OS.str(), D.getType(), IsDynInit, 77*5ffd83dbSDimitry Andric IsExcluded); 780b57cec5SDimitry Andric } 790b57cec5SDimitry Andric 800b57cec5SDimitry Andric void SanitizerMetadata::disableSanitizerForGlobal(llvm::GlobalVariable *GV) { 810b57cec5SDimitry Andric // For now, just make sure the global is not modified by the ASan 820b57cec5SDimitry Andric // instrumentation. 830b57cec5SDimitry Andric if (isAsanHwasanOrMemTag(CGM.getLangOpts().Sanitize)) 840b57cec5SDimitry Andric reportGlobalToASan(GV, SourceLocation(), "", QualType(), false, true); 850b57cec5SDimitry Andric } 860b57cec5SDimitry Andric 870b57cec5SDimitry Andric void SanitizerMetadata::disableSanitizerForInstruction(llvm::Instruction *I) { 880b57cec5SDimitry Andric I->setMetadata(CGM.getModule().getMDKindID("nosanitize"), 890b57cec5SDimitry Andric llvm::MDNode::get(CGM.getLLVMContext(), None)); 900b57cec5SDimitry Andric } 910b57cec5SDimitry Andric 920b57cec5SDimitry Andric llvm::MDNode *SanitizerMetadata::getLocationMetadata(SourceLocation Loc) { 930b57cec5SDimitry Andric PresumedLoc PLoc = CGM.getContext().getSourceManager().getPresumedLoc(Loc); 940b57cec5SDimitry Andric if (!PLoc.isValid()) 950b57cec5SDimitry Andric return nullptr; 960b57cec5SDimitry Andric llvm::LLVMContext &VMContext = CGM.getLLVMContext(); 970b57cec5SDimitry Andric llvm::Metadata *LocMetadata[] = { 980b57cec5SDimitry Andric llvm::MDString::get(VMContext, PLoc.getFilename()), 990b57cec5SDimitry Andric llvm::ConstantAsMetadata::get(llvm::ConstantInt::get( 1000b57cec5SDimitry Andric llvm::Type::getInt32Ty(VMContext), PLoc.getLine())), 1010b57cec5SDimitry Andric llvm::ConstantAsMetadata::get(llvm::ConstantInt::get( 1020b57cec5SDimitry Andric llvm::Type::getInt32Ty(VMContext), PLoc.getColumn())), 1030b57cec5SDimitry Andric }; 1040b57cec5SDimitry Andric return llvm::MDNode::get(VMContext, LocMetadata); 1050b57cec5SDimitry Andric } 106