• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===--- NSAPI.cpp - NSFoundation APIs ------------------------------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "clang/AST/NSAPI.h"
11 #include "clang/AST/ASTContext.h"
12 #include "clang/AST/Expr.h"
13 
14 using namespace clang;
15 
NSAPI(ASTContext & ctx)16 NSAPI::NSAPI(ASTContext &ctx)
17   : Ctx(ctx), ClassIds(), BOOLId(nullptr), NSIntegerId(nullptr),
18     NSUIntegerId(nullptr), NSASCIIStringEncodingId(nullptr),
19     NSUTF8StringEncodingId(nullptr) {}
20 
getNSClassId(NSClassIdKindKind K) const21 IdentifierInfo *NSAPI::getNSClassId(NSClassIdKindKind K) const {
22   static const char *ClassName[NumClassIds] = {
23     "NSObject",
24     "NSString",
25     "NSArray",
26     "NSMutableArray",
27     "NSDictionary",
28     "NSMutableDictionary",
29     "NSNumber"
30   };
31 
32   if (!ClassIds[K])
33     return (ClassIds[K] = &Ctx.Idents.get(ClassName[K]));
34 
35   return ClassIds[K];
36 }
37 
getNSStringSelector(NSStringMethodKind MK) const38 Selector NSAPI::getNSStringSelector(NSStringMethodKind MK) const {
39   if (NSStringSelectors[MK].isNull()) {
40     Selector Sel;
41     switch (MK) {
42     case NSStr_stringWithString:
43       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("stringWithString"));
44       break;
45     case NSStr_stringWithUTF8String:
46       Sel = Ctx.Selectors.getUnarySelector(
47                                        &Ctx.Idents.get("stringWithUTF8String"));
48       break;
49     case NSStr_stringWithCStringEncoding: {
50       IdentifierInfo *KeyIdents[] = {
51         &Ctx.Idents.get("stringWithCString"),
52         &Ctx.Idents.get("encoding")
53       };
54       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
55       break;
56     }
57     case NSStr_stringWithCString:
58       Sel= Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("stringWithCString"));
59       break;
60     case NSStr_initWithString:
61       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("initWithString"));
62       break;
63     }
64     return (NSStringSelectors[MK] = Sel);
65   }
66 
67   return NSStringSelectors[MK];
68 }
69 
70 Optional<NSAPI::NSStringMethodKind>
getNSStringMethodKind(Selector Sel) const71 NSAPI::getNSStringMethodKind(Selector Sel) const {
72   for (unsigned i = 0; i != NumNSStringMethods; ++i) {
73     NSStringMethodKind MK = NSStringMethodKind(i);
74     if (Sel == getNSStringSelector(MK))
75       return MK;
76   }
77 
78   return None;
79 }
80 
getNSArraySelector(NSArrayMethodKind MK) const81 Selector NSAPI::getNSArraySelector(NSArrayMethodKind MK) const {
82   if (NSArraySelectors[MK].isNull()) {
83     Selector Sel;
84     switch (MK) {
85     case NSArr_array:
86       Sel = Ctx.Selectors.getNullarySelector(&Ctx.Idents.get("array"));
87       break;
88     case NSArr_arrayWithArray:
89       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("arrayWithArray"));
90       break;
91     case NSArr_arrayWithObject:
92       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("arrayWithObject"));
93       break;
94     case NSArr_arrayWithObjects:
95       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("arrayWithObjects"));
96       break;
97     case NSArr_arrayWithObjectsCount: {
98       IdentifierInfo *KeyIdents[] = {
99         &Ctx.Idents.get("arrayWithObjects"),
100         &Ctx.Idents.get("count")
101       };
102       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
103       break;
104     }
105     case NSArr_initWithArray:
106       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("initWithArray"));
107       break;
108     case NSArr_initWithObjects:
109       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("initWithObjects"));
110       break;
111     case NSArr_objectAtIndex:
112       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("objectAtIndex"));
113       break;
114     case NSMutableArr_replaceObjectAtIndex: {
115       IdentifierInfo *KeyIdents[] = {
116         &Ctx.Idents.get("replaceObjectAtIndex"),
117         &Ctx.Idents.get("withObject")
118       };
119       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
120       break;
121     }
122     }
123     return (NSArraySelectors[MK] = Sel);
124   }
125 
126   return NSArraySelectors[MK];
127 }
128 
getNSArrayMethodKind(Selector Sel)129 Optional<NSAPI::NSArrayMethodKind> NSAPI::getNSArrayMethodKind(Selector Sel) {
130   for (unsigned i = 0; i != NumNSArrayMethods; ++i) {
131     NSArrayMethodKind MK = NSArrayMethodKind(i);
132     if (Sel == getNSArraySelector(MK))
133       return MK;
134   }
135 
136   return None;
137 }
138 
getNSDictionarySelector(NSDictionaryMethodKind MK) const139 Selector NSAPI::getNSDictionarySelector(
140                                        NSDictionaryMethodKind MK) const {
141   if (NSDictionarySelectors[MK].isNull()) {
142     Selector Sel;
143     switch (MK) {
144     case NSDict_dictionary:
145       Sel = Ctx.Selectors.getNullarySelector(&Ctx.Idents.get("dictionary"));
146       break;
147     case NSDict_dictionaryWithDictionary:
148       Sel = Ctx.Selectors.getUnarySelector(
149                                    &Ctx.Idents.get("dictionaryWithDictionary"));
150       break;
151     case NSDict_dictionaryWithObjectForKey: {
152       IdentifierInfo *KeyIdents[] = {
153         &Ctx.Idents.get("dictionaryWithObject"),
154         &Ctx.Idents.get("forKey")
155       };
156       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
157       break;
158     }
159     case NSDict_dictionaryWithObjectsForKeys: {
160       IdentifierInfo *KeyIdents[] = {
161         &Ctx.Idents.get("dictionaryWithObjects"),
162         &Ctx.Idents.get("forKeys")
163       };
164       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
165       break;
166     }
167     case NSDict_dictionaryWithObjectsForKeysCount: {
168       IdentifierInfo *KeyIdents[] = {
169         &Ctx.Idents.get("dictionaryWithObjects"),
170         &Ctx.Idents.get("forKeys"),
171         &Ctx.Idents.get("count")
172       };
173       Sel = Ctx.Selectors.getSelector(3, KeyIdents);
174       break;
175     }
176     case NSDict_dictionaryWithObjectsAndKeys:
177       Sel = Ctx.Selectors.getUnarySelector(
178                                &Ctx.Idents.get("dictionaryWithObjectsAndKeys"));
179       break;
180     case NSDict_initWithDictionary:
181       Sel = Ctx.Selectors.getUnarySelector(
182                                          &Ctx.Idents.get("initWithDictionary"));
183       break;
184     case NSDict_initWithObjectsAndKeys:
185       Sel = Ctx.Selectors.getUnarySelector(
186                                      &Ctx.Idents.get("initWithObjectsAndKeys"));
187       break;
188     case NSDict_initWithObjectsForKeys: {
189       IdentifierInfo *KeyIdents[] = {
190         &Ctx.Idents.get("initWithObjects"),
191         &Ctx.Idents.get("forKeys")
192       };
193       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
194       break;
195     }
196     case NSDict_objectForKey:
197       Sel = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get("objectForKey"));
198       break;
199     case NSMutableDict_setObjectForKey: {
200       IdentifierInfo *KeyIdents[] = {
201         &Ctx.Idents.get("setObject"),
202         &Ctx.Idents.get("forKey")
203       };
204       Sel = Ctx.Selectors.getSelector(2, KeyIdents);
205       break;
206     }
207     }
208     return (NSDictionarySelectors[MK] = Sel);
209   }
210 
211   return NSDictionarySelectors[MK];
212 }
213 
214 Optional<NSAPI::NSDictionaryMethodKind>
getNSDictionaryMethodKind(Selector Sel)215 NSAPI::getNSDictionaryMethodKind(Selector Sel) {
216   for (unsigned i = 0; i != NumNSDictionaryMethods; ++i) {
217     NSDictionaryMethodKind MK = NSDictionaryMethodKind(i);
218     if (Sel == getNSDictionarySelector(MK))
219       return MK;
220   }
221 
222   return None;
223 }
224 
getNSNumberLiteralSelector(NSNumberLiteralMethodKind MK,bool Instance) const225 Selector NSAPI::getNSNumberLiteralSelector(NSNumberLiteralMethodKind MK,
226                                            bool Instance) const {
227   static const char *ClassSelectorName[NumNSNumberLiteralMethods] = {
228     "numberWithChar",
229     "numberWithUnsignedChar",
230     "numberWithShort",
231     "numberWithUnsignedShort",
232     "numberWithInt",
233     "numberWithUnsignedInt",
234     "numberWithLong",
235     "numberWithUnsignedLong",
236     "numberWithLongLong",
237     "numberWithUnsignedLongLong",
238     "numberWithFloat",
239     "numberWithDouble",
240     "numberWithBool",
241     "numberWithInteger",
242     "numberWithUnsignedInteger"
243   };
244   static const char *InstanceSelectorName[NumNSNumberLiteralMethods] = {
245     "initWithChar",
246     "initWithUnsignedChar",
247     "initWithShort",
248     "initWithUnsignedShort",
249     "initWithInt",
250     "initWithUnsignedInt",
251     "initWithLong",
252     "initWithUnsignedLong",
253     "initWithLongLong",
254     "initWithUnsignedLongLong",
255     "initWithFloat",
256     "initWithDouble",
257     "initWithBool",
258     "initWithInteger",
259     "initWithUnsignedInteger"
260   };
261 
262   Selector *Sels;
263   const char **Names;
264   if (Instance) {
265     Sels = NSNumberInstanceSelectors;
266     Names = InstanceSelectorName;
267   } else {
268     Sels = NSNumberClassSelectors;
269     Names = ClassSelectorName;
270   }
271 
272   if (Sels[MK].isNull())
273     Sels[MK] = Ctx.Selectors.getUnarySelector(&Ctx.Idents.get(Names[MK]));
274   return Sels[MK];
275 }
276 
277 Optional<NSAPI::NSNumberLiteralMethodKind>
getNSNumberLiteralMethodKind(Selector Sel) const278 NSAPI::getNSNumberLiteralMethodKind(Selector Sel) const {
279   for (unsigned i = 0; i != NumNSNumberLiteralMethods; ++i) {
280     NSNumberLiteralMethodKind MK = NSNumberLiteralMethodKind(i);
281     if (isNSNumberLiteralSelector(MK, Sel))
282       return MK;
283   }
284 
285   return None;
286 }
287 
288 Optional<NSAPI::NSNumberLiteralMethodKind>
getNSNumberFactoryMethodKind(QualType T) const289 NSAPI::getNSNumberFactoryMethodKind(QualType T) const {
290   const BuiltinType *BT = T->getAs<BuiltinType>();
291   if (!BT)
292     return None;
293 
294   const TypedefType *TDT = T->getAs<TypedefType>();
295   if (TDT) {
296     QualType TDTTy = QualType(TDT, 0);
297     if (isObjCBOOLType(TDTTy))
298       return NSAPI::NSNumberWithBool;
299     if (isObjCNSIntegerType(TDTTy))
300       return NSAPI::NSNumberWithInteger;
301     if (isObjCNSUIntegerType(TDTTy))
302       return NSAPI::NSNumberWithUnsignedInteger;
303   }
304 
305   switch (BT->getKind()) {
306   case BuiltinType::Char_S:
307   case BuiltinType::SChar:
308     return NSAPI::NSNumberWithChar;
309   case BuiltinType::Char_U:
310   case BuiltinType::UChar:
311     return NSAPI::NSNumberWithUnsignedChar;
312   case BuiltinType::Short:
313     return NSAPI::NSNumberWithShort;
314   case BuiltinType::UShort:
315     return NSAPI::NSNumberWithUnsignedShort;
316   case BuiltinType::Int:
317     return NSAPI::NSNumberWithInt;
318   case BuiltinType::UInt:
319     return NSAPI::NSNumberWithUnsignedInt;
320   case BuiltinType::Long:
321     return NSAPI::NSNumberWithLong;
322   case BuiltinType::ULong:
323     return NSAPI::NSNumberWithUnsignedLong;
324   case BuiltinType::LongLong:
325     return NSAPI::NSNumberWithLongLong;
326   case BuiltinType::ULongLong:
327     return NSAPI::NSNumberWithUnsignedLongLong;
328   case BuiltinType::Float:
329     return NSAPI::NSNumberWithFloat;
330   case BuiltinType::Double:
331     return NSAPI::NSNumberWithDouble;
332   case BuiltinType::Bool:
333     return NSAPI::NSNumberWithBool;
334 
335   case BuiltinType::Void:
336   case BuiltinType::WChar_U:
337   case BuiltinType::WChar_S:
338   case BuiltinType::Char16:
339   case BuiltinType::Char32:
340   case BuiltinType::Int128:
341   case BuiltinType::LongDouble:
342   case BuiltinType::UInt128:
343   case BuiltinType::NullPtr:
344   case BuiltinType::ObjCClass:
345   case BuiltinType::ObjCId:
346   case BuiltinType::ObjCSel:
347   case BuiltinType::OCLImage1d:
348   case BuiltinType::OCLImage1dArray:
349   case BuiltinType::OCLImage1dBuffer:
350   case BuiltinType::OCLImage2d:
351   case BuiltinType::OCLImage2dArray:
352   case BuiltinType::OCLImage3d:
353   case BuiltinType::OCLSampler:
354   case BuiltinType::OCLEvent:
355   case BuiltinType::BoundMember:
356   case BuiltinType::Dependent:
357   case BuiltinType::Overload:
358   case BuiltinType::UnknownAny:
359   case BuiltinType::ARCUnbridgedCast:
360   case BuiltinType::Half:
361   case BuiltinType::PseudoObject:
362   case BuiltinType::BuiltinFn:
363     break;
364   }
365 
366   return None;
367 }
368 
369 /// \brief Returns true if \param T is a typedef of "BOOL" in objective-c.
isObjCBOOLType(QualType T) const370 bool NSAPI::isObjCBOOLType(QualType T) const {
371   return isObjCTypedef(T, "BOOL", BOOLId);
372 }
373 /// \brief Returns true if \param T is a typedef of "NSInteger" in objective-c.
isObjCNSIntegerType(QualType T) const374 bool NSAPI::isObjCNSIntegerType(QualType T) const {
375   return isObjCTypedef(T, "NSInteger", NSIntegerId);
376 }
377 /// \brief Returns true if \param T is a typedef of "NSUInteger" in objective-c.
isObjCNSUIntegerType(QualType T) const378 bool NSAPI::isObjCNSUIntegerType(QualType T) const {
379   return isObjCTypedef(T, "NSUInteger", NSUIntegerId);
380 }
381 
isObjCTypedef(QualType T,StringRef name,IdentifierInfo * & II) const382 bool NSAPI::isObjCTypedef(QualType T,
383                           StringRef name, IdentifierInfo *&II) const {
384   if (!Ctx.getLangOpts().ObjC1)
385     return false;
386   if (T.isNull())
387     return false;
388 
389   if (!II)
390     II = &Ctx.Idents.get(name);
391 
392   while (const TypedefType *TDT = T->getAs<TypedefType>()) {
393     if (TDT->getDecl()->getDeclName().getAsIdentifierInfo() == II)
394       return true;
395     T = TDT->desugar();
396   }
397 
398   return false;
399 }
400 
isObjCEnumerator(const Expr * E,StringRef name,IdentifierInfo * & II) const401 bool NSAPI::isObjCEnumerator(const Expr *E,
402                              StringRef name, IdentifierInfo *&II) const {
403   if (!Ctx.getLangOpts().ObjC1)
404     return false;
405   if (!E)
406     return false;
407 
408   if (!II)
409     II = &Ctx.Idents.get(name);
410 
411   if (const DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E->IgnoreParenImpCasts()))
412     if (const EnumConstantDecl *
413           EnumD = dyn_cast_or_null<EnumConstantDecl>(DRE->getDecl()))
414       return EnumD->getIdentifier() == II;
415 
416   return false;
417 }
418 
getOrInitSelector(ArrayRef<StringRef> Ids,Selector & Sel) const419 Selector NSAPI::getOrInitSelector(ArrayRef<StringRef> Ids,
420                                   Selector &Sel) const {
421   if (Sel.isNull()) {
422     SmallVector<IdentifierInfo *, 4> Idents;
423     for (ArrayRef<StringRef>::const_iterator
424            I = Ids.begin(), E = Ids.end(); I != E; ++I)
425       Idents.push_back(&Ctx.Idents.get(*I));
426     Sel = Ctx.Selectors.getSelector(Idents.size(), Idents.data());
427   }
428   return Sel;
429 }
430