summaryrefslogtreecommitdiff
path: root/src/interp/istream.cc
diff options
context:
space:
mode:
authorNg Zhi An <ngzhian@gmail.com>2021-03-22 15:22:02 -0700
committerGitHub <noreply@github.com>2021-03-22 15:22:02 -0700
commit2c7d2572414aad39341734d1513a6cd94759a320 (patch)
tree4b71cafd0fbee7c1a4920a96b8d4e11138f7264a /src/interp/istream.cc
parentc7293e42c587cab2b15eaf2934f574f84eeab9e5 (diff)
downloadwabt-2c7d2572414aad39341734d1513a6cd94759a320.tar.gz
wabt-2c7d2572414aad39341734d1513a6cd94759a320.tar.bz2
wabt-2c7d2572414aad39341734d1513a6cd94759a320.zip
[simd] Implement load lane (#1646)
This is a new kind of ir/ast node/instruction. It has 3 immediates: memarg align, memarg offset, and lane index. This required new visitor functions in all the places. Drive-by cleanup to share the simd lane parsing logic between shuffle, lane op and this new load lane instructions. This requires rebasing some tests because the error messages are slightly different now.
Diffstat (limited to 'src/interp/istream.cc')
-rw-r--r--src/interp/istream.cc24
1 files changed, 24 insertions, 0 deletions
diff --git a/src/interp/istream.cc b/src/interp/istream.cc
index 67025c85..4e24df7c 100644
--- a/src/interp/istream.cc
+++ b/src/interp/istream.cc
@@ -69,6 +69,13 @@ void Istream::Emit(Opcode::Enum op, u32 val1, u32 val2) {
EmitInternal(val2);
}
+void Istream::Emit(Opcode::Enum op, u32 val1, u32 val2, u8 val3) {
+ Emit(op);
+ EmitInternal(val1);
+ EmitInternal(val2);
+ EmitInternal(val3);
+}
+
void Istream::EmitDropKeep(u32 drop, u32 keep) {
if (drop > 0) {
if (drop == 1 && keep == 0) {
@@ -645,6 +652,17 @@ Instr Istream::Read(Offset* offset) const {
instr.imm_u32x2.snd = ReadAt<u32>(offset);
break;
+ case Opcode::V128Load8Lane:
+ case Opcode::V128Load16Lane:
+ case Opcode::V128Load32Lane:
+ case Opcode::V128Load64Lane:
+ // Index, memory offset, lane index immediates, 2 operands.
+ instr.kind = InstrKind::Imm_Index_Offset_Lane_Op_2;
+ instr.imm_u32x2_u8.fst = ReadAt<u32>(offset);
+ instr.imm_u32x2_u8.snd = ReadAt<u32>(offset);
+ instr.imm_u32x2_u8.idx = ReadAt<u8>(offset);
+ break;
+
case Opcode::I32AtomicRmw16CmpxchgU:
case Opcode::I32AtomicRmw8CmpxchgU:
case Opcode::I32AtomicRmwCmpxchg:
@@ -868,6 +886,12 @@ Istream::Offset Istream::Trace(Stream* stream,
source->Pick(1, instr).c_str());
break;
+ case InstrKind::Imm_Index_Offset_Lane_Op_2:
+ stream->Writef(" $%u:%s+$%u, %s (Lane imm: $%u)\n", instr.imm_u32x2_u8.fst,
+ source->Pick(2, instr).c_str(), instr.imm_u32x2_u8.snd,
+ source->Pick(1, instr).c_str(), instr.imm_u32x2_u8.idx);
+ break;
+
case InstrKind::Imm_I32_Op_0:
stream->Writef(" %u\n", instr.imm_u32);
break;