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( ...@@ -6650,11 +6650,12 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet(
Node* control = NodeProperties::GetControlInput(node); Node* control = NodeProperties::GetControlInput(node);
Node* receiver = NodeProperties::GetValueInput(node, 1); Node* receiver = NodeProperties::GetValueInput(node, 1);
Node* context = NodeProperties::GetContextInput(node);
Node* frame_state = NodeProperties::GetFrameStateInput(node);
CallParameters const& p = CallParametersOf(node->op()); CallParameters const& p = CallParametersOf(node->op());
if (p.speculation_mode() == SpeculationMode::kDisallowSpeculation) {
return NoChange();
}
Node* offset = node->op()->ValueInputCount() > 2 Node* offset = node->op()->ValueInputCount() > 2
? NodeProperties::GetValueInput(node, 2) ? NodeProperties::GetValueInput(node, 2)
: jsgraph()->ZeroConstant(); : jsgraph()->ZeroConstant();
...@@ -6674,7 +6675,7 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet( ...@@ -6674,7 +6675,7 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet(
jsgraph()->ZeroConstant(), offset); jsgraph()->ZeroConstant(), offset);
effect = graph()->NewNode( effect = graph()->NewNode(
simplified()->CheckIf(DeoptimizeReason::kNotASmi, p.feedback()), simplified()->CheckIf(DeoptimizeReason::kOutOfBounds, p.feedback()),
is_positive, effect, control); is_positive, effect, control);
// Coerce {is_little_endian} to boolean. // Coerce {is_little_endian} to boolean.
...@@ -6688,37 +6689,25 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet( ...@@ -6688,37 +6689,25 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet(
Node* check_neutered = effect = graph()->NewNode( Node* check_neutered = effect = graph()->NewNode(
simplified()->ArrayBufferWasNeutered(), buffer, effect, control); simplified()->ArrayBufferWasNeutered(), buffer, effect, control);
Node* branch_neutered = graph()->NewNode( check_neutered =
common()->Branch(BranchHint::kFalse), check_neutered, control); graph()->NewNode(simplified()->BooleanNot(), check_neutered);
// Raise an error if it was neuteured. // If the buffer was neutered, deopt and let the unoptimized code throw.
Node* if_true_neutered = effect = graph()->NewNode(
graph()->NewNode(common()->IfTrue(), branch_neutered); simplified()->CheckIf(DeoptimizeReason::kArrayBufferWasNeutered,
Node* etrue_neutered = effect; p.feedback()),
{ check_neutered, effect, control);
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;
// Get the byte offset and byte length of the {receiver}. // Get the byte offset and byte length of the {receiver}.
Node* byte_offset = efalse_neutered = Node* byte_offset = effect =
graph()->NewNode(simplified()->LoadField( graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSArrayBufferViewByteOffset()), AccessBuilder::ForJSArrayBufferViewByteOffset()),
receiver, efalse_neutered, if_false_neutered); receiver, effect, control);
Node* byte_length = efalse_neutered = Node* byte_length = effect =
graph()->NewNode(simplified()->LoadField( graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSArrayBufferViewByteLength()), AccessBuilder::ForJSArrayBufferViewByteLength()),
receiver, efalse_neutered, if_false_neutered); receiver, effect, control);
// The end offset is the offset plus the element size // The end offset is the offset plus the element size
// of the type that we want to load. // of the type that we want to load.
...@@ -6726,100 +6715,49 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet( ...@@ -6726,100 +6715,49 @@ Reduction JSCallReducer::ReduceDataViewPrototypeGet(
Node* end_offset = graph()->NewNode(simplified()->NumberAdd(), offset, Node* end_offset = graph()->NewNode(simplified()->NumberAdd(), offset,
jsgraph()->Constant(element_size)); jsgraph()->Constant(element_size));
// We need to check that {end_offset} <= {byte_length}, ie // We need to check that {end_offset} <= {byte_length}.
// throw a RangeError if {byte_length} < {end_offset}. Node* check_bounds = graph()->NewNode(simplified()->NumberLessThanOrEqual(),
Node* check_range = graph()->NewNode(simplified()->NumberLessThan(), end_offset, byte_length);
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); // Also deopt and let the unoptimized code throw in this case.
Node* etrue_range = efalse_neutered; effect = graph()->NewNode(
{ simplified()->CheckIf(DeoptimizeReason::kOutOfBounds, p.feedback()),
if_true_range = etrue_range = graph()->NewNode( check_bounds, effect, control);
javascript()->CallRuntime(Runtime::kThrowRangeError, 2),
jsgraph()->Constant(MessageTemplate::kInvalidDataViewAccessorOffset),
jsgraph()->HeapConstant(
factory()->NewStringFromAsciiChecked("DataView.prototype.get")),
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;
{
// Get the buffer's backing store. // Get the buffer's backing store.
Node* backing_store = efalse_range = Node* backing_store = effect = graph()->NewNode(
graph()->NewNode(simplified()->LoadField( simplified()->LoadField(AccessBuilder::ForJSArrayBufferBackingStore()),
AccessBuilder::ForJSArrayBufferBackingStore()), buffer, effect, control);
buffer, efalse_range, if_false_range);
// Compute the buffer index at which we'll read. // Compute the buffer index at which we'll read.
Node* buffer_index = Node* buffer_index =
graph()->NewNode(simplified()->NumberAdd(), offset, byte_offset); graph()->NewNode(simplified()->NumberAdd(), offset, byte_offset);
// Perform the load. // Perform the load.
vfalse_range = efalse_range = Node* value = effect = graph()->NewNode(
graph()->NewNode(simplified()->LoadDataViewElement(element_type), simplified()->LoadDataViewElement(element_type), buffer, backing_store,
buffer, backing_store, buffer_index, buffer_index, is_little_endian, effect, control);
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);
// We can't throw in LoadDataViewElement(),
// so we don't need to handle that path here.
// 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);
}
// 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);
// Continue on the regular path. // Continue on the regular path.
ReplaceWithValue(node, vfalse_range, efalse_range, if_false_range); ReplaceWithValue(node, value, effect, control);
return Changed(vfalse_range); return Changed(value);
} }
return NoChange(); return NoChange();
} }
Reduction JSCallReducer::ReduceDataViewPrototypeSet( Reduction JSCallReducer::ReduceDataViewPrototypeSet(
Node* node, ExternalArrayType element_type) { Node* node, ExternalArrayType element_type) {
Node* effect = NodeProperties::GetEffectInput(node); Node* effect = NodeProperties::GetEffectInput(node);
Node* control = NodeProperties::GetControlInput(node); Node* control = NodeProperties::GetControlInput(node);
Node* receiver = NodeProperties::GetValueInput(node, 1); Node* receiver = NodeProperties::GetValueInput(node, 1);
Node* context = NodeProperties::GetContextInput(node);
Node* frame_state = NodeProperties::GetFrameStateInput(node);
CallParameters const& p = CallParametersOf(node->op()); CallParameters const& p = CallParametersOf(node->op());
if (p.speculation_mode() == SpeculationMode::kDisallowSpeculation) {
return NoChange();
}
Node* offset = node->op()->ValueInputCount() > 2 Node* offset = node->op()->ValueInputCount() > 2
? NodeProperties::GetValueInput(node, 2) ? NodeProperties::GetValueInput(node, 2)
: jsgraph()->ZeroConstant(); : jsgraph()->ZeroConstant();
...@@ -6843,7 +6781,7 @@ Reduction JSCallReducer::ReduceDataViewPrototypeSet( ...@@ -6843,7 +6781,7 @@ Reduction JSCallReducer::ReduceDataViewPrototypeSet(
jsgraph()->ZeroConstant(), offset); jsgraph()->ZeroConstant(), offset);
effect = graph()->NewNode( effect = graph()->NewNode(
simplified()->CheckIf(DeoptimizeReason::kNotASmi, p.feedback()), simplified()->CheckIf(DeoptimizeReason::kOutOfBounds, p.feedback()),
is_positive, effect, control); is_positive, effect, control);
// Coerce {is_little_endian} to boolean. // Coerce {is_little_endian} to boolean.
...@@ -6863,37 +6801,25 @@ Reduction JSCallReducer::ReduceDataViewPrototypeSet( ...@@ -6863,37 +6801,25 @@ Reduction JSCallReducer::ReduceDataViewPrototypeSet(
Node* check_neutered = effect = graph()->NewNode( Node* check_neutered = effect = graph()->NewNode(
simplified()->ArrayBufferWasNeutered(), buffer, effect, control); simplified()->ArrayBufferWasNeutered(), buffer, effect, control);
Node* branch_neutered = graph()->NewNode( check_neutered =
common()->Branch(BranchHint::kFalse), check_neutered, control); graph()->NewNode(simplified()->BooleanNot(), check_neutered);
// Raise an error if it was neuteured. // If the buffer was neutered, deopt and let the unoptimized code throw.
Node* if_true_neutered = effect = graph()->NewNode(
graph()->NewNode(common()->IfTrue(), branch_neutered); simplified()->CheckIf(DeoptimizeReason::kArrayBufferWasNeutered,
Node* etrue_neutered = effect; p.feedback()),
{ check_neutered, effect, control);
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;
// Get the byte offset and byte length of the {receiver}. // Get the byte offset and byte length of the {receiver}.
Node* byte_offset = efalse_neutered = Node* byte_offset = effect =
graph()->NewNode(simplified()->LoadField( graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSArrayBufferViewByteOffset()), AccessBuilder::ForJSArrayBufferViewByteOffset()),
receiver, efalse_neutered, if_false_neutered); receiver, effect, control);
Node* byte_length = efalse_neutered = Node* byte_length = effect =
graph()->NewNode(simplified()->LoadField( graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSArrayBufferViewByteLength()), AccessBuilder::ForJSArrayBufferViewByteLength()),
receiver, efalse_neutered, if_false_neutered); receiver, effect, control);
// The end offset is the offset plus the element size // The end offset is the offset plus the element size
// of the type that we want to store. // of the type that we want to store.
...@@ -6901,85 +6827,34 @@ Reduction JSCallReducer::ReduceDataViewPrototypeSet( ...@@ -6901,85 +6827,34 @@ Reduction JSCallReducer::ReduceDataViewPrototypeSet(
Node* end_offset = graph()->NewNode(simplified()->NumberAdd(), offset, Node* end_offset = graph()->NewNode(simplified()->NumberAdd(), offset,
jsgraph()->Constant(element_size)); jsgraph()->Constant(element_size));
// We need to check that {end_offset} <= {byte_length}, ie // We need to check that {end_offset} <= {byte_length}.
// throw a RangeError if {byte_length} < {end_offset}. Node* check_bounds = graph()->NewNode(simplified()->NumberLessThanOrEqual(),
Node* check_range = graph()->NewNode(simplified()->NumberLessThan(), end_offset, byte_length);
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); // Also deopt and let the unoptimized code throw in this case.
Node* etrue_range = efalse_neutered; effect = graph()->NewNode(
{ simplified()->CheckIf(DeoptimizeReason::kOutOfBounds, p.feedback()),
if_true_range = etrue_range = graph()->NewNode( check_bounds, effect, control);
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. // Get the buffer's backing store.
Node* backing_store = efalse_range = Node* backing_store = effect = graph()->NewNode(
graph()->NewNode(simplified()->LoadField( simplified()->LoadField(AccessBuilder::ForJSArrayBufferBackingStore()),
AccessBuilder::ForJSArrayBufferBackingStore()), buffer, effect, control);
buffer, efalse_range, if_false_range);
// Compute the buffer index at which we'll write. // Compute the buffer index at which we'll write.
Node* buffer_index = Node* buffer_index =
graph()->NewNode(simplified()->NumberAdd(), offset, byte_offset); graph()->NewNode(simplified()->NumberAdd(), offset, byte_offset);
// Perform the store. // Perform the store.
efalse_range = effect = graph()->NewNode(simplified()->StoreDataViewElement(element_type),
graph()->NewNode(simplified()->StoreDataViewElement(element_type),
buffer, backing_store, buffer_index, value, buffer, backing_store, buffer_index, value,
is_little_endian, efalse_range, if_false_range); is_little_endian, effect, control);
}
// 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);
// We can't throw in StoreDataViewElement(), Node* value = jsgraph()->UndefinedConstant();
// so we don't need to handle that path here.
// 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);
}
// 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);
// Continue on the regular path. // Continue on the regular path.
ReplaceWithValue(node, vfalse_range, efalse_range, if_false_range); ReplaceWithValue(node, value, effect, control);
return Changed(vfalse_range); return Changed(value);
} }
return NoChange(); 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)); ...@@ -131,43 +131,56 @@ assertEquals(b4, readFloat64(16));
dataview.setFloat64(16, b4, true); dataview.setFloat64(16, b4, true);
assertEquals(b4, readFloat64(16, true)); assertEquals(b4, readFloat64(16, true));
// TurboFan out of bounds reads deopt.
// TurboFan out of bounds read, throw with exception handler.
assertOptimized(readInt8Handled); assertOptimized(readInt8Handled);
assertInstanceof(readInt8Handled(24), RangeError); assertInstanceof(readInt8Handled(24), RangeError);
assertOptimized(readInt8Handled); assertUnoptimized(readInt8Handled);
assertOptimized(readInt16Handled); assertOptimized(readInt16Handled);
assertInstanceof(readInt16Handled(23), RangeError); assertInstanceof(readInt16Handled(23), RangeError);
assertOptimized(readInt16Handled); assertUnoptimized(readInt16Handled);
assertOptimized(readInt32Handled); assertOptimized(readInt32Handled);
assertInstanceof(readInt32Handled(21), RangeError); assertInstanceof(readInt32Handled(21), RangeError);
assertOptimized(readInt32Handled); assertUnoptimized(readInt32Handled);
// Without exception handler. // Without exception handler.
assertOptimized(readUint8); assertOptimized(readUint8);
assertThrows(() => readUint8(24)); assertThrows(() => readUint8(24));
assertOptimized(readUint8); assertUnoptimized(readUint8);
assertOptimized(readFloat32); assertOptimized(readFloat32);
assertThrows(() => readFloat32(21)); assertThrows(() => readFloat32(21));
assertOptimized(readFloat32); assertUnoptimized(readFloat32);
assertOptimized(readFloat64); assertOptimized(readFloat64);
assertThrows(() => readFloat64(17)); assertThrows(() => readFloat64(17));
assertOptimized(readFloat64); assertUnoptimized(readFloat64);
// TurboFan deoptimizations. // Negative Smi deopts.
assertOptimized(readInt8Handled); (function() {
assertInstanceof(readInt8Handled(-1), RangeError); // Negative Smi deopts. function readInt8Handled(offset) {
assertUnoptimized(readInt8Handled); try { return dataview.getInt8(offset); } catch (e) { return e; }
}
warmup(readInt8Handled); warmup(readInt8Handled);
assertOptimized(readInt8Handled); assertOptimized(readInt8Handled);
assertEquals(values[3], readInt8Handled(3.14)); // Non-Smi index deopts. assertInstanceof(readInt8Handled(-1), RangeError);
assertUnoptimized(readInt8Handled); assertUnoptimized(readInt8Handled);
})();
// TurboFan neutered buffer.
warmup(readInt8Handled); // Non-Smi index deopts.
assertOptimized(readInt8Handled); (function() {
%ArrayBufferNeuter(buffer); function readUint8(offset) { return dataview.getUint8(offset); }
assertInstanceof(readInt8Handled(0), TypeError); warmup(readUint8);
assertOptimized(readInt8Handled); 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)); ...@@ -117,15 +117,15 @@ assertEquals(b4, dataview.getFloat64(8));
writeFloat64(8, b4, true); writeFloat64(8, b4, true);
assertEquals(b4, dataview.getFloat64(8, true)); assertEquals(b4, dataview.getFloat64(8, true));
// TurboFan out of bounds read, throw with exception handler. // TurboFan out of bounds read, deopt.
assertOptimized(writeInt8Handled); assertOptimized(writeInt8Handled);
assertInstanceof(writeInt8Handled(24, 0), RangeError); assertInstanceof(writeInt8Handled(24, 0), RangeError);
assertOptimized(writeInt8Handled); assertUnoptimized(writeInt8Handled);
// Without exception handler. // Without exception handler, deopt too.
assertOptimized(writeUint8); assertOptimized(writeUint8);
assertThrows(() => writeUint8(24, 0)); assertThrows(() => writeUint8(24, 0));
assertOptimized(writeUint8); assertUnoptimized(writeUint8);
// None of the stores wrote out of bounds. // None of the stores wrote out of bounds.
var bytes = new Uint8Array(buffer); 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