summaryrefslogtreecommitdiff
path: root/src/literal.h
diff options
context:
space:
mode:
authorAlon Zakai <azakai@google.com>2021-06-17 07:51:11 -0700
committerGitHub <noreply@github.com>2021-06-17 07:51:11 -0700
commitc36c6fa9e42f4e917864312780ba95fb996eda79 (patch)
tree1e171bbd8b1a10f09ab14a59bc03b39bbe494c1e /src/literal.h
parent9d279c08b9f37b6cf2c5a5fac564eee9ea4fb927 (diff)
downloadbinaryen-c36c6fa9e42f4e917864312780ba95fb996eda79.tar.gz
binaryen-c36c6fa9e42f4e917864312780ba95fb996eda79.tar.bz2
binaryen-c36c6fa9e42f4e917864312780ba95fb996eda79.zip
[Wasm GC] rtt.fresh_sub (#3936)
This is the same as rtt.sub, but creates a "new" rtt each time. See https://docs.google.com/document/d/1DklC3qVuOdLHSXB5UXghM_syCh-4cMinQ50ICiXnK3Q/edit# The old Literal implementation of rtts becomes a little more complex here, as it was designed for the original spec where only structure matters. It may be worth a complete redesign there, but for now as the spec is in flux I think the approach here is good enough.
Diffstat (limited to 'src/literal.h')
-rw-r--r--src/literal.h32
1 files changed, 28 insertions, 4 deletions
diff --git a/src/literal.h b/src/literal.h
index 14281d66f..1dd013c4c 100644
--- a/src/literal.h
+++ b/src/literal.h
@@ -31,9 +31,7 @@ namespace wasm {
class Literals;
struct GCData;
-// Subclass the vector type so that this is not easily confused with a vector of
-// types (which could be confusing on the Literal constructor).
-struct RttSupers : std::vector<Type> {};
+struct RttSupers;
class Literal {
// store only integers, whose bits are deterministic. floats
@@ -61,6 +59,9 @@ class Literal {
// would do, but it is simple.)
// The unique_ptr here is to avoid increasing the size of the union as well
// as the Literal class itself.
+ // To support the experimental RttFreshSub instruction, we not only store
+ // the type, but also a reference to an allocation.
+ // See struct RttSuper below for more details.
std::unique_ptr<RttSupers> rttSupers;
// TODO: Literals of type `externref` can only be `null` currently but we
// will need to represent extern values eventually, to
@@ -684,6 +685,28 @@ struct GCData {
GCData(Literal rtt, Literals values) : rtt(rtt), values(values) {}
};
+struct RttSuper {
+ // The type of the super.
+ Type type;
+ // A shared allocation, used to implement rtt.fresh_sub. This is null for a
+ // normal sub, and for a fresh one we allocate a value here, which can then be
+ // used to differentiate rtts. (The allocation is shared so that when copying
+ // an rtt we remain equal.)
+ // TODO: Remove or optimize this when the spec stabilizes.
+ std::shared_ptr<size_t> freshPtr;
+
+ RttSuper(Type type) : type(type) {}
+
+ void makeFresh() { freshPtr = std::make_shared<size_t>(); }
+
+ bool operator==(const RttSuper& other) const {
+ return type == other.type && freshPtr == other.freshPtr;
+ }
+ bool operator!=(const RttSuper& other) const { return !(*this == other); }
+};
+
+struct RttSupers : std::vector<RttSuper> {};
+
} // namespace wasm
namespace std {
@@ -743,7 +766,8 @@ template<> struct hash<wasm::Literal> {
const auto& supers = a.getRttSupers();
wasm::rehash(digest, supers.size());
for (auto super : supers) {
- wasm::rehash(digest, super.getID());
+ wasm::rehash(digest, super.type.getID());
+ wasm::rehash(digest, uintptr_t(super.freshPtr.get()));
}
return digest;
}