forked from swiftlang/swift
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathTypeInfo.h
485 lines (407 loc) · 19.3 KB
/
TypeInfo.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
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
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
//===--- TypeInfo.h - Abstract primitive operations on values ---*- C++ -*-===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2016 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 the interface used to perform primitive
// operations on swift values and objects.
//
// This interface is supplemented in two ways:
// - FixedTypeInfo provides a number of operations meaningful only
// for types with a fixed-size representation
// - ReferenceTypeInfo is a further refinement of FixedTypeInfo
// which provides operations meaningful only for types with
// reference semantics
//
//===----------------------------------------------------------------------===//
#ifndef SWIFT_IRGEN_TYPEINFO_H
#define SWIFT_IRGEN_TYPEINFO_H
#include "IRGen.h"
namespace llvm {
class Constant;
class Twine;
class Type;
}
namespace swift {
enum IsTake_t : bool;
class SILType;
namespace irgen {
class Address;
class ContainedAddress;
class IRGenFunction;
class IRGenModule;
class Explosion;
class ExplosionSchema;
enum OnHeap_t : unsigned char;
class OwnedAddress;
class RValue;
class RValueSchema;
/// Ways in which an object can fit into a fixed-size buffer.
enum class FixedPacking {
/// It fits at offset zero.
OffsetZero,
/// It doesn't fit and needs to be side-allocated.
Allocate,
/// It needs to be checked dynamically.
Dynamic
};
/// Information about the IR representation and generation of the
/// given type.
class TypeInfo {
TypeInfo(const TypeInfo &) = delete;
TypeInfo &operator=(const TypeInfo &) = delete;
friend class TypeConverter;
mutable const TypeInfo *NextConverted;
protected:
enum SpecialTypeInfoKind {
STIK_Unimplemented,
STIK_None,
/// Everything after this is statically fixed-size.
STIK_Fixed,
STIK_Weak,
/// Everything after this is loadable.
STIK_Loadable,
STIK_Reference,
};
TypeInfo(llvm::Type *Type, Alignment A, IsPOD_t pod,
IsBitwiseTakable_t bitwiseTakable,
IsFixedSize_t alwaysFixedSize,
SpecialTypeInfoKind stik)
: NextConverted(0), StorageType(Type), StorageAlignment(A),
POD(pod), BitwiseTakable(bitwiseTakable),
AlwaysFixedSize(alwaysFixedSize), STIK(stik),
SubclassKind(InvalidSubclassKind) {
assert(STIK >= STIK_Fixed || !AlwaysFixedSize);
}
/// Change the minimum alignment of a stored value of this type.
void setStorageAlignment(Alignment alignment) {
StorageAlignment = alignment;
}
public:
virtual ~TypeInfo() = default;
/// Unsafely cast this to the given subtype.
template <class T> const T &as() const {
// FIXME: maybe do an assert somehow if we have RTTI enabled.
return static_cast<const T &>(*this);
}
private:
/// The LLVM representation of a stored value of this type. For
/// non-fixed types, this is really useful only for forming pointers to it.
llvm::Type *StorageType;
/// The storage alignment of this type in bytes. This is never zero
/// for a completely-converted type.
Alignment StorageAlignment;
/// Whether this type is known to be POD.
unsigned POD : 1;
/// Whether this type is known to be bitwise-takable.
unsigned BitwiseTakable : 1;
/// Whether this type can be assumed to have a fixed size from all
/// resilience domains.
unsigned AlwaysFixedSize : 1;
/// The kind of supplemental API this type has, if any.
unsigned STIK : 3;
/// An arbitrary discriminator for the subclass. This is useful for
/// e.g. distinguishing between different TypeInfos that all
/// implement the same kind of type.
unsigned SubclassKind : 3;
enum { InvalidSubclassKind = 0x7 };
protected:
void setSubclassKind(unsigned kind) {
assert(kind != InvalidSubclassKind);
SubclassKind = kind;
assert(SubclassKind == kind && "kind was truncated?");
}
public:
/// Whether this type info has been completely converted.
bool isComplete() const { return !StorageAlignment.isZero(); }
/// Whether this type is known to be empty.
bool isKnownEmpty(ResilienceExpansion expansion) const;
/// Whether this type is known to be POD, i.e. to not require any
/// particular action on copy or destroy.
IsPOD_t isPOD(ResilienceExpansion expansion) const { return IsPOD_t(POD); }
/// Whether this type is known to be bitwise-takable, i.e. "initializeWithTake"
/// is equivalent to a memcpy.
IsBitwiseTakable_t isBitwiseTakable(ResilienceExpansion expansion) const {
return IsBitwiseTakable_t(BitwiseTakable);
}
/// Returns the type of special interface followed by this TypeInfo.
/// It is important for our design that this depends only on
/// immediate type structure and not on, say, properties that can
/// vary by resilience. Of course, generics can obscure these
/// properties on their parameter types, but then the program
/// can rely on them.
SpecialTypeInfoKind getSpecialTypeInfoKind() const {
return SpecialTypeInfoKind(STIK);
}
/// Returns whatever arbitrary data has been stash in the subclass
/// kind field. This mechanism allows an orthogonal dimension of
/// distinguishing between TypeInfos, which is useful when multiple
/// TypeInfo subclasses are used to implement the same kind of type.
unsigned getSubclassKind() const {
assert(SubclassKind != InvalidSubclassKind &&
"subclass kind has not been initialized!");
return SubclassKind;
}
/// Whether this type is known to be fixed-size in the local
/// resilience domain. If true, this TypeInfo can be cast to
/// FixedTypeInfo.
IsFixedSize_t isFixedSize() const {
return IsFixedSize_t(STIK >= STIK_Fixed);
}
/// Whether this type is known to be fixed-size in the given
/// resilience domain. If true, spare bits can be used.
IsFixedSize_t isFixedSize(ResilienceExpansion expansion) const {
switch (expansion) {
case ResilienceExpansion::Maximal:
return isFixedSize();
case ResilienceExpansion::Minimal:
// We can't be universally fixed size if we're not locally
// fixed size.
assert((isFixedSize() || AlwaysFixedSize == IsNotFixedSize) &&
"IsFixedSize vs IsAlwaysFixedSize mismatch");
return IsFixedSize_t(AlwaysFixedSize);
}
}
/// Whether this type is known to be loadable in the local
/// resilience domain. If true, this TypeInfo can be cast to
/// LoadableTypeInfo.
IsLoadable_t isLoadable() const {
return IsLoadable_t(STIK >= STIK_Loadable);
}
llvm::Type *getStorageType() const { return StorageType; }
Alignment getBestKnownAlignment() const {
return StorageAlignment;
}
/// Given a generic pointer to this type, produce an Address for it.
Address getAddressForPointer(llvm::Value *ptr) const;
/// Produce an undefined pointer to an object of this type.
Address getUndefAddress() const;
/// Return the size and alignment of this type.
virtual std::pair<llvm::Value*,llvm::Value*>
getSizeAndAlignmentMask(IRGenFunction &IGF, SILType T) const = 0;
virtual std::tuple<llvm::Value*,llvm::Value*,llvm::Value*>
getSizeAndAlignmentMaskAndStride(IRGenFunction &IGF, SILType T) const = 0;
virtual llvm::Value *getSize(IRGenFunction &IGF, SILType T) const = 0;
virtual llvm::Value *getAlignmentMask(IRGenFunction &IGF, SILType T) const = 0;
virtual llvm::Value *getStride(IRGenFunction &IGF, SILType T) const = 0;
virtual llvm::Value *getIsPOD(IRGenFunction &IGF, SILType T) const = 0;
virtual llvm::Value *isDynamicallyPackedInline(IRGenFunction &IGF,
SILType T) const = 0;
/// Return the statically-known size of this type, or null if it is
/// not known.
virtual llvm::Constant *getStaticSize(IRGenModule &IGM) const = 0;
/// Return the statically-known alignment mask for this type, or
/// null if it is not known.
virtual llvm::Constant *getStaticAlignmentMask(IRGenModule &IGM) const = 0;
/// Return the statically-known stride size of this type, or null if
/// it is not known.
virtual llvm::Constant *getStaticStride(IRGenModule &IGM) const = 0;
/// Add the information for exploding values of this type to the
/// given schema.
virtual void getSchema(ExplosionSchema &schema) const = 0;
/// A convenience for getting the schema of a single type.
ExplosionSchema getSchema() const;
/// Allocate a variable of this type on the stack.
virtual ContainedAddress allocateStack(IRGenFunction &IGF,
SILType T,
const llvm::Twine &name) const = 0;
/// Deallocate a variable of this type.
virtual void deallocateStack(IRGenFunction &IGF, Address addr,
SILType T) const = 0;
/// Destroy the value of a variable of this type, then deallocate its
/// memory.
virtual void destroyStack(IRGenFunction &IGF, Address addr,
SILType T) const = 0;
/// Copy or take a value out of one address and into another, destroying
/// old value in the destination. Equivalent to either assignWithCopy
/// or assignWithTake depending on the value of isTake.
void assign(IRGenFunction &IGF, Address dest, Address src, IsTake_t isTake,
SILType T) const;
/// Copy a value out of an object and into another, destroying the
/// old value in the destination.
virtual void assignWithCopy(IRGenFunction &IGF, Address dest,
Address src, SILType T) const = 0;
/// Move a value out of an object and into another, destroying the
/// old value there and leaving the source object in an invalid state.
virtual void assignWithTake(IRGenFunction &IGF, Address dest,
Address src, SILType T) const = 0;
/// Copy-initialize or take-initialize an uninitialized object
/// with the value from a different object. Equivalent to either
/// initializeWithCopy or initializeWithTake depending on the value
/// of isTake.
void initialize(IRGenFunction &IGF, Address dest, Address src,
IsTake_t isTake, SILType T) const;
/// Perform a "take-initialization" from the given object. A
/// take-initialization is like a C++ move-initialization, except that
/// the old object is actually no longer permitted to be destroyed.
virtual void initializeWithTake(IRGenFunction &IGF, Address destAddr,
Address srcAddr, SILType T) const = 0;
/// Perform a copy-initialization from the given object.
virtual void initializeWithCopy(IRGenFunction &IGF, Address destAddr,
Address srcAddr, SILType T) const = 0;
/// Allocate space for an object of this type within an uninitialized
/// fixed-size buffer.
virtual Address allocateBuffer(IRGenFunction &IGF, Address buffer,
SILType T) const;
/// Project the address of an object of this type from an initialized
/// fixed-size buffer.
virtual Address projectBuffer(IRGenFunction &IGF, Address buffer,
SILType T) const;
/// Perform a "take-initialization" from the given object into an
/// uninitialized fixed-size buffer, allocating the buffer if necessary.
/// Returns the address of the value inside the buffer.
///
/// This is equivalent to:
/// auto destAddress = allocateBuffer(IGF, destBuffer, T);
/// initializeWithTake(IGF, destAddr, srcAddr, T);
/// return destAddress;
/// but will be more efficient for dynamic types, since it uses a single
/// value witness call.
virtual Address initializeBufferWithTake(IRGenFunction &IGF,
Address destBuffer,
Address srcAddr,
SILType T) const;
/// Perform a copy-initialization from the given object into an
/// uninitialized fixed-size buffer, allocating the buffer if necessary.
/// Returns the address of the value inside the buffer.
///
/// This is equivalent to:
/// auto destAddress = allocateBuffer(IGF, destBuffer, T);
/// initializeWithCopy(IGF, destAddr, srcAddr, T);
/// return destAddress;
/// but will be more efficient for dynamic types, since it uses a single
/// value witness call.
virtual Address initializeBufferWithCopy(IRGenFunction &IGF,
Address destBuffer,
Address srcAddr,
SILType T) const;
/// Perform a copy-initialization from the given fixed-size buffer
/// into an uninitialized fixed-size buffer, allocating the buffer if
/// necessary. Returns the address of the value inside the buffer.
///
/// This is equivalent to:
/// auto srcAddress = projectBuffer(IGF, srcBuffer, T);
/// initializeBufferWithCopy(IGF, destBuffer, srcAddress, T);
/// but will be more efficient for dynamic types, since it uses a single
/// value witness call.
virtual Address initializeBufferWithCopyOfBuffer(IRGenFunction &IGF,
Address destBuffer,
Address srcBuffer,
SILType T) const;
/// Perform a take-initialization from the given fixed-size buffer
/// into an uninitialized fixed-size buffer, allocating the buffer if
/// necessary and deallocating the destination buffer. Returns the
/// address of the value inside the destination buffer.
///
/// This is equivalent to:
/// auto srcAddress = projectBuffer(IGF, srcBuffer, T);
/// initializeBufferWithTake(IGF, destBuffer, srcAddress, T);
/// deallocateBuffer(IGF, srcBuffer, T);
/// but may be able to re-use the buffer from the source buffer, and may
/// be more efficient for dynamic types, since it uses a single
/// value witness call.
virtual Address initializeBufferWithTakeOfBuffer(IRGenFunction &IGF,
Address destBuffer,
Address srcBuffer,
SILType T) const;
/// Destroy an object of this type within an initialized fixed-size buffer
/// and deallocate the buffer.
///
/// This is equivalent to:
/// auto valueAddr = projectBuffer(IGF, buffer, T);
/// destroy(IGF, valueAddr, T);
/// deallocateBuffer(IGF, buffer, T);
/// but will be more efficient for dynamic types, since it uses a single
/// value witness call.
virtual void destroyBuffer(IRGenFunction &IGF, Address buffer,
SILType T) const;
/// Deallocate the space for an object of this type within an initialized
/// fixed-size buffer.
virtual void deallocateBuffer(IRGenFunction &IGF, Address buffer,
SILType T) const;
/// Take-initialize an address from a parameter explosion.
virtual void initializeFromParams(IRGenFunction &IGF, Explosion ¶ms,
Address src, SILType T) const = 0;
/// Destroy an object of this type in memory.
virtual void destroy(IRGenFunction &IGF, Address address, SILType T) const = 0;
/// Should optimizations be enabled which rely on the representation
/// for this type being a single object pointer?
///
/// \return false by default
virtual bool isSingleRetainablePointer(ResilienceExpansion expansion,
ReferenceCounting *refcounting
= nullptr) const;
/// Should optimizations be enabled which rely on the representation
/// for this type being a single Swift-retainable object pointer?
///
/// \return false by default
bool isSingleSwiftRetainablePointer(ResilienceExpansion expansion) const {
ReferenceCounting refcounting;
return (isSingleRetainablePointer(expansion, &refcounting) &&
refcounting == ReferenceCounting::Native);
}
/// Does this type statically have extra inhabitants, or may it dynamically
/// have extra inhabitants based on type arguments?
virtual bool mayHaveExtraInhabitants(IRGenModule &IGM) const = 0;
/// Map an extra inhabitant representation in memory to a unique 31-bit
/// identifier, and map a valid representation of the type to -1.
///
/// Calls to this witness must be dominated by a runtime check that the type
/// has extra inhabitants.
virtual llvm::Value *getExtraInhabitantIndex(IRGenFunction &IGF,
Address src,
SILType T) const = 0;
/// Store the extra inhabitant representation indexed by a 31-bit identifier
/// to memory.
///
/// Calls to this witness must be dominated by a runtime check that the type
/// has extra inhabitants.
virtual void storeExtraInhabitant(IRGenFunction &IGF,
llvm::Value *index,
Address dest,
SILType T) const = 0;
/// Initialize a freshly instantiated value witness table. Should be a no-op
/// for fixed-size types.
virtual void initializeMetadata(IRGenFunction &IGF,
llvm::Value *metadata,
llvm::Value *vwtable,
SILType T) const = 0;
/// Compute the packing of values of this type into a fixed-size buffer.
FixedPacking getFixedPacking(IRGenModule &IGM) const;
/// Index into an array of objects of this type.
Address indexArray(IRGenFunction &IGF, Address base, llvm::Value *offset,
SILType T) const;
/// Destroy an array of objects of this type in memory.
virtual void destroyArray(IRGenFunction &IGF, Address base,
llvm::Value *count, SILType T) const;
/// Initialize an array of objects of this type in memory by copying the
/// values from another array. The arrays must not overlap.
virtual void initializeArrayWithCopy(IRGenFunction &IGF,
Address dest,
Address src,
llvm::Value *count, SILType T) const;
/// Initialize an array of objects of this type in memory by taking the
/// values from another array. The destination array may overlap the head of
/// the source array because the elements are taken as if in front-to-back
/// order.
virtual void initializeArrayWithTakeFrontToBack(IRGenFunction &IGF,
Address dest, Address src,
llvm::Value *count, SILType T) const;
/// Initialize an array of objects of this type in memory by taking the
/// values from another array. The destination array may overlap the tail of
/// the source array because the elements are taken as if in back-to-front
/// order.
virtual void initializeArrayWithTakeBackToFront(IRGenFunction &IGF,
Address dest, Address src,
llvm::Value *count, SILType T) const;
};
} // end namespace irgen
} // end namespace swift
#endif