Skip to content

Commit 65e6db5

Browse files
author
v8-autoroll
committed
Version 3.32.1 (based on 9def087)
[turbofan] Don't crash when typing load from a Uint8ClampedArray (Chromium issue 446156). [turbofan] Truncation of Bit/Word8/16 to Word32 is a no-op (Chromium issue 445859). [x64] Rearrange code for OOB integer loads (Chromium issue 445858). Fix %NeverOptimizeFunction() intrinsic (Chromium issue 445732). [turbofan] Fix invalid bounds check with overflowing offset (Chromium issue 445267). [turbofan] Raise max virtual registers and call parameter limit (issue 3786). Performance and stability improvements on all platforms. Cr-Commit-Position: refs/heads/candidates@{#25351}
1 parent 799ad2f commit 65e6db5

31 files changed

+488
-322
lines changed

ChangeLog

+21
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,24 @@
1+
2015-01-07: Version 3.32.1
2+
3+
[turbofan] Don't crash when typing load from a Uint8ClampedArray
4+
(Chromium issue 446156).
5+
6+
[turbofan] Truncation of Bit/Word8/16 to Word32 is a no-op (Chromium
7+
issue 445859).
8+
9+
[x64] Rearrange code for OOB integer loads (Chromium issue 445858).
10+
11+
Fix %NeverOptimizeFunction() intrinsic (Chromium issue 445732).
12+
13+
[turbofan] Fix invalid bounds check with overflowing offset (Chromium
14+
issue 445267).
15+
16+
[turbofan] Raise max virtual registers and call parameter limit (issue
17+
3786).
18+
19+
Performance and stability improvements on all platforms.
20+
21+
122
2014-12-23: Version 3.31.74
223

