Skip to content

Commit a461185

Browse files
committed
deps: V8: cherry-pick 254c7945eea2
Original commit message: Merged: [deoptimizer] Fix bug in OptimizedFrame::Summarize Revision: 3353a7d0b017146d543434be4036a81aaf7d25ae BUG=chromium:1182647 NOTRY=true NOPRESUBMIT=true NOTREECHECKS=true R=​[email protected] (cherry picked from commit c0c96b768a7d3463b11403874549e6496529740d) Change-Id: I86abd6a3f34169be5f99aa9f54bb7bb3706fa85a Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2780300 Reviewed-by: Georg Neis <[email protected]> Reviewed-by: Benedikt Meurer <[email protected]> Commit-Queue: Georg Neis <[email protected]> Cr-Original-Commit-Position: refs/branch-heads/8.9@{nodejs#49} Cr-Original-Branched-From: 16b9bbbd581c25391981aa03180b76aa60463a3e-refs/heads/8.9.255@{#1} Cr-Original-Branched-From: d16a2a688498bd1c3e6a49edb25d8c4ca56232dc-refs/heads/master@{#72039} Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2794427 Reviewed-by: Victor-Gabriel Savu <[email protected]> Commit-Queue: Artem Sumaneev <[email protected]> Cr-Commit-Position: refs/branch-heads/8.6@{nodejs#72} Cr-Branched-From: a64aed2333abf49e494d2a5ce24bbd14fff19f60-refs/heads/8.6.395@{#1} Cr-Branched-From: a626bc036236c9bf92ac7b87dc40c9e538b087e3-refs/heads/master@{#69472} Refs: v8/v8@254c794
1 parent 14f68be commit a461185

File tree

4 files changed

+93
-11
lines changed

4 files changed

+93
-11
lines changed

common.gypi

+1-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@
3636

3737
# Reset this number to 0 on major V8 upgrades.
3838
# Increment by one for each non-official patch applied to deps/v8.
39-
'v8_embedder_string': '-node.48',
39+
'v8_embedder_string': '-node.49',
4040

4141
##### V8 defaults for Node.js #####
4242

deps/v8/src/deoptimizer/deoptimizer.cc

+49-9
Original file line numberDiff line numberDiff line change
@@ -3266,7 +3266,8 @@ Address TranslatedState::DecompressIfNeeded(intptr_t value) {
32663266
}
32673267
}
32683268

