diff options
author | Thomas Lively <7121787+tlively@users.noreply.github.com> | 2022-05-02 17:45:57 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2022-05-03 00:45:57 +0000 |
commit | f124a11ca3a40c87ab6aa4498037449584689be9 (patch) | |
tree | 853e7d60fc6bf22d4ed47b336bcc29e4446fc788 /src/ir/module-utils.cpp | |
parent | 5d157836c4b684e6a9f2f4a431fb748fa6ab969c (diff) | |
download | binaryen-f124a11ca3a40c87ab6aa4498037449584689be9.tar.gz binaryen-f124a11ca3a40c87ab6aa4498037449584689be9.tar.bz2 binaryen-f124a11ca3a40c87ab6aa4498037449584689be9.zip |
Update nominal type ordering (#4631)
V8 requires that supertypes come before subtypes when it parses
isorecursive (i.e. standards-track) type definitions. Since 2268f2a we are
emitting nominal types using the standard isorecursive format, so respect the
ordering requirement.
Diffstat (limited to 'src/ir/module-utils.cpp')
-rw-r--r-- | src/ir/module-utils.cpp | 47 |
1 files changed, 33 insertions, 14 deletions
diff --git a/src/ir/module-utils.cpp b/src/ir/module-utils.cpp index 7861bde54..bf1a45205 100644 --- a/src/ir/module-utils.cpp +++ b/src/ir/module-utils.cpp @@ -203,9 +203,10 @@ std::vector<HeapType> collectHeapTypes(Module& wasm) { } IndexedHeapTypes getOptimizedIndexedHeapTypes(Module& wasm) { + TypeSystem system = getTypeSystem(); Counts counts = getHeapTypeCounts(wasm); - if (getTypeSystem() != TypeSystem::Isorecursive) { + if (system == TypeSystem::Equirecursive) { // Sort by frequency and then original insertion order. std::vector<std::pair<HeapType, size_t>> sorted(counts.begin(), counts.end()); @@ -223,9 +224,12 @@ IndexedHeapTypes getOptimizedIndexedHeapTypes(Module& wasm) { return indexedTypes; } - // Isorecursive types have to be arranged into topologically ordered recursion - // groups. Sort the groups by average use count among their members so that - // the topological sort will place frequently used types first. + // Types have to be arranged into topologically ordered recursion groups. + // Under isorecrsive typing, the topological sort has to take all referenced + // rec groups into account but under nominal typing it only has to take + // supertypes into account. First, sort the groups by average use count among + // their members so that the later topological sort will place frequently used + // types first. struct GroupInfo { size_t index; double useCount = 0; @@ -236,7 +240,7 @@ IndexedHeapTypes getOptimizedIndexedHeapTypes(Module& wasm) { if (useCount != other.useCount) { return useCount < other.useCount; } - return index < other.index; + return index > other.index; } }; @@ -257,20 +261,35 @@ IndexedHeapTypes getOptimizedIndexedHeapTypes(Module& wasm) { // Update the reference count. info.useCount += counts.at(type); // Collect predecessor groups. - for (auto child : type.getReferencedHeapTypes()) { - if (!child.isBasic()) { - RecGroup otherGroup = child.getRecGroup(); - if (otherGroup != group) { - info.preds.insert(otherGroup); + switch (system) { + case TypeSystem::Isorecursive: + for (auto child : type.getReferencedHeapTypes()) { + if (!child.isBasic()) { + RecGroup otherGroup = child.getRecGroup(); + if (otherGroup != group) { + info.preds.insert(otherGroup); + } + } } - } + break; + case TypeSystem::Nominal: + if (auto super = type.getSuperType()) { + info.preds.insert(super->getRecGroup()); + } + break; + case TypeSystem::Equirecursive: + WASM_UNREACHABLE( + "Equirecursive types should already have been handled"); } } // Fix up the use counts to be averages to ensure groups are used comensurate - // with the amount of index space they occupy. - for (auto& [group, info] : groupInfos) { - info.useCount /= group.size(); + // with the amount of index space they occupy. Skip this for nominal types + // since their internal group size is always 1. + if (system != TypeSystem::Nominal) { + for (auto& [group, info] : groupInfos) { + info.useCount /= group.size(); + } } // Sort the predecessors so the most used will be visited first. |