// Copyright 2018 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 --experimental-wasm-eh load("test/mjsunit/wasm/wasm-constants.js"); load("test/mjsunit/wasm/wasm-module-builder.js"); // Helper function to return a new exported exception with the {kSig_v_v} type // signature from an anonymous module. The underlying module is thrown away. function NewExportedException() { let builder = new WasmModuleBuilder(); let except = builder.addException(kSig_v_v); builder.addExportOfKind("ex", kExternalException, except); let instance = builder.instantiate(); return instance.exports.ex; } // Check that an instance matches an exception thrown by itself, even when the // exception is re-thrown by a regular JavaScript function. (function TestSingleInstance() { print(arguments.callee.name); let builder = new WasmModuleBuilder(); let sig_index = builder.addType(kSig_v_v); let fun = builder.addImport("m", "f", sig_index); let except = builder.addException(kSig_v_v); builder.addFunction("throw", kSig_v_v) .addBody([ kExprThrow, except ]).exportFunc(); builder.addFunction("catch", kSig_v_v) .addBody([ kExprTry, kWasmStmt, kExprCallFunction, fun, kExprCatch, kExprBrOnExn, 0, except, kExprRethrow, kExprEnd, ]).exportFunc(); let ex_obj = new Error("my exception"); let instance = builder.instantiate({ m: { f: function() { throw ex_obj }}}); assertThrows(() => instance.exports.throw(), WebAssembly.RuntimeError); assertThrowsEquals(() => instance.exports.catch(), ex_obj); try { instance.exports.throw(); } catch (e) { ex_obj = e; } assertDoesNotThrow(() => instance.exports.catch()); })(); // Check that two instances distinguish their individual exceptions if they are // not shared, even when declared by the same underlying module. (function TestMultiInstanceNonShared() { print(arguments.callee.name); let builder = new WasmModuleBuilder(); let sig_index = builder.addType(kSig_v_v); let fun = builder.addImport("m", "f", sig_index); let except = builder.addException(kSig_v_v); builder.addFunction("throw", kSig_v_v) .addBody([ kExprThrow, except ]).exportFunc(); builder.addFunction("catch", kSig_v_v) .addBody([ kExprTry, kWasmStmt, kExprCallFunction, fun, kExprCatch, kExprBrOnExn, 0, except, kExprRethrow, kExprEnd, ]).exportFunc(); let ex_obj = new Error("my exception"); let instance1 = builder.instantiate({ m: { f: assertUnreachable }}); let instance2 = builder.instantiate({ m: { f: function() { throw ex_obj }}}); assertThrows(() => instance1.exports.throw(), WebAssembly.RuntimeError); assertThrowsEquals(() => instance2.exports.catch(), ex_obj); try { instance1.exports.throw(); } catch (e) { ex_obj = e; } assertThrowsEquals(() => instance2.exports.catch(), ex_obj); })(); // Check that two instances match their exceptions if they are shared properly, // even if the local exception index of export and import is different. (function TestMultiInstanceShared() { print(arguments.callee.name); let builder = new WasmModuleBuilder(); let sig_index = builder.addType(kSig_v_v); let fun = builder.addImport("m", "f", sig_index); let except1 = builder.addImportedException("m", "ex1", kSig_v_v); let except2 = builder.addException(kSig_v_v); builder.addExportOfKind("ex2", kExternalException, except2); builder.addFunction("throw", kSig_v_v) .addBody([ kExprThrow, except2 ]).exportFunc(); builder.addFunction("catch", kSig_v_v) .addBody([ kExprTry, kWasmStmt, kExprCallFunction, fun, kExprCatch, kExprBrOnExn, 0, except1, kExprRethrow, kExprEnd, ]).exportFunc(); let ex_obj = new Error("my exception"); let instance1 = builder.instantiate({ m: { f: assertUnreachable, ex1: NewExportedException() }}); let instance2 = builder.instantiate({ m: { f: function() { throw ex_obj }, ex1: instance1.exports.ex2 }}); assertThrows(() => instance1.exports.throw(), WebAssembly.RuntimeError); assertThrowsEquals(() => instance2.exports.catch(), ex_obj); try { instance1.exports.throw(); } catch (e) { ex_obj = e; } assertDoesNotThrow(() => instance2.exports.catch()); })(); // Check that two instances based on different modules match their exceptions if // they are shared properly, even if the local exception index is different. (function TestMultiModuleShared() { print(arguments.callee.name); let builder1 = new WasmModuleBuilder(); let except1 = builder1.addException(kSig_v_v); let except2 = builder1.addException(kSig_v_v); builder1.addExportOfKind("ex", kExternalException, except2); builder1.addFunction("throw", kSig_v_v) .addBody([ kExprThrow, except2 ]).exportFunc(); let builder2 = new WasmModuleBuilder(); let sig_index = builder2.addType(kSig_v_v); let fun = builder2.addImport("m", "f", sig_index); let except = builder2.addImportedException("m", "ex", kSig_v_v); builder2.addFunction("catch", kSig_v_v) .addBody([ kExprTry, kWasmStmt, kExprCallFunction, fun, kExprCatch, kExprBrOnExn, 0, except, kExprRethrow, kExprEnd, ]).exportFunc(); let ex_obj = new Error("my exception"); let instance1 = builder1.instantiate(); let instance2 = builder2.instantiate({ m: { f: function() { throw ex_obj }, ex: instance1.exports.ex }}); assertThrows(() => instance1.exports.throw(), WebAssembly.RuntimeError); assertThrowsEquals(() => instance2.exports.catch(), ex_obj); try { instance1.exports.throw(); } catch (e) { ex_obj = e; } assertDoesNotThrow(() => instance2.exports.catch()); })();