summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/ir/possible-contents.cpp10
-rw-r--r--src/ir/possible-contents.h17
2 files changed, 27 insertions, 0 deletions
diff --git a/src/ir/possible-contents.cpp b/src/ir/possible-contents.cpp
index 7c9a4fc6b..e7e0cd4fd 100644
--- a/src/ir/possible-contents.cpp
+++ b/src/ir/possible-contents.cpp
@@ -1202,6 +1202,13 @@ struct InfoCollector
assert(handledPops == totalPops);
// Handle local.get/sets: each set must write to the proper gets.
+ //
+ // Note that we do not use LocalLocation because LocalGraph gives us more
+ // precise information: we generate direct links from sets to relevant gets
+ // rather than consider each local index a single location, which
+ // LocalLocation does. (LocalLocation is useful in cases where we do need a
+ // single location, such as when we consider what type to give the local;
+ // the type must be the same for all gets of that local.)
LocalGraph localGraph(func, getModule());
for (auto& [get, setsForGet] : localGraph.getSetses) {
@@ -2766,6 +2773,9 @@ void Flower::dump(Location location) {
} else if (auto* loc = std::get_if<ParamLocation>(&location)) {
std::cout << " paramloc " << loc->func->name << " : " << loc->index
<< '\n';
+ } else if (auto* loc = std::get_if<LocalLocation>(&location)) {
+ std::cout << " localloc " << loc->func->name << " : " << loc->index
+ << '\n';
} else if (auto* loc = std::get_if<ResultLocation>(&location)) {
std::cout << " resultloc $" << loc->func->name << " : " << loc->index
<< '\n';
diff --git a/src/ir/possible-contents.h b/src/ir/possible-contents.h
index 2fd6c39c7..2773c1d31 100644
--- a/src/ir/possible-contents.h
+++ b/src/ir/possible-contents.h
@@ -380,6 +380,15 @@ struct ParamLocation {
}
};
+// The location of a value in a local.
+struct LocalLocation {
+ Function* func;
+ Index index;
+ bool operator==(const LocalLocation& other) const {
+ return func == other.func && index == other.index;
+ }
+};
+
// The location of one of the results of a function.
struct ResultLocation {
Function* func;
@@ -494,6 +503,7 @@ struct ConeReadLocation {
// have.
using Location = std::variant<ExpressionLocation,
ParamLocation,
+ LocalLocation,
ResultLocation,
BreakTargetLocation,
GlobalLocation,
@@ -534,6 +544,13 @@ template<> struct hash<wasm::ParamLocation> {
}
};
+template<> struct hash<wasm::LocalLocation> {
+ size_t operator()(const wasm::LocalLocation& loc) const {
+ return std::hash<std::pair<size_t, wasm::Index>>{}(
+ {size_t(loc.func), loc.index});
+ }
+};
+
template<> struct hash<wasm::ResultLocation> {
size_t operator()(const wasm::ResultLocation& loc) const {
return std::hash<std::pair<size_t, wasm::Index>>{}(