diff options
author | Ashley Nelson <nashley@google.com> | 2022-12-09 16:18:44 -0800 |
---|---|---|
committer | GitHub <noreply@github.com> | 2022-12-09 16:18:44 -0800 |
commit | f5e71e6d2be82639681fc7d45794645e03d2ad93 (patch) | |
tree | a4438a50c46f3c96aefdc469e5fde272a87518dd /test/lit/passes/multi-memory-lowering.wast | |
parent | 082dbe25b7377809b1b3dc429cb334fc80fac286 (diff) | |
download | binaryen-f5e71e6d2be82639681fc7d45794645e03d2ad93.tar.gz binaryen-f5e71e6d2be82639681fc7d45794645e03d2ad93.tar.bz2 binaryen-f5e71e6d2be82639681fc7d45794645e03d2ad93.zip |
Adds bounds checks to Load/Store in Multi-Memories Lowering Pass (#5256)
Per the wasm spec guidelines for Load (rule 10) & Store (rule 12), this PR adds an option for bounds checking, producing a runtime error if the instruction exceeds the bounds of the particular memory within the combined memory.
Diffstat (limited to 'test/lit/passes/multi-memory-lowering.wast')
-rw-r--r-- | test/lit/passes/multi-memory-lowering.wast | 339 |
1 files changed, 339 insertions, 0 deletions
diff --git a/test/lit/passes/multi-memory-lowering.wast b/test/lit/passes/multi-memory-lowering.wast index fc402d877..d66db1b02 100644 --- a/test/lit/passes/multi-memory-lowering.wast +++ b/test/lit/passes/multi-memory-lowering.wast @@ -1,5 +1,6 @@ ;; NOTE: Assertions have been generated by update_lit_checks.py --all-items and should not be edited. ;; RUN: wasm-opt %s --enable-multi-memories --multi-memory-lowering --enable-bulk-memory --enable-extended-const -S -o - | filecheck %s +;; RUN: wasm-opt %s --enable-multi-memories --multi-memory-lowering-with-bounds-checks --enable-bulk-memory --enable-extended-const -S -o - | filecheck %s --check-prefix BOUNDS (module (memory $memory1 1) @@ -49,6 +50,105 @@ ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) + ;; BOUNDS: (type $none_=>_i32 (func (result i32))) + + ;; BOUNDS: (type $i32_=>_i32 (func (param i32) (result i32))) + + ;; BOUNDS: (type $none_=>_none (func)) + + ;; BOUNDS: (global $memory2_byte_offset (mut i32) (i32.const 65536)) + + ;; BOUNDS: (global $memory3_byte_offset (mut i32) (i32.const 196608)) + + ;; BOUNDS: (memory $combined_memory 6) + + ;; BOUNDS: (data (i32.const 0) "a") + + ;; BOUNDS: (data (i32.add + ;; BOUNDS-NEXT: (global.get $memory3_byte_offset) + ;; BOUNDS-NEXT: (i32.const 1) + ;; BOUNDS-NEXT: ) "123") + + ;; BOUNDS: (func $loads + ;; BOUNDS-NEXT: (local $0 i32) + ;; BOUNDS-NEXT: (local $1 i32) + ;; BOUNDS-NEXT: (local $2 i32) + ;; BOUNDS-NEXT: (drop + ;; BOUNDS-NEXT: (i32.load + ;; BOUNDS-NEXT: (block (result i32) + ;; BOUNDS-NEXT: (local.set $0 + ;; BOUNDS-NEXT: (i32.const 10) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (if + ;; BOUNDS-NEXT: (i32.gt_u + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (local.get $0) + ;; BOUNDS-NEXT: (i32.const 0) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.const 4) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (call $memory1_size) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (unreachable) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (local.get $0) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (drop + ;; BOUNDS-NEXT: (i32.load + ;; BOUNDS-NEXT: (block (result i32) + ;; BOUNDS-NEXT: (local.set $1 + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (global.get $memory2_byte_offset) + ;; BOUNDS-NEXT: (i32.const 11) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (if + ;; BOUNDS-NEXT: (i32.gt_u + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (local.get $1) + ;; BOUNDS-NEXT: (i32.const 0) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.const 4) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (call $memory2_size) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (unreachable) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (local.get $1) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (drop + ;; BOUNDS-NEXT: (i32.load + ;; BOUNDS-NEXT: (block (result i32) + ;; BOUNDS-NEXT: (local.set $2 + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (global.get $memory3_byte_offset) + ;; BOUNDS-NEXT: (i32.const 12) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (if + ;; BOUNDS-NEXT: (i32.gt_u + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (local.get $2) + ;; BOUNDS-NEXT: (i32.const 0) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.const 4) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (call $memory3_size) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (unreachable) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (local.get $2) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) (func $loads (drop (i32.load $memory1 @@ -86,6 +186,83 @@ ;; CHECK-NEXT: (i32.const 115) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) + ;; BOUNDS: (func $stores + ;; BOUNDS-NEXT: (local $0 i32) + ;; BOUNDS-NEXT: (local $1 i32) + ;; BOUNDS-NEXT: (local $2 i32) + ;; BOUNDS-NEXT: (i32.store + ;; BOUNDS-NEXT: (block (result i32) + ;; BOUNDS-NEXT: (local.set $0 + ;; BOUNDS-NEXT: (i32.const 10) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (if + ;; BOUNDS-NEXT: (i32.gt_u + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (local.get $0) + ;; BOUNDS-NEXT: (i32.const 0) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.const 4) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (call $memory1_size) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (unreachable) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (local.get $0) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.const 115) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.store + ;; BOUNDS-NEXT: (block (result i32) + ;; BOUNDS-NEXT: (local.set $1 + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (global.get $memory2_byte_offset) + ;; BOUNDS-NEXT: (i32.const 11) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (if + ;; BOUNDS-NEXT: (i32.gt_u + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (local.get $1) + ;; BOUNDS-NEXT: (i32.const 0) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.const 4) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (call $memory2_size) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (unreachable) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (local.get $1) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.const 115) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.store + ;; BOUNDS-NEXT: (block (result i32) + ;; BOUNDS-NEXT: (local.set $2 + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (global.get $memory3_byte_offset) + ;; BOUNDS-NEXT: (i32.const 12) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (if + ;; BOUNDS-NEXT: (i32.gt_u + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (i32.add + ;; BOUNDS-NEXT: (local.get $2) + ;; BOUNDS-NEXT: (i32.const 0) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.const 4) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (call $memory3_size) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (unreachable) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (local.get $2) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: (i32.const 115) + ;; BOUNDS-NEXT: ) + ;; BOUNDS-NEXT: ) (func $stores (i32.store $memory1 (i32.const 10) @@ -263,3 +440,165 @@ ;; CHECK-NEXT: ) ;; CHECK-NEXT: (local.get $return_size) ;; CHECK-NEXT: ) + +;; BOUNDS: (func $memory1_size (result i32) +;; BOUNDS-NEXT: (return +;; BOUNDS-NEXT: (i32.div_u +;; BOUNDS-NEXT: (global.get $memory2_byte_offset) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) + +;; BOUNDS: (func $memory2_size (result i32) +;; BOUNDS-NEXT: (return +;; BOUNDS-NEXT: (i32.sub +;; BOUNDS-NEXT: (i32.div_u +;; BOUNDS-NEXT: (global.get $memory3_byte_offset) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (i32.div_u +;; BOUNDS-NEXT: (global.get $memory2_byte_offset) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) + +;; BOUNDS: (func $memory3_size (result i32) +;; BOUNDS-NEXT: (return +;; BOUNDS-NEXT: (i32.sub +;; BOUNDS-NEXT: (memory.size) +;; BOUNDS-NEXT: (i32.div_u +;; BOUNDS-NEXT: (global.get $memory3_byte_offset) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) + +;; BOUNDS: (func $memory1_grow (param $page_delta i32) (result i32) +;; BOUNDS-NEXT: (local $return_size i32) +;; BOUNDS-NEXT: (local $memory_size i32) +;; BOUNDS-NEXT: (local.set $return_size +;; BOUNDS-NEXT: (call $memory1_size) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (local.set $memory_size +;; BOUNDS-NEXT: (memory.size) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (if +;; BOUNDS-NEXT: (i32.eq +;; BOUNDS-NEXT: (memory.grow +;; BOUNDS-NEXT: (local.get $page_delta) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (i32.const -1) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (return +;; BOUNDS-NEXT: (i32.const -1) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (memory.copy +;; BOUNDS-NEXT: (i32.add +;; BOUNDS-NEXT: (global.get $memory2_byte_offset) +;; BOUNDS-NEXT: (i32.mul +;; BOUNDS-NEXT: (local.get $page_delta) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (global.get $memory2_byte_offset) +;; BOUNDS-NEXT: (i32.sub +;; BOUNDS-NEXT: (i32.mul +;; BOUNDS-NEXT: (local.get $memory_size) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (global.get $memory2_byte_offset) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (global.set $memory2_byte_offset +;; BOUNDS-NEXT: (i32.add +;; BOUNDS-NEXT: (global.get $memory2_byte_offset) +;; BOUNDS-NEXT: (i32.mul +;; BOUNDS-NEXT: (local.get $page_delta) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (global.set $memory3_byte_offset +;; BOUNDS-NEXT: (i32.add +;; BOUNDS-NEXT: (global.get $memory3_byte_offset) +;; BOUNDS-NEXT: (i32.mul +;; BOUNDS-NEXT: (local.get $page_delta) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (local.get $return_size) +;; BOUNDS-NEXT: ) + +;; BOUNDS: (func $memory2_grow (param $page_delta i32) (result i32) +;; BOUNDS-NEXT: (local $return_size i32) +;; BOUNDS-NEXT: (local $memory_size i32) +;; BOUNDS-NEXT: (local.set $return_size +;; BOUNDS-NEXT: (call $memory2_size) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (local.set $memory_size +;; BOUNDS-NEXT: (memory.size) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (if +;; BOUNDS-NEXT: (i32.eq +;; BOUNDS-NEXT: (memory.grow +;; BOUNDS-NEXT: (local.get $page_delta) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (i32.const -1) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (return +;; BOUNDS-NEXT: (i32.const -1) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (memory.copy +;; BOUNDS-NEXT: (i32.add +;; BOUNDS-NEXT: (global.get $memory3_byte_offset) +;; BOUNDS-NEXT: (i32.mul +;; BOUNDS-NEXT: (local.get $page_delta) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (global.get $memory3_byte_offset) +;; BOUNDS-NEXT: (i32.sub +;; BOUNDS-NEXT: (i32.mul +;; BOUNDS-NEXT: (local.get $memory_size) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (global.get $memory3_byte_offset) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (global.set $memory3_byte_offset +;; BOUNDS-NEXT: (i32.add +;; BOUNDS-NEXT: (global.get $memory3_byte_offset) +;; BOUNDS-NEXT: (i32.mul +;; BOUNDS-NEXT: (local.get $page_delta) +;; BOUNDS-NEXT: (i32.const 65536) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (local.get $return_size) +;; BOUNDS-NEXT: ) + +;; BOUNDS: (func $memory3_grow (param $page_delta i32) (result i32) +;; BOUNDS-NEXT: (local $return_size i32) +;; BOUNDS-NEXT: (local.set $return_size +;; BOUNDS-NEXT: (call $memory3_size) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (if +;; BOUNDS-NEXT: (i32.eq +;; BOUNDS-NEXT: (memory.grow +;; BOUNDS-NEXT: (local.get $page_delta) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (i32.const -1) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (return +;; BOUNDS-NEXT: (i32.const -1) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: ) +;; BOUNDS-NEXT: (local.get $return_size) +;; BOUNDS-NEXT: ) |