xref: /openbsd-src/gnu/llvm/llvm/lib/Frontend/OpenMP/OMPContext.cpp (revision d415bd752c734aee168c4ee86ff32e8cc249eb16)
1097a140dSpatrick //===- OMPContext.cpp ------ Collection of helpers for OpenMP contexts ----===//
2097a140dSpatrick //
3097a140dSpatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4097a140dSpatrick // See https://llvm.org/LICENSE.txt for license information.
5097a140dSpatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6097a140dSpatrick //
7097a140dSpatrick //===----------------------------------------------------------------------===//
8097a140dSpatrick /// \file
9097a140dSpatrick ///
10097a140dSpatrick /// This file implements helper functions and classes to deal with OpenMP
11097a140dSpatrick /// contexts as used by `[begin/end] declare variant` and `metadirective`.
12097a140dSpatrick ///
13097a140dSpatrick //===----------------------------------------------------------------------===//
14097a140dSpatrick 
15097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPContext.h"
1673471bf0Spatrick #include "llvm/ADT/StringRef.h"
17097a140dSpatrick #include "llvm/ADT/StringSwitch.h"
1873471bf0Spatrick #include "llvm/ADT/Triple.h"
19097a140dSpatrick #include "llvm/Support/Debug.h"
20097a140dSpatrick #include "llvm/Support/raw_ostream.h"
21097a140dSpatrick 
22097a140dSpatrick #define DEBUG_TYPE "openmp-ir-builder"
23097a140dSpatrick 
24097a140dSpatrick using namespace llvm;
25097a140dSpatrick using namespace omp;
26097a140dSpatrick 
OMPContext(bool IsDeviceCompilation,Triple TargetTriple)27097a140dSpatrick OMPContext::OMPContext(bool IsDeviceCompilation, Triple TargetTriple) {
28097a140dSpatrick   // Add the appropriate device kind trait based on the triple and the
29097a140dSpatrick   // IsDeviceCompilation flag.
30097a140dSpatrick   ActiveTraits.set(unsigned(IsDeviceCompilation
31097a140dSpatrick                                 ? TraitProperty::device_kind_nohost
32097a140dSpatrick                                 : TraitProperty::device_kind_host));
33097a140dSpatrick   switch (TargetTriple.getArch()) {
34097a140dSpatrick   case Triple::arm:
35097a140dSpatrick   case Triple::armeb:
36097a140dSpatrick   case Triple::aarch64:
37097a140dSpatrick   case Triple::aarch64_be:
38097a140dSpatrick   case Triple::aarch64_32:
39097a140dSpatrick   case Triple::mips:
40097a140dSpatrick   case Triple::mipsel:
41097a140dSpatrick   case Triple::mips64:
42097a140dSpatrick   case Triple::mips64el:
43097a140dSpatrick   case Triple::ppc:
4473471bf0Spatrick   case Triple::ppcle:
45097a140dSpatrick   case Triple::ppc64:
46097a140dSpatrick   case Triple::ppc64le:
47097a140dSpatrick   case Triple::x86:
48097a140dSpatrick   case Triple::x86_64:
49097a140dSpatrick     ActiveTraits.set(unsigned(TraitProperty::device_kind_cpu));
50097a140dSpatrick     break;
51097a140dSpatrick   case Triple::amdgcn:
52097a140dSpatrick   case Triple::nvptx:
53097a140dSpatrick   case Triple::nvptx64:
54097a140dSpatrick     ActiveTraits.set(unsigned(TraitProperty::device_kind_gpu));
55097a140dSpatrick     break;
56097a140dSpatrick   default:
57097a140dSpatrick     break;
58097a140dSpatrick   }
59097a140dSpatrick 
60097a140dSpatrick   // Add the appropriate device architecture trait based on the triple.
61097a140dSpatrick #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str)         \
6273471bf0Spatrick   if (TraitSelector::TraitSelectorEnum == TraitSelector::device_arch) {        \
63097a140dSpatrick     if (TargetTriple.getArch() == TargetTriple.getArchTypeForLLVMName(Str))    \
6473471bf0Spatrick       ActiveTraits.set(unsigned(TraitProperty::Enum));                         \
6573471bf0Spatrick     if (StringRef(Str) == StringRef("x86_64") &&                               \
6673471bf0Spatrick         TargetTriple.getArch() == Triple::x86_64)                              \
6773471bf0Spatrick       ActiveTraits.set(unsigned(TraitProperty::Enum));                         \
6873471bf0Spatrick   }
69097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
70097a140dSpatrick 
71097a140dSpatrick   // TODO: What exactly do we want to see as device ISA trait?
72097a140dSpatrick   //       The discussion on the list did not seem to have come to an agreed
73097a140dSpatrick   //       upon solution.
74097a140dSpatrick 
75097a140dSpatrick   // LLVM is the "OpenMP vendor" but we could also interpret vendor as the
76097a140dSpatrick   // target vendor.
77097a140dSpatrick   ActiveTraits.set(unsigned(TraitProperty::implementation_vendor_llvm));
78097a140dSpatrick 
79097a140dSpatrick   // The user condition true is accepted but not false.
80097a140dSpatrick   ActiveTraits.set(unsigned(TraitProperty::user_condition_true));
81097a140dSpatrick 
82097a140dSpatrick   // This is for sure some device.
83097a140dSpatrick   ActiveTraits.set(unsigned(TraitProperty::device_kind_any));
84097a140dSpatrick 
85097a140dSpatrick   LLVM_DEBUG({
86097a140dSpatrick     dbgs() << "[" << DEBUG_TYPE
87097a140dSpatrick            << "] New OpenMP context with the following properties:\n";
88097a140dSpatrick     for (unsigned Bit : ActiveTraits.set_bits()) {
89097a140dSpatrick       TraitProperty Property = TraitProperty(Bit);
90097a140dSpatrick       dbgs() << "\t " << getOpenMPContextTraitPropertyFullName(Property)
91097a140dSpatrick              << "\n";
92097a140dSpatrick     }
93097a140dSpatrick   });
94097a140dSpatrick }
95097a140dSpatrick 
96097a140dSpatrick /// Return true if \p C0 is a subset of \p C1. Note that both arrays are
97097a140dSpatrick /// expected to be sorted.
isSubset(ArrayRef<T> C0,ArrayRef<T> C1)98097a140dSpatrick template <typename T> static bool isSubset(ArrayRef<T> C0, ArrayRef<T> C1) {
99097a140dSpatrick #ifdef EXPENSIVE_CHECKS
100097a140dSpatrick   assert(llvm::is_sorted(C0) && llvm::is_sorted(C1) &&
101097a140dSpatrick          "Expected sorted arrays!");
102097a140dSpatrick #endif
103097a140dSpatrick   if (C0.size() > C1.size())
104097a140dSpatrick     return false;
105097a140dSpatrick   auto It0 = C0.begin(), End0 = C0.end();
106097a140dSpatrick   auto It1 = C1.begin(), End1 = C1.end();
107097a140dSpatrick   while (It0 != End0) {
108097a140dSpatrick     if (It1 == End1)
109097a140dSpatrick       return false;
110097a140dSpatrick     if (*It0 == *It1) {
111097a140dSpatrick       ++It0;
112097a140dSpatrick       ++It1;
113097a140dSpatrick       continue;
114097a140dSpatrick     }
115097a140dSpatrick     ++It0;
116097a140dSpatrick   }
117097a140dSpatrick   return true;
118097a140dSpatrick }
119097a140dSpatrick 
120097a140dSpatrick /// Return true if \p C0 is a strict subset of \p C1. Note that both arrays are
121097a140dSpatrick /// expected to be sorted.
122097a140dSpatrick template <typename T>
isStrictSubset(ArrayRef<T> C0,ArrayRef<T> C1)123097a140dSpatrick static bool isStrictSubset(ArrayRef<T> C0, ArrayRef<T> C1) {
124097a140dSpatrick   if (C0.size() >= C1.size())
125097a140dSpatrick     return false;
126097a140dSpatrick   return isSubset<T>(C0, C1);
127097a140dSpatrick }
128097a140dSpatrick 
isStrictSubset(const VariantMatchInfo & VMI0,const VariantMatchInfo & VMI1)129097a140dSpatrick static bool isStrictSubset(const VariantMatchInfo &VMI0,
130097a140dSpatrick                            const VariantMatchInfo &VMI1) {
131097a140dSpatrick   // If all required traits are a strict subset and the ordered vectors storing
132097a140dSpatrick   // the construct traits, we say it is a strict subset. Note that the latter
133097a140dSpatrick   // relation is not required to be strict.
134097a140dSpatrick   if (VMI0.RequiredTraits.count() >= VMI1.RequiredTraits.count())
135097a140dSpatrick     return false;
136097a140dSpatrick   for (unsigned Bit : VMI0.RequiredTraits.set_bits())
137097a140dSpatrick     if (!VMI1.RequiredTraits.test(Bit))
138097a140dSpatrick       return false;
139097a140dSpatrick   if (!isSubset<TraitProperty>(VMI0.ConstructTraits, VMI1.ConstructTraits))
140097a140dSpatrick     return false;
141097a140dSpatrick   return true;
142097a140dSpatrick }
143097a140dSpatrick 
isVariantApplicableInContextHelper(const VariantMatchInfo & VMI,const OMPContext & Ctx,SmallVectorImpl<unsigned> * ConstructMatches,bool DeviceSetOnly)144097a140dSpatrick static int isVariantApplicableInContextHelper(
145097a140dSpatrick     const VariantMatchInfo &VMI, const OMPContext &Ctx,
146097a140dSpatrick     SmallVectorImpl<unsigned> *ConstructMatches, bool DeviceSetOnly) {
147097a140dSpatrick 
148097a140dSpatrick   // The match kind determines if we need to match all traits, any of the
149097a140dSpatrick   // traits, or none of the traits for it to be an applicable context.
150097a140dSpatrick   enum MatchKind { MK_ALL, MK_ANY, MK_NONE };
151097a140dSpatrick 
152097a140dSpatrick   MatchKind MK = MK_ALL;
153097a140dSpatrick   // Determine the match kind the user wants, "all" is the default and provided
154097a140dSpatrick   // to the user only for completeness.
155097a140dSpatrick   if (VMI.RequiredTraits.test(
156097a140dSpatrick           unsigned(TraitProperty::implementation_extension_match_any)))
157097a140dSpatrick     MK = MK_ANY;
158097a140dSpatrick   if (VMI.RequiredTraits.test(
159097a140dSpatrick           unsigned(TraitProperty::implementation_extension_match_none)))
160097a140dSpatrick     MK = MK_NONE;
161097a140dSpatrick 
162097a140dSpatrick   // Helper to deal with a single property that was (not) found in the OpenMP
163097a140dSpatrick   // context based on the match kind selected by the user via
164097a140dSpatrick   // `implementation={extensions(match_[all,any,none])}'
165097a140dSpatrick   auto HandleTrait = [MK](TraitProperty Property,
166*d415bd75Srobert                           bool WasFound) -> std::optional<bool> /* Result */ {
167097a140dSpatrick     // For kind "any" a single match is enough but we ignore non-matched
168097a140dSpatrick     // properties.
169097a140dSpatrick     if (MK == MK_ANY) {
170097a140dSpatrick       if (WasFound)
171097a140dSpatrick         return true;
172*d415bd75Srobert       return std::nullopt;
173097a140dSpatrick     }
174097a140dSpatrick 
175097a140dSpatrick     // In "all" or "none" mode we accept a matching or non-matching property
176097a140dSpatrick     // respectively and move on. We are not done yet!
177097a140dSpatrick     if ((WasFound && MK == MK_ALL) || (!WasFound && MK == MK_NONE))
178*d415bd75Srobert       return std::nullopt;
179097a140dSpatrick 
180097a140dSpatrick     // We missed a property, provide some debug output and indicate failure.
181097a140dSpatrick     LLVM_DEBUG({
182097a140dSpatrick       if (MK == MK_ALL)
183097a140dSpatrick         dbgs() << "[" << DEBUG_TYPE << "] Property "
18473471bf0Spatrick                << getOpenMPContextTraitPropertyName(Property, "")
185097a140dSpatrick                << " was not in the OpenMP context but match kind is all.\n";
186097a140dSpatrick       if (MK == MK_NONE)
187097a140dSpatrick         dbgs() << "[" << DEBUG_TYPE << "] Property "
18873471bf0Spatrick                << getOpenMPContextTraitPropertyName(Property, "")
189097a140dSpatrick                << " was in the OpenMP context but match kind is none.\n";
190097a140dSpatrick     });
191097a140dSpatrick     return false;
192097a140dSpatrick   };
193097a140dSpatrick 
194097a140dSpatrick   for (unsigned Bit : VMI.RequiredTraits.set_bits()) {
195097a140dSpatrick     TraitProperty Property = TraitProperty(Bit);
196097a140dSpatrick     if (DeviceSetOnly &&
197097a140dSpatrick         getOpenMPContextTraitSetForProperty(Property) != TraitSet::device)
198097a140dSpatrick       continue;
199097a140dSpatrick 
200097a140dSpatrick     // So far all extensions are handled elsewhere, we skip them here as they
201097a140dSpatrick     // are not part of the OpenMP context.
202097a140dSpatrick     if (getOpenMPContextTraitSelectorForProperty(Property) ==
203097a140dSpatrick         TraitSelector::implementation_extension)
204097a140dSpatrick       continue;
205097a140dSpatrick 
206097a140dSpatrick     bool IsActiveTrait = Ctx.ActiveTraits.test(unsigned(Property));
20773471bf0Spatrick 
20873471bf0Spatrick     // We overwrite the isa trait as it is actually up to the OMPContext hook to
20973471bf0Spatrick     // check the raw string(s).
21073471bf0Spatrick     if (Property == TraitProperty::device_isa___ANY)
21173471bf0Spatrick       IsActiveTrait = llvm::all_of(VMI.ISATraits, [&](StringRef RawString) {
21273471bf0Spatrick         return Ctx.matchesISATrait(RawString);
21373471bf0Spatrick       });
21473471bf0Spatrick 
215*d415bd75Srobert     if (std::optional<bool> Result = HandleTrait(Property, IsActiveTrait))
216*d415bd75Srobert       return *Result;
217097a140dSpatrick   }
218097a140dSpatrick 
219097a140dSpatrick   if (!DeviceSetOnly) {
220097a140dSpatrick     // We could use isSubset here but we also want to record the match
221097a140dSpatrick     // locations.
222097a140dSpatrick     unsigned ConstructIdx = 0, NoConstructTraits = Ctx.ConstructTraits.size();
223097a140dSpatrick     for (TraitProperty Property : VMI.ConstructTraits) {
224097a140dSpatrick       assert(getOpenMPContextTraitSetForProperty(Property) ==
225097a140dSpatrick                  TraitSet::construct &&
226097a140dSpatrick              "Variant context is ill-formed!");
227097a140dSpatrick 
228097a140dSpatrick       // Verify the nesting.
229097a140dSpatrick       bool FoundInOrder = false;
230097a140dSpatrick       while (!FoundInOrder && ConstructIdx != NoConstructTraits)
231097a140dSpatrick         FoundInOrder = (Ctx.ConstructTraits[ConstructIdx++] == Property);
232097a140dSpatrick       if (ConstructMatches)
233097a140dSpatrick         ConstructMatches->push_back(ConstructIdx - 1);
234097a140dSpatrick 
235*d415bd75Srobert       if (std::optional<bool> Result = HandleTrait(Property, FoundInOrder))
236*d415bd75Srobert         return *Result;
237097a140dSpatrick 
238097a140dSpatrick       if (!FoundInOrder) {
239097a140dSpatrick         LLVM_DEBUG(dbgs() << "[" << DEBUG_TYPE << "] Construct property "
24073471bf0Spatrick                           << getOpenMPContextTraitPropertyName(Property, "")
241097a140dSpatrick                           << " was not nested properly.\n");
242097a140dSpatrick         return false;
243097a140dSpatrick       }
244097a140dSpatrick 
245097a140dSpatrick       // TODO: Verify SIMD
246097a140dSpatrick     }
247097a140dSpatrick 
248097a140dSpatrick     assert(isSubset<TraitProperty>(VMI.ConstructTraits, Ctx.ConstructTraits) &&
249097a140dSpatrick            "Broken invariant!");
250097a140dSpatrick   }
251097a140dSpatrick 
252097a140dSpatrick   if (MK == MK_ANY) {
253097a140dSpatrick     LLVM_DEBUG(dbgs() << "[" << DEBUG_TYPE
254097a140dSpatrick                       << "] None of the properties was in the OpenMP context "
255097a140dSpatrick                          "but match kind is any.\n");
256097a140dSpatrick     return false;
257097a140dSpatrick   }
258097a140dSpatrick 
259097a140dSpatrick   return true;
260097a140dSpatrick }
261097a140dSpatrick 
isVariantApplicableInContext(const VariantMatchInfo & VMI,const OMPContext & Ctx,bool DeviceSetOnly)262097a140dSpatrick bool llvm::omp::isVariantApplicableInContext(const VariantMatchInfo &VMI,
263097a140dSpatrick                                              const OMPContext &Ctx,
264097a140dSpatrick                                              bool DeviceSetOnly) {
265097a140dSpatrick   return isVariantApplicableInContextHelper(
266097a140dSpatrick       VMI, Ctx, /* ConstructMatches */ nullptr, DeviceSetOnly);
267097a140dSpatrick }
268097a140dSpatrick 
getVariantMatchScore(const VariantMatchInfo & VMI,const OMPContext & Ctx,SmallVectorImpl<unsigned> & ConstructMatches)269097a140dSpatrick static APInt getVariantMatchScore(const VariantMatchInfo &VMI,
270097a140dSpatrick                                   const OMPContext &Ctx,
271097a140dSpatrick                                   SmallVectorImpl<unsigned> &ConstructMatches) {
272097a140dSpatrick   APInt Score(64, 1);
273097a140dSpatrick 
274097a140dSpatrick   unsigned NoConstructTraits = VMI.ConstructTraits.size();
275097a140dSpatrick   for (unsigned Bit : VMI.RequiredTraits.set_bits()) {
276097a140dSpatrick     TraitProperty Property = TraitProperty(Bit);
277097a140dSpatrick     // If there is a user score attached, use it.
278097a140dSpatrick     if (VMI.ScoreMap.count(Property)) {
279097a140dSpatrick       const APInt &UserScore = VMI.ScoreMap.lookup(Property);
280097a140dSpatrick       assert(UserScore.uge(0) && "Expect non-negative user scores!");
281097a140dSpatrick       Score += UserScore.getZExtValue();
282097a140dSpatrick       continue;
283097a140dSpatrick     }
284097a140dSpatrick 
285097a140dSpatrick     switch (getOpenMPContextTraitSetForProperty(Property)) {
286097a140dSpatrick     case TraitSet::construct:
287097a140dSpatrick       // We handle the construct traits later via the VMI.ConstructTraits
288097a140dSpatrick       // container.
289097a140dSpatrick       continue;
290097a140dSpatrick     case TraitSet::implementation:
291097a140dSpatrick       // No effect on the score (implementation defined).
292097a140dSpatrick       continue;
293097a140dSpatrick     case TraitSet::user:
294097a140dSpatrick       // No effect on the score.
295097a140dSpatrick       continue;
296097a140dSpatrick     case TraitSet::device:
297097a140dSpatrick       // Handled separately below.
298097a140dSpatrick       break;
299097a140dSpatrick     case TraitSet::invalid:
300097a140dSpatrick       llvm_unreachable("Unknown trait set is not to be used!");
301097a140dSpatrick     }
302097a140dSpatrick 
303097a140dSpatrick     // device={kind(any)} is "as if" no kind selector was specified.
304097a140dSpatrick     if (Property == TraitProperty::device_kind_any)
305097a140dSpatrick       continue;
306097a140dSpatrick 
307097a140dSpatrick     switch (getOpenMPContextTraitSelectorForProperty(Property)) {
308097a140dSpatrick     case TraitSelector::device_kind:
309097a140dSpatrick       Score += (1ULL << (NoConstructTraits + 0));
310097a140dSpatrick       continue;
311097a140dSpatrick     case TraitSelector::device_arch:
312097a140dSpatrick       Score += (1ULL << (NoConstructTraits + 1));
313097a140dSpatrick       continue;
314097a140dSpatrick     case TraitSelector::device_isa:
315097a140dSpatrick       Score += (1ULL << (NoConstructTraits + 2));
316097a140dSpatrick       continue;
317097a140dSpatrick     default:
318097a140dSpatrick       continue;
319097a140dSpatrick     }
320097a140dSpatrick   }
321097a140dSpatrick 
322097a140dSpatrick   unsigned ConstructIdx = 0;
323097a140dSpatrick   assert(NoConstructTraits == ConstructMatches.size() &&
324097a140dSpatrick          "Mismatch in the construct traits!");
325097a140dSpatrick   for (TraitProperty Property : VMI.ConstructTraits) {
326097a140dSpatrick     assert(getOpenMPContextTraitSetForProperty(Property) ==
327097a140dSpatrick                TraitSet::construct &&
328097a140dSpatrick            "Ill-formed variant match info!");
329097a140dSpatrick     (void)Property;
330097a140dSpatrick     // ConstructMatches is the position p - 1 and we need 2^(p-1).
331097a140dSpatrick     Score += (1ULL << ConstructMatches[ConstructIdx++]);
332097a140dSpatrick   }
333097a140dSpatrick 
334097a140dSpatrick   LLVM_DEBUG(dbgs() << "[" << DEBUG_TYPE << "] Variant has a score of " << Score
335097a140dSpatrick                     << "\n");
336097a140dSpatrick   return Score;
337097a140dSpatrick }
338097a140dSpatrick 
getBestVariantMatchForContext(const SmallVectorImpl<VariantMatchInfo> & VMIs,const OMPContext & Ctx)339097a140dSpatrick int llvm::omp::getBestVariantMatchForContext(
340097a140dSpatrick     const SmallVectorImpl<VariantMatchInfo> &VMIs, const OMPContext &Ctx) {
341097a140dSpatrick 
342097a140dSpatrick   APInt BestScore(64, 0);
343097a140dSpatrick   int BestVMIIdx = -1;
344097a140dSpatrick   const VariantMatchInfo *BestVMI = nullptr;
345097a140dSpatrick 
346097a140dSpatrick   for (unsigned u = 0, e = VMIs.size(); u < e; ++u) {
347097a140dSpatrick     const VariantMatchInfo &VMI = VMIs[u];
348097a140dSpatrick 
349097a140dSpatrick     SmallVector<unsigned, 8> ConstructMatches;
350097a140dSpatrick     // If the variant is not applicable its not the best.
351097a140dSpatrick     if (!isVariantApplicableInContextHelper(VMI, Ctx, &ConstructMatches,
352097a140dSpatrick                                             /* DeviceSetOnly */ false))
353097a140dSpatrick       continue;
354097a140dSpatrick     // Check if its clearly not the best.
355097a140dSpatrick     APInt Score = getVariantMatchScore(VMI, Ctx, ConstructMatches);
356097a140dSpatrick     if (Score.ult(BestScore))
357097a140dSpatrick       continue;
358097a140dSpatrick     // Equal score need subset checks.
359097a140dSpatrick     if (Score.eq(BestScore)) {
360097a140dSpatrick       // Strict subset are never best.
361097a140dSpatrick       if (isStrictSubset(VMI, *BestVMI))
362097a140dSpatrick         continue;
363097a140dSpatrick       // Same score and the current best is no strict subset so we keep it.
364097a140dSpatrick       if (!isStrictSubset(*BestVMI, VMI))
365097a140dSpatrick         continue;
366097a140dSpatrick     }
367097a140dSpatrick     // New best found.
368097a140dSpatrick     BestVMI = &VMI;
369097a140dSpatrick     BestVMIIdx = u;
370097a140dSpatrick     BestScore = Score;
371097a140dSpatrick   }
372097a140dSpatrick 
373097a140dSpatrick   return BestVMIIdx;
374097a140dSpatrick }
375097a140dSpatrick 
getOpenMPContextTraitSetKind(StringRef S)376097a140dSpatrick TraitSet llvm::omp::getOpenMPContextTraitSetKind(StringRef S) {
377097a140dSpatrick   return StringSwitch<TraitSet>(S)
378097a140dSpatrick #define OMP_TRAIT_SET(Enum, Str) .Case(Str, TraitSet::Enum)
379097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
380097a140dSpatrick       .Default(TraitSet::invalid);
381097a140dSpatrick }
382097a140dSpatrick 
383097a140dSpatrick TraitSet
getOpenMPContextTraitSetForSelector(TraitSelector Selector)384097a140dSpatrick llvm::omp::getOpenMPContextTraitSetForSelector(TraitSelector Selector) {
385097a140dSpatrick   switch (Selector) {
386097a140dSpatrick #define OMP_TRAIT_SELECTOR(Enum, TraitSetEnum, Str, ReqProp)                   \
387097a140dSpatrick   case TraitSelector::Enum:                                                    \
388097a140dSpatrick     return TraitSet::TraitSetEnum;
389097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
390097a140dSpatrick   }
391097a140dSpatrick   llvm_unreachable("Unknown trait selector!");
392097a140dSpatrick }
393097a140dSpatrick TraitSet
getOpenMPContextTraitSetForProperty(TraitProperty Property)394097a140dSpatrick llvm::omp::getOpenMPContextTraitSetForProperty(TraitProperty Property) {
395097a140dSpatrick   switch (Property) {
396097a140dSpatrick #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str)         \
397097a140dSpatrick   case TraitProperty::Enum:                                                    \
398097a140dSpatrick     return TraitSet::TraitSetEnum;
399097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
400097a140dSpatrick   }
401097a140dSpatrick   llvm_unreachable("Unknown trait set!");
402097a140dSpatrick }
getOpenMPContextTraitSetName(TraitSet Kind)403097a140dSpatrick StringRef llvm::omp::getOpenMPContextTraitSetName(TraitSet Kind) {
404097a140dSpatrick   switch (Kind) {
405097a140dSpatrick #define OMP_TRAIT_SET(Enum, Str)                                               \
406097a140dSpatrick   case TraitSet::Enum:                                                         \
407097a140dSpatrick     return Str;
408097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
409097a140dSpatrick   }
410097a140dSpatrick   llvm_unreachable("Unknown trait set!");
411097a140dSpatrick }
412097a140dSpatrick 
getOpenMPContextTraitSelectorKind(StringRef S)413097a140dSpatrick TraitSelector llvm::omp::getOpenMPContextTraitSelectorKind(StringRef S) {
414097a140dSpatrick   return StringSwitch<TraitSelector>(S)
415097a140dSpatrick #define OMP_TRAIT_SELECTOR(Enum, TraitSetEnum, Str, ReqProp)                   \
416097a140dSpatrick   .Case(Str, TraitSelector::Enum)
417097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
418097a140dSpatrick       .Default(TraitSelector::invalid);
419097a140dSpatrick }
420097a140dSpatrick TraitSelector
getOpenMPContextTraitSelectorForProperty(TraitProperty Property)421097a140dSpatrick llvm::omp::getOpenMPContextTraitSelectorForProperty(TraitProperty Property) {
422097a140dSpatrick   switch (Property) {
423097a140dSpatrick #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str)         \
424097a140dSpatrick   case TraitProperty::Enum:                                                    \
425097a140dSpatrick     return TraitSelector::TraitSelectorEnum;
426097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
427097a140dSpatrick   }
428097a140dSpatrick   llvm_unreachable("Unknown trait set!");
429097a140dSpatrick }
getOpenMPContextTraitSelectorName(TraitSelector Kind)430097a140dSpatrick StringRef llvm::omp::getOpenMPContextTraitSelectorName(TraitSelector Kind) {
431097a140dSpatrick   switch (Kind) {
432097a140dSpatrick #define OMP_TRAIT_SELECTOR(Enum, TraitSetEnum, Str, ReqProp)                   \
433097a140dSpatrick   case TraitSelector::Enum:                                                    \
434097a140dSpatrick     return Str;
435097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
436097a140dSpatrick   }
437097a140dSpatrick   llvm_unreachable("Unknown trait selector!");
438097a140dSpatrick }
439097a140dSpatrick 
getOpenMPContextTraitPropertyKind(TraitSet Set,TraitSelector Selector,StringRef S)44073471bf0Spatrick TraitProperty llvm::omp::getOpenMPContextTraitPropertyKind(
44173471bf0Spatrick     TraitSet Set, TraitSelector Selector, StringRef S) {
44273471bf0Spatrick   // Special handling for `device={isa(...)}` as we accept anything here. It is
44373471bf0Spatrick   // up to the target to decide if the feature is available.
44473471bf0Spatrick   if (Set == TraitSet::device && Selector == TraitSelector::device_isa)
44573471bf0Spatrick     return TraitProperty::device_isa___ANY;
446097a140dSpatrick #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str)         \
447097a140dSpatrick   if (Set == TraitSet::TraitSetEnum && Str == S)                               \
448097a140dSpatrick     return TraitProperty::Enum;
449097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
450097a140dSpatrick   return TraitProperty::invalid;
451097a140dSpatrick }
452097a140dSpatrick TraitProperty
getOpenMPContextTraitPropertyForSelector(TraitSelector Selector)453097a140dSpatrick llvm::omp::getOpenMPContextTraitPropertyForSelector(TraitSelector Selector) {
454097a140dSpatrick   return StringSwitch<TraitProperty>(
455097a140dSpatrick              getOpenMPContextTraitSelectorName(Selector))
456097a140dSpatrick #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str)         \
457097a140dSpatrick   .Case(Str, Selector == TraitSelector::TraitSelectorEnum                      \
458097a140dSpatrick                  ? TraitProperty::Enum                                         \
459097a140dSpatrick                  : TraitProperty::invalid)
460097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
461097a140dSpatrick       .Default(TraitProperty::invalid);
462097a140dSpatrick }
getOpenMPContextTraitPropertyName(TraitProperty Kind,StringRef RawString)46373471bf0Spatrick StringRef llvm::omp::getOpenMPContextTraitPropertyName(TraitProperty Kind,
46473471bf0Spatrick                                                        StringRef RawString) {
46573471bf0Spatrick   if (Kind == TraitProperty::device_isa___ANY)
46673471bf0Spatrick     return RawString;
467097a140dSpatrick   switch (Kind) {
468097a140dSpatrick #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str)         \
469097a140dSpatrick   case TraitProperty::Enum:                                                    \
470097a140dSpatrick     return Str;
471097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
472097a140dSpatrick   }
473097a140dSpatrick   llvm_unreachable("Unknown trait property!");
474097a140dSpatrick }
getOpenMPContextTraitPropertyFullName(TraitProperty Kind)475097a140dSpatrick StringRef llvm::omp::getOpenMPContextTraitPropertyFullName(TraitProperty Kind) {
476097a140dSpatrick   switch (Kind) {
477097a140dSpatrick #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str)         \
478097a140dSpatrick   case TraitProperty::Enum:                                                    \
479097a140dSpatrick     return "(" #TraitSetEnum "," #TraitSelectorEnum "," Str ")";
480097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
481097a140dSpatrick   }
482097a140dSpatrick   llvm_unreachable("Unknown trait property!");
483097a140dSpatrick }
484097a140dSpatrick 
isValidTraitSelectorForTraitSet(TraitSelector Selector,TraitSet Set,bool & AllowsTraitScore,bool & RequiresProperty)485097a140dSpatrick bool llvm::omp::isValidTraitSelectorForTraitSet(TraitSelector Selector,
486097a140dSpatrick                                                 TraitSet Set,
487097a140dSpatrick                                                 bool &AllowsTraitScore,
488097a140dSpatrick                                                 bool &RequiresProperty) {
489097a140dSpatrick   AllowsTraitScore = Set != TraitSet::construct && Set != TraitSet::device;
490097a140dSpatrick   switch (Selector) {
491097a140dSpatrick #define OMP_TRAIT_SELECTOR(Enum, TraitSetEnum, Str, ReqProp)                   \
492097a140dSpatrick   case TraitSelector::Enum:                                                    \
493097a140dSpatrick     RequiresProperty = ReqProp;                                                \
494097a140dSpatrick     return Set == TraitSet::TraitSetEnum;
495097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
496097a140dSpatrick   }
497097a140dSpatrick   llvm_unreachable("Unknown trait selector!");
498097a140dSpatrick }
499097a140dSpatrick 
isValidTraitPropertyForTraitSetAndSelector(TraitProperty Property,TraitSelector Selector,TraitSet Set)500097a140dSpatrick bool llvm::omp::isValidTraitPropertyForTraitSetAndSelector(
501097a140dSpatrick     TraitProperty Property, TraitSelector Selector, TraitSet Set) {
502097a140dSpatrick   switch (Property) {
503097a140dSpatrick #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str)         \
504097a140dSpatrick   case TraitProperty::Enum:                                                    \
505097a140dSpatrick     return Set == TraitSet::TraitSetEnum &&                                    \
506097a140dSpatrick            Selector == TraitSelector::TraitSelectorEnum;
507097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
508097a140dSpatrick   }
509097a140dSpatrick   llvm_unreachable("Unknown trait property!");
510097a140dSpatrick }
511097a140dSpatrick 
listOpenMPContextTraitSets()512097a140dSpatrick std::string llvm::omp::listOpenMPContextTraitSets() {
513097a140dSpatrick   std::string S;
514097a140dSpatrick #define OMP_TRAIT_SET(Enum, Str)                                               \
515097a140dSpatrick   if (StringRef(Str) != "invalid")                                             \
516097a140dSpatrick     S.append("'").append(Str).append("'").append(" ");
517097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
518097a140dSpatrick   S.pop_back();
519097a140dSpatrick   return S;
520097a140dSpatrick }
521097a140dSpatrick 
listOpenMPContextTraitSelectors(TraitSet Set)522097a140dSpatrick std::string llvm::omp::listOpenMPContextTraitSelectors(TraitSet Set) {
523097a140dSpatrick   std::string S;
524097a140dSpatrick #define OMP_TRAIT_SELECTOR(Enum, TraitSetEnum, Str, ReqProp)                   \
525097a140dSpatrick   if (TraitSet::TraitSetEnum == Set && StringRef(Str) != "Invalid")            \
526097a140dSpatrick     S.append("'").append(Str).append("'").append(" ");
527097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
528097a140dSpatrick   S.pop_back();
529097a140dSpatrick   return S;
530097a140dSpatrick }
531097a140dSpatrick 
532097a140dSpatrick std::string
listOpenMPContextTraitProperties(TraitSet Set,TraitSelector Selector)533097a140dSpatrick llvm::omp::listOpenMPContextTraitProperties(TraitSet Set,
534097a140dSpatrick                                             TraitSelector Selector) {
535097a140dSpatrick   std::string S;
536097a140dSpatrick #define OMP_TRAIT_PROPERTY(Enum, TraitSetEnum, TraitSelectorEnum, Str)         \
537097a140dSpatrick   if (TraitSet::TraitSetEnum == Set &&                                         \
538097a140dSpatrick       TraitSelector::TraitSelectorEnum == Selector &&                          \
539097a140dSpatrick       StringRef(Str) != "invalid")                                             \
540097a140dSpatrick     S.append("'").append(Str).append("'").append(" ");
541097a140dSpatrick #include "llvm/Frontend/OpenMP/OMPKinds.def"
542097a140dSpatrick   if (S.empty())
543097a140dSpatrick     return "<none>";
544097a140dSpatrick   S.pop_back();
545097a140dSpatrick   return S;
546097a140dSpatrick }
547