Commit 311808ee authored by Théotime Grohens's avatar Théotime Grohens Committed by Commit Bot

[dataview] Deopt in TurboFan instead of raising exceptions

This CL simplifies the implementation of inlined DataView
methods in TurboFan. It removes the explicit exception handling,
and just deopts and relies on the baseline code to handle
exceptions instead.

It also adapts the DataView test files in mjsunit/compiler/
accordingly.

Change-Id: I013c76970e1480df2b755d17d397bd0f9f26f0ec
Reviewed-on: https://chromium-review.googlesource.com/1148207
Commit-Queue: Théotime Grohens <theotime@google.com>
Reviewed-by: 's avatarBenedikt Meurer <bmeurer@chromium.org>
Reviewed-by: 's avatarJaroslav Sevcik <jarin@chromium.org>
Cr-Commit-Position: refs/heads/master@{#54648}
parent e0670b22
......@@ -6650,11 +6650,12 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet(
Node* control = NodeProperties::GetControlInput(node);
Node* receiver = NodeProperties::GetValueInput(node, 1);
Node* context = NodeProperties::GetContextInput(node);
Node* frame_state = NodeProperties::GetFrameStateInput(node);
CallParameters const& p = CallParametersOf(node->op());
if (p.speculation_mode() == SpeculationMode::kDisallowSpeculation) {
return NoChange();
}
Node* offset = node->op()->ValueInputCount() > 2
? NodeProperties::GetValueInput(node, 2)
: jsgraph()->ZeroConstant();
......@@ -6674,7 +6675,7 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet(
jsgraph()->ZeroConstant(), offset);
effect = graph()->NewNode(
simplified()->CheckIf(DeoptimizeReason::kNotASmi, p.feedback()),
simplified()->CheckIf(DeoptimizeReason::kOutOfBounds, p.feedback()),
is_positive, effect, control);
// Coerce {is_little_endian} to boolean.
......@@ -6688,37 +6689,25 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet(
Node* check_neutered = effect = graph()->NewNode(
simplified()->ArrayBufferWasNeutered(), buffer, effect, control);
Node* branch_neutered = graph()->NewNode(
common()->Branch(BranchHint::kFalse), check_neutered, control);
check_neutered =
graph()->NewNode(simplified()->BooleanNot(), check_neutered);
// Raise an error if it was neuteured.
Node* if_true_neutered =
graph()->NewNode(common()->IfTrue(), branch_neutered);
Node* etrue_neutered = effect;
{
if_true_neutered = etrue_neutered = graph()->NewNode(
javascript()->CallRuntime(Runtime::kThrowTypeError, 2),
jsgraph()->Constant(MessageTemplate::kDetachedOperation),
jsgraph()->HeapConstant(
factory()->NewStringFromAsciiChecked("DataView.prototype.get")),
context, frame_state, etrue_neutered, if_true_neutered);
}
// Otherwise, proceed.
Node* if_false_neutered =
graph()->NewNode(common()->IfFalse(), branch_neutered);
Node* efalse_neutered = effect;
// If the buffer was neutered, deopt and let the unoptimized code throw.
effect = graph()->NewNode(
simplified()->CheckIf(DeoptimizeReason::kArrayBufferWasNeutered,
p.feedback()),
check_neutered, effect, control);
// Get the byte offset and byte length of the {receiver}.
Node* byte_offset = efalse_neutered =
Node* byte_offset = effect =
graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSArrayBufferViewByteOffset()),
receiver, efalse_neutered, if_false_neutered);
receiver, effect, control);
Node* byte_length = efalse_neutered =
Node* byte_length = effect =
graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSArrayBufferViewByteLength()),
receiver, efalse_neutered, if_false_neutered);
receiver, effect, control);
// The end offset is the offset plus the element size
// of the type that we want to load.
......@@ -6726,100 +6715,49 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet(
Node* end_offset = graph()->NewNode(simplified()->NumberAdd(), offset,
jsgraph()->Constant(element_size));
// We need to check that {end_offset} <= {byte_length}, ie
// throw a RangeError if {byte_length} < {end_offset}.
Node* check_range = graph()->NewNode(simplified()->NumberLessThan(),
byte_length, end_offset);
Node* branch_range = graph()->NewNode(common()->Branch(BranchHint::kFalse),
check_range, if_false_neutered);
Node* if_true_range = graph()->NewNode(common()->IfTrue(), branch_range);
Node* etrue_range = efalse_neutered;
{
if_true_range = etrue_range = graph()->NewNode(
javascript()->CallRuntime(Runtime::kThrowRangeError, 2),
jsgraph()->Constant(MessageTemplate::kInvalidDataViewAccessorOffset),
jsgraph()->HeapConstant(
factory()->NewStringFromAsciiChecked("DataView.prototype.get")),
context, frame_state, etrue_range, if_true_range);
}
// We need to check that {end_offset} <= {byte_length}.
Node* check_bounds = graph()->NewNode(simplified()->NumberLessThanOrEqual(),
end_offset, byte_length);
Node* if_false_range = graph()->NewNode(common()->IfFalse(), branch_range);
Node* efalse_range = efalse_neutered;
Node* vfalse_range;
{
// Get the buffer's backing store.
Node* backing_store = efalse_range =
graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSArrayBufferBackingStore()),
buffer, efalse_range, if_false_range);
// Compute the buffer index at which we'll read.
Node* buffer_index =
graph()->NewNode(simplified()->NumberAdd(), offset, byte_offset);
// Perform the load.
vfalse_range = efalse_range =
graph()->NewNode(simplified()->LoadDataViewElement(element_type),
buffer, backing_store, buffer_index,
is_little_endian, efalse_range, if_false_range);
}
// Rewire potential exception edges.
Node* on_exception = nullptr;
if (NodeProperties::IsExceptionalCall(node, &on_exception)) {
// Create appropriate {IfException} and {IfSuccess} nodes.
Node* extrue_neutered = graph()->NewNode(
common()->IfException(), etrue_neutered,
if_true_neutered); // We threw because the array was neutered.
if_true_neutered =
graph()->NewNode(common()->IfSuccess(), if_true_neutered);
Node* extrue_range =
graph()->NewNode(common()->IfException(), etrue_range,
if_true_range); // We threw because out of bounds.
if_true_range = graph()->NewNode(common()->IfSuccess(), if_true_range);
// Also deopt and let the unoptimized code throw in this case.
effect = graph()->NewNode(
simplified()->CheckIf(DeoptimizeReason::kOutOfBounds, p.feedback()),
check_bounds, effect, control);
// We can't throw in LoadDataViewElement(),
// so we don't need to handle that path here.
// Get the buffer's backing store.
Node* backing_store = effect = graph()->NewNode(
simplified()->LoadField(AccessBuilder::ForJSArrayBufferBackingStore()),
buffer, effect, control);
// Join the exception edges.
Node* merge =
graph()->NewNode(common()->Merge(2), extrue_neutered, extrue_range);
Node* ephi = graph()->NewNode(common()->EffectPhi(2), extrue_neutered,
extrue_range, merge);
Node* phi =
graph()->NewNode(common()->Phi(MachineRepresentation::kTagged, 2),
extrue_neutered, extrue_range, merge);
ReplaceWithValue(on_exception, phi, ephi, merge);
}
// Compute the buffer index at which we'll read.
Node* buffer_index =
graph()->NewNode(simplified()->NumberAdd(), offset, byte_offset);
// Connect the throwing paths to end.
if_true_neutered =
graph()->NewNode(common()->Throw(), etrue_neutered, if_true_neutered);
NodeProperties::MergeControlToEnd(graph(), common(), if_true_neutered);
if_true_range =
graph()->NewNode(common()->Throw(), etrue_range, if_true_range);
NodeProperties::MergeControlToEnd(graph(), common(), if_true_range);
// Perform the load.
Node* value = effect = graph()->NewNode(
simplified()->LoadDataViewElement(element_type), buffer, backing_store,
buffer_index, is_little_endian, effect, control);
// Continue on the regular path.
ReplaceWithValue(node, vfalse_range, efalse_range, if_false_range);
return Changed(vfalse_range);
ReplaceWithValue(node, value, effect, control);
return Changed(value);
}
return NoChange();
}
Reduction JSCallReducer::ReduceDataViewPrototypeSet(
Node* node, ExternalArrayType element_type) {
Node* effect = NodeProperties::GetEffectInput(node);
Node* control = NodeProperties::GetControlInput(node);
Node* receiver = NodeProperties::GetValueInput(node, 1);
Node* context = NodeProperties::GetContextInput(node);
Node* frame_state = NodeProperties::GetFrameStateInput(node);
CallParameters const& p = CallParametersOf(node->op());
if (p.speculation_mode() == SpeculationMode::kDisallowSpeculation) {
return NoChange();
}
Node* offset = node->op()->ValueInputCount() > 2
? NodeProperties::GetValueInput(node, 2)
: jsgraph()->ZeroConstant();
......@@ -6843,7 +6781,7 @@ Reduction JSCallReducer::ReduceDataViewPrototypeSet(
jsgraph()->ZeroConstant(), offset);
effect = graph()->NewNode(
simplified()->CheckIf(DeoptimizeReason::kNotASmi, p.feedback()),
simplified()->CheckIf(DeoptimizeReason::kOutOfBounds, p.feedback()),
is_positive, effect, control);
// Coerce {is_little_endian} to boolean.
......@@ -6863,37 +6801,25 @@ Reduction JSCallReducer::ReduceDataViewPrototypeSet(
Node* check_neutered = effect = graph()->NewNode(
simplified()->ArrayBufferWasNeutered(), buffer, effect, control);
Node* branch_neutered = graph()->NewNode(
common()->Branch(BranchHint::kFalse), check_neutered, control);
check_neutered =
graph()->NewNode(simplified()->BooleanNot(), check_neutered);
// Raise an error if it was neuteured.
Node* if_true_neutered =
graph()->NewNode(common()->IfTrue(), branch_neutered);
Node* etrue_neutered = effect;
{
if_true_neutered = etrue_neutered = graph()->NewNode(
javascript()->CallRuntime(Runtime::kThrowTypeError, 2),
jsgraph()->Constant(MessageTemplate::kDetachedOperation),
jsgraph()->HeapConstant(
factory()->NewStringFromAsciiChecked("DataView.prototype.set")),
context, frame_state, etrue_neutered, if_true_neutered);
}
// Otherwise, proceed.
Node* if_false_neutered =
graph()->NewNode(common()->IfFalse(), branch_neutered);
Node* efalse_neutered = effect;
// If the buffer was neutered, deopt and let the unoptimized code throw.
effect = graph()->NewNode(
simplified()->CheckIf(DeoptimizeReason::kArrayBufferWasNeutered,
p.feedback()),
check_neutered, effect, control);
// Get the byte offset and byte length of the {receiver}.
Node* byte_offset = efalse_neutered =
Node* byte_offset = effect =
graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSArrayBufferViewByteOffset()),
receiver, efalse_neutered, if_false_neutered);
receiver, effect, control);
Node* byte_length = efalse_neutered =
Node* byte_length = effect =
graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSArrayBufferViewByteLength()),
receiver, efalse_neutered, if_false_neutered);
receiver, effect, control);
// The end offset is the offset plus the element size
// of the type that we want to store.
......@@ -6901,85 +6827,34 @@ Reduction JSCallReducer::ReduceDataViewPrototypeSet(
Node* end_offset = graph()->NewNode(simplified()->NumberAdd(), offset,
jsgraph()->Constant(element_size));
// We need to check that {end_offset} <= {byte_length}, ie
// throw a RangeError if {byte_length} < {end_offset}.
Node* check_range = graph()->NewNode(simplified()->NumberLessThan(),
byte_length, end_offset);
Node* branch_range = graph()->NewNode(common()->Branch(BranchHint::kFalse),
check_range, if_false_neutered);
Node* if_true_range = graph()->NewNode(common()->IfTrue(), branch_range);
Node* etrue_range = efalse_neutered;
{
if_true_range = etrue_range = graph()->NewNode(
javascript()->CallRuntime(Runtime::kThrowRangeError, 2),
jsgraph()->Constant(MessageTemplate::kInvalidDataViewAccessorOffset),
jsgraph()->HeapConstant(
factory()->NewStringFromAsciiChecked("DataView.prototype.set")),
context, frame_state, etrue_range, if_true_range);
}
Node* if_false_range = graph()->NewNode(common()->IfFalse(), branch_range);
Node* efalse_range = efalse_neutered;
Node* vfalse_range = jsgraph()->UndefinedConstant(); // Return value.
{
// Get the buffer's backing store.
Node* backing_store = efalse_range =
graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSArrayBufferBackingStore()),
buffer, efalse_range, if_false_range);
// Compute the buffer index at which we'll write.
Node* buffer_index =
graph()->NewNode(simplified()->NumberAdd(), offset, byte_offset);
// Perform the store.
efalse_range =
graph()->NewNode(simplified()->StoreDataViewElement(element_type),
buffer, backing_store, buffer_index, value,
is_little_endian, efalse_range, if_false_range);
}
// We need to check that {end_offset} <= {byte_length}.
Node* check_bounds = graph()->NewNode(simplified()->NumberLessThanOrEqual(),
end_offset, byte_length);
// Rewire potential exception edges.
Node* on_exception = nullptr;
if (NodeProperties::IsExceptionalCall(node, &on_exception)) {
// Create appropriate {IfException} and {IfSuccess} nodes.
Node* extrue_neutered = graph()->NewNode(
common()->IfException(), etrue_neutered,
if_true_neutered); // We threw because the array was neutered.
if_true_neutered =
graph()->NewNode(common()->IfSuccess(), if_true_neutered);
// Also deopt and let the unoptimized code throw in this case.
effect = graph()->NewNode(
simplified()->CheckIf(DeoptimizeReason::kOutOfBounds, p.feedback()),
check_bounds, effect, control);
Node* extrue_range =
graph()->NewNode(common()->IfException(), etrue_range,
if_true_range); // We threw because out of bounds.
if_true_range = graph()->NewNode(common()->IfSuccess(), if_true_range);
// Get the buffer's backing store.
Node* backing_store = effect = graph()->NewNode(
simplified()->LoadField(AccessBuilder::ForJSArrayBufferBackingStore()),
buffer, effect, control);
// We can't throw in StoreDataViewElement(),
// so we don't need to handle that path here.
// Compute the buffer index at which we'll write.
Node* buffer_index =
graph()->NewNode(simplified()->NumberAdd(), offset, byte_offset);
// Join the exception edges.
Node* merge =
graph()->NewNode(common()->Merge(2), extrue_neutered, extrue_range);
Node* ephi = graph()->NewNode(common()->EffectPhi(2), extrue_neutered,
extrue_range, merge);
Node* phi =
graph()->NewNode(common()->Phi(MachineRepresentation::kTagged, 2),
extrue_neutered, extrue_range, merge);
ReplaceWithValue(on_exception, phi, ephi, merge);
}
// Perform the store.
effect = graph()->NewNode(simplified()->StoreDataViewElement(element_type),
buffer, backing_store, buffer_index, value,
is_little_endian, effect, control);
// Connect the throwing paths to end.
if_true_neutered =
graph()->NewNode(common()->Throw(), etrue_neutered, if_true_neutered);
NodeProperties::MergeControlToEnd(graph(), common(), if_true_neutered);
if_true_range =
graph()->NewNode(common()->Throw(), etrue_range, if_true_range);
NodeProperties::MergeControlToEnd(graph(), common(), if_true_range);
Node* value = jsgraph()->UndefinedConstant();
// Continue on the regular path.
ReplaceWithValue(node, vfalse_range, efalse_range, if_false_range);
return Changed(vfalse_range);
ReplaceWithValue(node, value, effect, control);
return Changed(value);
}
return NoChange();
......
// 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: --allow-natives-syntax --opt --no-always-opt
// Check that there are no deopt loops for DataView methods.
var buffer = new ArrayBuffer(64);
var dataview = new DataView(buffer, 8, 24);
// Check DataView getters.
function readUint8(offset) {
return dataview.getUint8(offset);
}
function warmupRead(f) {
f(0);
f(1);
%OptimizeFunctionOnNextCall(f);
f(2);
f(3);
}
warmupRead(readUint8);
assertOptimized(readUint8);
readUint8(0.5); // Deopts.
assertUnoptimized(readUint8);
warmupRead(readUint8);
assertOptimized(readUint8);
readUint8(1.5); // Doesn't deopt because getUint8 didn't get inlined this time.
assertOptimized(readUint8);
// Check DataView setters.
function writeUint8(offset, value) {
dataview.setUint8(offset, value);
}
function warmupWrite(f) {
f(0, 0);
f(0, 1);
%OptimizeFunctionOnNextCall(f);
f(0, 2);
f(0, 3);
}
warmupWrite(writeUint8);
assertOptimized(writeUint8);
writeUint8(0.5, 0); // Deopts.
assertUnoptimized(writeUint8);
warmupWrite(writeUint8);
assertOptimized(writeUint8);
writeUint8(1.5, 0); // Doesn't deopt.
assertOptimized(writeUint8);
......@@ -131,43 +131,56 @@ assertEquals(b4, readFloat64(16));
dataview.setFloat64(16, b4, true);
assertEquals(b4, readFloat64(16, true));
// TurboFan out of bounds read, throw with exception handler.
// TurboFan out of bounds reads deopt.
assertOptimized(readInt8Handled);
assertInstanceof(readInt8Handled(24), RangeError);
assertOptimized(readInt8Handled);
assertUnoptimized(readInt8Handled);
assertOptimized(readInt16Handled);
assertInstanceof(readInt16Handled(23), RangeError);
assertOptimized(readInt16Handled);
assertUnoptimized(readInt16Handled);
assertOptimized(readInt32Handled);
assertInstanceof(readInt32Handled(21), RangeError);
assertOptimized(readInt32Handled);
assertUnoptimized(readInt32Handled);
// Without exception handler.
assertOptimized(readUint8);
assertThrows(() => readUint8(24));
assertOptimized(readUint8);
assertUnoptimized(readUint8);
assertOptimized(readFloat32);
assertThrows(() => readFloat32(21));
assertOptimized(readFloat32);
assertUnoptimized(readFloat32);
assertOptimized(readFloat64);
assertThrows(() => readFloat64(17));
assertOptimized(readFloat64);
assertUnoptimized(readFloat64);
// TurboFan deoptimizations.
assertOptimized(readInt8Handled);
assertInstanceof(readInt8Handled(-1), RangeError); // Negative Smi deopts.
assertUnoptimized(readInt8Handled);
warmup(readInt8Handled);
assertOptimized(readInt8Handled);
assertEquals(values[3], readInt8Handled(3.14)); // Non-Smi index deopts.
assertUnoptimized(readInt8Handled);
// TurboFan neutered buffer.
warmup(readInt8Handled);
assertOptimized(readInt8Handled);
%ArrayBufferNeuter(buffer);
assertInstanceof(readInt8Handled(0), TypeError);
assertOptimized(readInt8Handled);
// Negative Smi deopts.
(function() {
function readInt8Handled(offset) {
try { return dataview.getInt8(offset); } catch (e) { return e; }
}
warmup(readInt8Handled);
assertOptimized(readInt8Handled);
assertInstanceof(readInt8Handled(-1), RangeError);
assertUnoptimized(readInt8Handled);
})();
// Non-Smi index deopts.
(function() {
function readUint8(offset) { return dataview.getUint8(offset); }
warmup(readUint8);
assertOptimized(readUint8);
assertEquals(values[3], readUint8(3.14));
assertUnoptimized(readUint8);
})();
// TurboFan neutered buffer deopts.
(function() {
function readInt8Handled(offset) {
try { return dataview.getInt8(offset); } catch (e) { return e; }
}
warmup(readInt8Handled);
%ArrayBufferNeuter(buffer);
assertOptimized(readInt8Handled);
assertInstanceof(readInt8Handled(0), TypeError);
assertUnoptimized(readInt8Handled);
})();
......@@ -117,15 +117,15 @@ assertEquals(b4, dataview.getFloat64(8));
writeFloat64(8, b4, true);
assertEquals(b4, dataview.getFloat64(8, true));
// TurboFan out of bounds read, throw with exception handler.
// TurboFan out of bounds read, deopt.
assertOptimized(writeInt8Handled);
assertInstanceof(writeInt8Handled(24, 0), RangeError);
assertOptimized(writeInt8Handled);
assertUnoptimized(writeInt8Handled);
// Without exception handler.
// Without exception handler, deopt too.
assertOptimized(writeUint8);
assertThrows(() => writeUint8(24, 0));
assertOptimized(writeUint8);
assertUnoptimized(writeUint8);
// None of the stores wrote out of bounds.
var bytes = new Uint8Array(buffer);
......
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