xref: /freebsd/contrib/llvm-project/clang/lib/AST/NSAPI.cpp (revision 4fbb9c43aa44d9145151bb5f77d302ba01fb7551)
1 //===--- NSAPI.cpp - NSFoundation APIs ------------------------------------===//
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 
9 #include "clang/AST/NSAPI.h"
10 #include "clang/AST/ASTContext.h"
11 #include "clang/AST/DeclObjC.h"
12 #include "clang/AST/Expr.h"
13 #include "llvm/ADT/StringSwitch.h"
14 #include <optional>
15 
16 using namespace clang;
17 
18 NSAPI::NSAPI(ASTContext &ctx)
19   : Ctx(ctx), ClassIds(), BOOLId(nullptr), NSIntegerId(nullptr),
20     NSUIntegerId(nullptr), NSASCIIStringEncodingId(nullptr),
21     NSUTF8StringEncodingId(nullptr) {}
22 
23 IdentifierInfo *NSAPI::getNSClassId(NSClassIdKindKind K) const {
24   static const char *ClassName[NumClassIds] = {
25     "NSObject",
26     "NSString",
27     "NSArray",
28     "NSMutableArray",
29     "NSDictionary",
30     "NSMutableDictionary",
31     "NSNumber",
32     "NSMutableSet",
33     "NSMutableOrderedSet",
34     "NSValue"
35   };
36 
37   if (!ClassIds[K])
38     return (ClassIds[K] = &Ctx.Idents.get(ClassName[K]));
39 
40   return ClassIds[K];
41 }
42 
43 Selector NSAPI::getNSStringSelector(NSStringMethodKind MK) const {
44   if (NSStringSelectors[MK].isNull()) {
45     Selector Sel;
46     switch (MK) {
47     case NSStr_stringWithString:
48       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("stringWithString"));
49       break;
50     case NSStr_stringWithUTF8String:
51       Sel = Ctx.Selectors.getUnarySelector(
52                                        &Ctx.Idents.get("stringWithUTF8String"));
53       break;
54     case NSStr_initWithUTF8String:
55       Sel = Ctx.Selectors.getUnarySelector(
56                                        &Ctx.Idents.get("initWithUTF8String"));
57       break;
58     case NSStr_stringWithCStringEncoding: {
59       IdentifierInfo *KeyIdents[] = {
60         &Ctx.Idents.get("stringWithCString"),
61         &Ctx.Idents.get("encoding")
62       };
63       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
64       break;
65     }
66     case NSStr_stringWithCString:
67       Sel= Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("stringWithCString"));
68       break;
69     case NSStr_initWithString:
70       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("initWithString"));
71       break;
72     }
73     return (NSStringSelectors[MK] = Sel);
74   }
75 
76   return NSStringSelectors[MK];
77 }
78 
79 Selector NSAPI::getNSArraySelector(NSArrayMethodKind MK) const {
80   if (NSArraySelectors[MK].isNull()) {
81     Selector Sel;
82     switch (MK) {
83     case NSArr_array:
84       Sel = Ctx.Selectors.getNullarySelector(&Ctx.Idents.get("array"));
85       break;
86     case NSArr_arrayWithArray:
87       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("arrayWithArray"));
88       break;
89     case NSArr_arrayWithObject:
90       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("arrayWithObject"));
91       break;
92     case NSArr_arrayWithObjects:
93       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("arrayWithObjects"));
94       break;
95     case NSArr_arrayWithObjectsCount: {
96       IdentifierInfo *KeyIdents[] = {
97         &Ctx.Idents.get("arrayWithObjects"),
98         &Ctx.Idents.get("count")
99       };
100       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
101       break;
102     }
103     case NSArr_initWithArray:
104       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("initWithArray"));
105       break;
106     case NSArr_initWithObjects:
107       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("initWithObjects"));
108       break;
109     case NSArr_objectAtIndex:
110       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("objectAtIndex"));
111       break;
112     case NSMutableArr_replaceObjectAtIndex: {
113       IdentifierInfo *KeyIdents[] = {
114         &Ctx.Idents.get("replaceObjectAtIndex"),
115         &Ctx.Idents.get("withObject")
116       };
117       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
118       break;
119     }
120     case NSMutableArr_addObject:
121       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("addObject"));
122       break;
123     case NSMutableArr_insertObjectAtIndex: {
124       IdentifierInfo *KeyIdents[] = {
125         &Ctx.Idents.get("insertObject"),
126         &Ctx.Idents.get("atIndex")
127       };
128       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
129       break;
130     }
131     case NSMutableArr_setObjectAtIndexedSubscript: {
132       IdentifierInfo *KeyIdents[] = {
133         &Ctx.Idents.get("setObject"),
134         &Ctx.Idents.get("atIndexedSubscript")
135       };
136       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
137       break;
138     }
139     }
140     return (NSArraySelectors[MK] = Sel);
141   }
142 
143   return NSArraySelectors[MK];
144 }
145 
146 std::optional<NSAPI::NSArrayMethodKind>
147 NSAPI::getNSArrayMethodKind(Selector Sel) {
148   for (unsigned i = 0; i != NumNSArrayMethods; ++i) {
149     NSArrayMethodKind MK = NSArrayMethodKind(i);
150     if (Sel == getNSArraySelector(MK))
151       return MK;
152   }
153 
154   return std::nullopt;
155 }
156 
157 Selector NSAPI::getNSDictionarySelector(
158                                        NSDictionaryMethodKind MK) const {
159   if (NSDictionarySelectors[MK].isNull()) {
160     Selector Sel;
161     switch (MK) {
162     case NSDict_dictionary:
163       Sel = Ctx.Selectors.getNullarySelector(&Ctx.Idents.get("dictionary"));
164       break;
165     case NSDict_dictionaryWithDictionary:
166       Sel = Ctx.Selectors.getUnarySelector(
167                                    &Ctx.Idents.get("dictionaryWithDictionary"));
168       break;
169     case NSDict_dictionaryWithObjectForKey: {
170       IdentifierInfo *KeyIdents[] = {
171         &Ctx.Idents.get("dictionaryWithObject"),
172         &Ctx.Idents.get("forKey")
173       };
174       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
175       break;
176     }
177     case NSDict_dictionaryWithObjectsForKeys: {
178       IdentifierInfo *KeyIdents[] = {
179         &Ctx.Idents.get("dictionaryWithObjects"),
180         &Ctx.Idents.get("forKeys")
181       };
182       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
183       break;
184     }
185     case NSDict_dictionaryWithObjectsForKeysCount: {
186       IdentifierInfo *KeyIdents[] = {
187         &Ctx.Idents.get("dictionaryWithObjects"),
188         &Ctx.Idents.get("forKeys"),
189         &Ctx.Idents.get("count")
190       };
191       Sel = Ctx.Selectors.getSelector(3, KeyIdents);
192       break;
193     }
194     case NSDict_dictionaryWithObjectsAndKeys:
195       Sel = Ctx.Selectors.getUnarySelector(
196                                &Ctx.Idents.get("dictionaryWithObjectsAndKeys"));
197       break;
198     case NSDict_initWithDictionary:
199       Sel = Ctx.Selectors.getUnarySelector(
200                                          &Ctx.Idents.get("initWithDictionary"));
201       break;
202     case NSDict_initWithObjectsAndKeys:
203       Sel = Ctx.Selectors.getUnarySelector(
204                                      &Ctx.Idents.get("initWithObjectsAndKeys"));
205       break;
206     case NSDict_initWithObjectsForKeys: {
207       IdentifierInfo *KeyIdents[] = {
208         &Ctx.Idents.get("initWithObjects"),
209         &Ctx.Idents.get("forKeys")
210       };
211       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
212       break;
213     }
214     case NSDict_objectForKey:
215       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("objectForKey"));
216       break;
217     case NSMutableDict_setObjectForKey: {
218       IdentifierInfo *KeyIdents[] = {
219         &Ctx.Idents.get("setObject"),
220         &Ctx.Idents.get("forKey")
221       };
222       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
223       break;
224     }
225     case NSMutableDict_setObjectForKeyedSubscript: {
226       IdentifierInfo *KeyIdents[] = {
227         &Ctx.Idents.get("setObject"),
228         &Ctx.Idents.get("forKeyedSubscript")
229       };
230       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
231       break;
232     }
233     case NSMutableDict_setValueForKey: {
234       IdentifierInfo *KeyIdents[] = {
235         &Ctx.Idents.get("setValue"),
236         &Ctx.Idents.get("forKey")
237       };
238       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
239       break;
240     }
241     }
242     return (NSDictionarySelectors[MK] = Sel);
243   }
244 
245   return NSDictionarySelectors[MK];
246 }
247 
248 std::optional<NSAPI::NSDictionaryMethodKind>
249 NSAPI::getNSDictionaryMethodKind(Selector Sel) {
250   for (unsigned i = 0; i != NumNSDictionaryMethods; ++i) {
251     NSDictionaryMethodKind MK = NSDictionaryMethodKind(i);
252     if (Sel == getNSDictionarySelector(MK))
253       return MK;
254   }
255 
256   return std::nullopt;
257 }
258 
259 Selector NSAPI::getNSSetSelector(NSSetMethodKind MK) const {
260   if (NSSetSelectors[MK].isNull()) {
261     Selector Sel;
262     switch (MK) {
263     case NSMutableSet_addObject:
264       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("addObject"));
265       break;
266     case NSOrderedSet_insertObjectAtIndex: {
267       IdentifierInfo *KeyIdents[] = {
268         &Ctx.Idents.get("insertObject"),
269         &Ctx.Idents.get("atIndex")
270       };
271       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
272       break;
273     }
274     case NSOrderedSet_setObjectAtIndex: {
275       IdentifierInfo *KeyIdents[] = {
276         &Ctx.Idents.get("setObject"),
277         &Ctx.Idents.get("atIndex")
278       };
279       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
280       break;
281     }
282     case NSOrderedSet_setObjectAtIndexedSubscript: {
283       IdentifierInfo *KeyIdents[] = {
284         &Ctx.Idents.get("setObject"),
285         &Ctx.Idents.get("atIndexedSubscript")
286       };
287       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
288       break;
289     }
290     case NSOrderedSet_replaceObjectAtIndexWithObject: {
291       IdentifierInfo *KeyIdents[] = {
292         &Ctx.Idents.get("replaceObjectAtIndex"),
293         &Ctx.Idents.get("withObject")
294       };
295       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
296       break;
297     }
298     }
299     return (NSSetSelectors[MK] = Sel);
300   }
301 
302   return NSSetSelectors[MK];
303 }
304 
305 std::optional<NSAPI::NSSetMethodKind> NSAPI::getNSSetMethodKind(Selector Sel) {
306   for (unsigned i = 0; i != NumNSSetMethods; ++i) {
307     NSSetMethodKind MK = NSSetMethodKind(i);
308     if (Sel == getNSSetSelector(MK))
309       return MK;
310   }
311 
312   return std::nullopt;
313 }
314 
315 Selector NSAPI::getNSNumberLiteralSelector(NSNumberLiteralMethodKind MK,
316                                            bool Instance) const {
317   static const char *ClassSelectorName[NumNSNumberLiteralMethods] = {
318     "numberWithChar",
319     "numberWithUnsignedChar",
320     "numberWithShort",
321     "numberWithUnsignedShort",
322     "numberWithInt",
323     "numberWithUnsignedInt",
324     "numberWithLong",
325     "numberWithUnsignedLong",
326     "numberWithLongLong",
327     "numberWithUnsignedLongLong",
328     "numberWithFloat",
329     "numberWithDouble",
330     "numberWithBool",
331     "numberWithInteger",
332     "numberWithUnsignedInteger"
333   };
334   static const char *InstanceSelectorName[NumNSNumberLiteralMethods] = {
335     "initWithChar",
336     "initWithUnsignedChar",
337     "initWithShort",
338     "initWithUnsignedShort",
339     "initWithInt",
340     "initWithUnsignedInt",
341     "initWithLong",
342     "initWithUnsignedLong",
343     "initWithLongLong",
344     "initWithUnsignedLongLong",
345     "initWithFloat",
346     "initWithDouble",
347     "initWithBool",
348     "initWithInteger",
349     "initWithUnsignedInteger"
350   };
351 
352   Selector *Sels;
353   const char **Names;
354   if (Instance) {
355     Sels = NSNumberInstanceSelectors;
356     Names = InstanceSelectorName;
357   } else {
358     Sels = NSNumberClassSelectors;
359     Names = ClassSelectorName;
360   }
361 
362   if (Sels[MK].isNull())
363     Sels[MK] = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get(Names[MK]));
364   return Sels[MK];
365 }
366 
367 std::optional<NSAPI::NSNumberLiteralMethodKind>
368 NSAPI::getNSNumberLiteralMethodKind(Selector Sel) const {
369   for (unsigned i = 0; i != NumNSNumberLiteralMethods; ++i) {
370     NSNumberLiteralMethodKind MK = NSNumberLiteralMethodKind(i);
371     if (isNSNumberLiteralSelector(MK, Sel))
372       return MK;
373   }
374 
375   return std::nullopt;
376 }
377 
378 std::optional<NSAPI::NSNumberLiteralMethodKind>
379 NSAPI::getNSNumberFactoryMethodKind(QualType T) const {
380   const BuiltinType *BT = T->getAs<BuiltinType>();
381   if (!BT)
382     return std::nullopt;
383 
384   const TypedefType *TDT = T->getAs<TypedefType>();
385   if (TDT) {
386     QualType TDTTy = QualType(TDT, 0);
387     if (isObjCBOOLType(TDTTy))
388       return NSAPI::NSNumberWithBool;
389     if (isObjCNSIntegerType(TDTTy))
390       return NSAPI::NSNumberWithInteger;
391     if (isObjCNSUIntegerType(TDTTy))
392       return NSAPI::NSNumberWithUnsignedInteger;
393   }
394 
395   switch (BT->getKind()) {
396   case BuiltinType::Char_S:
397   case BuiltinType::SChar:
398     return NSAPI::NSNumberWithChar;
399   case BuiltinType::Char_U:
400   case BuiltinType::UChar:
401     return NSAPI::NSNumberWithUnsignedChar;
402   case BuiltinType::Short:
403     return NSAPI::NSNumberWithShort;
404   case BuiltinType::UShort:
405     return NSAPI::NSNumberWithUnsignedShort;
406   case BuiltinType::Int:
407     return NSAPI::NSNumberWithInt;
408   case BuiltinType::UInt:
409     return NSAPI::NSNumberWithUnsignedInt;
410   case BuiltinType::Long:
411     return NSAPI::NSNumberWithLong;
412   case BuiltinType::ULong:
413     return NSAPI::NSNumberWithUnsignedLong;
414   case BuiltinType::LongLong:
415     return NSAPI::NSNumberWithLongLong;
416   case BuiltinType::ULongLong:
417     return NSAPI::NSNumberWithUnsignedLongLong;
418   case BuiltinType::Float:
419     return NSAPI::NSNumberWithFloat;
420   case BuiltinType::Double:
421     return NSAPI::NSNumberWithDouble;
422   case BuiltinType::Bool:
423     return NSAPI::NSNumberWithBool;
424 
425   case BuiltinType::Void:
426   case BuiltinType::WChar_U:
427   case BuiltinType::WChar_S:
428   case BuiltinType::Char8:
429   case BuiltinType::Char16:
430   case BuiltinType::Char32:
431   case BuiltinType::Int128:
432   case BuiltinType::LongDouble:
433   case BuiltinType::ShortAccum:
434   case BuiltinType::Accum:
435   case BuiltinType::LongAccum:
436   case BuiltinType::UShortAccum:
437   case BuiltinType::UAccum:
438   case BuiltinType::ULongAccum:
439   case BuiltinType::ShortFract:
440   case BuiltinType::Fract:
441   case BuiltinType::LongFract:
442   case BuiltinType::UShortFract:
443   case BuiltinType::UFract:
444   case BuiltinType::ULongFract:
445   case BuiltinType::SatShortAccum:
446   case BuiltinType::SatAccum:
447   case BuiltinType::SatLongAccum:
448   case BuiltinType::SatUShortAccum:
449   case BuiltinType::SatUAccum:
450   case BuiltinType::SatULongAccum:
451   case BuiltinType::SatShortFract:
452   case BuiltinType::SatFract:
453   case BuiltinType::SatLongFract:
454   case BuiltinType::SatUShortFract:
455   case BuiltinType::SatUFract:
456   case BuiltinType::SatULongFract:
457   case BuiltinType::UInt128:
458   case BuiltinType::Float16:
459   case BuiltinType::Float128:
460   case BuiltinType::Ibm128:
461   case BuiltinType::NullPtr:
462   case BuiltinType::ObjCClass:
463   case BuiltinType::ObjCId:
464   case BuiltinType::ObjCSel:
465 #define IMAGE_TYPE(ImgType, Id, SingletonId, Access, Suffix) \
466   case BuiltinType::Id:
467 #include "clang/Basic/OpenCLImageTypes.def"
468 #define EXT_OPAQUE_TYPE(ExtType, Id, Ext) \
469   case BuiltinType::Id:
470 #include "clang/Basic/OpenCLExtensionTypes.def"
471   case BuiltinType::OCLSampler:
472   case BuiltinType::OCLEvent:
473   case BuiltinType::OCLClkEvent:
474   case BuiltinType::OCLQueue:
475   case BuiltinType::OCLReserveID:
476 #define SVE_TYPE(Name, Id, SingletonId) \
477   case BuiltinType::Id:
478 #include "clang/Basic/AArch64SVEACLETypes.def"
479 #define PPC_VECTOR_TYPE(Name, Id, Size) \
480   case BuiltinType::Id:
481 #include "clang/Basic/PPCTypes.def"
482 #define RVV_TYPE(Name, Id, SingletonId) case BuiltinType::Id:
483 #include "clang/Basic/RISCVVTypes.def"
484   case BuiltinType::BoundMember:
485   case BuiltinType::Dependent:
486   case BuiltinType::Overload:
487   case BuiltinType::UnknownAny:
488   case BuiltinType::ARCUnbridgedCast:
489   case BuiltinType::Half:
490   case BuiltinType::PseudoObject:
491   case BuiltinType::BuiltinFn:
492   case BuiltinType::IncompleteMatrixIdx:
493   case BuiltinType::OMPArraySection:
494   case BuiltinType::OMPArrayShaping:
495   case BuiltinType::OMPIterator:
496   case BuiltinType::BFloat16:
497     break;
498   }
499 
500   return std::nullopt;
501 }
502 
503 /// Returns true if \param T is a typedef of "BOOL" in objective-c.
504 bool NSAPI::isObjCBOOLType(QualType T) const {
505   return isObjCTypedef(T, "BOOL", BOOLId);
506 }
507 /// Returns true if \param T is a typedef of "NSInteger" in objective-c.
508 bool NSAPI::isObjCNSIntegerType(QualType T) const {
509   return isObjCTypedef(T, "NSInteger", NSIntegerId);
510 }
511 /// Returns true if \param T is a typedef of "NSUInteger" in objective-c.
512 bool NSAPI::isObjCNSUIntegerType(QualType T) const {
513   return isObjCTypedef(T, "NSUInteger", NSUIntegerId);
514 }
515 
516 StringRef NSAPI::GetNSIntegralKind(QualType T) const {
517   if (!Ctx.getLangOpts().ObjC || T.isNull())
518     return StringRef();
519 
520   while (const TypedefType *TDT = T->getAs<TypedefType>()) {
521     StringRef NSIntegralResust =
522       llvm::StringSwitch<StringRef>(
523         TDT->getDecl()->getDeclName().getAsIdentifierInfo()->getName())
524     .Case("int8_t", "int8_t")
525     .Case("int16_t", "int16_t")
526     .Case("int32_t", "int32_t")
527     .Case("NSInteger", "NSInteger")
528     .Case("int64_t", "int64_t")
529     .Case("uint8_t", "uint8_t")
530     .Case("uint16_t", "uint16_t")
531     .Case("uint32_t", "uint32_t")
532     .Case("NSUInteger", "NSUInteger")
533     .Case("uint64_t", "uint64_t")
534     .Default(StringRef());
535     if (!NSIntegralResust.empty())
536       return NSIntegralResust;
537     T = TDT->desugar();
538   }
539   return StringRef();
540 }
541 
542 bool NSAPI::isMacroDefined(StringRef Id) const {
543   // FIXME: Check whether the relevant module macros are visible.
544   return Ctx.Idents.get(Id).hasMacroDefinition();
545 }
546 
547 bool NSAPI::isSubclassOfNSClass(ObjCInterfaceDecl *InterfaceDecl,
548                                 NSClassIdKindKind NSClassKind) const {
549   if (!InterfaceDecl) {
550     return false;
551   }
552 
553   IdentifierInfo *NSClassID = getNSClassId(NSClassKind);
554 
555   bool IsSubclass = false;
556   do {
557     IsSubclass = NSClassID == InterfaceDecl->getIdentifier();
558 
559     if (IsSubclass) {
560       break;
561     }
562   } while ((InterfaceDecl = InterfaceDecl->getSuperClass()));
563 
564   return IsSubclass;
565 }
566 
567 bool NSAPI::isObjCTypedef(QualType T,
568                           StringRef name, IdentifierInfo *&II) const {
569   if (!Ctx.getLangOpts().ObjC)
570     return false;
571   if (T.isNull())
572     return false;
573 
574   if (!II)
575     II = &Ctx.Idents.get(name);
576 
577   while (const TypedefType *TDT = T->getAs<TypedefType>()) {
578     if (TDT->getDecl()->getDeclName().getAsIdentifierInfo() == II)
579       return true;
580     T = TDT->desugar();
581   }
582 
583   return false;
584 }
585 
586 bool NSAPI::isObjCEnumerator(const Expr *E,
587                              StringRef name, IdentifierInfo *&II) const {
588   if (!Ctx.getLangOpts().ObjC)
589     return false;
590   if (!E)
591     return false;
592 
593   if (!II)
594     II = &Ctx.Idents.get(name);
595 
596   if (const DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E->IgnoreParenImpCasts()))
597     if (const EnumConstantDecl *
598           EnumD = dyn_cast_or_null<EnumConstantDecl>(DRE->getDecl()))
599       return EnumD->getIdentifier() == II;
600 
601   return false;
602 }
603 
604 Selector NSAPI::getOrInitSelector(ArrayRef<StringRef> Ids,
605                                   Selector &Sel) const {
606   if (Sel.isNull()) {
607     SmallVector<IdentifierInfo *, 4> Idents;
608     for (ArrayRef<StringRef>::const_iterator
609            I = Ids.begin(), E = Ids.end(); I != E; ++I)
610       Idents.push_back(&Ctx.Idents.get(*I));
611     Sel = Ctx.Selectors.getSelector(Idents.size(), Idents.data());
612   }
613   return Sel;
614 }
615 
616 Selector NSAPI::getOrInitNullarySelector(StringRef Id, Selector &Sel) const {
617   if (Sel.isNull()) {
618     IdentifierInfo *Ident = &Ctx.Idents.get(Id);
619     Sel = Ctx.Selectors.getSelector(0, &Ident);
620   }
621   return Sel;
622 }
623