xref: /freebsd/contrib/llvm-project/clang/lib/Edit/RewriteObjCFoundationAPI.cpp (revision 734e82fe33aa764367791a7d603b383996c6b40b)
1 //===--- RewriteObjCFoundationAPI.cpp - Foundation API Rewriter -----------===//
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 // Rewrites legacy method calls to modern syntax.
10 //
11 //===----------------------------------------------------------------------===//
12 
13 #include "clang/Edit/Rewriters.h"
14 #include "clang/AST/ASTContext.h"
15 #include "clang/AST/ExprCXX.h"
16 #include "clang/AST/ExprObjC.h"
17 #include "clang/AST/NSAPI.h"
18 #include "clang/AST/ParentMap.h"
19 #include "clang/Edit/Commit.h"
20 #include "clang/Lex/Lexer.h"
21 #include <optional>
22 
23 using namespace clang;
24 using namespace edit;
25 
26 static bool checkForLiteralCreation(const ObjCMessageExpr *Msg,
27                                     IdentifierInfo *&ClassId,
28                                     const LangOptions &LangOpts) {
29   if (!Msg || Msg->isImplicit() || !Msg->getMethodDecl())
30     return false;
31 
32   const ObjCInterfaceDecl *Receiver = Msg->getReceiverInterface();
33   if (!Receiver)
34     return false;
35   ClassId = Receiver->getIdentifier();
36 
37   if (Msg->getReceiverKind() == ObjCMessageExpr::Class)
38     return true;
39 
40   // When in ARC mode we also convert "[[.. alloc] init]" messages to literals,
41   // since the change from +1 to +0 will be handled fine by ARC.
42   if (LangOpts.ObjCAutoRefCount) {
43     if (Msg->getReceiverKind() == ObjCMessageExpr::Instance) {
44       if (const ObjCMessageExpr *Rec = dyn_cast<ObjCMessageExpr>(
45                            Msg->getInstanceReceiver()->IgnoreParenImpCasts())) {
46         if (Rec->getMethodFamily() == OMF_alloc)
47           return true;
48       }
49     }
50   }
51 
52   return false;
53 }
54 
55 //===----------------------------------------------------------------------===//
56 // rewriteObjCRedundantCallWithLiteral.
57 //===----------------------------------------------------------------------===//
58 
59 bool edit::rewriteObjCRedundantCallWithLiteral(const ObjCMessageExpr *Msg,
60                                               const NSAPI &NS, Commit &commit) {
61   IdentifierInfo *II = nullptr;
62   if (!checkForLiteralCreation(Msg, II, NS.getASTContext().getLangOpts()))
63     return false;
64   if (Msg->getNumArgs() != 1)
65     return false;
66 
67   const Expr *Arg = Msg->getArg(0)->IgnoreParenImpCasts();
68   Selector Sel = Msg->getSelector();
69 
70   if ((isa<ObjCStringLiteral>(Arg) &&
71        NS.getNSClassId(NSAPI::ClassId_NSString) == II &&
72        (NS.getNSStringSelector(NSAPI::NSStr_stringWithString) == Sel ||
73         NS.getNSStringSelector(NSAPI::NSStr_initWithString) == Sel))   ||
74 
75       (isa<ObjCArrayLiteral>(Arg) &&
76        NS.getNSClassId(NSAPI::ClassId_NSArray) == II &&
77        (NS.getNSArraySelector(NSAPI::NSArr_arrayWithArray) == Sel ||
78         NS.getNSArraySelector(NSAPI::NSArr_initWithArray) == Sel))     ||
79 
80       (isa<ObjCDictionaryLiteral>(Arg) &&
81        NS.getNSClassId(NSAPI::ClassId_NSDictionary) == II &&
82        (NS.getNSDictionarySelector(
83                               NSAPI::NSDict_dictionaryWithDictionary) == Sel ||
84         NS.getNSDictionarySelector(NSAPI::NSDict_initWithDictionary) == Sel))) {
85 
86     commit.replaceWithInner(Msg->getSourceRange(),
87                            Msg->getArg(0)->getSourceRange());
88     return true;
89   }
90 
91   return false;
92 }
93 
94 //===----------------------------------------------------------------------===//
95 // rewriteToObjCSubscriptSyntax.
96 //===----------------------------------------------------------------------===//
97 
98 /// Check for classes that accept 'objectForKey:' (or the other selectors
99 /// that the migrator handles) but return their instances as 'id', resulting
100 /// in the compiler resolving 'objectForKey:' as the method from NSDictionary.
101 ///
102 /// When checking if we can convert to subscripting syntax, check whether
103 /// the receiver is a result of a class method from a hardcoded list of
104 /// such classes. In such a case return the specific class as the interface
105 /// of the receiver.
106 ///
107 /// FIXME: Remove this when these classes start using 'instancetype'.
108 static const ObjCInterfaceDecl *
109 maybeAdjustInterfaceForSubscriptingCheck(const ObjCInterfaceDecl *IFace,
110                                          const Expr *Receiver,
111                                          ASTContext &Ctx) {
112   assert(IFace && Receiver);
113 
114   // If the receiver has type 'id'...
115   if (!Ctx.isObjCIdType(Receiver->getType().getUnqualifiedType()))
116     return IFace;
117 
118   const ObjCMessageExpr *
119     InnerMsg = dyn_cast<ObjCMessageExpr>(Receiver->IgnoreParenCasts());
120   if (!InnerMsg)
121     return IFace;
122 
123   QualType ClassRec;
124   switch (InnerMsg->getReceiverKind()) {
125   case ObjCMessageExpr::Instance:
126   case ObjCMessageExpr::SuperInstance:
127     return IFace;
128 
129   case ObjCMessageExpr::Class:
130     ClassRec = InnerMsg->getClassReceiver();
131     break;
132   case ObjCMessageExpr::SuperClass:
133     ClassRec = InnerMsg->getSuperType();
134     break;
135   }
136 
137   if (ClassRec.isNull())
138     return IFace;
139 
140   // ...and it is the result of a class message...
141 
142   const ObjCObjectType *ObjTy = ClassRec->getAs<ObjCObjectType>();
143   if (!ObjTy)
144     return IFace;
145   const ObjCInterfaceDecl *OID = ObjTy->getInterface();
146 
147   // ...and the receiving class is NSMapTable or NSLocale, return that
148   // class as the receiving interface.
149   if (OID->getName() == "NSMapTable" ||
150       OID->getName() == "NSLocale")
151     return OID;
152 
153   return IFace;
154 }
155 
156 static bool canRewriteToSubscriptSyntax(const ObjCInterfaceDecl *&IFace,
157                                         const ObjCMessageExpr *Msg,
158                                         ASTContext &Ctx,
159                                         Selector subscriptSel) {
160   const Expr *Rec = Msg->getInstanceReceiver();
161   if (!Rec)
162     return false;
163   IFace = maybeAdjustInterfaceForSubscriptingCheck(IFace, Rec, Ctx);
164 
165   if (const ObjCMethodDecl *MD = IFace->lookupInstanceMethod(subscriptSel)) {
166     if (!MD->isUnavailable())
167       return true;
168   }
169   return false;
170 }
171 
172 static bool subscriptOperatorNeedsParens(const Expr *FullExpr);
173 
174 static void maybePutParensOnReceiver(const Expr *Receiver, Commit &commit) {
175   if (subscriptOperatorNeedsParens(Receiver)) {
176     SourceRange RecRange = Receiver->getSourceRange();
177     commit.insertWrap("(", RecRange, ")");
178   }
179 }
180 
181 static bool rewriteToSubscriptGetCommon(const ObjCMessageExpr *Msg,
182                                         Commit &commit) {
183   if (Msg->getNumArgs() != 1)
184     return false;
185   const Expr *Rec = Msg->getInstanceReceiver();
186   if (!Rec)
187     return false;
188 
189   SourceRange MsgRange = Msg->getSourceRange();
190   SourceRange RecRange = Rec->getSourceRange();
191   SourceRange ArgRange = Msg->getArg(0)->getSourceRange();
192 
193   commit.replaceWithInner(CharSourceRange::getCharRange(MsgRange.getBegin(),
194                                                        ArgRange.getBegin()),
195                          CharSourceRange::getTokenRange(RecRange));
196   commit.replaceWithInner(SourceRange(ArgRange.getBegin(), MsgRange.getEnd()),
197                          ArgRange);
198   commit.insertWrap("[", ArgRange, "]");
199   maybePutParensOnReceiver(Rec, commit);
200   return true;
201 }
202 
203 static bool rewriteToArraySubscriptGet(const ObjCInterfaceDecl *IFace,
204                                        const ObjCMessageExpr *Msg,
205                                        const NSAPI &NS,
206                                        Commit &commit) {
207   if (!canRewriteToSubscriptSyntax(IFace, Msg, NS.getASTContext(),
208                                    NS.getObjectAtIndexedSubscriptSelector()))
209     return false;
210   return rewriteToSubscriptGetCommon(Msg, commit);
211 }
212 
213 static bool rewriteToDictionarySubscriptGet(const ObjCInterfaceDecl *IFace,
214                                             const ObjCMessageExpr *Msg,
215                                             const NSAPI &NS,
216                                             Commit &commit) {
217   if (!canRewriteToSubscriptSyntax(IFace, Msg, NS.getASTContext(),
218                                   NS.getObjectForKeyedSubscriptSelector()))
219     return false;
220   return rewriteToSubscriptGetCommon(Msg, commit);
221 }
222 
223 static bool rewriteToArraySubscriptSet(const ObjCInterfaceDecl *IFace,
224                                        const ObjCMessageExpr *Msg,
225                                        const NSAPI &NS,
226                                        Commit &commit) {
227   if (!canRewriteToSubscriptSyntax(IFace, Msg, NS.getASTContext(),
228                                    NS.getSetObjectAtIndexedSubscriptSelector()))
229     return false;
230 
231   if (Msg->getNumArgs() != 2)
232     return false;
233   const Expr *Rec = Msg->getInstanceReceiver();
234   if (!Rec)
235     return false;
236 
237   SourceRange MsgRange = Msg->getSourceRange();
238   SourceRange RecRange = Rec->getSourceRange();
239   SourceRange Arg0Range = Msg->getArg(0)->getSourceRange();
240   SourceRange Arg1Range = Msg->getArg(1)->getSourceRange();
241 
242   commit.replaceWithInner(CharSourceRange::getCharRange(MsgRange.getBegin(),
243                                                        Arg0Range.getBegin()),
244                          CharSourceRange::getTokenRange(RecRange));
245   commit.replaceWithInner(CharSourceRange::getCharRange(Arg0Range.getBegin(),
246                                                        Arg1Range.getBegin()),
247                          CharSourceRange::getTokenRange(Arg0Range));
248   commit.replaceWithInner(SourceRange(Arg1Range.getBegin(), MsgRange.getEnd()),
249                          Arg1Range);
250   commit.insertWrap("[", CharSourceRange::getCharRange(Arg0Range.getBegin(),
251                                                        Arg1Range.getBegin()),
252                     "] = ");
253   maybePutParensOnReceiver(Rec, commit);
254   return true;
255 }
256 
257 static bool rewriteToDictionarySubscriptSet(const ObjCInterfaceDecl *IFace,
258                                             const ObjCMessageExpr *Msg,
259                                             const NSAPI &NS,
260                                             Commit &commit) {
261   if (!canRewriteToSubscriptSyntax(IFace, Msg, NS.getASTContext(),
262                                    NS.getSetObjectForKeyedSubscriptSelector()))
263     return false;
264 
265   if (Msg->getNumArgs() != 2)
266     return false;
267   const Expr *Rec = Msg->getInstanceReceiver();
268   if (!Rec)
269     return false;
270 
271   SourceRange MsgRange = Msg->getSourceRange();
272   SourceRange RecRange = Rec->getSourceRange();
273   SourceRange Arg0Range = Msg->getArg(0)->getSourceRange();
274   SourceRange Arg1Range = Msg->getArg(1)->getSourceRange();
275 
276   SourceLocation LocBeforeVal = Arg0Range.getBegin();
277   commit.insertBefore(LocBeforeVal, "] = ");
278   commit.insertFromRange(LocBeforeVal, Arg1Range, /*afterToken=*/false,
279                          /*beforePreviousInsertions=*/true);
280   commit.insertBefore(LocBeforeVal, "[");
281   commit.replaceWithInner(CharSourceRange::getCharRange(MsgRange.getBegin(),
282                                                        Arg0Range.getBegin()),
283                          CharSourceRange::getTokenRange(RecRange));
284   commit.replaceWithInner(SourceRange(Arg0Range.getBegin(), MsgRange.getEnd()),
285                          Arg0Range);
286   maybePutParensOnReceiver(Rec, commit);
287   return true;
288 }
289 
290 bool edit::rewriteToObjCSubscriptSyntax(const ObjCMessageExpr *Msg,
291                                         const NSAPI &NS, Commit &commit) {
292   if (!Msg || Msg->isImplicit() ||
293       Msg->getReceiverKind() != ObjCMessageExpr::Instance)
294     return false;
295   const ObjCMethodDecl *Method = Msg->getMethodDecl();
296   if (!Method)
297     return false;
298 
299   const ObjCInterfaceDecl *IFace =
300       NS.getASTContext().getObjContainingInterface(Method);
301   if (!IFace)
302     return false;
303   Selector Sel = Msg->getSelector();
304 
305   if (Sel == NS.getNSArraySelector(NSAPI::NSArr_objectAtIndex))
306     return rewriteToArraySubscriptGet(IFace, Msg, NS, commit);
307 
308   if (Sel == NS.getNSDictionarySelector(NSAPI::NSDict_objectForKey))
309     return rewriteToDictionarySubscriptGet(IFace, Msg, NS, commit);
310 
311   if (Msg->getNumArgs() != 2)
312     return false;
313 
314   if (Sel == NS.getNSArraySelector(NSAPI::NSMutableArr_replaceObjectAtIndex))
315     return rewriteToArraySubscriptSet(IFace, Msg, NS, commit);
316 
317   if (Sel == NS.getNSDictionarySelector(NSAPI::NSMutableDict_setObjectForKey))
318     return rewriteToDictionarySubscriptSet(IFace, Msg, NS, commit);
319 
320   return false;
321 }
322 
323 //===----------------------------------------------------------------------===//
324 // rewriteToObjCLiteralSyntax.
325 //===----------------------------------------------------------------------===//
326 
327 static bool rewriteToArrayLiteral(const ObjCMessageExpr *Msg,
328                                   const NSAPI &NS, Commit &commit,
329                                   const ParentMap *PMap);
330 static bool rewriteToDictionaryLiteral(const ObjCMessageExpr *Msg,
331                                   const NSAPI &NS, Commit &commit);
332 static bool rewriteToNumberLiteral(const ObjCMessageExpr *Msg,
333                                   const NSAPI &NS, Commit &commit);
334 static bool rewriteToNumericBoxedExpression(const ObjCMessageExpr *Msg,
335                                             const NSAPI &NS, Commit &commit);
336 static bool rewriteToStringBoxedExpression(const ObjCMessageExpr *Msg,
337                                            const NSAPI &NS, Commit &commit);
338 
339 bool edit::rewriteToObjCLiteralSyntax(const ObjCMessageExpr *Msg,
340                                       const NSAPI &NS, Commit &commit,
341                                       const ParentMap *PMap) {
342   IdentifierInfo *II = nullptr;
343   if (!checkForLiteralCreation(Msg, II, NS.getASTContext().getLangOpts()))
344     return false;
345 
346   if (II == NS.getNSClassId(NSAPI::ClassId_NSArray))
347     return rewriteToArrayLiteral(Msg, NS, commit, PMap);
348   if (II == NS.getNSClassId(NSAPI::ClassId_NSDictionary))
349     return rewriteToDictionaryLiteral(Msg, NS, commit);
350   if (II == NS.getNSClassId(NSAPI::ClassId_NSNumber))
351     return rewriteToNumberLiteral(Msg, NS, commit);
352   if (II == NS.getNSClassId(NSAPI::ClassId_NSString))
353     return rewriteToStringBoxedExpression(Msg, NS, commit);
354 
355   return false;
356 }
357 
358 /// Returns true if the immediate message arguments of \c Msg should not
359 /// be rewritten because it will interfere with the rewrite of the parent
360 /// message expression. e.g.
361 /// \code
362 ///   [NSDictionary dictionaryWithObjects:
363 ///                                 [NSArray arrayWithObjects:@"1", @"2", nil]
364 ///                         forKeys:[NSArray arrayWithObjects:@"A", @"B", nil]];
365 /// \endcode
366 /// It will return true for this because we are going to rewrite this directly
367 /// to a dictionary literal without any array literals.
368 static bool shouldNotRewriteImmediateMessageArgs(const ObjCMessageExpr *Msg,
369                                                  const NSAPI &NS);
370 
371 //===----------------------------------------------------------------------===//
372 // rewriteToArrayLiteral.
373 //===----------------------------------------------------------------------===//
374 
375 /// Adds an explicit cast to 'id' if the type is not objc object.
376 static void objectifyExpr(const Expr *E, Commit &commit);
377 
378 static bool rewriteToArrayLiteral(const ObjCMessageExpr *Msg,
379                                   const NSAPI &NS, Commit &commit,
380                                   const ParentMap *PMap) {
381   if (PMap) {
382     const ObjCMessageExpr *ParentMsg =
383         dyn_cast_or_null<ObjCMessageExpr>(PMap->getParentIgnoreParenCasts(Msg));
384     if (shouldNotRewriteImmediateMessageArgs(ParentMsg, NS))
385       return false;
386   }
387 
388   Selector Sel = Msg->getSelector();
389   SourceRange MsgRange = Msg->getSourceRange();
390 
391   if (Sel == NS.getNSArraySelector(NSAPI::NSArr_array)) {
392     if (Msg->getNumArgs() != 0)
393       return false;
394     commit.replace(MsgRange, "@[]");
395     return true;
396   }
397 
398   if (Sel == NS.getNSArraySelector(NSAPI::NSArr_arrayWithObject)) {
399     if (Msg->getNumArgs() != 1)
400       return false;
401     objectifyExpr(Msg->getArg(0), commit);
402     SourceRange ArgRange = Msg->getArg(0)->getSourceRange();
403     commit.replaceWithInner(MsgRange, ArgRange);
404     commit.insertWrap("@[", ArgRange, "]");
405     return true;
406   }
407 
408   if (Sel == NS.getNSArraySelector(NSAPI::NSArr_arrayWithObjects) ||
409       Sel == NS.getNSArraySelector(NSAPI::NSArr_initWithObjects)) {
410     if (Msg->getNumArgs() == 0)
411       return false;
412     const Expr *SentinelExpr = Msg->getArg(Msg->getNumArgs() - 1);
413     if (!NS.getASTContext().isSentinelNullExpr(SentinelExpr))
414       return false;
415 
416     for (unsigned i = 0, e = Msg->getNumArgs() - 1; i != e; ++i)
417       objectifyExpr(Msg->getArg(i), commit);
418 
419     if (Msg->getNumArgs() == 1) {
420       commit.replace(MsgRange, "@[]");
421       return true;
422     }
423     SourceRange ArgRange(Msg->getArg(0)->getBeginLoc(),
424                          Msg->getArg(Msg->getNumArgs() - 2)->getEndLoc());
425     commit.replaceWithInner(MsgRange, ArgRange);
426     commit.insertWrap("@[", ArgRange, "]");
427     return true;
428   }
429 
430   return false;
431 }
432 
433 //===----------------------------------------------------------------------===//
434 // rewriteToDictionaryLiteral.
435 //===----------------------------------------------------------------------===//
436 
437 /// If \c Msg is an NSArray creation message or literal, this gets the
438 /// objects that were used to create it.
439 /// \returns true if it is an NSArray and we got objects, or false otherwise.
440 static bool getNSArrayObjects(const Expr *E, const NSAPI &NS,
441                               SmallVectorImpl<const Expr *> &Objs) {
442   if (!E)
443     return false;
444 
445   E = E->IgnoreParenCasts();
446   if (!E)
447     return false;
448 
449   if (const ObjCMessageExpr *Msg = dyn_cast<ObjCMessageExpr>(E)) {
450     IdentifierInfo *Cls = nullptr;
451     if (!checkForLiteralCreation(Msg, Cls, NS.getASTContext().getLangOpts()))
452       return false;
453 
454     if (Cls != NS.getNSClassId(NSAPI::ClassId_NSArray))
455       return false;
456 
457     Selector Sel = Msg->getSelector();
458     if (Sel == NS.getNSArraySelector(NSAPI::NSArr_array))
459       return true; // empty array.
460 
461     if (Sel == NS.getNSArraySelector(NSAPI::NSArr_arrayWithObject)) {
462       if (Msg->getNumArgs() != 1)
463         return false;
464       Objs.push_back(Msg->getArg(0));
465       return true;
466     }
467 
468     if (Sel == NS.getNSArraySelector(NSAPI::NSArr_arrayWithObjects) ||
469         Sel == NS.getNSArraySelector(NSAPI::NSArr_initWithObjects)) {
470       if (Msg->getNumArgs() == 0)
471         return false;
472       const Expr *SentinelExpr = Msg->getArg(Msg->getNumArgs() - 1);
473       if (!NS.getASTContext().isSentinelNullExpr(SentinelExpr))
474         return false;
475 
476       for (unsigned i = 0, e = Msg->getNumArgs() - 1; i != e; ++i)
477         Objs.push_back(Msg->getArg(i));
478       return true;
479     }
480 
481   } else if (const ObjCArrayLiteral *ArrLit = dyn_cast<ObjCArrayLiteral>(E)) {
482     for (unsigned i = 0, e = ArrLit->getNumElements(); i != e; ++i)
483       Objs.push_back(ArrLit->getElement(i));
484     return true;
485   }
486 
487   return false;
488 }
489 
490 static bool rewriteToDictionaryLiteral(const ObjCMessageExpr *Msg,
491                                        const NSAPI &NS, Commit &commit) {
492   Selector Sel = Msg->getSelector();
493   SourceRange MsgRange = Msg->getSourceRange();
494 
495   if (Sel == NS.getNSDictionarySelector(NSAPI::NSDict_dictionary)) {
496     if (Msg->getNumArgs() != 0)
497       return false;
498     commit.replace(MsgRange, "@{}");
499     return true;
500   }
501 
502   if (Sel == NS.getNSDictionarySelector(
503                                     NSAPI::NSDict_dictionaryWithObjectForKey)) {
504     if (Msg->getNumArgs() != 2)
505       return false;
506 
507     objectifyExpr(Msg->getArg(0), commit);
508     objectifyExpr(Msg->getArg(1), commit);
509 
510     SourceRange ValRange = Msg->getArg(0)->getSourceRange();
511     SourceRange KeyRange = Msg->getArg(1)->getSourceRange();
512     // Insert key before the value.
513     commit.insertBefore(ValRange.getBegin(), ": ");
514     commit.insertFromRange(ValRange.getBegin(),
515                            CharSourceRange::getTokenRange(KeyRange),
516                        /*afterToken=*/false, /*beforePreviousInsertions=*/true);
517     commit.insertBefore(ValRange.getBegin(), "@{");
518     commit.insertAfterToken(ValRange.getEnd(), "}");
519     commit.replaceWithInner(MsgRange, ValRange);
520     return true;
521   }
522 
523   if (Sel == NS.getNSDictionarySelector(
524                                   NSAPI::NSDict_dictionaryWithObjectsAndKeys) ||
525       Sel == NS.getNSDictionarySelector(NSAPI::NSDict_initWithObjectsAndKeys)) {
526     if (Msg->getNumArgs() % 2 != 1)
527       return false;
528     unsigned SentinelIdx = Msg->getNumArgs() - 1;
529     const Expr *SentinelExpr = Msg->getArg(SentinelIdx);
530     if (!NS.getASTContext().isSentinelNullExpr(SentinelExpr))
531       return false;
532 
533     if (Msg->getNumArgs() == 1) {
534       commit.replace(MsgRange, "@{}");
535       return true;
536     }
537 
538     for (unsigned i = 0; i < SentinelIdx; i += 2) {
539       objectifyExpr(Msg->getArg(i), commit);
540       objectifyExpr(Msg->getArg(i+1), commit);
541 
542       SourceRange ValRange = Msg->getArg(i)->getSourceRange();
543       SourceRange KeyRange = Msg->getArg(i+1)->getSourceRange();
544       // Insert value after key.
545       commit.insertAfterToken(KeyRange.getEnd(), ": ");
546       commit.insertFromRange(KeyRange.getEnd(), ValRange, /*afterToken=*/true);
547       commit.remove(CharSourceRange::getCharRange(ValRange.getBegin(),
548                                                   KeyRange.getBegin()));
549     }
550     // Range of arguments up until and including the last key.
551     // The sentinel and first value are cut off, the value will move after the
552     // key.
553     SourceRange ArgRange(Msg->getArg(1)->getBeginLoc(),
554                          Msg->getArg(SentinelIdx - 1)->getEndLoc());
555     commit.insertWrap("@{", ArgRange, "}");
556     commit.replaceWithInner(MsgRange, ArgRange);
557     return true;
558   }
559 
560   if (Sel == NS.getNSDictionarySelector(
561                                   NSAPI::NSDict_dictionaryWithObjectsForKeys) ||
562       Sel == NS.getNSDictionarySelector(NSAPI::NSDict_initWithObjectsForKeys)) {
563     if (Msg->getNumArgs() != 2)
564       return false;
565 
566     SmallVector<const Expr *, 8> Vals;
567     if (!getNSArrayObjects(Msg->getArg(0), NS, Vals))
568       return false;
569 
570     SmallVector<const Expr *, 8> Keys;
571     if (!getNSArrayObjects(Msg->getArg(1), NS, Keys))
572       return false;
573 
574     if (Vals.size() != Keys.size())
575       return false;
576 
577     if (Vals.empty()) {
578       commit.replace(MsgRange, "@{}");
579       return true;
580     }
581 
582     for (unsigned i = 0, n = Vals.size(); i < n; ++i) {
583       objectifyExpr(Vals[i], commit);
584       objectifyExpr(Keys[i], commit);
585 
586       SourceRange ValRange = Vals[i]->getSourceRange();
587       SourceRange KeyRange = Keys[i]->getSourceRange();
588       // Insert value after key.
589       commit.insertAfterToken(KeyRange.getEnd(), ": ");
590       commit.insertFromRange(KeyRange.getEnd(), ValRange, /*afterToken=*/true);
591     }
592     // Range of arguments up until and including the last key.
593     // The first value is cut off, the value will move after the key.
594     SourceRange ArgRange(Keys.front()->getBeginLoc(), Keys.back()->getEndLoc());
595     commit.insertWrap("@{", ArgRange, "}");
596     commit.replaceWithInner(MsgRange, ArgRange);
597     return true;
598   }
599 
600   return false;
601 }
602 
603 static bool shouldNotRewriteImmediateMessageArgs(const ObjCMessageExpr *Msg,
604                                                  const NSAPI &NS) {
605   if (!Msg)
606     return false;
607 
608   IdentifierInfo *II = nullptr;
609   if (!checkForLiteralCreation(Msg, II, NS.getASTContext().getLangOpts()))
610     return false;
611 
612   if (II != NS.getNSClassId(NSAPI::ClassId_NSDictionary))
613     return false;
614 
615   Selector Sel = Msg->getSelector();
616   if (Sel == NS.getNSDictionarySelector(
617                                   NSAPI::NSDict_dictionaryWithObjectsForKeys) ||
618       Sel == NS.getNSDictionarySelector(NSAPI::NSDict_initWithObjectsForKeys)) {
619     if (Msg->getNumArgs() != 2)
620       return false;
621 
622     SmallVector<const Expr *, 8> Vals;
623     if (!getNSArrayObjects(Msg->getArg(0), NS, Vals))
624       return false;
625 
626     SmallVector<const Expr *, 8> Keys;
627     if (!getNSArrayObjects(Msg->getArg(1), NS, Keys))
628       return false;
629 
630     if (Vals.size() != Keys.size())
631       return false;
632 
633     return true;
634   }
635 
636   return false;
637 }
638 
639 //===----------------------------------------------------------------------===//
640 // rewriteToNumberLiteral.
641 //===----------------------------------------------------------------------===//
642 
643 static bool rewriteToCharLiteral(const ObjCMessageExpr *Msg,
644                                    const CharacterLiteral *Arg,
645                                    const NSAPI &NS, Commit &commit) {
646   if (Arg->getKind() != CharacterLiteral::Ascii)
647     return false;
648   if (NS.isNSNumberLiteralSelector(NSAPI::NSNumberWithChar,
649                                    Msg->getSelector())) {
650     SourceRange ArgRange = Arg->getSourceRange();
651     commit.replaceWithInner(Msg->getSourceRange(), ArgRange);
652     commit.insert(ArgRange.getBegin(), "@");
653     return true;
654   }
655 
656   return rewriteToNumericBoxedExpression(Msg, NS, commit);
657 }
658 
659 static bool rewriteToBoolLiteral(const ObjCMessageExpr *Msg,
660                                    const Expr *Arg,
661                                    const NSAPI &NS, Commit &commit) {
662   if (NS.isNSNumberLiteralSelector(NSAPI::NSNumberWithBool,
663                                    Msg->getSelector())) {
664     SourceRange ArgRange = Arg->getSourceRange();
665     commit.replaceWithInner(Msg->getSourceRange(), ArgRange);
666     commit.insert(ArgRange.getBegin(), "@");
667     return true;
668   }
669 
670   return rewriteToNumericBoxedExpression(Msg, NS, commit);
671 }
672 
673 namespace {
674 
675 struct LiteralInfo {
676   bool Hex, Octal;
677   StringRef U, F, L, LL;
678   CharSourceRange WithoutSuffRange;
679 };
680 
681 }
682 
683 static bool getLiteralInfo(SourceRange literalRange,
684                            bool isFloat, bool isIntZero,
685                           ASTContext &Ctx, LiteralInfo &Info) {
686   if (literalRange.getBegin().isMacroID() ||
687       literalRange.getEnd().isMacroID())
688     return false;
689   StringRef text = Lexer::getSourceText(
690                                   CharSourceRange::getTokenRange(literalRange),
691                                   Ctx.getSourceManager(), Ctx.getLangOpts());
692   if (text.empty())
693     return false;
694 
695   std::optional<bool> UpperU, UpperL;
696   bool UpperF = false;
697 
698   struct Suff {
699     static bool has(StringRef suff, StringRef &text) {
700       if (text.endswith(suff)) {
701         text = text.substr(0, text.size()-suff.size());
702         return true;
703       }
704       return false;
705     }
706   };
707 
708   while (true) {
709     if (Suff::has("u", text)) {
710       UpperU = false;
711     } else if (Suff::has("U", text)) {
712       UpperU = true;
713     } else if (Suff::has("ll", text)) {
714       UpperL = false;
715     } else if (Suff::has("LL", text)) {
716       UpperL = true;
717     } else if (Suff::has("l", text)) {
718       UpperL = false;
719     } else if (Suff::has("L", text)) {
720       UpperL = true;
721     } else if (isFloat && Suff::has("f", text)) {
722       UpperF = false;
723     } else if (isFloat && Suff::has("F", text)) {
724       UpperF = true;
725     } else
726       break;
727   }
728 
729   if (!UpperU && !UpperL)
730     UpperU = UpperL = true;
731   else if (UpperU && !UpperL)
732     UpperL = UpperU;
733   else if (UpperL && !UpperU)
734     UpperU = UpperL;
735 
736   Info.U = *UpperU ? "U" : "u";
737   Info.L = *UpperL ? "L" : "l";
738   Info.LL = *UpperL ? "LL" : "ll";
739   Info.F = UpperF ? "F" : "f";
740 
741   Info.Hex = Info.Octal = false;
742   if (text.startswith("0x"))
743     Info.Hex = true;
744   else if (!isFloat && !isIntZero && text.startswith("0"))
745     Info.Octal = true;
746 
747   SourceLocation B = literalRange.getBegin();
748   Info.WithoutSuffRange =
749       CharSourceRange::getCharRange(B, B.getLocWithOffset(text.size()));
750   return true;
751 }
752 
753 static bool rewriteToNumberLiteral(const ObjCMessageExpr *Msg,
754                                    const NSAPI &NS, Commit &commit) {
755   if (Msg->getNumArgs() != 1)
756     return false;
757 
758   const Expr *Arg = Msg->getArg(0)->IgnoreParenImpCasts();
759   if (const CharacterLiteral *CharE = dyn_cast<CharacterLiteral>(Arg))
760     return rewriteToCharLiteral(Msg, CharE, NS, commit);
761   if (const ObjCBoolLiteralExpr *BE = dyn_cast<ObjCBoolLiteralExpr>(Arg))
762     return rewriteToBoolLiteral(Msg, BE, NS, commit);
763   if (const CXXBoolLiteralExpr *BE = dyn_cast<CXXBoolLiteralExpr>(Arg))
764     return rewriteToBoolLiteral(Msg, BE, NS, commit);
765 
766   const Expr *literalE = Arg;
767   if (const UnaryOperator *UOE = dyn_cast<UnaryOperator>(literalE)) {
768     if (UOE->getOpcode() == UO_Plus || UOE->getOpcode() == UO_Minus)
769       literalE = UOE->getSubExpr();
770   }
771 
772   // Only integer and floating literals, otherwise try to rewrite to boxed
773   // expression.
774   if (!isa<IntegerLiteral>(literalE) && !isa<FloatingLiteral>(literalE))
775     return rewriteToNumericBoxedExpression(Msg, NS, commit);
776 
777   ASTContext &Ctx = NS.getASTContext();
778   Selector Sel = Msg->getSelector();
779   std::optional<NSAPI::NSNumberLiteralMethodKind> MKOpt =
780       NS.getNSNumberLiteralMethodKind(Sel);
781   if (!MKOpt)
782     return false;
783   NSAPI::NSNumberLiteralMethodKind MK = *MKOpt;
784 
785   bool CallIsUnsigned = false, CallIsLong = false, CallIsLongLong = false;
786   bool CallIsFloating = false, CallIsDouble = false;
787 
788   switch (MK) {
789   // We cannot have these calls with int/float literals.
790   case NSAPI::NSNumberWithChar:
791   case NSAPI::NSNumberWithUnsignedChar:
792   case NSAPI::NSNumberWithShort:
793   case NSAPI::NSNumberWithUnsignedShort:
794   case NSAPI::NSNumberWithBool:
795     return rewriteToNumericBoxedExpression(Msg, NS, commit);
796 
797   case NSAPI::NSNumberWithUnsignedInt:
798   case NSAPI::NSNumberWithUnsignedInteger:
799     CallIsUnsigned = true;
800     [[fallthrough]];
801   case NSAPI::NSNumberWithInt:
802   case NSAPI::NSNumberWithInteger:
803     break;
804 
805   case NSAPI::NSNumberWithUnsignedLong:
806     CallIsUnsigned = true;
807     [[fallthrough]];
808   case NSAPI::NSNumberWithLong:
809     CallIsLong = true;
810     break;
811 
812   case NSAPI::NSNumberWithUnsignedLongLong:
813     CallIsUnsigned = true;
814     [[fallthrough]];
815   case NSAPI::NSNumberWithLongLong:
816     CallIsLongLong = true;
817     break;
818 
819   case NSAPI::NSNumberWithDouble:
820     CallIsDouble = true;
821     [[fallthrough]];
822   case NSAPI::NSNumberWithFloat:
823     CallIsFloating = true;
824     break;
825   }
826 
827   SourceRange ArgRange = Arg->getSourceRange();
828   QualType ArgTy = Arg->getType();
829   QualType CallTy = Msg->getArg(0)->getType();
830 
831   // Check for the easy case, the literal maps directly to the call.
832   if (Ctx.hasSameType(ArgTy, CallTy)) {
833     commit.replaceWithInner(Msg->getSourceRange(), ArgRange);
834     commit.insert(ArgRange.getBegin(), "@");
835     return true;
836   }
837 
838   // We will need to modify the literal suffix to get the same type as the call.
839   // Try with boxed expression if it came from a macro.
840   if (ArgRange.getBegin().isMacroID())
841     return rewriteToNumericBoxedExpression(Msg, NS, commit);
842 
843   bool LitIsFloat = ArgTy->isFloatingType();
844   // For a float passed to integer call, don't try rewriting to objc literal.
845   // It is difficult and a very uncommon case anyway.
846   // But try with boxed expression.
847   if (LitIsFloat && !CallIsFloating)
848     return rewriteToNumericBoxedExpression(Msg, NS, commit);
849 
850   // Try to modify the literal make it the same type as the method call.
851   // -Modify the suffix, and/or
852   // -Change integer to float
853 
854   LiteralInfo LitInfo;
855   bool isIntZero = false;
856   if (const IntegerLiteral *IntE = dyn_cast<IntegerLiteral>(literalE))
857     isIntZero = !IntE->getValue().getBoolValue();
858   if (!getLiteralInfo(ArgRange, LitIsFloat, isIntZero, Ctx, LitInfo))
859     return rewriteToNumericBoxedExpression(Msg, NS, commit);
860 
861   // Not easy to do int -> float with hex/octal and uncommon anyway.
862   if (!LitIsFloat && CallIsFloating && (LitInfo.Hex || LitInfo.Octal))
863     return rewriteToNumericBoxedExpression(Msg, NS, commit);
864 
865   SourceLocation LitB = LitInfo.WithoutSuffRange.getBegin();
866   SourceLocation LitE = LitInfo.WithoutSuffRange.getEnd();
867 
868   commit.replaceWithInner(CharSourceRange::getTokenRange(Msg->getSourceRange()),
869                          LitInfo.WithoutSuffRange);
870   commit.insert(LitB, "@");
871 
872   if (!LitIsFloat && CallIsFloating)
873     commit.insert(LitE, ".0");
874 
875   if (CallIsFloating) {
876     if (!CallIsDouble)
877       commit.insert(LitE, LitInfo.F);
878   } else {
879     if (CallIsUnsigned)
880       commit.insert(LitE, LitInfo.U);
881 
882     if (CallIsLong)
883       commit.insert(LitE, LitInfo.L);
884     else if (CallIsLongLong)
885       commit.insert(LitE, LitInfo.LL);
886   }
887   return true;
888 }
889 
890 // FIXME: Make determination of operator precedence more general and
891 // make it broadly available.
892 static bool subscriptOperatorNeedsParens(const Expr *FullExpr) {
893   const Expr* Expr = FullExpr->IgnoreImpCasts();
894   if (isa<ArraySubscriptExpr>(Expr) ||
895       isa<CallExpr>(Expr) ||
896       isa<DeclRefExpr>(Expr) ||
897       isa<CXXNamedCastExpr>(Expr) ||
898       isa<CXXConstructExpr>(Expr) ||
899       isa<CXXThisExpr>(Expr) ||
900       isa<CXXTypeidExpr>(Expr) ||
901       isa<CXXUnresolvedConstructExpr>(Expr) ||
902       isa<ObjCMessageExpr>(Expr) ||
903       isa<ObjCPropertyRefExpr>(Expr) ||
904       isa<ObjCProtocolExpr>(Expr) ||
905       isa<MemberExpr>(Expr) ||
906       isa<ObjCIvarRefExpr>(Expr) ||
907       isa<ParenExpr>(FullExpr) ||
908       isa<ParenListExpr>(Expr) ||
909       isa<SizeOfPackExpr>(Expr))
910     return false;
911 
912   return true;
913 }
914 static bool castOperatorNeedsParens(const Expr *FullExpr) {
915   const Expr* Expr = FullExpr->IgnoreImpCasts();
916   if (isa<ArraySubscriptExpr>(Expr) ||
917       isa<CallExpr>(Expr) ||
918       isa<DeclRefExpr>(Expr) ||
919       isa<CastExpr>(Expr) ||
920       isa<CXXNewExpr>(Expr) ||
921       isa<CXXConstructExpr>(Expr) ||
922       isa<CXXDeleteExpr>(Expr) ||
923       isa<CXXNoexceptExpr>(Expr) ||
924       isa<CXXPseudoDestructorExpr>(Expr) ||
925       isa<CXXScalarValueInitExpr>(Expr) ||
926       isa<CXXThisExpr>(Expr) ||
927       isa<CXXTypeidExpr>(Expr) ||
928       isa<CXXUnresolvedConstructExpr>(Expr) ||
929       isa<ObjCMessageExpr>(Expr) ||
930       isa<ObjCPropertyRefExpr>(Expr) ||
931       isa<ObjCProtocolExpr>(Expr) ||
932       isa<MemberExpr>(Expr) ||
933       isa<ObjCIvarRefExpr>(Expr) ||
934       isa<ParenExpr>(FullExpr) ||
935       isa<ParenListExpr>(Expr) ||
936       isa<SizeOfPackExpr>(Expr) ||
937       isa<UnaryOperator>(Expr))
938     return false;
939 
940   return true;
941 }
942 
943 static void objectifyExpr(const Expr *E, Commit &commit) {
944   if (!E) return;
945 
946   QualType T = E->getType();
947   if (T->isObjCObjectPointerType()) {
948     if (const ImplicitCastExpr *ICE = dyn_cast<ImplicitCastExpr>(E)) {
949       if (ICE->getCastKind() != CK_CPointerToObjCPointerCast)
950         return;
951     } else {
952       return;
953     }
954   } else if (!T->isPointerType()) {
955     return;
956   }
957 
958   SourceRange Range = E->getSourceRange();
959   if (castOperatorNeedsParens(E))
960     commit.insertWrap("(", Range, ")");
961   commit.insertBefore(Range.getBegin(), "(id)");
962 }
963 
964 //===----------------------------------------------------------------------===//
965 // rewriteToNumericBoxedExpression.
966 //===----------------------------------------------------------------------===//
967 
968 static bool isEnumConstant(const Expr *E) {
969   if (const DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E->IgnoreParenImpCasts()))
970     if (const ValueDecl *VD = DRE->getDecl())
971       return isa<EnumConstantDecl>(VD);
972 
973   return false;
974 }
975 
976 static bool rewriteToNumericBoxedExpression(const ObjCMessageExpr *Msg,
977                                             const NSAPI &NS, Commit &commit) {
978   if (Msg->getNumArgs() != 1)
979     return false;
980 
981   const Expr *Arg = Msg->getArg(0);
982   if (Arg->isTypeDependent())
983     return false;
984 
985   ASTContext &Ctx = NS.getASTContext();
986   Selector Sel = Msg->getSelector();
987   std::optional<NSAPI::NSNumberLiteralMethodKind> MKOpt =
988       NS.getNSNumberLiteralMethodKind(Sel);
989   if (!MKOpt)
990     return false;
991   NSAPI::NSNumberLiteralMethodKind MK = *MKOpt;
992 
993   const Expr *OrigArg = Arg->IgnoreImpCasts();
994   QualType FinalTy = Arg->getType();
995   QualType OrigTy = OrigArg->getType();
996   uint64_t FinalTySize = Ctx.getTypeSize(FinalTy);
997   uint64_t OrigTySize = Ctx.getTypeSize(OrigTy);
998 
999   bool isTruncated = FinalTySize < OrigTySize;
1000   bool needsCast = false;
1001 
1002   if (const ImplicitCastExpr *ICE = dyn_cast<ImplicitCastExpr>(Arg)) {
1003     switch (ICE->getCastKind()) {
1004     case CK_LValueToRValue:
1005     case CK_NoOp:
1006     case CK_UserDefinedConversion:
1007       break;
1008 
1009     case CK_IntegralCast: {
1010       if (MK == NSAPI::NSNumberWithBool && OrigTy->isBooleanType())
1011         break;
1012       // Be more liberal with Integer/UnsignedInteger which are very commonly
1013       // used.
1014       if ((MK == NSAPI::NSNumberWithInteger ||
1015            MK == NSAPI::NSNumberWithUnsignedInteger) &&
1016           !isTruncated) {
1017         if (OrigTy->getAs<EnumType>() || isEnumConstant(OrigArg))
1018           break;
1019         if ((MK==NSAPI::NSNumberWithInteger) == OrigTy->isSignedIntegerType() &&
1020             OrigTySize >= Ctx.getTypeSize(Ctx.IntTy))
1021           break;
1022       }
1023 
1024       needsCast = true;
1025       break;
1026     }
1027 
1028     case CK_PointerToBoolean:
1029     case CK_IntegralToBoolean:
1030     case CK_IntegralToFloating:
1031     case CK_FloatingToIntegral:
1032     case CK_FloatingToBoolean:
1033     case CK_FloatingCast:
1034     case CK_FloatingComplexToReal:
1035     case CK_FloatingComplexToBoolean:
1036     case CK_IntegralComplexToReal:
1037     case CK_IntegralComplexToBoolean:
1038     case CK_AtomicToNonAtomic:
1039     case CK_AddressSpaceConversion:
1040       needsCast = true;
1041       break;
1042 
1043     case CK_Dependent:
1044     case CK_BitCast:
1045     case CK_LValueBitCast:
1046     case CK_LValueToRValueBitCast:
1047     case CK_BaseToDerived:
1048     case CK_DerivedToBase:
1049     case CK_UncheckedDerivedToBase:
1050     case CK_Dynamic:
1051     case CK_ToUnion:
1052     case CK_ArrayToPointerDecay:
1053     case CK_FunctionToPointerDecay:
1054     case CK_NullToPointer:
1055     case CK_NullToMemberPointer:
1056     case CK_BaseToDerivedMemberPointer:
1057     case CK_DerivedToBaseMemberPointer:
1058     case CK_MemberPointerToBoolean:
1059     case CK_ReinterpretMemberPointer:
1060     case CK_ConstructorConversion:
1061     case CK_IntegralToPointer:
1062     case CK_PointerToIntegral:
1063     case CK_ToVoid:
1064     case CK_VectorSplat:
1065     case CK_CPointerToObjCPointerCast:
1066     case CK_BlockPointerToObjCPointerCast:
1067     case CK_AnyPointerToBlockPointerCast:
1068     case CK_ObjCObjectLValueCast:
1069     case CK_FloatingRealToComplex:
1070     case CK_FloatingComplexCast:
1071     case CK_FloatingComplexToIntegralComplex:
1072     case CK_IntegralRealToComplex:
1073     case CK_IntegralComplexCast:
1074     case CK_IntegralComplexToFloatingComplex:
1075     case CK_ARCProduceObject:
1076     case CK_ARCConsumeObject:
1077     case CK_ARCReclaimReturnedObject:
1078     case CK_ARCExtendBlockObject:
1079     case CK_NonAtomicToAtomic:
1080     case CK_CopyAndAutoreleaseBlockObject:
1081     case CK_BuiltinFnToFnPtr:
1082     case CK_ZeroToOCLOpaqueType:
1083     case CK_IntToOCLSampler:
1084     case CK_MatrixCast:
1085       return false;
1086 
1087     case CK_BooleanToSignedIntegral:
1088       llvm_unreachable("OpenCL-specific cast in Objective-C?");
1089 
1090     case CK_FloatingToFixedPoint:
1091     case CK_FixedPointToFloating:
1092     case CK_FixedPointCast:
1093     case CK_FixedPointToBoolean:
1094     case CK_FixedPointToIntegral:
1095     case CK_IntegralToFixedPoint:
1096       llvm_unreachable("Fixed point types are disabled for Objective-C");
1097     }
1098   }
1099 
1100   if (needsCast) {
1101     DiagnosticsEngine &Diags = Ctx.getDiagnostics();
1102     // FIXME: Use a custom category name to distinguish migration diagnostics.
1103     unsigned diagID = Diags.getCustomDiagID(DiagnosticsEngine::Warning,
1104                        "converting to boxing syntax requires casting %0 to %1");
1105     Diags.Report(Msg->getExprLoc(), diagID) << OrigTy << FinalTy
1106         << Msg->getSourceRange();
1107     return false;
1108   }
1109 
1110   SourceRange ArgRange = OrigArg->getSourceRange();
1111   commit.replaceWithInner(Msg->getSourceRange(), ArgRange);
1112 
1113   if (isa<ParenExpr>(OrigArg) || isa<IntegerLiteral>(OrigArg))
1114     commit.insertBefore(ArgRange.getBegin(), "@");
1115   else
1116     commit.insertWrap("@(", ArgRange, ")");
1117 
1118   return true;
1119 }
1120 
1121 //===----------------------------------------------------------------------===//
1122 // rewriteToStringBoxedExpression.
1123 //===----------------------------------------------------------------------===//
1124 
1125 static bool doRewriteToUTF8StringBoxedExpressionHelper(
1126                                               const ObjCMessageExpr *Msg,
1127                                               const NSAPI &NS, Commit &commit) {
1128   const Expr *Arg = Msg->getArg(0);
1129   if (Arg->isTypeDependent())
1130     return false;
1131 
1132   ASTContext &Ctx = NS.getASTContext();
1133 
1134   const Expr *OrigArg = Arg->IgnoreImpCasts();
1135   QualType OrigTy = OrigArg->getType();
1136   if (OrigTy->isArrayType())
1137     OrigTy = Ctx.getArrayDecayedType(OrigTy);
1138 
1139   if (const StringLiteral *
1140         StrE = dyn_cast<StringLiteral>(OrigArg->IgnoreParens())) {
1141     commit.replaceWithInner(Msg->getSourceRange(), StrE->getSourceRange());
1142     commit.insert(StrE->getBeginLoc(), "@");
1143     return true;
1144   }
1145 
1146   if (const PointerType *PT = OrigTy->getAs<PointerType>()) {
1147     QualType PointeeType = PT->getPointeeType();
1148     if (Ctx.hasSameUnqualifiedType(PointeeType, Ctx.CharTy)) {
1149       SourceRange ArgRange = OrigArg->getSourceRange();
1150       commit.replaceWithInner(Msg->getSourceRange(), ArgRange);
1151 
1152       if (isa<ParenExpr>(OrigArg) || isa<IntegerLiteral>(OrigArg))
1153         commit.insertBefore(ArgRange.getBegin(), "@");
1154       else
1155         commit.insertWrap("@(", ArgRange, ")");
1156 
1157       return true;
1158     }
1159   }
1160 
1161   return false;
1162 }
1163 
1164 static bool rewriteToStringBoxedExpression(const ObjCMessageExpr *Msg,
1165                                            const NSAPI &NS, Commit &commit) {
1166   Selector Sel = Msg->getSelector();
1167 
1168   if (Sel == NS.getNSStringSelector(NSAPI::NSStr_stringWithUTF8String) ||
1169       Sel == NS.getNSStringSelector(NSAPI::NSStr_stringWithCString) ||
1170       Sel == NS.getNSStringSelector(NSAPI::NSStr_initWithUTF8String)) {
1171     if (Msg->getNumArgs() != 1)
1172       return false;
1173     return doRewriteToUTF8StringBoxedExpressionHelper(Msg, NS, commit);
1174   }
1175 
1176   if (Sel == NS.getNSStringSelector(NSAPI::NSStr_stringWithCStringEncoding)) {
1177     if (Msg->getNumArgs() != 2)
1178       return false;
1179 
1180     const Expr *encodingArg = Msg->getArg(1);
1181     if (NS.isNSUTF8StringEncodingConstant(encodingArg) ||
1182         NS.isNSASCIIStringEncodingConstant(encodingArg))
1183       return doRewriteToUTF8StringBoxedExpressionHelper(Msg, NS, commit);
1184   }
1185 
1186   return false;
1187 }
1188