annotate win32-mingw/include/capnp/dynamic.h @ 149:279b18cc7785

Update Win32 capnp builds to v0.6
author Chris Cannam <cannam@all-day-breakfast.com>
date Tue, 23 May 2017 09:16:54 +0100
parents 38d1c0e7850b
children
rev   line source
cannam@149 1 // Copyright (c) 2013-2014 Sandstorm Development Group, Inc. and contributors
cannam@149 2 // Licensed under the MIT License:
cannam@149 3 //
cannam@149 4 // Permission is hereby granted, free of charge, to any person obtaining a copy
cannam@149 5 // of this software and associated documentation files (the "Software"), to deal
cannam@149 6 // in the Software without restriction, including without limitation the rights
cannam@149 7 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
cannam@149 8 // copies of the Software, and to permit persons to whom the Software is
cannam@149 9 // furnished to do so, subject to the following conditions:
cannam@149 10 //
cannam@149 11 // The above copyright notice and this permission notice shall be included in
cannam@149 12 // all copies or substantial portions of the Software.
cannam@149 13 //
cannam@149 14 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
cannam@149 15 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
cannam@149 16 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
cannam@149 17 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
cannam@149 18 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
cannam@149 19 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
cannam@149 20 // THE SOFTWARE.
cannam@149 21
cannam@149 22 // This file defines classes that can be used to manipulate messages based on schemas that are not
cannam@149 23 // known until runtime. This is also useful for writing generic code that uses schemas to handle
cannam@149 24 // arbitrary types in a generic way.
cannam@149 25 //
cannam@149 26 // Each of the classes defined here has a to() template method which converts an instance back to a
cannam@149 27 // native type. This method will throw an exception if the requested type does not match the
cannam@149 28 // schema. To convert native types to dynamic, use DynamicFactory.
cannam@149 29 //
cannam@149 30 // As always, underlying data is validated lazily, so you have to actually traverse the whole
cannam@149 31 // message if you want to validate all content.
cannam@149 32
cannam@149 33 #ifndef CAPNP_DYNAMIC_H_
cannam@149 34 #define CAPNP_DYNAMIC_H_
cannam@149 35
cannam@149 36 #if defined(__GNUC__) && !defined(CAPNP_HEADER_WARNINGS)
cannam@149 37 #pragma GCC system_header
cannam@149 38 #endif
cannam@149 39
cannam@149 40 #include "schema.h"
cannam@149 41 #include "layout.h"
cannam@149 42 #include "message.h"
cannam@149 43 #include "any.h"
cannam@149 44 #include "capability.h"
cannam@149 45
cannam@149 46 namespace capnp {
cannam@149 47
cannam@149 48 class MessageReader;
cannam@149 49 class MessageBuilder;
cannam@149 50
cannam@149 51 struct DynamicValue {
cannam@149 52 DynamicValue() = delete;
cannam@149 53
cannam@149 54 enum Type {
cannam@149 55 UNKNOWN,
cannam@149 56 // Means that the value has unknown type and content because it comes from a newer version of
cannam@149 57 // the schema, or from a newer version of Cap'n Proto that has new features that this version
cannam@149 58 // doesn't understand.
cannam@149 59
cannam@149 60 VOID,
cannam@149 61 BOOL,
cannam@149 62 INT,
cannam@149 63 UINT,
cannam@149 64 FLOAT,
cannam@149 65 TEXT,
cannam@149 66 DATA,
cannam@149 67 LIST,
cannam@149 68 ENUM,
cannam@149 69 STRUCT,
cannam@149 70 CAPABILITY,
cannam@149 71 ANY_POINTER
cannam@149 72 };
cannam@149 73
cannam@149 74 class Reader;
cannam@149 75 class Builder;
cannam@149 76 class Pipeline;
cannam@149 77 };
cannam@149 78 class DynamicEnum;
cannam@149 79 struct DynamicStruct {
cannam@149 80 DynamicStruct() = delete;
cannam@149 81 class Reader;
cannam@149 82 class Builder;
cannam@149 83 class Pipeline;
cannam@149 84 };
cannam@149 85 struct DynamicList {
cannam@149 86 DynamicList() = delete;
cannam@149 87 class Reader;
cannam@149 88 class Builder;
cannam@149 89 };
cannam@149 90 struct DynamicCapability {
cannam@149 91 DynamicCapability() = delete;
cannam@149 92 class Client;
cannam@149 93 class Server;
cannam@149 94 };
cannam@149 95 template <> class Orphan<DynamicValue>;
cannam@149 96
cannam@149 97 template <Kind k> struct DynamicTypeFor_;
cannam@149 98 template <> struct DynamicTypeFor_<Kind::ENUM> { typedef DynamicEnum Type; };
cannam@149 99 template <> struct DynamicTypeFor_<Kind::STRUCT> { typedef DynamicStruct Type; };
cannam@149 100 template <> struct DynamicTypeFor_<Kind::LIST> { typedef DynamicList Type; };
cannam@149 101 template <> struct DynamicTypeFor_<Kind::INTERFACE> { typedef DynamicCapability Type; };
cannam@149 102
cannam@149 103 template <typename T>
cannam@149 104 using DynamicTypeFor = typename DynamicTypeFor_<kind<T>()>::Type;
cannam@149 105
cannam@149 106 template <typename T>
cannam@149 107 ReaderFor<DynamicTypeFor<FromReader<T>>> toDynamic(T&& value);
cannam@149 108 template <typename T>
cannam@149 109 BuilderFor<DynamicTypeFor<FromBuilder<T>>> toDynamic(T&& value);
cannam@149 110 template <typename T>
cannam@149 111 DynamicTypeFor<TypeIfEnum<T>> toDynamic(T&& value);
cannam@149 112 template <typename T>
cannam@149 113 typename DynamicTypeFor<FromServer<T>>::Client toDynamic(kj::Own<T>&& value);
cannam@149 114
cannam@149 115 namespace _ { // private
cannam@149 116
cannam@149 117 template <> struct Kind_<DynamicValue > { static constexpr Kind kind = Kind::OTHER; };
cannam@149 118 template <> struct Kind_<DynamicEnum > { static constexpr Kind kind = Kind::OTHER; };
cannam@149 119 template <> struct Kind_<DynamicStruct > { static constexpr Kind kind = Kind::OTHER; };
cannam@149 120 template <> struct Kind_<DynamicList > { static constexpr Kind kind = Kind::OTHER; };
cannam@149 121 template <> struct Kind_<DynamicCapability> { static constexpr Kind kind = Kind::OTHER; };
cannam@149 122
cannam@149 123 } // namespace _ (private)
cannam@149 124
cannam@149 125 template <> inline constexpr Style style<DynamicValue >() { return Style::POINTER; }
cannam@149 126 template <> inline constexpr Style style<DynamicEnum >() { return Style::PRIMITIVE; }
cannam@149 127 template <> inline constexpr Style style<DynamicStruct >() { return Style::STRUCT; }
cannam@149 128 template <> inline constexpr Style style<DynamicList >() { return Style::POINTER; }
cannam@149 129 template <> inline constexpr Style style<DynamicCapability>() { return Style::CAPABILITY; }
cannam@149 130
cannam@149 131 // -------------------------------------------------------------------
cannam@149 132
cannam@149 133 class DynamicEnum {
cannam@149 134 public:
cannam@149 135 DynamicEnum() = default;
cannam@149 136 inline DynamicEnum(EnumSchema::Enumerant enumerant)
cannam@149 137 : schema(enumerant.getContainingEnum()), value(enumerant.getOrdinal()) {}
cannam@149 138 inline DynamicEnum(EnumSchema schema, uint16_t value)
cannam@149 139 : schema(schema), value(value) {}
cannam@149 140
cannam@149 141 template <typename T, typename = kj::EnableIf<kind<T>() == Kind::ENUM>>
cannam@149 142 inline DynamicEnum(T&& value): DynamicEnum(toDynamic(value)) {}
cannam@149 143
cannam@149 144 template <typename T>
cannam@149 145 inline T as() const { return static_cast<T>(asImpl(typeId<T>())); }
cannam@149 146 // Cast to a native enum type.
cannam@149 147
cannam@149 148 inline EnumSchema getSchema() const { return schema; }
cannam@149 149
cannam@149 150 kj::Maybe<EnumSchema::Enumerant> getEnumerant() const;
cannam@149 151 // Get which enumerant this enum value represents. Returns nullptr if the numeric value does not
cannam@149 152 // correspond to any enumerant in the schema -- this can happen if the data was built using a
cannam@149 153 // newer schema that has more values defined.
cannam@149 154
cannam@149 155 inline uint16_t getRaw() const { return value; }
cannam@149 156 // Returns the raw underlying enum value.
cannam@149 157
cannam@149 158 private:
cannam@149 159 EnumSchema schema;
cannam@149 160 uint16_t value;
cannam@149 161
cannam@149 162 uint16_t asImpl(uint64_t requestedTypeId) const;
cannam@149 163
cannam@149 164 friend struct DynamicStruct;
cannam@149 165 friend struct DynamicList;
cannam@149 166 friend struct DynamicValue;
cannam@149 167 template <typename T>
cannam@149 168 friend DynamicTypeFor<TypeIfEnum<T>> toDynamic(T&& value);
cannam@149 169 };
cannam@149 170
cannam@149 171 // -------------------------------------------------------------------
cannam@149 172
cannam@149 173 class DynamicStruct::Reader {
cannam@149 174 public:
cannam@149 175 typedef DynamicStruct Reads;
cannam@149 176
cannam@149 177 Reader() = default;
cannam@149 178
cannam@149 179 template <typename T, typename = kj::EnableIf<kind<FromReader<T>>() == Kind::STRUCT>>
cannam@149 180 inline Reader(T&& value): Reader(toDynamic(value)) {}
cannam@149 181
cannam@149 182 inline MessageSize totalSize() const { return reader.totalSize().asPublic(); }
cannam@149 183
cannam@149 184 template <typename T>
cannam@149 185 typename T::Reader as() const;
cannam@149 186 // Convert the dynamic struct to its compiled-in type.
cannam@149 187
cannam@149 188 inline StructSchema getSchema() const { return schema; }
cannam@149 189
cannam@149 190 DynamicValue::Reader get(StructSchema::Field field) const;
cannam@149 191 // Read the given field value.
cannam@149 192
cannam@149 193 bool has(StructSchema::Field field) const;
cannam@149 194 // Tests whether the given field is set to its default value. For pointer values, this does
cannam@149 195 // not actually traverse the value comparing it with the default, but simply returns true if the
cannam@149 196 // pointer is non-null. For members of unions, has() returns false if the union member is not
cannam@149 197 // active, but does not necessarily return true if the member is active (depends on the field's
cannam@149 198 // value).
cannam@149 199
cannam@149 200 kj::Maybe<StructSchema::Field> which() const;
cannam@149 201 // If the struct contains an (unnamed) union, and the currently-active field within that union
cannam@149 202 // is known, this returns that field. Otherwise, it returns null. In other words, this returns
cannam@149 203 // null if there is no union present _or_ if the union's discriminant is set to an unrecognized
cannam@149 204 // value. This could happen in particular when receiving a message from a sender who has a
cannam@149 205 // newer version of the protocol and is using a field of the union that you don't know about yet.
cannam@149 206
cannam@149 207 DynamicValue::Reader get(kj::StringPtr name) const;
cannam@149 208 bool has(kj::StringPtr name) const;
cannam@149 209 // Shortcuts to access fields by name. These throw exceptions if no such field exists.
cannam@149 210
cannam@149 211 private:
cannam@149 212 StructSchema schema;
cannam@149 213 _::StructReader reader;
cannam@149 214
cannam@149 215 inline Reader(StructSchema schema, _::StructReader reader)
cannam@149 216 : schema(schema), reader(reader) {}
cannam@149 217 Reader(StructSchema schema, const _::OrphanBuilder& orphan);
cannam@149 218
cannam@149 219 bool isSetInUnion(StructSchema::Field field) const;
cannam@149 220 void verifySetInUnion(StructSchema::Field field) const;
cannam@149 221 static DynamicValue::Reader getImpl(_::StructReader reader, StructSchema::Field field);
cannam@149 222
cannam@149 223 template <typename T, Kind K>
cannam@149 224 friend struct _::PointerHelpers;
cannam@149 225 friend class DynamicStruct::Builder;
cannam@149 226 friend struct DynamicList;
cannam@149 227 friend class MessageReader;
cannam@149 228 friend class MessageBuilder;
cannam@149 229 template <typename T, ::capnp::Kind k>
cannam@149 230 friend struct ::capnp::ToDynamic_;
cannam@149 231 friend kj::StringTree _::structString(
cannam@149 232 _::StructReader reader, const _::RawBrandedSchema& schema);
cannam@149 233 friend class Orphanage;
cannam@149 234 friend class Orphan<DynamicStruct>;
cannam@149 235 friend class Orphan<DynamicValue>;
cannam@149 236 friend class Orphan<AnyPointer>;
cannam@149 237 };
cannam@149 238
cannam@149 239 class DynamicStruct::Builder {
cannam@149 240 public:
cannam@149 241 typedef DynamicStruct Builds;
cannam@149 242
cannam@149 243 Builder() = default;
cannam@149 244 inline Builder(decltype(nullptr)) {}
cannam@149 245
cannam@149 246 template <typename T, typename = kj::EnableIf<kind<FromBuilder<T>>() == Kind::STRUCT>>
cannam@149 247 inline Builder(T&& value): Builder(toDynamic(value)) {}
cannam@149 248
cannam@149 249 inline MessageSize totalSize() const { return asReader().totalSize(); }
cannam@149 250
cannam@149 251 template <typename T>
cannam@149 252 typename T::Builder as();
cannam@149 253 // Cast to a particular struct type.
cannam@149 254
cannam@149 255 inline StructSchema getSchema() const { return schema; }
cannam@149 256
cannam@149 257 DynamicValue::Builder get(StructSchema::Field field);
cannam@149 258 // Read the given field value.
cannam@149 259
cannam@149 260 inline bool has(StructSchema::Field field) { return asReader().has(field); }
cannam@149 261 // Tests whether the given field is set to its default value. For pointer values, this does
cannam@149 262 // not actually traverse the value comparing it with the default, but simply returns true if the
cannam@149 263 // pointer is non-null. For members of unions, has() returns whether the field is currently
cannam@149 264 // active and the union as a whole is non-default -- so, the only time has() will return false
cannam@149 265 // for an active union field is if it is the default active field and it has its default value.
cannam@149 266
cannam@149 267 kj::Maybe<StructSchema::Field> which();
cannam@149 268 // If the struct contains an (unnamed) union, and the currently-active field within that union
cannam@149 269 // is known, this returns that field. Otherwise, it returns null. In other words, this returns
cannam@149 270 // null if there is no union present _or_ if the union's discriminant is set to an unrecognized
cannam@149 271 // value. This could happen in particular when receiving a message from a sender who has a
cannam@149 272 // newer version of the protocol and is using a field of the union that you don't know about yet.
cannam@149 273
cannam@149 274 void set(StructSchema::Field field, const DynamicValue::Reader& value);
cannam@149 275 // Set the given field value.
cannam@149 276
cannam@149 277 DynamicValue::Builder init(StructSchema::Field field);
cannam@149 278 DynamicValue::Builder init(StructSchema::Field field, uint size);
cannam@149 279 // Init a struct, list, or blob field.
cannam@149 280
cannam@149 281 void adopt(StructSchema::Field field, Orphan<DynamicValue>&& orphan);
cannam@149 282 Orphan<DynamicValue> disown(StructSchema::Field field);
cannam@149 283 // Adopt/disown. This works even for non-pointer fields: adopt() becomes equivalent to set()
cannam@149 284 // and disown() becomes like get() followed by clear().
cannam@149 285
cannam@149 286 void clear(StructSchema::Field field);
cannam@149 287 // Clear a field, setting it to its default value. For pointer fields, this actually makes the
cannam@149 288 // field null.
cannam@149 289
cannam@149 290 DynamicValue::Builder get(kj::StringPtr name);
cannam@149 291 bool has(kj::StringPtr name);
cannam@149 292 void set(kj::StringPtr name, const DynamicValue::Reader& value);
cannam@149 293 void set(kj::StringPtr name, std::initializer_list<DynamicValue::Reader> value);
cannam@149 294 DynamicValue::Builder init(kj::StringPtr name);
cannam@149 295 DynamicValue::Builder init(kj::StringPtr name, uint size);
cannam@149 296 void adopt(kj::StringPtr name, Orphan<DynamicValue>&& orphan);
cannam@149 297 Orphan<DynamicValue> disown(kj::StringPtr name);
cannam@149 298 void clear(kj::StringPtr name);
cannam@149 299 // Shortcuts to access fields by name. These throw exceptions if no such field exists.
cannam@149 300
cannam@149 301 Reader asReader() const;
cannam@149 302
cannam@149 303 private:
cannam@149 304 StructSchema schema;
cannam@149 305 _::StructBuilder builder;
cannam@149 306
cannam@149 307 inline Builder(StructSchema schema, _::StructBuilder builder)
cannam@149 308 : schema(schema), builder(builder) {}
cannam@149 309 Builder(StructSchema schema, _::OrphanBuilder& orphan);
cannam@149 310
cannam@149 311 bool isSetInUnion(StructSchema::Field field);
cannam@149 312 void verifySetInUnion(StructSchema::Field field);
cannam@149 313 void setInUnion(StructSchema::Field field);
cannam@149 314
cannam@149 315 template <typename T, Kind k>
cannam@149 316 friend struct _::PointerHelpers;
cannam@149 317 friend struct DynamicList;
cannam@149 318 friend class MessageReader;
cannam@149 319 friend class MessageBuilder;
cannam@149 320 template <typename T, ::capnp::Kind k>
cannam@149 321 friend struct ::capnp::ToDynamic_;
cannam@149 322 friend class Orphanage;
cannam@149 323 friend class Orphan<DynamicStruct>;
cannam@149 324 friend class Orphan<DynamicValue>;
cannam@149 325 friend class Orphan<AnyPointer>;
cannam@149 326 };
cannam@149 327
cannam@149 328 class DynamicStruct::Pipeline {
cannam@149 329 public:
cannam@149 330 typedef DynamicStruct Pipelines;
cannam@149 331
cannam@149 332 inline Pipeline(decltype(nullptr)): typeless(nullptr) {}
cannam@149 333
cannam@149 334 template <typename T>
cannam@149 335 typename T::Pipeline releaseAs();
cannam@149 336 // Convert the dynamic pipeline to its compiled-in type.
cannam@149 337
cannam@149 338 inline StructSchema getSchema() { return schema; }
cannam@149 339
cannam@149 340 DynamicValue::Pipeline get(StructSchema::Field field);
cannam@149 341 // Read the given field value.
cannam@149 342
cannam@149 343 DynamicValue::Pipeline get(kj::StringPtr name);
cannam@149 344 // Get by string name.
cannam@149 345
cannam@149 346 private:
cannam@149 347 StructSchema schema;
cannam@149 348 AnyPointer::Pipeline typeless;
cannam@149 349
cannam@149 350 inline explicit Pipeline(StructSchema schema, AnyPointer::Pipeline&& typeless)
cannam@149 351 : schema(schema), typeless(kj::mv(typeless)) {}
cannam@149 352
cannam@149 353 friend class Request<DynamicStruct, DynamicStruct>;
cannam@149 354 };
cannam@149 355
cannam@149 356 // -------------------------------------------------------------------
cannam@149 357
cannam@149 358 class DynamicList::Reader {
cannam@149 359 public:
cannam@149 360 typedef DynamicList Reads;
cannam@149 361
cannam@149 362 inline Reader(): reader(ElementSize::VOID) {}
cannam@149 363
cannam@149 364 template <typename T, typename = kj::EnableIf<kind<FromReader<T>>() == Kind::LIST>>
cannam@149 365 inline Reader(T&& value): Reader(toDynamic(value)) {}
cannam@149 366
cannam@149 367 template <typename T>
cannam@149 368 typename T::Reader as() const;
cannam@149 369 // Try to convert to any List<T>, Data, or Text. Throws an exception if the underlying data
cannam@149 370 // can't possibly represent the requested type.
cannam@149 371
cannam@149 372 inline ListSchema getSchema() const { return schema; }
cannam@149 373
cannam@149 374 inline uint size() const { return unbound(reader.size() / ELEMENTS); }
cannam@149 375 DynamicValue::Reader operator[](uint index) const;
cannam@149 376
cannam@149 377 typedef _::IndexingIterator<const Reader, DynamicValue::Reader> Iterator;
cannam@149 378 inline Iterator begin() const { return Iterator(this, 0); }
cannam@149 379 inline Iterator end() const { return Iterator(this, size()); }
cannam@149 380
cannam@149 381 private:
cannam@149 382 ListSchema schema;
cannam@149 383 _::ListReader reader;
cannam@149 384
cannam@149 385 Reader(ListSchema schema, _::ListReader reader): schema(schema), reader(reader) {}
cannam@149 386 Reader(ListSchema schema, const _::OrphanBuilder& orphan);
cannam@149 387
cannam@149 388 template <typename T, Kind k>
cannam@149 389 friend struct _::PointerHelpers;
cannam@149 390 friend struct DynamicStruct;
cannam@149 391 friend class DynamicList::Builder;
cannam@149 392 template <typename T, ::capnp::Kind k>
cannam@149 393 friend struct ::capnp::ToDynamic_;
cannam@149 394 friend class Orphanage;
cannam@149 395 friend class Orphan<DynamicList>;
cannam@149 396 friend class Orphan<DynamicValue>;
cannam@149 397 friend class Orphan<AnyPointer>;
cannam@149 398 };
cannam@149 399
cannam@149 400 class DynamicList::Builder {
cannam@149 401 public:
cannam@149 402 typedef DynamicList Builds;
cannam@149 403
cannam@149 404 inline Builder(): builder(ElementSize::VOID) {}
cannam@149 405 inline Builder(decltype(nullptr)): builder(ElementSize::VOID) {}
cannam@149 406
cannam@149 407 template <typename T, typename = kj::EnableIf<kind<FromBuilder<T>>() == Kind::LIST>>
cannam@149 408 inline Builder(T&& value): Builder(toDynamic(value)) {}
cannam@149 409
cannam@149 410 template <typename T>
cannam@149 411 typename T::Builder as();
cannam@149 412 // Try to convert to any List<T>, Data, or Text. Throws an exception if the underlying data
cannam@149 413 // can't possibly represent the requested type.
cannam@149 414
cannam@149 415 inline ListSchema getSchema() const { return schema; }
cannam@149 416
cannam@149 417 inline uint size() const { return unbound(builder.size() / ELEMENTS); }
cannam@149 418 DynamicValue::Builder operator[](uint index);
cannam@149 419 void set(uint index, const DynamicValue::Reader& value);
cannam@149 420 DynamicValue::Builder init(uint index, uint size);
cannam@149 421 void adopt(uint index, Orphan<DynamicValue>&& orphan);
cannam@149 422 Orphan<DynamicValue> disown(uint index);
cannam@149 423
cannam@149 424 typedef _::IndexingIterator<Builder, DynamicStruct::Builder> Iterator;
cannam@149 425 inline Iterator begin() { return Iterator(this, 0); }
cannam@149 426 inline Iterator end() { return Iterator(this, size()); }
cannam@149 427
cannam@149 428 void copyFrom(std::initializer_list<DynamicValue::Reader> value);
cannam@149 429
cannam@149 430 Reader asReader() const;
cannam@149 431
cannam@149 432 private:
cannam@149 433 ListSchema schema;
cannam@149 434 _::ListBuilder builder;
cannam@149 435
cannam@149 436 Builder(ListSchema schema, _::ListBuilder builder): schema(schema), builder(builder) {}
cannam@149 437 Builder(ListSchema schema, _::OrphanBuilder& orphan);
cannam@149 438
cannam@149 439 template <typename T, Kind k>
cannam@149 440 friend struct _::PointerHelpers;
cannam@149 441 friend struct DynamicStruct;
cannam@149 442 template <typename T, ::capnp::Kind k>
cannam@149 443 friend struct ::capnp::ToDynamic_;
cannam@149 444 friend class Orphanage;
cannam@149 445 template <typename T, Kind k>
cannam@149 446 friend struct _::OrphanGetImpl;
cannam@149 447 friend class Orphan<DynamicList>;
cannam@149 448 friend class Orphan<DynamicValue>;
cannam@149 449 friend class Orphan<AnyPointer>;
cannam@149 450 };
cannam@149 451
cannam@149 452 // -------------------------------------------------------------------
cannam@149 453
cannam@149 454 class DynamicCapability::Client: public Capability::Client {
cannam@149 455 public:
cannam@149 456 typedef DynamicCapability Calls;
cannam@149 457 typedef DynamicCapability Reads;
cannam@149 458
cannam@149 459 Client() = default;
cannam@149 460
cannam@149 461 template <typename T, typename = kj::EnableIf<kind<FromClient<T>>() == Kind::INTERFACE>>
cannam@149 462 inline Client(T&& client);
cannam@149 463
cannam@149 464 template <typename T, typename = kj::EnableIf<kj::canConvert<T*, DynamicCapability::Server*>()>>
cannam@149 465 inline Client(kj::Own<T>&& server);
cannam@149 466
cannam@149 467 template <typename T, typename = kj::EnableIf<kind<T>() == Kind::INTERFACE>>
cannam@149 468 typename T::Client as();
cannam@149 469 template <typename T, typename = kj::EnableIf<kind<T>() == Kind::INTERFACE>>
cannam@149 470 typename T::Client releaseAs();
cannam@149 471 // Convert to any client type.
cannam@149 472
cannam@149 473 Client upcast(InterfaceSchema requestedSchema);
cannam@149 474 // Upcast to a superclass. Throws an exception if `schema` is not a superclass.
cannam@149 475
cannam@149 476 inline InterfaceSchema getSchema() { return schema; }
cannam@149 477
cannam@149 478 Request<DynamicStruct, DynamicStruct> newRequest(
cannam@149 479 InterfaceSchema::Method method, kj::Maybe<MessageSize> sizeHint = nullptr);
cannam@149 480 Request<DynamicStruct, DynamicStruct> newRequest(
cannam@149 481 kj::StringPtr methodName, kj::Maybe<MessageSize> sizeHint = nullptr);
cannam@149 482
cannam@149 483 private:
cannam@149 484 InterfaceSchema schema;
cannam@149 485
cannam@149 486 Client(InterfaceSchema schema, kj::Own<ClientHook>&& hook)
cannam@149 487 : Capability::Client(kj::mv(hook)), schema(schema) {}
cannam@149 488
cannam@149 489 template <typename T>
cannam@149 490 inline Client(InterfaceSchema schema, kj::Own<T>&& server);
cannam@149 491
cannam@149 492 friend struct Capability;
cannam@149 493 friend struct DynamicStruct;
cannam@149 494 friend struct DynamicList;
cannam@149 495 friend struct DynamicValue;
cannam@149 496 friend class Orphan<DynamicCapability>;
cannam@149 497 friend class Orphan<DynamicValue>;
cannam@149 498 friend class Orphan<AnyPointer>;
cannam@149 499 template <typename T, Kind k>
cannam@149 500 friend struct _::PointerHelpers;
cannam@149 501 };
cannam@149 502
cannam@149 503 class DynamicCapability::Server: public Capability::Server {
cannam@149 504 public:
cannam@149 505 typedef DynamicCapability Serves;
cannam@149 506
cannam@149 507 Server(InterfaceSchema schema): schema(schema) {}
cannam@149 508
cannam@149 509 virtual kj::Promise<void> call(InterfaceSchema::Method method,
cannam@149 510 CallContext<DynamicStruct, DynamicStruct> context) = 0;
cannam@149 511
cannam@149 512 kj::Promise<void> dispatchCall(uint64_t interfaceId, uint16_t methodId,
cannam@149 513 CallContext<AnyPointer, AnyPointer> context) override final;
cannam@149 514
cannam@149 515 inline InterfaceSchema getSchema() const { return schema; }
cannam@149 516
cannam@149 517 private:
cannam@149 518 InterfaceSchema schema;
cannam@149 519 };
cannam@149 520
cannam@149 521 template <>
cannam@149 522 class Request<DynamicStruct, DynamicStruct>: public DynamicStruct::Builder {
cannam@149 523 // Specialization of `Request<T, U>` for DynamicStruct.
cannam@149 524
cannam@149 525 public:
cannam@149 526 inline Request(DynamicStruct::Builder builder, kj::Own<RequestHook>&& hook,
cannam@149 527 StructSchema resultSchema)
cannam@149 528 : DynamicStruct::Builder(builder), hook(kj::mv(hook)), resultSchema(resultSchema) {}
cannam@149 529
cannam@149 530 RemotePromise<DynamicStruct> send();
cannam@149 531 // Send the call and return a promise for the results.
cannam@149 532
cannam@149 533 private:
cannam@149 534 kj::Own<RequestHook> hook;
cannam@149 535 StructSchema resultSchema;
cannam@149 536
cannam@149 537 friend class Capability::Client;
cannam@149 538 friend struct DynamicCapability;
cannam@149 539 template <typename, typename>
cannam@149 540 friend class CallContext;
cannam@149 541 friend class RequestHook;
cannam@149 542 };
cannam@149 543
cannam@149 544 template <>
cannam@149 545 class CallContext<DynamicStruct, DynamicStruct>: public kj::DisallowConstCopy {
cannam@149 546 // Wrapper around CallContextHook with a specific return type.
cannam@149 547 //
cannam@149 548 // Methods of this class may only be called from within the server's event loop, not from other
cannam@149 549 // threads.
cannam@149 550
cannam@149 551 public:
cannam@149 552 explicit CallContext(CallContextHook& hook, StructSchema paramType, StructSchema resultType);
cannam@149 553
cannam@149 554 DynamicStruct::Reader getParams();
cannam@149 555 void releaseParams();
cannam@149 556 DynamicStruct::Builder getResults(kj::Maybe<MessageSize> sizeHint = nullptr);
cannam@149 557 DynamicStruct::Builder initResults(kj::Maybe<MessageSize> sizeHint = nullptr);
cannam@149 558 void setResults(DynamicStruct::Reader value);
cannam@149 559 void adoptResults(Orphan<DynamicStruct>&& value);
cannam@149 560 Orphanage getResultsOrphanage(kj::Maybe<MessageSize> sizeHint = nullptr);
cannam@149 561 template <typename SubParams>
cannam@149 562 kj::Promise<void> tailCall(Request<SubParams, DynamicStruct>&& tailRequest);
cannam@149 563 void allowCancellation();
cannam@149 564
cannam@149 565 private:
cannam@149 566 CallContextHook* hook;
cannam@149 567 StructSchema paramType;
cannam@149 568 StructSchema resultType;
cannam@149 569
cannam@149 570 friend class DynamicCapability::Server;
cannam@149 571 };
cannam@149 572
cannam@149 573 // -------------------------------------------------------------------
cannam@149 574
cannam@149 575 // Make sure ReaderFor<T> and BuilderFor<T> work for DynamicEnum, DynamicStruct, and
cannam@149 576 // DynamicList, so that we can define DynamicValue::as().
cannam@149 577
cannam@149 578 template <> struct ReaderFor_ <DynamicEnum, Kind::OTHER> { typedef DynamicEnum Type; };
cannam@149 579 template <> struct BuilderFor_<DynamicEnum, Kind::OTHER> { typedef DynamicEnum Type; };
cannam@149 580 template <> struct ReaderFor_ <DynamicStruct, Kind::OTHER> { typedef DynamicStruct::Reader Type; };
cannam@149 581 template <> struct BuilderFor_<DynamicStruct, Kind::OTHER> { typedef DynamicStruct::Builder Type; };
cannam@149 582 template <> struct ReaderFor_ <DynamicList, Kind::OTHER> { typedef DynamicList::Reader Type; };
cannam@149 583 template <> struct BuilderFor_<DynamicList, Kind::OTHER> { typedef DynamicList::Builder Type; };
cannam@149 584 template <> struct ReaderFor_ <DynamicCapability, Kind::OTHER> { typedef DynamicCapability::Client Type; };
cannam@149 585 template <> struct BuilderFor_<DynamicCapability, Kind::OTHER> { typedef DynamicCapability::Client Type; };
cannam@149 586 template <> struct PipelineFor_<DynamicCapability, Kind::OTHER> { typedef DynamicCapability::Client Type; };
cannam@149 587
cannam@149 588 class DynamicValue::Reader {
cannam@149 589 public:
cannam@149 590 typedef DynamicValue Reads;
cannam@149 591
cannam@149 592 inline Reader(decltype(nullptr) n = nullptr); // UNKNOWN
cannam@149 593 inline Reader(Void value);
cannam@149 594 inline Reader(bool value);
cannam@149 595 inline Reader(char value);
cannam@149 596 inline Reader(signed char value);
cannam@149 597 inline Reader(short value);
cannam@149 598 inline Reader(int value);
cannam@149 599 inline Reader(long value);
cannam@149 600 inline Reader(long long value);
cannam@149 601 inline Reader(unsigned char value);
cannam@149 602 inline Reader(unsigned short value);
cannam@149 603 inline Reader(unsigned int value);
cannam@149 604 inline Reader(unsigned long value);
cannam@149 605 inline Reader(unsigned long long value);
cannam@149 606 inline Reader(float value);
cannam@149 607 inline Reader(double value);
cannam@149 608 inline Reader(const char* value); // Text
cannam@149 609 inline Reader(const Text::Reader& value);
cannam@149 610 inline Reader(const Data::Reader& value);
cannam@149 611 inline Reader(const DynamicList::Reader& value);
cannam@149 612 inline Reader(DynamicEnum value);
cannam@149 613 inline Reader(const DynamicStruct::Reader& value);
cannam@149 614 inline Reader(const AnyPointer::Reader& value);
cannam@149 615 inline Reader(DynamicCapability::Client& value);
cannam@149 616 inline Reader(DynamicCapability::Client&& value);
cannam@149 617 template <typename T, typename = kj::EnableIf<kj::canConvert<T*, DynamicCapability::Server*>()>>
cannam@149 618 inline Reader(kj::Own<T>&& value);
cannam@149 619 Reader(ConstSchema constant);
cannam@149 620
cannam@149 621 template <typename T, typename = decltype(toDynamic(kj::instance<T>()))>
cannam@149 622 inline Reader(T&& value): Reader(toDynamic(kj::mv(value))) {}
cannam@149 623
cannam@149 624 Reader(const Reader& other);
cannam@149 625 Reader(Reader&& other) noexcept;
cannam@149 626 ~Reader() noexcept(false);
cannam@149 627 Reader& operator=(const Reader& other);
cannam@149 628 Reader& operator=(Reader&& other);
cannam@149 629 // Unfortunately, we cannot use the implicit definitions of these since DynamicCapability is not
cannam@149 630 // trivially copyable.
cannam@149 631
cannam@149 632 template <typename T>
cannam@149 633 inline ReaderFor<T> as() const { return AsImpl<T>::apply(*this); }
cannam@149 634 // Use to interpret the value as some Cap'n Proto type. Allowed types are:
cannam@149 635 // - Void, bool, [u]int{8,16,32,64}_t, float, double, any enum: Returns the raw value.
cannam@149 636 // - Text, Data, AnyPointer, any struct type: Returns the corresponding Reader.
cannam@149 637 // - List<T> for any T listed above: Returns List<T>::Reader.
cannam@149 638 // - DynamicEnum: Returns the corresponding type.
cannam@149 639 // - DynamicStruct, DynamicList: Returns the corresponding Reader.
cannam@149 640 // - Any capability type, including DynamicCapability: Returns the corresponding Client.
cannam@149 641 // - DynamicValue: Returns an identical Reader. Useful to avoid special-casing in generic code.
cannam@149 642 // (TODO(perf): On GCC 4.8 / Clang 3.3, provide rvalue-qualified version that avoids
cannam@149 643 // refcounting.)
cannam@149 644 //
cannam@149 645 // DynamicValue allows various implicit conversions, mostly just to make the interface friendlier.
cannam@149 646 // - Any integer can be converted to any other integer type so long as the actual value is within
cannam@149 647 // the new type's range.
cannam@149 648 // - Floating-point types can be converted to integers as long as no information would be lost
cannam@149 649 // in the conversion.
cannam@149 650 // - Integers can be converted to floating points. This may lose information, but won't throw.
cannam@149 651 // - Float32/Float64 can be converted between each other. Converting Float64 -> Float32 may lose
cannam@149 652 // information, but won't throw.
cannam@149 653 // - Text can be converted to an enum, if the Text matches one of the enumerant names (but not
cannam@149 654 // vice-versa).
cannam@149 655 // - Capabilities can be upcast (cast to a supertype), but not downcast.
cannam@149 656 //
cannam@149 657 // Any other conversion attempt will throw an exception.
cannam@149 658
cannam@149 659 inline Type getType() const { return type; }
cannam@149 660 // Get the type of this value.
cannam@149 661
cannam@149 662 private:
cannam@149 663 Type type;
cannam@149 664
cannam@149 665 union {
cannam@149 666 Void voidValue;
cannam@149 667 bool boolValue;
cannam@149 668 int64_t intValue;
cannam@149 669 uint64_t uintValue;
cannam@149 670 double floatValue;
cannam@149 671 Text::Reader textValue;
cannam@149 672 Data::Reader dataValue;
cannam@149 673 DynamicList::Reader listValue;
cannam@149 674 DynamicEnum enumValue;
cannam@149 675 DynamicStruct::Reader structValue;
cannam@149 676 AnyPointer::Reader anyPointerValue;
cannam@149 677
cannam@149 678 mutable DynamicCapability::Client capabilityValue;
cannam@149 679 // Declared mutable because `Client`s normally cannot be const.
cannam@149 680
cannam@149 681 // Warning: Copy/move constructors assume all these types are trivially copyable except
cannam@149 682 // Capability.
cannam@149 683 };
cannam@149 684
cannam@149 685 template <typename T, Kind kind = kind<T>()> struct AsImpl;
cannam@149 686 // Implementation backing the as() method. Needs to be a struct to allow partial
cannam@149 687 // specialization. Has a method apply() which does the work.
cannam@149 688
cannam@149 689 friend class Orphanage; // to speed up newOrphanCopy(DynamicValue::Reader)
cannam@149 690 };
cannam@149 691
cannam@149 692 class DynamicValue::Builder {
cannam@149 693 public:
cannam@149 694 typedef DynamicValue Builds;
cannam@149 695
cannam@149 696 inline Builder(decltype(nullptr) n = nullptr); // UNKNOWN
cannam@149 697 inline Builder(Void value);
cannam@149 698 inline Builder(bool value);
cannam@149 699 inline Builder(char value);
cannam@149 700 inline Builder(signed char value);
cannam@149 701 inline Builder(short value);
cannam@149 702 inline Builder(int value);
cannam@149 703 inline Builder(long value);
cannam@149 704 inline Builder(long long value);
cannam@149 705 inline Builder(unsigned char value);
cannam@149 706 inline Builder(unsigned short value);
cannam@149 707 inline Builder(unsigned int value);
cannam@149 708 inline Builder(unsigned long value);
cannam@149 709 inline Builder(unsigned long long value);
cannam@149 710 inline Builder(float value);
cannam@149 711 inline Builder(double value);
cannam@149 712 inline Builder(Text::Builder value);
cannam@149 713 inline Builder(Data::Builder value);
cannam@149 714 inline Builder(DynamicList::Builder value);
cannam@149 715 inline Builder(DynamicEnum value);
cannam@149 716 inline Builder(DynamicStruct::Builder value);
cannam@149 717 inline Builder(AnyPointer::Builder value);
cannam@149 718 inline Builder(DynamicCapability::Client& value);
cannam@149 719 inline Builder(DynamicCapability::Client&& value);
cannam@149 720
cannam@149 721 template <typename T, typename = decltype(toDynamic(kj::instance<T>()))>
cannam@149 722 inline Builder(T value): Builder(toDynamic(value)) {}
cannam@149 723
cannam@149 724 Builder(Builder& other);
cannam@149 725 Builder(Builder&& other) noexcept;
cannam@149 726 ~Builder() noexcept(false);
cannam@149 727 Builder& operator=(Builder& other);
cannam@149 728 Builder& operator=(Builder&& other);
cannam@149 729 // Unfortunately, we cannot use the implicit definitions of these since DynamicCapability is not
cannam@149 730 // trivially copyable.
cannam@149 731
cannam@149 732 template <typename T>
cannam@149 733 inline BuilderFor<T> as() { return AsImpl<T>::apply(*this); }
cannam@149 734 // See DynamicValue::Reader::as().
cannam@149 735
cannam@149 736 inline Type getType() { return type; }
cannam@149 737 // Get the type of this value.
cannam@149 738
cannam@149 739 Reader asReader() const;
cannam@149 740
cannam@149 741 private:
cannam@149 742 Type type;
cannam@149 743
cannam@149 744 union {
cannam@149 745 Void voidValue;
cannam@149 746 bool boolValue;
cannam@149 747 int64_t intValue;
cannam@149 748 uint64_t uintValue;
cannam@149 749 double floatValue;
cannam@149 750 Text::Builder textValue;
cannam@149 751 Data::Builder dataValue;
cannam@149 752 DynamicList::Builder listValue;
cannam@149 753 DynamicEnum enumValue;
cannam@149 754 DynamicStruct::Builder structValue;
cannam@149 755 AnyPointer::Builder anyPointerValue;
cannam@149 756
cannam@149 757 mutable DynamicCapability::Client capabilityValue;
cannam@149 758 // Declared mutable because `Client`s normally cannot be const.
cannam@149 759 };
cannam@149 760
cannam@149 761 template <typename T, Kind kind = kind<T>()> struct AsImpl;
cannam@149 762 // Implementation backing the as() method. Needs to be a struct to allow partial
cannam@149 763 // specialization. Has a method apply() which does the work.
cannam@149 764
cannam@149 765 friend class Orphan<DynamicValue>;
cannam@149 766 };
cannam@149 767
cannam@149 768 class DynamicValue::Pipeline {
cannam@149 769 public:
cannam@149 770 typedef DynamicValue Pipelines;
cannam@149 771
cannam@149 772 inline Pipeline(decltype(nullptr) n = nullptr);
cannam@149 773 inline Pipeline(DynamicStruct::Pipeline&& value);
cannam@149 774 inline Pipeline(DynamicCapability::Client&& value);
cannam@149 775
cannam@149 776 Pipeline(Pipeline&& other) noexcept;
cannam@149 777 Pipeline& operator=(Pipeline&& other);
cannam@149 778 ~Pipeline() noexcept(false);
cannam@149 779
cannam@149 780 template <typename T>
cannam@149 781 inline PipelineFor<T> releaseAs() { return AsImpl<T>::apply(*this); }
cannam@149 782
cannam@149 783 inline Type getType() { return type; }
cannam@149 784 // Get the type of this value.
cannam@149 785
cannam@149 786 private:
cannam@149 787 Type type;
cannam@149 788 union {
cannam@149 789 DynamicStruct::Pipeline structValue;
cannam@149 790 DynamicCapability::Client capabilityValue;
cannam@149 791 };
cannam@149 792
cannam@149 793 template <typename T, Kind kind = kind<T>()> struct AsImpl;
cannam@149 794 // Implementation backing the releaseAs() method. Needs to be a struct to allow partial
cannam@149 795 // specialization. Has a method apply() which does the work.
cannam@149 796 };
cannam@149 797
cannam@149 798 kj::StringTree KJ_STRINGIFY(const DynamicValue::Reader& value);
cannam@149 799 kj::StringTree KJ_STRINGIFY(const DynamicValue::Builder& value);
cannam@149 800 kj::StringTree KJ_STRINGIFY(DynamicEnum value);
cannam@149 801 kj::StringTree KJ_STRINGIFY(const DynamicStruct::Reader& value);
cannam@149 802 kj::StringTree KJ_STRINGIFY(const DynamicStruct::Builder& value);
cannam@149 803 kj::StringTree KJ_STRINGIFY(const DynamicList::Reader& value);
cannam@149 804 kj::StringTree KJ_STRINGIFY(const DynamicList::Builder& value);
cannam@149 805
cannam@149 806 // -------------------------------------------------------------------
cannam@149 807 // Orphan <-> Dynamic glue
cannam@149 808
cannam@149 809 template <>
cannam@149 810 class Orphan<DynamicStruct> {
cannam@149 811 public:
cannam@149 812 Orphan() = default;
cannam@149 813 KJ_DISALLOW_COPY(Orphan);
cannam@149 814 Orphan(Orphan&&) = default;
cannam@149 815 Orphan& operator=(Orphan&&) = default;
cannam@149 816
cannam@149 817 template <typename T, typename = kj::EnableIf<kind<T>() == Kind::STRUCT>>
cannam@149 818 inline Orphan(Orphan<T>&& other): schema(Schema::from<T>()), builder(kj::mv(other.builder)) {}
cannam@149 819
cannam@149 820 DynamicStruct::Builder get();
cannam@149 821 DynamicStruct::Reader getReader() const;
cannam@149 822
cannam@149 823 template <typename T>
cannam@149 824 Orphan<T> releaseAs();
cannam@149 825 // Like DynamicStruct::Builder::as(), but coerces the Orphan type. Since Orphans are move-only,
cannam@149 826 // the original Orphan<DynamicStruct> is no longer valid after this call; ownership is
cannam@149 827 // transferred to the returned Orphan<T>.
cannam@149 828
cannam@149 829 inline bool operator==(decltype(nullptr)) const { return builder == nullptr; }
cannam@149 830 inline bool operator!=(decltype(nullptr)) const { return builder != nullptr; }
cannam@149 831
cannam@149 832 private:
cannam@149 833 StructSchema schema;
cannam@149 834 _::OrphanBuilder builder;
cannam@149 835
cannam@149 836 inline Orphan(StructSchema schema, _::OrphanBuilder&& builder)
cannam@149 837 : schema(schema), builder(kj::mv(builder)) {}
cannam@149 838
cannam@149 839 template <typename, Kind>
cannam@149 840 friend struct _::PointerHelpers;
cannam@149 841 friend struct DynamicList;
cannam@149 842 friend class Orphanage;
cannam@149 843 friend class Orphan<DynamicValue>;
cannam@149 844 friend class Orphan<AnyPointer>;
cannam@149 845 friend class MessageBuilder;
cannam@149 846 };
cannam@149 847
cannam@149 848 template <>
cannam@149 849 class Orphan<DynamicList> {
cannam@149 850 public:
cannam@149 851 Orphan() = default;
cannam@149 852 KJ_DISALLOW_COPY(Orphan);
cannam@149 853 Orphan(Orphan&&) = default;
cannam@149 854 Orphan& operator=(Orphan&&) = default;
cannam@149 855
cannam@149 856 template <typename T, typename = kj::EnableIf<kind<T>() == Kind::LIST>>
cannam@149 857 inline Orphan(Orphan<T>&& other): schema(Schema::from<T>()), builder(kj::mv(other.builder)) {}
cannam@149 858
cannam@149 859 DynamicList::Builder get();
cannam@149 860 DynamicList::Reader getReader() const;
cannam@149 861
cannam@149 862 template <typename T>
cannam@149 863 Orphan<T> releaseAs();
cannam@149 864 // Like DynamicList::Builder::as(), but coerces the Orphan type. Since Orphans are move-only,
cannam@149 865 // the original Orphan<DynamicStruct> is no longer valid after this call; ownership is
cannam@149 866 // transferred to the returned Orphan<T>.
cannam@149 867
cannam@149 868 // TODO(someday): Support truncate().
cannam@149 869
cannam@149 870 inline bool operator==(decltype(nullptr)) const { return builder == nullptr; }
cannam@149 871 inline bool operator!=(decltype(nullptr)) const { return builder != nullptr; }
cannam@149 872
cannam@149 873 private:
cannam@149 874 ListSchema schema;
cannam@149 875 _::OrphanBuilder builder;
cannam@149 876
cannam@149 877 inline Orphan(ListSchema schema, _::OrphanBuilder&& builder)
cannam@149 878 : schema(schema), builder(kj::mv(builder)) {}
cannam@149 879
cannam@149 880 template <typename, Kind>
cannam@149 881 friend struct _::PointerHelpers;
cannam@149 882 friend struct DynamicList;
cannam@149 883 friend class Orphanage;
cannam@149 884 friend class Orphan<DynamicValue>;
cannam@149 885 friend class Orphan<AnyPointer>;
cannam@149 886 };
cannam@149 887
cannam@149 888 template <>
cannam@149 889 class Orphan<DynamicCapability> {
cannam@149 890 public:
cannam@149 891 Orphan() = default;
cannam@149 892 KJ_DISALLOW_COPY(Orphan);
cannam@149 893 Orphan(Orphan&&) = default;
cannam@149 894 Orphan& operator=(Orphan&&) = default;
cannam@149 895
cannam@149 896 template <typename T, typename = kj::EnableIf<kind<T>() == Kind::INTERFACE>>
cannam@149 897 inline Orphan(Orphan<T>&& other): schema(Schema::from<T>()), builder(kj::mv(other.builder)) {}
cannam@149 898
cannam@149 899 DynamicCapability::Client get();
cannam@149 900 DynamicCapability::Client getReader() const;
cannam@149 901
cannam@149 902 template <typename T>
cannam@149 903 Orphan<T> releaseAs();
cannam@149 904 // Like DynamicCapability::Client::as(), but coerces the Orphan type. Since Orphans are move-only,
cannam@149 905 // the original Orphan<DynamicCapability> is no longer valid after this call; ownership is
cannam@149 906 // transferred to the returned Orphan<T>.
cannam@149 907
cannam@149 908 inline bool operator==(decltype(nullptr)) const { return builder == nullptr; }
cannam@149 909 inline bool operator!=(decltype(nullptr)) const { return builder != nullptr; }
cannam@149 910
cannam@149 911 private:
cannam@149 912 InterfaceSchema schema;
cannam@149 913 _::OrphanBuilder builder;
cannam@149 914
cannam@149 915 inline Orphan(InterfaceSchema schema, _::OrphanBuilder&& builder)
cannam@149 916 : schema(schema), builder(kj::mv(builder)) {}
cannam@149 917
cannam@149 918 template <typename, Kind>
cannam@149 919 friend struct _::PointerHelpers;
cannam@149 920 friend struct DynamicList;
cannam@149 921 friend class Orphanage;
cannam@149 922 friend class Orphan<DynamicValue>;
cannam@149 923 friend class Orphan<AnyPointer>;
cannam@149 924 };
cannam@149 925
cannam@149 926 template <>
cannam@149 927 class Orphan<DynamicValue> {
cannam@149 928 public:
cannam@149 929 inline Orphan(decltype(nullptr) n = nullptr): type(DynamicValue::UNKNOWN) {}
cannam@149 930 inline Orphan(Void value);
cannam@149 931 inline Orphan(bool value);
cannam@149 932 inline Orphan(char value);
cannam@149 933 inline Orphan(signed char value);
cannam@149 934 inline Orphan(short value);
cannam@149 935 inline Orphan(int value);
cannam@149 936 inline Orphan(long value);
cannam@149 937 inline Orphan(long long value);
cannam@149 938 inline Orphan(unsigned char value);
cannam@149 939 inline Orphan(unsigned short value);
cannam@149 940 inline Orphan(unsigned int value);
cannam@149 941 inline Orphan(unsigned long value);
cannam@149 942 inline Orphan(unsigned long long value);
cannam@149 943 inline Orphan(float value);
cannam@149 944 inline Orphan(double value);
cannam@149 945 inline Orphan(DynamicEnum value);
cannam@149 946 Orphan(Orphan&&) = default;
cannam@149 947 template <typename T>
cannam@149 948 Orphan(Orphan<T>&&);
cannam@149 949 Orphan(Orphan<AnyPointer>&&);
cannam@149 950 Orphan(void*) = delete; // So Orphan(bool) doesn't accept pointers.
cannam@149 951 KJ_DISALLOW_COPY(Orphan);
cannam@149 952
cannam@149 953 Orphan& operator=(Orphan&&) = default;
cannam@149 954
cannam@149 955 inline DynamicValue::Type getType() { return type; }
cannam@149 956
cannam@149 957 DynamicValue::Builder get();
cannam@149 958 DynamicValue::Reader getReader() const;
cannam@149 959
cannam@149 960 template <typename T>
cannam@149 961 Orphan<T> releaseAs();
cannam@149 962 // Like DynamicValue::Builder::as(), but coerces the Orphan type. Since Orphans are move-only,
cannam@149 963 // the original Orphan<DynamicStruct> is no longer valid after this call; ownership is
cannam@149 964 // transferred to the returned Orphan<T>.
cannam@149 965
cannam@149 966 private:
cannam@149 967 DynamicValue::Type type;
cannam@149 968 union {
cannam@149 969 Void voidValue;
cannam@149 970 bool boolValue;
cannam@149 971 int64_t intValue;
cannam@149 972 uint64_t uintValue;
cannam@149 973 double floatValue;
cannam@149 974 DynamicEnum enumValue;
cannam@149 975 StructSchema structSchema;
cannam@149 976 ListSchema listSchema;
cannam@149 977 InterfaceSchema interfaceSchema;
cannam@149 978 };
cannam@149 979
cannam@149 980 _::OrphanBuilder builder;
cannam@149 981 // Only used if `type` is a pointer type.
cannam@149 982
cannam@149 983 Orphan(DynamicValue::Builder value, _::OrphanBuilder&& builder);
cannam@149 984 Orphan(DynamicValue::Type type, _::OrphanBuilder&& builder)
cannam@149 985 : type(type), builder(kj::mv(builder)) {}
cannam@149 986 Orphan(StructSchema structSchema, _::OrphanBuilder&& builder)
cannam@149 987 : type(DynamicValue::STRUCT), structSchema(structSchema), builder(kj::mv(builder)) {}
cannam@149 988 Orphan(ListSchema listSchema, _::OrphanBuilder&& builder)
cannam@149 989 : type(DynamicValue::LIST), listSchema(listSchema), builder(kj::mv(builder)) {}
cannam@149 990
cannam@149 991 template <typename, Kind>
cannam@149 992 friend struct _::PointerHelpers;
cannam@149 993 friend struct DynamicStruct;
cannam@149 994 friend struct DynamicList;
cannam@149 995 friend struct AnyPointer;
cannam@149 996 friend class Orphanage;
cannam@149 997 };
cannam@149 998
cannam@149 999 template <typename T>
cannam@149 1000 inline Orphan<DynamicValue>::Orphan(Orphan<T>&& other)
cannam@149 1001 : Orphan(other.get(), kj::mv(other.builder)) {}
cannam@149 1002
cannam@149 1003 inline Orphan<DynamicValue>::Orphan(Orphan<AnyPointer>&& other)
cannam@149 1004 : type(DynamicValue::ANY_POINTER), builder(kj::mv(other.builder)) {}
cannam@149 1005
cannam@149 1006 template <typename T>
cannam@149 1007 Orphan<T> Orphan<DynamicStruct>::releaseAs() {
cannam@149 1008 get().as<T>(); // type check
cannam@149 1009 return Orphan<T>(kj::mv(builder));
cannam@149 1010 }
cannam@149 1011
cannam@149 1012 template <typename T>
cannam@149 1013 Orphan<T> Orphan<DynamicList>::releaseAs() {
cannam@149 1014 get().as<T>(); // type check
cannam@149 1015 return Orphan<T>(kj::mv(builder));
cannam@149 1016 }
cannam@149 1017
cannam@149 1018 template <typename T>
cannam@149 1019 Orphan<T> Orphan<DynamicCapability>::releaseAs() {
cannam@149 1020 get().as<T>(); // type check
cannam@149 1021 return Orphan<T>(kj::mv(builder));
cannam@149 1022 }
cannam@149 1023
cannam@149 1024 template <typename T>
cannam@149 1025 Orphan<T> Orphan<DynamicValue>::releaseAs() {
cannam@149 1026 get().as<T>(); // type check
cannam@149 1027 type = DynamicValue::UNKNOWN;
cannam@149 1028 return Orphan<T>(kj::mv(builder));
cannam@149 1029 }
cannam@149 1030
cannam@149 1031 template <>
cannam@149 1032 Orphan<AnyPointer> Orphan<DynamicValue>::releaseAs<AnyPointer>();
cannam@149 1033 template <>
cannam@149 1034 Orphan<DynamicStruct> Orphan<DynamicValue>::releaseAs<DynamicStruct>();
cannam@149 1035 template <>
cannam@149 1036 Orphan<DynamicList> Orphan<DynamicValue>::releaseAs<DynamicList>();
cannam@149 1037 template <>
cannam@149 1038 Orphan<DynamicCapability> Orphan<DynamicValue>::releaseAs<DynamicCapability>();
cannam@149 1039
cannam@149 1040 template <>
cannam@149 1041 struct Orphanage::GetInnerBuilder<DynamicStruct, Kind::OTHER> {
cannam@149 1042 static inline _::StructBuilder apply(DynamicStruct::Builder& t) {
cannam@149 1043 return t.builder;
cannam@149 1044 }
cannam@149 1045 };
cannam@149 1046
cannam@149 1047 template <>
cannam@149 1048 struct Orphanage::GetInnerBuilder<DynamicList, Kind::OTHER> {
cannam@149 1049 static inline _::ListBuilder apply(DynamicList::Builder& t) {
cannam@149 1050 return t.builder;
cannam@149 1051 }
cannam@149 1052 };
cannam@149 1053
cannam@149 1054 template <>
cannam@149 1055 inline Orphan<DynamicStruct> Orphanage::newOrphanCopy<DynamicStruct::Reader>(
cannam@149 1056 DynamicStruct::Reader copyFrom) const {
cannam@149 1057 return Orphan<DynamicStruct>(
cannam@149 1058 copyFrom.getSchema(), _::OrphanBuilder::copy(arena, capTable, copyFrom.reader));
cannam@149 1059 }
cannam@149 1060
cannam@149 1061 template <>
cannam@149 1062 inline Orphan<DynamicList> Orphanage::newOrphanCopy<DynamicList::Reader>(
cannam@149 1063 DynamicList::Reader copyFrom) const {
cannam@149 1064 return Orphan<DynamicList>(copyFrom.getSchema(),
cannam@149 1065 _::OrphanBuilder::copy(arena, capTable, copyFrom.reader));
cannam@149 1066 }
cannam@149 1067
cannam@149 1068 template <>
cannam@149 1069 inline Orphan<DynamicCapability> Orphanage::newOrphanCopy<DynamicCapability::Client>(
cannam@149 1070 DynamicCapability::Client copyFrom) const {
cannam@149 1071 return Orphan<DynamicCapability>(
cannam@149 1072 copyFrom.getSchema(), _::OrphanBuilder::copy(arena, capTable, copyFrom.hook->addRef()));
cannam@149 1073 }
cannam@149 1074
cannam@149 1075 template <>
cannam@149 1076 Orphan<DynamicValue> Orphanage::newOrphanCopy<DynamicValue::Reader>(
cannam@149 1077 DynamicValue::Reader copyFrom) const;
cannam@149 1078
cannam@149 1079 namespace _ { // private
cannam@149 1080
cannam@149 1081 template <>
cannam@149 1082 struct PointerHelpers<DynamicStruct, Kind::OTHER> {
cannam@149 1083 // getDynamic() is used when an AnyPointer's get() accessor is passed arguments, because for
cannam@149 1084 // non-dynamic types PointerHelpers::get() takes a default value as the third argument, and we
cannam@149 1085 // don't want people to accidentally be able to provide their own default value.
cannam@149 1086 static DynamicStruct::Reader getDynamic(PointerReader reader, StructSchema schema);
cannam@149 1087 static DynamicStruct::Builder getDynamic(PointerBuilder builder, StructSchema schema);
cannam@149 1088 static void set(PointerBuilder builder, const DynamicStruct::Reader& value);
cannam@149 1089 static DynamicStruct::Builder init(PointerBuilder builder, StructSchema schema);
cannam@149 1090 static inline void adopt(PointerBuilder builder, Orphan<DynamicStruct>&& value) {
cannam@149 1091 builder.adopt(kj::mv(value.builder));
cannam@149 1092 }
cannam@149 1093 static inline Orphan<DynamicStruct> disown(PointerBuilder builder, StructSchema schema) {
cannam@149 1094 return Orphan<DynamicStruct>(schema, builder.disown());
cannam@149 1095 }
cannam@149 1096 };
cannam@149 1097
cannam@149 1098 template <>
cannam@149 1099 struct PointerHelpers<DynamicList, Kind::OTHER> {
cannam@149 1100 // getDynamic() is used when an AnyPointer's get() accessor is passed arguments, because for
cannam@149 1101 // non-dynamic types PointerHelpers::get() takes a default value as the third argument, and we
cannam@149 1102 // don't want people to accidentally be able to provide their own default value.
cannam@149 1103 static DynamicList::Reader getDynamic(PointerReader reader, ListSchema schema);
cannam@149 1104 static DynamicList::Builder getDynamic(PointerBuilder builder, ListSchema schema);
cannam@149 1105 static void set(PointerBuilder builder, const DynamicList::Reader& value);
cannam@149 1106 static DynamicList::Builder init(PointerBuilder builder, ListSchema schema, uint size);
cannam@149 1107 static inline void adopt(PointerBuilder builder, Orphan<DynamicList>&& value) {
cannam@149 1108 builder.adopt(kj::mv(value.builder));
cannam@149 1109 }
cannam@149 1110 static inline Orphan<DynamicList> disown(PointerBuilder builder, ListSchema schema) {
cannam@149 1111 return Orphan<DynamicList>(schema, builder.disown());
cannam@149 1112 }
cannam@149 1113 };
cannam@149 1114
cannam@149 1115 template <>
cannam@149 1116 struct PointerHelpers<DynamicCapability, Kind::OTHER> {
cannam@149 1117 // getDynamic() is used when an AnyPointer's get() accessor is passed arguments, because for
cannam@149 1118 // non-dynamic types PointerHelpers::get() takes a default value as the third argument, and we
cannam@149 1119 // don't want people to accidentally be able to provide their own default value.
cannam@149 1120 static DynamicCapability::Client getDynamic(PointerReader reader, InterfaceSchema schema);
cannam@149 1121 static DynamicCapability::Client getDynamic(PointerBuilder builder, InterfaceSchema schema);
cannam@149 1122 static void set(PointerBuilder builder, DynamicCapability::Client& value);
cannam@149 1123 static void set(PointerBuilder builder, DynamicCapability::Client&& value);
cannam@149 1124 static inline void adopt(PointerBuilder builder, Orphan<DynamicCapability>&& value) {
cannam@149 1125 builder.adopt(kj::mv(value.builder));
cannam@149 1126 }
cannam@149 1127 static inline Orphan<DynamicCapability> disown(PointerBuilder builder, InterfaceSchema schema) {
cannam@149 1128 return Orphan<DynamicCapability>(schema, builder.disown());
cannam@149 1129 }
cannam@149 1130 };
cannam@149 1131
cannam@149 1132 } // namespace _ (private)
cannam@149 1133
cannam@149 1134 template <typename T>
cannam@149 1135 inline ReaderFor<T> AnyPointer::Reader::getAs(StructSchema schema) const {
cannam@149 1136 return _::PointerHelpers<T>::getDynamic(reader, schema);
cannam@149 1137 }
cannam@149 1138 template <typename T>
cannam@149 1139 inline ReaderFor<T> AnyPointer::Reader::getAs(ListSchema schema) const {
cannam@149 1140 return _::PointerHelpers<T>::getDynamic(reader, schema);
cannam@149 1141 }
cannam@149 1142 template <typename T>
cannam@149 1143 inline ReaderFor<T> AnyPointer::Reader::getAs(InterfaceSchema schema) const {
cannam@149 1144 return _::PointerHelpers<T>::getDynamic(reader, schema);
cannam@149 1145 }
cannam@149 1146 template <typename T>
cannam@149 1147 inline BuilderFor<T> AnyPointer::Builder::getAs(StructSchema schema) {
cannam@149 1148 return _::PointerHelpers<T>::getDynamic(builder, schema);
cannam@149 1149 }
cannam@149 1150 template <typename T>
cannam@149 1151 inline BuilderFor<T> AnyPointer::Builder::getAs(ListSchema schema) {
cannam@149 1152 return _::PointerHelpers<T>::getDynamic(builder, schema);
cannam@149 1153 }
cannam@149 1154 template <typename T>
cannam@149 1155 inline BuilderFor<T> AnyPointer::Builder::getAs(InterfaceSchema schema) {
cannam@149 1156 return _::PointerHelpers<T>::getDynamic(builder, schema);
cannam@149 1157 }
cannam@149 1158 template <typename T>
cannam@149 1159 inline BuilderFor<T> AnyPointer::Builder::initAs(StructSchema schema) {
cannam@149 1160 return _::PointerHelpers<T>::init(builder, schema);
cannam@149 1161 }
cannam@149 1162 template <typename T>
cannam@149 1163 inline BuilderFor<T> AnyPointer::Builder::initAs(ListSchema schema, uint elementCount) {
cannam@149 1164 return _::PointerHelpers<T>::init(builder, schema, elementCount);
cannam@149 1165 }
cannam@149 1166 template <>
cannam@149 1167 inline void AnyPointer::Builder::setAs<DynamicStruct>(DynamicStruct::Reader value) {
cannam@149 1168 return _::PointerHelpers<DynamicStruct>::set(builder, value);
cannam@149 1169 }
cannam@149 1170 template <>
cannam@149 1171 inline void AnyPointer::Builder::setAs<DynamicList>(DynamicList::Reader value) {
cannam@149 1172 return _::PointerHelpers<DynamicList>::set(builder, value);
cannam@149 1173 }
cannam@149 1174 template <>
cannam@149 1175 inline void AnyPointer::Builder::setAs<DynamicCapability>(DynamicCapability::Client value) {
cannam@149 1176 return _::PointerHelpers<DynamicCapability>::set(builder, kj::mv(value));
cannam@149 1177 }
cannam@149 1178 template <>
cannam@149 1179 void AnyPointer::Builder::adopt<DynamicValue>(Orphan<DynamicValue>&& orphan);
cannam@149 1180 template <typename T>
cannam@149 1181 inline Orphan<T> AnyPointer::Builder::disownAs(StructSchema schema) {
cannam@149 1182 return _::PointerHelpers<T>::disown(builder, schema);
cannam@149 1183 }
cannam@149 1184 template <typename T>
cannam@149 1185 inline Orphan<T> AnyPointer::Builder::disownAs(ListSchema schema) {
cannam@149 1186 return _::PointerHelpers<T>::disown(builder, schema);
cannam@149 1187 }
cannam@149 1188 template <typename T>
cannam@149 1189 inline Orphan<T> AnyPointer::Builder::disownAs(InterfaceSchema schema) {
cannam@149 1190 return _::PointerHelpers<T>::disown(builder, schema);
cannam@149 1191 }
cannam@149 1192
cannam@149 1193 // We have to declare the methods below inline because Clang and GCC disagree about how to mangle
cannam@149 1194 // their symbol names.
cannam@149 1195 template <>
cannam@149 1196 inline DynamicStruct::Builder Orphan<AnyPointer>::getAs<DynamicStruct>(StructSchema schema) {
cannam@149 1197 return DynamicStruct::Builder(schema, builder);
cannam@149 1198 }
cannam@149 1199 template <>
cannam@149 1200 inline DynamicStruct::Reader Orphan<AnyPointer>::getAsReader<DynamicStruct>(
cannam@149 1201 StructSchema schema) const {
cannam@149 1202 return DynamicStruct::Reader(schema, builder);
cannam@149 1203 }
cannam@149 1204 template <>
cannam@149 1205 inline Orphan<DynamicStruct> Orphan<AnyPointer>::releaseAs<DynamicStruct>(StructSchema schema) {
cannam@149 1206 return Orphan<DynamicStruct>(schema, kj::mv(builder));
cannam@149 1207 }
cannam@149 1208 template <>
cannam@149 1209 inline DynamicList::Builder Orphan<AnyPointer>::getAs<DynamicList>(ListSchema schema) {
cannam@149 1210 return DynamicList::Builder(schema, builder);
cannam@149 1211 }
cannam@149 1212 template <>
cannam@149 1213 inline DynamicList::Reader Orphan<AnyPointer>::getAsReader<DynamicList>(ListSchema schema) const {
cannam@149 1214 return DynamicList::Reader(schema, builder);
cannam@149 1215 }
cannam@149 1216 template <>
cannam@149 1217 inline Orphan<DynamicList> Orphan<AnyPointer>::releaseAs<DynamicList>(ListSchema schema) {
cannam@149 1218 return Orphan<DynamicList>(schema, kj::mv(builder));
cannam@149 1219 }
cannam@149 1220 template <>
cannam@149 1221 inline DynamicCapability::Client Orphan<AnyPointer>::getAs<DynamicCapability>(
cannam@149 1222 InterfaceSchema schema) {
cannam@149 1223 return DynamicCapability::Client(schema, builder.asCapability());
cannam@149 1224 }
cannam@149 1225 template <>
cannam@149 1226 inline DynamicCapability::Client Orphan<AnyPointer>::getAsReader<DynamicCapability>(
cannam@149 1227 InterfaceSchema schema) const {
cannam@149 1228 return DynamicCapability::Client(schema, builder.asCapability());
cannam@149 1229 }
cannam@149 1230 template <>
cannam@149 1231 inline Orphan<DynamicCapability> Orphan<AnyPointer>::releaseAs<DynamicCapability>(
cannam@149 1232 InterfaceSchema schema) {
cannam@149 1233 return Orphan<DynamicCapability>(schema, kj::mv(builder));
cannam@149 1234 }
cannam@149 1235
cannam@149 1236 // =======================================================================================
cannam@149 1237 // Inline implementation details.
cannam@149 1238
cannam@149 1239 template <typename T>
cannam@149 1240 struct ToDynamic_<T, Kind::STRUCT> {
cannam@149 1241 static inline DynamicStruct::Reader apply(const typename T::Reader& value) {
cannam@149 1242 return DynamicStruct::Reader(Schema::from<T>(), value._reader);
cannam@149 1243 }
cannam@149 1244 static inline DynamicStruct::Builder apply(typename T::Builder& value) {
cannam@149 1245 return DynamicStruct::Builder(Schema::from<T>(), value._builder);
cannam@149 1246 }
cannam@149 1247 };
cannam@149 1248
cannam@149 1249 template <typename T>
cannam@149 1250 struct ToDynamic_<T, Kind::LIST> {
cannam@149 1251 static inline DynamicList::Reader apply(const typename T::Reader& value) {
cannam@149 1252 return DynamicList::Reader(Schema::from<T>(), value.reader);
cannam@149 1253 }
cannam@149 1254 static inline DynamicList::Builder apply(typename T::Builder& value) {
cannam@149 1255 return DynamicList::Builder(Schema::from<T>(), value.builder);
cannam@149 1256 }
cannam@149 1257 };
cannam@149 1258
cannam@149 1259 template <typename T>
cannam@149 1260 struct ToDynamic_<T, Kind::INTERFACE> {
cannam@149 1261 static inline DynamicCapability::Client apply(typename T::Client value) {
cannam@149 1262 return DynamicCapability::Client(kj::mv(value));
cannam@149 1263 }
cannam@149 1264 static inline DynamicCapability::Client apply(typename T::Client&& value) {
cannam@149 1265 return DynamicCapability::Client(kj::mv(value));
cannam@149 1266 }
cannam@149 1267 };
cannam@149 1268
cannam@149 1269 template <typename T>
cannam@149 1270 ReaderFor<DynamicTypeFor<FromReader<T>>> toDynamic(T&& value) {
cannam@149 1271 return ToDynamic_<FromReader<T>>::apply(value);
cannam@149 1272 }
cannam@149 1273 template <typename T>
cannam@149 1274 BuilderFor<DynamicTypeFor<FromBuilder<T>>> toDynamic(T&& value) {
cannam@149 1275 return ToDynamic_<FromBuilder<T>>::apply(value);
cannam@149 1276 }
cannam@149 1277 template <typename T>
cannam@149 1278 DynamicTypeFor<TypeIfEnum<T>> toDynamic(T&& value) {
cannam@149 1279 return DynamicEnum(Schema::from<kj::Decay<T>>(), static_cast<uint16_t>(value));
cannam@149 1280 }
cannam@149 1281 template <typename T>
cannam@149 1282 typename DynamicTypeFor<FromServer<T>>::Client toDynamic(kj::Own<T>&& value) {
cannam@149 1283 return typename FromServer<T>::Client(kj::mv(value));
cannam@149 1284 }
cannam@149 1285
cannam@149 1286 inline DynamicValue::Reader::Reader(std::nullptr_t n): type(UNKNOWN) {}
cannam@149 1287 inline DynamicValue::Builder::Builder(std::nullptr_t n): type(UNKNOWN) {}
cannam@149 1288
cannam@149 1289 #define CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(cppType, typeTag, fieldName) \
cannam@149 1290 inline DynamicValue::Reader::Reader(cppType value) \
cannam@149 1291 : type(typeTag), fieldName##Value(value) {} \
cannam@149 1292 inline DynamicValue::Builder::Builder(cppType value) \
cannam@149 1293 : type(typeTag), fieldName##Value(value) {} \
cannam@149 1294 inline Orphan<DynamicValue>::Orphan(cppType value) \
cannam@149 1295 : type(DynamicValue::typeTag), fieldName##Value(value) {}
cannam@149 1296
cannam@149 1297 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(Void, VOID, void);
cannam@149 1298 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(bool, BOOL, bool);
cannam@149 1299 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(char, INT, int);
cannam@149 1300 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(signed char, INT, int);
cannam@149 1301 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(short, INT, int);
cannam@149 1302 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(int, INT, int);
cannam@149 1303 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(long, INT, int);
cannam@149 1304 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(long long, INT, int);
cannam@149 1305 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(unsigned char, UINT, uint);
cannam@149 1306 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(unsigned short, UINT, uint);
cannam@149 1307 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(unsigned int, UINT, uint);
cannam@149 1308 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(unsigned long, UINT, uint);
cannam@149 1309 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(unsigned long long, UINT, uint);
cannam@149 1310 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(float, FLOAT, float);
cannam@149 1311 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(double, FLOAT, float);
cannam@149 1312 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(DynamicEnum, ENUM, enum);
cannam@149 1313 #undef CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR
cannam@149 1314
cannam@149 1315 #define CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(cppType, typeTag, fieldName) \
cannam@149 1316 inline DynamicValue::Reader::Reader(const cppType::Reader& value) \
cannam@149 1317 : type(typeTag), fieldName##Value(value) {} \
cannam@149 1318 inline DynamicValue::Builder::Builder(cppType::Builder value) \
cannam@149 1319 : type(typeTag), fieldName##Value(value) {}
cannam@149 1320
cannam@149 1321 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(Text, TEXT, text);
cannam@149 1322 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(Data, DATA, data);
cannam@149 1323 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(DynamicList, LIST, list);
cannam@149 1324 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(DynamicStruct, STRUCT, struct);
cannam@149 1325 CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR(AnyPointer, ANY_POINTER, anyPointer);
cannam@149 1326
cannam@149 1327 #undef CAPNP_DECLARE_DYNAMIC_VALUE_CONSTRUCTOR
cannam@149 1328
cannam@149 1329 inline DynamicValue::Reader::Reader(DynamicCapability::Client& value)
cannam@149 1330 : type(CAPABILITY), capabilityValue(value) {}
cannam@149 1331 inline DynamicValue::Reader::Reader(DynamicCapability::Client&& value)
cannam@149 1332 : type(CAPABILITY), capabilityValue(kj::mv(value)) {}
cannam@149 1333 template <typename T, typename>
cannam@149 1334 inline DynamicValue::Reader::Reader(kj::Own<T>&& value)
cannam@149 1335 : type(CAPABILITY), capabilityValue(kj::mv(value)) {}
cannam@149 1336 inline DynamicValue::Builder::Builder(DynamicCapability::Client& value)
cannam@149 1337 : type(CAPABILITY), capabilityValue(value) {}
cannam@149 1338 inline DynamicValue::Builder::Builder(DynamicCapability::Client&& value)
cannam@149 1339 : type(CAPABILITY), capabilityValue(kj::mv(value)) {}
cannam@149 1340
cannam@149 1341 inline DynamicValue::Reader::Reader(const char* value): Reader(Text::Reader(value)) {}
cannam@149 1342
cannam@149 1343 #define CAPNP_DECLARE_TYPE(discrim, typeName) \
cannam@149 1344 template <> \
cannam@149 1345 struct DynamicValue::Reader::AsImpl<typeName> { \
cannam@149 1346 static ReaderFor<typeName> apply(const Reader& reader); \
cannam@149 1347 }; \
cannam@149 1348 template <> \
cannam@149 1349 struct DynamicValue::Builder::AsImpl<typeName> { \
cannam@149 1350 static BuilderFor<typeName> apply(Builder& builder); \
cannam@149 1351 };
cannam@149 1352
cannam@149 1353 //CAPNP_DECLARE_TYPE(VOID, Void)
cannam@149 1354 CAPNP_DECLARE_TYPE(BOOL, bool)
cannam@149 1355 CAPNP_DECLARE_TYPE(INT8, int8_t)
cannam@149 1356 CAPNP_DECLARE_TYPE(INT16, int16_t)
cannam@149 1357 CAPNP_DECLARE_TYPE(INT32, int32_t)
cannam@149 1358 CAPNP_DECLARE_TYPE(INT64, int64_t)
cannam@149 1359 CAPNP_DECLARE_TYPE(UINT8, uint8_t)
cannam@149 1360 CAPNP_DECLARE_TYPE(UINT16, uint16_t)
cannam@149 1361 CAPNP_DECLARE_TYPE(UINT32, uint32_t)
cannam@149 1362 CAPNP_DECLARE_TYPE(UINT64, uint64_t)
cannam@149 1363 CAPNP_DECLARE_TYPE(FLOAT32, float)
cannam@149 1364 CAPNP_DECLARE_TYPE(FLOAT64, double)
cannam@149 1365
cannam@149 1366 CAPNP_DECLARE_TYPE(TEXT, Text)
cannam@149 1367 CAPNP_DECLARE_TYPE(DATA, Data)
cannam@149 1368 CAPNP_DECLARE_TYPE(LIST, DynamicList)
cannam@149 1369 CAPNP_DECLARE_TYPE(STRUCT, DynamicStruct)
cannam@149 1370 CAPNP_DECLARE_TYPE(INTERFACE, DynamicCapability)
cannam@149 1371 CAPNP_DECLARE_TYPE(ENUM, DynamicEnum)
cannam@149 1372 CAPNP_DECLARE_TYPE(ANY_POINTER, AnyPointer)
cannam@149 1373 #undef CAPNP_DECLARE_TYPE
cannam@149 1374
cannam@149 1375 // CAPNP_DECLARE_TYPE(Void) causes gcc 4.7 to segfault. If I do it manually and remove the
cannam@149 1376 // ReaderFor<> and BuilderFor<> wrappers, it works.
cannam@149 1377 template <>
cannam@149 1378 struct DynamicValue::Reader::AsImpl<Void> {
cannam@149 1379 static Void apply(const Reader& reader);
cannam@149 1380 };
cannam@149 1381 template <>
cannam@149 1382 struct DynamicValue::Builder::AsImpl<Void> {
cannam@149 1383 static Void apply(Builder& builder);
cannam@149 1384 };
cannam@149 1385
cannam@149 1386 template <typename T>
cannam@149 1387 struct DynamicValue::Reader::AsImpl<T, Kind::ENUM> {
cannam@149 1388 static T apply(const Reader& reader) {
cannam@149 1389 return reader.as<DynamicEnum>().as<T>();
cannam@149 1390 }
cannam@149 1391 };
cannam@149 1392 template <typename T>
cannam@149 1393 struct DynamicValue::Builder::AsImpl<T, Kind::ENUM> {
cannam@149 1394 static T apply(Builder& builder) {
cannam@149 1395 return builder.as<DynamicEnum>().as<T>();
cannam@149 1396 }
cannam@149 1397 };
cannam@149 1398
cannam@149 1399 template <typename T>
cannam@149 1400 struct DynamicValue::Reader::AsImpl<T, Kind::STRUCT> {
cannam@149 1401 static typename T::Reader apply(const Reader& reader) {
cannam@149 1402 return reader.as<DynamicStruct>().as<T>();
cannam@149 1403 }
cannam@149 1404 };
cannam@149 1405 template <typename T>
cannam@149 1406 struct DynamicValue::Builder::AsImpl<T, Kind::STRUCT> {
cannam@149 1407 static typename T::Builder apply(Builder& builder) {
cannam@149 1408 return builder.as<DynamicStruct>().as<T>();
cannam@149 1409 }
cannam@149 1410 };
cannam@149 1411
cannam@149 1412 template <typename T>
cannam@149 1413 struct DynamicValue::Reader::AsImpl<T, Kind::LIST> {
cannam@149 1414 static typename T::Reader apply(const Reader& reader) {
cannam@149 1415 return reader.as<DynamicList>().as<T>();
cannam@149 1416 }
cannam@149 1417 };
cannam@149 1418 template <typename T>
cannam@149 1419 struct DynamicValue::Builder::AsImpl<T, Kind::LIST> {
cannam@149 1420 static typename T::Builder apply(Builder& builder) {
cannam@149 1421 return builder.as<DynamicList>().as<T>();
cannam@149 1422 }
cannam@149 1423 };
cannam@149 1424
cannam@149 1425 template <typename T>
cannam@149 1426 struct DynamicValue::Reader::AsImpl<T, Kind::INTERFACE> {
cannam@149 1427 static typename T::Client apply(const Reader& reader) {
cannam@149 1428 return reader.as<DynamicCapability>().as<T>();
cannam@149 1429 }
cannam@149 1430 };
cannam@149 1431 template <typename T>
cannam@149 1432 struct DynamicValue::Builder::AsImpl<T, Kind::INTERFACE> {
cannam@149 1433 static typename T::Client apply(Builder& builder) {
cannam@149 1434 return builder.as<DynamicCapability>().as<T>();
cannam@149 1435 }
cannam@149 1436 };
cannam@149 1437
cannam@149 1438 template <>
cannam@149 1439 struct DynamicValue::Reader::AsImpl<DynamicValue> {
cannam@149 1440 static DynamicValue::Reader apply(const Reader& reader) {
cannam@149 1441 return reader;
cannam@149 1442 }
cannam@149 1443 };
cannam@149 1444 template <>
cannam@149 1445 struct DynamicValue::Builder::AsImpl<DynamicValue> {
cannam@149 1446 static DynamicValue::Builder apply(Builder& builder) {
cannam@149 1447 return builder;
cannam@149 1448 }
cannam@149 1449 };
cannam@149 1450
cannam@149 1451 inline DynamicValue::Pipeline::Pipeline(std::nullptr_t n): type(UNKNOWN) {}
cannam@149 1452 inline DynamicValue::Pipeline::Pipeline(DynamicStruct::Pipeline&& value)
cannam@149 1453 : type(STRUCT), structValue(kj::mv(value)) {}
cannam@149 1454 inline DynamicValue::Pipeline::Pipeline(DynamicCapability::Client&& value)
cannam@149 1455 : type(CAPABILITY), capabilityValue(kj::mv(value)) {}
cannam@149 1456
cannam@149 1457 template <typename T>
cannam@149 1458 struct DynamicValue::Pipeline::AsImpl<T, Kind::STRUCT> {
cannam@149 1459 static typename T::Pipeline apply(Pipeline& pipeline) {
cannam@149 1460 return pipeline.releaseAs<DynamicStruct>().releaseAs<T>();
cannam@149 1461 }
cannam@149 1462 };
cannam@149 1463 template <typename T>
cannam@149 1464 struct DynamicValue::Pipeline::AsImpl<T, Kind::INTERFACE> {
cannam@149 1465 static typename T::Client apply(Pipeline& pipeline) {
cannam@149 1466 return pipeline.releaseAs<DynamicCapability>().releaseAs<T>();
cannam@149 1467 }
cannam@149 1468 };
cannam@149 1469 template <>
cannam@149 1470 struct DynamicValue::Pipeline::AsImpl<DynamicStruct, Kind::OTHER> {
cannam@149 1471 static PipelineFor<DynamicStruct> apply(Pipeline& pipeline);
cannam@149 1472 };
cannam@149 1473 template <>
cannam@149 1474 struct DynamicValue::Pipeline::AsImpl<DynamicCapability, Kind::OTHER> {
cannam@149 1475 static PipelineFor<DynamicCapability> apply(Pipeline& pipeline);
cannam@149 1476 };
cannam@149 1477
cannam@149 1478 // -------------------------------------------------------------------
cannam@149 1479
cannam@149 1480 template <typename T>
cannam@149 1481 typename T::Reader DynamicStruct::Reader::as() const {
cannam@149 1482 static_assert(kind<T>() == Kind::STRUCT,
cannam@149 1483 "DynamicStruct::Reader::as<T>() can only convert to struct types.");
cannam@149 1484 schema.requireUsableAs<T>();
cannam@149 1485 return typename T::Reader(reader);
cannam@149 1486 }
cannam@149 1487
cannam@149 1488 template <typename T>
cannam@149 1489 typename T::Builder DynamicStruct::Builder::as() {
cannam@149 1490 static_assert(kind<T>() == Kind::STRUCT,
cannam@149 1491 "DynamicStruct::Builder::as<T>() can only convert to struct types.");
cannam@149 1492 schema.requireUsableAs<T>();
cannam@149 1493 return typename T::Builder(builder);
cannam@149 1494 }
cannam@149 1495
cannam@149 1496 template <>
cannam@149 1497 inline DynamicStruct::Reader DynamicStruct::Reader::as<DynamicStruct>() const {
cannam@149 1498 return *this;
cannam@149 1499 }
cannam@149 1500 template <>
cannam@149 1501 inline DynamicStruct::Builder DynamicStruct::Builder::as<DynamicStruct>() {
cannam@149 1502 return *this;
cannam@149 1503 }
cannam@149 1504
cannam@149 1505 inline DynamicStruct::Reader DynamicStruct::Builder::asReader() const {
cannam@149 1506 return DynamicStruct::Reader(schema, builder.asReader());
cannam@149 1507 }
cannam@149 1508
cannam@149 1509 template <>
cannam@149 1510 inline AnyStruct::Reader DynamicStruct::Reader::as<AnyStruct>() const {
cannam@149 1511 return AnyStruct::Reader(reader);
cannam@149 1512 }
cannam@149 1513
cannam@149 1514 template <>
cannam@149 1515 inline AnyStruct::Builder DynamicStruct::Builder::as<AnyStruct>() {
cannam@149 1516 return AnyStruct::Builder(builder);
cannam@149 1517 }
cannam@149 1518
cannam@149 1519 template <typename T>
cannam@149 1520 typename T::Pipeline DynamicStruct::Pipeline::releaseAs() {
cannam@149 1521 static_assert(kind<T>() == Kind::STRUCT,
cannam@149 1522 "DynamicStruct::Pipeline::releaseAs<T>() can only convert to struct types.");
cannam@149 1523 schema.requireUsableAs<T>();
cannam@149 1524 return typename T::Pipeline(kj::mv(typeless));
cannam@149 1525 }
cannam@149 1526
cannam@149 1527 // -------------------------------------------------------------------
cannam@149 1528
cannam@149 1529 template <typename T>
cannam@149 1530 typename T::Reader DynamicList::Reader::as() const {
cannam@149 1531 static_assert(kind<T>() == Kind::LIST,
cannam@149 1532 "DynamicStruct::Reader::as<T>() can only convert to list types.");
cannam@149 1533 schema.requireUsableAs<T>();
cannam@149 1534 return typename T::Reader(reader);
cannam@149 1535 }
cannam@149 1536 template <typename T>
cannam@149 1537 typename T::Builder DynamicList::Builder::as() {
cannam@149 1538 static_assert(kind<T>() == Kind::LIST,
cannam@149 1539 "DynamicStruct::Builder::as<T>() can only convert to list types.");
cannam@149 1540 schema.requireUsableAs<T>();
cannam@149 1541 return typename T::Builder(builder);
cannam@149 1542 }
cannam@149 1543
cannam@149 1544 template <>
cannam@149 1545 inline DynamicList::Reader DynamicList::Reader::as<DynamicList>() const {
cannam@149 1546 return *this;
cannam@149 1547 }
cannam@149 1548 template <>
cannam@149 1549 inline DynamicList::Builder DynamicList::Builder::as<DynamicList>() {
cannam@149 1550 return *this;
cannam@149 1551 }
cannam@149 1552
cannam@149 1553 template <>
cannam@149 1554 inline AnyList::Reader DynamicList::Reader::as<AnyList>() const {
cannam@149 1555 return AnyList::Reader(reader);
cannam@149 1556 }
cannam@149 1557
cannam@149 1558 template <>
cannam@149 1559 inline AnyList::Builder DynamicList::Builder::as<AnyList>() {
cannam@149 1560 return AnyList::Builder(builder);
cannam@149 1561 }
cannam@149 1562
cannam@149 1563 // -------------------------------------------------------------------
cannam@149 1564
cannam@149 1565 template <typename T, typename>
cannam@149 1566 inline DynamicCapability::Client::Client(T&& client)
cannam@149 1567 : Capability::Client(kj::mv(client)), schema(Schema::from<FromClient<T>>()) {}
cannam@149 1568
cannam@149 1569 template <typename T, typename>
cannam@149 1570 inline DynamicCapability::Client::Client(kj::Own<T>&& server)
cannam@149 1571 : Client(server->getSchema(), kj::mv(server)) {}
cannam@149 1572 template <typename T>
cannam@149 1573 inline DynamicCapability::Client::Client(InterfaceSchema schema, kj::Own<T>&& server)
cannam@149 1574 : Capability::Client(kj::mv(server)), schema(schema) {}
cannam@149 1575
cannam@149 1576 template <typename T, typename>
cannam@149 1577 typename T::Client DynamicCapability::Client::as() {
cannam@149 1578 static_assert(kind<T>() == Kind::INTERFACE,
cannam@149 1579 "DynamicCapability::Client::as<T>() can only convert to interface types.");
cannam@149 1580 schema.requireUsableAs<T>();
cannam@149 1581 return typename T::Client(hook->addRef());
cannam@149 1582 }
cannam@149 1583
cannam@149 1584 template <typename T, typename>
cannam@149 1585 typename T::Client DynamicCapability::Client::releaseAs() {
cannam@149 1586 static_assert(kind<T>() == Kind::INTERFACE,
cannam@149 1587 "DynamicCapability::Client::as<T>() can only convert to interface types.");
cannam@149 1588 schema.requireUsableAs<T>();
cannam@149 1589 return typename T::Client(kj::mv(hook));
cannam@149 1590 }
cannam@149 1591
cannam@149 1592 inline CallContext<DynamicStruct, DynamicStruct>::CallContext(
cannam@149 1593 CallContextHook& hook, StructSchema paramType, StructSchema resultType)
cannam@149 1594 : hook(&hook), paramType(paramType), resultType(resultType) {}
cannam@149 1595 inline DynamicStruct::Reader CallContext<DynamicStruct, DynamicStruct>::getParams() {
cannam@149 1596 return hook->getParams().getAs<DynamicStruct>(paramType);
cannam@149 1597 }
cannam@149 1598 inline void CallContext<DynamicStruct, DynamicStruct>::releaseParams() {
cannam@149 1599 hook->releaseParams();
cannam@149 1600 }
cannam@149 1601 inline DynamicStruct::Builder CallContext<DynamicStruct, DynamicStruct>::getResults(
cannam@149 1602 kj::Maybe<MessageSize> sizeHint) {
cannam@149 1603 return hook->getResults(sizeHint).getAs<DynamicStruct>(resultType);
cannam@149 1604 }
cannam@149 1605 inline DynamicStruct::Builder CallContext<DynamicStruct, DynamicStruct>::initResults(
cannam@149 1606 kj::Maybe<MessageSize> sizeHint) {
cannam@149 1607 return hook->getResults(sizeHint).initAs<DynamicStruct>(resultType);
cannam@149 1608 }
cannam@149 1609 inline void CallContext<DynamicStruct, DynamicStruct>::setResults(DynamicStruct::Reader value) {
cannam@149 1610 hook->getResults(value.totalSize()).setAs<DynamicStruct>(value);
cannam@149 1611 }
cannam@149 1612 inline void CallContext<DynamicStruct, DynamicStruct>::adoptResults(Orphan<DynamicStruct>&& value) {
cannam@149 1613 hook->getResults(MessageSize { 0, 0 }).adopt(kj::mv(value));
cannam@149 1614 }
cannam@149 1615 inline Orphanage CallContext<DynamicStruct, DynamicStruct>::getResultsOrphanage(
cannam@149 1616 kj::Maybe<MessageSize> sizeHint) {
cannam@149 1617 return Orphanage::getForMessageContaining(hook->getResults(sizeHint));
cannam@149 1618 }
cannam@149 1619 template <typename SubParams>
cannam@149 1620 inline kj::Promise<void> CallContext<DynamicStruct, DynamicStruct>::tailCall(
cannam@149 1621 Request<SubParams, DynamicStruct>&& tailRequest) {
cannam@149 1622 return hook->tailCall(kj::mv(tailRequest.hook));
cannam@149 1623 }
cannam@149 1624 inline void CallContext<DynamicStruct, DynamicStruct>::allowCancellation() {
cannam@149 1625 hook->allowCancellation();
cannam@149 1626 }
cannam@149 1627
cannam@149 1628 template <>
cannam@149 1629 inline DynamicCapability::Client Capability::Client::castAs<DynamicCapability>(
cannam@149 1630 InterfaceSchema schema) {
cannam@149 1631 return DynamicCapability::Client(schema, hook->addRef());
cannam@149 1632 }
cannam@149 1633
cannam@149 1634 // -------------------------------------------------------------------
cannam@149 1635
cannam@149 1636 template <typename T>
cannam@149 1637 ReaderFor<T> ConstSchema::as() const {
cannam@149 1638 return DynamicValue::Reader(*this).as<T>();
cannam@149 1639 }
cannam@149 1640
cannam@149 1641 } // namespace capnp
cannam@149 1642
cannam@149 1643 #endif // CAPNP_DYNAMIC_H_