Commit bac984e6 authored by Jakob Kummerow's avatar Jakob Kummerow Committed by V8 LUCI CQ

[wasm-gc] Fix roundtrips of JS functions through Wasm

When passing anyref-typed things to Wasm, we cannot expect that
all functions are WasmExternalFunctions. Instead of adding a
relatively expensive type check to such calls, this patch disables
function unwrapping for anyref-typed values.

Fixed: v8:12789
Change-Id: Ied57187bac7fde0326634f7b4fc428ad21dc9c2f
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3605231
Auto-Submit: Jakob Kummerow <jkummerow@chromium.org>
Reviewed-by: 's avatarAndreas Haas <ahaas@chromium.org>
Commit-Queue: Andreas Haas <ahaas@chromium.org>
Cr-Commit-Position: refs/heads/main@{#80179}
parent 755b7d86
...@@ -6493,26 +6493,35 @@ class WasmWrapperGraphBuilder : public WasmGraphBuilder { ...@@ -6493,26 +6493,35 @@ class WasmWrapperGraphBuilder : public WasmGraphBuilder {
Operator::kEliminatable, input, context); Operator::kEliminatable, input, context);
} }
enum UnwrapExternalFunctions : bool {
kUnwrapWasmExternalFunctions = true,
kLeaveFunctionsAlone = false
};
// Assumes {input} has been checked for validity against the target wasm type. // Assumes {input} has been checked for validity against the target wasm type.
// If {input} is a function, returns the WasmInternalFunction associated with // If {input} is a function, returns the WasmInternalFunction associated with
// it. If {input} has the {wasm_wrapped_object_symbol} property, returns the // it. If {input} has the {wasm_wrapped_object_symbol} property, returns the
// value of that property. Otherwise, returns {input}. // value of that property. Otherwise, returns {input}.
Node* BuildUnpackObjectWrapper(Node* input, Node* context) { Node* BuildUnpackObjectWrapper(
auto not_a_function = gasm_->MakeLabel(); Node* input, Node* context,
UnwrapExternalFunctions unwrap_wasm_external_functions) {
auto end = gasm_->MakeLabel(MachineRepresentation::kTaggedPointer); auto end = gasm_->MakeLabel(MachineRepresentation::kTaggedPointer);
gasm_->GotoIfNot(gasm_->HasInstanceType(input, JS_FUNCTION_TYPE), if (unwrap_wasm_external_functions) {
&not_a_function); auto not_a_function = gasm_->MakeLabel();
gasm_->GotoIf(IsSmi(input), &not_a_function);
gasm_->GotoIfNot(gasm_->HasInstanceType(input, JS_FUNCTION_TYPE),
&not_a_function);
Node* function_data = gasm_->LoadFunctionDataFromJSFunction(input); Node* function_data = gasm_->LoadFunctionDataFromJSFunction(input);
// Due to type checking, {function_data} will be a WasmFunctionData. // Due to type checking, {function_data} will be a WasmFunctionData.
Node* internal = gasm_->LoadFromObject( Node* internal = gasm_->LoadFromObject(
MachineType::TaggedPointer(), function_data, MachineType::TaggedPointer(), function_data,
wasm::ObjectAccess::ToTagged(WasmFunctionData::kInternalOffset)); wasm::ObjectAccess::ToTagged(WasmFunctionData::kInternalOffset));
gasm_->Goto(&end, internal); gasm_->Goto(&end, internal);
gasm_->Bind(&not_a_function); gasm_->Bind(&not_a_function);
}
if (!FLAG_wasm_gc_js_interop) { if (!FLAG_wasm_gc_js_interop) {
Node* obj = gasm_->CallBuiltin( Node* obj = gasm_->CallBuiltin(
Builtin::kWasmGetOwnProperty, Operator::kEliminatable, input, Builtin::kWasmGetOwnProperty, Operator::kEliminatable, input,
...@@ -6605,10 +6614,19 @@ class WasmWrapperGraphBuilder : public WasmGraphBuilder { ...@@ -6605,10 +6614,19 @@ class WasmWrapperGraphBuilder : public WasmGraphBuilder {
if (!enabled_features_.has_gc()) return input; if (!enabled_features_.has_gc()) return input;
// If this is a wrapper for arrays/structs/i31s, unpack it. // If this is a wrapper for arrays/structs/i31s, unpack it.
// TODO(7748): Update this when JS interop has settled. // TODO(7748): Update this when JS interop has settled.
return BuildUnpackObjectWrapper(input, js_context); // We prefer not to unwrap functions here, because deciding whether
// that's valid for a given function is expensive (specifically,
// it's not valid for plain JS functions, because they don't have
// a WasmFunctionData/WasmInternalFunction inside). The consequence
// is that passing a funcref to JS and taking it back as {anyref}
// turns it into an opaque pointer that can't be cast back to
// {funcref}.
return BuildUnpackObjectWrapper(input, js_context,
kLeaveFunctionsAlone);
case wasm::HeapType::kFunc: case wasm::HeapType::kFunc:
BuildCheckValidRefValue(input, js_context, type); BuildCheckValidRefValue(input, js_context, type);
return BuildUnpackObjectWrapper(input, js_context); return BuildUnpackObjectWrapper(input, js_context,
kUnwrapWasmExternalFunctions);
case wasm::HeapType::kData: case wasm::HeapType::kData:
case wasm::HeapType::kArray: case wasm::HeapType::kArray:
case wasm::HeapType::kEq: case wasm::HeapType::kEq:
...@@ -6617,11 +6635,15 @@ class WasmWrapperGraphBuilder : public WasmGraphBuilder { ...@@ -6617,11 +6635,15 @@ class WasmWrapperGraphBuilder : public WasmGraphBuilder {
BuildCheckValidRefValue(input, js_context, type); BuildCheckValidRefValue(input, js_context, type);
// This will just return {input} if the object is not wrapped, i.e. // This will just return {input} if the object is not wrapped, i.e.
// if it is null (given the check just above). // if it is null (given the check just above).
return BuildUnpackObjectWrapper(input, js_context); // Skip function unpacking here to save code size (they can't occur
// anyway).
return BuildUnpackObjectWrapper(input, js_context,
kLeaveFunctionsAlone);
default: default:
if (module_->has_signature(type.ref_index())) { if (module_->has_signature(type.ref_index())) {
BuildCheckValidRefValue(input, js_context, type); BuildCheckValidRefValue(input, js_context, type);
return BuildUnpackObjectWrapper(input, js_context); return BuildUnpackObjectWrapper(input, js_context,
kUnwrapWasmExternalFunctions);
} }
// If this is reached, then IsJSCompatibleSignature() is too // If this is reached, then IsJSCompatibleSignature() is too
// permissive. // permissive.
......
// Copyright 2022 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: --experimental-wasm-gc --wasm-gc-js-interop
d8.file.execute("test/mjsunit/wasm/wasm-module-builder.js");
var instance = (function () {
var builder = new WasmModuleBuilder();
let struct_index = builder.addStruct([makeField(kWasmI32, true)]);
let callback = builder.addImport(
'import', 'callback', {params: [kWasmAnyRef], results: []});
builder.addFunction("object", { params: [], results: [kWasmEqRef] })
.addBody([kGCPrefix, kExprStructNewDefault, struct_index]).exportFunc();
builder.addFunction(
'roundtrip', {params: [kWasmEqRef, kWasmAnyRef], results: []})
.addBody([
kExprLocalGet, 1,
kExprCallFunction, callback,
])
.exportFunc();
return builder.instantiate({
import: {
callback: function(f) {
assertEquals("function", typeof f);
}
}
});
})();
var c = instance.exports.object();
instance.exports.roundtrip(c, () => 12);
instance.exports.roundtrip(34, () => 56);
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment