cannam@147
|
1 // Copyright (c) 2013-2014 Sandstorm Development Group, Inc. and contributors
|
cannam@147
|
2 // Licensed under the MIT License:
|
cannam@147
|
3 //
|
cannam@147
|
4 // Permission is hereby granted, free of charge, to any person obtaining a copy
|
cannam@147
|
5 // of this software and associated documentation files (the "Software"), to deal
|
cannam@147
|
6 // in the Software without restriction, including without limitation the rights
|
cannam@147
|
7 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
cannam@147
|
8 // copies of the Software, and to permit persons to whom the Software is
|
cannam@147
|
9 // furnished to do so, subject to the following conditions:
|
cannam@147
|
10 //
|
cannam@147
|
11 // The above copyright notice and this permission notice shall be included in
|
cannam@147
|
12 // all copies or substantial portions of the Software.
|
cannam@147
|
13 //
|
cannam@147
|
14 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
cannam@147
|
15 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
cannam@147
|
16 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
cannam@147
|
17 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
cannam@147
|
18 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
cannam@147
|
19 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
cannam@147
|
20 // THE SOFTWARE.
|
cannam@147
|
21
|
cannam@147
|
22 #ifndef CAPNP_SCHEMA_H_
|
cannam@147
|
23 #define CAPNP_SCHEMA_H_
|
cannam@147
|
24
|
cannam@147
|
25 #if defined(__GNUC__) && !defined(CAPNP_HEADER_WARNINGS)
|
cannam@147
|
26 #pragma GCC system_header
|
cannam@147
|
27 #endif
|
cannam@147
|
28
|
cannam@147
|
29 #if CAPNP_LITE
|
cannam@147
|
30 #error "Reflection APIs, including this header, are not available in lite mode."
|
cannam@147
|
31 #endif
|
cannam@147
|
32
|
cannam@147
|
33 #include <capnp/schema.capnp.h>
|
cannam@147
|
34
|
cannam@147
|
35 namespace capnp {
|
cannam@147
|
36
|
cannam@147
|
37 class Schema;
|
cannam@147
|
38 class StructSchema;
|
cannam@147
|
39 class EnumSchema;
|
cannam@147
|
40 class InterfaceSchema;
|
cannam@147
|
41 class ConstSchema;
|
cannam@147
|
42 class ListSchema;
|
cannam@147
|
43 class Type;
|
cannam@147
|
44
|
cannam@147
|
45 template <typename T, Kind k = kind<T>()> struct SchemaType_ { typedef Schema Type; };
|
cannam@147
|
46 template <typename T> struct SchemaType_<T, Kind::PRIMITIVE> { typedef schema::Type::Which Type; };
|
cannam@147
|
47 template <typename T> struct SchemaType_<T, Kind::BLOB> { typedef schema::Type::Which Type; };
|
cannam@147
|
48 template <typename T> struct SchemaType_<T, Kind::ENUM> { typedef EnumSchema Type; };
|
cannam@147
|
49 template <typename T> struct SchemaType_<T, Kind::STRUCT> { typedef StructSchema Type; };
|
cannam@147
|
50 template <typename T> struct SchemaType_<T, Kind::INTERFACE> { typedef InterfaceSchema Type; };
|
cannam@147
|
51 template <typename T> struct SchemaType_<T, Kind::LIST> { typedef ListSchema Type; };
|
cannam@147
|
52
|
cannam@147
|
53 template <typename T>
|
cannam@147
|
54 using SchemaType = typename SchemaType_<T>::Type;
|
cannam@147
|
55 // SchemaType<T> is the type of T's schema, e.g. StructSchema if T is a struct.
|
cannam@147
|
56
|
cannam@147
|
57 namespace _ { // private
|
cannam@147
|
58 extern const RawSchema NULL_SCHEMA;
|
cannam@147
|
59 extern const RawSchema NULL_STRUCT_SCHEMA;
|
cannam@147
|
60 extern const RawSchema NULL_ENUM_SCHEMA;
|
cannam@147
|
61 extern const RawSchema NULL_INTERFACE_SCHEMA;
|
cannam@147
|
62 extern const RawSchema NULL_CONST_SCHEMA;
|
cannam@147
|
63 // The schema types default to these null (empty) schemas in case of error, especially when
|
cannam@147
|
64 // exceptions are disabled.
|
cannam@147
|
65 } // namespace _ (private)
|
cannam@147
|
66
|
cannam@147
|
67 class Schema {
|
cannam@147
|
68 // Convenience wrapper around capnp::schema::Node.
|
cannam@147
|
69
|
cannam@147
|
70 public:
|
cannam@147
|
71 inline Schema(): raw(&_::NULL_SCHEMA.defaultBrand) {}
|
cannam@147
|
72
|
cannam@147
|
73 template <typename T>
|
cannam@147
|
74 static inline SchemaType<T> from() { return SchemaType<T>::template fromImpl<T>(); }
|
cannam@147
|
75 // Get the Schema for a particular compiled-in type.
|
cannam@147
|
76
|
cannam@147
|
77 schema::Node::Reader getProto() const;
|
cannam@147
|
78 // Get the underlying Cap'n Proto representation of the schema node. (Note that this accessor
|
cannam@147
|
79 // has performance comparable to accessors of struct-typed fields on Reader classes.)
|
cannam@147
|
80
|
cannam@147
|
81 kj::ArrayPtr<const word> asUncheckedMessage() const;
|
cannam@147
|
82 // Get the encoded schema node content as a single message segment. It is safe to read as an
|
cannam@147
|
83 // unchecked message.
|
cannam@147
|
84
|
cannam@147
|
85 Schema getDependency(uint64_t id) const KJ_DEPRECATED("Does not handle generics correctly.");
|
cannam@147
|
86 // DEPRECATED: This method cannot correctly account for generic type parameter bindings that
|
cannam@147
|
87 // may apply to the dependency. Instead of using this method, use a method of the Schema API
|
cannam@147
|
88 // that corresponds to the exact kind of dependency. For example, to get a field type, use
|
cannam@147
|
89 // StructSchema::Field::getType().
|
cannam@147
|
90 //
|
cannam@147
|
91 // Gets the Schema for one of this Schema's dependencies. For example, if this Schema is for a
|
cannam@147
|
92 // struct, you could look up the schema for one of its fields' types. Throws an exception if this
|
cannam@147
|
93 // schema doesn't actually depend on the given id.
|
cannam@147
|
94 //
|
cannam@147
|
95 // Note that not all type IDs found in the schema node are considered "dependencies" -- only the
|
cannam@147
|
96 // ones that are needed to implement the dynamic API are. That includes:
|
cannam@147
|
97 // - Field types.
|
cannam@147
|
98 // - Group types.
|
cannam@147
|
99 // - scopeId for group nodes, but NOT otherwise.
|
cannam@147
|
100 // - Method parameter and return types.
|
cannam@147
|
101 //
|
cannam@147
|
102 // The following are NOT considered dependencies:
|
cannam@147
|
103 // - Nested nodes.
|
cannam@147
|
104 // - scopeId for a non-group node.
|
cannam@147
|
105 // - Annotations.
|
cannam@147
|
106 //
|
cannam@147
|
107 // To obtain schemas for those, you would need a SchemaLoader.
|
cannam@147
|
108
|
cannam@147
|
109 bool isBranded() const;
|
cannam@147
|
110 // Returns true if this schema represents a non-default parameterization of this type.
|
cannam@147
|
111
|
cannam@147
|
112 Schema getGeneric() const;
|
cannam@147
|
113 // Get the version of this schema with any brands removed.
|
cannam@147
|
114
|
cannam@147
|
115 class BrandArgumentList;
|
cannam@147
|
116 BrandArgumentList getBrandArgumentsAtScope(uint64_t scopeId) const;
|
cannam@147
|
117 // Gets the values bound to the brand parameters at the given scope.
|
cannam@147
|
118
|
cannam@147
|
119 StructSchema asStruct() const;
|
cannam@147
|
120 EnumSchema asEnum() const;
|
cannam@147
|
121 InterfaceSchema asInterface() const;
|
cannam@147
|
122 ConstSchema asConst() const;
|
cannam@147
|
123 // Cast the Schema to a specific type. Throws an exception if the type doesn't match. Use
|
cannam@147
|
124 // getProto() to determine type, e.g. getProto().isStruct().
|
cannam@147
|
125
|
cannam@147
|
126 inline bool operator==(const Schema& other) const { return raw == other.raw; }
|
cannam@147
|
127 inline bool operator!=(const Schema& other) const { return raw != other.raw; }
|
cannam@147
|
128 // Determine whether two Schemas are wrapping the exact same underlying data, by identity. If
|
cannam@147
|
129 // you want to check if two Schemas represent the same type (but possibly different versions of
|
cannam@147
|
130 // it), compare their IDs instead.
|
cannam@147
|
131
|
cannam@147
|
132 template <typename T>
|
cannam@147
|
133 void requireUsableAs() const;
|
cannam@147
|
134 // Throws an exception if a value with this Schema cannot safely be cast to a native value of
|
cannam@147
|
135 // the given type. This passes if either:
|
cannam@147
|
136 // - *this == from<T>()
|
cannam@147
|
137 // - This schema was loaded with SchemaLoader, the type ID matches typeId<T>(), and
|
cannam@147
|
138 // loadCompiledTypeAndDependencies<T>() was called on the SchemaLoader.
|
cannam@147
|
139
|
cannam@147
|
140 kj::StringPtr getShortDisplayName() const;
|
cannam@147
|
141 // Get the short version of the node's display name.
|
cannam@147
|
142
|
cannam@147
|
143 private:
|
cannam@147
|
144 const _::RawBrandedSchema* raw;
|
cannam@147
|
145
|
cannam@147
|
146 inline explicit Schema(const _::RawBrandedSchema* raw): raw(raw) {
|
cannam@147
|
147 KJ_IREQUIRE(raw->lazyInitializer == nullptr,
|
cannam@147
|
148 "Must call ensureInitialized() on RawSchema before constructing Schema.");
|
cannam@147
|
149 }
|
cannam@147
|
150
|
cannam@147
|
151 template <typename T> static inline Schema fromImpl() {
|
cannam@147
|
152 return Schema(&_::rawSchema<T>());
|
cannam@147
|
153 }
|
cannam@147
|
154
|
cannam@147
|
155 void requireUsableAs(const _::RawSchema* expected) const;
|
cannam@147
|
156
|
cannam@147
|
157 uint32_t getSchemaOffset(const schema::Value::Reader& value) const;
|
cannam@147
|
158
|
cannam@147
|
159 Type getBrandBinding(uint64_t scopeId, uint index) const;
|
cannam@147
|
160 // Look up the binding for a brand parameter used by this Schema. Returns `AnyPointer` if the
|
cannam@147
|
161 // parameter is not bound.
|
cannam@147
|
162 //
|
cannam@147
|
163 // TODO(someday): Public interface for iterating over all bindings?
|
cannam@147
|
164
|
cannam@147
|
165 Schema getDependency(uint64_t id, uint location) const;
|
cannam@147
|
166 // Look up schema for a particular dependency of this schema. `location` is the dependency
|
cannam@147
|
167 // location number as defined in _::RawBrandedSchema.
|
cannam@147
|
168
|
cannam@147
|
169 Type interpretType(schema::Type::Reader proto, uint location) const;
|
cannam@147
|
170 // Interpret a schema::Type in the given location within the schema, compiling it into a
|
cannam@147
|
171 // Type object.
|
cannam@147
|
172
|
cannam@147
|
173 friend class StructSchema;
|
cannam@147
|
174 friend class EnumSchema;
|
cannam@147
|
175 friend class InterfaceSchema;
|
cannam@147
|
176 friend class ConstSchema;
|
cannam@147
|
177 friend class ListSchema;
|
cannam@147
|
178 friend class SchemaLoader;
|
cannam@147
|
179 friend class Type;
|
cannam@147
|
180 friend kj::StringTree _::structString(
|
cannam@147
|
181 _::StructReader reader, const _::RawBrandedSchema& schema);
|
cannam@147
|
182 friend kj::String _::enumString(uint16_t value, const _::RawBrandedSchema& schema);
|
cannam@147
|
183 };
|
cannam@147
|
184
|
cannam@147
|
185 kj::StringPtr KJ_STRINGIFY(const Schema& schema);
|
cannam@147
|
186
|
cannam@147
|
187 class Schema::BrandArgumentList {
|
cannam@147
|
188 // A list of generic parameter bindings for parameters of some particular type. Note that since
|
cannam@147
|
189 // parameters on an outer type apply to all inner types as well, a deeply-nested type can have
|
cannam@147
|
190 // multiple BrandArgumentLists that apply to it.
|
cannam@147
|
191 //
|
cannam@147
|
192 // A BrandArgumentList only represents the arguments that the client of the type specified. Since
|
cannam@147
|
193 // new parameters can be added over time, this list may not cover all defined parameters for the
|
cannam@147
|
194 // type. Missing parameters should be treated as AnyPointer. This class's implementation of
|
cannam@147
|
195 // operator[] already does this for you; out-of-bounds access will safely return AnyPointer.
|
cannam@147
|
196
|
cannam@147
|
197 public:
|
cannam@147
|
198 inline BrandArgumentList(): scopeId(0), size_(0), bindings(nullptr) {}
|
cannam@147
|
199
|
cannam@147
|
200 inline uint size() const { return size_; }
|
cannam@147
|
201 Type operator[](uint index) const;
|
cannam@147
|
202
|
cannam@147
|
203 typedef _::IndexingIterator<const BrandArgumentList, Type> Iterator;
|
cannam@147
|
204 inline Iterator begin() const { return Iterator(this, 0); }
|
cannam@147
|
205 inline Iterator end() const { return Iterator(this, size()); }
|
cannam@147
|
206
|
cannam@147
|
207 private:
|
cannam@147
|
208 uint64_t scopeId;
|
cannam@147
|
209 uint size_;
|
cannam@147
|
210 bool isUnbound;
|
cannam@147
|
211 const _::RawBrandedSchema::Binding* bindings;
|
cannam@147
|
212
|
cannam@147
|
213 inline BrandArgumentList(uint64_t scopeId, bool isUnbound)
|
cannam@147
|
214 : scopeId(scopeId), size_(0), isUnbound(isUnbound), bindings(nullptr) {}
|
cannam@147
|
215 inline BrandArgumentList(uint64_t scopeId, uint size,
|
cannam@147
|
216 const _::RawBrandedSchema::Binding* bindings)
|
cannam@147
|
217 : scopeId(scopeId), size_(size), isUnbound(false), bindings(bindings) {}
|
cannam@147
|
218
|
cannam@147
|
219 friend class Schema;
|
cannam@147
|
220 };
|
cannam@147
|
221
|
cannam@147
|
222 // -------------------------------------------------------------------
|
cannam@147
|
223
|
cannam@147
|
224 class StructSchema: public Schema {
|
cannam@147
|
225 public:
|
cannam@147
|
226 inline StructSchema(): Schema(&_::NULL_STRUCT_SCHEMA.defaultBrand) {}
|
cannam@147
|
227
|
cannam@147
|
228 class Field;
|
cannam@147
|
229 class FieldList;
|
cannam@147
|
230 class FieldSubset;
|
cannam@147
|
231
|
cannam@147
|
232 FieldList getFields() const;
|
cannam@147
|
233 // List top-level fields of this struct. This list will contain top-level groups (including
|
cannam@147
|
234 // named unions) but not the members of those groups. The list does, however, contain the
|
cannam@147
|
235 // members of the unnamed union, if there is one.
|
cannam@147
|
236
|
cannam@147
|
237 FieldSubset getUnionFields() const;
|
cannam@147
|
238 // If the field contains an unnamed union, get a list of fields in the union, ordered by
|
cannam@147
|
239 // ordinal. Since discriminant values are assigned sequentially by ordinal, you may index this
|
cannam@147
|
240 // list by discriminant value.
|
cannam@147
|
241
|
cannam@147
|
242 FieldSubset getNonUnionFields() const;
|
cannam@147
|
243 // Get the fields of this struct which are not in an unnamed union, ordered by ordinal.
|
cannam@147
|
244
|
cannam@147
|
245 kj::Maybe<Field> findFieldByName(kj::StringPtr name) const;
|
cannam@147
|
246 // Find the field with the given name, or return null if there is no such field. If the struct
|
cannam@147
|
247 // contains an unnamed union, then this will find fields of that union in addition to fields
|
cannam@147
|
248 // of the outer struct, since they exist in the same namespace. It will not, however, find
|
cannam@147
|
249 // members of groups (including named unions) -- you must first look up the group itself,
|
cannam@147
|
250 // then dig into its type.
|
cannam@147
|
251
|
cannam@147
|
252 Field getFieldByName(kj::StringPtr name) const;
|
cannam@147
|
253 // Like findFieldByName() but throws an exception on failure.
|
cannam@147
|
254
|
cannam@147
|
255 kj::Maybe<Field> getFieldByDiscriminant(uint16_t discriminant) const;
|
cannam@147
|
256 // Finds the field whose `discriminantValue` is equal to the given value, or returns null if
|
cannam@147
|
257 // there is no such field. (If the schema does not represent a union or a struct containing
|
cannam@147
|
258 // an unnamed union, then this always returns null.)
|
cannam@147
|
259
|
cannam@147
|
260 private:
|
cannam@147
|
261 StructSchema(Schema base): Schema(base) {}
|
cannam@147
|
262 template <typename T> static inline StructSchema fromImpl() {
|
cannam@147
|
263 return StructSchema(Schema(&_::rawBrandedSchema<T>()));
|
cannam@147
|
264 }
|
cannam@147
|
265 friend class Schema;
|
cannam@147
|
266 friend class Type;
|
cannam@147
|
267 };
|
cannam@147
|
268
|
cannam@147
|
269 class StructSchema::Field {
|
cannam@147
|
270 public:
|
cannam@147
|
271 Field() = default;
|
cannam@147
|
272
|
cannam@147
|
273 inline schema::Field::Reader getProto() const { return proto; }
|
cannam@147
|
274 inline StructSchema getContainingStruct() const { return parent; }
|
cannam@147
|
275
|
cannam@147
|
276 inline uint getIndex() const { return index; }
|
cannam@147
|
277 // Get the index of this field within the containing struct or union.
|
cannam@147
|
278
|
cannam@147
|
279 Type getType() const;
|
cannam@147
|
280 // Get the type of this field. Note that this is preferred over getProto().getType() as this
|
cannam@147
|
281 // method will apply generics.
|
cannam@147
|
282
|
cannam@147
|
283 uint32_t getDefaultValueSchemaOffset() const;
|
cannam@147
|
284 // For struct, list, and object fields, returns the offset, in words, within the first segment of
|
cannam@147
|
285 // the struct's schema, where this field's default value pointer is located. The schema is
|
cannam@147
|
286 // always stored as a single-segment unchecked message, which in turn means that the default
|
cannam@147
|
287 // value pointer itself can be treated as the root of an unchecked message -- if you know where
|
cannam@147
|
288 // to find it, which is what this method helps you with.
|
cannam@147
|
289 //
|
cannam@147
|
290 // For blobs, returns the offset of the beginning of the blob's content within the first segment
|
cannam@147
|
291 // of the struct's schema.
|
cannam@147
|
292 //
|
cannam@147
|
293 // This is primarily useful for code generators. The C++ code generator, for example, embeds
|
cannam@147
|
294 // the entire schema as a raw word array within the generated code. Of course, to implement
|
cannam@147
|
295 // field accessors, it needs access to those fields' default values. Embedding separate copies
|
cannam@147
|
296 // of those default values would be redundant since they are already included in the schema, but
|
cannam@147
|
297 // seeking through the schema at runtime to find the default values would be ugly. Instead,
|
cannam@147
|
298 // the code generator can use getDefaultValueSchemaOffset() to find the offset of the default
|
cannam@147
|
299 // value within the schema, and can simply apply that offset at runtime.
|
cannam@147
|
300 //
|
cannam@147
|
301 // If the above does not make sense, you probably don't need this method.
|
cannam@147
|
302
|
cannam@147
|
303 inline bool operator==(const Field& other) const;
|
cannam@147
|
304 inline bool operator!=(const Field& other) const { return !(*this == other); }
|
cannam@147
|
305
|
cannam@147
|
306 private:
|
cannam@147
|
307 StructSchema parent;
|
cannam@147
|
308 uint index;
|
cannam@147
|
309 schema::Field::Reader proto;
|
cannam@147
|
310
|
cannam@147
|
311 inline Field(StructSchema parent, uint index, schema::Field::Reader proto)
|
cannam@147
|
312 : parent(parent), index(index), proto(proto) {}
|
cannam@147
|
313
|
cannam@147
|
314 friend class StructSchema;
|
cannam@147
|
315 };
|
cannam@147
|
316
|
cannam@147
|
317 kj::StringPtr KJ_STRINGIFY(const StructSchema::Field& field);
|
cannam@147
|
318
|
cannam@147
|
319 class StructSchema::FieldList {
|
cannam@147
|
320 public:
|
cannam@147
|
321 FieldList() = default; // empty list
|
cannam@147
|
322
|
cannam@147
|
323 inline uint size() const { return list.size(); }
|
cannam@147
|
324 inline Field operator[](uint index) const { return Field(parent, index, list[index]); }
|
cannam@147
|
325
|
cannam@147
|
326 typedef _::IndexingIterator<const FieldList, Field> Iterator;
|
cannam@147
|
327 inline Iterator begin() const { return Iterator(this, 0); }
|
cannam@147
|
328 inline Iterator end() const { return Iterator(this, size()); }
|
cannam@147
|
329
|
cannam@147
|
330 private:
|
cannam@147
|
331 StructSchema parent;
|
cannam@147
|
332 List<schema::Field>::Reader list;
|
cannam@147
|
333
|
cannam@147
|
334 inline FieldList(StructSchema parent, List<schema::Field>::Reader list)
|
cannam@147
|
335 : parent(parent), list(list) {}
|
cannam@147
|
336
|
cannam@147
|
337 friend class StructSchema;
|
cannam@147
|
338 };
|
cannam@147
|
339
|
cannam@147
|
340 class StructSchema::FieldSubset {
|
cannam@147
|
341 public:
|
cannam@147
|
342 FieldSubset() = default; // empty list
|
cannam@147
|
343
|
cannam@147
|
344 inline uint size() const { return size_; }
|
cannam@147
|
345 inline Field operator[](uint index) const {
|
cannam@147
|
346 return Field(parent, indices[index], list[indices[index]]);
|
cannam@147
|
347 }
|
cannam@147
|
348
|
cannam@147
|
349 typedef _::IndexingIterator<const FieldSubset, Field> Iterator;
|
cannam@147
|
350 inline Iterator begin() const { return Iterator(this, 0); }
|
cannam@147
|
351 inline Iterator end() const { return Iterator(this, size()); }
|
cannam@147
|
352
|
cannam@147
|
353 private:
|
cannam@147
|
354 StructSchema parent;
|
cannam@147
|
355 List<schema::Field>::Reader list;
|
cannam@147
|
356 const uint16_t* indices;
|
cannam@147
|
357 uint size_;
|
cannam@147
|
358
|
cannam@147
|
359 inline FieldSubset(StructSchema parent, List<schema::Field>::Reader list,
|
cannam@147
|
360 const uint16_t* indices, uint size)
|
cannam@147
|
361 : parent(parent), list(list), indices(indices), size_(size) {}
|
cannam@147
|
362
|
cannam@147
|
363 friend class StructSchema;
|
cannam@147
|
364 };
|
cannam@147
|
365
|
cannam@147
|
366 // -------------------------------------------------------------------
|
cannam@147
|
367
|
cannam@147
|
368 class EnumSchema: public Schema {
|
cannam@147
|
369 public:
|
cannam@147
|
370 inline EnumSchema(): Schema(&_::NULL_ENUM_SCHEMA.defaultBrand) {}
|
cannam@147
|
371
|
cannam@147
|
372 class Enumerant;
|
cannam@147
|
373 class EnumerantList;
|
cannam@147
|
374
|
cannam@147
|
375 EnumerantList getEnumerants() const;
|
cannam@147
|
376
|
cannam@147
|
377 kj::Maybe<Enumerant> findEnumerantByName(kj::StringPtr name) const;
|
cannam@147
|
378
|
cannam@147
|
379 Enumerant getEnumerantByName(kj::StringPtr name) const;
|
cannam@147
|
380 // Like findEnumerantByName() but throws an exception on failure.
|
cannam@147
|
381
|
cannam@147
|
382 private:
|
cannam@147
|
383 EnumSchema(Schema base): Schema(base) {}
|
cannam@147
|
384 template <typename T> static inline EnumSchema fromImpl() {
|
cannam@147
|
385 return EnumSchema(Schema(&_::rawBrandedSchema<T>()));
|
cannam@147
|
386 }
|
cannam@147
|
387 friend class Schema;
|
cannam@147
|
388 friend class Type;
|
cannam@147
|
389 };
|
cannam@147
|
390
|
cannam@147
|
391 class EnumSchema::Enumerant {
|
cannam@147
|
392 public:
|
cannam@147
|
393 Enumerant() = default;
|
cannam@147
|
394
|
cannam@147
|
395 inline schema::Enumerant::Reader getProto() const { return proto; }
|
cannam@147
|
396 inline EnumSchema getContainingEnum() const { return parent; }
|
cannam@147
|
397
|
cannam@147
|
398 inline uint16_t getOrdinal() const { return ordinal; }
|
cannam@147
|
399 inline uint getIndex() const { return ordinal; }
|
cannam@147
|
400
|
cannam@147
|
401 inline bool operator==(const Enumerant& other) const;
|
cannam@147
|
402 inline bool operator!=(const Enumerant& other) const { return !(*this == other); }
|
cannam@147
|
403
|
cannam@147
|
404 private:
|
cannam@147
|
405 EnumSchema parent;
|
cannam@147
|
406 uint16_t ordinal;
|
cannam@147
|
407 schema::Enumerant::Reader proto;
|
cannam@147
|
408
|
cannam@147
|
409 inline Enumerant(EnumSchema parent, uint16_t ordinal, schema::Enumerant::Reader proto)
|
cannam@147
|
410 : parent(parent), ordinal(ordinal), proto(proto) {}
|
cannam@147
|
411
|
cannam@147
|
412 friend class EnumSchema;
|
cannam@147
|
413 };
|
cannam@147
|
414
|
cannam@147
|
415 class EnumSchema::EnumerantList {
|
cannam@147
|
416 public:
|
cannam@147
|
417 EnumerantList() = default; // empty list
|
cannam@147
|
418
|
cannam@147
|
419 inline uint size() const { return list.size(); }
|
cannam@147
|
420 inline Enumerant operator[](uint index) const { return Enumerant(parent, index, list[index]); }
|
cannam@147
|
421
|
cannam@147
|
422 typedef _::IndexingIterator<const EnumerantList, Enumerant> Iterator;
|
cannam@147
|
423 inline Iterator begin() const { return Iterator(this, 0); }
|
cannam@147
|
424 inline Iterator end() const { return Iterator(this, size()); }
|
cannam@147
|
425
|
cannam@147
|
426 private:
|
cannam@147
|
427 EnumSchema parent;
|
cannam@147
|
428 List<schema::Enumerant>::Reader list;
|
cannam@147
|
429
|
cannam@147
|
430 inline EnumerantList(EnumSchema parent, List<schema::Enumerant>::Reader list)
|
cannam@147
|
431 : parent(parent), list(list) {}
|
cannam@147
|
432
|
cannam@147
|
433 friend class EnumSchema;
|
cannam@147
|
434 };
|
cannam@147
|
435
|
cannam@147
|
436 // -------------------------------------------------------------------
|
cannam@147
|
437
|
cannam@147
|
438 class InterfaceSchema: public Schema {
|
cannam@147
|
439 public:
|
cannam@147
|
440 inline InterfaceSchema(): Schema(&_::NULL_INTERFACE_SCHEMA.defaultBrand) {}
|
cannam@147
|
441
|
cannam@147
|
442 class Method;
|
cannam@147
|
443 class MethodList;
|
cannam@147
|
444
|
cannam@147
|
445 MethodList getMethods() const;
|
cannam@147
|
446
|
cannam@147
|
447 kj::Maybe<Method> findMethodByName(kj::StringPtr name) const;
|
cannam@147
|
448
|
cannam@147
|
449 Method getMethodByName(kj::StringPtr name) const;
|
cannam@147
|
450 // Like findMethodByName() but throws an exception on failure.
|
cannam@147
|
451
|
cannam@147
|
452 class SuperclassList;
|
cannam@147
|
453
|
cannam@147
|
454 SuperclassList getSuperclasses() const;
|
cannam@147
|
455 // Get the immediate superclasses of this type, after applying generics.
|
cannam@147
|
456
|
cannam@147
|
457 bool extends(InterfaceSchema other) const;
|
cannam@147
|
458 // Returns true if `other` is a superclass of this interface (including if `other == *this`).
|
cannam@147
|
459
|
cannam@147
|
460 kj::Maybe<InterfaceSchema> findSuperclass(uint64_t typeId) const;
|
cannam@147
|
461 // Find the superclass of this interface with the given type ID. Returns null if the interface
|
cannam@147
|
462 // extends no such type.
|
cannam@147
|
463
|
cannam@147
|
464 private:
|
cannam@147
|
465 InterfaceSchema(Schema base): Schema(base) {}
|
cannam@147
|
466 template <typename T> static inline InterfaceSchema fromImpl() {
|
cannam@147
|
467 return InterfaceSchema(Schema(&_::rawBrandedSchema<T>()));
|
cannam@147
|
468 }
|
cannam@147
|
469 friend class Schema;
|
cannam@147
|
470 friend class Type;
|
cannam@147
|
471
|
cannam@147
|
472 kj::Maybe<Method> findMethodByName(kj::StringPtr name, uint& counter) const;
|
cannam@147
|
473 bool extends(InterfaceSchema other, uint& counter) const;
|
cannam@147
|
474 kj::Maybe<InterfaceSchema> findSuperclass(uint64_t typeId, uint& counter) const;
|
cannam@147
|
475 // We protect against malicious schemas with large or cyclic hierarchies by cutting off the
|
cannam@147
|
476 // search when the counter reaches a threshold.
|
cannam@147
|
477 };
|
cannam@147
|
478
|
cannam@147
|
479 class InterfaceSchema::Method {
|
cannam@147
|
480 public:
|
cannam@147
|
481 Method() = default;
|
cannam@147
|
482
|
cannam@147
|
483 inline schema::Method::Reader getProto() const { return proto; }
|
cannam@147
|
484 inline InterfaceSchema getContainingInterface() const { return parent; }
|
cannam@147
|
485
|
cannam@147
|
486 inline uint16_t getOrdinal() const { return ordinal; }
|
cannam@147
|
487 inline uint getIndex() const { return ordinal; }
|
cannam@147
|
488
|
cannam@147
|
489 StructSchema getParamType() const;
|
cannam@147
|
490 StructSchema getResultType() const;
|
cannam@147
|
491 // Get the parameter and result types, including substituting generic parameters.
|
cannam@147
|
492
|
cannam@147
|
493 inline bool operator==(const Method& other) const;
|
cannam@147
|
494 inline bool operator!=(const Method& other) const { return !(*this == other); }
|
cannam@147
|
495
|
cannam@147
|
496 private:
|
cannam@147
|
497 InterfaceSchema parent;
|
cannam@147
|
498 uint16_t ordinal;
|
cannam@147
|
499 schema::Method::Reader proto;
|
cannam@147
|
500
|
cannam@147
|
501 inline Method(InterfaceSchema parent, uint16_t ordinal,
|
cannam@147
|
502 schema::Method::Reader proto)
|
cannam@147
|
503 : parent(parent), ordinal(ordinal), proto(proto) {}
|
cannam@147
|
504
|
cannam@147
|
505 friend class InterfaceSchema;
|
cannam@147
|
506 };
|
cannam@147
|
507
|
cannam@147
|
508 class InterfaceSchema::MethodList {
|
cannam@147
|
509 public:
|
cannam@147
|
510 MethodList() = default; // empty list
|
cannam@147
|
511
|
cannam@147
|
512 inline uint size() const { return list.size(); }
|
cannam@147
|
513 inline Method operator[](uint index) const { return Method(parent, index, list[index]); }
|
cannam@147
|
514
|
cannam@147
|
515 typedef _::IndexingIterator<const MethodList, Method> Iterator;
|
cannam@147
|
516 inline Iterator begin() const { return Iterator(this, 0); }
|
cannam@147
|
517 inline Iterator end() const { return Iterator(this, size()); }
|
cannam@147
|
518
|
cannam@147
|
519 private:
|
cannam@147
|
520 InterfaceSchema parent;
|
cannam@147
|
521 List<schema::Method>::Reader list;
|
cannam@147
|
522
|
cannam@147
|
523 inline MethodList(InterfaceSchema parent, List<schema::Method>::Reader list)
|
cannam@147
|
524 : parent(parent), list(list) {}
|
cannam@147
|
525
|
cannam@147
|
526 friend class InterfaceSchema;
|
cannam@147
|
527 };
|
cannam@147
|
528
|
cannam@147
|
529 class InterfaceSchema::SuperclassList {
|
cannam@147
|
530 public:
|
cannam@147
|
531 SuperclassList() = default; // empty list
|
cannam@147
|
532
|
cannam@147
|
533 inline uint size() const { return list.size(); }
|
cannam@147
|
534 InterfaceSchema operator[](uint index) const;
|
cannam@147
|
535
|
cannam@147
|
536 typedef _::IndexingIterator<const SuperclassList, InterfaceSchema> Iterator;
|
cannam@147
|
537 inline Iterator begin() const { return Iterator(this, 0); }
|
cannam@147
|
538 inline Iterator end() const { return Iterator(this, size()); }
|
cannam@147
|
539
|
cannam@147
|
540 private:
|
cannam@147
|
541 InterfaceSchema parent;
|
cannam@147
|
542 List<schema::Superclass>::Reader list;
|
cannam@147
|
543
|
cannam@147
|
544 inline SuperclassList(InterfaceSchema parent, List<schema::Superclass>::Reader list)
|
cannam@147
|
545 : parent(parent), list(list) {}
|
cannam@147
|
546
|
cannam@147
|
547 friend class InterfaceSchema;
|
cannam@147
|
548 };
|
cannam@147
|
549
|
cannam@147
|
550 // -------------------------------------------------------------------
|
cannam@147
|
551
|
cannam@147
|
552 class ConstSchema: public Schema {
|
cannam@147
|
553 // Represents a constant declaration.
|
cannam@147
|
554 //
|
cannam@147
|
555 // `ConstSchema` can be implicitly cast to DynamicValue to read its value.
|
cannam@147
|
556
|
cannam@147
|
557 public:
|
cannam@147
|
558 inline ConstSchema(): Schema(&_::NULL_CONST_SCHEMA.defaultBrand) {}
|
cannam@147
|
559
|
cannam@147
|
560 template <typename T>
|
cannam@147
|
561 ReaderFor<T> as() const;
|
cannam@147
|
562 // Read the constant's value. This is a convenience method equivalent to casting the ConstSchema
|
cannam@147
|
563 // to a DynamicValue and then calling its `as<T>()` method. For dependency reasons, this method
|
cannam@147
|
564 // is defined in <capnp/dynamic.h>, which you must #include explicitly.
|
cannam@147
|
565
|
cannam@147
|
566 uint32_t getValueSchemaOffset() const;
|
cannam@147
|
567 // Much like StructSchema::Field::getDefaultValueSchemaOffset(), if the constant has pointer
|
cannam@147
|
568 // type, this gets the offset from the beginning of the constant's schema node to a pointer
|
cannam@147
|
569 // representing the constant value.
|
cannam@147
|
570
|
cannam@147
|
571 Type getType() const;
|
cannam@147
|
572
|
cannam@147
|
573 private:
|
cannam@147
|
574 ConstSchema(Schema base): Schema(base) {}
|
cannam@147
|
575 friend class Schema;
|
cannam@147
|
576 };
|
cannam@147
|
577
|
cannam@147
|
578 // -------------------------------------------------------------------
|
cannam@147
|
579
|
cannam@147
|
580 class Type {
|
cannam@147
|
581 public:
|
cannam@147
|
582 struct BrandParameter {
|
cannam@147
|
583 uint64_t scopeId;
|
cannam@147
|
584 uint index;
|
cannam@147
|
585 };
|
cannam@147
|
586 struct ImplicitParameter {
|
cannam@147
|
587 uint index;
|
cannam@147
|
588 };
|
cannam@147
|
589
|
cannam@147
|
590 inline Type();
|
cannam@147
|
591 inline Type(schema::Type::Which primitive);
|
cannam@147
|
592 inline Type(StructSchema schema);
|
cannam@147
|
593 inline Type(EnumSchema schema);
|
cannam@147
|
594 inline Type(InterfaceSchema schema);
|
cannam@147
|
595 inline Type(ListSchema schema);
|
cannam@147
|
596 inline Type(schema::Type::AnyPointer::Unconstrained::Which anyPointerKind);
|
cannam@147
|
597 inline Type(BrandParameter param);
|
cannam@147
|
598 inline Type(ImplicitParameter param);
|
cannam@147
|
599
|
cannam@147
|
600 template <typename T>
|
cannam@147
|
601 inline static Type from();
|
cannam@147
|
602
|
cannam@147
|
603 inline schema::Type::Which which() const;
|
cannam@147
|
604
|
cannam@147
|
605 StructSchema asStruct() const;
|
cannam@147
|
606 EnumSchema asEnum() const;
|
cannam@147
|
607 InterfaceSchema asInterface() const;
|
cannam@147
|
608 ListSchema asList() const;
|
cannam@147
|
609 // Each of these methods may only be called if which() returns the corresponding type.
|
cannam@147
|
610
|
cannam@147
|
611 kj::Maybe<BrandParameter> getBrandParameter() const;
|
cannam@147
|
612 // Only callable if which() returns ANY_POINTER. Returns null if the type is just a regular
|
cannam@147
|
613 // AnyPointer and not a parameter.
|
cannam@147
|
614
|
cannam@147
|
615 kj::Maybe<ImplicitParameter> getImplicitParameter() const;
|
cannam@147
|
616 // Only callable if which() returns ANY_POINTER. Returns null if the type is just a regular
|
cannam@147
|
617 // AnyPointer and not a parameter. "Implicit parameters" refer to type parameters on methods.
|
cannam@147
|
618
|
cannam@147
|
619 inline schema::Type::AnyPointer::Unconstrained::Which whichAnyPointerKind() const;
|
cannam@147
|
620 // Only callable if which() returns ANY_POINTER.
|
cannam@147
|
621
|
cannam@147
|
622 inline bool isVoid() const;
|
cannam@147
|
623 inline bool isBool() const;
|
cannam@147
|
624 inline bool isInt8() const;
|
cannam@147
|
625 inline bool isInt16() const;
|
cannam@147
|
626 inline bool isInt32() const;
|
cannam@147
|
627 inline bool isInt64() const;
|
cannam@147
|
628 inline bool isUInt8() const;
|
cannam@147
|
629 inline bool isUInt16() const;
|
cannam@147
|
630 inline bool isUInt32() const;
|
cannam@147
|
631 inline bool isUInt64() const;
|
cannam@147
|
632 inline bool isFloat32() const;
|
cannam@147
|
633 inline bool isFloat64() const;
|
cannam@147
|
634 inline bool isText() const;
|
cannam@147
|
635 inline bool isData() const;
|
cannam@147
|
636 inline bool isList() const;
|
cannam@147
|
637 inline bool isEnum() const;
|
cannam@147
|
638 inline bool isStruct() const;
|
cannam@147
|
639 inline bool isInterface() const;
|
cannam@147
|
640 inline bool isAnyPointer() const;
|
cannam@147
|
641
|
cannam@147
|
642 bool operator==(const Type& other) const;
|
cannam@147
|
643 inline bool operator!=(const Type& other) const { return !(*this == other); }
|
cannam@147
|
644
|
cannam@147
|
645 size_t hashCode() const;
|
cannam@147
|
646
|
cannam@147
|
647 inline Type wrapInList(uint depth = 1) const;
|
cannam@147
|
648 // Return the Type formed by wrapping this type in List() `depth` times.
|
cannam@147
|
649
|
cannam@147
|
650 inline Type(schema::Type::Which derived, const _::RawBrandedSchema* schema);
|
cannam@147
|
651 // For internal use.
|
cannam@147
|
652
|
cannam@147
|
653 private:
|
cannam@147
|
654 schema::Type::Which baseType; // type not including applications of List()
|
cannam@147
|
655 uint8_t listDepth; // 0 for T, 1 for List(T), 2 for List(List(T)), ...
|
cannam@147
|
656
|
cannam@147
|
657 bool isImplicitParam;
|
cannam@147
|
658 // If true, this refers to an implicit method parameter. baseType must be ANY_POINTER, scopeId
|
cannam@147
|
659 // must be zero, and paramIndex indicates the parameter index.
|
cannam@147
|
660
|
cannam@147
|
661 union {
|
cannam@147
|
662 uint16_t paramIndex;
|
cannam@147
|
663 // If baseType is ANY_POINTER but this Type actually refers to a type parameter, this is the
|
cannam@147
|
664 // index of the parameter among the parameters at its scope, and `scopeId` below is the type ID
|
cannam@147
|
665 // of the scope where the parameter was defined.
|
cannam@147
|
666
|
cannam@147
|
667 schema::Type::AnyPointer::Unconstrained::Which anyPointerKind;
|
cannam@147
|
668 // If scopeId is zero and isImplicitParam is false.
|
cannam@147
|
669 };
|
cannam@147
|
670
|
cannam@147
|
671 union {
|
cannam@147
|
672 const _::RawBrandedSchema* schema; // if type is struct, enum, interface...
|
cannam@147
|
673 uint64_t scopeId; // if type is AnyPointer but it's actually a type parameter...
|
cannam@147
|
674 };
|
cannam@147
|
675
|
cannam@147
|
676 Type(schema::Type::Which baseType, uint8_t listDepth, const _::RawBrandedSchema* schema)
|
cannam@147
|
677 : baseType(baseType), listDepth(listDepth), schema(schema) {
|
cannam@147
|
678 KJ_IREQUIRE(baseType != schema::Type::ANY_POINTER);
|
cannam@147
|
679 }
|
cannam@147
|
680
|
cannam@147
|
681 void requireUsableAs(Type expected) const;
|
cannam@147
|
682
|
cannam@147
|
683 friend class ListSchema; // only for requireUsableAs()
|
cannam@147
|
684 };
|
cannam@147
|
685
|
cannam@147
|
686 // -------------------------------------------------------------------
|
cannam@147
|
687
|
cannam@147
|
688 class ListSchema {
|
cannam@147
|
689 // ListSchema is a little different because list types are not described by schema nodes. So,
|
cannam@147
|
690 // ListSchema doesn't subclass Schema.
|
cannam@147
|
691
|
cannam@147
|
692 public:
|
cannam@147
|
693 ListSchema() = default;
|
cannam@147
|
694
|
cannam@147
|
695 static ListSchema of(schema::Type::Which primitiveType);
|
cannam@147
|
696 static ListSchema of(StructSchema elementType);
|
cannam@147
|
697 static ListSchema of(EnumSchema elementType);
|
cannam@147
|
698 static ListSchema of(InterfaceSchema elementType);
|
cannam@147
|
699 static ListSchema of(ListSchema elementType);
|
cannam@147
|
700 static ListSchema of(Type elementType);
|
cannam@147
|
701 // Construct the schema for a list of the given type.
|
cannam@147
|
702
|
cannam@147
|
703 static ListSchema of(schema::Type::Reader elementType, Schema context)
|
cannam@147
|
704 KJ_DEPRECATED("Does not handle generics correctly.");
|
cannam@147
|
705 // DEPRECATED: This method cannot correctly account for generic type parameter bindings that
|
cannam@147
|
706 // may apply to the input type. Instead of using this method, use a method of the Schema API
|
cannam@147
|
707 // that corresponds to the exact kind of dependency. For example, to get a field type, use
|
cannam@147
|
708 // StructSchema::Field::getType().
|
cannam@147
|
709 //
|
cannam@147
|
710 // Construct from an element type schema. Requires a context which can handle getDependency()
|
cannam@147
|
711 // requests for any type ID found in the schema.
|
cannam@147
|
712
|
cannam@147
|
713 Type getElementType() const;
|
cannam@147
|
714
|
cannam@147
|
715 inline schema::Type::Which whichElementType() const;
|
cannam@147
|
716 // Get the element type's "which()". ListSchema does not actually store a schema::Type::Reader
|
cannam@147
|
717 // describing the element type, but if it did, this would be equivalent to calling
|
cannam@147
|
718 // .getBody().which() on that type.
|
cannam@147
|
719
|
cannam@147
|
720 StructSchema getStructElementType() const;
|
cannam@147
|
721 EnumSchema getEnumElementType() const;
|
cannam@147
|
722 InterfaceSchema getInterfaceElementType() const;
|
cannam@147
|
723 ListSchema getListElementType() const;
|
cannam@147
|
724 // Get the schema for complex element types. Each of these throws an exception if the element
|
cannam@147
|
725 // type is not of the requested kind.
|
cannam@147
|
726
|
cannam@147
|
727 inline bool operator==(const ListSchema& other) const { return elementType == other.elementType; }
|
cannam@147
|
728 inline bool operator!=(const ListSchema& other) const { return elementType != other.elementType; }
|
cannam@147
|
729
|
cannam@147
|
730 template <typename T>
|
cannam@147
|
731 void requireUsableAs() const;
|
cannam@147
|
732
|
cannam@147
|
733 private:
|
cannam@147
|
734 Type elementType;
|
cannam@147
|
735
|
cannam@147
|
736 inline explicit ListSchema(Type elementType): elementType(elementType) {}
|
cannam@147
|
737
|
cannam@147
|
738 template <typename T>
|
cannam@147
|
739 struct FromImpl;
|
cannam@147
|
740 template <typename T> static inline ListSchema fromImpl() {
|
cannam@147
|
741 return FromImpl<T>::get();
|
cannam@147
|
742 }
|
cannam@147
|
743
|
cannam@147
|
744 void requireUsableAs(ListSchema expected) const;
|
cannam@147
|
745
|
cannam@147
|
746 friend class Schema;
|
cannam@147
|
747 };
|
cannam@147
|
748
|
cannam@147
|
749 // =======================================================================================
|
cannam@147
|
750 // inline implementation
|
cannam@147
|
751
|
cannam@147
|
752 template <> inline schema::Type::Which Schema::from<Void>() { return schema::Type::VOID; }
|
cannam@147
|
753 template <> inline schema::Type::Which Schema::from<bool>() { return schema::Type::BOOL; }
|
cannam@147
|
754 template <> inline schema::Type::Which Schema::from<int8_t>() { return schema::Type::INT8; }
|
cannam@147
|
755 template <> inline schema::Type::Which Schema::from<int16_t>() { return schema::Type::INT16; }
|
cannam@147
|
756 template <> inline schema::Type::Which Schema::from<int32_t>() { return schema::Type::INT32; }
|
cannam@147
|
757 template <> inline schema::Type::Which Schema::from<int64_t>() { return schema::Type::INT64; }
|
cannam@147
|
758 template <> inline schema::Type::Which Schema::from<uint8_t>() { return schema::Type::UINT8; }
|
cannam@147
|
759 template <> inline schema::Type::Which Schema::from<uint16_t>() { return schema::Type::UINT16; }
|
cannam@147
|
760 template <> inline schema::Type::Which Schema::from<uint32_t>() { return schema::Type::UINT32; }
|
cannam@147
|
761 template <> inline schema::Type::Which Schema::from<uint64_t>() { return schema::Type::UINT64; }
|
cannam@147
|
762 template <> inline schema::Type::Which Schema::from<float>() { return schema::Type::FLOAT32; }
|
cannam@147
|
763 template <> inline schema::Type::Which Schema::from<double>() { return schema::Type::FLOAT64; }
|
cannam@147
|
764 template <> inline schema::Type::Which Schema::from<Text>() { return schema::Type::TEXT; }
|
cannam@147
|
765 template <> inline schema::Type::Which Schema::from<Data>() { return schema::Type::DATA; }
|
cannam@147
|
766
|
cannam@147
|
767 inline Schema Schema::getDependency(uint64_t id) const {
|
cannam@147
|
768 return getDependency(id, 0);
|
cannam@147
|
769 }
|
cannam@147
|
770
|
cannam@147
|
771 inline bool Schema::isBranded() const {
|
cannam@147
|
772 return raw != &raw->generic->defaultBrand;
|
cannam@147
|
773 }
|
cannam@147
|
774
|
cannam@147
|
775 inline Schema Schema::getGeneric() const {
|
cannam@147
|
776 return Schema(&raw->generic->defaultBrand);
|
cannam@147
|
777 }
|
cannam@147
|
778
|
cannam@147
|
779 template <typename T>
|
cannam@147
|
780 inline void Schema::requireUsableAs() const {
|
cannam@147
|
781 requireUsableAs(&_::rawSchema<T>());
|
cannam@147
|
782 }
|
cannam@147
|
783
|
cannam@147
|
784 inline bool StructSchema::Field::operator==(const Field& other) const {
|
cannam@147
|
785 return parent == other.parent && index == other.index;
|
cannam@147
|
786 }
|
cannam@147
|
787 inline bool EnumSchema::Enumerant::operator==(const Enumerant& other) const {
|
cannam@147
|
788 return parent == other.parent && ordinal == other.ordinal;
|
cannam@147
|
789 }
|
cannam@147
|
790 inline bool InterfaceSchema::Method::operator==(const Method& other) const {
|
cannam@147
|
791 return parent == other.parent && ordinal == other.ordinal;
|
cannam@147
|
792 }
|
cannam@147
|
793
|
cannam@147
|
794 inline ListSchema ListSchema::of(StructSchema elementType) {
|
cannam@147
|
795 return ListSchema(Type(elementType));
|
cannam@147
|
796 }
|
cannam@147
|
797 inline ListSchema ListSchema::of(EnumSchema elementType) {
|
cannam@147
|
798 return ListSchema(Type(elementType));
|
cannam@147
|
799 }
|
cannam@147
|
800 inline ListSchema ListSchema::of(InterfaceSchema elementType) {
|
cannam@147
|
801 return ListSchema(Type(elementType));
|
cannam@147
|
802 }
|
cannam@147
|
803 inline ListSchema ListSchema::of(ListSchema elementType) {
|
cannam@147
|
804 return ListSchema(Type(elementType));
|
cannam@147
|
805 }
|
cannam@147
|
806 inline ListSchema ListSchema::of(Type elementType) {
|
cannam@147
|
807 return ListSchema(elementType);
|
cannam@147
|
808 }
|
cannam@147
|
809
|
cannam@147
|
810 inline Type ListSchema::getElementType() const {
|
cannam@147
|
811 return elementType;
|
cannam@147
|
812 }
|
cannam@147
|
813
|
cannam@147
|
814 inline schema::Type::Which ListSchema::whichElementType() const {
|
cannam@147
|
815 return elementType.which();
|
cannam@147
|
816 }
|
cannam@147
|
817
|
cannam@147
|
818 inline StructSchema ListSchema::getStructElementType() const {
|
cannam@147
|
819 return elementType.asStruct();
|
cannam@147
|
820 }
|
cannam@147
|
821
|
cannam@147
|
822 inline EnumSchema ListSchema::getEnumElementType() const {
|
cannam@147
|
823 return elementType.asEnum();
|
cannam@147
|
824 }
|
cannam@147
|
825
|
cannam@147
|
826 inline InterfaceSchema ListSchema::getInterfaceElementType() const {
|
cannam@147
|
827 return elementType.asInterface();
|
cannam@147
|
828 }
|
cannam@147
|
829
|
cannam@147
|
830 inline ListSchema ListSchema::getListElementType() const {
|
cannam@147
|
831 return elementType.asList();
|
cannam@147
|
832 }
|
cannam@147
|
833
|
cannam@147
|
834 template <typename T>
|
cannam@147
|
835 inline void ListSchema::requireUsableAs() const {
|
cannam@147
|
836 static_assert(kind<T>() == Kind::LIST,
|
cannam@147
|
837 "ListSchema::requireUsableAs<T>() requires T is a list type.");
|
cannam@147
|
838 requireUsableAs(Schema::from<T>());
|
cannam@147
|
839 }
|
cannam@147
|
840
|
cannam@147
|
841 inline void ListSchema::requireUsableAs(ListSchema expected) const {
|
cannam@147
|
842 elementType.requireUsableAs(expected.elementType);
|
cannam@147
|
843 }
|
cannam@147
|
844
|
cannam@147
|
845 template <typename T>
|
cannam@147
|
846 struct ListSchema::FromImpl<List<T>> {
|
cannam@147
|
847 static inline ListSchema get() { return of(Schema::from<T>()); }
|
cannam@147
|
848 };
|
cannam@147
|
849
|
cannam@147
|
850 inline Type::Type(): baseType(schema::Type::VOID), listDepth(0), schema(nullptr) {}
|
cannam@147
|
851 inline Type::Type(schema::Type::Which primitive)
|
cannam@147
|
852 : baseType(primitive), listDepth(0), isImplicitParam(false) {
|
cannam@147
|
853 KJ_IREQUIRE(primitive != schema::Type::STRUCT &&
|
cannam@147
|
854 primitive != schema::Type::ENUM &&
|
cannam@147
|
855 primitive != schema::Type::INTERFACE &&
|
cannam@147
|
856 primitive != schema::Type::LIST);
|
cannam@147
|
857 if (primitive == schema::Type::ANY_POINTER) {
|
cannam@147
|
858 scopeId = 0;
|
cannam@147
|
859 anyPointerKind = schema::Type::AnyPointer::Unconstrained::ANY_KIND;
|
cannam@147
|
860 } else {
|
cannam@147
|
861 schema = nullptr;
|
cannam@147
|
862 }
|
cannam@147
|
863 }
|
cannam@147
|
864 inline Type::Type(schema::Type::Which derived, const _::RawBrandedSchema* schema)
|
cannam@147
|
865 : baseType(derived), listDepth(0), isImplicitParam(false), schema(schema) {
|
cannam@147
|
866 KJ_IREQUIRE(derived == schema::Type::STRUCT ||
|
cannam@147
|
867 derived == schema::Type::ENUM ||
|
cannam@147
|
868 derived == schema::Type::INTERFACE);
|
cannam@147
|
869 }
|
cannam@147
|
870
|
cannam@147
|
871 inline Type::Type(StructSchema schema)
|
cannam@147
|
872 : baseType(schema::Type::STRUCT), listDepth(0), schema(schema.raw) {}
|
cannam@147
|
873 inline Type::Type(EnumSchema schema)
|
cannam@147
|
874 : baseType(schema::Type::ENUM), listDepth(0), schema(schema.raw) {}
|
cannam@147
|
875 inline Type::Type(InterfaceSchema schema)
|
cannam@147
|
876 : baseType(schema::Type::INTERFACE), listDepth(0), schema(schema.raw) {}
|
cannam@147
|
877 inline Type::Type(ListSchema schema)
|
cannam@147
|
878 : Type(schema.getElementType()) { ++listDepth; }
|
cannam@147
|
879 inline Type::Type(schema::Type::AnyPointer::Unconstrained::Which anyPointerKind)
|
cannam@147
|
880 : baseType(schema::Type::ANY_POINTER), listDepth(0), isImplicitParam(false),
|
cannam@147
|
881 anyPointerKind(anyPointerKind), scopeId(0) {}
|
cannam@147
|
882 inline Type::Type(BrandParameter param)
|
cannam@147
|
883 : baseType(schema::Type::ANY_POINTER), listDepth(0), isImplicitParam(false),
|
cannam@147
|
884 paramIndex(param.index), scopeId(param.scopeId) {}
|
cannam@147
|
885 inline Type::Type(ImplicitParameter param)
|
cannam@147
|
886 : baseType(schema::Type::ANY_POINTER), listDepth(0), isImplicitParam(true),
|
cannam@147
|
887 paramIndex(param.index), scopeId(0) {}
|
cannam@147
|
888
|
cannam@147
|
889 inline schema::Type::Which Type::which() const {
|
cannam@147
|
890 return listDepth > 0 ? schema::Type::LIST : baseType;
|
cannam@147
|
891 }
|
cannam@147
|
892
|
cannam@147
|
893 inline schema::Type::AnyPointer::Unconstrained::Which Type::whichAnyPointerKind() const {
|
cannam@147
|
894 KJ_IREQUIRE(baseType == schema::Type::ANY_POINTER);
|
cannam@147
|
895 return !isImplicitParam && scopeId == 0 ? anyPointerKind
|
cannam@147
|
896 : schema::Type::AnyPointer::Unconstrained::ANY_KIND;
|
cannam@147
|
897 }
|
cannam@147
|
898
|
cannam@147
|
899 template <typename T>
|
cannam@147
|
900 inline Type Type::from() { return Type(Schema::from<T>()); }
|
cannam@147
|
901
|
cannam@147
|
902 inline bool Type::isVoid () const { return baseType == schema::Type::VOID && listDepth == 0; }
|
cannam@147
|
903 inline bool Type::isBool () const { return baseType == schema::Type::BOOL && listDepth == 0; }
|
cannam@147
|
904 inline bool Type::isInt8 () const { return baseType == schema::Type::INT8 && listDepth == 0; }
|
cannam@147
|
905 inline bool Type::isInt16 () const { return baseType == schema::Type::INT16 && listDepth == 0; }
|
cannam@147
|
906 inline bool Type::isInt32 () const { return baseType == schema::Type::INT32 && listDepth == 0; }
|
cannam@147
|
907 inline bool Type::isInt64 () const { return baseType == schema::Type::INT64 && listDepth == 0; }
|
cannam@147
|
908 inline bool Type::isUInt8 () const { return baseType == schema::Type::UINT8 && listDepth == 0; }
|
cannam@147
|
909 inline bool Type::isUInt16 () const { return baseType == schema::Type::UINT16 && listDepth == 0; }
|
cannam@147
|
910 inline bool Type::isUInt32 () const { return baseType == schema::Type::UINT32 && listDepth == 0; }
|
cannam@147
|
911 inline bool Type::isUInt64 () const { return baseType == schema::Type::UINT64 && listDepth == 0; }
|
cannam@147
|
912 inline bool Type::isFloat32() const { return baseType == schema::Type::FLOAT32 && listDepth == 0; }
|
cannam@147
|
913 inline bool Type::isFloat64() const { return baseType == schema::Type::FLOAT64 && listDepth == 0; }
|
cannam@147
|
914 inline bool Type::isText () const { return baseType == schema::Type::TEXT && listDepth == 0; }
|
cannam@147
|
915 inline bool Type::isData () const { return baseType == schema::Type::DATA && listDepth == 0; }
|
cannam@147
|
916 inline bool Type::isList () const { return listDepth > 0; }
|
cannam@147
|
917 inline bool Type::isEnum () const { return baseType == schema::Type::ENUM && listDepth == 0; }
|
cannam@147
|
918 inline bool Type::isStruct () const { return baseType == schema::Type::STRUCT && listDepth == 0; }
|
cannam@147
|
919 inline bool Type::isInterface() const {
|
cannam@147
|
920 return baseType == schema::Type::INTERFACE && listDepth == 0;
|
cannam@147
|
921 }
|
cannam@147
|
922 inline bool Type::isAnyPointer() const {
|
cannam@147
|
923 return baseType == schema::Type::ANY_POINTER && listDepth == 0;
|
cannam@147
|
924 }
|
cannam@147
|
925
|
cannam@147
|
926 inline Type Type::wrapInList(uint depth) const {
|
cannam@147
|
927 Type result = *this;
|
cannam@147
|
928 result.listDepth += depth;
|
cannam@147
|
929 return result;
|
cannam@147
|
930 }
|
cannam@147
|
931
|
cannam@147
|
932 } // namespace capnp
|
cannam@147
|
933
|
cannam@147
|
934 #endif // CAPNP_SCHEMA_H_
|