blob: 8bb1018825e0d1d87811e0642dba714ebcc1b2af [file] [log] [blame]
// Copyright 2017 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Flags: --expose-wasm --stress-compaction
load('test/mjsunit/wasm/wasm-constants.js');
load('test/mjsunit/wasm/wasm-module-builder.js');
let initialPages = 1;
let maximumPages = 6;
function generateBuilder() {
let builder = new WasmModuleBuilder();
builder.addMemory(initialPages, maximumPages, true);
builder.addFunction('store', kSig_i_ii)
.addBody([
kExprGetLocal, 0, kExprGetLocal, 1, kExprI32StoreMem, 0, 0,
kExprGetLocal, 1
])
.exportFunc();
return builder;
}
// This test verifies that the effects of growing memory inside a loop
// affect the result of current_memory when the loop is over.
(function TestGrowMemoryInsideLoop() {
print('TestGrowMemoryInsideLoop ...');
let deltaPages = 1;
let builder = generateBuilder();
builder.addFunction('main', kSig_i_i)
.addBody([
// clang-format off
kExprLoop, kWasmStmt, // while
kExprGetLocal, 0, // -
kExprIf, kWasmStmt, // if <param0> != 0
// Grow memory.
kExprI32Const, deltaPages, // -
kExprGrowMemory, kMemoryZero, // grow memory
kExprDrop, // drop the result of grow
// Decrease loop variable.
kExprGetLocal, 0, // -
kExprI32Const, 1, // -
kExprI32Sub, // -
kExprSetLocal, 0, // decrease <param0>
kExprBr, 1, // continue
kExprEnd, // end if
kExprEnd, // end loop
// Return the memory size.
kExprMemorySize, kMemoryZero // put memory size on stack
// clang-format on
])
.exportFunc();
{
// Avoid the loop branch (not growing memory).
let instance = builder.instantiate();
let iterations = 0;
let expectedPages = initialPages + iterations * deltaPages;
assertTrue(expectedPages <= maximumPages);
assertEquals(expectedPages, instance.exports.main(iterations));
}
{
// Enter the loop branch (growing memory).
let instance = builder.instantiate();
let iterations = 2;
let expectedPages = initialPages + iterations * deltaPages;
assertTrue(expectedPages <= maximumPages);
assertEquals(expectedPages, instance.exports.main(iterations));
}
})();
// This test verifies that a loop does not affect the result of current_memory
// when the memory is grown both inside and outside the loop.
(function TestGrowMemoryInsideAndOutsideLoop() {
print('TestGrowMemoryInsideAndOutsideLoop ...');
let deltaPagesIn = 1;
let deltaPagesOut = 2;
let builder = generateBuilder();
builder.addFunction('main', kSig_i_i)
.addBody([
// clang-format off
// Grow memory.
kExprI32Const, deltaPagesOut, // -
kExprGrowMemory, kMemoryZero, // grow memory
kExprDrop, // drop the result of grow
kExprLoop, kWasmStmt, // while
kExprGetLocal, 0, // -
kExprIf, kWasmStmt, // if <param0> != 0
// Grow memory.
kExprI32Const, deltaPagesIn, // -
kExprGrowMemory, kMemoryZero, // grow memory
kExprDrop, // drop the result of grow
// Decrease loop variable.
kExprGetLocal, 0, // -
kExprI32Const, 1, // -
kExprI32Sub, // -
kExprSetLocal, 0, // decrease <param0>
kExprBr, 1, // continue
kExprEnd, // end if
kExprEnd, // end loop
// Return memory size.
kExprMemorySize, kMemoryZero // put memory size on stack
// clang-format on
])
.exportFunc();
{
// Avoid the loop branch (grow memory by deltaPagesOut).
let instance = builder.instantiate();
let iterations = 0;
let expectedPages = initialPages + deltaPagesOut;
assertTrue(expectedPages <= maximumPages);
assertEquals(expectedPages, instance.exports.main(iterations));
}
{
// Avoid the loop branch (grow memory by deltaPagesOut
// + iterations * deltaPagesIn).
let instance = builder.instantiate();
let iterations = 3;
let expectedPages =
initialPages + deltaPagesOut + (iterations * deltaPagesIn);
assertTrue(expectedPages <= maximumPages);
assertEquals(expectedPages, instance.exports.main(iterations));
}
})();
// This test verifies that the effects of writing to memory grown inside a loop
// are retained when the loop is over.
(function TestGrowMemoryAndStoreInsideLoop() {
print('TestGrowMemoryAndStoreInsideLoop ...');
let deltaPages = 1;
let builder = generateBuilder();
builder.addFunction('main', kSig_i_ii)
.addBody([
// clang-format off
kExprLoop, kWasmStmt, // while
kExprGetLocal, 0, // -
kExprIf, kWasmStmt, // if <param0> != 0
// Grow memory.
kExprI32Const, deltaPages, // -
kExprGrowMemory, kMemoryZero, // grow memory
kExprDrop, // drop the result of grow
// Increase counter in memory.
kExprGetLocal, 1, // put index (for store)
kExprGetLocal, 1, // put index (for load)
kExprI32LoadMem, 0, 0, // load from grown memory
kExprI32Const, 1, // -
kExprI32Add, // increase counter
kExprI32StoreMem, 0, 0, // store counter in memory
// Decrease loop variable.
kExprGetLocal, 0, // -
kExprI32Const, 1, // -
kExprI32Sub, // -
kExprSetLocal, 0, // decrease <param0>
kExprBr, 1, // continue
kExprEnd, // end if
kExprEnd, // end loop
// Increase counter in memory.
kExprGetLocal, 1, // -
kExprI32LoadMem, 0, 0 // load from grown memory
// clang-format on
])
.exportFunc();
let index = 0;
let initialValue = 1;
{
// Avoid the loop (not growing memory).
let instance = builder.instantiate();
let iterations = 0;
let expectedValue = initialValue + iterations;
instance.exports.store(index, initialValue);
assertEquals(expectedValue, instance.exports.main(iterations, index));
}
{
// Enter the loop (growing memory + increasing counter in grown memory).
let instance = builder.instantiate();
let iterations = 2;
let expectedValue = initialValue + iterations;
instance.exports.store(index, initialValue);
assertEquals(expectedValue, instance.exports.main(iterations, index));
}
})();
// This test verifies that a loop does not affect the memory when the
// memory is grown both inside and outside the loop.
(function TestGrowMemoryAndStoreInsideAndOutsideLoop() {
print('TestGrowMemoryAndStoreInsideAndOutsideLoop ...');
let deltaPagesIn = 1;
let deltaPagesOut = 2;
let builder = generateBuilder();
builder.addFunction('main', kSig_i_ii)
.addBody([
// clang-format off
// Grow memory.
kExprI32Const, deltaPagesOut, // -
kExprGrowMemory, kMemoryZero, // grow memory
kExprDrop, // drop the result of grow
// Increase counter in memory.
kExprGetLocal, 1, // put index (for store)
kExprGetLocal, 1, // put index (for load)
kExprI32LoadMem, 0, 0, // load from grown memory
kExprI32Const, 1, // -
kExprI32Add, // increase value on stack
kExprI32StoreMem, 0, 0, // store new value
// Start loop.
kExprLoop, kWasmStmt, // while
kExprGetLocal, 0, // -
kExprIf, kWasmStmt, // if <param0> != 0
// Grow memory.
kExprI32Const, deltaPagesIn, // -
kExprGrowMemory, kMemoryZero, // grow memory
kExprDrop, // drop the result of grow
// Increase counter in memory.
kExprGetLocal, 1, // put index (for store)
kExprGetLocal, 1, // put index (for load)
kExprI32LoadMem, 0, 0, // load from grown memory
kExprI32Const, 1, // -
kExprI32Add, // increase value on stack
kExprI32StoreMem, 0, 0, // store new value
// Decrease loop variable.
kExprGetLocal, 0, // -
kExprI32Const, 1, // -
kExprI32Sub, // -
kExprSetLocal, 0, // decrease <param0>
kExprBr, 1, // continue
kExprEnd, // end if
kExprEnd, // end loop
// Return counter from memory.
kExprGetLocal, 1, // put index on stack
kExprI32LoadMem, 0, 0 // load from grown memory
// clang-format on
])
.exportFunc();
let index = 0;
let initialValue = 1;
{
// Avoid the loop (grow memory and increment counter only outside the loop).
let instance = builder.instantiate();
let iterations = 0;
let expectedValue = initialValue + 1;
instance.exports.store(index, initialValue);
assertEquals(expectedValue, instance.exports.main(iterations, index));
}
{
// Enter the loop (grow memory and increment counter outside/inside loop).
let instance = builder.instantiate();
let iterations = 3;
let expectedValue = initialValue + iterations + 1;
instance.exports.store(index, initialValue);
assertEquals(expectedValue, instance.exports.main(iterations, index));
}
})();