3269-
TranslatedState::TranslatedState(const JavaScriptFrame* frame) {
3269+
TranslatedState::TranslatedState(const JavaScriptFrame* frame)
3270+
: purpose_(kFrameInspection) {
32703271
int deopt_index = Safepoint::kNoDeoptimizationIndex;
32713272
DeoptimizationData data =
32723273
static_cast<const OptimizedFrame*>(frame)->GetDeoptimizationData(
@@ -3641,25 +3642,63 @@ void TranslatedState::EnsureCapturedObjectAllocatedAt(
36413642
}
36423643

36433644
default:
3644-
CHECK(map->IsJSObjectMap());
36453645
EnsureJSObjectAllocated(slot, map);
3646-
TranslatedValue* properties_slot = &(frame->values_[value_index]);
3647-
value_index++;
3646+
int remaining_children_count = slot->GetChildrenCount() - 1;
3647+
3648+
TranslatedValue* properties_slot = frame->ValueAt(value_index);
3649+
value_index++, remaining_children_count--;
36483650
if (properties_slot->kind() == TranslatedValue::kCapturedObject) {
3649-
// If we are materializing the property array, make sure we put
3650-
// the mutable heap numbers at the right places.
3651+
// We are materializing the property array, so make sure we put the
3652+
// mutable heap numbers at the right places.
36513653
EnsurePropertiesAllocatedAndMarked(properties_slot, map);
36523654
EnsureChildrenAllocated(properties_slot->GetChildrenCount(), frame,
36533655
&value_index, worklist);
3656+
} else {
3657+
CHECK_EQ(properties_slot->kind(), TranslatedValue::kTagged);
36543658
}
3655-
// Make sure all the remaining children (after the map and properties) are
3656-
// allocated.
3657-
return EnsureChildrenAllocated(slot->GetChildrenCount() - 2, frame,
3659+
3660+
TranslatedValue* elements_slot = frame->ValueAt(value_index);
3661+
value_index++, remaining_children_count--;
3662+
if (elements_slot->kind() == TranslatedValue::kCapturedObject ||
3663+
!map->IsJSArrayMap()) {
3664+
// Handle this case with the other remaining children below.
3665+
value_index--, remaining_children_count++;
3666+
} else {
3667+
CHECK_EQ(elements_slot->kind(), TranslatedValue::kTagged);
3668+
elements_slot->GetValue();
3669+
if (purpose_ == kFrameInspection) {
3670+
// We are materializing a JSArray for the purpose of frame inspection.
3671+
// If we were to construct it with the above elements value then an
3672+
// actual deopt later on might create another JSArray instance with
3673+
// the same elements store. That would violate the key assumption
3674+
// behind left-trimming.
3675+
elements_slot->ReplaceElementsArrayWithCopy();
3676+
}
3677+
}
3678+
3679+
// Make sure all the remaining children (after the map, properties store,
3680+
// and possibly elements store) are allocated.
3681+
return EnsureChildrenAllocated(remaining_children_count, frame,
36583682
&value_index, worklist);
36593683
}
36603684
UNREACHABLE();
36613685
}
36623686

3687+
void TranslatedValue::ReplaceElementsArrayWithCopy() {
3688+
DCHECK_EQ(kind(), TranslatedValue::kTagged);
3689+
DCHECK_EQ(materialization_state(), TranslatedValue::kFinished);
3690+
auto elements = Handle<FixedArrayBase>::cast(GetValue());
3691+
DCHECK(elements->IsFixedArray() || elements->IsFixedDoubleArray());
3692+
if (elements->IsFixedDoubleArray()) {
3693+
DCHECK(!elements->IsCowArray());
3694+
set_storage(isolate()->factory()->CopyFixedDoubleArray(
3695+
Handle<FixedDoubleArray>::cast(elements)));
3696+
} else if (!elements->IsCowArray()) {
3697+
set_storage(isolate()->factory()->CopyFixedArray(
3698+
Handle<FixedArray>::cast(elements)));
3699+
}
3700+
}
3701+
36633702
void TranslatedState::EnsureChildrenAllocated(int count, TranslatedFrame* frame,
36643703
int* value_index,
36653704
std::stack<int>* worklist) {
@@ -3724,6 +3763,7 @@ Handle<ByteArray> TranslatedState::AllocateStorageFor(TranslatedValue* slot) {
37243763

37253764
void TranslatedState::EnsureJSObjectAllocated(TranslatedValue* slot,
37263765
Handle<Map> map) {
3766+
CHECK(map->IsJSObjectMap());
37273767
CHECK_EQ(map->instance_size(), slot->GetChildrenCount() * kTaggedSize);
37283768

37293769
Handle<ByteArray> object_storage = AllocateStorageFor(slot);

deps/v8/src/deoptimizer/deoptimizer.h

+18-1
Original file line numberDiff line numberDiff line change
@@ -117,6 +117,8 @@ class TranslatedValue {
117117
return storage_;
118118
}
119119

120+
void ReplaceElementsArrayWithCopy();
121+
120122
Kind kind_;
121123
MaterializationState materialization_state_ = kUninitialized;
122124
TranslatedState* container_; // This is only needed for materialization of
@@ -313,7 +315,15 @@ class TranslatedFrame {
313315

314316
class TranslatedState {
315317
public:
316-
TranslatedState() = default;
318+
// There are two constructors, each for a different purpose:
319+
320+
// The default constructor is for the purpose of deoptimizing an optimized
321+
// frame (replacing it with one or several unoptimized frames). It is used by
322+
// the Deoptimizer.
323+
TranslatedState() : purpose_(kDeoptimization) {}
324+
325+
// This constructor is for the purpose of merely inspecting an optimized
326+
// frame. It is used by stack trace generation and various debugging features.
317327
explicit TranslatedState(const JavaScriptFrame* frame);
318328

319329
void Prepare(Address stack_frame_pointer);
@@ -347,6 +357,12 @@ class TranslatedState {
347357
private:
348358
friend TranslatedValue;
349359

360+
// See the description of the constructors for an explanation of the two
361+
// purposes. The only actual difference is that in the kFrameInspection case
362+
// extra work is needed to not violate assumptions made by left-trimming. For
363+
// details, see the code around ReplaceElementsArrayWithCopy.
364+
enum Purpose { kDeoptimization, kFrameInspection };
365+
350366
TranslatedFrame CreateNextTranslatedFrame(TranslationIterator* iterator,
351367
FixedArray literal_array,
352368
Address fp, FILE* trace_file);
@@ -404,6 +420,7 @@ class TranslatedState {
404420
static Float32 GetFloatSlot(Address fp, int slot_index);
405421
static Float64 GetDoubleSlot(Address fp, int slot_index);
406422

423+
Purpose const purpose_;
407424
std::vector<TranslatedFrame> frames_;
408425
Isolate* isolate_ = nullptr;
409426
Address stack_frame_pointer_ = kNullAddress;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
// Copyright 2021 the V8 project authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style license that can be
3+
// found in the LICENSE file.
4+
5+
// Flags: --allow-natives-syntax --verify-heap
6+
7+
function foo() {
8+
const arr = Array(1000);
9+
10+
function bar() {
11+
try { ({a: p4nda, b: arr.length}); } catch(e) {}
12+
}
13+
14+
for (var i = 0; i < 25; i++) bar();
15+
16+
/p4nda/.test({}); // Deopt here.
17+
18+
arr.shift();
19+
}
20+
21+
%PrepareFunctionForOptimization(foo);
22+
foo();
23+
foo();
24+
%OptimizeFunctionOnNextCall(foo);
25+
foo();

0 commit comments

Comments
 (0)