/* * Copyright 2022 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_printing_h #define wasm_wasm_type_printing_h #include #include #include #include "support/name.h" #include "support/utilities.h" #include "wasm-type.h" #include "wasm.h" namespace wasm { // CRTP base that all other type name generators should subclass. Provides the // ability to use the generator as a function to print Types and HeapTypes to // streams. template struct TypeNameGeneratorBase { TypeNames getNames(HeapType type) { static_assert(&TypeNameGeneratorBase::getNames != &Subclass::getNames, "Derived class must implement getNames"); WASM_UNREACHABLE("Derived class must implement getNames"); } HeapType::Printed operator()(HeapType type) { return type.print( [&](HeapType ht) { return static_cast(this)->getNames(ht); }); } Type::Printed operator()(Type type) { return type.print( [&](HeapType ht) { return static_cast(this)->getNames(ht); }); } }; // Generates names like "func.0", "struct.1", "array.2", etc. Struct fields are // not given names. struct DefaultTypeNameGenerator : TypeNameGeneratorBase { size_t funcCount = 0; size_t contCount = 0; size_t structCount = 0; size_t arrayCount = 0; // Cached names for types that have already been seen. std::unordered_map nameCache; TypeNames getNames(HeapType type); }; // Generates names based on the indices of types in some collection, falling // back to the given FallbackGenerator when encountering a type not in the // collection. Struct fields are not given names. template struct IndexedTypeNameGenerator : TypeNameGeneratorBase> { DefaultTypeNameGenerator defaultGenerator; FallbackGenerator& fallback; std::unordered_map names; template IndexedTypeNameGenerator(T& types, FallbackGenerator& fallback, const std::string& prefix = "") : fallback(fallback) { for (size_t i = 0; i < types.size(); ++i) { names.insert({types[i], {prefix + std::to_string(i), {}}}); } } template IndexedTypeNameGenerator(T& types, const std::string& prefix = "") : IndexedTypeNameGenerator(types, defaultGenerator, prefix) {} TypeNames getNames(HeapType type) { if (auto it = names.find(type); it != names.end()) { return it->second; } else { return fallback.getNames(type); } } }; // Deduction guide. template IndexedTypeNameGenerator(T&) -> IndexedTypeNameGenerator<>; // Prints heap types stored in a module, falling back to the given // FallbackGenerator if the module does not have a name for type type. template struct ModuleTypeNameGenerator : TypeNameGeneratorBase> { const Module& wasm; DefaultTypeNameGenerator defaultGenerator; FallbackGenerator& fallback; ModuleTypeNameGenerator(const Module& wasm, FallbackGenerator& fallback) : wasm(wasm), fallback(fallback) {} // TODO: Use C++20 `requires` to clean this up. template ModuleTypeNameGenerator( const Module& wasm, std::enable_if_t>* = nullptr) : ModuleTypeNameGenerator(wasm, defaultGenerator) {} TypeNames getNames(HeapType type) { if (auto it = wasm.typeNames.find(type); it != wasm.typeNames.end()) { return it->second; } return fallback.getNames(type); } }; // Deduction guide. ModuleTypeNameGenerator(const Module&) -> ModuleTypeNameGenerator<>; } // namespace wasm #endif // wasm_wasm_type_printing_h