summaryrefslogtreecommitdiff
path: root/test/regress/recursion-issue.txt
diff options
context:
space:
mode:
authorSoni L. <EnderMoneyMod@gmail.com>2024-09-18 15:35:55 -0300
committerGitHub <noreply@github.com>2024-09-18 11:35:55 -0700
commit874caa4433e51013132c0f3d6ca479b759a6f23f (patch)
tree15be9f7e3c057aafb6fb0dabe60650c32af4c885 /test/regress/recursion-issue.txt
parentc6853eb4857908072fd1d7c28dffb1d2613a0ee6 (diff)
downloadwabt-874caa4433e51013132c0f3d6ca479b759a6f23f.tar.gz
wabt-874caa4433e51013132c0f3d6ca479b759a6f23f.tar.bz2
wabt-874caa4433e51013132c0f3d6ca479b759a6f23f.zip
[wasm-interp] Fix memory corruption with recursive call_indirect (#2464)
The interpreter could overflow the stack without trapping properly in `call_indirect` situations. While it would set the `out_trap` to the trap reason, it would return `RunResult::Ok` and the interpreter code would only check `RunResult::Ok` to decide whether or not to keep running. In other words, while the stack overflow meant the interpreter wouldn't push a frame onto the call stack, the interpreter loop would continue advancing instructions, resulting in instructions after the runaway `call_indirect` running. If the offending `call_indirect` didn't have return values, it would be as if the call returned normally. If it did have return values, nothing would be pushed onto the value stack, yet the return types would be pushed onto the type stack. With careful manipulation of the following instructions, this could be used to cause all sorts of memory corruption. As it turns out, the function exit code, as well as a handful of other instructions, do check the state of the value and type stacks and can safely reproduce the bug without the memory corruption, so that's what we made the test do. The obvious fix was to make `call_indirect` propagate `RunResult::Trap` properly. Additionally, we made it so `assert_exhaustion` checks both the `RunResult` *and* the `out_trap`, and asserts if they don't match. This should help catch similar bugs in the future. Closes #2462 Fixes #2398
Diffstat (limited to 'test/regress/recursion-issue.txt')
-rw-r--r--test/regress/recursion-issue.txt20
1 files changed, 20 insertions, 0 deletions
diff --git a/test/regress/recursion-issue.txt b/test/regress/recursion-issue.txt
new file mode 100644
index 00000000..a5fa660b
--- /dev/null
+++ b/test/regress/recursion-issue.txt
@@ -0,0 +1,20 @@
+;;; TOOL: run-interp-spec
+;;; NOTE: ref: issue-2398
+(module
+ (type $out-i32 (func (result i32)))
+
+ (table funcref
+ (elem
+ $runaway-with-return
+ )
+ )
+
+ (func $runaway-with-return (export "runaway-with-return") (type $out-i32)
+ (call_indirect (type $out-i32) (i32.const 0))
+ )
+)
+
+(assert_exhaustion (invoke "runaway-with-return") "call stack exhausted")
+(;; STDOUT ;;;
+2/2 tests passed.
+;;; STDOUT ;;)