forked from swiftlang/swift
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathFixedTypeInfo.h
234 lines (197 loc) · 9.24 KB
/
FixedTypeInfo.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
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
//===--- FixedTypeInfo.h - Supplement for fixed-layout types ----*- C++ -*-===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2015 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See http://swift.org/LICENSE.txt for license information
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
//
// This file defines FixedTypeInfo, which supplements the TypeInfo
// interface for classes with (at least locally) fixed-layout type
// implementations.
//
//===----------------------------------------------------------------------===//
#ifndef SWIFT_IRGEN_FIXEDTYPEINFO_H
#define SWIFT_IRGEN_FIXEDTYPEINFO_H
#include "Address.h"
#include "TypeInfo.h"
#include "swift/Basic/ClusteredBitVector.h"
#include "swift/SIL/SILType.h"
namespace llvm {
class ConstantInt;
}
namespace swift {
namespace irgen {
/// FixedTypeInfo - An abstract class designed for use when
/// implementing a type that has a statically known layout.
class FixedTypeInfo : public TypeInfo {
private:
/// The storage size of this type in bytes. This may be zero even
/// for well-formed and complete types, such as a trivial enum or
/// tuple.
Size StorageSize;
/// The spare bit mask for this type. SpareBits[0] is the LSB of the first
/// byte. This may be empty if the type has no spare bits.
SpareBitVector SpareBits;
protected:
FixedTypeInfo(llvm::Type *type, Size size,
const SpareBitVector &spareBits,
Alignment align, IsPOD_t pod, IsBitwiseTakable_t bt,
IsFixedSize_t alwaysFixedSize,
SpecialTypeInfoKind stik = STIK_Fixed)
: TypeInfo(type, align, pod, bt, alwaysFixedSize, stik),
StorageSize(size), SpareBits(spareBits) {
assert(SpareBits.size() == size.getValueInBits());
assert(isFixedSize());
}
FixedTypeInfo(llvm::Type *type, Size size,
SpareBitVector &&spareBits,
Alignment align, IsPOD_t pod, IsBitwiseTakable_t bt,
IsFixedSize_t alwaysFixedSize,
SpecialTypeInfoKind stik = STIK_Fixed)
: TypeInfo(type, align, pod, bt, alwaysFixedSize, stik),
StorageSize(size), SpareBits(std::move(spareBits)) {
assert(SpareBits.size() == size.getValueInBits());
assert(isFixedSize());
}
public:
// This is useful for metaprogramming.
static bool isFixed() { return true; }
/// Whether this type is known to be empty.
bool isKnownEmpty() const { return StorageSize.isZero(); }
ContainedAddress allocateStack(IRGenFunction &IGF, SILType T,
const llvm::Twine &name) const override;
void deallocateStack(IRGenFunction &IGF, Address addr, SILType T) const override;
// We can give these reasonable default implementations.
void initializeWithTake(IRGenFunction &IGF, Address destAddr,
Address srcAddr, SILType T) const override;
std::pair<llvm::Value*, llvm::Value*>
getSizeAndAlignmentMask(IRGenFunction &IGF, SILType T) const override;
std::tuple<llvm::Value*,llvm::Value*,llvm::Value*>
getSizeAndAlignmentMaskAndStride(IRGenFunction &IGF, SILType T) const override;
llvm::Value *getSize(IRGenFunction &IGF, SILType T) const override;
llvm::Value *getAlignmentMask(IRGenFunction &IGF, SILType T) const override;
llvm::Value *getStride(IRGenFunction &IGF, SILType T) const override;
llvm::Value *getIsPOD(IRGenFunction &IGF, SILType T) const override;
llvm::Value *isDynamicallyPackedInline(IRGenFunction &IGF,
SILType T) const override;
llvm::Constant *getStaticSize(IRGenModule &IGM) const override;
llvm::Constant *getStaticAlignmentMask(IRGenModule &IGM) const override;
llvm::Constant *getStaticStride(IRGenModule &IGM) const override;
void completeFixed(Size size, Alignment alignment) {
StorageSize = size;
setStorageAlignment(alignment);
}
/// Returns the known, fixed alignment of a stored value of this type.
Alignment getFixedAlignment() const {
return getBestKnownAlignment();
}
/// Returns the known, fixed size required to store a value of this type.
Size getFixedSize() const {
return StorageSize;
}
/// Returns the (assumed fixed) stride of the storage for this
/// object. The stride is the storage size rounded up to the
/// alignment; its practical use is that, in an array, it is the
/// offset from the size of one element to the offset of the next.
Size getFixedStride() const {
return StorageSize.roundUpToAlignment(getFixedAlignment());
}
/// Returns the fixed number of "extra inhabitants" (that is, bit
/// patterns that don't represent valid values of the type) in the type
/// representation.
virtual unsigned getFixedExtraInhabitantCount(IRGenModule &IGM) const {
return getSpareBitExtraInhabitantCount();
}
/// Returns the number of extra inhabitants available by exercising spare
/// bits.
unsigned getSpareBitExtraInhabitantCount() const;
/// We can statically determine the presence of extra inhabitants for fixed
/// types.
bool mayHaveExtraInhabitants(IRGenModule &IGM) const override {
return getFixedExtraInhabitantCount(IGM) > 0;
}
/// Get the bit mask that must be applied before testing an extra inhabitant.
virtual APInt getFixedExtraInhabitantMask(IRGenModule &IGM) const {
return APInt::getAllOnesValue(getFixedSize().getValueInBits());
}
/// Create a constant of the given bit width holding one of the extra
/// inhabitants of the type.
/// The index must be less than the value returned by
/// getFixedExtraInhabitantCount().
virtual APInt getFixedExtraInhabitantValue(IRGenModule &IGM,
unsigned bits,
unsigned index) const {
return getSpareBitFixedExtraInhabitantValue(IGM, bits, index);
}
/// Create an extra inhabitant constant using the spare bits of the type.
APInt getSpareBitFixedExtraInhabitantValue(IRGenModule &IGM,
unsigned bits,
unsigned index) const;
/// Map an extra inhabitant representation in memory to a unique 31-bit
/// identifier, and map a valid representation of the type to -1.
llvm::Value *getExtraInhabitantIndex(IRGenFunction &IGF,
Address src, SILType T) const override {
return getSpareBitExtraInhabitantIndex(IGF, src);
}
/// Map an extra inhabitant representation derived from spare bits to an
/// index.
llvm::Value *getSpareBitExtraInhabitantIndex(IRGenFunction &IGF,
Address src) const;
/// Store the extra inhabitant representation indexed by a 31-bit identifier
/// to memory.
void storeExtraInhabitant(IRGenFunction &IGF,
llvm::Value *index,
Address dest, SILType T) const override {
storeSpareBitExtraInhabitant(IGF, index, dest);
}
/// Store the indexed spare-bit-derived extra inhabitant to memory.
void storeSpareBitExtraInhabitant(IRGenFunction &IGF,
llvm::Value *index,
Address dest) const;
/// Get the spare bit mask for the type.
const SpareBitVector &getSpareBits() const { return SpareBits; }
/// True if the type representation has statically "spare" unused bits.
bool hasFixedSpareBits() const {
return SpareBits.any();
}
/// Applies the fixed spare bits mask for this type to the given BitVector,
/// clearing any bits used by valid representations of the type.
///
/// If the bitvector is empty or smaller than this type, it is grown and
/// filled with bits direct from the spare bits mask. If the bitvector is
/// larger than this type, the trailing bits are untouched.
///
/// The intent is that, for all the data types of an enum, you should be able
/// to do this:
///
/// SpareBitVector spareBits;
/// for (EnumElementDecl *elt : u->getAllElements())
/// getFragileTypeInfo(elt->getArgumentType())
/// .applyFixedSpareBitsMask(spareBits, 0);
///
/// and end up with a spare bits mask for the entire enum.
void applyFixedSpareBitsMask(SpareBitVector &mask) const;
/// Applies a fixed spare bits mask to the given BitVector,
/// clearing any bits used by valid representations of the type.
///
/// If the bitvector is empty or smaller than this type, it is grown and
/// filled with bits direct from the spare bits mask. If the bitvector is
/// larger than this type, the trailing bits are untouched.
static void applyFixedSpareBitsMask(SpareBitVector &mask,
const SpareBitVector &spareBits);
/// Fixed-size types never need dynamic value witness table instantiation.
void initializeMetadata(IRGenFunction &IGF,
llvm::Value *metadata,
llvm::Value *vwtable,
SILType T) const override {}
static bool classof(const FixedTypeInfo *type) { return true; }
static bool classof(const TypeInfo *type) { return type->isFixedSize(); }
};
}
}
#endif