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