1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295
|
//===--- AvailabilityConstraint.cpp - Swift Availability Constraints ------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
#include "swift/AST/AvailabilityConstraint.h"
#include "swift/AST/ASTContext.h"
#include "swift/AST/AvailabilityContext.h"
#include "swift/AST/AvailabilityInference.h"
#include "swift/AST/Decl.h"
using namespace swift;
AvailabilityDomainAndRange
AvailabilityConstraint::getDomainAndRange(const ASTContext &ctx) const {
switch (getReason()) {
case Reason::UnconditionallyUnavailable:
// Technically, unconditional unavailability doesn't have an associated
// range. However, if you view it as a special case of obsoletion, then an
// unconditionally unavailable declaration is "always obsoleted."
return AvailabilityDomainAndRange(getDomain().getRemappedDomain(ctx),
AvailabilityRange::alwaysAvailable());
case Reason::Obsoleted:
return getAttr().getObsoletedDomainAndRange(ctx).value();
case Reason::UnavailableForDeployment:
case Reason::PotentiallyUnavailable:
return getAttr().getIntroducedDomainAndRange(ctx).value();
}
}
bool AvailabilityConstraint::isActiveForRuntimeQueries(
const ASTContext &ctx) const {
if (getAttr().getPlatform() == PlatformKind::none)
return true;
return swift::isPlatformActive(getAttr().getPlatform(), ctx.LangOpts,
/*forTargetVariant=*/false,
/*forRuntimeQuery=*/true);
}
static bool constraintIsStronger(const AvailabilityConstraint &lhs,
const AvailabilityConstraint &rhs) {
DEBUG_ASSERT(lhs.getDomain() == rhs.getDomain());
// If the constraints have matching domains but different reasons, the
// constraint with the lowest reason is "strongest".
if (lhs.getReason() != rhs.getReason())
return lhs.getReason() < rhs.getReason();
switch (lhs.getReason()) {
case AvailabilityConstraint::Reason::UnconditionallyUnavailable:
// Just keep the first.
return false;
case AvailabilityConstraint::Reason::Obsoleted:
// Pick the larger obsoleted range.
return *lhs.getAttr().getObsoleted() < *rhs.getAttr().getObsoleted();
case AvailabilityConstraint::Reason::UnavailableForDeployment:
case AvailabilityConstraint::Reason::PotentiallyUnavailable:
// Pick the smaller introduced range.
return *lhs.getAttr().getIntroduced() > *rhs.getAttr().getIntroduced();
}
}
void addConstraint(llvm::SmallVector<AvailabilityConstraint, 4> &constraints,
const AvailabilityConstraint &constraint,
const ASTContext &ctx) {
auto iter = llvm::find_if(
constraints, [&constraint](AvailabilityConstraint &existing) {
return constraint.getDomain() == existing.getDomain();
});
// There's no existing constraint for the same domain so just add it.
if (iter == constraints.end()) {
constraints.emplace_back(constraint);
return;
}
if (constraintIsStronger(constraint, *iter)) {
constraints.erase(iter);
constraints.emplace_back(constraint);
}
}
std::optional<AvailabilityConstraint>
DeclAvailabilityConstraints::getPrimaryConstraint() const {
std::optional<AvailabilityConstraint> result;
auto isStrongerConstraint = [](const AvailabilityConstraint &lhs,
const AvailabilityConstraint &rhs) {
// Constraint reasons are defined in descending order of strength.
if (lhs.getReason() != rhs.getReason())
return lhs.getReason() < rhs.getReason();
// Pick the constraint from the broader domain.
if (lhs.getDomain() != rhs.getDomain())
return rhs.getDomain().contains(lhs.getDomain());
return false;
};
// Pick the strongest constraint.
for (auto const &constraint : constraints) {
if (!result || isStrongerConstraint(constraint, *result))
result.emplace(constraint);
}
return result;
}
static bool canIgnoreConstraintInUnavailableContexts(
const Decl *decl, const AvailabilityConstraint &constraint) {
auto domain = constraint.getDomain();
switch (constraint.getReason()) {
case AvailabilityConstraint::Reason::UnconditionallyUnavailable:
// Always reject uses of universally unavailable declarations, regardless
// of context, since there are no possible compilation configurations in
// which they are available. However, make an exception for types and
// conformances, which can sometimes be awkward to avoid references to.
if (!isa<TypeDecl>(decl) && !isa<ExtensionDecl>(decl)) {
if (domain.isUniversal() || domain.isSwiftLanguage())
return false;
}
return true;
case AvailabilityConstraint::Reason::PotentiallyUnavailable:
switch (domain.getKind()) {
case AvailabilityDomain::Kind::Universal:
case AvailabilityDomain::Kind::SwiftLanguage:
case AvailabilityDomain::Kind::PackageDescription:
case AvailabilityDomain::Kind::Embedded:
case AvailabilityDomain::Kind::Custom:
return false;
case AvailabilityDomain::Kind::Platform:
// Platform availability only applies to the target triple that the
// binary is being compiled for. Since the same declaration can be
// potentially unavailable from a given context when compiling for one
// platform, but available from that context when compiling for a
// different platform, it is overly strict to enforce potential platform
// unavailability constraints in contexts that are unavailable to that
// platform.
return true;
}
return constraint.getDomain().isPlatform();
case AvailabilityConstraint::Reason::Obsoleted:
case AvailabilityConstraint::Reason::UnavailableForDeployment:
return false;
}
}
static bool
shouldIgnoreConstraintInContext(const Decl *decl,
const AvailabilityConstraint &constraint,
const AvailabilityContext &context) {
if (!context.isUnavailable())
return false;
if (!canIgnoreConstraintInUnavailableContexts(decl, constraint))
return false;
return context.containsUnavailableDomain(constraint.getDomain());
}
/// Returns the `AvailabilityConstraint` that describes how \p attr restricts
/// use of \p decl in \p context or `std::nullopt` if there is no restriction.
static std::optional<AvailabilityConstraint>
getAvailabilityConstraintForAttr(const Decl *decl,
const SemanticAvailableAttr &attr,
const AvailabilityContext &context) {
// Is the decl unconditionally unavailable?
if (attr.isUnconditionallyUnavailable())
return AvailabilityConstraint::unconditionallyUnavailable(attr);
auto &ctx = decl->getASTContext();
auto domain = attr.getDomain();
auto deploymentRange = domain.getDeploymentRange(ctx);
// Is the decl obsoleted in the deployment context?
if (auto obsoletedRange = attr.getObsoletedRange(ctx)) {
if (deploymentRange && deploymentRange->isContainedIn(*obsoletedRange))
return AvailabilityConstraint::obsoleted(attr);
}
// Is the decl not yet introduced in the local context?
if (auto introducedRange = attr.getIntroducedRange(ctx)) {
if (domain.supportsContextRefinement()) {
auto availableRange = context.getAvailabilityRange(domain, ctx);
if (!availableRange || !availableRange->isContainedIn(*introducedRange))
return AvailabilityConstraint::potentiallyUnavailable(attr);
return std::nullopt;
}
// Is the decl not yet introduced in the deployment context?
if (deploymentRange && !deploymentRange->isContainedIn(*introducedRange))
return AvailabilityConstraint::unavailableForDeployment(attr);
}
// FIXME: [availability] Model deprecation as an availability constraint.
return std::nullopt;
}
/// Returns the most specific platform domain from the availability attributes
/// attached to \p decl or `std::nullopt` if there are none. Platform specific
/// `@available` attributes for other platforms should be ignored. For example,
/// if a declaration has attributes for both iOS and macCatalyst, only the
/// macCatalyst attributes take effect when compiling for a macCatalyst target.
static std::optional<AvailabilityDomain>
activePlatformDomainForDecl(const Decl *decl) {
std::optional<AvailabilityDomain> activeDomain;
for (auto attr :
decl->getSemanticAvailableAttrs(/*includingInactive=*/false)) {
auto domain = attr.getDomain();
if (!domain.isPlatform())
continue;
if (activeDomain && domain.contains(*activeDomain))
continue;
activeDomain.emplace(domain);
}
return activeDomain;
}
static void getAvailabilityConstraintsForDecl(
llvm::SmallVector<AvailabilityConstraint, 4> &constraints, const Decl *decl,
const AvailabilityContext &context, AvailabilityConstraintFlags flags) {
auto &ctx = decl->getASTContext();
auto activePlatformDomain = activePlatformDomainForDecl(decl);
bool includeAllDomains =
flags.contains(AvailabilityConstraintFlag::IncludeAllDomains);
for (auto attr : decl->getSemanticAvailableAttrs(includeAllDomains)) {
auto domain = attr.getDomain();
if (!includeAllDomains && domain.isPlatform() && activePlatformDomain &&
!activePlatformDomain->contains(domain))
continue;
if (auto constraint = getAvailabilityConstraintForAttr(decl, attr, context))
addConstraint(constraints, *constraint, ctx);
}
// After resolving constraints, remove any constraints that indicate the
// declaration is unconditionally unavailable in a domain for which
// the context is already unavailable.
llvm::erase_if(constraints, [&](const AvailabilityConstraint &constraint) {
return shouldIgnoreConstraintInContext(decl, constraint, context);
});
}
DeclAvailabilityConstraints
swift::getAvailabilityConstraintsForDecl(const Decl *decl,
const AvailabilityContext &context,
AvailabilityConstraintFlags flags) {
llvm::SmallVector<AvailabilityConstraint, 4> constraints;
// Generic parameters are always available.
if (isa<GenericTypeParamDecl>(decl))
return DeclAvailabilityConstraints();
decl = decl->getAbstractSyntaxDeclForAttributes();
getAvailabilityConstraintsForDecl(constraints, decl, context, flags);
if (flags.contains(AvailabilityConstraintFlag::SkipEnclosingExtension))
return constraints;
// If decl is an extension member, query the attributes of the extension, too.
//
// Skip decls imported from Clang, though, as they could be associated to the
// wrong extension and inherit unavailability incorrectly. ClangImporter
// associates Objective-C protocol members to the first category where the
// protocol is directly or indirectly adopted, no matter its availability
// and the availability of other categories. rdar://problem/53956555
if (decl->getClangNode())
return constraints;
auto parent = AvailabilityInference::parentDeclForInferredAvailability(decl);
if (auto extension = dyn_cast_or_null<ExtensionDecl>(parent))
getAvailabilityConstraintsForDecl(constraints, extension, context, flags);
return constraints;
}
|