forked from swiftlang/swift
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathNonFixedTypeInfo.h
150 lines (126 loc) · 5.2 KB
/
NonFixedTypeInfo.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
//===--- NonFixedTypeInfo.h - Non-fixed-layout types ------------*- 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 classes that are useful for implementing types
// that do not have a fixed representation and cannot be laid out
// statically.
//
// These classes are useful only for creating TypeInfo
// implementations; unlike the similarly-named FixedTypeInfo, they
// do not provide a supplemental API.
//
//===----------------------------------------------------------------------===//
#ifndef SWIFT_IRGEN_NONFIXEDTYPEINFO_H
#define SWIFT_IRGEN_NONFIXEDTYPEINFO_H
#include "GenOpaque.h"
#include "IndirectTypeInfo.h"
namespace swift {
namespace irgen {
/// An abstract CRTP class designed for types whose storage size,
/// alignment, and stride need to be fetched from the value witness
/// table for the type.
template <class Impl>
class WitnessSizedTypeInfo : public IndirectTypeInfo<Impl, TypeInfo> {
private:
typedef IndirectTypeInfo<Impl, TypeInfo> super;
protected:
const Impl &asImpl() const { return static_cast<const Impl &>(*this); }
WitnessSizedTypeInfo(llvm::Type *type, Alignment align, IsPOD_t pod,
IsBitwiseTakable_t bt)
: super(type, align, pod, bt, IsNotFixedSize, TypeInfo::STIK_None) {}
private:
/// Bit-cast the given pointer to the right type and assume it as an
/// address of this type.
Address getAsBitCastAddress(IRGenFunction &IGF, llvm::Value *addr) const {
addr = IGF.Builder.CreateBitCast(addr,
this->getStorageType()->getPointerTo());
return this->getAddressForPointer(addr);
}
public:
// This is useful for metaprogramming.
static bool isFixed() { return false; }
ContainedAddress allocateStack(IRGenFunction &IGF,
SILType T,
const llvm::Twine &name) const override {
// Make a fixed-size buffer.
Address buffer = IGF.createFixedSizeBufferAlloca(name);
IGF.Builder.CreateLifetimeStart(buffer, getFixedBufferSize(IGF.IGM));
// Allocate an object of the appropriate type within it.
llvm::Value *address = emitAllocateBufferCall(IGF, T, buffer);
return { buffer, getAsBitCastAddress(IGF, address) };
}
void deallocateStack(IRGenFunction &IGF, Address buffer,
SILType T) const override {
emitDeallocateBufferCall(IGF, T, buffer);
IGF.Builder.CreateLifetimeEnd(buffer, getFixedBufferSize(IGF.IGM));
}
void destroyStack(IRGenFunction &IGF, Address buffer,
SILType T) const override {
emitDestroyBufferCall(IGF, T, buffer);
IGF.Builder.CreateLifetimeEnd(buffer, getFixedBufferSize(IGF.IGM));
}
llvm::Value *getValueWitnessTable(IRGenFunction &IGF, SILType T) const {
return IGF.emitValueWitnessTableRefForLayout(T);
}
std::pair<llvm::Value*,llvm::Value*>
getSizeAndAlignmentMask(IRGenFunction &IGF, SILType T) const override {
auto size = emitLoadOfSize(IGF, T);
auto align = emitLoadOfAlignmentMask(IGF, T);
return { size, align };
}
std::tuple<llvm::Value*,llvm::Value*,llvm::Value*>
getSizeAndAlignmentMaskAndStride(IRGenFunction &IGF, SILType T) const override {
auto size = emitLoadOfSize(IGF, T);
auto align = emitLoadOfAlignmentMask(IGF, T);
auto stride = emitLoadOfStride(IGF, T);
return std::make_tuple(size, align, stride);
}
llvm::Value *getSize(IRGenFunction &IGF, SILType T) const override {
return emitLoadOfSize(IGF, T);
}
llvm::Value *getAlignmentMask(IRGenFunction &IGF, SILType T) const override {
return emitLoadOfAlignmentMask(IGF, T);
}
llvm::Value *getStride(IRGenFunction &IGF, SILType T) const override {
return emitLoadOfStride(IGF, T);
}
llvm::Value *getIsPOD(IRGenFunction &IGF, SILType T) const override {
return emitLoadOfIsPOD(IGF, T);
}
llvm::Value *isDynamicallyPackedInline(IRGenFunction &IGF,
SILType T) const override {
return emitLoadOfIsInline(IGF, T);
}
/// FIXME: Dynamic extra inhabitant lookup.
bool mayHaveExtraInhabitants(IRGenModule &) const override { return false; }
llvm::Value *getExtraInhabitantIndex(IRGenFunction &IGF,
Address src, SILType T) const override {
llvm_unreachable("dynamic extra inhabitants not supported");
}
void storeExtraInhabitant(IRGenFunction &IGF,
llvm::Value *index,
Address dest, SILType T) const override {
llvm_unreachable("dynamic extra inhabitants not supported");
}
llvm::Constant *getStaticSize(IRGenModule &IGM) const override {
return nullptr;
}
llvm::Constant *getStaticAlignmentMask(IRGenModule &IGM) const override {
return nullptr;
}
llvm::Constant *getStaticStride(IRGenModule &IGM) const override {
return nullptr;
}
};
}
}
#endif