//===- Assumptions.cpp ------ Collection of helpers for assumptions -------===// // // 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 implements helper functions for accessing assumption infomration // inside of the "llvm.assume" metadata. // //===----------------------------------------------------------------------===// #include "llvm/IR/Assumptions.h" #include "llvm/ADT/SetOperations.h" #include "llvm/ADT/StringExtras.h" #include "llvm/IR/Attributes.h" #include "llvm/IR/Function.h" #include "llvm/IR/InstrTypes.h" using namespace llvm; namespace { bool hasAssumption(const Attribute &A, const KnownAssumptionString &AssumptionStr) { if (!A.isValid()) return false; assert(A.isStringAttribute() && "Expected a string attribute!"); SmallVector Strings; A.getValueAsString().split(Strings, ","); return llvm::is_contained(Strings, AssumptionStr); } DenseSet getAssumptions(const Attribute &A) { if (!A.isValid()) return DenseSet(); assert(A.isStringAttribute() && "Expected a string attribute!"); DenseSet Assumptions; SmallVector Strings; A.getValueAsString().split(Strings, ","); for (StringRef Str : Strings) Assumptions.insert(Str); return Assumptions; } template bool addAssumptionsImpl(AttrSite &Site, const DenseSet &Assumptions) { if (Assumptions.empty()) return false; DenseSet CurAssumptions = getAssumptions(Site); if (!set_union(CurAssumptions, Assumptions)) return false; LLVMContext &Ctx = Site.getContext(); Site.addFnAttr(llvm::Attribute::get( Ctx, llvm::AssumptionAttrKey, llvm::join(CurAssumptions.begin(), CurAssumptions.end(), ","))); return true; } } // namespace bool llvm::hasAssumption(const Function &F, const KnownAssumptionString &AssumptionStr) { const Attribute &A = F.getFnAttribute(AssumptionAttrKey); return ::hasAssumption(A, AssumptionStr); } bool llvm::hasAssumption(const CallBase &CB, const KnownAssumptionString &AssumptionStr) { if (Function *F = CB.getCalledFunction()) if (hasAssumption(*F, AssumptionStr)) return true; const Attribute &A = CB.getFnAttr(AssumptionAttrKey); return ::hasAssumption(A, AssumptionStr); } DenseSet llvm::getAssumptions(const Function &F) { const Attribute &A = F.getFnAttribute(AssumptionAttrKey); return ::getAssumptions(A); } DenseSet llvm::getAssumptions(const CallBase &CB) { const Attribute &A = CB.getFnAttr(AssumptionAttrKey); return ::getAssumptions(A); } bool llvm::addAssumptions(Function &F, const DenseSet &Assumptions) { return ::addAssumptionsImpl(F, Assumptions); } bool llvm::addAssumptions(CallBase &CB, const DenseSet &Assumptions) { return ::addAssumptionsImpl(CB, Assumptions); } StringSet<> llvm::KnownAssumptionStrings({ "omp_no_openmp", // OpenMP 5.1 "omp_no_openmp_routines", // OpenMP 5.1 "omp_no_parallelism", // OpenMP 5.1 "ompx_spmd_amenable", // OpenMPOpt extension "ompx_no_call_asm", // OpenMPOpt extension });