1c953409fSJohannes Doerfert //===- OMPContext.cpp ------ Collection of helpers for OpenMP contexts ----===// 2c953409fSJohannes Doerfert // 3c953409fSJohannes Doerfert // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4c953409fSJohannes Doerfert // See https://llvm.org/LICENSE.txt for license information. 5c953409fSJohannes Doerfert // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6c953409fSJohannes Doerfert // 7c953409fSJohannes Doerfert //===----------------------------------------------------------------------===// 8c953409fSJohannes Doerfert /// \file 9c953409fSJohannes Doerfert /// 10c953409fSJohannes Doerfert /// This file implements helper functions and classes to deal with OpenMP 11c953409fSJohannes Doerfert /// contexts as used by `[begin/end] declare variant` and `metadirective`. 12c953409fSJohannes Doerfert /// 13c953409fSJohannes Doerfert //===----------------------------------------------------------------------===// 14c953409fSJohannes Doerfert 15c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPContext.h" 16d970a285SJohannes Doerfert #include "llvm/ADT/StringRef.h" 17c953409fSJohannes Doerfert #include "llvm/ADT/StringSwitch.h" 18c953409fSJohannes Doerfert #include "llvm/Support/Debug.h" 19c953409fSJohannes Doerfert #include "llvm/Support/raw_ostream.h" 2062c7f035SArchibald Elliott #include "llvm/TargetParser/Triple.h" 21c953409fSJohannes Doerfert 22c953409fSJohannes Doerfert #define DEBUG_TYPE "openmp-ir-builder" 23c953409fSJohannes Doerfert 24c953409fSJohannes Doerfert using namespace llvm; 25c953409fSJohannes Doerfert using namespace omp; 26c953409fSJohannes Doerfert 27c953409fSJohannes Doerfert OMPContext::OMPContext(bool IsDeviceCompilation, Triple TargetTriple) { 28c953409fSJohannes Doerfert // Add the appropriate device kind trait based on the triple and the 29c953409fSJohannes Doerfert // IsDeviceCompilation flag. 306b57d7f5SJohannes Doerfert ActiveTraits.set(unsigned(IsDeviceCompilation 316b57d7f5SJohannes Doerfert ? TraitProperty::device_kind_nohost 326b57d7f5SJohannes Doerfert : TraitProperty::device_kind_host)); 33c953409fSJohannes Doerfert switch (TargetTriple.getArch()) { 34c953409fSJohannes Doerfert case Triple::arm: 35c953409fSJohannes Doerfert case Triple::armeb: 36c953409fSJohannes Doerfert case Triple::aarch64: 37c953409fSJohannes Doerfert case Triple::aarch64_be: 38c953409fSJohannes Doerfert case Triple::aarch64_32: 39*bdf72706Swanglei case Triple::loongarch64: 40c953409fSJohannes Doerfert case Triple::mips: 41c953409fSJohannes Doerfert case Triple::mipsel: 42c953409fSJohannes Doerfert case Triple::mips64: 43c953409fSJohannes Doerfert case Triple::mips64el: 44c953409fSJohannes Doerfert case Triple::ppc: 4522883197SBrandon Bergren case Triple::ppcle: 46c953409fSJohannes Doerfert case Triple::ppc64: 47c953409fSJohannes Doerfert case Triple::ppc64le: 48c9062e8fSUlrich Weigand case Triple::systemz: 49c953409fSJohannes Doerfert case Triple::x86: 50c953409fSJohannes Doerfert case Triple::x86_64: 516b57d7f5SJohannes Doerfert ActiveTraits.set(unsigned(TraitProperty::device_kind_cpu)); 52c953409fSJohannes Doerfert break; 53c953409fSJohannes Doerfert case Triple::amdgcn: 54c953409fSJohannes Doerfert case Triple::nvptx: 55c953409fSJohannes Doerfert case Triple::nvptx64: 566b57d7f5SJohannes Doerfert ActiveTraits.set(unsigned(TraitProperty::device_kind_gpu)); 57c953409fSJohannes Doerfert break; 58c953409fSJohannes Doerfert default: 59c953409fSJohannes Doerfert break; 60c953409fSJohannes Doerfert } 61c953409fSJohannes Doerfert 62c953409fSJohannes Doerfert // Add the appropriate device architecture trait based on the triple. 63c953409fSJohannes Doerfert #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str) \ 64d970a285SJohannes Doerfert if (TraitSelector::TraitSelectorEnum == TraitSelector::device_arch) { \ 65c953409fSJohannes Doerfert if (TargetTriple.getArch() == TargetTriple.getArchTypeForLLVMName(Str)) \ 66d970a285SJohannes Doerfert ActiveTraits.set(unsigned(TraitProperty::Enum)); \ 6714030d71SKazu Hirata if (StringRef(Str) == "x86_64" && \ 680b0f2e6eSJohannes Doerfert TargetTriple.getArch() == Triple::x86_64) \ 69d970a285SJohannes Doerfert ActiveTraits.set(unsigned(TraitProperty::Enum)); \ 70d970a285SJohannes Doerfert } 71c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 72c953409fSJohannes Doerfert 73c953409fSJohannes Doerfert // TODO: What exactly do we want to see as device ISA trait? 74c953409fSJohannes Doerfert // The discussion on the list did not seem to have come to an agreed 75c953409fSJohannes Doerfert // upon solution. 76c953409fSJohannes Doerfert 77c953409fSJohannes Doerfert // LLVM is the "OpenMP vendor" but we could also interpret vendor as the 78c953409fSJohannes Doerfert // target vendor. 796b57d7f5SJohannes Doerfert ActiveTraits.set(unsigned(TraitProperty::implementation_vendor_llvm)); 80c953409fSJohannes Doerfert 81c953409fSJohannes Doerfert // The user condition true is accepted but not false. 826b57d7f5SJohannes Doerfert ActiveTraits.set(unsigned(TraitProperty::user_condition_true)); 83c953409fSJohannes Doerfert 84c953409fSJohannes Doerfert // This is for sure some device. 856b57d7f5SJohannes Doerfert ActiveTraits.set(unsigned(TraitProperty::device_kind_any)); 86c953409fSJohannes Doerfert 87c953409fSJohannes Doerfert LLVM_DEBUG({ 88c953409fSJohannes Doerfert dbgs() << "[" << DEBUG_TYPE 89c953409fSJohannes Doerfert << "] New OpenMP context with the following properties:\n"; 9099913ef3SFlorian Hahn for (unsigned Bit : ActiveTraits.set_bits()) { 9199913ef3SFlorian Hahn TraitProperty Property = TraitProperty(Bit); 92c953409fSJohannes Doerfert dbgs() << "\t " << getOpenMPContextTraitPropertyFullName(Property) 93c953409fSJohannes Doerfert << "\n"; 946b57d7f5SJohannes Doerfert } 95c953409fSJohannes Doerfert }); 96c953409fSJohannes Doerfert } 97c953409fSJohannes Doerfert 98c953409fSJohannes Doerfert /// Return true if \p C0 is a subset of \p C1. Note that both arrays are 99c953409fSJohannes Doerfert /// expected to be sorted. 100c953409fSJohannes Doerfert template <typename T> static bool isSubset(ArrayRef<T> C0, ArrayRef<T> C1) { 101c953409fSJohannes Doerfert #ifdef EXPENSIVE_CHECKS 1021647ff6eSGeorgii Rymar assert(llvm::is_sorted(C0) && llvm::is_sorted(C1) && 1031647ff6eSGeorgii Rymar "Expected sorted arrays!"); 104c953409fSJohannes Doerfert #endif 105c953409fSJohannes Doerfert if (C0.size() > C1.size()) 106c953409fSJohannes Doerfert return false; 107c953409fSJohannes Doerfert auto It0 = C0.begin(), End0 = C0.end(); 108c953409fSJohannes Doerfert auto It1 = C1.begin(), End1 = C1.end(); 109c953409fSJohannes Doerfert while (It0 != End0) { 110c953409fSJohannes Doerfert if (It1 == End1) 111c953409fSJohannes Doerfert return false; 112c953409fSJohannes Doerfert if (*It0 == *It1) { 113c953409fSJohannes Doerfert ++It0; 114c953409fSJohannes Doerfert ++It1; 115c953409fSJohannes Doerfert continue; 116c953409fSJohannes Doerfert } 117c953409fSJohannes Doerfert ++It0; 118c953409fSJohannes Doerfert } 119c953409fSJohannes Doerfert return true; 120c953409fSJohannes Doerfert } 121c953409fSJohannes Doerfert 122c953409fSJohannes Doerfert /// Return true if \p C0 is a strict subset of \p C1. Note that both arrays are 123c953409fSJohannes Doerfert /// expected to be sorted. 124c953409fSJohannes Doerfert template <typename T> 125c953409fSJohannes Doerfert static bool isStrictSubset(ArrayRef<T> C0, ArrayRef<T> C1) { 126c953409fSJohannes Doerfert if (C0.size() >= C1.size()) 127c953409fSJohannes Doerfert return false; 128c953409fSJohannes Doerfert return isSubset<T>(C0, C1); 129c953409fSJohannes Doerfert } 130c953409fSJohannes Doerfert 131c953409fSJohannes Doerfert static bool isStrictSubset(const VariantMatchInfo &VMI0, 132c953409fSJohannes Doerfert const VariantMatchInfo &VMI1) { 133c953409fSJohannes Doerfert // If all required traits are a strict subset and the ordered vectors storing 134c953409fSJohannes Doerfert // the construct traits, we say it is a strict subset. Note that the latter 135c953409fSJohannes Doerfert // relation is not required to be strict. 1366b57d7f5SJohannes Doerfert if (VMI0.RequiredTraits.count() >= VMI1.RequiredTraits.count()) 1376b57d7f5SJohannes Doerfert return false; 13899913ef3SFlorian Hahn for (unsigned Bit : VMI0.RequiredTraits.set_bits()) 13999913ef3SFlorian Hahn if (!VMI1.RequiredTraits.test(Bit)) 1406b57d7f5SJohannes Doerfert return false; 1416b57d7f5SJohannes Doerfert if (!isSubset<TraitProperty>(VMI0.ConstructTraits, VMI1.ConstructTraits)) 1426b57d7f5SJohannes Doerfert return false; 1436b57d7f5SJohannes Doerfert return true; 144c953409fSJohannes Doerfert } 145c953409fSJohannes Doerfert 146c953409fSJohannes Doerfert static int isVariantApplicableInContextHelper( 147c953409fSJohannes Doerfert const VariantMatchInfo &VMI, const OMPContext &Ctx, 148a19eb1deSJohannes Doerfert SmallVectorImpl<unsigned> *ConstructMatches, bool DeviceSetOnly) { 149a19eb1deSJohannes Doerfert 150a19eb1deSJohannes Doerfert // The match kind determines if we need to match all traits, any of the 151a19eb1deSJohannes Doerfert // traits, or none of the traits for it to be an applicable context. 152a19eb1deSJohannes Doerfert enum MatchKind { MK_ALL, MK_ANY, MK_NONE }; 153a19eb1deSJohannes Doerfert 154a19eb1deSJohannes Doerfert MatchKind MK = MK_ALL; 155a19eb1deSJohannes Doerfert // Determine the match kind the user wants, "all" is the default and provided 156a19eb1deSJohannes Doerfert // to the user only for completeness. 157a19eb1deSJohannes Doerfert if (VMI.RequiredTraits.test( 158a19eb1deSJohannes Doerfert unsigned(TraitProperty::implementation_extension_match_any))) 159a19eb1deSJohannes Doerfert MK = MK_ANY; 160a19eb1deSJohannes Doerfert if (VMI.RequiredTraits.test( 161a19eb1deSJohannes Doerfert unsigned(TraitProperty::implementation_extension_match_none))) 162a19eb1deSJohannes Doerfert MK = MK_NONE; 163a19eb1deSJohannes Doerfert 164a19eb1deSJohannes Doerfert // Helper to deal with a single property that was (not) found in the OpenMP 165a19eb1deSJohannes Doerfert // context based on the match kind selected by the user via 166a19eb1deSJohannes Doerfert // `implementation={extensions(match_[all,any,none])}' 167a19eb1deSJohannes Doerfert auto HandleTrait = [MK](TraitProperty Property, 16877c90c8cSKazu Hirata bool WasFound) -> std::optional<bool> /* Result */ { 169a19eb1deSJohannes Doerfert // For kind "any" a single match is enough but we ignore non-matched 170a19eb1deSJohannes Doerfert // properties. 171a19eb1deSJohannes Doerfert if (MK == MK_ANY) { 17249ed3032SJohannes Doerfert if (WasFound) 173a19eb1deSJohannes Doerfert return true; 174aadaafacSKazu Hirata return std::nullopt; 175a19eb1deSJohannes Doerfert } 176a19eb1deSJohannes Doerfert 177a19eb1deSJohannes Doerfert // In "all" or "none" mode we accept a matching or non-matching property 178a19eb1deSJohannes Doerfert // respectively and move on. We are not done yet! 179a19eb1deSJohannes Doerfert if ((WasFound && MK == MK_ALL) || (!WasFound && MK == MK_NONE)) 180aadaafacSKazu Hirata return std::nullopt; 181a19eb1deSJohannes Doerfert 182a19eb1deSJohannes Doerfert // We missed a property, provide some debug output and indicate failure. 183a19eb1deSJohannes Doerfert LLVM_DEBUG({ 184a19eb1deSJohannes Doerfert if (MK == MK_ALL) 185a19eb1deSJohannes Doerfert dbgs() << "[" << DEBUG_TYPE << "] Property " 186ee05167cSJohannes Doerfert << getOpenMPContextTraitPropertyName(Property, "") 187a19eb1deSJohannes Doerfert << " was not in the OpenMP context but match kind is all.\n"; 188a19eb1deSJohannes Doerfert if (MK == MK_NONE) 189a19eb1deSJohannes Doerfert dbgs() << "[" << DEBUG_TYPE << "] Property " 190ee05167cSJohannes Doerfert << getOpenMPContextTraitPropertyName(Property, "") 191a19eb1deSJohannes Doerfert << " was in the OpenMP context but match kind is none.\n"; 192a19eb1deSJohannes Doerfert }); 193a19eb1deSJohannes Doerfert return false; 194a19eb1deSJohannes Doerfert }; 195c953409fSJohannes Doerfert 19699913ef3SFlorian Hahn for (unsigned Bit : VMI.RequiredTraits.set_bits()) { 19799913ef3SFlorian Hahn TraitProperty Property = TraitProperty(Bit); 198a19eb1deSJohannes Doerfert if (DeviceSetOnly && 199a19eb1deSJohannes Doerfert getOpenMPContextTraitSetForProperty(Property) != TraitSet::device) 200a19eb1deSJohannes Doerfert continue; 201a19eb1deSJohannes Doerfert 202a19eb1deSJohannes Doerfert // So far all extensions are handled elsewhere, we skip them here as they 203a19eb1deSJohannes Doerfert // are not part of the OpenMP context. 204a19eb1deSJohannes Doerfert if (getOpenMPContextTraitSelectorForProperty(Property) == 205a19eb1deSJohannes Doerfert TraitSelector::implementation_extension) 206a19eb1deSJohannes Doerfert continue; 207c953409fSJohannes Doerfert 2086b57d7f5SJohannes Doerfert bool IsActiveTrait = Ctx.ActiveTraits.test(unsigned(Property)); 209ee05167cSJohannes Doerfert 210ee05167cSJohannes Doerfert // We overwrite the isa trait as it is actually up to the OMPContext hook to 211ee05167cSJohannes Doerfert // check the raw string(s). 212ee05167cSJohannes Doerfert if (Property == TraitProperty::device_isa___ANY) 213ee05167cSJohannes Doerfert IsActiveTrait = llvm::all_of(VMI.ISATraits, [&](StringRef RawString) { 214ee05167cSJohannes Doerfert return Ctx.matchesISATrait(RawString); 215ee05167cSJohannes Doerfert }); 216ee05167cSJohannes Doerfert 21777c90c8cSKazu Hirata if (std::optional<bool> Result = HandleTrait(Property, IsActiveTrait)) 218a3209b0fSFangrui Song return *Result; 219c953409fSJohannes Doerfert } 220c953409fSJohannes Doerfert 221a19eb1deSJohannes Doerfert if (!DeviceSetOnly) { 222a19eb1deSJohannes Doerfert // We could use isSubset here but we also want to record the match 223a19eb1deSJohannes Doerfert // locations. 224c953409fSJohannes Doerfert unsigned ConstructIdx = 0, NoConstructTraits = Ctx.ConstructTraits.size(); 225c953409fSJohannes Doerfert for (TraitProperty Property : VMI.ConstructTraits) { 226c953409fSJohannes Doerfert assert(getOpenMPContextTraitSetForProperty(Property) == 227c953409fSJohannes Doerfert TraitSet::construct && 228c953409fSJohannes Doerfert "Variant context is ill-formed!"); 229c953409fSJohannes Doerfert 230c953409fSJohannes Doerfert // Verify the nesting. 231c953409fSJohannes Doerfert bool FoundInOrder = false; 232c953409fSJohannes Doerfert while (!FoundInOrder && ConstructIdx != NoConstructTraits) 233c953409fSJohannes Doerfert FoundInOrder = (Ctx.ConstructTraits[ConstructIdx++] == Property); 234c953409fSJohannes Doerfert if (ConstructMatches) 235c953409fSJohannes Doerfert ConstructMatches->push_back(ConstructIdx - 1); 236c953409fSJohannes Doerfert 23777c90c8cSKazu Hirata if (std::optional<bool> Result = HandleTrait(Property, FoundInOrder)) 238a3209b0fSFangrui Song return *Result; 239a19eb1deSJohannes Doerfert 240c953409fSJohannes Doerfert if (!FoundInOrder) { 241c953409fSJohannes Doerfert LLVM_DEBUG(dbgs() << "[" << DEBUG_TYPE << "] Construct property " 242ee05167cSJohannes Doerfert << getOpenMPContextTraitPropertyName(Property, "") 243c953409fSJohannes Doerfert << " was not nested properly.\n"); 244c953409fSJohannes Doerfert return false; 245c953409fSJohannes Doerfert } 246c953409fSJohannes Doerfert 247c953409fSJohannes Doerfert // TODO: Verify SIMD 248c953409fSJohannes Doerfert } 249c953409fSJohannes Doerfert 250c953409fSJohannes Doerfert assert(isSubset<TraitProperty>(VMI.ConstructTraits, Ctx.ConstructTraits) && 251c953409fSJohannes Doerfert "Broken invariant!"); 252a19eb1deSJohannes Doerfert } 253a19eb1deSJohannes Doerfert 254a19eb1deSJohannes Doerfert if (MK == MK_ANY) { 255a19eb1deSJohannes Doerfert LLVM_DEBUG(dbgs() << "[" << DEBUG_TYPE 256a19eb1deSJohannes Doerfert << "] None of the properties was in the OpenMP context " 257a19eb1deSJohannes Doerfert "but match kind is any.\n"); 258a19eb1deSJohannes Doerfert return false; 259a19eb1deSJohannes Doerfert } 260a19eb1deSJohannes Doerfert 261c953409fSJohannes Doerfert return true; 262c953409fSJohannes Doerfert } 263c953409fSJohannes Doerfert 264c953409fSJohannes Doerfert bool llvm::omp::isVariantApplicableInContext(const VariantMatchInfo &VMI, 265a19eb1deSJohannes Doerfert const OMPContext &Ctx, 266a19eb1deSJohannes Doerfert bool DeviceSetOnly) { 267a19eb1deSJohannes Doerfert return isVariantApplicableInContextHelper( 268a19eb1deSJohannes Doerfert VMI, Ctx, /* ConstructMatches */ nullptr, DeviceSetOnly); 269c953409fSJohannes Doerfert } 270c953409fSJohannes Doerfert 271c953409fSJohannes Doerfert static APInt getVariantMatchScore(const VariantMatchInfo &VMI, 272c953409fSJohannes Doerfert const OMPContext &Ctx, 273c953409fSJohannes Doerfert SmallVectorImpl<unsigned> &ConstructMatches) { 274c953409fSJohannes Doerfert APInt Score(64, 1); 275c953409fSJohannes Doerfert 276c953409fSJohannes Doerfert unsigned NoConstructTraits = VMI.ConstructTraits.size(); 27799913ef3SFlorian Hahn for (unsigned Bit : VMI.RequiredTraits.set_bits()) { 27899913ef3SFlorian Hahn TraitProperty Property = TraitProperty(Bit); 279c953409fSJohannes Doerfert // If there is a user score attached, use it. 280c953409fSJohannes Doerfert if (VMI.ScoreMap.count(Property)) { 281c953409fSJohannes Doerfert const APInt &UserScore = VMI.ScoreMap.lookup(Property); 282c953409fSJohannes Doerfert assert(UserScore.uge(0) && "Expect non-negative user scores!"); 283c953409fSJohannes Doerfert Score += UserScore.getZExtValue(); 284c953409fSJohannes Doerfert continue; 285c953409fSJohannes Doerfert } 286c953409fSJohannes Doerfert 287c953409fSJohannes Doerfert switch (getOpenMPContextTraitSetForProperty(Property)) { 288c953409fSJohannes Doerfert case TraitSet::construct: 289c953409fSJohannes Doerfert // We handle the construct traits later via the VMI.ConstructTraits 290c953409fSJohannes Doerfert // container. 291c953409fSJohannes Doerfert continue; 292c953409fSJohannes Doerfert case TraitSet::implementation: 293c953409fSJohannes Doerfert // No effect on the score (implementation defined). 294c953409fSJohannes Doerfert continue; 295c953409fSJohannes Doerfert case TraitSet::user: 296c953409fSJohannes Doerfert // No effect on the score. 297c953409fSJohannes Doerfert continue; 298c953409fSJohannes Doerfert case TraitSet::device: 299c953409fSJohannes Doerfert // Handled separately below. 300c953409fSJohannes Doerfert break; 301c953409fSJohannes Doerfert case TraitSet::invalid: 302c953409fSJohannes Doerfert llvm_unreachable("Unknown trait set is not to be used!"); 303c953409fSJohannes Doerfert } 304c953409fSJohannes Doerfert 305c953409fSJohannes Doerfert // device={kind(any)} is "as if" no kind selector was specified. 306c953409fSJohannes Doerfert if (Property == TraitProperty::device_kind_any) 307c953409fSJohannes Doerfert continue; 308c953409fSJohannes Doerfert 309c953409fSJohannes Doerfert switch (getOpenMPContextTraitSelectorForProperty(Property)) { 310c953409fSJohannes Doerfert case TraitSelector::device_kind: 311ae5d3e8cSSimon Pilgrim Score += (1ULL << (NoConstructTraits + 0)); 312c953409fSJohannes Doerfert continue; 313c953409fSJohannes Doerfert case TraitSelector::device_arch: 314ae5d3e8cSSimon Pilgrim Score += (1ULL << (NoConstructTraits + 1)); 315c953409fSJohannes Doerfert continue; 316c953409fSJohannes Doerfert case TraitSelector::device_isa: 317ae5d3e8cSSimon Pilgrim Score += (1ULL << (NoConstructTraits + 2)); 318c953409fSJohannes Doerfert continue; 319c953409fSJohannes Doerfert default: 320c953409fSJohannes Doerfert continue; 321c953409fSJohannes Doerfert } 322c953409fSJohannes Doerfert } 323c953409fSJohannes Doerfert 324c953409fSJohannes Doerfert unsigned ConstructIdx = 0; 325c953409fSJohannes Doerfert assert(NoConstructTraits == ConstructMatches.size() && 326c953409fSJohannes Doerfert "Mismatch in the construct traits!"); 327c953409fSJohannes Doerfert for (TraitProperty Property : VMI.ConstructTraits) { 328c953409fSJohannes Doerfert assert(getOpenMPContextTraitSetForProperty(Property) == 329c953409fSJohannes Doerfert TraitSet::construct && 330c953409fSJohannes Doerfert "Ill-formed variant match info!"); 3317b6e49a2SDmitri Gribenko (void)Property; 332c953409fSJohannes Doerfert // ConstructMatches is the position p - 1 and we need 2^(p-1). 333ae5d3e8cSSimon Pilgrim Score += (1ULL << ConstructMatches[ConstructIdx++]); 334c953409fSJohannes Doerfert } 335c953409fSJohannes Doerfert 336c953409fSJohannes Doerfert LLVM_DEBUG(dbgs() << "[" << DEBUG_TYPE << "] Variant has a score of " << Score 337c953409fSJohannes Doerfert << "\n"); 338c953409fSJohannes Doerfert return Score; 3392dc45bf3SMartin Storsjö } 340c953409fSJohannes Doerfert 341c953409fSJohannes Doerfert int llvm::omp::getBestVariantMatchForContext( 342c953409fSJohannes Doerfert const SmallVectorImpl<VariantMatchInfo> &VMIs, const OMPContext &Ctx) { 343c953409fSJohannes Doerfert 344c953409fSJohannes Doerfert APInt BestScore(64, 0); 345c953409fSJohannes Doerfert int BestVMIIdx = -1; 346c953409fSJohannes Doerfert const VariantMatchInfo *BestVMI = nullptr; 347c953409fSJohannes Doerfert 348c953409fSJohannes Doerfert for (unsigned u = 0, e = VMIs.size(); u < e; ++u) { 349c953409fSJohannes Doerfert const VariantMatchInfo &VMI = VMIs[u]; 350c953409fSJohannes Doerfert 351c953409fSJohannes Doerfert SmallVector<unsigned, 8> ConstructMatches; 352c953409fSJohannes Doerfert // If the variant is not applicable its not the best. 353a19eb1deSJohannes Doerfert if (!isVariantApplicableInContextHelper(VMI, Ctx, &ConstructMatches, 354a19eb1deSJohannes Doerfert /* DeviceSetOnly */ false)) 355c953409fSJohannes Doerfert continue; 356c953409fSJohannes Doerfert // Check if its clearly not the best. 357c953409fSJohannes Doerfert APInt Score = getVariantMatchScore(VMI, Ctx, ConstructMatches); 358c953409fSJohannes Doerfert if (Score.ult(BestScore)) 359c953409fSJohannes Doerfert continue; 360c953409fSJohannes Doerfert // Equal score need subset checks. 361c953409fSJohannes Doerfert if (Score.eq(BestScore)) { 362c953409fSJohannes Doerfert // Strict subset are never best. 363c953409fSJohannes Doerfert if (isStrictSubset(VMI, *BestVMI)) 364c953409fSJohannes Doerfert continue; 365c953409fSJohannes Doerfert // Same score and the current best is no strict subset so we keep it. 366c953409fSJohannes Doerfert if (!isStrictSubset(*BestVMI, VMI)) 367c953409fSJohannes Doerfert continue; 368c953409fSJohannes Doerfert } 369c953409fSJohannes Doerfert // New best found. 370c953409fSJohannes Doerfert BestVMI = &VMI; 371c953409fSJohannes Doerfert BestVMIIdx = u; 372c953409fSJohannes Doerfert BestScore = Score; 373c953409fSJohannes Doerfert } 374c953409fSJohannes Doerfert 375c953409fSJohannes Doerfert return BestVMIIdx; 376c953409fSJohannes Doerfert } 377c953409fSJohannes Doerfert 378c953409fSJohannes Doerfert TraitSet llvm::omp::getOpenMPContextTraitSetKind(StringRef S) { 379c953409fSJohannes Doerfert return StringSwitch<TraitSet>(S) 380c953409fSJohannes Doerfert #define OMP_TRAIT_SET(Enum, Str) .Case(Str, TraitSet::Enum) 381c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 382c953409fSJohannes Doerfert .Default(TraitSet::invalid); 383c953409fSJohannes Doerfert } 3841228d42dSJohannes Doerfert 3851228d42dSJohannes Doerfert TraitSet 3861228d42dSJohannes Doerfert llvm::omp::getOpenMPContextTraitSetForSelector(TraitSelector Selector) { 3871228d42dSJohannes Doerfert switch (Selector) { 3881228d42dSJohannes Doerfert #define OMP_TRAIT_SELECTOR(Enum, TraitSetEnum, Str, ReqProp) \ 3891228d42dSJohannes Doerfert case TraitSelector::Enum: \ 3901228d42dSJohannes Doerfert return TraitSet::TraitSetEnum; 3911228d42dSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 3921228d42dSJohannes Doerfert } 393b91c2673SJohannes Doerfert llvm_unreachable("Unknown trait selector!"); 3941228d42dSJohannes Doerfert } 395c953409fSJohannes Doerfert TraitSet 396c953409fSJohannes Doerfert llvm::omp::getOpenMPContextTraitSetForProperty(TraitProperty Property) { 397c953409fSJohannes Doerfert switch (Property) { 398c953409fSJohannes Doerfert #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str) \ 399c953409fSJohannes Doerfert case TraitProperty::Enum: \ 400c953409fSJohannes Doerfert return TraitSet::TraitSetEnum; 401c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 402c953409fSJohannes Doerfert } 4032dc45bf3SMartin Storsjö llvm_unreachable("Unknown trait set!"); 404c953409fSJohannes Doerfert } 405c953409fSJohannes Doerfert StringRef llvm::omp::getOpenMPContextTraitSetName(TraitSet Kind) { 406c953409fSJohannes Doerfert switch (Kind) { 407c953409fSJohannes Doerfert #define OMP_TRAIT_SET(Enum, Str) \ 408c953409fSJohannes Doerfert case TraitSet::Enum: \ 409c953409fSJohannes Doerfert return Str; 410c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 411c953409fSJohannes Doerfert } 412c953409fSJohannes Doerfert llvm_unreachable("Unknown trait set!"); 413c953409fSJohannes Doerfert } 414c953409fSJohannes Doerfert 415c953409fSJohannes Doerfert TraitSelector llvm::omp::getOpenMPContextTraitSelectorKind(StringRef S) { 416c953409fSJohannes Doerfert return StringSwitch<TraitSelector>(S) 417c953409fSJohannes Doerfert #define OMP_TRAIT_SELECTOR(Enum, TraitSetEnum, Str, ReqProp) \ 418c953409fSJohannes Doerfert .Case(Str, TraitSelector::Enum) 419c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 420c953409fSJohannes Doerfert .Default(TraitSelector::invalid); 421c953409fSJohannes Doerfert } 422c953409fSJohannes Doerfert TraitSelector 423c953409fSJohannes Doerfert llvm::omp::getOpenMPContextTraitSelectorForProperty(TraitProperty Property) { 424c953409fSJohannes Doerfert switch (Property) { 425c953409fSJohannes Doerfert #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str) \ 426c953409fSJohannes Doerfert case TraitProperty::Enum: \ 427c953409fSJohannes Doerfert return TraitSelector::TraitSelectorEnum; 428c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 429c953409fSJohannes Doerfert } 4302dc45bf3SMartin Storsjö llvm_unreachable("Unknown trait set!"); 431c953409fSJohannes Doerfert } 432c953409fSJohannes Doerfert StringRef llvm::omp::getOpenMPContextTraitSelectorName(TraitSelector Kind) { 433c953409fSJohannes Doerfert switch (Kind) { 434c953409fSJohannes Doerfert #define OMP_TRAIT_SELECTOR(Enum, TraitSetEnum, Str, ReqProp) \ 435c953409fSJohannes Doerfert case TraitSelector::Enum: \ 436c953409fSJohannes Doerfert return Str; 437c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 438c953409fSJohannes Doerfert } 439c953409fSJohannes Doerfert llvm_unreachable("Unknown trait selector!"); 440c953409fSJohannes Doerfert } 441c953409fSJohannes Doerfert 442ee05167cSJohannes Doerfert TraitProperty llvm::omp::getOpenMPContextTraitPropertyKind( 443ee05167cSJohannes Doerfert TraitSet Set, TraitSelector Selector, StringRef S) { 444ee05167cSJohannes Doerfert // Special handling for `device={isa(...)}` as we accept anything here. It is 445ee05167cSJohannes Doerfert // up to the target to decide if the feature is available. 446ee05167cSJohannes Doerfert if (Set == TraitSet::device && Selector == TraitSelector::device_isa) 447ee05167cSJohannes Doerfert return TraitProperty::device_isa___ANY; 448c953409fSJohannes Doerfert #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str) \ 4491228d42dSJohannes Doerfert if (Set == TraitSet::TraitSetEnum && Str == S) \ 450c953409fSJohannes Doerfert return TraitProperty::Enum; 451c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 452c953409fSJohannes Doerfert return TraitProperty::invalid; 453c953409fSJohannes Doerfert } 454c953409fSJohannes Doerfert TraitProperty 455c953409fSJohannes Doerfert llvm::omp::getOpenMPContextTraitPropertyForSelector(TraitSelector Selector) { 456c953409fSJohannes Doerfert return StringSwitch<TraitProperty>( 457c953409fSJohannes Doerfert getOpenMPContextTraitSelectorName(Selector)) 458c953409fSJohannes Doerfert #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str) \ 459c953409fSJohannes Doerfert .Case(Str, Selector == TraitSelector::TraitSelectorEnum \ 460c953409fSJohannes Doerfert ? TraitProperty::Enum \ 461c953409fSJohannes Doerfert : TraitProperty::invalid) 462c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 463c953409fSJohannes Doerfert .Default(TraitProperty::invalid); 464c953409fSJohannes Doerfert } 465ee05167cSJohannes Doerfert StringRef llvm::omp::getOpenMPContextTraitPropertyName(TraitProperty Kind, 466ee05167cSJohannes Doerfert StringRef RawString) { 467ee05167cSJohannes Doerfert if (Kind == TraitProperty::device_isa___ANY) 468ee05167cSJohannes Doerfert return RawString; 469c953409fSJohannes Doerfert switch (Kind) { 470c953409fSJohannes Doerfert #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str) \ 471c953409fSJohannes Doerfert case TraitProperty::Enum: \ 472c953409fSJohannes Doerfert return Str; 473c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 474c953409fSJohannes Doerfert } 475c953409fSJohannes Doerfert llvm_unreachable("Unknown trait property!"); 476c953409fSJohannes Doerfert } 477c953409fSJohannes Doerfert StringRef llvm::omp::getOpenMPContextTraitPropertyFullName(TraitProperty Kind) { 478c953409fSJohannes Doerfert switch (Kind) { 479c953409fSJohannes Doerfert #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str) \ 480c953409fSJohannes Doerfert case TraitProperty::Enum: \ 481c953409fSJohannes Doerfert return "(" #TraitSetEnum "," #TraitSelectorEnum "," Str ")"; 482c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 483c953409fSJohannes Doerfert } 484c953409fSJohannes Doerfert llvm_unreachable("Unknown trait property!"); 485c953409fSJohannes Doerfert } 486c953409fSJohannes Doerfert 487c953409fSJohannes Doerfert bool llvm::omp::isValidTraitSelectorForTraitSet(TraitSelector Selector, 488c953409fSJohannes Doerfert TraitSet Set, 489c953409fSJohannes Doerfert bool &AllowsTraitScore, 490c953409fSJohannes Doerfert bool &RequiresProperty) { 491c953409fSJohannes Doerfert AllowsTraitScore = Set != TraitSet::construct && Set != TraitSet::device; 492c953409fSJohannes Doerfert switch (Selector) { 493c953409fSJohannes Doerfert #define OMP_TRAIT_SELECTOR(Enum, TraitSetEnum, Str, ReqProp) \ 494c953409fSJohannes Doerfert case TraitSelector::Enum: \ 495c953409fSJohannes Doerfert RequiresProperty = ReqProp; \ 496c953409fSJohannes Doerfert return Set == TraitSet::TraitSetEnum; 497c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 498c953409fSJohannes Doerfert } 499c953409fSJohannes Doerfert llvm_unreachable("Unknown trait selector!"); 500c953409fSJohannes Doerfert } 501c953409fSJohannes Doerfert 502c953409fSJohannes Doerfert bool llvm::omp::isValidTraitPropertyForTraitSetAndSelector( 503c953409fSJohannes Doerfert TraitProperty Property, TraitSelector Selector, TraitSet Set) { 504c953409fSJohannes Doerfert switch (Property) { 505c953409fSJohannes Doerfert #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str) \ 506c953409fSJohannes Doerfert case TraitProperty::Enum: \ 507c953409fSJohannes Doerfert return Set == TraitSet::TraitSetEnum && \ 508c953409fSJohannes Doerfert Selector == TraitSelector::TraitSelectorEnum; 509c953409fSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 510c953409fSJohannes Doerfert } 511c953409fSJohannes Doerfert llvm_unreachable("Unknown trait property!"); 512c953409fSJohannes Doerfert } 5131228d42dSJohannes Doerfert 5141228d42dSJohannes Doerfert std::string llvm::omp::listOpenMPContextTraitSets() { 5151228d42dSJohannes Doerfert std::string S; 5161228d42dSJohannes Doerfert #define OMP_TRAIT_SET(Enum, Str) \ 517a8f03963SJohannes Doerfert if (StringRef(Str) != "invalid") \ 5181228d42dSJohannes Doerfert S.append("'").append(Str).append("'").append(" "); 5191228d42dSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 5201228d42dSJohannes Doerfert S.pop_back(); 5211228d42dSJohannes Doerfert return S; 5221228d42dSJohannes Doerfert } 5231228d42dSJohannes Doerfert 5241228d42dSJohannes Doerfert std::string llvm::omp::listOpenMPContextTraitSelectors(TraitSet Set) { 5251228d42dSJohannes Doerfert std::string S; 5261228d42dSJohannes Doerfert #define OMP_TRAIT_SELECTOR(Enum, TraitSetEnum, Str, ReqProp) \ 527a8f03963SJohannes Doerfert if (TraitSet::TraitSetEnum == Set && StringRef(Str) != "Invalid") \ 5281228d42dSJohannes Doerfert S.append("'").append(Str).append("'").append(" "); 5291228d42dSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 5301228d42dSJohannes Doerfert S.pop_back(); 5311228d42dSJohannes Doerfert return S; 5321228d42dSJohannes Doerfert } 5331228d42dSJohannes Doerfert 5341228d42dSJohannes Doerfert std::string 5351228d42dSJohannes Doerfert llvm::omp::listOpenMPContextTraitProperties(TraitSet Set, 5361228d42dSJohannes Doerfert TraitSelector Selector) { 5371228d42dSJohannes Doerfert std::string S; 5381228d42dSJohannes Doerfert #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str) \ 5391228d42dSJohannes Doerfert if (TraitSet::TraitSetEnum == Set && \ 5401a93285cSReid Kleckner TraitSelector::TraitSelectorEnum == Selector && \ 541a8f03963SJohannes Doerfert StringRef(Str) != "invalid") \ 5421228d42dSJohannes Doerfert S.append("'").append(Str).append("'").append(" "); 5431228d42dSJohannes Doerfert #include "llvm/Frontend/OpenMP/OMPKinds.def" 544095cecbeSJohannes Doerfert if (S.empty()) 545095cecbeSJohannes Doerfert return "<none>"; 5461228d42dSJohannes Doerfert S.pop_back(); 5471228d42dSJohannes Doerfert return S; 5481228d42dSJohannes Doerfert } 549