-
Notifications
You must be signed in to change notification settings - Fork 10.4k
/
Copy pathForeignRepresentationInfo.h
121 lines (100 loc) · 4.2 KB
/
ForeignRepresentationInfo.h
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
//===--- ForeignRepresentationInfo.h - Used in bridging queries -*- C++ -*-===//
//
// 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
//
//===----------------------------------------------------------------------===//
#ifndef SWIFT_FOREIGNREPRESENTATIONINFO_H
#define SWIFT_FOREIGNREPRESENTATIONINFO_H
#include "swift/AST/Type.h"
#include "swift/Basic/LLVM.h"
#include "llvm/ADT/PointerEmbeddedInt.h"
#include "llvm/ADT/PointerIntPair.h"
namespace swift {
class ProtocolConformance;
class ForeignRepresentationInfo {
using PayloadTy =
llvm::PointerEmbeddedInt<uintptr_t, sizeof(uintptr_t) * CHAR_BIT - 3>;
/// The low three bits store a ForeignRepresentableKind.
///
/// When the ForeignRepresentableKind == None, the upper bits are
/// the generation count at which this negative result was last checked.
/// When it's Bridged, it's the conformance that describes the bridging.
/// When it's Trivial, it's simply a flag stating whether Optional is
/// supported.
llvm::PointerIntPair<PayloadTy, 3, ForeignRepresentableKind> Storage;
public:
/// Retrieve a cache entry for a non-foreign-representable type.
static ForeignRepresentationInfo forNone(unsigned generation = 0) {
ForeignRepresentationInfo result;
result.Storage = { generation, ForeignRepresentableKind::None };
return result;
}
// Retrieve a cache entry for a trivially representable type.
static ForeignRepresentationInfo forTrivial() {
ForeignRepresentationInfo result;
result.Storage = { 0, ForeignRepresentableKind::Trivial };
return result;
}
// Retrieve a cache entry for a trivially representable type that can also
// be optional.
static ForeignRepresentationInfo forTrivialWithOptional() {
ForeignRepresentationInfo result;
result.Storage = { 1, ForeignRepresentableKind::Trivial };
return result;
}
// Retrieve a cache entry for a bridged representable type.
static ForeignRepresentationInfo
forBridged(ProtocolConformance *conformance) {
using PayloadTraits = llvm::PointerLikeTypeTraits<PayloadTy>;
ForeignRepresentationInfo result;
result.Storage = {PayloadTraits::getFromVoidPointer(conformance),
ForeignRepresentableKind::Bridged};
assert(result.getConformance() == conformance && "insufficiently aligned");
return result;
}
// Retrieve a cache entry for a trivially representable type that can also
// be optional.
static ForeignRepresentationInfo forBridgedError() {
ForeignRepresentationInfo result;
result.Storage = { 0, ForeignRepresentableKind::BridgedError };
return result;
}
/// Retrieve the foreign representable kind.
ForeignRepresentableKind getKind() const {
return Storage.getInt();
}
/// Retrieve the generation for a non-representable type.
unsigned getGeneration() const {
assert(getKind() == ForeignRepresentableKind::None);
return Storage.getPointer();
}
/// Retrieve the protocol conformance that makes it representable.
ProtocolConformance *getConformance() const {
switch (getKind()) {
case ForeignRepresentableKind::None:
llvm_unreachable("this type is not representable");
case ForeignRepresentableKind::Trivial:
case ForeignRepresentableKind::BridgedError:
return nullptr;
case ForeignRepresentableKind::Bridged: {
using PayloadTraits = llvm::PointerLikeTypeTraits<PayloadTy>;
auto payload = PayloadTraits::getAsVoidPointer(Storage.getPointer());
return static_cast<ProtocolConformance *>(payload);
}
case ForeignRepresentableKind::Object:
case ForeignRepresentableKind::StaticBridged:
llvm_unreachable("unexpected kind in ForeignRepresentableCacheEntry");
}
llvm_unreachable("Unhandled ForeignRepresentableKind in switch.");
}
/// Returns true if the optional version of this type is also representable.
bool isRepresentableAsOptional() const;
};
} // end namespace swift
#endif // SWIFT_FOREIGNREPRESENTATIONINFO_H