forked from pytorch/pytorch
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Dict.h
374 lines (307 loc) · 12.1 KB
/
Dict.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
#pragma once
#include <c10/macros/Macros.h>
#include <c10/util/TypeTraits.h>
#include <c10/util/TypeList.h>
#include <c10/util/flat_hash_map.h>
#include <c10/util/intrusive_ptr.h>
#include <c10/util/Optional.h>
#include <ATen/core/Tensor.h>
namespace c10 {
struct IValue;
template<class Key, class Value> class Dict;
struct Type;
using TypePtr = std::shared_ptr<Type>;
namespace impl {
bool shallowEquals(const IValue& lhs, const IValue& rhs);
using valid_dict_key_types = guts::typelist::typelist<
int64_t,
std::string,
double,
bool,
at::Tensor
>;
}
namespace detail {
struct DictKeyHash {
size_t operator()(const IValue& ivalue) const;
};
struct DictKeyEqualTo {
bool operator()(const IValue& lhs, const IValue& rhs) const {
return impl::shallowEquals(lhs, rhs);
}
};
struct DictImpl final : public c10::intrusive_ptr_target {
using dict_map_type = ska::flat_hash_map<IValue, IValue, DictKeyHash, DictKeyEqualTo>;
struct DictElementTypes final {
TypePtr keyType;
TypePtr valueType;
};
explicit DictImpl(dict_map_type dict_, optional<DictElementTypes> elementTypes_)
: dict(std::move(dict_))
, elementTypes(std::move(elementTypes_)) {
TORCH_INTERNAL_ASSERT(!elementTypes.has_value() || (nullptr != elementTypes->keyType.get() && nullptr != elementTypes->valueType.get()), "Key and value type must not be nullptr");
}
dict_map_type dict;
// TODO Right now, this is optional, but we want to make it mandatory for all dicts to know their types
optional<DictElementTypes> elementTypes;
intrusive_ptr<DictImpl> copy() const;
};
}
namespace impl {
template<class Key, class Value, class Iterator> class DictIterator;
/**
* A reference to an entry in the Dict.
* Use the `key()` and `value()` methods to read the element.
*/
template<class Key, class Value, class Iterator>
class DictEntryRef final {
public:
explicit DictEntryRef(Iterator iterator)
: iterator_(std::move(iterator)) {}
Key key() const {
return iterator_->first.template to<Key>();
}
Value value() const {
return iterator_->second.template to<Value>();
}
template<class Value_>
void setValue(Value_&& value) const {
static_assert(std::is_constructible<Value, Value_>::value, "Wrong type for the value argument of setValue()");
iterator_->second = Value(std::forward<Value_>(value));
}
private:
// allow copying and moving, but only our friends (i.e. the Dict class) can do
// it. Copying/moving this reference wrapper would be too ambiguous to allow it
// in the public API.
DictEntryRef(const DictEntryRef&) = default;
DictEntryRef& operator=(const DictEntryRef&) = default;
DictEntryRef(DictEntryRef&&) noexcept = default;
DictEntryRef& operator=(DictEntryRef&& rhs) & noexcept = default;
Iterator iterator_;
friend class DictIterator<Key, Value, Iterator>;
friend class Dict<Key, Value>;
};
// this wraps map_type::iterator to make sure user code can't rely
// on it being the type of the underlying map.
template<class Key, class Value, class Iterator>
class DictIterator final : public std::iterator<std::forward_iterator_tag, DictEntryRef<Key, Value, Iterator>> {
public:
explicit DictIterator() = default;
~DictIterator() = default;
DictIterator(const DictIterator& rhs): entryRef_(rhs.entryRef_) {}
DictIterator(DictIterator&& rhs) noexcept: entryRef_(std::move(rhs.entryRef_)) {}
DictIterator& operator=(const DictIterator& rhs) {
entryRef_ = rhs.entryRef_;
return *this;
}
DictIterator& operator=(DictIterator&& rhs) noexcept {
entryRef_ = std::move(rhs.entryRef_);
return *this;
}
DictIterator& operator++() {
++entryRef_.iterator_;
return *this;
}
DictIterator operator++(int) {
DictIterator copy(*this);
++*this;
return copy;
}
const DictEntryRef<Key, Value, Iterator>& operator*() const {
return entryRef_;
}
const DictEntryRef<Key, Value, Iterator>* operator->() const {
return &entryRef_;
}
friend typename std::iterator<std::random_access_iterator_tag, DictEntryRef<Key, Value, Iterator>>::difference_type operator-(const DictIterator& lhs, const DictIterator& rhs) {
return lhs.entryRef_.iterator_ - rhs.entryRef_.iterator_;
}
private:
explicit DictIterator(Iterator iterator): entryRef_(std::move(iterator)) {}
const Iterator& get_iterator_() const {
return entryRef_.iterator_;
}
friend bool operator==(const DictIterator& lhs, const DictIterator& rhs) {
return lhs.get_iterator_() == rhs.get_iterator_();
}
friend bool operator!=(const DictIterator& lhs, const DictIterator& rhs) {
return lhs.get_iterator_() != rhs.get_iterator_();
}
friend bool operator<(const DictIterator& lhs, const DictIterator& rhs) {
return lhs.get_iterator_() < rhs.get_iterator_();
}
friend bool operator<=(const DictIterator& lhs, const DictIterator& rhs) {
return lhs.get_iterator_() <= rhs.get_iterator_();
}
friend bool operator>(const DictIterator& lhs, const DictIterator& rhs) {
return lhs.get_iterator_() > rhs.get_iterator_();
}
friend bool operator>=(const DictIterator& lhs, const DictIterator& rhs) {
return lhs.get_iterator_() >= rhs.get_iterator_();
}
DictEntryRef<Key, Value, Iterator> entryRef_;
friend class DictIterator<Key, Value, typename detail::DictImpl::dict_map_type::iterator>;
friend class Dict<Key, Value>;
};
template<class Key, class Value> Dict<Key, Value> toTypedDict(Dict<IValue, IValue> dict);
template<class Key, class Value> Dict<IValue, IValue> toGenericDict(Dict<Key, Value> dict);
struct deprecatedUntypedDict final {};
}
/**
* An object of this class stores a map from Key to Value.
*
* This is a pointer type. After a copy, both Dicts
* will share the same storage:
*
* > Dict<int, string> a;
* > Dict<int, string> b = a;
* > b.insert(3, "three");
* > ASSERT("three" == a.at(3));
*
* We use this class in the PyTorch kernel API because that
* allows us to do optimizations and switch out the underlying
* map implementation without breaking backwards compatibility
* for the kernel API.
*/
template<class Key, class Value>
class Dict final {
private:
static_assert((std::is_same<IValue, Key>::value && std::is_same<IValue, Value>::value) || guts::typelist::contains<impl::valid_dict_key_types, Key>::value, "Invalid Key type for Dict. We only support int64_t, double, bool, and string.");
// impl_ stores the underlying map as a ska::flat_hash_map.
// We intentionally don't offer conversion from/to
// ska::flat_hash_map, return references to it or something like that,
// because such operations would get expensive if we switch out
// the actual map implementation.
// This is an intrusive_ptr because Dict is a pointer type.
// Invariant: This will never be a nullptr, there will always be a valid
// DictImpl.
c10::intrusive_ptr<detail::DictImpl> impl_;
explicit Dict(c10::intrusive_ptr<detail::DictImpl>&& impl);
friend struct IValue;
template<class K, class V> friend Dict<K, V> impl::toTypedDict(Dict<IValue, IValue>);
template<class K, class V> friend Dict<IValue, IValue> impl::toGenericDict(Dict<K, V>);
public:
using key_type = Key;
using mapped_type = Value;
using size_type = typename detail::DictImpl::dict_map_type::size_type;
using iterator = impl::DictIterator<Key, Value, typename detail::DictImpl::dict_map_type::iterator>;
/**
* Creates an empty dict.
*/
explicit Dict();
/**
* Create a generic dict with runtime type information.
* This only works for c10::impl::GenericDict and is not part of the public API
* but only supposed to be used internally by PyTorch.
*/
explicit Dict(TypePtr keyType, TypePtr valueType);
/**
* Creates an untyped dict, i.e. a Dict that doesn't know its types and
* doesn't do type checking.
* Please don't use this if you can avoid it. We want to get rid of untyped
* dicts.
*/
explicit Dict(impl::deprecatedUntypedDict);
~Dict() = default;
Dict(const Dict&) = default;
Dict& operator=(const Dict&) = default;
Dict(Dict&&) noexcept;
Dict& operator=(Dict&&) noexcept;
/**
* Create a new Dict pointing to a deep copy of the same data.
* The Dict returned is a new dict with separate storage.
* Changes in it are not reflected in the original dict or vice versa.
*/
Dict copy() const;
/**
* Returns an iterator to the first element of the container.
* If the container is empty, the returned iterator will be equal to end().
*/
iterator begin() const;
/**
* Returns an iterator to the element following the last element of the container.
* This element acts as a placeholder; attempting to access it results in undefined behavior.
*/
iterator end() const;
/**
* Checks if the container has no elements.
*/
bool empty() const;
/**
* Returns the number of elements in the container.
*/
size_type size() const;
/**
* Erases all elements from the container. After this call, size() returns zero.
* Invalidates any references, pointers, or iterators referring to contained elements. May also invalidate past-the-end iterators.
*/
void clear() const;
/**
* Inserts element(s) into the container, if the container doesn't already contain an element with an equivalent key.
* May invalidate any references, pointers, or iterators referring to contained elements.
*
* @return A pair consisting of an iterator to the inserted element (or to the element that prevented the insertion) and a bool denoting whether the insertion took place.
*/
template<class Key_, class Value_>
std::pair<iterator, bool> insert(Key_&& key, Value_&& value) const;
/**
* If an element with the given key already exists, it is overwritten with the given value.
* Otherwise, a new element with the given key and value are inserted.
* May invalidate any references, pointers, or iterators referring to contained elements.
*
* @return The bool component is true if the insertion took place and false if the assignment took place. The iterator component is pointing at the element that was inserted or updated.
*/
template<class Key_, class Value_>
std::pair<iterator, bool> insert_or_assign(Key_&& key, Value_&& value) const;
/**
* Removes the element pointed to by iter.
* May invalidate any references, pointers, or iterators referring to contained elements.
* The iterator iter must be valid and dereferenceable. Thus the end() iterator (which is valid, but is not dereferenceable) cannot be used as a value for iter.
*/
void erase(iterator iter) const;
/**
* Removes the element with the given key, if it exists.
* May invalidate any references, pointers, or iterators referring to contained elements.
*
* @return The number of elements removed. This is either '1' if an element with the key existed, or '0' if it didn't.
*/
C10_NODISCARD size_t erase(const Key& key) const;
/**
* Returns the mapped value of the element with key equivalent to key.
* If no such element exists, an exception of type std::out_of_range is thrown.
*/
Value at(const Key& key) const;
/**
* Finds an element with key equivalent to key.
*
* @return Iterator to an element with key equivalent to key.
* If no such element is found, past-the-end (see end()) iterator is returned.
*/
iterator find(const Key& key) const;
/**
* Checks if there is an element with key equivalent to key in the container.
*
* @return true if there is such an element, otherwise false.
*/
bool contains(const Key& key) const;
/**
* Increase the capacity so that at least count elements can be stored without
* having to reallocate or rehash.
*/
void reserve(size_type count) const;
// private API for now because the return type will change to TypePtr
// instead of optional<TypePtr> once types are mandatory.
optional<TypePtr> _keyType() const;
optional<TypePtr> _valueType() const;
};
namespace impl {
// GenericDict is how IValue stores dicts. It is, however, not part of the
// public API. Kernels should use Dicts with concrete Key, Value types instead
// (maybe except for some internal prim ops).
using GenericDict = Dict<IValue, IValue>;
}
}
namespace torch {
template<class Key, class Value> using Dict = c10::Dict<Key, Value>;
}
#include <ATen/core/Dict_inl.h>