/* * Copyright 2017 WebAssembly Community Group participants * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef wasm_wasm_type_h #define wasm_wasm_type_h #include #include #include #include #include #include #include #include "support/index.h" #include "support/name.h" #include "support/parent_index_iterator.h" #include "wasm-features.h" // TODO: At various code locations we were assuming that single types are basic // types, but this is going to change with the introduction of the compound // Signature, Struct and Array types that will be single but not basic. To // prepare for this change, the following macro marks affected code locations. #define TODO_SINGLE_COMPOUND(type) \ assert(!type.isTuple() && "Unexpected tuple type"); \ assert(type.isBasic() && "TODO: handle compound types"); namespace wasm { // Dangerous! Frees all types and heap types that have ever been created and // resets the type system's internal state. This is only really meant to be used // for tests. void destroyAllTypesForTestingPurposesOnly(); // The types defined in this file. All of them are small and typically passed by // value except for `Tuple` and `Struct`, which may own an unbounded amount of // data. class Type; class HeapType; class RecGroup; struct Signature; struct Continuation; struct Field; struct Struct; struct Array; using TypeList = std::vector; using Tuple = TypeList; enum Nullability { NonNullable, Nullable }; enum Mutability { Immutable, Mutable }; // HeapType name information used for printing. struct TypeNames { // The name of the type. Name name; // For a Struct, names of fields. std::unordered_map fieldNames; }; // Used to generate HeapType names. using HeapTypeNameGenerator = std::function; // The type used for interning IDs in the public interfaces of Type and // HeapType. using TypeID = uint64_t; enum Shareability { Shared, Unshared }; enum class HeapTypeKind { Basic, Func, Struct, Array, Cont, }; class HeapType { // Unlike `Type`, which represents the types of values on the WebAssembly // stack, `HeapType` is used to describe the structures that reference types // refer to. HeapTypes are canonicalized and interned exactly like Types and // should also be passed by value. uintptr_t id; public: // Bits 0 and 1 are used by the Type representation, so need to be left free. // Bit 2 determines whether the basic heap type is shared (1) or unshared (0). enum BasicHeapType : uint32_t { ext = 1 << 3, func = 2 << 3, cont = 3 << 3, any = 4 << 3, eq = 5 << 3, i31 = 6 << 3, struct_ = 7 << 3, array = 8 << 3, exn = 9 << 3, string = 10 << 3, none = 11 << 3, noext = 12 << 3, nofunc = 13 << 3, nocont = 14 << 3, noexn = 15 << 3, }; static constexpr BasicHeapType _last_basic_type = BasicHeapType(noexn + (1 << 2)); // BasicHeapType can be implicitly upgraded to HeapType constexpr HeapType(BasicHeapType id) : id(id) {} // But converting raw TypeID is more dangerous, so make it explicit explicit HeapType(TypeID id) : id(id) {} // Choose an arbitrary heap type as the default. constexpr HeapType() : HeapType(func) {} // Construct a HeapType referring to the single canonical HeapType for the // given signature. In nominal mode, this is the first HeapType created with // this signature. HeapType(Signature signature); HeapType(Continuation cont); // Create a HeapType with the given structure. In equirecursive mode, this may // be the same as a previous HeapType created with the same contents. In // nominal mode, this will be a fresh type distinct from all previously // created HeapTypes. // TODO: make these explicit to differentiate them. HeapType(const Struct& struct_); HeapType(Struct&& struct_); HeapType(Array array); HeapTypeKind getKind() const; constexpr bool isBasic() const { return id <= _last_basic_type; } bool isFunction() const { return isMaybeShared(func) || getKind() == HeapTypeKind::Func; } bool isData() const { auto kind = getKind(); return isMaybeShared(string) || kind == HeapTypeKind::Struct || kind == HeapTypeKind::Array; } bool isSignature() const { return getKind() == HeapTypeKind::Func; } bool isContinuation() const { return getKind() == HeapTypeKind::Cont; } bool isStruct() const { return getKind() == HeapTypeKind::Struct; } bool isArray() const { return getKind() == HeapTypeKind::Array; } bool isExn() const { return isMaybeShared(HeapType::exn); } bool isString() const { return isMaybeShared(HeapType::string); } bool isBottom() const; bool isOpen() const; bool isShared() const { return getShared() == Shared; } Shareability getShared() const; // Check if the type is a given basic heap type, while ignoring whether it is // shared or not. bool isMaybeShared(BasicHeapType type) const { return isBasic() && getBasic(Unshared) == type; } Signature getSignature() const; Continuation getContinuation() const; const Struct& getStruct() const; Array getArray() const; // If there is a nontrivial (i.e. non-basic, one that was declared by the // module) nominal supertype, return it, else an empty optional. std::optional getDeclaredSuperType() const; // As |getDeclaredSuperType|, but also handles basic types, that is, if the // super is a basic type, then we return it here. Declared types are returned // as well, just like |getDeclaredSuperType|. std::optional getSuperType() const; // Return the depth of this heap type in the nominal type hierarchy, i.e. the // number of supertypes in its supertype chain. size_t getDepth() const; // Get the bottom heap type for this heap type's hierarchy. BasicHeapType getUnsharedBottom() const; BasicHeapType getBottom() const { return HeapType(getUnsharedBottom()).getBasic(getShared()); } // Get the top heap type for this heap type's hierarchy. BasicHeapType getUnsharedTop() const; BasicHeapType getTop() const { return HeapType(getUnsharedTop()).getBasic(getShared()); } // Get the recursion group for this non-basic type. RecGroup getRecGroup() const; // Get the index of this non-basic type within its recursion group. size_t getRecGroupIndex() const; constexpr TypeID getID() const { return id; } // Get the shared or unshared version of this basic heap type. constexpr BasicHeapType getBasic(Shareability share) const { assert(isBasic()); return BasicHeapType(share == Shared ? (id | 4) : (id & ~4)); } // (In)equality must be defined for both HeapType and BasicHeapType because it // is otherwise ambiguous whether to convert both this and other to int or // convert other to HeapType. bool operator==(const HeapType& other) const { return id == other.id; } bool operator==(const BasicHeapType& other) const { return id == other; } bool operator!=(const HeapType& other) const { return id != other.id; } bool operator!=(const BasicHeapType& other) const { return id != other; } // Returns true if left is a subtype of right. Subtype includes itself. static bool isSubType(HeapType left, HeapType right); std::vector getTypeChildren() const; // Return the ordered HeapType children, looking through child Types. std::vector getHeapTypeChildren() const; // Similar to `getHeapTypeChildren`, but also includes the supertype if it // exists. std::vector getReferencedHeapTypes() const; // Return the LUB of two HeapTypes, which may or may not exist. static std::optional getLeastUpperBound(HeapType a, HeapType b); // Returns the feature set required to use this type. FeatureSet getFeatures() const; // Helper allowing the value of `print(...)` to be sent to an ostream. Stores // a `TypeID` because `Type` is incomplete at this point and using a reference // makes it less convenient to use. struct Printed { TypeID typeID; HeapTypeNameGenerator generateName; }; // Given a function for generating HeapType names, print the definition of // this HeapType to `os`. `generateName` should return the same // name each time it is called with the same HeapType and it should return // different names for different types. Printed print(HeapTypeNameGenerator generateName) { return Printed{getID(), generateName}; } std::string toString() const; }; class Type { // The `id` uniquely represents each type, so type equality is just a // comparison of the ids. The basic types are packed at the bottom of the // expressible range, and after that tuple types are distinguished by having // bit 0 set. When that bit is masked off, they are pointers to the underlying // vectors of types. Otherwise, the type is a reference type, and is // represented as a heap type with bit 1 set iff the reference type is // nullable. // // Since `Type` is really just a single integer, it should be passed by value. // This is a uintptr_t rather than a TypeID (uint64_t) to save memory on // 32-bit platforms. uintptr_t id; public: enum BasicType : uint32_t { none = 0, unreachable = 1, i32 = 2, i64 = 3, f32 = 4, f64 = 5, v128 = 6, }; static constexpr BasicType _last_basic_type = v128; Type() : id(none) {} // BasicType can be implicitly upgraded to Type constexpr Type(BasicType id) : id(id) {} // But converting raw TypeID is more dangerous, so make it explicit explicit Type(TypeID id) : id(id) {} // Construct tuple from a list of single types Type(std::initializer_list); // Construct from tuple description Type(const Tuple&); Type(Tuple&&); // Construct from a heap type description. Also covers construction from // Signature, Struct or Array via implicit conversion to HeapType. Type(HeapType heapType, Nullability nullable) : Type(heapType.getID() | (nullable == Nullable ? 2 : 0)) {} // Predicates // Compound Concrete // Type Basic │ Single│ // ╒═════════════╦═│═╤═│═╤═│═╤═│═╤═══════╕ // │ none ║ x │ │ │ │ │ // │ unreachable ║ x │ │ │ │ │ // ├─────────────╫───┼───┼───┼───┤───────┤ // │ i32 ║ x │ │ x │ x │ I │ ┐ Number // │ i64 ║ x │ │ x │ x │ I │ │ I_nteger // │ f32 ║ x │ │ x │ x │ F │ │ F_loat // │ f64 ║ x │ │ x │ x │ F │ │ V_ector // │ v128 ║ x │ │ x │ x │ V │ ┘ // ├─ Aliases ───╫───┼───┼───┼───┤───────┤ // │ funcref ║ x │ │ x │ x │ f n │ ┐ Ref // │ anyref ║ x │ │ x │ x │ f? n │ │ f_unc // │ eqref ║ x │ │ x │ x │ n │ │ n_ullable // │ i31ref ║ x │ │ x │ x │ n │ │ // │ structref ║ x │ │ x │ x │ n │ │ // │ arrayref ║ x │ │ x │ x │ n │ │ // │ exnref ║ x │ │ x │ x │ n │ │ // │ stringref ║ x │ │ x │ x │ n │ │ // ├─ Compound ──╫───┼───┼───┼───┤───────┤ │ // │ Ref ║ │ x │ x │ x │ f? n? │◄┘ // │ Tuple ║ │ x │ │ x │ │ // └─────────────╨───┴───┴───┴───┴───────┘ constexpr bool isBasic() const { return id <= _last_basic_type; } constexpr bool isConcrete() const { return id >= i32; } constexpr bool isInteger() const { return id == i32 || id == i64; } constexpr bool isFloat() const { return id == f32 || id == f64; } constexpr bool isVector() const { return id == v128; }; constexpr bool isNumber() const { return id >= i32 && id <= v128; } bool isSingle() const { return isConcrete() && !isTuple(); } // Tuples, refs, etc. are quickly handled using isBasic(), leaving the non- // basic case for the underlying implementation. // TODO: Experiment with leaving bit 0 free in basic types. bool isTuple() const { return !isBasic() && (id & 1); } const Tuple& getTuple() const { assert(isTuple()); return *(Tuple*)(id & ~1); } bool isRef() const { return !isBasic() && !(id & 1); } bool isNullable() const { return isRef() && (id & 2); } bool isNonNullable() const { return isRef() && !(id & 2); } HeapType getHeapType() const { assert(isRef()); return HeapType(id & ~2); } bool isFunction() const { return isRef() && getHeapType().isFunction(); } bool isSignature() const { return isRef() && getHeapType().isSignature(); } bool isData() const { return isRef() && getHeapType().isData(); } // Whether this type is only inhabited by null values. bool isNull() const { return isRef() && getHeapType().isBottom(); } bool isStruct() const { return isRef() && getHeapType().isStruct(); } bool isArray() const { return isRef() && getHeapType().isArray(); } bool isExn() const { return isRef() && getHeapType().isExn(); } bool isString() const { return isRef() && getHeapType().isString(); } bool isDefaultable() const; // TODO: Allow this only for reference types. Nullability getNullability() const { return isNullable() ? Nullable : NonNullable; } private: template bool hasPredicate() { for (const auto& type : *this) { if ((type.*pred)()) { return true; } } return false; } public: bool hasVector() { return hasPredicate<&Type::isVector>(); } bool hasRef() { return hasPredicate<&Type::isRef>(); } constexpr TypeID getID() const { return id; } constexpr BasicType getBasic() const { assert(isBasic() && "Basic type expected"); return static_cast(id); } // (In)equality must be defined for both Type and BasicType because it is // otherwise ambiguous whether to convert both this and other to int or // convert other to Type. bool operator==(const Type& other) const { return id == other.id; } bool operator==(const BasicType& other) const { return id == other; } bool operator!=(const Type& other) const { return id != other.id; } bool operator!=(const BasicType& other) const { return id != other; } // Returns the type size in bytes. Only single types are supported. unsigned getByteSize() const; // Returns whether the type has a size in bytes. This is the same as whether // it can be stored in linear memory. Things like references do not have this // property, while numbers do. Tuples may or may not depending on their // contents. unsigned hasByteSize() const; // Reinterpret an integer type to a float type with the same size and vice // versa. Only single integer and float types are supported. Type reinterpret() const; // Returns the feature set required to use this type. FeatureSet getFeatures() const; // Returns a number type based on its size in bytes and whether it is a float // type. static Type get(unsigned byteSize, bool float_); // Returns true if left is a subtype of right. Subtype includes itself. static bool isSubType(Type left, Type right); // Return the ordered HeapType children, looking through child Types. std::vector getHeapTypeChildren(); // Computes the least upper bound from the type lattice. // If one of the type is unreachable, the other type becomes the result. If // the common supertype does not exist, returns none, a poison value. static bool hasLeastUpperBound(Type a, Type b); static Type getLeastUpperBound(Type a, Type b); template static bool hasLeastUpperBound(const T& types) { auto first = types.begin(), end = types.end(); if (first == end) { return false; } for (auto second = std::next(first); second != end;) { if (!hasLeastUpperBound(*first++, *second++)) { return false; } } return true; } template static Type getLeastUpperBound(const T& types) { auto it = types.begin(), end = types.end(); if (it == end) { return Type::none; } Type lub = *it++; for (; it != end; ++it) { lub = getLeastUpperBound(lub, *it); if (lub == Type::none) { return Type::none; } } return lub; } static Type getGreatestLowerBound(Type a, Type b); // Helper allowing the value of `print(...)` to be sent to an ostream. Stores // a `TypeID` because `Type` is incomplete at this point and using a reference // makes it less convenient to use. struct Printed { TypeID typeID; HeapTypeNameGenerator generateName; }; // Given a function for generating non-basic HeapType names, print this Type // to `os`.`generateName` should return the same name each time it is called // with the same HeapType and it should return different names for different // types. Printed print(HeapTypeNameGenerator generateName) { return Printed{getID(), generateName}; } std::string toString() const; size_t size() const { return isTuple() ? getTuple().size() : size_t(id != Type::none); } struct Iterator : ParentIndexIterator { using value_type = Type; using pointer = const Type*; using reference = const Type&; reference operator*() const; }; Iterator begin() const { return Iterator{{this, 0}}; } Iterator end() const { return Iterator{{this, size()}}; } std::reverse_iterator rbegin() const { return std::make_reverse_iterator(end()); } std::reverse_iterator rend() const { return std::make_reverse_iterator(begin()); } const Type& operator[](size_t i) const { return *Iterator{{this, i}}; } }; namespace HeapTypes { constexpr HeapType ext = HeapType::ext; constexpr HeapType func = HeapType::func; constexpr HeapType cont = HeapType::cont; constexpr HeapType any = HeapType::any; constexpr HeapType eq = HeapType::eq; constexpr HeapType i31 = HeapType::i31; constexpr HeapType struct_ = HeapType::struct_; constexpr HeapType array = HeapType::array; constexpr HeapType exn = HeapType::exn; constexpr HeapType string = HeapType::string; constexpr HeapType none = HeapType::none; constexpr HeapType noext = HeapType::noext; constexpr HeapType nofunc = HeapType::nofunc; constexpr HeapType nocont = HeapType::nocont; constexpr HeapType noexn = HeapType::noexn; } // namespace HeapTypes // A recursion group consisting of one or more HeapTypes. HeapTypes with single // members are encoded without using any additional memory, which is why // `getHeapTypes` has to return a vector by value; it might have to create one // on the fly. class RecGroup { uintptr_t id; public: explicit RecGroup(uintptr_t id) : id(id) {} constexpr TypeID getID() const { return id; } bool operator==(const RecGroup& other) const { return id == other.id; } bool operator!=(const RecGroup& other) const { return id != other.id; } size_t size() const; struct Iterator : ParentIndexIterator { using value_type = HeapType; using pointer = const HeapType*; using reference = const HeapType&; value_type operator*() const; }; Iterator begin() const { return Iterator{{this, 0}}; } Iterator end() const { return Iterator{{this, size()}}; } HeapType operator[](size_t i) const { return *Iterator{{this, i}}; } }; struct Signature { Type params; Type results; Signature() : params(Type::none), results(Type::none) {} Signature(Type params, Type results) : params(params), results(results) {} bool operator==(const Signature& other) const { return params == other.params && results == other.results; } bool operator!=(const Signature& other) const { return !(*this == other); } std::string toString() const; }; struct Continuation { HeapType type; Continuation(HeapType type) : type(type) {} bool operator==(const Continuation& other) const { return type == other.type; } bool operator!=(const Continuation& other) const { return !(*this == other); } std::string toString() const; }; struct Field { Type type; enum PackedType { not_packed, i8, i16, } packedType; // applicable iff type=i32 Mutability mutable_; // Arbitrary defaults for convenience. Field() : type(Type::i32), packedType(not_packed), mutable_(Mutable) {} Field(Type type, Mutability mutable_) : type(type), packedType(not_packed), mutable_(mutable_) {} Field(PackedType packedType, Mutability mutable_) : type(Type::i32), packedType(packedType), mutable_(mutable_) {} constexpr bool isPacked() const { if (packedType != not_packed) { assert(type == Type::i32 && "unexpected type"); return true; } return false; } bool operator==(const Field& other) const { return type == other.type && packedType == other.packedType && mutable_ == other.mutable_; } bool operator!=(const Field& other) const { return !(*this == other); } std::string toString() const; unsigned getByteSize() const; }; using FieldList = std::vector; // Passed by reference rather than by value because it can own an unbounded // amount of data. struct Struct { FieldList fields; Struct() = default; Struct(const Struct& other) : fields(other.fields) {} Struct(const FieldList& fields) : fields(fields) {} Struct(FieldList&& fields) : fields(std::move(fields)) {} bool operator==(const Struct& other) const { return fields == other.fields; } bool operator!=(const Struct& other) const { return !(*this == other); } std::string toString() const; // Prevent accidental copies Struct& operator=(const Struct&) = delete; Struct& operator=(Struct&&) = default; }; struct Array { Field element; Array() = default; Array(const Array& other) : element(other.element) {} Array(Field element) : element(element) {} bool operator==(const Array& other) const { return element == other.element; } bool operator!=(const Array& other) const { return !(*this == other); } std::string toString() const; Array& operator=(const Array& other) = default; }; // TypeBuilder - allows for the construction of recursive types. Contains a // table of `n` mutable HeapTypes and can construct temporary types that are // backed by those HeapTypes, refering to them by reference. Those temporary // types are owned by the TypeBuilder and should only be used in the // construction of HeapTypes to insert into the TypeBuilder. Temporary types // should never be used in the construction of normal Types, only other // temporary types. struct TypeBuilder { struct Impl; std::unique_ptr impl; TypeBuilder(size_t n); TypeBuilder() : TypeBuilder(0) {} ~TypeBuilder(); TypeBuilder(TypeBuilder& other) = delete; TypeBuilder& operator=(TypeBuilder&) = delete; TypeBuilder(TypeBuilder&& other); TypeBuilder& operator=(TypeBuilder&& other); // Append `n` new uninitialized HeapType slots to the end of the TypeBuilder. void grow(size_t n); // The number of HeapType slots in the TypeBuilder. size_t size(); // Sets the heap type at index `i`. May only be called before `build`. void setHeapType(size_t i, Signature signature); void setHeapType(size_t i, Continuation continuation); void setHeapType(size_t i, const Struct& struct_); void setHeapType(size_t i, Struct&& struct_); void setHeapType(size_t i, Array array); // Sets the heap type at index `i` to be a copy of the given heap type with // its referenced HeapTypes to be replaced according to the provided mapping // function. template void copyHeapType(size_t i, HeapType type, F map) { assert(!type.isBasic()); if (auto super = type.getDeclaredSuperType()) { setSubType(i, map(*super)); } setOpen(i, type.isOpen()); setShared(i, type.getShared()); auto copySingleType = [&](Type t) -> Type { if (t.isBasic()) { return t; } assert(t.isRef()); return getTempRefType(map(t.getHeapType()), t.getNullability()); }; auto copyType = [&](Type t) -> Type { if (t.isTuple()) { std::vector elems; elems.reserve(t.size()); for (auto elem : t) { elems.push_back(copySingleType(elem)); } return getTempTupleType(elems); } return copySingleType(t); }; switch (type.getKind()) { case HeapTypeKind::Func: { auto sig = type.getSignature(); setHeapType(i, Signature(copyType(sig.params), copyType(sig.results))); return; } case HeapTypeKind::Struct: { const auto& struct_ = type.getStruct(); std::vector fields; fields.reserve(struct_.fields.size()); for (auto field : struct_.fields) { field.type = copyType(field.type); fields.push_back(field); } setHeapType(i, Struct(fields)); return; } case HeapTypeKind::Array: { auto elem = type.getArray().element; elem.type = copyType(elem.type); // MSVC gets confused without this disambiguation. setHeapType(i, wasm::Array(elem)); return; } case HeapTypeKind::Cont: setHeapType(i, Continuation(map(type.getContinuation().type))); return; case HeapTypeKind::Basic: WASM_UNREACHABLE("unexpected kind"); } } // Gets the temporary HeapType at index `i`. This HeapType should only be used // to construct temporary Types using the methods below. HeapType getTempHeapType(size_t i); // Gets a temporary type or heap type for use in initializing the // TypeBuilder's HeapTypes. For Ref types, the HeapType may be a temporary // HeapType owned by this builder or a canonical HeapType. Type getTempTupleType(const Tuple&); Type getTempRefType(HeapType heapType, Nullability nullable); // Declare the HeapType being built at index `i` to be an immediate subtype of // the given HeapType. void setSubType(size_t i, std::optional super); // Create a new recursion group covering slots [i, i + length). Groups must // not overlap or go out of bounds. void createRecGroup(size_t i, size_t length); void setOpen(size_t i, bool open = true); void setShared(size_t i, Shareability share = Shared); enum class ErrorReason { // There is a cycle in the supertype relation. SelfSupertype, // The declared supertype of a type is invalid. InvalidSupertype, // The declared supertype is an invalid forward reference. ForwardSupertypeReference, // A child of the type is an invalid forward reference. ForwardChildReference, // A continuation reference that does not refer to a function type. InvalidFuncType, // A non-shared field of a shared heap type. InvalidUnsharedField, }; struct Error { // The index of the type causing the failure. size_t index; ErrorReason reason; }; struct BuildResult : std::variant, Error> { operator bool() const { return bool(std::get_if>(this)); } const std::vector& operator*() const { return std::get>(*this); } const std::vector* operator->() const { return &*(*this); } const Error* getError() const { return std::get_if(this); } }; // Returns all of the newly constructed heap types. May only be called once // all of the heap types have been initialized with `setHeapType`. In nominal // mode, all of the constructed HeapTypes will be fresh and distinct. In // nominal mode, will also produce a fatal error if the declared subtype // relationships are not valid. BuildResult build(); // Utility for ergonomically using operator[] instead of explicit setHeapType // and getTempHeapType methods. struct Entry { TypeBuilder& builder; size_t index; operator HeapType() const { return builder.getTempHeapType(index); } Entry& operator=(Signature signature) { builder.setHeapType(index, signature); return *this; } Entry& operator=(Continuation continuation) { builder.setHeapType(index, continuation); return *this; } Entry& operator=(const Struct& struct_) { builder.setHeapType(index, struct_); return *this; } Entry& operator=(Struct&& struct_) { builder.setHeapType(index, std::move(struct_)); return *this; } Entry& operator=(Array array) { builder.setHeapType(index, array); return *this; } Entry& subTypeOf(std::optional other) { builder.setSubType(index, other); return *this; } Entry& setOpen(bool open = true) { builder.setOpen(index, open); return *this; } Entry& setShared(Shareability share = Shared) { builder.setShared(index, share); return *this; } template Entry& copy(HeapType type, F map) { builder.copyHeapType(index, type, map); return *this; } Entry& copy(HeapType type) { return copy(type, [](HeapType t) { return t; }); } }; Entry operator[](size_t i) { return Entry{*this, i}; } void dump(); }; // We consider certain specific types to always be public, to allow closed- // world to operate even if they escape. Specifically, "plain old data" types // like array of i8 and i16, which are used to represent strings, may cross // the boundary in Web environments. // // These are "ignorable as public", because we do not error on them being // public. That is, we // // 1. Consider them public, so that passes that do not operate on public types // do not in fact operate on them, and // 2. Are ok with them being public in the validator. // std::unordered_set getIgnorablePublicTypes(); std::ostream& operator<<(std::ostream&, Type); std::ostream& operator<<(std::ostream&, Type::Printed); std::ostream& operator<<(std::ostream&, HeapType); std::ostream& operator<<(std::ostream&, HeapType::Printed); std::ostream& operator<<(std::ostream&, Tuple); std::ostream& operator<<(std::ostream&, Signature); std::ostream& operator<<(std::ostream&, Continuation); std::ostream& operator<<(std::ostream&, Field); std::ostream& operator<<(std::ostream&, Struct); std::ostream& operator<<(std::ostream&, Array); std::ostream& operator<<(std::ostream&, TypeBuilder::ErrorReason); // Inline some nontrivial methods here for performance reasons. inline bool HeapType::isBottom() const { if (isBasic()) { switch (getBasic(Unshared)) { case ext: case func: case cont: case any: case eq: case i31: case struct_: case array: case exn: case string: return false; case none: case noext: case nofunc: case nocont: case noexn: return true; } } return false; } } // namespace wasm namespace std { template<> class hash { public: size_t operator()(const wasm::Type&) const; }; template<> class hash { public: size_t operator()(const wasm::Signature&) const; }; template<> class hash { public: size_t operator()(const wasm::Continuation&) const; }; template<> class hash { public: size_t operator()(const wasm::Field&) const; }; template<> class hash { public: size_t operator()(const wasm::Struct&) const; }; template<> class hash { public: size_t operator()(const wasm::Array&) const; }; template<> class hash { public: size_t operator()(const wasm::HeapType&) const; }; template<> class hash { public: size_t operator()(const wasm::RecGroup&) const; }; } // namespace std #endif // wasm_wasm_type_h