324
[turbofan] Turn DCHECK for fixed slot index into a CHECK (Chromium issue

DEPS

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ deps = {
1818
"v8/testing/gmock":
1919
Var("git_url") + "/external/googlemock.git" + "@" + "29763965ab52f24565299976b936d1265cb6a271", # from svn revision 501
2020
"v8/tools/clang":
21-
Var("git_url") + "/chromium/src/tools/clang.git" + "@" + "90fb65e7a9a5c9d6d9613dfb0e78921c52ca9cfc",
21+
Var("git_url") + "/chromium/src/tools/clang.git" + "@" + "c945be21f6485fa177b43814f910b76cce921653",
2222
}
2323

2424
deps_os = {

src/compiler/control-reducer.cc

+8
Original file line numberDiff line numberDiff line change
@@ -403,6 +403,14 @@ class ControlReducerImpl {
403403
if (n <= 1) return dead(); // No non-control inputs.
404404
if (n == 2) return node->InputAt(0); // Only one non-control input.
405405

406+
// Never remove an effect phi from a (potentially non-terminating) loop.
407+
// Otherwise, we might end up eliminating effect nodes, such as calls,
408+
// before the loop.
409+
if (node->opcode() == IrOpcode::kEffectPhi &&
410+
NodeProperties::GetControlInput(node)->opcode() == IrOpcode::kLoop) {
411+
return node;
412+
}
413+
406414
Node* replacement = NULL;
407415
Node::Inputs inputs = node->inputs();
408416
for (InputIter it = inputs.begin(); n > 1; --n, ++it) {

src/compiler/js-typed-lowering.cc

+23-135
Original file line numberDiff line numberDiff line change
@@ -490,124 +490,34 @@ Reduction JSTypedLowering::ReduceJSStrictEqual(Node* node, bool invert) {
490490
}
491491

492492

493-
Reduction JSTypedLowering::ReduceJSToBooleanInput(Node* input) {
494-
if (input->opcode() == IrOpcode::kJSToBoolean) {
495-
// Recursively try to reduce the input first.
496-
Reduction result = ReduceJSToBoolean(input);
497-
if (result.Changed()) return result;
498-
return Changed(input); // JSToBoolean(JSToBoolean(x)) => JSToBoolean(x)
499-
}
500-
// Check if we have a cached conversion.
501-
Node* conversion = FindConversion<IrOpcode::kJSToBoolean>(input);
502-
if (conversion) return Replace(conversion);
493+
Reduction JSTypedLowering::ReduceJSUnaryNot(Node* node) {
494+
Node* input = node->InputAt(0);
503495
Type* input_type = NodeProperties::GetBounds(input).upper;
504496
if (input_type->Is(Type::Boolean())) {
505-
return Changed(input); // JSToBoolean(x:boolean) => x
506-
}
507-
if (input_type->Is(Type::Undefined())) {
508-
// JSToBoolean(undefined) => #false
509-
return Replace(jsgraph()->FalseConstant());
510-
}
511-
if (input_type->Is(Type::Null())) {
512-
// JSToBoolean(null) => #false
513-
return Replace(jsgraph()->FalseConstant());
514-
}
515-
if (input_type->Is(Type::DetectableReceiver())) {
516-
// JSToBoolean(x:detectable) => #true
517-
return Replace(jsgraph()->TrueConstant());
518-
}
519-
if (input_type->Is(Type::Undetectable())) {
520-
// JSToBoolean(x:undetectable) => #false
521-
return Replace(jsgraph()->FalseConstant());
522-
}
523-
if (input_type->Is(Type::OrderedNumber())) {
524-
// JSToBoolean(x:ordered-number) => BooleanNot(NumberEqual(x, #0))
525-
Node* cmp = graph()->NewNode(simplified()->NumberEqual(), input,
526-
jsgraph()->ZeroConstant());
527-
Node* inv = graph()->NewNode(simplified()->BooleanNot(), cmp);
528-
return Replace(inv);
529-
}
530-
if (input_type->Is(Type::String())) {
531-
// JSToBoolean(x:string) => BooleanNot(NumberEqual(x.length, #0))
532-
FieldAccess access = AccessBuilder::ForStringLength();
533-
Node* length = graph()->NewNode(simplified()->LoadField(access), input,
534-
graph()->start(), graph()->start());
535-
Node* cmp = graph()->NewNode(simplified()->NumberEqual(), length,
536-
jsgraph()->ZeroConstant());
537-
Node* inv = graph()->NewNode(simplified()->BooleanNot(), cmp);
538-
return Replace(inv);
497+
// JSUnaryNot(x:boolean,context) => BooleanNot(x)
498+
node->set_op(simplified()->BooleanNot());
499+
node->TrimInputCount(1);
500+
return Changed(node);
539501
}
540-
return NoChange();
502+
// JSUnaryNot(x,context) => BooleanNot(AnyToBoolean(x))
503+
node->set_op(simplified()->BooleanNot());
504+
node->ReplaceInput(0, graph()->NewNode(simplified()->AnyToBoolean(), input));
505+
node->TrimInputCount(1);
506+
return Changed(node);
541507
}
542508

543509

544510
Reduction JSTypedLowering::ReduceJSToBoolean(Node* node) {
545-
// Try to reduce the input first.
546-
Node* const input = node->InputAt(0);
547-
Reduction reduction = ReduceJSToBooleanInput(input);
548-
if (reduction.Changed()) return reduction;
549-
if (input->opcode() == IrOpcode::kPhi) {
550-
// JSToBoolean(phi(x1,...,xn,control),context)
551-
// => phi(JSToBoolean(x1,no-context),...,JSToBoolean(xn,no-context))
552-
int const input_count = input->InputCount() - 1;
553-
Node* const control = input->InputAt(input_count);
554-
DCHECK_LE(0, input_count);
555-
DCHECK(NodeProperties::IsControl(control));
556-
DCHECK(NodeProperties::GetBounds(node).upper->Is(Type::Boolean()));
557-
DCHECK(!NodeProperties::GetBounds(input).upper->Is(Type::Boolean()));
558-
node->set_op(common()->Phi(kMachAnyTagged, input_count));
559-
for (int i = 0; i < input_count; ++i) {
560-
// We must be very careful not to introduce cycles when pushing
561-
// operations into phis. It is safe for {value}, since it appears
562-
// as input to the phi that we are replacing, but it's not safe
563-
// to simply reuse the context of the {node}. However, ToBoolean()
564-
// does not require a context anyways, so it's safe to discard it
565-
// here and pass the dummy context.
566-
Node* const value = ConvertToBoolean(input->InputAt(i));
567-
if (i < node->InputCount()) {
568-
node->ReplaceInput(i, value);
569-
} else {
570-
node->AppendInput(graph()->zone(), value);
571-
}
572-
}
573-
if (input_count < node->InputCount()) {
574-
node->ReplaceInput(input_count, control);
575-
} else {
576-
node->AppendInput(graph()->zone(), control);
577-
}
578-
node->TrimInputCount(input_count + 1);
579-
return Changed(node);
580-
}
581-
if (input->opcode() == IrOpcode::kSelect) {
582-
// JSToBoolean(select(c,x1,x2),context)
583-
// => select(c,JSToBoolean(x1,no-context),...,JSToBoolean(x2,no-context))
584-
int const input_count = input->InputCount();
585-
BranchHint const input_hint = SelectParametersOf(input->op()).hint();
586-
DCHECK_EQ(3, input_count);
587-
DCHECK(NodeProperties::GetBounds(node).upper->Is(Type::Boolean()));
588-
DCHECK(!NodeProperties::GetBounds(input).upper->Is(Type::Boolean()));
589-
node->set_op(common()->Select(kMachAnyTagged, input_hint));
590-
node->InsertInput(graph()->zone(), 0, input->InputAt(0));
591-
for (int i = 1; i < input_count; ++i) {
592-
// We must be very careful not to introduce cycles when pushing
593-
// operations into selects. It is safe for {value}, since it appears
594-
// as input to the select that we are replacing, but it's not safe
595-
// to simply reuse the context of the {node}. However, ToBoolean()
596-
// does not require a context anyways, so it's safe to discard it
597-
// here and pass the dummy context.
598-
Node* const value = ConvertToBoolean(input->InputAt(i));
599-
node->ReplaceInput(i, value);
600-
}
601-
DCHECK_EQ(3, node->InputCount());
602-
return Changed(node);
603-
}
604-
InsertConversion(node);
605-
if (node->InputAt(1) != jsgraph()->NoContextConstant()) {
606-
// JSToBoolean(x,context) => JSToBoolean(x,no-context)
607-
node->ReplaceInput(1, jsgraph()->NoContextConstant());
608-
return Changed(node);
511+
Node* input = node->InputAt(0);
512+
Type* input_type = NodeProperties::GetBounds(input).upper;
513+
if (input_type->Is(Type::Boolean())) {
514+
// JSToBoolean(x:boolean,context) => x
515+
return Replace(input);
609516
}
610-
return NoChange();
517+
// JSToBoolean(x,context) => AnyToBoolean(x)
518+
node->set_op(simplified()->AnyToBoolean());
519+
node->TrimInputCount(1);
520+
return Changed(node);
611521
}
612522

613523

@@ -972,18 +882,8 @@ Reduction JSTypedLowering::Reduce(Node* node) {
972882
return ReduceNumberBinop(node, simplified()->NumberDivide());
973883
case IrOpcode::kJSModulus:
974884
return ReduceNumberBinop(node, simplified()->NumberModulus());
975-
case IrOpcode::kJSUnaryNot: {
976-
Reduction result = ReduceJSToBooleanInput(node->InputAt(0));
977-
if (result.Changed()) {
978-
// JSUnaryNot(x:boolean) => BooleanNot(x)
979-
node = result.replacement();
980-
} else {
981-
// JSUnaryNot(x) => BooleanNot(JSToBoolean(x))
982-
node->set_op(javascript()->ToBoolean());
983-
}
984-
Node* value = graph()->NewNode(simplified()->BooleanNot(), node);
985-
return Replace(value);
986-
}
885+
case IrOpcode::kJSUnaryNot:
886+
return ReduceJSUnaryNot(node);
987887
case IrOpcode::kJSToBoolean:
988888
return ReduceJSToBoolean(node);
989889
case IrOpcode::kJSToNumber:
@@ -1005,17 +905,6 @@ Reduction JSTypedLowering::Reduce(Node* node) {
1005905
}
1006906

1007907

1008-
Node* JSTypedLowering::ConvertToBoolean(Node* input) {
1009-
// Avoid inserting too many eager ToBoolean() operations.
1010-
Reduction const reduction = ReduceJSToBooleanInput(input);
1011-
if (reduction.Changed()) return reduction.replacement();
1012-
Node* const conversion = graph()->NewNode(javascript()->ToBoolean(), input,
1013-
jsgraph()->NoContextConstant());
1014-
InsertConversion(conversion);
1015-
return conversion;
1016-
}
1017-
1018-
1019908
Node* JSTypedLowering::ConvertToNumber(Node* input) {
1020909
DCHECK(NodeProperties::GetBounds(input).upper->Is(Type::PlainPrimitive()));
1021910
// Avoid inserting too many eager ToNumber() operations.
@@ -1043,8 +932,7 @@ Node* JSTypedLowering::FindConversion(Node* input) {
1043932

1044933

1045934
void JSTypedLowering::InsertConversion(Node* conversion) {
1046-
DCHECK(conversion->opcode() == IrOpcode::kJSToBoolean ||
1047-
conversion->opcode() == IrOpcode::kJSToNumber);
935+
DCHECK(conversion->opcode() == IrOpcode::kJSToNumber);
1048936
size_t const input_id = conversion->InputAt(0)->id();
1049937
if (input_id >= conversions_.size()) {
1050938
conversions_.resize(2 * input_id + 1);

src/compiler/js-typed-lowering.h

+1-2
Original file line numberDiff line numberDiff line change
@@ -41,7 +41,7 @@ class JSTypedLowering FINAL : public Reducer {
4141
Reduction ReduceJSStoreContext(Node* node);
4242
Reduction ReduceJSEqual(Node* node, bool invert);
4343
Reduction ReduceJSStrictEqual(Node* node, bool invert);
44-
Reduction ReduceJSToBooleanInput(Node* input);
44+
Reduction ReduceJSUnaryNot(Node* node);
4545
Reduction ReduceJSToBoolean(Node* node);
4646
Reduction ReduceJSToNumberInput(Node* input);
4747
Reduction ReduceJSToNumber(Node* node);
@@ -52,7 +52,6 @@ class JSTypedLowering FINAL : public Reducer {
5252
Reduction ReduceUI32Shift(Node* node, Signedness left_signedness,
5353
const Operator* shift_op);
5454

55-
Node* ConvertToBoolean(Node* input);
5655
Node* ConvertToNumber(Node* input);
5756
template <IrOpcode::Value>
5857
Node* FindConversion(Node* input);

src/compiler/opcodes.h

+1
Original file line numberDiff line numberDiff line change
@@ -132,6 +132,7 @@
132132

133133
// Opcodes for VirtuaMachine-level operators.
134134
#define SIMPLIFIED_OP_LIST(V) \
135+
V(AnyToBoolean) \
135136
V(BooleanNot) \
136137
V(BooleanToNumber) \
137138
V(NumberEqual) \

src/compiler/pipeline.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -441,7 +441,7 @@ struct SimplifiedLoweringPhase {
441441
void Run(PipelineData* data, Zone* temp_zone) {
442442
SourcePositionTable::Scope pos(data->source_positions(),
443443
SourcePosition::Unknown());
444-
SimplifiedLowering lowering(data->jsgraph());
444+
SimplifiedLowering lowering(data->jsgraph(), temp_zone);
445445
lowering.LowerAllNodes();
446446
ValueNumberingReducer vn_reducer(temp_zone);
447447
SimplifiedOperatorReducer simple_reducer(data->jsgraph());

src/compiler/simplified-lowering.cc

+33-2
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,9 @@ class RepresentationSelector {
7777
memset(info_, 0, sizeof(NodeInfo) * count_);
7878

7979
Factory* f = zone->isolate()->factory();
80+
safe_bit_range_ =
81+
Type::Union(Type::Boolean(),
82+
Type::Range(f->NewNumber(0), f->NewNumber(1), zone), zone);
8083
safe_int_additive_range_ =
8184
Type::Range(f->NewNumber(-std::pow(2.0, 52.0)),
8285
f->NewNumber(std::pow(2.0, 52.0)), zone);
@@ -320,7 +323,7 @@ class RepresentationSelector {
320323
} else {
321324
return kRepFloat64;
322325
}
323-
} else if (upper->Is(Type::Boolean())) {
326+
} else if (IsSafeBitOperand(node)) {
324327
// multiple uses => pick kRepBit.
325328
return kRepBit;
326329
} else if (upper->Is(Type::Number())) {
@@ -414,6 +417,11 @@ class RepresentationSelector {
414417
return BothInputsAre(node, Type::Signed32()) && !CanObserveNonInt32(use);
415418
}
416419

420+
bool IsSafeBitOperand(Node* node) {
421+
Type* type = NodeProperties::GetBounds(node).upper;
422+
return type->Is(safe_bit_range_);
423+
}
424+
417425
bool IsSafeIntAdditiveOperand(Node* node) {
418426
Type* type = NodeProperties::GetBounds(node).upper;
419427
// TODO(jarin): Unfortunately, bitset types are not subtypes of larger
@@ -521,6 +529,28 @@ class RepresentationSelector {
521529
//------------------------------------------------------------------
522530
// Simplified operators.
523531
//------------------------------------------------------------------
532+
case IrOpcode::kAnyToBoolean: {
533+
if (IsSafeBitOperand(node->InputAt(0))) {
534+
VisitUnop(node, kRepBit, kRepBit);
535+
if (lower()) DeferReplacement(node, node->InputAt(0));
536+
} else {
537+
VisitUnop(node, kMachAnyTagged, kTypeBool | kRepTagged);
538+
if (lower()) {
539+
// AnyToBoolean(x) => Call(ToBooleanStub, x, no-context)
540+
Operator::Properties properties = node->op()->properties();
541+
Callable callable = CodeFactory::ToBoolean(
542+
jsgraph_->isolate(), ToBooleanStub::RESULT_AS_ODDBALL);
543+
CallDescriptor::Flags flags = CallDescriptor::kPatchableCallSite;
544+
CallDescriptor* desc = Linkage::GetStubCallDescriptor(
545+
callable.descriptor(), 0, flags, properties, jsgraph_->zone());
546+
node->set_op(jsgraph_->common()->Call(desc));
547+
node->InsertInput(jsgraph_->zone(), 0,
548+
jsgraph_->HeapConstant(callable.code()));
549+
node->AppendInput(jsgraph_->zone(), jsgraph_->NoContextConstant());
550+
}
551+
}
552+
break;
553+
}
524554
case IrOpcode::kBooleanNot: {
525555
if (lower()) {
526556
MachineTypeUnion input = GetInfo(node->InputAt(0))->output;
@@ -1034,6 +1064,7 @@ class RepresentationSelector {
10341064
Phase phase_; // current phase of algorithm
10351065
RepresentationChanger* changer_; // for inserting representation changes
10361066
ZoneQueue<Node*> queue_; // queue for traversing the graph
1067+
Type* safe_bit_range_;
10371068
Type* safe_int_additive_range_;
10381069

10391070
NodeInfo* GetInfo(Node* node) {
@@ -1058,7 +1089,7 @@ void SimplifiedLowering::LowerAllNodes() {
10581089
SimplifiedOperatorBuilder simplified(graph()->zone());
10591090
RepresentationChanger changer(jsgraph(), &simplified,
10601091
graph()->zone()->isolate());
1061-
RepresentationSelector selector(jsgraph(), zone(), &changer);
1092+
RepresentationSelector selector(jsgraph(), zone_, &changer);
10621093
selector.Run(this);
10631094
}
10641095

src/compiler/simplified-lowering.h

+4-2
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,8 @@ class RepresentationChanger;
2020

2121
class SimplifiedLowering FINAL {
2222
public:
23-
explicit SimplifiedLowering(JSGraph* jsgraph) : jsgraph_(jsgraph) {}
23+
SimplifiedLowering(JSGraph* jsgraph, Zone* zone)
24+
: jsgraph_(jsgraph), zone_(zone) {}
2425
~SimplifiedLowering() {}
2526

2627
void LowerAllNodes();
@@ -41,7 +42,8 @@ class SimplifiedLowering FINAL {
4142
void DoStringLessThanOrEqual(Node* node);
4243

4344
private:
44-
JSGraph* jsgraph_;
45+
JSGraph* const jsgraph_;
46+
Zone* const zone_;
4547

4648
Node* SmiTag(Node* node);
4749
Node* IsTagged(Node* node);

0 commit comments

Comments
 (0)