diff options
author | Alon Zakai <alonzakai@gmail.com> | 2018-05-10 08:58:09 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2018-05-10 08:58:09 -0700 |
commit | 111b4f1950467d51a78211af183f4ae6398aac49 (patch) | |
tree | 7f53ab6389e2bcaacf761cbf33846ac0210bd098 /src/ir/equivalent_sets.h | |
parent | 6a9ececa2fc9eca99a12b65ca130612942babdce (diff) | |
download | binaryen-111b4f1950467d51a78211af183f4ae6398aac49.tar.gz binaryen-111b4f1950467d51a78211af183f4ae6398aac49.tar.bz2 binaryen-111b4f1950467d51a78211af183f4ae6398aac49.zip |
Optimize equivalent locals (#1540)
If locals are known to contain the same value, we can
* Pick which local to use for a get_local of any of them. Makes sense to prefer the most common, to increase the chance of one dropping to zero uses.
* Remove copies between a local and one that we know contains the same value.
This is a consistent win, small though, around 0.1-0.2%.
Diffstat (limited to 'src/ir/equivalent_sets.h')
-rw-r--r-- | src/ir/equivalent_sets.h | 94 |
1 files changed, 94 insertions, 0 deletions
diff --git a/src/ir/equivalent_sets.h b/src/ir/equivalent_sets.h new file mode 100644 index 000000000..d1a957bbd --- /dev/null +++ b/src/ir/equivalent_sets.h @@ -0,0 +1,94 @@ +/* + * Copyright 2018 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_ir_equivalent_sets_h +#define wasm_ir_equivalent_sets_h + +#include <wasm.h> + +namespace wasm { + +// +// A map of each index to all those it is equivalent to, and some helpers. +// +struct EquivalentSets { + // A set of indexes. + typedef std::unordered_set<Index> Set; + + std::unordered_map<Index, std::shared_ptr<Set>> indexSets; + + // Clears the state completely, removing all equivalences. + void clear() { + indexSets.clear(); + } + + // Resets an index, removing any equivalences between it and others. + void reset(Index index) { + auto iter = indexSets.find(index); + if (iter != indexSets.end()) { + auto& set = iter->second; + assert(!set->empty()); // can't be empty - we are equal to ourselves! + if (set->size() > 1) { + // We are not the last item, fix things up + set->erase(index); + } + indexSets.erase(iter); + } + } + + // Adds a new equivalence between two indexes. + // `justReset` is an index that was just reset, and has no + // equivalences. `other` may have existing equivalences. + void add(Index justReset, Index other) { + auto iter = indexSets.find(other); + if (iter != indexSets.end()) { + auto& set = iter->second; + set->insert(justReset); + indexSets[justReset] = set; + } else { + auto set = std::make_shared<Set>(); + set->insert(justReset); + set->insert(other); + indexSets[justReset] = set; + indexSets[other] = set; + } + } + + // Checks whether two indexes contain the same data. + bool check(Index a, Index b) { + if (a == b) return true; + if (auto* set = getEquivalents(a)) { + if (set->find(b) != set->end()) { + return true; + } + } + return false; + } + + // Returns the equivalent set, or nullptr + Set* getEquivalents(Index index) { + auto iter = indexSets.find(index); + if (iter != indexSets.end()) { + return iter->second.get(); + } + return nullptr; + } +}; + +} // namespace wasm + +#endif // wasm_ir_equivalent_sets_h + |