xref: /freebsd/contrib/llvm-project/clang/lib/Sema/SemaAvailability.cpp (revision 963f5dc7a30624e95d72fb7f87b8892651164e46)
1 //===--- SemaAvailability.cpp - Availability attribute handling -----------===//
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 //  This file processes the availability attribute.
10 //
11 //===----------------------------------------------------------------------===//
12 
13 #include "clang/AST/Attr.h"
14 #include "clang/AST/Decl.h"
15 #include "clang/AST/RecursiveASTVisitor.h"
16 #include "clang/Basic/DiagnosticSema.h"
17 #include "clang/Basic/TargetInfo.h"
18 #include "clang/Lex/Preprocessor.h"
19 #include "clang/Sema/DelayedDiagnostic.h"
20 #include "clang/Sema/ScopeInfo.h"
21 #include "clang/Sema/Sema.h"
22 
23 using namespace clang;
24 using namespace sema;
25 
26 static const AvailabilityAttr *getAttrForPlatform(ASTContext &Context,
27                                                   const Decl *D) {
28   // Check each AvailabilityAttr to find the one for this platform.
29   for (const auto *A : D->attrs()) {
30     if (const auto *Avail = dyn_cast<AvailabilityAttr>(A)) {
31       // FIXME: this is copied from CheckAvailability. We should try to
32       // de-duplicate.
33 
34       // Check if this is an App Extension "platform", and if so chop off
35       // the suffix for matching with the actual platform.
36       StringRef ActualPlatform = Avail->getPlatform()->getName();
37       StringRef RealizedPlatform = ActualPlatform;
38       if (Context.getLangOpts().AppExt) {
39         size_t suffix = RealizedPlatform.rfind("_app_extension");
40         if (suffix != StringRef::npos)
41           RealizedPlatform = RealizedPlatform.slice(0, suffix);
42       }
43 
44       StringRef TargetPlatform = Context.getTargetInfo().getPlatformName();
45 
46       // Match the platform name.
47       if (RealizedPlatform == TargetPlatform)
48         return Avail;
49     }
50   }
51   return nullptr;
52 }
53 
54 /// The diagnostic we should emit for \c D, and the declaration that
55 /// originated it, or \c AR_Available.
56 ///
57 /// \param D The declaration to check.
58 /// \param Message If non-null, this will be populated with the message from
59 /// the availability attribute that is selected.
60 /// \param ClassReceiver If we're checking the the method of a class message
61 /// send, the class. Otherwise nullptr.
62 static std::pair<AvailabilityResult, const NamedDecl *>
63 ShouldDiagnoseAvailabilityOfDecl(Sema &S, const NamedDecl *D,
64                                  std::string *Message,
65                                  ObjCInterfaceDecl *ClassReceiver) {
66   AvailabilityResult Result = D->getAvailability(Message);
67 
68   // For typedefs, if the typedef declaration appears available look
69   // to the underlying type to see if it is more restrictive.
70   while (const auto *TD = dyn_cast<TypedefNameDecl>(D)) {
71     if (Result == AR_Available) {
72       if (const auto *TT = TD->getUnderlyingType()->getAs<TagType>()) {
73         D = TT->getDecl();
74         Result = D->getAvailability(Message);
75         continue;
76       }
77     }
78     break;
79   }
80 
81   // Forward class declarations get their attributes from their definition.
82   if (const auto *IDecl = dyn_cast<ObjCInterfaceDecl>(D)) {
83     if (IDecl->getDefinition()) {
84       D = IDecl->getDefinition();
85       Result = D->getAvailability(Message);
86     }
87   }
88 
89   if (const auto *ECD = dyn_cast<EnumConstantDecl>(D))
90     if (Result == AR_Available) {
91       const DeclContext *DC = ECD->getDeclContext();
92       if (const auto *TheEnumDecl = dyn_cast<EnumDecl>(DC)) {
93         Result = TheEnumDecl->getAvailability(Message);
94         D = TheEnumDecl;
95       }
96     }
97 
98   // For +new, infer availability from -init.
99   if (const auto *MD = dyn_cast<ObjCMethodDecl>(D)) {
100     if (S.NSAPIObj && ClassReceiver) {
101       ObjCMethodDecl *Init = ClassReceiver->lookupInstanceMethod(
102           S.NSAPIObj->getInitSelector());
103       if (Init && Result == AR_Available && MD->isClassMethod() &&
104           MD->getSelector() == S.NSAPIObj->getNewSelector() &&
105           MD->definedInNSObject(S.getASTContext())) {
106         Result = Init->getAvailability(Message);
107         D = Init;
108       }
109     }
110   }
111 
112   return {Result, D};
113 }
114 
115 
116 /// whether we should emit a diagnostic for \c K and \c DeclVersion in
117 /// the context of \c Ctx. For example, we should emit an unavailable diagnostic
118 /// in a deprecated context, but not the other way around.
119 static bool
120 ShouldDiagnoseAvailabilityInContext(Sema &S, AvailabilityResult K,
121                                     VersionTuple DeclVersion, Decl *Ctx,
122                                     const NamedDecl *OffendingDecl) {
123   assert(K != AR_Available && "Expected an unavailable declaration here!");
124 
125   // Checks if we should emit the availability diagnostic in the context of C.
126   auto CheckContext = [&](const Decl *C) {
127     if (K == AR_NotYetIntroduced) {
128       if (const AvailabilityAttr *AA = getAttrForPlatform(S.Context, C))
129         if (AA->getIntroduced() >= DeclVersion)
130           return true;
131     } else if (K == AR_Deprecated) {
132       if (C->isDeprecated())
133         return true;
134     } else if (K == AR_Unavailable) {
135       // It is perfectly fine to refer to an 'unavailable' Objective-C method
136       // when it is referenced from within the @implementation itself. In this
137       // context, we interpret unavailable as a form of access control.
138       if (const auto *MD = dyn_cast<ObjCMethodDecl>(OffendingDecl)) {
139         if (const auto *Impl = dyn_cast<ObjCImplDecl>(C)) {
140           if (MD->getClassInterface() == Impl->getClassInterface())
141             return true;
142         }
143       }
144     }
145 
146     if (C->isUnavailable())
147       return true;
148     return false;
149   };
150 
151   do {
152     if (CheckContext(Ctx))
153       return false;
154 
155     // An implementation implicitly has the availability of the interface.
156     // Unless it is "+load" method.
157     if (const auto *MethodD = dyn_cast<ObjCMethodDecl>(Ctx))
158       if (MethodD->isClassMethod() &&
159           MethodD->getSelector().getAsString() == "load")
160         return true;
161 
162     if (const auto *CatOrImpl = dyn_cast<ObjCImplDecl>(Ctx)) {
163       if (const ObjCInterfaceDecl *Interface = CatOrImpl->getClassInterface())
164         if (CheckContext(Interface))
165           return false;
166     }
167     // A category implicitly has the availability of the interface.
168     else if (const auto *CatD = dyn_cast<ObjCCategoryDecl>(Ctx))
169       if (const ObjCInterfaceDecl *Interface = CatD->getClassInterface())
170         if (CheckContext(Interface))
171           return false;
172   } while ((Ctx = cast_or_null<Decl>(Ctx->getDeclContext())));
173 
174   return true;
175 }
176 
177 static bool
178 shouldDiagnoseAvailabilityByDefault(const ASTContext &Context,
179                                     const VersionTuple &DeploymentVersion,
180                                     const VersionTuple &DeclVersion) {
181   const auto &Triple = Context.getTargetInfo().getTriple();
182   VersionTuple ForceAvailabilityFromVersion;
183   switch (Triple.getOS()) {
184   case llvm::Triple::IOS:
185   case llvm::Triple::TvOS:
186     ForceAvailabilityFromVersion = VersionTuple(/*Major=*/11);
187     break;
188   case llvm::Triple::WatchOS:
189     ForceAvailabilityFromVersion = VersionTuple(/*Major=*/4);
190     break;
191   case llvm::Triple::Darwin:
192   case llvm::Triple::MacOSX:
193     ForceAvailabilityFromVersion = VersionTuple(/*Major=*/10, /*Minor=*/13);
194     break;
195   default:
196     // New targets should always warn about availability.
197     return Triple.getVendor() == llvm::Triple::Apple;
198   }
199   return DeploymentVersion >= ForceAvailabilityFromVersion ||
200          DeclVersion >= ForceAvailabilityFromVersion;
201 }
202 
203 static NamedDecl *findEnclosingDeclToAnnotate(Decl *OrigCtx) {
204   for (Decl *Ctx = OrigCtx; Ctx;
205        Ctx = cast_or_null<Decl>(Ctx->getDeclContext())) {
206     if (isa<TagDecl>(Ctx) || isa<FunctionDecl>(Ctx) || isa<ObjCMethodDecl>(Ctx))
207       return cast<NamedDecl>(Ctx);
208     if (auto *CD = dyn_cast<ObjCContainerDecl>(Ctx)) {
209       if (auto *Imp = dyn_cast<ObjCImplDecl>(Ctx))
210         return Imp->getClassInterface();
211       return CD;
212     }
213   }
214 
215   return dyn_cast<NamedDecl>(OrigCtx);
216 }
217 
218 namespace {
219 
220 struct AttributeInsertion {
221   StringRef Prefix;
222   SourceLocation Loc;
223   StringRef Suffix;
224 
225   static AttributeInsertion createInsertionAfter(const NamedDecl *D) {
226     return {" ", D->getEndLoc(), ""};
227   }
228   static AttributeInsertion createInsertionAfter(SourceLocation Loc) {
229     return {" ", Loc, ""};
230   }
231   static AttributeInsertion createInsertionBefore(const NamedDecl *D) {
232     return {"", D->getBeginLoc(), "\n"};
233   }
234 };
235 
236 } // end anonymous namespace
237 
238 /// Tries to parse a string as ObjC method name.
239 ///
240 /// \param Name The string to parse. Expected to originate from availability
241 /// attribute argument.
242 /// \param SlotNames The vector that will be populated with slot names. In case
243 /// of unsuccessful parsing can contain invalid data.
244 /// \returns A number of method parameters if parsing was successful, None
245 /// otherwise.
246 static Optional<unsigned>
247 tryParseObjCMethodName(StringRef Name, SmallVectorImpl<StringRef> &SlotNames,
248                        const LangOptions &LangOpts) {
249   // Accept replacements starting with - or + as valid ObjC method names.
250   if (!Name.empty() && (Name.front() == '-' || Name.front() == '+'))
251     Name = Name.drop_front(1);
252   if (Name.empty())
253     return None;
254   Name.split(SlotNames, ':');
255   unsigned NumParams;
256   if (Name.back() == ':') {
257     // Remove an empty string at the end that doesn't represent any slot.
258     SlotNames.pop_back();
259     NumParams = SlotNames.size();
260   } else {
261     if (SlotNames.size() != 1)
262       // Not a valid method name, just a colon-separated string.
263       return None;
264     NumParams = 0;
265   }
266   // Verify all slot names are valid.
267   bool AllowDollar = LangOpts.DollarIdents;
268   for (StringRef S : SlotNames) {
269     if (S.empty())
270       continue;
271     if (!isValidIdentifier(S, AllowDollar))
272       return None;
273   }
274   return NumParams;
275 }
276 
277 /// Returns a source location in which it's appropriate to insert a new
278 /// attribute for the given declaration \D.
279 static Optional<AttributeInsertion>
280 createAttributeInsertion(const NamedDecl *D, const SourceManager &SM,
281                          const LangOptions &LangOpts) {
282   if (isa<ObjCPropertyDecl>(D))
283     return AttributeInsertion::createInsertionAfter(D);
284   if (const auto *MD = dyn_cast<ObjCMethodDecl>(D)) {
285     if (MD->hasBody())
286       return None;
287     return AttributeInsertion::createInsertionAfter(D);
288   }
289   if (const auto *TD = dyn_cast<TagDecl>(D)) {
290     SourceLocation Loc =
291         Lexer::getLocForEndOfToken(TD->getInnerLocStart(), 0, SM, LangOpts);
292     if (Loc.isInvalid())
293       return None;
294     // Insert after the 'struct'/whatever keyword.
295     return AttributeInsertion::createInsertionAfter(Loc);
296   }
297   return AttributeInsertion::createInsertionBefore(D);
298 }
299 
300 /// Actually emit an availability diagnostic for a reference to an unavailable
301 /// decl.
302 ///
303 /// \param Ctx The context that the reference occurred in
304 /// \param ReferringDecl The exact declaration that was referenced.
305 /// \param OffendingDecl A related decl to \c ReferringDecl that has an
306 /// availability attribute corresponding to \c K attached to it. Note that this
307 /// may not be the same as ReferringDecl, i.e. if an EnumDecl is annotated and
308 /// we refer to a member EnumConstantDecl, ReferringDecl is the EnumConstantDecl
309 /// and OffendingDecl is the EnumDecl.
310 static void DoEmitAvailabilityWarning(Sema &S, AvailabilityResult K,
311                                       Decl *Ctx, const NamedDecl *ReferringDecl,
312                                       const NamedDecl *OffendingDecl,
313                                       StringRef Message,
314                                       ArrayRef<SourceLocation> Locs,
315                                       const ObjCInterfaceDecl *UnknownObjCClass,
316                                       const ObjCPropertyDecl *ObjCProperty,
317                                       bool ObjCPropertyAccess) {
318   // Diagnostics for deprecated or unavailable.
319   unsigned diag, diag_message, diag_fwdclass_message;
320   unsigned diag_available_here = diag::note_availability_specified_here;
321   SourceLocation NoteLocation = OffendingDecl->getLocation();
322 
323   // Matches 'diag::note_property_attribute' options.
324   unsigned property_note_select;
325 
326   // Matches diag::note_availability_specified_here.
327   unsigned available_here_select_kind;
328 
329   VersionTuple DeclVersion;
330   if (const AvailabilityAttr *AA = getAttrForPlatform(S.Context, OffendingDecl))
331     DeclVersion = AA->getIntroduced();
332 
333   if (!ShouldDiagnoseAvailabilityInContext(S, K, DeclVersion, Ctx,
334                                            OffendingDecl))
335     return;
336 
337   SourceLocation Loc = Locs.front();
338 
339   // The declaration can have multiple availability attributes, we are looking
340   // at one of them.
341   const AvailabilityAttr *A = getAttrForPlatform(S.Context, OffendingDecl);
342   if (A && A->isInherited()) {
343     for (const Decl *Redecl = OffendingDecl->getMostRecentDecl(); Redecl;
344          Redecl = Redecl->getPreviousDecl()) {
345       const AvailabilityAttr *AForRedecl =
346           getAttrForPlatform(S.Context, Redecl);
347       if (AForRedecl && !AForRedecl->isInherited()) {
348         // If D is a declaration with inherited attributes, the note should
349         // point to the declaration with actual attributes.
350         NoteLocation = Redecl->getLocation();
351         break;
352       }
353     }
354   }
355 
356   switch (K) {
357   case AR_NotYetIntroduced: {
358     // We would like to emit the diagnostic even if -Wunguarded-availability is
359     // not specified for deployment targets >= to iOS 11 or equivalent or
360     // for declarations that were introduced in iOS 11 (macOS 10.13, ...) or
361     // later.
362     const AvailabilityAttr *AA =
363         getAttrForPlatform(S.getASTContext(), OffendingDecl);
364     VersionTuple Introduced = AA->getIntroduced();
365 
366     bool UseNewWarning = shouldDiagnoseAvailabilityByDefault(
367         S.Context, S.Context.getTargetInfo().getPlatformMinVersion(),
368         Introduced);
369     unsigned Warning = UseNewWarning ? diag::warn_unguarded_availability_new
370                                      : diag::warn_unguarded_availability;
371 
372     std::string PlatformName(AvailabilityAttr::getPrettyPlatformName(
373         S.getASTContext().getTargetInfo().getPlatformName()));
374 
375     S.Diag(Loc, Warning) << OffendingDecl << PlatformName
376                          << Introduced.getAsString();
377 
378     S.Diag(OffendingDecl->getLocation(),
379            diag::note_partial_availability_specified_here)
380         << OffendingDecl << PlatformName << Introduced.getAsString()
381         << S.Context.getTargetInfo().getPlatformMinVersion().getAsString();
382 
383     if (const auto *Enclosing = findEnclosingDeclToAnnotate(Ctx)) {
384       if (const auto *TD = dyn_cast<TagDecl>(Enclosing))
385         if (TD->getDeclName().isEmpty()) {
386           S.Diag(TD->getLocation(),
387                  diag::note_decl_unguarded_availability_silence)
388               << /*Anonymous*/ 1 << TD->getKindName();
389           return;
390         }
391       auto FixitNoteDiag =
392           S.Diag(Enclosing->getLocation(),
393                  diag::note_decl_unguarded_availability_silence)
394           << /*Named*/ 0 << Enclosing;
395       // Don't offer a fixit for declarations with availability attributes.
396       if (Enclosing->hasAttr<AvailabilityAttr>())
397         return;
398       if (!S.getPreprocessor().isMacroDefined("API_AVAILABLE"))
399         return;
400       Optional<AttributeInsertion> Insertion = createAttributeInsertion(
401           Enclosing, S.getSourceManager(), S.getLangOpts());
402       if (!Insertion)
403         return;
404       std::string PlatformName =
405           AvailabilityAttr::getPlatformNameSourceSpelling(
406               S.getASTContext().getTargetInfo().getPlatformName())
407               .lower();
408       std::string Introduced =
409           OffendingDecl->getVersionIntroduced().getAsString();
410       FixitNoteDiag << FixItHint::CreateInsertion(
411           Insertion->Loc,
412           (llvm::Twine(Insertion->Prefix) + "API_AVAILABLE(" + PlatformName +
413            "(" + Introduced + "))" + Insertion->Suffix)
414               .str());
415     }
416     return;
417   }
418   case AR_Deprecated:
419     diag = !ObjCPropertyAccess ? diag::warn_deprecated
420                                : diag::warn_property_method_deprecated;
421     diag_message = diag::warn_deprecated_message;
422     diag_fwdclass_message = diag::warn_deprecated_fwdclass_message;
423     property_note_select = /* deprecated */ 0;
424     available_here_select_kind = /* deprecated */ 2;
425     if (const auto *AL = OffendingDecl->getAttr<DeprecatedAttr>())
426       NoteLocation = AL->getLocation();
427     break;
428 
429   case AR_Unavailable:
430     diag = !ObjCPropertyAccess ? diag::err_unavailable
431                                : diag::err_property_method_unavailable;
432     diag_message = diag::err_unavailable_message;
433     diag_fwdclass_message = diag::warn_unavailable_fwdclass_message;
434     property_note_select = /* unavailable */ 1;
435     available_here_select_kind = /* unavailable */ 0;
436 
437     if (auto AL = OffendingDecl->getAttr<UnavailableAttr>()) {
438       if (AL->isImplicit() && AL->getImplicitReason()) {
439         // Most of these failures are due to extra restrictions in ARC;
440         // reflect that in the primary diagnostic when applicable.
441         auto flagARCError = [&] {
442           if (S.getLangOpts().ObjCAutoRefCount &&
443               S.getSourceManager().isInSystemHeader(
444                   OffendingDecl->getLocation()))
445             diag = diag::err_unavailable_in_arc;
446         };
447 
448         switch (AL->getImplicitReason()) {
449         case UnavailableAttr::IR_None: break;
450 
451         case UnavailableAttr::IR_ARCForbiddenType:
452           flagARCError();
453           diag_available_here = diag::note_arc_forbidden_type;
454           break;
455 
456         case UnavailableAttr::IR_ForbiddenWeak:
457           if (S.getLangOpts().ObjCWeakRuntime)
458             diag_available_here = diag::note_arc_weak_disabled;
459           else
460             diag_available_here = diag::note_arc_weak_no_runtime;
461           break;
462 
463         case UnavailableAttr::IR_ARCForbiddenConversion:
464           flagARCError();
465           diag_available_here = diag::note_performs_forbidden_arc_conversion;
466           break;
467 
468         case UnavailableAttr::IR_ARCInitReturnsUnrelated:
469           flagARCError();
470           diag_available_here = diag::note_arc_init_returns_unrelated;
471           break;
472 
473         case UnavailableAttr::IR_ARCFieldWithOwnership:
474           flagARCError();
475           diag_available_here = diag::note_arc_field_with_ownership;
476           break;
477         }
478       }
479     }
480     break;
481 
482   case AR_Available:
483     llvm_unreachable("Warning for availability of available declaration?");
484   }
485 
486   SmallVector<FixItHint, 12> FixIts;
487   if (K == AR_Deprecated) {
488     StringRef Replacement;
489     if (auto AL = OffendingDecl->getAttr<DeprecatedAttr>())
490       Replacement = AL->getReplacement();
491     if (auto AL = getAttrForPlatform(S.Context, OffendingDecl))
492       Replacement = AL->getReplacement();
493 
494     CharSourceRange UseRange;
495     if (!Replacement.empty())
496       UseRange =
497           CharSourceRange::getCharRange(Loc, S.getLocForEndOfToken(Loc));
498     if (UseRange.isValid()) {
499       if (const auto *MethodDecl = dyn_cast<ObjCMethodDecl>(ReferringDecl)) {
500         Selector Sel = MethodDecl->getSelector();
501         SmallVector<StringRef, 12> SelectorSlotNames;
502         Optional<unsigned> NumParams = tryParseObjCMethodName(
503             Replacement, SelectorSlotNames, S.getLangOpts());
504         if (NumParams && NumParams.getValue() == Sel.getNumArgs()) {
505           assert(SelectorSlotNames.size() == Locs.size());
506           for (unsigned I = 0; I < Locs.size(); ++I) {
507             if (!Sel.getNameForSlot(I).empty()) {
508               CharSourceRange NameRange = CharSourceRange::getCharRange(
509                   Locs[I], S.getLocForEndOfToken(Locs[I]));
510               FixIts.push_back(FixItHint::CreateReplacement(
511                   NameRange, SelectorSlotNames[I]));
512             } else
513               FixIts.push_back(
514                   FixItHint::CreateInsertion(Locs[I], SelectorSlotNames[I]));
515           }
516         } else
517           FixIts.push_back(FixItHint::CreateReplacement(UseRange, Replacement));
518       } else
519         FixIts.push_back(FixItHint::CreateReplacement(UseRange, Replacement));
520     }
521   }
522 
523   if (!Message.empty()) {
524     S.Diag(Loc, diag_message) << ReferringDecl << Message << FixIts;
525     if (ObjCProperty)
526       S.Diag(ObjCProperty->getLocation(), diag::note_property_attribute)
527           << ObjCProperty->getDeclName() << property_note_select;
528   } else if (!UnknownObjCClass) {
529     S.Diag(Loc, diag) << ReferringDecl << FixIts;
530     if (ObjCProperty)
531       S.Diag(ObjCProperty->getLocation(), diag::note_property_attribute)
532           << ObjCProperty->getDeclName() << property_note_select;
533   } else {
534     S.Diag(Loc, diag_fwdclass_message) << ReferringDecl << FixIts;
535     S.Diag(UnknownObjCClass->getLocation(), diag::note_forward_class);
536   }
537 
538   S.Diag(NoteLocation, diag_available_here)
539     << OffendingDecl << available_here_select_kind;
540 }
541 
542 void Sema::handleDelayedAvailabilityCheck(DelayedDiagnostic &DD, Decl *Ctx) {
543   assert(DD.Kind == DelayedDiagnostic::Availability &&
544          "Expected an availability diagnostic here");
545 
546   DD.Triggered = true;
547   DoEmitAvailabilityWarning(
548       *this, DD.getAvailabilityResult(), Ctx, DD.getAvailabilityReferringDecl(),
549       DD.getAvailabilityOffendingDecl(), DD.getAvailabilityMessage(),
550       DD.getAvailabilitySelectorLocs(), DD.getUnknownObjCClass(),
551       DD.getObjCProperty(), false);
552 }
553 
554 static void EmitAvailabilityWarning(Sema &S, AvailabilityResult AR,
555                                     const NamedDecl *ReferringDecl,
556                                     const NamedDecl *OffendingDecl,
557                                     StringRef Message,
558                                     ArrayRef<SourceLocation> Locs,
559                                     const ObjCInterfaceDecl *UnknownObjCClass,
560                                     const ObjCPropertyDecl *ObjCProperty,
561                                     bool ObjCPropertyAccess) {
562   // Delay if we're currently parsing a declaration.
563   if (S.DelayedDiagnostics.shouldDelayDiagnostics()) {
564     S.DelayedDiagnostics.add(
565         DelayedDiagnostic::makeAvailability(
566             AR, Locs, ReferringDecl, OffendingDecl, UnknownObjCClass,
567             ObjCProperty, Message, ObjCPropertyAccess));
568     return;
569   }
570 
571   Decl *Ctx = cast<Decl>(S.getCurLexicalContext());
572   DoEmitAvailabilityWarning(S, AR, Ctx, ReferringDecl, OffendingDecl,
573                             Message, Locs, UnknownObjCClass, ObjCProperty,
574                             ObjCPropertyAccess);
575 }
576 
577 namespace {
578 
579 /// Returns true if the given statement can be a body-like child of \p Parent.
580 bool isBodyLikeChildStmt(const Stmt *S, const Stmt *Parent) {
581   switch (Parent->getStmtClass()) {
582   case Stmt::IfStmtClass:
583     return cast<IfStmt>(Parent)->getThen() == S ||
584            cast<IfStmt>(Parent)->getElse() == S;
585   case Stmt::WhileStmtClass:
586     return cast<WhileStmt>(Parent)->getBody() == S;
587   case Stmt::DoStmtClass:
588     return cast<DoStmt>(Parent)->getBody() == S;
589   case Stmt::ForStmtClass:
590     return cast<ForStmt>(Parent)->getBody() == S;
591   case Stmt::CXXForRangeStmtClass:
592     return cast<CXXForRangeStmt>(Parent)->getBody() == S;
593   case Stmt::ObjCForCollectionStmtClass:
594     return cast<ObjCForCollectionStmt>(Parent)->getBody() == S;
595   case Stmt::CaseStmtClass:
596   case Stmt::DefaultStmtClass:
597     return cast<SwitchCase>(Parent)->getSubStmt() == S;
598   default:
599     return false;
600   }
601 }
602 
603 class StmtUSEFinder : public RecursiveASTVisitor<StmtUSEFinder> {
604   const Stmt *Target;
605 
606 public:
607   bool VisitStmt(Stmt *S) { return S != Target; }
608 
609   /// Returns true if the given statement is present in the given declaration.
610   static bool isContained(const Stmt *Target, const Decl *D) {
611     StmtUSEFinder Visitor;
612     Visitor.Target = Target;
613     return !Visitor.TraverseDecl(const_cast<Decl *>(D));
614   }
615 };
616 
617 /// Traverses the AST and finds the last statement that used a given
618 /// declaration.
619 class LastDeclUSEFinder : public RecursiveASTVisitor<LastDeclUSEFinder> {
620   const Decl *D;
621 
622 public:
623   bool VisitDeclRefExpr(DeclRefExpr *DRE) {
624     if (DRE->getDecl() == D)
625       return false;
626     return true;
627   }
628 
629   static const Stmt *findLastStmtThatUsesDecl(const Decl *D,
630                                               const CompoundStmt *Scope) {
631     LastDeclUSEFinder Visitor;
632     Visitor.D = D;
633     for (auto I = Scope->body_rbegin(), E = Scope->body_rend(); I != E; ++I) {
634       const Stmt *S = *I;
635       if (!Visitor.TraverseStmt(const_cast<Stmt *>(S)))
636         return S;
637     }
638     return nullptr;
639   }
640 };
641 
642 /// This class implements -Wunguarded-availability.
643 ///
644 /// This is done with a traversal of the AST of a function that makes reference
645 /// to a partially available declaration. Whenever we encounter an \c if of the
646 /// form: \c if(@available(...)), we use the version from the condition to visit
647 /// the then statement.
648 class DiagnoseUnguardedAvailability
649     : public RecursiveASTVisitor<DiagnoseUnguardedAvailability> {
650   typedef RecursiveASTVisitor<DiagnoseUnguardedAvailability> Base;
651 
652   Sema &SemaRef;
653   Decl *Ctx;
654 
655   /// Stack of potentially nested 'if (@available(...))'s.
656   SmallVector<VersionTuple, 8> AvailabilityStack;
657   SmallVector<const Stmt *, 16> StmtStack;
658 
659   void DiagnoseDeclAvailability(NamedDecl *D, SourceRange Range,
660                                 ObjCInterfaceDecl *ClassReceiver = nullptr);
661 
662 public:
663   DiagnoseUnguardedAvailability(Sema &SemaRef, Decl *Ctx)
664       : SemaRef(SemaRef), Ctx(Ctx) {
665     AvailabilityStack.push_back(
666         SemaRef.Context.getTargetInfo().getPlatformMinVersion());
667   }
668 
669   bool TraverseStmt(Stmt *S) {
670     if (!S)
671       return true;
672     StmtStack.push_back(S);
673     bool Result = Base::TraverseStmt(S);
674     StmtStack.pop_back();
675     return Result;
676   }
677 
678   void IssueDiagnostics(Stmt *S) { TraverseStmt(S); }
679 
680   bool TraverseIfStmt(IfStmt *If);
681 
682   // for 'case X:' statements, don't bother looking at the 'X'; it can't lead
683   // to any useful diagnostics.
684   bool TraverseCaseStmt(CaseStmt *CS) { return TraverseStmt(CS->getSubStmt()); }
685 
686   bool VisitObjCPropertyRefExpr(ObjCPropertyRefExpr *PRE) { return true; }
687 
688   bool VisitObjCMessageExpr(ObjCMessageExpr *Msg) {
689     if (ObjCMethodDecl *D = Msg->getMethodDecl()) {
690       ObjCInterfaceDecl *ID = nullptr;
691       QualType ReceiverTy = Msg->getClassReceiver();
692       if (!ReceiverTy.isNull() && ReceiverTy->getAsObjCInterfaceType())
693         ID = ReceiverTy->getAsObjCInterfaceType()->getInterface();
694 
695       DiagnoseDeclAvailability(
696           D, SourceRange(Msg->getSelectorStartLoc(), Msg->getEndLoc()), ID);
697     }
698     return true;
699   }
700 
701   bool VisitDeclRefExpr(DeclRefExpr *DRE) {
702     DiagnoseDeclAvailability(DRE->getDecl(),
703                              SourceRange(DRE->getBeginLoc(), DRE->getEndLoc()));
704     return true;
705   }
706 
707   bool VisitMemberExpr(MemberExpr *ME) {
708     DiagnoseDeclAvailability(ME->getMemberDecl(),
709                              SourceRange(ME->getBeginLoc(), ME->getEndLoc()));
710     return true;
711   }
712 
713   bool VisitObjCAvailabilityCheckExpr(ObjCAvailabilityCheckExpr *E) {
714     SemaRef.Diag(E->getBeginLoc(), diag::warn_at_available_unchecked_use)
715         << (!SemaRef.getLangOpts().ObjC);
716     return true;
717   }
718 
719   bool VisitTypeLoc(TypeLoc Ty);
720 };
721 
722 void DiagnoseUnguardedAvailability::DiagnoseDeclAvailability(
723     NamedDecl *D, SourceRange Range, ObjCInterfaceDecl *ReceiverClass) {
724   AvailabilityResult Result;
725   const NamedDecl *OffendingDecl;
726   std::tie(Result, OffendingDecl) =
727       ShouldDiagnoseAvailabilityOfDecl(SemaRef, D, nullptr, ReceiverClass);
728   if (Result != AR_Available) {
729     // All other diagnostic kinds have already been handled in
730     // DiagnoseAvailabilityOfDecl.
731     if (Result != AR_NotYetIntroduced)
732       return;
733 
734     const AvailabilityAttr *AA =
735       getAttrForPlatform(SemaRef.getASTContext(), OffendingDecl);
736     VersionTuple Introduced = AA->getIntroduced();
737 
738     if (AvailabilityStack.back() >= Introduced)
739       return;
740 
741     // If the context of this function is less available than D, we should not
742     // emit a diagnostic.
743     if (!ShouldDiagnoseAvailabilityInContext(SemaRef, Result, Introduced, Ctx,
744                                              OffendingDecl))
745       return;
746 
747     // We would like to emit the diagnostic even if -Wunguarded-availability is
748     // not specified for deployment targets >= to iOS 11 or equivalent or
749     // for declarations that were introduced in iOS 11 (macOS 10.13, ...) or
750     // later.
751     unsigned DiagKind =
752         shouldDiagnoseAvailabilityByDefault(
753             SemaRef.Context,
754             SemaRef.Context.getTargetInfo().getPlatformMinVersion(), Introduced)
755             ? diag::warn_unguarded_availability_new
756             : diag::warn_unguarded_availability;
757 
758     std::string PlatformName(AvailabilityAttr::getPrettyPlatformName(
759         SemaRef.getASTContext().getTargetInfo().getPlatformName()));
760 
761     SemaRef.Diag(Range.getBegin(), DiagKind)
762         << Range << D << PlatformName << Introduced.getAsString();
763 
764     SemaRef.Diag(OffendingDecl->getLocation(),
765                  diag::note_partial_availability_specified_here)
766         << OffendingDecl << PlatformName << Introduced.getAsString()
767         << SemaRef.Context.getTargetInfo()
768                .getPlatformMinVersion()
769                .getAsString();
770 
771     auto FixitDiag =
772         SemaRef.Diag(Range.getBegin(), diag::note_unguarded_available_silence)
773         << Range << D
774         << (SemaRef.getLangOpts().ObjC ? /*@available*/ 0
775                                        : /*__builtin_available*/ 1);
776 
777     // Find the statement which should be enclosed in the if @available check.
778     if (StmtStack.empty())
779       return;
780     const Stmt *StmtOfUse = StmtStack.back();
781     const CompoundStmt *Scope = nullptr;
782     for (const Stmt *S : llvm::reverse(StmtStack)) {
783       if (const auto *CS = dyn_cast<CompoundStmt>(S)) {
784         Scope = CS;
785         break;
786       }
787       if (isBodyLikeChildStmt(StmtOfUse, S)) {
788         // The declaration won't be seen outside of the statement, so we don't
789         // have to wrap the uses of any declared variables in if (@available).
790         // Therefore we can avoid setting Scope here.
791         break;
792       }
793       StmtOfUse = S;
794     }
795     const Stmt *LastStmtOfUse = nullptr;
796     if (isa<DeclStmt>(StmtOfUse) && Scope) {
797       for (const Decl *D : cast<DeclStmt>(StmtOfUse)->decls()) {
798         if (StmtUSEFinder::isContained(StmtStack.back(), D)) {
799           LastStmtOfUse = LastDeclUSEFinder::findLastStmtThatUsesDecl(D, Scope);
800           break;
801         }
802       }
803     }
804 
805     const SourceManager &SM = SemaRef.getSourceManager();
806     SourceLocation IfInsertionLoc =
807         SM.getExpansionLoc(StmtOfUse->getBeginLoc());
808     SourceLocation StmtEndLoc =
809         SM.getExpansionRange(
810               (LastStmtOfUse ? LastStmtOfUse : StmtOfUse)->getEndLoc())
811             .getEnd();
812     if (SM.getFileID(IfInsertionLoc) != SM.getFileID(StmtEndLoc))
813       return;
814 
815     StringRef Indentation = Lexer::getIndentationForLine(IfInsertionLoc, SM);
816     const char *ExtraIndentation = "    ";
817     std::string FixItString;
818     llvm::raw_string_ostream FixItOS(FixItString);
819     FixItOS << "if (" << (SemaRef.getLangOpts().ObjC ? "@available"
820                                                      : "__builtin_available")
821             << "("
822             << AvailabilityAttr::getPlatformNameSourceSpelling(
823                    SemaRef.getASTContext().getTargetInfo().getPlatformName())
824             << " " << Introduced.getAsString() << ", *)) {\n"
825             << Indentation << ExtraIndentation;
826     FixitDiag << FixItHint::CreateInsertion(IfInsertionLoc, FixItOS.str());
827     SourceLocation ElseInsertionLoc = Lexer::findLocationAfterToken(
828         StmtEndLoc, tok::semi, SM, SemaRef.getLangOpts(),
829         /*SkipTrailingWhitespaceAndNewLine=*/false);
830     if (ElseInsertionLoc.isInvalid())
831       ElseInsertionLoc =
832           Lexer::getLocForEndOfToken(StmtEndLoc, 0, SM, SemaRef.getLangOpts());
833     FixItOS.str().clear();
834     FixItOS << "\n"
835             << Indentation << "} else {\n"
836             << Indentation << ExtraIndentation
837             << "// Fallback on earlier versions\n"
838             << Indentation << "}";
839     FixitDiag << FixItHint::CreateInsertion(ElseInsertionLoc, FixItOS.str());
840   }
841 }
842 
843 bool DiagnoseUnguardedAvailability::VisitTypeLoc(TypeLoc Ty) {
844   const Type *TyPtr = Ty.getTypePtr();
845   SourceRange Range{Ty.getBeginLoc(), Ty.getEndLoc()};
846 
847   if (Range.isInvalid())
848     return true;
849 
850   if (const auto *TT = dyn_cast<TagType>(TyPtr)) {
851     TagDecl *TD = TT->getDecl();
852     DiagnoseDeclAvailability(TD, Range);
853 
854   } else if (const auto *TD = dyn_cast<TypedefType>(TyPtr)) {
855     TypedefNameDecl *D = TD->getDecl();
856     DiagnoseDeclAvailability(D, Range);
857 
858   } else if (const auto *ObjCO = dyn_cast<ObjCObjectType>(TyPtr)) {
859     if (NamedDecl *D = ObjCO->getInterface())
860       DiagnoseDeclAvailability(D, Range);
861   }
862 
863   return true;
864 }
865 
866 bool DiagnoseUnguardedAvailability::TraverseIfStmt(IfStmt *If) {
867   VersionTuple CondVersion;
868   if (auto *E = dyn_cast<ObjCAvailabilityCheckExpr>(If->getCond())) {
869     CondVersion = E->getVersion();
870 
871     // If we're using the '*' case here or if this check is redundant, then we
872     // use the enclosing version to check both branches.
873     if (CondVersion.empty() || CondVersion <= AvailabilityStack.back())
874       return TraverseStmt(If->getThen()) && TraverseStmt(If->getElse());
875   } else {
876     // This isn't an availability checking 'if', we can just continue.
877     return Base::TraverseIfStmt(If);
878   }
879 
880   AvailabilityStack.push_back(CondVersion);
881   bool ShouldContinue = TraverseStmt(If->getThen());
882   AvailabilityStack.pop_back();
883 
884   return ShouldContinue && TraverseStmt(If->getElse());
885 }
886 
887 } // end anonymous namespace
888 
889 void Sema::DiagnoseUnguardedAvailabilityViolations(Decl *D) {
890   Stmt *Body = nullptr;
891 
892   if (auto *FD = D->getAsFunction()) {
893     // FIXME: We only examine the pattern decl for availability violations now,
894     // but we should also examine instantiated templates.
895     if (FD->isTemplateInstantiation())
896       return;
897 
898     Body = FD->getBody();
899   } else if (auto *MD = dyn_cast<ObjCMethodDecl>(D))
900     Body = MD->getBody();
901   else if (auto *BD = dyn_cast<BlockDecl>(D))
902     Body = BD->getBody();
903 
904   assert(Body && "Need a body here!");
905 
906   DiagnoseUnguardedAvailability(*this, D).IssueDiagnostics(Body);
907 }
908 
909 FunctionScopeInfo *Sema::getCurFunctionAvailabilityContext() {
910   if (FunctionScopes.empty())
911     return nullptr;
912 
913   // Conservatively search the entire current function scope context for
914   // availability violations. This ensures we always correctly analyze nested
915   // classes, blocks, lambdas, etc. that may or may not be inside if(@available)
916   // checks themselves.
917   return FunctionScopes.front();
918 }
919 
920 void Sema::DiagnoseAvailabilityOfDecl(NamedDecl *D,
921                                       ArrayRef<SourceLocation> Locs,
922                                       const ObjCInterfaceDecl *UnknownObjCClass,
923                                       bool ObjCPropertyAccess,
924                                       bool AvoidPartialAvailabilityChecks,
925                                       ObjCInterfaceDecl *ClassReceiver) {
926   std::string Message;
927   AvailabilityResult Result;
928   const NamedDecl* OffendingDecl;
929   // See if this declaration is unavailable, deprecated, or partial.
930   std::tie(Result, OffendingDecl) =
931       ShouldDiagnoseAvailabilityOfDecl(*this, D, &Message, ClassReceiver);
932   if (Result == AR_Available)
933     return;
934 
935   if (Result == AR_NotYetIntroduced) {
936     if (AvoidPartialAvailabilityChecks)
937       return;
938 
939     // We need to know the @available context in the current function to
940     // diagnose this use, let DiagnoseUnguardedAvailabilityViolations do that
941     // when we're done parsing the current function.
942     if (FunctionScopeInfo *Context = getCurFunctionAvailabilityContext()) {
943       Context->HasPotentialAvailabilityViolations = true;
944       return;
945     }
946   }
947 
948   const ObjCPropertyDecl *ObjCPDecl = nullptr;
949   if (const auto *MD = dyn_cast<ObjCMethodDecl>(D)) {
950     if (const ObjCPropertyDecl *PD = MD->findPropertyDecl()) {
951       AvailabilityResult PDeclResult = PD->getAvailability(nullptr);
952       if (PDeclResult == Result)
953         ObjCPDecl = PD;
954     }
955   }
956 
957   EmitAvailabilityWarning(*this, Result, D, OffendingDecl, Message, Locs,
958                           UnknownObjCClass, ObjCPDecl, ObjCPropertyAccess);
959 }
960