2016-04-18 11:57:06 +00:00
|
|
|
// Copyright 2015 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.
|
|
|
|
|
|
|
|
#include "src/compiler/code-assembler.h"
|
|
|
|
|
|
|
|
#include <ostream>
|
|
|
|
|
|
|
|
#include "src/code-factory.h"
|
2018-11-12 14:12:52 +00:00
|
|
|
#include "src/compiler/backend/instruction-selector.h"
|
2016-04-18 11:57:06 +00:00
|
|
|
#include "src/compiler/graph.h"
|
|
|
|
#include "src/compiler/linkage.h"
|
2016-05-19 15:49:03 +00:00
|
|
|
#include "src/compiler/node-matchers.h"
|
2016-04-18 11:57:06 +00:00
|
|
|
#include "src/compiler/pipeline.h"
|
|
|
|
#include "src/compiler/raw-machine-assembler.h"
|
|
|
|
#include "src/compiler/schedule.h"
|
|
|
|
#include "src/frames.h"
|
|
|
|
#include "src/interface-descriptors.h"
|
|
|
|
#include "src/interpreter/bytecodes.h"
|
2018-12-04 21:21:28 +00:00
|
|
|
#include "src/lsan.h"
|
2016-04-18 11:57:06 +00:00
|
|
|
#include "src/machine-type.h"
|
|
|
|
#include "src/macro-assembler.h"
|
2018-11-29 16:59:26 +00:00
|
|
|
#include "src/memcopy.h"
|
2017-02-23 11:46:29 +00:00
|
|
|
#include "src/objects-inl.h"
|
2018-11-03 00:13:22 +00:00
|
|
|
#include "src/objects/smi.h"
|
2016-09-20 16:07:25 +00:00
|
|
|
#include "src/zone/zone.h"
|
2016-04-18 11:57:06 +00:00
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
2017-10-20 15:18:53 +00:00
|
|
|
|
|
|
|
constexpr MachineType MachineTypeOf<Smi>::value;
|
|
|
|
constexpr MachineType MachineTypeOf<Object>::value;
|
|
|
|
|
2016-04-18 11:57:06 +00:00
|
|
|
namespace compiler {
|
|
|
|
|
2017-10-20 15:18:53 +00:00
|
|
|
static_assert(std::is_convertible<TNode<Number>, TNode<Object>>::value,
|
|
|
|
"test subtyping");
|
|
|
|
static_assert(std::is_convertible<TNode<UnionT<Smi, HeapNumber>>,
|
|
|
|
TNode<UnionT<Smi, HeapObject>>>::value,
|
|
|
|
"test subtyping");
|
|
|
|
static_assert(
|
|
|
|
!std::is_convertible<TNode<UnionT<Smi, HeapObject>>, TNode<Number>>::value,
|
|
|
|
"test subtyping");
|
|
|
|
|
2016-11-16 11:48:07 +00:00
|
|
|
CodeAssemblerState::CodeAssemblerState(
|
|
|
|
Isolate* isolate, Zone* zone, const CallInterfaceDescriptor& descriptor,
|
2018-04-30 12:13:54 +00:00
|
|
|
Code::Kind kind, const char* name, PoisoningMitigationLevel poisoning_level,
|
2018-12-07 14:53:22 +00:00
|
|
|
int32_t builtin_index)
|
2018-02-11 19:17:27 +00:00
|
|
|
// TODO(rmcilroy): Should we use Linkage::GetBytecodeDispatchDescriptor for
|
|
|
|
// bytecode handlers?
|
2016-11-16 11:48:07 +00:00
|
|
|
: CodeAssemblerState(
|
2016-04-18 11:57:06 +00:00
|
|
|
isolate, zone,
|
|
|
|
Linkage::GetStubCallDescriptor(
|
2018-06-18 15:14:29 +00:00
|
|
|
zone, descriptor, descriptor.GetStackParameterCount(),
|
2018-06-21 08:47:48 +00:00
|
|
|
CallDescriptor::kNoFlags, Operator::kNoProperties),
|
2018-12-07 14:53:22 +00:00
|
|
|
kind, name, poisoning_level, builtin_index) {}
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2018-04-30 12:13:54 +00:00
|
|
|
CodeAssemblerState::CodeAssemblerState(Isolate* isolate, Zone* zone,
|
|
|
|
int parameter_count, Code::Kind kind,
|
|
|
|
const char* name,
|
|
|
|
PoisoningMitigationLevel poisoning_level,
|
|
|
|
int32_t builtin_index)
|
2017-09-29 14:59:24 +00:00
|
|
|
: CodeAssemblerState(
|
|
|
|
isolate, zone,
|
2018-10-02 15:50:33 +00:00
|
|
|
Linkage::GetJSCallDescriptor(
|
|
|
|
zone, false, parameter_count,
|
|
|
|
(kind == Code::BUILTIN ? CallDescriptor::kPushArgumentCount
|
|
|
|
: CallDescriptor::kNoFlags) |
|
|
|
|
CallDescriptor::kCanUseRoots),
|
2018-12-07 14:53:22 +00:00
|
|
|
kind, name, poisoning_level, builtin_index) {}
|
2016-11-16 11:48:07 +00:00
|
|
|
|
2018-04-30 12:13:54 +00:00
|
|
|
CodeAssemblerState::CodeAssemblerState(Isolate* isolate, Zone* zone,
|
|
|
|
CallDescriptor* call_descriptor,
|
|
|
|
Code::Kind kind, const char* name,
|
|
|
|
PoisoningMitigationLevel poisoning_level,
|
2018-12-07 14:53:22 +00:00
|
|
|
int32_t builtin_index)
|
2016-04-18 11:57:06 +00:00
|
|
|
: raw_assembler_(new RawMachineAssembler(
|
|
|
|
isolate, new (zone) Graph(zone), call_descriptor,
|
|
|
|
MachineType::PointerRepresentation(),
|
2016-07-22 20:55:03 +00:00
|
|
|
InstructionSelector::SupportedMachineOperatorFlags(),
|
2018-04-30 12:13:54 +00:00
|
|
|
InstructionSelector::AlignmentRequirements(), poisoning_level)),
|
2017-09-29 14:59:24 +00:00
|
|
|
kind_(kind),
|
2016-04-18 11:57:06 +00:00
|
|
|
name_(name),
|
2017-11-16 12:35:58 +00:00
|
|
|
builtin_index_(builtin_index),
|
2016-04-18 11:57:06 +00:00
|
|
|
code_generated_(false),
|
|
|
|
variables_(zone) {}
|
|
|
|
|
2018-09-17 11:30:48 +00:00
|
|
|
CodeAssemblerState::~CodeAssemblerState() = default;
|
2016-11-16 11:48:07 +00:00
|
|
|
|
2017-01-02 17:22:30 +00:00
|
|
|
int CodeAssemblerState::parameter_count() const {
|
|
|
|
return static_cast<int>(raw_assembler_->call_descriptor()->ParameterCount());
|
|
|
|
}
|
|
|
|
|
2018-09-17 11:30:48 +00:00
|
|
|
CodeAssembler::~CodeAssembler() = default;
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2017-04-05 17:40:36 +00:00
|
|
|
#if DEBUG
|
|
|
|
void CodeAssemblerState::PrintCurrentBlock(std::ostream& os) {
|
|
|
|
raw_assembler_->PrintCurrentBlock(os);
|
|
|
|
}
|
2018-11-13 14:01:35 +00:00
|
|
|
#endif
|
2017-11-15 19:35:10 +00:00
|
|
|
|
|
|
|
bool CodeAssemblerState::InsideBlock() { return raw_assembler_->InsideBlock(); }
|
2017-04-05 17:40:36 +00:00
|
|
|
|
|
|
|
void CodeAssemblerState::SetInitialDebugInformation(const char* msg,
|
|
|
|
const char* file,
|
|
|
|
int line) {
|
|
|
|
#if DEBUG
|
|
|
|
AssemblerDebugInfo debug_info = {msg, file, line};
|
|
|
|
raw_assembler_->SetInitialDebugInformation(debug_info);
|
|
|
|
#endif // DEBUG
|
|
|
|
}
|
|
|
|
|
2016-12-15 13:32:27 +00:00
|
|
|
class BreakOnNodeDecorator final : public GraphDecorator {
|
|
|
|
public:
|
|
|
|
explicit BreakOnNodeDecorator(NodeId node_id) : node_id_(node_id) {}
|
|
|
|
|
|
|
|
void Decorate(Node* node) final {
|
|
|
|
if (node->id() == node_id_) {
|
|
|
|
base::OS::DebugBreak();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
NodeId node_id_;
|
|
|
|
};
|
|
|
|
|
|
|
|
void CodeAssembler::BreakOnNode(int node_id) {
|
|
|
|
Graph* graph = raw_assembler()->graph();
|
|
|
|
Zone* zone = graph->zone();
|
|
|
|
GraphDecorator* decorator =
|
|
|
|
new (zone) BreakOnNodeDecorator(static_cast<NodeId>(node_id));
|
|
|
|
graph->AddDecorator(decorator);
|
|
|
|
}
|
|
|
|
|
2016-12-28 15:47:34 +00:00
|
|
|
void CodeAssembler::RegisterCallGenerationCallbacks(
|
|
|
|
const CodeAssemblerCallback& call_prologue,
|
|
|
|
const CodeAssemblerCallback& call_epilogue) {
|
|
|
|
// The callback can be registered only once.
|
|
|
|
DCHECK(!state_->call_prologue_);
|
|
|
|
DCHECK(!state_->call_epilogue_);
|
|
|
|
state_->call_prologue_ = call_prologue;
|
|
|
|
state_->call_epilogue_ = call_epilogue;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeAssembler::UnregisterCallGenerationCallbacks() {
|
|
|
|
state_->call_prologue_ = nullptr;
|
|
|
|
state_->call_epilogue_ = nullptr;
|
|
|
|
}
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2016-12-28 15:47:34 +00:00
|
|
|
void CodeAssembler::CallPrologue() {
|
|
|
|
if (state_->call_prologue_) {
|
|
|
|
state_->call_prologue_();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeAssembler::CallEpilogue() {
|
|
|
|
if (state_->call_epilogue_) {
|
|
|
|
state_->call_epilogue_();
|
|
|
|
}
|
|
|
|
}
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2017-11-13 13:04:22 +00:00
|
|
|
bool CodeAssembler::Word32ShiftIsSafe() const {
|
|
|
|
return raw_assembler()->machine()->Word32ShiftIsSafe();
|
|
|
|
}
|
|
|
|
|
2018-04-30 12:13:54 +00:00
|
|
|
PoisoningMitigationLevel CodeAssembler::poisoning_level() const {
|
|
|
|
return raw_assembler()->poisoning_level();
|
2018-03-26 15:44:44 +00:00
|
|
|
}
|
|
|
|
|
2016-11-16 11:48:07 +00:00
|
|
|
// static
|
2018-07-06 08:58:43 +00:00
|
|
|
Handle<Code> CodeAssembler::GenerateCode(CodeAssemblerState* state,
|
|
|
|
const AssemblerOptions& options) {
|
2016-11-16 11:48:07 +00:00
|
|
|
DCHECK(!state->code_generated_);
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2016-11-16 11:48:07 +00:00
|
|
|
RawMachineAssembler* rasm = state->raw_assembler_.get();
|
2017-08-24 13:14:44 +00:00
|
|
|
|
2018-11-13 16:28:21 +00:00
|
|
|
Handle<Code> code;
|
|
|
|
if (FLAG_optimize_csa) {
|
|
|
|
// TODO(tebbi): Support jump rewriting also when FLAG_optimize_csa.
|
|
|
|
DCHECK(!FLAG_turbo_rewrite_far_jumps);
|
|
|
|
Graph* graph = rasm->ExportForOptimization();
|
|
|
|
|
|
|
|
code = Pipeline::GenerateCodeForCodeStub(
|
|
|
|
rasm->isolate(), rasm->call_descriptor(), graph, nullptr,
|
2018-12-07 14:53:22 +00:00
|
|
|
state->kind_, state->name_, state->builtin_index_, nullptr,
|
|
|
|
rasm->poisoning_level(), options)
|
2018-11-13 16:28:21 +00:00
|
|
|
.ToHandleChecked();
|
|
|
|
} else {
|
|
|
|
Schedule* schedule = rasm->Export();
|
2017-08-23 03:08:51 +00:00
|
|
|
|
2018-11-13 16:28:21 +00:00
|
|
|
JumpOptimizationInfo jump_opt;
|
|
|
|
bool should_optimize_jumps =
|
|
|
|
rasm->isolate()->serializer_enabled() && FLAG_turbo_rewrite_far_jumps;
|
2017-08-23 03:08:51 +00:00
|
|
|
|
2018-12-07 14:53:22 +00:00
|
|
|
code = Pipeline::GenerateCodeForCodeStub(
|
|
|
|
rasm->isolate(), rasm->call_descriptor(), rasm->graph(),
|
|
|
|
schedule, state->kind_, state->name_, state->builtin_index_,
|
|
|
|
should_optimize_jumps ? &jump_opt : nullptr,
|
|
|
|
rasm->poisoning_level(), options)
|
|
|
|
.ToHandleChecked();
|
2018-11-13 16:28:21 +00:00
|
|
|
|
|
|
|
if (jump_opt.is_optimizable()) {
|
|
|
|
jump_opt.set_optimizing();
|
|
|
|
|
|
|
|
// Regenerate machine code
|
|
|
|
code = Pipeline::GenerateCodeForCodeStub(
|
|
|
|
rasm->isolate(), rasm->call_descriptor(), rasm->graph(),
|
2018-12-07 14:53:22 +00:00
|
|
|
schedule, state->kind_, state->name_, state->builtin_index_,
|
|
|
|
&jump_opt, rasm->poisoning_level(), options)
|
2018-11-13 16:28:21 +00:00
|
|
|
.ToHandleChecked();
|
|
|
|
}
|
2017-08-23 03:08:51 +00:00
|
|
|
}
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2016-11-16 11:48:07 +00:00
|
|
|
state->code_generated_ = true;
|
2016-04-18 11:57:06 +00:00
|
|
|
return code;
|
|
|
|
}
|
|
|
|
|
2016-11-16 11:48:07 +00:00
|
|
|
bool CodeAssembler::Is64() const { return raw_assembler()->machine()->Is64(); }
|
2016-04-18 11:57:06 +00:00
|
|
|
|
|
|
|
bool CodeAssembler::IsFloat64RoundUpSupported() const {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->machine()->Float64RoundUp().IsSupported();
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeAssembler::IsFloat64RoundDownSupported() const {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->machine()->Float64RoundDown().IsSupported();
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2016-11-30 15:23:49 +00:00
|
|
|
bool CodeAssembler::IsFloat64RoundTiesEvenSupported() const {
|
|
|
|
return raw_assembler()->machine()->Float64RoundTiesEven().IsSupported();
|
|
|
|
}
|
|
|
|
|
2016-04-18 11:57:06 +00:00
|
|
|
bool CodeAssembler::IsFloat64RoundTruncateSupported() const {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->machine()->Float64RoundTruncate().IsSupported();
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-03-31 14:12:46 +00:00
|
|
|
bool CodeAssembler::IsInt32AbsWithOverflowSupported() const {
|
|
|
|
return raw_assembler()->machine()->Int32AbsWithOverflow().IsSupported();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeAssembler::IsInt64AbsWithOverflowSupported() const {
|
|
|
|
return raw_assembler()->machine()->Int64AbsWithOverflow().IsSupported();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeAssembler::IsIntPtrAbsWithOverflowSupported() const {
|
|
|
|
return Is64() ? IsInt64AbsWithOverflowSupported()
|
|
|
|
: IsInt32AbsWithOverflowSupported();
|
|
|
|
}
|
|
|
|
|
2018-05-23 11:19:44 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
void CodeAssembler::GenerateCheckMaybeObjectIsObject(Node* node,
|
|
|
|
const char* location) {
|
|
|
|
Label ok(this);
|
|
|
|
GotoIf(WordNotEqual(WordAnd(BitcastMaybeObjectToWord(node),
|
|
|
|
IntPtrConstant(kHeapObjectTagMask)),
|
|
|
|
IntPtrConstant(kWeakHeapObjectTag)),
|
|
|
|
&ok);
|
|
|
|
Node* message_node = StringConstant(location);
|
|
|
|
DebugAbort(message_node);
|
|
|
|
Unreachable();
|
|
|
|
Bind(&ok);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<Int32T> CodeAssembler::Int32Constant(int32_t value) {
|
|
|
|
return UncheckedCast<Int32T>(raw_assembler()->Int32Constant(value));
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<Int64T> CodeAssembler::Int64Constant(int64_t value) {
|
|
|
|
return UncheckedCast<Int64T>(raw_assembler()->Int64Constant(value));
|
2016-05-19 15:49:03 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<IntPtrT> CodeAssembler::IntPtrConstant(intptr_t value) {
|
|
|
|
return UncheckedCast<IntPtrT>(raw_assembler()->IntPtrConstant(value));
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-10-20 15:18:53 +00:00
|
|
|
TNode<Number> CodeAssembler::NumberConstant(double value) {
|
[turbofan] Lower NumberConstant nodes to IntPtrConstant.
If a NumberConstant can be represented as a Smi, then lower it to a
IntPtrConstant node during simplified lowering. Thanks to this, all backends can
match Smi values that can also be encoded as immediates in the instruction
selector. Additionally, we can apply the same lowering to the CodeAssembler for
the snapshot.
As a result, we can remove `mov` instructions generated because Int32Matcher and
Int64Matcher didn't not recognize Smis:
For 32-bit target, it's common for Smis also be immediates: "if (a < 100) {}"
~~~
mov r1, #200 -> cmp r0, #200
cmp r0, r1 -> blt <>
blt <> ->
~~~
On Arm64 particularly, we lose opportunites to use `cbz`: "if (a == 0) {}"
~~~
movz x0, #0x0 -> cbz x1 <>
cmp x1, x0 ->
b.eq <> ->
~~~
Overall, we do not see an impact on benchmarks such as webtooling. However, we
do see noteworthy code size reduction, from 0.5% to 1.5%.
Bug:
Change-Id: I7fbb718ad51b9036c3514fa31c1326bdd6f2b0e6
Reviewed-on: https://chromium-review.googlesource.com/848814
Reviewed-by: Jaroslav Sevcik <jarin@chromium.org>
Reviewed-by: Benedikt Meurer <bmeurer@chromium.org>
Commit-Queue: Pierre Langlois <pierre.langlois@arm.com>
Cr-Commit-Position: refs/heads/master@{#50569}
2018-01-03 18:30:45 +00:00
|
|
|
int smi_value;
|
|
|
|
if (DoubleToSmiInteger(value, &smi_value)) {
|
|
|
|
return UncheckedCast<Number>(SmiConstant(smi_value));
|
|
|
|
} else {
|
2018-04-26 14:38:00 +00:00
|
|
|
// We allocate the heap number constant eagerly at this point instead of
|
|
|
|
// deferring allocation to code generation
|
|
|
|
// (see AllocateAndInstallRequestedHeapObjects) since that makes it easier
|
|
|
|
// to generate constant lookups for embedded builtins.
|
2018-06-26 11:01:19 +00:00
|
|
|
return UncheckedCast<Number>(
|
|
|
|
HeapConstant(isolate()->factory()->NewHeapNumber(value, TENURED)));
|
[turbofan] Lower NumberConstant nodes to IntPtrConstant.
If a NumberConstant can be represented as a Smi, then lower it to a
IntPtrConstant node during simplified lowering. Thanks to this, all backends can
match Smi values that can also be encoded as immediates in the instruction
selector. Additionally, we can apply the same lowering to the CodeAssembler for
the snapshot.
As a result, we can remove `mov` instructions generated because Int32Matcher and
Int64Matcher didn't not recognize Smis:
For 32-bit target, it's common for Smis also be immediates: "if (a < 100) {}"
~~~
mov r1, #200 -> cmp r0, #200
cmp r0, r1 -> blt <>
blt <> ->
~~~
On Arm64 particularly, we lose opportunites to use `cbz`: "if (a == 0) {}"
~~~
movz x0, #0x0 -> cbz x1 <>
cmp x1, x0 ->
b.eq <> ->
~~~
Overall, we do not see an impact on benchmarks such as webtooling. However, we
do see noteworthy code size reduction, from 0.5% to 1.5%.
Bug:
Change-Id: I7fbb718ad51b9036c3514fa31c1326bdd6f2b0e6
Reviewed-on: https://chromium-review.googlesource.com/848814
Reviewed-by: Jaroslav Sevcik <jarin@chromium.org>
Reviewed-by: Benedikt Meurer <bmeurer@chromium.org>
Commit-Queue: Pierre Langlois <pierre.langlois@arm.com>
Cr-Commit-Position: refs/heads/master@{#50569}
2018-01-03 18:30:45 +00:00
|
|
|
}
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2018-11-03 00:13:22 +00:00
|
|
|
TNode<Smi> CodeAssembler::SmiConstant(Smi value) {
|
|
|
|
return UncheckedCast<Smi>(BitcastWordToTaggedSigned(
|
|
|
|
IntPtrConstant(static_cast<intptr_t>(value.ptr()))));
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<Smi> CodeAssembler::SmiConstant(int value) {
|
2016-10-25 08:30:37 +00:00
|
|
|
return SmiConstant(Smi::FromInt(value));
|
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<HeapObject> CodeAssembler::UntypedHeapConstant(
|
|
|
|
Handle<HeapObject> object) {
|
|
|
|
return UncheckedCast<HeapObject>(raw_assembler()->HeapConstant(object));
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<String> CodeAssembler::StringConstant(const char* str) {
|
2018-02-22 10:11:06 +00:00
|
|
|
Handle<String> internalized_string =
|
|
|
|
factory()->InternalizeOneByteString(OneByteVector(str));
|
|
|
|
return UncheckedCast<String>(HeapConstant(internalized_string));
|
2017-02-24 17:48:49 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<Oddball> CodeAssembler::BooleanConstant(bool value) {
|
2018-10-27 15:41:07 +00:00
|
|
|
Handle<Object> object = isolate()->factory()->ToBoolean(value);
|
|
|
|
return UncheckedCast<Oddball>(
|
|
|
|
raw_assembler()->HeapConstant(Handle<HeapObject>::cast(object)));
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<ExternalReference> CodeAssembler::ExternalConstant(
|
|
|
|
ExternalReference address) {
|
|
|
|
return UncheckedCast<ExternalReference>(
|
|
|
|
raw_assembler()->ExternalConstant(address));
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<Float64T> CodeAssembler::Float64Constant(double value) {
|
|
|
|
return UncheckedCast<Float64T>(raw_assembler()->Float64Constant(value));
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<HeapNumber> CodeAssembler::NaNConstant() {
|
2018-09-20 10:14:46 +00:00
|
|
|
return UncheckedCast<HeapNumber>(LoadRoot(RootIndex::kNanValue));
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2016-05-19 15:49:03 +00:00
|
|
|
bool CodeAssembler::ToInt32Constant(Node* node, int32_t& out_value) {
|
2018-04-13 09:11:42 +00:00
|
|
|
{
|
|
|
|
Int64Matcher m(node);
|
|
|
|
if (m.HasValue() && m.IsInRange(std::numeric_limits<int32_t>::min(),
|
|
|
|
std::numeric_limits<int32_t>::max())) {
|
|
|
|
out_value = static_cast<int32_t>(m.Value());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
Int32Matcher m(node);
|
|
|
|
if (m.HasValue()) {
|
|
|
|
out_value = m.Value();
|
|
|
|
return true;
|
|
|
|
}
|
2016-05-19 15:49:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeAssembler::ToInt64Constant(Node* node, int64_t& out_value) {
|
|
|
|
Int64Matcher m(node);
|
|
|
|
if (m.HasValue()) out_value = m.Value();
|
|
|
|
return m.HasValue();
|
|
|
|
}
|
|
|
|
|
2018-11-03 00:13:22 +00:00
|
|
|
bool CodeAssembler::ToSmiConstant(Node* node, Smi* out_value) {
|
2016-10-18 06:34:40 +00:00
|
|
|
if (node->opcode() == IrOpcode::kBitcastWordToTaggedSigned) {
|
|
|
|
node = node->InputAt(0);
|
|
|
|
}
|
|
|
|
IntPtrMatcher m(node);
|
|
|
|
if (m.HasValue()) {
|
2017-10-17 06:54:48 +00:00
|
|
|
intptr_t value = m.Value();
|
|
|
|
// Make sure that the value is actually a smi
|
|
|
|
CHECK_EQ(0, value & ((static_cast<intptr_t>(1) << kSmiShiftSize) - 1));
|
2018-11-03 00:13:22 +00:00
|
|
|
*out_value = Smi(static_cast<Address>(value));
|
2016-10-18 06:34:40 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-05-19 15:49:03 +00:00
|
|
|
bool CodeAssembler::ToIntPtrConstant(Node* node, intptr_t& out_value) {
|
2016-12-28 16:59:03 +00:00
|
|
|
if (node->opcode() == IrOpcode::kBitcastWordToTaggedSigned ||
|
|
|
|
node->opcode() == IrOpcode::kBitcastWordToTagged) {
|
|
|
|
node = node->InputAt(0);
|
|
|
|
}
|
2016-05-19 15:49:03 +00:00
|
|
|
IntPtrMatcher m(node);
|
|
|
|
if (m.HasValue()) out_value = m.Value();
|
|
|
|
return m.HasValue();
|
|
|
|
}
|
|
|
|
|
2018-04-25 08:14:09 +00:00
|
|
|
bool CodeAssembler::IsUndefinedConstant(TNode<Object> node) {
|
|
|
|
compiler::HeapObjectMatcher m(node);
|
|
|
|
return m.Is(isolate()->factory()->undefined_value());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeAssembler::IsNullConstant(TNode<Object> node) {
|
|
|
|
compiler::HeapObjectMatcher m(node);
|
|
|
|
return m.Is(isolate()->factory()->null_value());
|
|
|
|
}
|
|
|
|
|
2018-06-22 13:32:53 +00:00
|
|
|
Node* CodeAssembler::Parameter(int index) {
|
|
|
|
if (index == kTargetParameterIndex) return raw_assembler()->TargetParameter();
|
|
|
|
return raw_assembler()->Parameter(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CodeAssembler::IsJSFunctionCall() const {
|
|
|
|
auto call_descriptor = raw_assembler()->call_descriptor();
|
|
|
|
return call_descriptor->IsJSFunctionCall();
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<Context> CodeAssembler::GetJSContextParameter() {
|
2018-02-09 19:19:25 +00:00
|
|
|
auto call_descriptor = raw_assembler()->call_descriptor();
|
|
|
|
DCHECK(call_descriptor->IsJSFunctionCall());
|
2017-08-03 14:27:11 +00:00
|
|
|
return CAST(Parameter(Linkage::GetJSCallContextParamIndex(
|
2018-02-09 19:19:25 +00:00
|
|
|
static_cast<int>(call_descriptor->JSParameterCount()))));
|
2017-01-18 19:13:49 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
void CodeAssembler::Return(SloppyTNode<Object> value) {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->Return(value);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
void CodeAssembler::Return(SloppyTNode<Object> value1,
|
|
|
|
SloppyTNode<Object> value2) {
|
2017-02-08 14:26:56 +00:00
|
|
|
return raw_assembler()->Return(value1, value2);
|
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
void CodeAssembler::Return(SloppyTNode<Object> value1,
|
|
|
|
SloppyTNode<Object> value2,
|
|
|
|
SloppyTNode<Object> value3) {
|
2017-02-08 14:26:56 +00:00
|
|
|
return raw_assembler()->Return(value1, value2, value3);
|
|
|
|
}
|
|
|
|
|
2016-11-02 13:15:39 +00:00
|
|
|
void CodeAssembler::PopAndReturn(Node* pop, Node* value) {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->PopAndReturn(pop, value);
|
2016-11-02 13:15:39 +00:00
|
|
|
}
|
|
|
|
|
[async-iteration] implement AsyncGenerator
- Introduce new struct AsyncGeneratorRequest, which holds
information pertinent to resuming execution of an
AsyncGenerator, such as the Promise associated with the async
generator request. It is intended to be used as a singly
linked list, and holds a pointer to the next item in te queue.
- Introduce JSAsyncGeneratorObject (subclass of
JSGeneratorObject), which includes several new internal fields
(`queue` which contains a singly linked list of
AsyncGeneratorRequest objects, and `await_input` which
contains the sent value from an Await expression (This is
necessary to prevent function.sent (used by yield*) from
having the sent value observably overwritten during
execution).
- Modify SuspendGenerator to accept a set of Flags, which
indicate whether the suspend is for a Yield or Await, and
whether it takes place on an async generator or ES6
generator.
- Introduce interpreter intrinsics and TF intrinsic lowering for
accessing the await input of an async generator
- Modify the JSGeneratorStore operator to understand whether or
not it's suspending for a normal yield, or an AsyncGenerator
Await. This ensures appropriate registers are stored.
- Add versions of ResumeGeneratorTrampoline which store the
input value in a different field depending on wether it's an
AsyncGenerator Await resume, or an ordinary resume. Also modifies
whether debug code will assert that the generator object is a
JSGeneratorObject or a JSAsyncGeneratorObject depending on the
resume type.
BUG=v8:5855
R=bmeurer@chromium.org, rmcilroy@chromium.org, jgruber@chromium.org,
littledan@chromium.org, neis@chromium.org
TBR=marja@chromium.org
Change-Id: I9d58df1d344465fc937fe7eed322424204497187
Reviewed-on: https://chromium-review.googlesource.com/446961
Commit-Queue: Caitlin Potter <caitp@igalia.com>
Reviewed-by: Ross McIlroy <rmcilroy@chromium.org>
Reviewed-by: Hannes Payer <hpayer@chromium.org>
Reviewed-by: Benedikt Meurer <bmeurer@chromium.org>
Reviewed-by: Jakob Gruber <jgruber@chromium.org>
Cr-Commit-Position: refs/heads/master@{#44240}
2017-03-29 13:41:45 +00:00
|
|
|
void CodeAssembler::ReturnIf(Node* condition, Node* value) {
|
|
|
|
Label if_return(this), if_continue(this);
|
|
|
|
Branch(condition, &if_return, &if_continue);
|
|
|
|
Bind(&if_return);
|
|
|
|
Return(value);
|
|
|
|
Bind(&if_continue);
|
|
|
|
}
|
|
|
|
|
2018-06-21 16:17:31 +00:00
|
|
|
void CodeAssembler::ReturnRaw(Node* value) {
|
|
|
|
return raw_assembler()->Return(value);
|
|
|
|
}
|
|
|
|
|
2017-08-23 14:59:39 +00:00
|
|
|
void CodeAssembler::DebugAbort(Node* message) {
|
|
|
|
raw_assembler()->DebugAbort(message);
|
|
|
|
}
|
|
|
|
|
2016-11-16 11:48:07 +00:00
|
|
|
void CodeAssembler::DebugBreak() { raw_assembler()->DebugBreak(); }
|
2016-05-19 09:46:34 +00:00
|
|
|
|
2017-02-22 12:02:34 +00:00
|
|
|
void CodeAssembler::Unreachable() {
|
|
|
|
DebugBreak();
|
|
|
|
raw_assembler()->Unreachable();
|
|
|
|
}
|
|
|
|
|
2018-12-04 21:21:28 +00:00
|
|
|
void CodeAssembler::Comment(const char* format, ...) {
|
2016-06-09 07:25:53 +00:00
|
|
|
if (!FLAG_code_comments) return;
|
2018-12-04 21:21:28 +00:00
|
|
|
char buffer[4 * KB];
|
|
|
|
StringBuilder builder(buffer, arraysize(buffer));
|
|
|
|
va_list arguments;
|
|
|
|
va_start(arguments, format);
|
|
|
|
builder.AddFormattedList(format, arguments);
|
|
|
|
va_end(arguments);
|
|
|
|
|
|
|
|
// Copy the string before recording it in the assembler to avoid
|
|
|
|
// issues when the stack allocated buffer goes out of scope.
|
|
|
|
const int prefix_len = 2;
|
|
|
|
int length = builder.position() + 1;
|
|
|
|
char* copy = reinterpret_cast<char*>(malloc(length + prefix_len));
|
|
|
|
LSAN_IGNORE_OBJECT(copy);
|
|
|
|
MemCopy(copy + prefix_len, builder.Finalize(), length);
|
|
|
|
copy[0] = ';';
|
|
|
|
copy[1] = ' ';
|
|
|
|
raw_assembler()->Comment(copy);
|
2016-06-09 07:25:53 +00:00
|
|
|
}
|
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
void CodeAssembler::Bind(Label* label) { return label->Bind(); }
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2017-04-05 17:40:36 +00:00
|
|
|
#if DEBUG
|
|
|
|
void CodeAssembler::Bind(Label* label, AssemblerDebugInfo debug_info) {
|
|
|
|
return label->Bind(debug_info);
|
|
|
|
}
|
|
|
|
#endif // DEBUG
|
|
|
|
|
2016-04-18 11:57:06 +00:00
|
|
|
Node* CodeAssembler::LoadFramePointer() {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->LoadFramePointer();
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* CodeAssembler::LoadParentFramePointer() {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->LoadParentFramePointer();
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* CodeAssembler::LoadStackPointer() {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->LoadStackPointer();
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2018-04-30 12:13:54 +00:00
|
|
|
TNode<Object> CodeAssembler::TaggedPoisonOnSpeculation(
|
2018-03-26 15:44:44 +00:00
|
|
|
SloppyTNode<Object> value) {
|
|
|
|
return UncheckedCast<Object>(
|
2018-04-30 12:13:54 +00:00
|
|
|
raw_assembler()->TaggedPoisonOnSpeculation(value));
|
2018-03-26 15:44:44 +00:00
|
|
|
}
|
|
|
|
|
2018-04-30 12:13:54 +00:00
|
|
|
TNode<WordT> CodeAssembler::WordPoisonOnSpeculation(SloppyTNode<WordT> value) {
|
|
|
|
return UncheckedCast<WordT>(raw_assembler()->WordPoisonOnSpeculation(value));
|
2018-02-11 19:17:27 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
#define DEFINE_CODE_ASSEMBLER_BINARY_OP(name, ResType, Arg1Type, Arg2Type) \
|
|
|
|
TNode<ResType> CodeAssembler::name(SloppyTNode<Arg1Type> a, \
|
|
|
|
SloppyTNode<Arg2Type> b) { \
|
|
|
|
return UncheckedCast<ResType>(raw_assembler()->name(a, b)); \
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
CODE_ASSEMBLER_BINARY_OP_LIST(DEFINE_CODE_ASSEMBLER_BINARY_OP)
|
|
|
|
#undef DEFINE_CODE_ASSEMBLER_BINARY_OP
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<WordT> CodeAssembler::IntPtrAdd(SloppyTNode<WordT> left,
|
|
|
|
SloppyTNode<WordT> right) {
|
2016-12-28 16:59:03 +00:00
|
|
|
intptr_t left_constant;
|
|
|
|
bool is_left_constant = ToIntPtrConstant(left, left_constant);
|
|
|
|
intptr_t right_constant;
|
|
|
|
bool is_right_constant = ToIntPtrConstant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return IntPtrConstant(left_constant + right_constant);
|
|
|
|
}
|
|
|
|
if (left_constant == 0) {
|
|
|
|
return right;
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
2017-08-03 14:27:11 +00:00
|
|
|
return UncheckedCast<WordT>(raw_assembler()->IntPtrAdd(left, right));
|
2016-12-28 16:59:03 +00:00
|
|
|
}
|
|
|
|
|
2018-10-18 12:53:22 +00:00
|
|
|
TNode<IntPtrT> CodeAssembler::IntPtrDiv(TNode<IntPtrT> left,
|
|
|
|
TNode<IntPtrT> right) {
|
|
|
|
intptr_t left_constant;
|
|
|
|
bool is_left_constant = ToIntPtrConstant(left, left_constant);
|
|
|
|
intptr_t right_constant;
|
|
|
|
bool is_right_constant = ToIntPtrConstant(right, right_constant);
|
|
|
|
if (is_right_constant) {
|
|
|
|
if (is_left_constant) {
|
|
|
|
return IntPtrConstant(left_constant / right_constant);
|
|
|
|
}
|
|
|
|
if (base::bits::IsPowerOfTwo(right_constant)) {
|
|
|
|
return WordSar(left, WhichPowerOf2(right_constant));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<IntPtrT>(raw_assembler()->IntPtrDiv(left, right));
|
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<WordT> CodeAssembler::IntPtrSub(SloppyTNode<WordT> left,
|
|
|
|
SloppyTNode<WordT> right) {
|
2016-12-28 16:59:03 +00:00
|
|
|
intptr_t left_constant;
|
|
|
|
bool is_left_constant = ToIntPtrConstant(left, left_constant);
|
|
|
|
intptr_t right_constant;
|
|
|
|
bool is_right_constant = ToIntPtrConstant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return IntPtrConstant(left_constant - right_constant);
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
2017-08-03 14:27:11 +00:00
|
|
|
return UncheckedCast<IntPtrT>(raw_assembler()->IntPtrSub(left, right));
|
2016-12-28 16:59:03 +00:00
|
|
|
}
|
|
|
|
|
2017-09-11 13:44:42 +00:00
|
|
|
TNode<WordT> CodeAssembler::IntPtrMul(SloppyTNode<WordT> left,
|
|
|
|
SloppyTNode<WordT> right) {
|
|
|
|
intptr_t left_constant;
|
|
|
|
bool is_left_constant = ToIntPtrConstant(left, left_constant);
|
|
|
|
intptr_t right_constant;
|
|
|
|
bool is_right_constant = ToIntPtrConstant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return IntPtrConstant(left_constant * right_constant);
|
|
|
|
}
|
2017-11-10 16:11:17 +00:00
|
|
|
if (base::bits::IsPowerOfTwo(left_constant)) {
|
|
|
|
return WordShl(right, WhichPowerOf2(left_constant));
|
2017-09-11 13:44:42 +00:00
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
2017-11-10 16:11:17 +00:00
|
|
|
if (base::bits::IsPowerOfTwo(right_constant)) {
|
|
|
|
return WordShl(left, WhichPowerOf2(right_constant));
|
2017-09-11 13:44:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<IntPtrT>(raw_assembler()->IntPtrMul(left, right));
|
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<WordT> CodeAssembler::WordShl(SloppyTNode<WordT> value, int shift) {
|
|
|
|
return (shift != 0) ? WordShl(value, IntPtrConstant(shift)) : value;
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<WordT> CodeAssembler::WordShr(SloppyTNode<WordT> value, int shift) {
|
|
|
|
return (shift != 0) ? WordShr(value, IntPtrConstant(shift)) : value;
|
2016-07-20 08:07:04 +00:00
|
|
|
}
|
|
|
|
|
2018-03-22 16:09:55 +00:00
|
|
|
TNode<WordT> CodeAssembler::WordSar(SloppyTNode<WordT> value, int shift) {
|
|
|
|
return (shift != 0) ? WordSar(value, IntPtrConstant(shift)) : value;
|
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<Word32T> CodeAssembler::Word32Shr(SloppyTNode<Word32T> value, int shift) {
|
|
|
|
return (shift != 0) ? Word32Shr(value, Int32Constant(shift)) : value;
|
2016-04-29 14:19:22 +00:00
|
|
|
}
|
|
|
|
|
2017-09-11 13:44:42 +00:00
|
|
|
TNode<WordT> CodeAssembler::WordOr(SloppyTNode<WordT> left,
|
|
|
|
SloppyTNode<WordT> right) {
|
|
|
|
intptr_t left_constant;
|
|
|
|
bool is_left_constant = ToIntPtrConstant(left, left_constant);
|
|
|
|
intptr_t right_constant;
|
|
|
|
bool is_right_constant = ToIntPtrConstant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return IntPtrConstant(left_constant | right_constant);
|
|
|
|
}
|
|
|
|
if (left_constant == 0) {
|
|
|
|
return right;
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<WordT>(raw_assembler()->WordOr(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<WordT> CodeAssembler::WordAnd(SloppyTNode<WordT> left,
|
|
|
|
SloppyTNode<WordT> right) {
|
|
|
|
intptr_t left_constant;
|
|
|
|
bool is_left_constant = ToIntPtrConstant(left, left_constant);
|
|
|
|
intptr_t right_constant;
|
|
|
|
bool is_right_constant = ToIntPtrConstant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return IntPtrConstant(left_constant & right_constant);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<WordT>(raw_assembler()->WordAnd(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<WordT> CodeAssembler::WordXor(SloppyTNode<WordT> left,
|
|
|
|
SloppyTNode<WordT> right) {
|
|
|
|
intptr_t left_constant;
|
|
|
|
bool is_left_constant = ToIntPtrConstant(left, left_constant);
|
|
|
|
intptr_t right_constant;
|
|
|
|
bool is_right_constant = ToIntPtrConstant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return IntPtrConstant(left_constant ^ right_constant);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<WordT>(raw_assembler()->WordXor(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<WordT> CodeAssembler::WordShl(SloppyTNode<WordT> left,
|
|
|
|
SloppyTNode<IntegralT> right) {
|
|
|
|
intptr_t left_constant;
|
|
|
|
bool is_left_constant = ToIntPtrConstant(left, left_constant);
|
|
|
|
intptr_t right_constant;
|
|
|
|
bool is_right_constant = ToIntPtrConstant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return IntPtrConstant(left_constant << right_constant);
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<WordT>(raw_assembler()->WordShl(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<WordT> CodeAssembler::WordShr(SloppyTNode<WordT> left,
|
|
|
|
SloppyTNode<IntegralT> right) {
|
|
|
|
intptr_t left_constant;
|
|
|
|
bool is_left_constant = ToIntPtrConstant(left, left_constant);
|
|
|
|
intptr_t right_constant;
|
|
|
|
bool is_right_constant = ToIntPtrConstant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return IntPtrConstant(static_cast<uintptr_t>(left_constant) >>
|
|
|
|
right_constant);
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<WordT>(raw_assembler()->WordShr(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<WordT> CodeAssembler::WordSar(SloppyTNode<WordT> left,
|
|
|
|
SloppyTNode<IntegralT> right) {
|
|
|
|
intptr_t left_constant;
|
|
|
|
bool is_left_constant = ToIntPtrConstant(left, left_constant);
|
|
|
|
intptr_t right_constant;
|
|
|
|
bool is_right_constant = ToIntPtrConstant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return IntPtrConstant(left_constant >> right_constant);
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<WordT>(raw_assembler()->WordSar(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word32T> CodeAssembler::Word32Or(SloppyTNode<Word32T> left,
|
|
|
|
SloppyTNode<Word32T> right) {
|
|
|
|
int32_t left_constant;
|
|
|
|
bool is_left_constant = ToInt32Constant(left, left_constant);
|
|
|
|
int32_t right_constant;
|
|
|
|
bool is_right_constant = ToInt32Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int32Constant(left_constant | right_constant);
|
|
|
|
}
|
|
|
|
if (left_constant == 0) {
|
|
|
|
return right;
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word32T>(raw_assembler()->Word32Or(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word32T> CodeAssembler::Word32And(SloppyTNode<Word32T> left,
|
|
|
|
SloppyTNode<Word32T> right) {
|
|
|
|
int32_t left_constant;
|
|
|
|
bool is_left_constant = ToInt32Constant(left, left_constant);
|
|
|
|
int32_t right_constant;
|
|
|
|
bool is_right_constant = ToInt32Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int32Constant(left_constant & right_constant);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word32T>(raw_assembler()->Word32And(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word32T> CodeAssembler::Word32Xor(SloppyTNode<Word32T> left,
|
|
|
|
SloppyTNode<Word32T> right) {
|
|
|
|
int32_t left_constant;
|
|
|
|
bool is_left_constant = ToInt32Constant(left, left_constant);
|
|
|
|
int32_t right_constant;
|
|
|
|
bool is_right_constant = ToInt32Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int32Constant(left_constant ^ right_constant);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word32T>(raw_assembler()->Word32Xor(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word32T> CodeAssembler::Word32Shl(SloppyTNode<Word32T> left,
|
|
|
|
SloppyTNode<Word32T> right) {
|
|
|
|
int32_t left_constant;
|
|
|
|
bool is_left_constant = ToInt32Constant(left, left_constant);
|
|
|
|
int32_t right_constant;
|
|
|
|
bool is_right_constant = ToInt32Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int32Constant(left_constant << right_constant);
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word32T>(raw_assembler()->Word32Shl(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word32T> CodeAssembler::Word32Shr(SloppyTNode<Word32T> left,
|
|
|
|
SloppyTNode<Word32T> right) {
|
|
|
|
int32_t left_constant;
|
|
|
|
bool is_left_constant = ToInt32Constant(left, left_constant);
|
|
|
|
int32_t right_constant;
|
|
|
|
bool is_right_constant = ToInt32Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int32Constant(static_cast<uint32_t>(left_constant) >>
|
|
|
|
right_constant);
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word32T>(raw_assembler()->Word32Shr(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word32T> CodeAssembler::Word32Sar(SloppyTNode<Word32T> left,
|
|
|
|
SloppyTNode<Word32T> right) {
|
|
|
|
int32_t left_constant;
|
|
|
|
bool is_left_constant = ToInt32Constant(left, left_constant);
|
|
|
|
int32_t right_constant;
|
|
|
|
bool is_right_constant = ToInt32Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int32Constant(left_constant >> right_constant);
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word32T>(raw_assembler()->Word32Sar(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word64T> CodeAssembler::Word64Or(SloppyTNode<Word64T> left,
|
|
|
|
SloppyTNode<Word64T> right) {
|
|
|
|
int64_t left_constant;
|
|
|
|
bool is_left_constant = ToInt64Constant(left, left_constant);
|
|
|
|
int64_t right_constant;
|
|
|
|
bool is_right_constant = ToInt64Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int64Constant(left_constant | right_constant);
|
|
|
|
}
|
|
|
|
if (left_constant == 0) {
|
|
|
|
return right;
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word64T>(raw_assembler()->Word64Or(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word64T> CodeAssembler::Word64And(SloppyTNode<Word64T> left,
|
|
|
|
SloppyTNode<Word64T> right) {
|
|
|
|
int64_t left_constant;
|
|
|
|
bool is_left_constant = ToInt64Constant(left, left_constant);
|
|
|
|
int64_t right_constant;
|
|
|
|
bool is_right_constant = ToInt64Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int64Constant(left_constant & right_constant);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word64T>(raw_assembler()->Word64And(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word64T> CodeAssembler::Word64Xor(SloppyTNode<Word64T> left,
|
|
|
|
SloppyTNode<Word64T> right) {
|
|
|
|
int64_t left_constant;
|
|
|
|
bool is_left_constant = ToInt64Constant(left, left_constant);
|
|
|
|
int64_t right_constant;
|
|
|
|
bool is_right_constant = ToInt64Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int64Constant(left_constant ^ right_constant);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word64T>(raw_assembler()->Word64Xor(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word64T> CodeAssembler::Word64Shl(SloppyTNode<Word64T> left,
|
|
|
|
SloppyTNode<Word64T> right) {
|
|
|
|
int64_t left_constant;
|
|
|
|
bool is_left_constant = ToInt64Constant(left, left_constant);
|
|
|
|
int64_t right_constant;
|
|
|
|
bool is_right_constant = ToInt64Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int64Constant(left_constant << right_constant);
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word64T>(raw_assembler()->Word64Shl(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word64T> CodeAssembler::Word64Shr(SloppyTNode<Word64T> left,
|
|
|
|
SloppyTNode<Word64T> right) {
|
|
|
|
int64_t left_constant;
|
|
|
|
bool is_left_constant = ToInt64Constant(left, left_constant);
|
|
|
|
int64_t right_constant;
|
|
|
|
bool is_right_constant = ToInt64Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int64Constant(static_cast<uint64_t>(left_constant) >>
|
|
|
|
right_constant);
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word64T>(raw_assembler()->Word64Shr(left, right));
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Word64T> CodeAssembler::Word64Sar(SloppyTNode<Word64T> left,
|
|
|
|
SloppyTNode<Word64T> right) {
|
|
|
|
int64_t left_constant;
|
|
|
|
bool is_left_constant = ToInt64Constant(left, left_constant);
|
|
|
|
int64_t right_constant;
|
|
|
|
bool is_right_constant = ToInt64Constant(right, right_constant);
|
|
|
|
if (is_left_constant) {
|
|
|
|
if (is_right_constant) {
|
|
|
|
return Int64Constant(left_constant >> right_constant);
|
|
|
|
}
|
|
|
|
} else if (is_right_constant) {
|
|
|
|
if (right_constant == 0) {
|
|
|
|
return left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UncheckedCast<Word64T>(raw_assembler()->Word64Sar(left, right));
|
|
|
|
}
|
|
|
|
|
2018-06-13 15:45:16 +00:00
|
|
|
#define CODE_ASSEMBLER_COMPARE(Name, ArgT, VarT, ToConstant, op) \
|
|
|
|
TNode<BoolT> CodeAssembler::Name(SloppyTNode<ArgT> left, \
|
|
|
|
SloppyTNode<ArgT> right) { \
|
|
|
|
VarT lhs, rhs; \
|
|
|
|
if (ToConstant(left, lhs) && ToConstant(right, rhs)) { \
|
|
|
|
return BoolConstant(lhs op rhs); \
|
|
|
|
} \
|
|
|
|
return UncheckedCast<BoolT>(raw_assembler()->Name(left, right)); \
|
|
|
|
}
|
|
|
|
|
|
|
|
CODE_ASSEMBLER_COMPARE(IntPtrEqual, WordT, intptr_t, ToIntPtrConstant, ==)
|
|
|
|
CODE_ASSEMBLER_COMPARE(WordEqual, WordT, intptr_t, ToIntPtrConstant, ==)
|
|
|
|
CODE_ASSEMBLER_COMPARE(WordNotEqual, WordT, intptr_t, ToIntPtrConstant, !=)
|
|
|
|
CODE_ASSEMBLER_COMPARE(Word32Equal, Word32T, int32_t, ToInt32Constant, ==)
|
|
|
|
CODE_ASSEMBLER_COMPARE(Word32NotEqual, Word32T, int32_t, ToInt32Constant, !=)
|
|
|
|
CODE_ASSEMBLER_COMPARE(Word64Equal, Word64T, int64_t, ToInt64Constant, ==)
|
|
|
|
CODE_ASSEMBLER_COMPARE(Word64NotEqual, Word64T, int64_t, ToInt64Constant, !=)
|
|
|
|
#undef CODE_ASSEMBLER_COMPARE
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<UintPtrT> CodeAssembler::ChangeUint32ToWord(SloppyTNode<Word32T> value) {
|
2016-11-16 11:48:07 +00:00
|
|
|
if (raw_assembler()->machine()->Is64()) {
|
2017-08-03 14:27:11 +00:00
|
|
|
return UncheckedCast<UintPtrT>(
|
|
|
|
raw_assembler()->ChangeUint32ToUint64(value));
|
2016-04-18 19:41:09 +00:00
|
|
|
}
|
2017-10-20 15:18:53 +00:00
|
|
|
return ReinterpretCast<UintPtrT>(value);
|
2016-04-18 19:41:09 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
TNode<IntPtrT> CodeAssembler::ChangeInt32ToIntPtr(SloppyTNode<Word32T> value) {
|
2016-11-16 11:48:07 +00:00
|
|
|
if (raw_assembler()->machine()->Is64()) {
|
2017-10-20 15:18:53 +00:00
|
|
|
return ReinterpretCast<IntPtrT>(raw_assembler()->ChangeInt32ToInt64(value));
|
2016-04-22 09:17:58 +00:00
|
|
|
}
|
2017-10-20 15:18:53 +00:00
|
|
|
return ReinterpretCast<IntPtrT>(value);
|
2016-04-22 09:17:58 +00:00
|
|
|
}
|
|
|
|
|
2017-10-20 15:18:53 +00:00
|
|
|
TNode<UintPtrT> CodeAssembler::ChangeFloat64ToUintPtr(
|
|
|
|
SloppyTNode<Float64T> value) {
|
2017-03-13 13:40:25 +00:00
|
|
|
if (raw_assembler()->machine()->Is64()) {
|
2017-10-20 15:18:53 +00:00
|
|
|
return ReinterpretCast<UintPtrT>(
|
|
|
|
raw_assembler()->ChangeFloat64ToUint64(value));
|
2017-03-13 13:40:25 +00:00
|
|
|
}
|
2017-10-20 15:18:53 +00:00
|
|
|
return ReinterpretCast<UintPtrT>(
|
|
|
|
raw_assembler()->ChangeFloat64ToUint32(value));
|
2017-03-13 13:40:25 +00:00
|
|
|
}
|
|
|
|
|
2018-09-13 17:18:18 +00:00
|
|
|
TNode<Float64T> CodeAssembler::ChangeUintPtrToFloat64(TNode<UintPtrT> value) {
|
|
|
|
if (raw_assembler()->machine()->Is64()) {
|
|
|
|
// TODO(turbofan): Maybe we should introduce a ChangeUint64ToFloat64
|
|
|
|
// machine operator to TurboFan here?
|
|
|
|
return ReinterpretCast<Float64T>(
|
|
|
|
raw_assembler()->RoundUint64ToFloat64(value));
|
|
|
|
}
|
|
|
|
return ReinterpretCast<Float64T>(
|
|
|
|
raw_assembler()->ChangeUint32ToFloat64(value));
|
|
|
|
}
|
|
|
|
|
2016-09-06 16:17:08 +00:00
|
|
|
Node* CodeAssembler::RoundIntPtrToFloat64(Node* value) {
|
2016-11-16 11:48:07 +00:00
|
|
|
if (raw_assembler()->machine()->Is64()) {
|
|
|
|
return raw_assembler()->RoundInt64ToFloat64(value);
|
2016-09-06 16:17:08 +00:00
|
|
|
}
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->ChangeInt32ToFloat64(value);
|
2016-09-06 16:17:08 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
#define DEFINE_CODE_ASSEMBLER_UNARY_OP(name, ResType, ArgType) \
|
|
|
|
TNode<ResType> CodeAssembler::name(SloppyTNode<ArgType> a) { \
|
|
|
|
return UncheckedCast<ResType>(raw_assembler()->name(a)); \
|
|
|
|
}
|
2016-04-18 11:57:06 +00:00
|
|
|
CODE_ASSEMBLER_UNARY_OP_LIST(DEFINE_CODE_ASSEMBLER_UNARY_OP)
|
|
|
|
#undef DEFINE_CODE_ASSEMBLER_UNARY_OP
|
|
|
|
|
2018-03-26 15:44:44 +00:00
|
|
|
Node* CodeAssembler::Load(MachineType rep, Node* base,
|
|
|
|
LoadSensitivity needs_poisoning) {
|
|
|
|
return raw_assembler()->Load(rep, base, needs_poisoning);
|
2016-04-21 06:45:44 +00:00
|
|
|
}
|
|
|
|
|
2018-03-26 15:44:44 +00:00
|
|
|
Node* CodeAssembler::Load(MachineType rep, Node* base, Node* offset,
|
|
|
|
LoadSensitivity needs_poisoning) {
|
|
|
|
return raw_assembler()->Load(rep, base, offset, needs_poisoning);
|
2016-04-21 06:45:44 +00:00
|
|
|
}
|
|
|
|
|
2016-12-01 13:09:17 +00:00
|
|
|
Node* CodeAssembler::AtomicLoad(MachineType rep, Node* base, Node* offset) {
|
|
|
|
return raw_assembler()->AtomicLoad(rep, base, offset);
|
2016-04-21 06:45:44 +00:00
|
|
|
}
|
|
|
|
|
2018-09-20 10:14:46 +00:00
|
|
|
TNode<Object> CodeAssembler::LoadRoot(RootIndex root_index) {
|
2018-10-09 23:19:09 +00:00
|
|
|
if (RootsTable::IsImmortalImmovable(root_index)) {
|
2018-10-11 08:43:21 +00:00
|
|
|
Handle<Object> root = isolate()->root_handle(root_index);
|
2016-04-18 11:57:06 +00:00
|
|
|
if (root->IsSmi()) {
|
|
|
|
return SmiConstant(Smi::cast(*root));
|
|
|
|
} else {
|
|
|
|
return HeapConstant(Handle<HeapObject>::cast(root));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-12 11:49:58 +00:00
|
|
|
// TODO(jgruber): In theory we could generate better code for this by
|
|
|
|
// letting the macro assembler decide how to load from the roots list. In most
|
|
|
|
// cases, it would boil down to loading from a fixed kRootRegister offset.
|
2018-10-24 22:52:04 +00:00
|
|
|
Node* isolate_root =
|
|
|
|
ExternalConstant(ExternalReference::isolate_root(isolate()));
|
|
|
|
int offset = IsolateData::root_slot_offset(root_index);
|
|
|
|
return UncheckedCast<Object>(
|
|
|
|
Load(MachineType::AnyTagged(), isolate_root, IntPtrConstant(offset)));
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2016-12-01 13:12:00 +00:00
|
|
|
Node* CodeAssembler::Store(Node* base, Node* value) {
|
|
|
|
return raw_assembler()->Store(MachineRepresentation::kTagged, base, value,
|
|
|
|
kFullWriteBarrier);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2016-12-01 13:12:00 +00:00
|
|
|
Node* CodeAssembler::Store(Node* base, Node* offset, Node* value) {
|
|
|
|
return raw_assembler()->Store(MachineRepresentation::kTagged, base, offset,
|
|
|
|
value, kFullWriteBarrier);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2016-12-01 16:51:09 +00:00
|
|
|
Node* CodeAssembler::StoreWithMapWriteBarrier(Node* base, Node* offset,
|
|
|
|
Node* value) {
|
|
|
|
return raw_assembler()->Store(MachineRepresentation::kTagged, base, offset,
|
|
|
|
value, kMapWriteBarrier);
|
|
|
|
}
|
|
|
|
|
2016-04-18 11:57:06 +00:00
|
|
|
Node* CodeAssembler::StoreNoWriteBarrier(MachineRepresentation rep, Node* base,
|
|
|
|
Node* value) {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->Store(rep, base, value, kNoWriteBarrier);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* CodeAssembler::StoreNoWriteBarrier(MachineRepresentation rep, Node* base,
|
2016-12-01 13:09:17 +00:00
|
|
|
Node* offset, Node* value) {
|
|
|
|
return raw_assembler()->Store(rep, base, offset, value, kNoWriteBarrier);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2016-05-03 17:27:07 +00:00
|
|
|
Node* CodeAssembler::AtomicStore(MachineRepresentation rep, Node* base,
|
[atomics] Implement 64-bit Atomics operations
Atomics.{load,store,add,sub,and,or,xor,exchange,compareExchange}
are updated to support BigInt and BigInt64Array/BigUint64Array inputs.
Atomics.{wait,wake,isLockFree} are left unchanged for now.
Bug: v8:8100
Cq-Include-Trybots: luci.v8.try:v8_linux_noi18n_rel_ng
Change-Id: I8862d7e18c58ae08784535e9c010ba94f067a0ee
Reviewed-on: https://chromium-review.googlesource.com/1237294
Commit-Queue: Jakob Kummerow <jkummerow@chromium.org>
Reviewed-by: Deepti Gandluri <gdeepti@chromium.org>
Reviewed-by: Georg Neis <neis@chromium.org>
Cr-Commit-Position: refs/heads/master@{#56228}
2018-09-26 00:28:59 +00:00
|
|
|
Node* offset, Node* value, Node* value_high) {
|
|
|
|
return raw_assembler()->AtomicStore(rep, base, offset, value, value_high);
|
2016-05-03 17:27:07 +00:00
|
|
|
}
|
|
|
|
|
[atomics] Implement 64-bit Atomics operations
Atomics.{load,store,add,sub,and,or,xor,exchange,compareExchange}
are updated to support BigInt and BigInt64Array/BigUint64Array inputs.
Atomics.{wait,wake,isLockFree} are left unchanged for now.
Bug: v8:8100
Cq-Include-Trybots: luci.v8.try:v8_linux_noi18n_rel_ng
Change-Id: I8862d7e18c58ae08784535e9c010ba94f067a0ee
Reviewed-on: https://chromium-review.googlesource.com/1237294
Commit-Queue: Jakob Kummerow <jkummerow@chromium.org>
Reviewed-by: Deepti Gandluri <gdeepti@chromium.org>
Reviewed-by: Georg Neis <neis@chromium.org>
Cr-Commit-Position: refs/heads/master@{#56228}
2018-09-26 00:28:59 +00:00
|
|
|
#define ATOMIC_FUNCTION(name) \
|
|
|
|
Node* CodeAssembler::Atomic##name(MachineType type, Node* base, \
|
|
|
|
Node* offset, Node* value, \
|
|
|
|
Node* value_high) { \
|
|
|
|
return raw_assembler()->Atomic##name(type, base, offset, value, \
|
|
|
|
value_high); \
|
2017-04-11 00:09:37 +00:00
|
|
|
}
|
|
|
|
ATOMIC_FUNCTION(Exchange);
|
|
|
|
ATOMIC_FUNCTION(Add);
|
|
|
|
ATOMIC_FUNCTION(Sub);
|
|
|
|
ATOMIC_FUNCTION(And);
|
|
|
|
ATOMIC_FUNCTION(Or);
|
|
|
|
ATOMIC_FUNCTION(Xor);
|
|
|
|
#undef ATOMIC_FUNCTION
|
2017-03-07 00:07:44 +00:00
|
|
|
|
2017-03-16 22:29:02 +00:00
|
|
|
Node* CodeAssembler::AtomicCompareExchange(MachineType type, Node* base,
|
|
|
|
Node* offset, Node* old_value,
|
[atomics] Implement 64-bit Atomics operations
Atomics.{load,store,add,sub,and,or,xor,exchange,compareExchange}
are updated to support BigInt and BigInt64Array/BigUint64Array inputs.
Atomics.{wait,wake,isLockFree} are left unchanged for now.
Bug: v8:8100
Cq-Include-Trybots: luci.v8.try:v8_linux_noi18n_rel_ng
Change-Id: I8862d7e18c58ae08784535e9c010ba94f067a0ee
Reviewed-on: https://chromium-review.googlesource.com/1237294
Commit-Queue: Jakob Kummerow <jkummerow@chromium.org>
Reviewed-by: Deepti Gandluri <gdeepti@chromium.org>
Reviewed-by: Georg Neis <neis@chromium.org>
Cr-Commit-Position: refs/heads/master@{#56228}
2018-09-26 00:28:59 +00:00
|
|
|
Node* new_value,
|
|
|
|
Node* old_value_high,
|
|
|
|
Node* new_value_high) {
|
|
|
|
return raw_assembler()->AtomicCompareExchange(
|
|
|
|
type, base, offset, old_value, old_value_high, new_value, new_value_high);
|
2017-03-16 22:29:02 +00:00
|
|
|
}
|
|
|
|
|
2018-09-20 10:14:46 +00:00
|
|
|
Node* CodeAssembler::StoreRoot(RootIndex root_index, Node* value) {
|
2018-10-09 23:19:09 +00:00
|
|
|
DCHECK(!RootsTable::IsImmortalImmovable(root_index));
|
2018-10-24 22:52:04 +00:00
|
|
|
Node* isolate_root =
|
|
|
|
ExternalConstant(ExternalReference::isolate_root(isolate()));
|
|
|
|
int offset = IsolateData::root_slot_offset(root_index);
|
|
|
|
return StoreNoWriteBarrier(MachineRepresentation::kTagged, isolate_root,
|
2018-09-20 10:14:46 +00:00
|
|
|
IntPtrConstant(offset), value);
|
2016-05-17 11:23:59 +00:00
|
|
|
}
|
|
|
|
|
2016-09-14 09:28:04 +00:00
|
|
|
Node* CodeAssembler::Retain(Node* value) {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->Retain(value);
|
2016-09-14 09:28:04 +00:00
|
|
|
}
|
|
|
|
|
2016-04-18 11:57:06 +00:00
|
|
|
Node* CodeAssembler::Projection(int index, Node* value) {
|
2018-11-28 11:54:27 +00:00
|
|
|
DCHECK_LT(index, value->op()->ValueOutputCount());
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->Projection(index, value);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2016-08-22 15:58:01 +00:00
|
|
|
void CodeAssembler::GotoIfException(Node* node, Label* if_exception,
|
|
|
|
Variable* exception_var) {
|
Reland "[builtins] port Promise.all to CSA"
Simplifies the implementation of IteratorClose in IteratorBuiltinsAssembler, and makes clear that it is only invoked when an exception occurs. Adds exception handling support to GetIterator, IteratorStep, and IteratorCloseOnException.
Moves the Promise.all resolveElement closure and it's caller to
builtins-promise-gen.cc.
Instead of creating an internal array (and copying its elements into a
result
array), a single JSArray is allocated, and appended with
BuildAppendJSArray(),
falling back to %CreateDataProperty(), and elements are updated in the
resolve
closure the same way. This should always be unobservable.
This CL increases the size of snapshot_blob.bin on an x64.release build
by 8.51kb
BUG=v8:5343
R=cbruni@chromium.org, gsathysa@chromium.org, jgruber@chromium.org, hpayer@chromium.org, tebbi@chromium.org
Change-Id: I29c4a529154ef49ad65555ce6ddc2c5b7c9de6b3
Reviewed-on: https://chromium-review.googlesource.com/508473
Commit-Queue: Caitlin Potter <caitp@igalia.com>
Reviewed-by: Hannes Payer <hpayer@chromium.org>
Reviewed-by: Tobias Tebbi <tebbi@chromium.org>
Reviewed-by: Camillo Bruni <cbruni@chromium.org>
Reviewed-by: Sathya Gunasekaran <gsathya@chromium.org>
Reviewed-by: Jakob Gruber <jgruber@chromium.org>
Cr-Commit-Position: refs/heads/master@{#45946}
2017-06-13 17:14:57 +00:00
|
|
|
if (if_exception == nullptr) {
|
|
|
|
// If no handler is supplied, don't add continuations
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-10-31 13:00:51 +00:00
|
|
|
// No catch handlers should be active if we're using catch labels
|
|
|
|
DCHECK_EQ(state()->exception_handler_labels_.size(), 0);
|
2018-02-19 12:32:56 +00:00
|
|
|
DCHECK(!node->op()->HasProperty(Operator::kNoThrow));
|
|
|
|
|
2016-08-22 15:58:01 +00:00
|
|
|
Label success(this), exception(this, Label::kDeferred);
|
|
|
|
success.MergeVariables();
|
|
|
|
exception.MergeVariables();
|
|
|
|
|
2016-11-16 11:48:07 +00:00
|
|
|
raw_assembler()->Continuations(node, success.label_, exception.label_);
|
2016-08-22 15:58:01 +00:00
|
|
|
|
|
|
|
Bind(&exception);
|
2016-11-16 11:48:07 +00:00
|
|
|
const Operator* op = raw_assembler()->common()->IfException();
|
|
|
|
Node* exception_value = raw_assembler()->AddNode(op, node, node);
|
2016-08-22 15:58:01 +00:00
|
|
|
if (exception_var != nullptr) {
|
|
|
|
exception_var->Bind(exception_value);
|
|
|
|
}
|
|
|
|
Goto(if_exception);
|
|
|
|
|
|
|
|
Bind(&success);
|
2018-11-13 16:28:21 +00:00
|
|
|
raw_assembler()->AddNode(raw_assembler()->common()->IfSuccess(), node);
|
2016-08-22 15:58:01 +00:00
|
|
|
}
|
|
|
|
|
2018-10-31 13:00:51 +00:00
|
|
|
void CodeAssembler::HandleException(Node* node) {
|
|
|
|
if (state_->exception_handler_labels_.size() == 0) return;
|
|
|
|
CodeAssemblerExceptionHandlerLabel* label =
|
|
|
|
state_->exception_handler_labels_.back();
|
|
|
|
|
|
|
|
if (node->op()->HasProperty(Operator::kNoThrow)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Label success(this), exception(this, Label::kDeferred);
|
|
|
|
success.MergeVariables();
|
|
|
|
exception.MergeVariables();
|
|
|
|
|
|
|
|
raw_assembler()->Continuations(node, success.label_, exception.label_);
|
|
|
|
|
|
|
|
Bind(&exception);
|
|
|
|
const Operator* op = raw_assembler()->common()->IfException();
|
|
|
|
Node* exception_value = raw_assembler()->AddNode(op, node, node);
|
|
|
|
label->AddInputs({UncheckedCast<Object>(exception_value)});
|
|
|
|
Goto(label->plain_label());
|
2018-11-13 16:28:21 +00:00
|
|
|
|
2018-10-31 13:00:51 +00:00
|
|
|
Bind(&success);
|
2018-11-13 16:28:21 +00:00
|
|
|
raw_assembler()->AddNode(raw_assembler()->common()->IfSuccess(), node);
|
2018-10-31 13:00:51 +00:00
|
|
|
}
|
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
namespace {
|
|
|
|
template <size_t kMaxSize>
|
|
|
|
class NodeArray {
|
|
|
|
public:
|
|
|
|
void Add(Node* node) {
|
|
|
|
DCHECK_GT(kMaxSize, size());
|
|
|
|
*ptr_++ = node;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* const* data() const { return arr_; }
|
|
|
|
int size() const { return static_cast<int>(ptr_ - arr_); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
Node* arr_[kMaxSize];
|
|
|
|
Node** ptr_ = arr_;
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
TNode<Object> CodeAssembler::CallRuntimeImpl(
|
|
|
|
Runtime::FunctionId function, TNode<Object> context,
|
|
|
|
std::initializer_list<TNode<Object>> args) {
|
2018-06-21 16:17:31 +00:00
|
|
|
int result_size = Runtime::FunctionForId(function)->result_size;
|
|
|
|
TNode<Code> centry =
|
|
|
|
HeapConstant(CodeFactory::RuntimeCEntry(isolate(), result_size));
|
|
|
|
return CallRuntimeWithCEntryImpl(function, centry, context, args);
|
|
|
|
}
|
|
|
|
|
|
|
|
TNode<Object> CodeAssembler::CallRuntimeWithCEntryImpl(
|
|
|
|
Runtime::FunctionId function, TNode<Code> centry, TNode<Object> context,
|
|
|
|
std::initializer_list<TNode<Object>> args) {
|
2018-06-18 09:58:09 +00:00
|
|
|
constexpr size_t kMaxNumArgs = 6;
|
|
|
|
DCHECK_GE(kMaxNumArgs, args.size());
|
|
|
|
int argc = static_cast<int>(args.size());
|
2018-02-09 19:19:25 +00:00
|
|
|
auto call_descriptor = Linkage::GetRuntimeCallDescriptor(
|
2016-12-16 09:58:51 +00:00
|
|
|
zone(), function, argc, Operator::kNoProperties,
|
|
|
|
CallDescriptor::kNoFlags);
|
|
|
|
|
2018-04-25 07:28:14 +00:00
|
|
|
Node* ref = ExternalConstant(ExternalReference::Create(function));
|
2016-12-16 09:58:51 +00:00
|
|
|
Node* arity = Int32Constant(argc);
|
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
NodeArray<kMaxNumArgs + 4> inputs;
|
|
|
|
inputs.Add(centry);
|
|
|
|
for (auto arg : args) inputs.Add(arg);
|
|
|
|
inputs.Add(ref);
|
|
|
|
inputs.Add(arity);
|
|
|
|
inputs.Add(context);
|
2016-12-16 09:58:51 +00:00
|
|
|
|
2016-12-16 10:40:16 +00:00
|
|
|
CallPrologue();
|
2018-02-09 19:19:25 +00:00
|
|
|
Node* return_value =
|
2018-06-18 09:58:09 +00:00
|
|
|
raw_assembler()->CallN(call_descriptor, inputs.size(), inputs.data());
|
2018-10-31 13:00:51 +00:00
|
|
|
HandleException(return_value);
|
2016-04-18 11:57:06 +00:00
|
|
|
CallEpilogue();
|
2017-08-03 14:27:11 +00:00
|
|
|
return UncheckedCast<Object>(return_value);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
void CodeAssembler::TailCallRuntimeImpl(
|
|
|
|
Runtime::FunctionId function, TNode<Int32T> arity, TNode<Object> context,
|
|
|
|
std::initializer_list<TNode<Object>> args) {
|
2018-06-06 11:20:51 +00:00
|
|
|
int result_size = Runtime::FunctionForId(function)->result_size;
|
2018-06-12 17:30:48 +00:00
|
|
|
TNode<Code> centry =
|
2018-06-06 11:20:51 +00:00
|
|
|
HeapConstant(CodeFactory::RuntimeCEntry(isolate(), result_size));
|
2018-06-18 09:58:09 +00:00
|
|
|
return TailCallRuntimeWithCEntryImpl(function, arity, centry, context, args);
|
2018-06-06 11:20:51 +00:00
|
|
|
}
|
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
void CodeAssembler::TailCallRuntimeWithCEntryImpl(
|
|
|
|
Runtime::FunctionId function, TNode<Int32T> arity, TNode<Code> centry,
|
|
|
|
TNode<Object> context, std::initializer_list<TNode<Object>> args) {
|
|
|
|
constexpr size_t kMaxNumArgs = 6;
|
|
|
|
DCHECK_GE(kMaxNumArgs, args.size());
|
|
|
|
int argc = static_cast<int>(args.size());
|
2018-02-09 19:19:25 +00:00
|
|
|
auto call_descriptor = Linkage::GetRuntimeCallDescriptor(
|
2016-12-16 09:58:51 +00:00
|
|
|
zone(), function, argc, Operator::kNoProperties,
|
2017-11-29 10:48:29 +00:00
|
|
|
CallDescriptor::kNoFlags);
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2018-04-25 07:28:14 +00:00
|
|
|
Node* ref = ExternalConstant(ExternalReference::Create(function));
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
NodeArray<kMaxNumArgs + 4> inputs;
|
|
|
|
inputs.Add(centry);
|
|
|
|
for (auto arg : args) inputs.Add(arg);
|
|
|
|
inputs.Add(ref);
|
|
|
|
inputs.Add(arity);
|
|
|
|
inputs.Add(context);
|
2016-09-07 10:12:52 +00:00
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
raw_assembler()->TailCallN(call_descriptor, inputs.size(), inputs.data());
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2016-12-16 10:40:16 +00:00
|
|
|
Node* CodeAssembler::CallStubN(const CallInterfaceDescriptor& descriptor,
|
|
|
|
size_t result_size, int input_count,
|
2018-06-21 20:05:00 +00:00
|
|
|
Node* const* inputs) {
|
2017-11-06 14:10:18 +00:00
|
|
|
// implicit nodes are target and optionally context.
|
2018-06-21 20:05:00 +00:00
|
|
|
int implicit_nodes = descriptor.HasContextParameter() ? 2 : 1;
|
2017-11-06 14:10:18 +00:00
|
|
|
DCHECK_LE(implicit_nodes, input_count);
|
|
|
|
int argc = input_count - implicit_nodes;
|
2016-12-16 10:40:16 +00:00
|
|
|
DCHECK_LE(descriptor.GetParameterCount(), argc);
|
|
|
|
// Extra arguments not mentioned in the descriptor are passed on the stack.
|
|
|
|
int stack_parameter_count = argc - descriptor.GetRegisterParameterCount();
|
|
|
|
DCHECK_LE(descriptor.GetStackParameterCount(), stack_parameter_count);
|
2018-06-21 08:47:48 +00:00
|
|
|
DCHECK_EQ(result_size, descriptor.GetReturnCount());
|
2018-06-21 20:05:00 +00:00
|
|
|
|
2018-02-09 19:19:25 +00:00
|
|
|
auto call_descriptor = Linkage::GetStubCallDescriptor(
|
2018-06-18 15:14:29 +00:00
|
|
|
zone(), descriptor, stack_parameter_count, CallDescriptor::kNoFlags,
|
2018-06-21 20:05:00 +00:00
|
|
|
Operator::kNoProperties);
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2016-12-16 10:40:16 +00:00
|
|
|
CallPrologue();
|
2018-02-09 19:19:25 +00:00
|
|
|
Node* return_value =
|
|
|
|
raw_assembler()->CallN(call_descriptor, input_count, inputs);
|
2018-10-31 13:00:51 +00:00
|
|
|
HandleException(return_value);
|
2016-12-16 10:40:16 +00:00
|
|
|
CallEpilogue();
|
|
|
|
return return_value;
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2018-06-12 17:30:48 +00:00
|
|
|
void CodeAssembler::TailCallStubImpl(const CallInterfaceDescriptor& descriptor,
|
|
|
|
TNode<Code> target, TNode<Object> context,
|
2018-06-18 09:58:09 +00:00
|
|
|
std::initializer_list<Node*> args) {
|
|
|
|
constexpr size_t kMaxNumArgs = 11;
|
|
|
|
DCHECK_GE(kMaxNumArgs, args.size());
|
|
|
|
DCHECK_EQ(descriptor.GetParameterCount(), args.size());
|
2018-02-09 19:19:25 +00:00
|
|
|
auto call_descriptor = Linkage::GetStubCallDescriptor(
|
2018-06-18 15:14:29 +00:00
|
|
|
zone(), descriptor, descriptor.GetStackParameterCount(),
|
2018-06-21 08:47:48 +00:00
|
|
|
CallDescriptor::kNoFlags, Operator::kNoProperties);
|
2016-09-27 07:20:39 +00:00
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
NodeArray<kMaxNumArgs + 2> inputs;
|
|
|
|
inputs.Add(target);
|
|
|
|
for (auto arg : args) inputs.Add(arg);
|
2018-06-21 20:05:00 +00:00
|
|
|
if (descriptor.HasContextParameter()) {
|
|
|
|
inputs.Add(context);
|
|
|
|
}
|
2018-06-18 09:58:09 +00:00
|
|
|
|
|
|
|
raw_assembler()->TailCallN(call_descriptor, inputs.size(), inputs.data());
|
2016-09-27 07:20:39 +00:00
|
|
|
}
|
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
Node* CodeAssembler::CallStubRImpl(const CallInterfaceDescriptor& descriptor,
|
|
|
|
size_t result_size, SloppyTNode<Code> target,
|
|
|
|
SloppyTNode<Object> context,
|
|
|
|
std::initializer_list<Node*> args) {
|
|
|
|
constexpr size_t kMaxNumArgs = 10;
|
|
|
|
DCHECK_GE(kMaxNumArgs, args.size());
|
2016-10-19 10:11:25 +00:00
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
NodeArray<kMaxNumArgs + 2> inputs;
|
|
|
|
inputs.Add(target);
|
|
|
|
for (auto arg : args) inputs.Add(arg);
|
2018-06-21 20:05:00 +00:00
|
|
|
if (descriptor.HasContextParameter()) {
|
|
|
|
inputs.Add(context);
|
|
|
|
}
|
2018-06-18 09:58:09 +00:00
|
|
|
|
2018-06-21 20:05:00 +00:00
|
|
|
return CallStubN(descriptor, result_size, inputs.size(), inputs.data());
|
2018-06-18 09:58:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* CodeAssembler::TailCallStubThenBytecodeDispatchImpl(
|
2017-08-25 20:56:04 +00:00
|
|
|
const CallInterfaceDescriptor& descriptor, Node* target, Node* context,
|
2018-06-18 09:58:09 +00:00
|
|
|
std::initializer_list<Node*> args) {
|
|
|
|
constexpr size_t kMaxNumArgs = 6;
|
|
|
|
DCHECK_GE(kMaxNumArgs, args.size());
|
|
|
|
|
|
|
|
DCHECK_LE(descriptor.GetParameterCount(), args.size());
|
|
|
|
int argc = static_cast<int>(args.size());
|
2017-08-25 20:56:04 +00:00
|
|
|
// Extra arguments not mentioned in the descriptor are passed on the stack.
|
2018-06-18 09:58:09 +00:00
|
|
|
int stack_parameter_count = argc - descriptor.GetRegisterParameterCount();
|
2017-08-25 20:56:04 +00:00
|
|
|
DCHECK_LE(descriptor.GetStackParameterCount(), stack_parameter_count);
|
2018-02-09 19:19:25 +00:00
|
|
|
auto call_descriptor = Linkage::GetStubCallDescriptor(
|
2018-06-18 15:14:29 +00:00
|
|
|
zone(), descriptor, stack_parameter_count, CallDescriptor::kNoFlags,
|
2018-06-21 08:47:48 +00:00
|
|
|
Operator::kNoProperties);
|
2017-08-25 20:56:04 +00:00
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
NodeArray<kMaxNumArgs + 2> inputs;
|
|
|
|
inputs.Add(target);
|
|
|
|
for (auto arg : args) inputs.Add(arg);
|
|
|
|
inputs.Add(context);
|
2017-08-25 20:56:04 +00:00
|
|
|
|
2018-06-18 09:58:09 +00:00
|
|
|
return raw_assembler()->TailCallN(call_descriptor, inputs.size(),
|
|
|
|
inputs.data());
|
|
|
|
}
|
2017-08-25 20:56:04 +00:00
|
|
|
|
2016-12-16 12:58:19 +00:00
|
|
|
template <class... TArgs>
|
2016-04-18 11:57:06 +00:00
|
|
|
Node* CodeAssembler::TailCallBytecodeDispatch(
|
2016-12-16 12:58:19 +00:00
|
|
|
const CallInterfaceDescriptor& descriptor, Node* target, TArgs... args) {
|
|
|
|
DCHECK_EQ(descriptor.GetParameterCount(), sizeof...(args));
|
2018-02-09 19:19:25 +00:00
|
|
|
auto call_descriptor = Linkage::GetBytecodeDispatchCallDescriptor(
|
2018-06-18 15:14:29 +00:00
|
|
|
zone(), descriptor, descriptor.GetStackParameterCount());
|
2016-12-16 12:58:19 +00:00
|
|
|
|
|
|
|
Node* nodes[] = {target, args...};
|
[builtins] Separate Array.prototype.* CSA builtins into two parts
Previous to this CL, CSA-optimized Array builtins--like forEach, some, and
every--were written in a single, monolithic block of CSA code.
This CL teases the code for each of these builtins apart into two chunks, a main
body with optimizations for fast cases, and a "continuation" builtin that
performs a spec-compliant, but slower version of the main loop of the
builtin. The general idea is that when the "fast" main body builtin encounters
an unexpected condition that invalidates assumptions allowing fast-case code, it
tail calls to the slow, correct version of the loop that finishes the builtin
execution.
This separation currently doens't really provide any specific advantage over the
combined version. However, it paves the way to TF-optimized inlined Array
builtins. Inlined Array builtins may trigger deopts during the execution of the
builtin's loop, and those deopt must continue execution from the point at which
they failed. With some massaging of the deoptimizer, it will be possible to make
those deopt points create an extra frame on the top of the stack which resumes
execution in the slow-loop builtin created in this CL.
BUG=v8:1956
LOG=N
Review-Url: https://codereview.chromium.org/2753793002
Cr-Commit-Position: refs/heads/master@{#43867}
2017-03-16 15:34:01 +00:00
|
|
|
CHECK_EQ(descriptor.GetParameterCount() + 1, arraysize(nodes));
|
2018-02-09 19:19:25 +00:00
|
|
|
return raw_assembler()->TailCallN(call_descriptor, arraysize(nodes), nodes);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
[builtins] Separate Array.prototype.* CSA builtins into two parts
Previous to this CL, CSA-optimized Array builtins--like forEach, some, and
every--were written in a single, monolithic block of CSA code.
This CL teases the code for each of these builtins apart into two chunks, a main
body with optimizations for fast cases, and a "continuation" builtin that
performs a spec-compliant, but slower version of the main loop of the
builtin. The general idea is that when the "fast" main body builtin encounters
an unexpected condition that invalidates assumptions allowing fast-case code, it
tail calls to the slow, correct version of the loop that finishes the builtin
execution.
This separation currently doens't really provide any specific advantage over the
combined version. However, it paves the way to TF-optimized inlined Array
builtins. Inlined Array builtins may trigger deopts during the execution of the
builtin's loop, and those deopt must continue execution from the point at which
they failed. With some massaging of the deoptimizer, it will be possible to make
those deopt points create an extra frame on the top of the stack which resumes
execution in the slow-loop builtin created in this CL.
BUG=v8:1956
LOG=N
Review-Url: https://codereview.chromium.org/2753793002
Cr-Commit-Position: refs/heads/master@{#43867}
2017-03-16 15:34:01 +00:00
|
|
|
// Instantiate TailCallBytecodeDispatch() for argument counts used by
|
|
|
|
// CSA-generated code
|
2016-12-16 12:58:19 +00:00
|
|
|
template V8_EXPORT_PRIVATE Node* CodeAssembler::TailCallBytecodeDispatch(
|
|
|
|
const CallInterfaceDescriptor& descriptor, Node* target, Node*, Node*,
|
2018-02-11 19:17:27 +00:00
|
|
|
Node*, Node*);
|
2016-12-16 12:58:19 +00:00
|
|
|
|
2018-06-07 15:21:27 +00:00
|
|
|
TNode<Object> CodeAssembler::TailCallJSCode(TNode<Code> code,
|
|
|
|
TNode<Context> context,
|
|
|
|
TNode<JSFunction> function,
|
|
|
|
TNode<Object> new_target,
|
|
|
|
TNode<Int32T> arg_count) {
|
2018-06-18 15:14:29 +00:00
|
|
|
JSTrampolineDescriptor descriptor;
|
2018-06-07 15:21:27 +00:00
|
|
|
auto call_descriptor = Linkage::GetStubCallDescriptor(
|
2018-06-18 15:14:29 +00:00
|
|
|
zone(), descriptor, descriptor.GetStackParameterCount(),
|
2018-06-21 08:47:48 +00:00
|
|
|
CallDescriptor::kFixedTargetRegister, Operator::kNoProperties);
|
2018-06-07 15:21:27 +00:00
|
|
|
|
|
|
|
Node* nodes[] = {code, function, new_target, arg_count, context};
|
|
|
|
CHECK_EQ(descriptor.GetParameterCount() + 2, arraysize(nodes));
|
|
|
|
return UncheckedCast<Object>(
|
|
|
|
raw_assembler()->TailCallN(call_descriptor, arraysize(nodes), nodes));
|
|
|
|
}
|
|
|
|
|
2017-01-18 19:13:49 +00:00
|
|
|
Node* CodeAssembler::CallCFunctionN(Signature<MachineType>* signature,
|
|
|
|
int input_count, Node* const* inputs) {
|
2018-02-09 19:19:25 +00:00
|
|
|
auto call_descriptor = Linkage::GetSimplifiedCDescriptor(zone(), signature);
|
|
|
|
return raw_assembler()->CallN(call_descriptor, input_count, inputs);
|
2017-01-18 19:13:49 +00:00
|
|
|
}
|
|
|
|
|
2017-04-13 14:41:22 +00:00
|
|
|
Node* CodeAssembler::CallCFunction1(MachineType return_type,
|
|
|
|
MachineType arg0_type, Node* function,
|
|
|
|
Node* arg0) {
|
|
|
|
return raw_assembler()->CallCFunction1(return_type, arg0_type, function,
|
|
|
|
arg0);
|
|
|
|
}
|
|
|
|
|
2017-08-21 15:23:17 +00:00
|
|
|
Node* CodeAssembler::CallCFunction1WithCallerSavedRegisters(
|
2017-09-22 09:13:07 +00:00
|
|
|
MachineType return_type, MachineType arg0_type, Node* function, Node* arg0,
|
|
|
|
SaveFPRegsMode mode) {
|
2017-08-21 15:23:17 +00:00
|
|
|
DCHECK(return_type.LessThanOrEqualPointerSize());
|
|
|
|
return raw_assembler()->CallCFunction1WithCallerSavedRegisters(
|
2017-09-22 09:13:07 +00:00
|
|
|
return_type, arg0_type, function, arg0, mode);
|
2017-08-21 15:23:17 +00:00
|
|
|
}
|
|
|
|
|
2016-09-15 08:19:59 +00:00
|
|
|
Node* CodeAssembler::CallCFunction2(MachineType return_type,
|
|
|
|
MachineType arg0_type,
|
|
|
|
MachineType arg1_type, Node* function,
|
|
|
|
Node* arg0, Node* arg1) {
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->CallCFunction2(return_type, arg0_type, arg1_type,
|
|
|
|
function, arg0, arg1);
|
2016-09-15 08:19:59 +00:00
|
|
|
}
|
|
|
|
|
2016-12-16 13:24:07 +00:00
|
|
|
Node* CodeAssembler::CallCFunction3(MachineType return_type,
|
|
|
|
MachineType arg0_type,
|
|
|
|
MachineType arg1_type,
|
|
|
|
MachineType arg2_type, Node* function,
|
|
|
|
Node* arg0, Node* arg1, Node* arg2) {
|
|
|
|
return raw_assembler()->CallCFunction3(return_type, arg0_type, arg1_type,
|
|
|
|
arg2_type, function, arg0, arg1, arg2);
|
|
|
|
}
|
|
|
|
|
2017-08-21 15:23:17 +00:00
|
|
|
Node* CodeAssembler::CallCFunction3WithCallerSavedRegisters(
|
|
|
|
MachineType return_type, MachineType arg0_type, MachineType arg1_type,
|
2017-09-22 09:13:07 +00:00
|
|
|
MachineType arg2_type, Node* function, Node* arg0, Node* arg1, Node* arg2,
|
|
|
|
SaveFPRegsMode mode) {
|
2017-08-21 15:23:17 +00:00
|
|
|
DCHECK(return_type.LessThanOrEqualPointerSize());
|
|
|
|
return raw_assembler()->CallCFunction3WithCallerSavedRegisters(
|
2017-09-22 09:13:07 +00:00
|
|
|
return_type, arg0_type, arg1_type, arg2_type, function, arg0, arg1, arg2,
|
|
|
|
mode);
|
2017-08-21 15:23:17 +00:00
|
|
|
}
|
|
|
|
|
2017-11-27 16:58:41 +00:00
|
|
|
Node* CodeAssembler::CallCFunction4(
|
|
|
|
MachineType return_type, MachineType arg0_type, MachineType arg1_type,
|
|
|
|
MachineType arg2_type, MachineType arg3_type, Node* function, Node* arg0,
|
|
|
|
Node* arg1, Node* arg2, Node* arg3) {
|
|
|
|
return raw_assembler()->CallCFunction4(return_type, arg0_type, arg1_type,
|
|
|
|
arg2_type, arg3_type, function, arg0,
|
|
|
|
arg1, arg2, arg3);
|
|
|
|
}
|
|
|
|
|
2017-11-27 12:34:23 +00:00
|
|
|
Node* CodeAssembler::CallCFunction5(
|
|
|
|
MachineType return_type, MachineType arg0_type, MachineType arg1_type,
|
|
|
|
MachineType arg2_type, MachineType arg3_type, MachineType arg4_type,
|
|
|
|
Node* function, Node* arg0, Node* arg1, Node* arg2, Node* arg3,
|
|
|
|
Node* arg4) {
|
|
|
|
return raw_assembler()->CallCFunction5(
|
|
|
|
return_type, arg0_type, arg1_type, arg2_type, arg3_type, arg4_type,
|
|
|
|
function, arg0, arg1, arg2, arg3, arg4);
|
|
|
|
}
|
|
|
|
|
2017-04-19 10:47:03 +00:00
|
|
|
Node* CodeAssembler::CallCFunction6(
|
|
|
|
MachineType return_type, MachineType arg0_type, MachineType arg1_type,
|
|
|
|
MachineType arg2_type, MachineType arg3_type, MachineType arg4_type,
|
|
|
|
MachineType arg5_type, Node* function, Node* arg0, Node* arg1, Node* arg2,
|
|
|
|
Node* arg3, Node* arg4, Node* arg5) {
|
|
|
|
return raw_assembler()->CallCFunction6(
|
|
|
|
return_type, arg0_type, arg1_type, arg2_type, arg3_type, arg4_type,
|
|
|
|
arg5_type, function, arg0, arg1, arg2, arg3, arg4, arg5);
|
|
|
|
}
|
|
|
|
|
2017-04-21 14:03:34 +00:00
|
|
|
Node* CodeAssembler::CallCFunction9(
|
|
|
|
MachineType return_type, MachineType arg0_type, MachineType arg1_type,
|
|
|
|
MachineType arg2_type, MachineType arg3_type, MachineType arg4_type,
|
|
|
|
MachineType arg5_type, MachineType arg6_type, MachineType arg7_type,
|
|
|
|
MachineType arg8_type, Node* function, Node* arg0, Node* arg1, Node* arg2,
|
|
|
|
Node* arg3, Node* arg4, Node* arg5, Node* arg6, Node* arg7, Node* arg8) {
|
|
|
|
return raw_assembler()->CallCFunction9(
|
|
|
|
return_type, arg0_type, arg1_type, arg2_type, arg3_type, arg4_type,
|
|
|
|
arg5_type, arg6_type, arg7_type, arg8_type, function, arg0, arg1, arg2,
|
|
|
|
arg3, arg4, arg5, arg6, arg7, arg8);
|
|
|
|
}
|
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
void CodeAssembler::Goto(Label* label) {
|
2016-04-18 11:57:06 +00:00
|
|
|
label->MergeVariables();
|
2016-11-16 11:48:07 +00:00
|
|
|
raw_assembler()->Goto(label->label_);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
void CodeAssembler::GotoIf(SloppyTNode<IntegralT> condition,
|
|
|
|
Label* true_label) {
|
2016-04-18 11:57:06 +00:00
|
|
|
Label false_label(this);
|
|
|
|
Branch(condition, true_label, &false_label);
|
|
|
|
Bind(&false_label);
|
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
void CodeAssembler::GotoIfNot(SloppyTNode<IntegralT> condition,
|
|
|
|
Label* false_label) {
|
2016-04-18 11:57:06 +00:00
|
|
|
Label true_label(this);
|
|
|
|
Branch(condition, &true_label, false_label);
|
|
|
|
Bind(&true_label);
|
|
|
|
}
|
|
|
|
|
2017-08-03 14:27:11 +00:00
|
|
|
void CodeAssembler::Branch(SloppyTNode<IntegralT> condition, Label* true_label,
|
2016-12-06 10:29:13 +00:00
|
|
|
Label* false_label) {
|
2018-08-23 08:41:07 +00:00
|
|
|
int32_t constant;
|
|
|
|
if (ToInt32Constant(condition, constant)) {
|
|
|
|
if ((true_label->is_used() || true_label->is_bound()) &&
|
|
|
|
(false_label->is_used() || false_label->is_bound())) {
|
|
|
|
return Goto(constant ? true_label : false_label);
|
|
|
|
}
|
|
|
|
}
|
2016-04-18 11:57:06 +00:00
|
|
|
true_label->MergeVariables();
|
|
|
|
false_label->MergeVariables();
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->Branch(condition, true_label->label_,
|
|
|
|
false_label->label_);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 05:55:37 +00:00
|
|
|
void CodeAssembler::Branch(TNode<BoolT> condition,
|
2018-09-18 09:54:14 +00:00
|
|
|
const std::function<void()>& true_body,
|
|
|
|
const std::function<void()>& false_body) {
|
2018-08-09 05:55:37 +00:00
|
|
|
int32_t constant;
|
|
|
|
if (ToInt32Constant(condition, constant)) {
|
|
|
|
return constant ? true_body() : false_body();
|
|
|
|
}
|
|
|
|
|
2018-08-14 19:51:44 +00:00
|
|
|
Label vtrue(this), vfalse(this);
|
2018-08-09 05:55:37 +00:00
|
|
|
Branch(condition, &vtrue, &vfalse);
|
2018-08-23 08:41:07 +00:00
|
|
|
|
2018-08-09 05:55:37 +00:00
|
|
|
Bind(&vtrue);
|
2018-08-23 08:41:07 +00:00
|
|
|
true_body();
|
|
|
|
|
|
|
|
Bind(&vfalse);
|
|
|
|
false_body();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeAssembler::Branch(TNode<BoolT> condition, Label* true_label,
|
2018-09-18 09:54:14 +00:00
|
|
|
const std::function<void()>& false_body) {
|
2018-08-23 08:41:07 +00:00
|
|
|
int32_t constant;
|
|
|
|
if (ToInt32Constant(condition, constant)) {
|
|
|
|
return constant ? Goto(true_label) : false_body();
|
2018-08-09 05:55:37 +00:00
|
|
|
}
|
2018-08-23 08:41:07 +00:00
|
|
|
|
|
|
|
Label vfalse(this);
|
|
|
|
Branch(condition, true_label, &vfalse);
|
2018-08-09 05:55:37 +00:00
|
|
|
Bind(&vfalse);
|
2018-08-23 08:41:07 +00:00
|
|
|
false_body();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeAssembler::Branch(TNode<BoolT> condition,
|
2018-09-18 09:54:14 +00:00
|
|
|
const std::function<void()>& true_body,
|
2018-08-23 08:41:07 +00:00
|
|
|
Label* false_label) {
|
|
|
|
int32_t constant;
|
|
|
|
if (ToInt32Constant(condition, constant)) {
|
|
|
|
return constant ? true_body() : Goto(false_label);
|
2018-08-09 05:55:37 +00:00
|
|
|
}
|
2018-08-23 08:41:07 +00:00
|
|
|
|
|
|
|
Label vtrue(this);
|
|
|
|
Branch(condition, &vtrue, false_label);
|
|
|
|
Bind(&vtrue);
|
|
|
|
true_body();
|
2018-08-09 05:55:37 +00:00
|
|
|
}
|
|
|
|
|
2016-04-18 11:57:06 +00:00
|
|
|
void CodeAssembler::Switch(Node* index, Label* default_label,
|
2016-08-03 03:26:58 +00:00
|
|
|
const int32_t* case_values, Label** case_labels,
|
2016-04-18 11:57:06 +00:00
|
|
|
size_t case_count) {
|
|
|
|
RawMachineLabel** labels =
|
|
|
|
new (zone()->New(sizeof(RawMachineLabel*) * case_count))
|
|
|
|
RawMachineLabel*[case_count];
|
|
|
|
for (size_t i = 0; i < case_count; ++i) {
|
|
|
|
labels[i] = case_labels[i]->label_;
|
|
|
|
case_labels[i]->MergeVariables();
|
|
|
|
}
|
2017-08-30 20:28:04 +00:00
|
|
|
default_label->MergeVariables();
|
2016-11-16 11:48:07 +00:00
|
|
|
return raw_assembler()->Switch(index, default_label->label_, case_values,
|
|
|
|
labels, case_count);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-08-03 08:11:31 +00:00
|
|
|
bool CodeAssembler::UnalignedLoadSupported(MachineRepresentation rep) const {
|
|
|
|
return raw_assembler()->machine()->UnalignedLoadSupported(rep);
|
2017-08-02 19:11:56 +00:00
|
|
|
}
|
2017-08-03 08:11:31 +00:00
|
|
|
bool CodeAssembler::UnalignedStoreSupported(MachineRepresentation rep) const {
|
|
|
|
return raw_assembler()->machine()->UnalignedStoreSupported(rep);
|
2017-04-08 22:15:38 +00:00
|
|
|
}
|
|
|
|
|
2016-04-18 11:57:06 +00:00
|
|
|
// RawMachineAssembler delegate helpers:
|
2016-11-16 11:48:07 +00:00
|
|
|
Isolate* CodeAssembler::isolate() const { return raw_assembler()->isolate(); }
|
2016-04-18 11:57:06 +00:00
|
|
|
|
|
|
|
Factory* CodeAssembler::factory() const { return isolate()->factory(); }
|
|
|
|
|
2016-11-16 11:48:07 +00:00
|
|
|
Zone* CodeAssembler::zone() const { return raw_assembler()->zone(); }
|
|
|
|
|
2018-10-31 13:00:51 +00:00
|
|
|
bool CodeAssembler::IsExceptionHandlerActive() const {
|
|
|
|
return state_->exception_handler_labels_.size() != 0;
|
|
|
|
}
|
|
|
|
|
2016-11-16 11:48:07 +00:00
|
|
|
RawMachineAssembler* CodeAssembler::raw_assembler() const {
|
|
|
|
return state_->raw_assembler_.get();
|
|
|
|
}
|
2016-04-18 11:57:06 +00:00
|
|
|
|
|
|
|
// The core implementation of Variable is stored through an indirection so
|
|
|
|
// that it can outlive the often block-scoped Variable declarations. This is
|
|
|
|
// needed to ensure that variable binding and merging through phis can
|
|
|
|
// properly be verified.
|
2016-12-06 10:29:13 +00:00
|
|
|
class CodeAssemblerVariable::Impl : public ZoneObject {
|
2016-04-18 11:57:06 +00:00
|
|
|
public:
|
2018-11-06 12:23:53 +00:00
|
|
|
explicit Impl(MachineRepresentation rep, CodeAssemblerState::VariableId id)
|
2017-04-05 17:40:36 +00:00
|
|
|
:
|
|
|
|
#if DEBUG
|
|
|
|
debug_info_(AssemblerDebugInfo(nullptr, nullptr, -1)),
|
|
|
|
#endif
|
|
|
|
value_(nullptr),
|
2018-11-06 12:23:53 +00:00
|
|
|
rep_(rep),
|
|
|
|
var_id_(id) {
|
2017-04-05 17:40:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#if DEBUG
|
|
|
|
AssemblerDebugInfo debug_info() const { return debug_info_; }
|
|
|
|
void set_debug_info(AssemblerDebugInfo debug_info) {
|
|
|
|
debug_info_ = debug_info;
|
|
|
|
}
|
|
|
|
|
|
|
|
AssemblerDebugInfo debug_info_;
|
|
|
|
#endif // DEBUG
|
2018-11-06 12:23:53 +00:00
|
|
|
bool operator<(const CodeAssemblerVariable::Impl& other) const {
|
|
|
|
return var_id_ < other.var_id_;
|
|
|
|
}
|
2016-04-18 11:57:06 +00:00
|
|
|
Node* value_;
|
|
|
|
MachineRepresentation rep_;
|
2018-11-06 12:23:53 +00:00
|
|
|
CodeAssemblerState::VariableId var_id_;
|
2016-04-18 11:57:06 +00:00
|
|
|
};
|
|
|
|
|
2018-11-06 12:23:53 +00:00
|
|
|
bool CodeAssemblerVariable::ImplComparator::operator()(
|
|
|
|
const CodeAssemblerVariable::Impl* a,
|
|
|
|
const CodeAssemblerVariable::Impl* b) const {
|
|
|
|
return *a < *b;
|
|
|
|
}
|
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
CodeAssemblerVariable::CodeAssemblerVariable(CodeAssembler* assembler,
|
|
|
|
MachineRepresentation rep)
|
2018-11-06 12:23:53 +00:00
|
|
|
: impl_(new (assembler->zone())
|
|
|
|
Impl(rep, assembler->state()->NextVariableId())),
|
|
|
|
state_(assembler->state()) {
|
2016-11-16 11:48:07 +00:00
|
|
|
state_->variables_.insert(impl_);
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
|
2017-01-25 10:00:38 +00:00
|
|
|
CodeAssemblerVariable::CodeAssemblerVariable(CodeAssembler* assembler,
|
|
|
|
MachineRepresentation rep,
|
|
|
|
Node* initial_value)
|
|
|
|
: CodeAssemblerVariable(assembler, rep) {
|
|
|
|
Bind(initial_value);
|
|
|
|
}
|
|
|
|
|
2017-04-05 17:40:36 +00:00
|
|
|
#if DEBUG
|
|
|
|
CodeAssemblerVariable::CodeAssemblerVariable(CodeAssembler* assembler,
|
|
|
|
AssemblerDebugInfo debug_info,
|
|
|
|
MachineRepresentation rep)
|
2018-11-06 12:23:53 +00:00
|
|
|
: impl_(new (assembler->zone())
|
|
|
|
Impl(rep, assembler->state()->NextVariableId())),
|
|
|
|
state_(assembler->state()) {
|
2017-04-05 17:40:36 +00:00
|
|
|
impl_->set_debug_info(debug_info);
|
|
|
|
state_->variables_.insert(impl_);
|
|
|
|
}
|
|
|
|
|
|
|
|
CodeAssemblerVariable::CodeAssemblerVariable(CodeAssembler* assembler,
|
|
|
|
AssemblerDebugInfo debug_info,
|
|
|
|
MachineRepresentation rep,
|
|
|
|
Node* initial_value)
|
|
|
|
: CodeAssemblerVariable(assembler, debug_info, rep) {
|
|
|
|
impl_->set_debug_info(debug_info);
|
|
|
|
Bind(initial_value);
|
|
|
|
}
|
|
|
|
#endif // DEBUG
|
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
CodeAssemblerVariable::~CodeAssemblerVariable() {
|
|
|
|
state_->variables_.erase(impl_);
|
|
|
|
}
|
2016-06-02 21:24:24 +00:00
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
void CodeAssemblerVariable::Bind(Node* value) { impl_->value_ = value; }
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
Node* CodeAssemblerVariable::value() const {
|
2017-04-05 17:40:36 +00:00
|
|
|
#if DEBUG
|
|
|
|
if (!IsBound()) {
|
|
|
|
std::stringstream str;
|
|
|
|
str << "#Use of unbound variable:"
|
2017-11-15 19:35:10 +00:00
|
|
|
<< "#\n Variable: " << *this << "#\n Current Block: ";
|
|
|
|
state_->PrintCurrentBlock(str);
|
2017-12-18 16:19:23 +00:00
|
|
|
FATAL("%s", str.str().c_str());
|
2017-11-15 19:35:10 +00:00
|
|
|
}
|
|
|
|
if (!state_->InsideBlock()) {
|
|
|
|
std::stringstream str;
|
|
|
|
str << "#Accessing variable value outside a block:"
|
2017-04-05 17:40:36 +00:00
|
|
|
<< "#\n Variable: " << *this;
|
2017-12-18 16:19:23 +00:00
|
|
|
FATAL("%s", str.str().c_str());
|
2017-04-05 17:40:36 +00:00
|
|
|
}
|
|
|
|
#endif // DEBUG
|
2016-04-18 11:57:06 +00:00
|
|
|
return impl_->value_;
|
|
|
|
}
|
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
MachineRepresentation CodeAssemblerVariable::rep() const { return impl_->rep_; }
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
bool CodeAssemblerVariable::IsBound() const { return impl_->value_ != nullptr; }
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2017-04-05 17:40:36 +00:00
|
|
|
std::ostream& operator<<(std::ostream& os,
|
|
|
|
const CodeAssemblerVariable::Impl& impl) {
|
|
|
|
#if DEBUG
|
|
|
|
AssemblerDebugInfo info = impl.debug_info();
|
|
|
|
if (info.name) os << "V" << info;
|
|
|
|
#endif // DEBUG
|
|
|
|
return os;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream& operator<<(std::ostream& os,
|
|
|
|
const CodeAssemblerVariable& variable) {
|
|
|
|
os << *variable.impl_;
|
|
|
|
return os;
|
|
|
|
}
|
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
CodeAssemblerLabel::CodeAssemblerLabel(CodeAssembler* assembler,
|
|
|
|
size_t vars_count,
|
2017-03-21 15:57:38 +00:00
|
|
|
CodeAssemblerVariable* const* vars,
|
2016-12-06 10:29:13 +00:00
|
|
|
CodeAssemblerLabel::Type type)
|
2016-11-16 11:48:07 +00:00
|
|
|
: bound_(false),
|
|
|
|
merge_count_(0),
|
2016-12-06 10:29:13 +00:00
|
|
|
state_(assembler->state()),
|
2016-11-16 11:48:07 +00:00
|
|
|
label_(nullptr) {
|
2016-04-18 11:57:06 +00:00
|
|
|
void* buffer = assembler->zone()->New(sizeof(RawMachineLabel));
|
|
|
|
label_ = new (buffer)
|
|
|
|
RawMachineLabel(type == kDeferred ? RawMachineLabel::kDeferred
|
|
|
|
: RawMachineLabel::kNonDeferred);
|
2016-11-02 13:18:50 +00:00
|
|
|
for (size_t i = 0; i < vars_count; ++i) {
|
|
|
|
variable_phis_[vars[i]->impl_] = nullptr;
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-18 14:39:01 +00:00
|
|
|
CodeAssemblerLabel::~CodeAssemblerLabel() { label_->~RawMachineLabel(); }
|
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
void CodeAssemblerLabel::MergeVariables() {
|
2016-04-18 11:57:06 +00:00
|
|
|
++merge_count_;
|
2017-04-05 17:40:36 +00:00
|
|
|
for (CodeAssemblerVariable::Impl* var : state_->variables_) {
|
2016-04-18 11:57:06 +00:00
|
|
|
size_t count = 0;
|
|
|
|
Node* node = var->value_;
|
|
|
|
if (node != nullptr) {
|
|
|
|
auto i = variable_merges_.find(var);
|
|
|
|
if (i != variable_merges_.end()) {
|
|
|
|
i->second.push_back(node);
|
|
|
|
count = i->second.size();
|
|
|
|
} else {
|
|
|
|
count = 1;
|
|
|
|
variable_merges_[var] = std::vector<Node*>(1, node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If the following asserts, then you've jumped to a label without a bound
|
|
|
|
// variable along that path that expects to merge its value into a phi.
|
|
|
|
DCHECK(variable_phis_.find(var) == variable_phis_.end() ||
|
|
|
|
count == merge_count_);
|
|
|
|
USE(count);
|
|
|
|
|
|
|
|
// If the label is already bound, we already know the set of variables to
|
|
|
|
// merge and phi nodes have already been created.
|
|
|
|
if (bound_) {
|
|
|
|
auto phi = variable_phis_.find(var);
|
|
|
|
if (phi != variable_phis_.end()) {
|
|
|
|
DCHECK_NOT_NULL(phi->second);
|
2016-11-16 11:48:07 +00:00
|
|
|
state_->raw_assembler_->AppendPhiInput(phi->second, node);
|
2016-04-18 11:57:06 +00:00
|
|
|
} else {
|
|
|
|
auto i = variable_merges_.find(var);
|
|
|
|
if (i != variable_merges_.end()) {
|
|
|
|
// If the following assert fires, then you've declared a variable that
|
|
|
|
// has the same bound value along all paths up until the point you
|
|
|
|
// bound this label, but then later merged a path with a new value for
|
|
|
|
// the variable after the label bind (it's not possible to add phis to
|
|
|
|
// the bound label after the fact, just make sure to list the variable
|
|
|
|
// in the label's constructor's list of merged variables).
|
2017-04-05 17:40:36 +00:00
|
|
|
#if DEBUG
|
|
|
|
if (find_if(i->second.begin(), i->second.end(),
|
|
|
|
[node](Node* e) -> bool { return node != e; }) !=
|
|
|
|
i->second.end()) {
|
|
|
|
std::stringstream str;
|
|
|
|
str << "Unmerged variable found when jumping to block. \n"
|
|
|
|
<< "# Variable: " << *var;
|
|
|
|
if (bound_) {
|
|
|
|
str << "\n# Target block: " << *label_->block();
|
|
|
|
}
|
|
|
|
str << "\n# Current Block: ";
|
|
|
|
state_->PrintCurrentBlock(str);
|
2017-12-18 16:19:23 +00:00
|
|
|
FATAL("%s", str.str().c_str());
|
2017-04-05 17:40:36 +00:00
|
|
|
}
|
|
|
|
#endif // DEBUG
|
2016-04-18 11:57:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-05 17:40:36 +00:00
|
|
|
#if DEBUG
|
|
|
|
void CodeAssemblerLabel::Bind(AssemblerDebugInfo debug_info) {
|
2017-05-15 08:39:41 +00:00
|
|
|
if (bound_) {
|
|
|
|
std::stringstream str;
|
|
|
|
str << "Cannot bind the same label twice:"
|
|
|
|
<< "\n# current: " << debug_info
|
|
|
|
<< "\n# previous: " << *label_->block();
|
2017-12-18 16:19:23 +00:00
|
|
|
FATAL("%s", str.str().c_str());
|
2017-05-15 08:39:41 +00:00
|
|
|
}
|
2017-04-05 17:40:36 +00:00
|
|
|
state_->raw_assembler_->Bind(label_, debug_info);
|
|
|
|
UpdateVariablesAfterBind();
|
|
|
|
}
|
|
|
|
#endif // DEBUG
|
|
|
|
|
2016-12-06 10:29:13 +00:00
|
|
|
void CodeAssemblerLabel::Bind() {
|
2016-04-18 11:57:06 +00:00
|
|
|
DCHECK(!bound_);
|
2016-11-16 11:48:07 +00:00
|
|
|
state_->raw_assembler_->Bind(label_);
|
2017-04-05 17:40:36 +00:00
|
|
|
UpdateVariablesAfterBind();
|
|
|
|
}
|
2016-04-18 11:57:06 +00:00
|
|
|
|
2017-04-05 17:40:36 +00:00
|
|
|
void CodeAssemblerLabel::UpdateVariablesAfterBind() {
|
2016-04-18 11:57:06 +00:00
|
|
|
// Make sure that all variables that have changed along any path up to this
|
|
|
|
// point are marked as merge variables.
|
2016-11-16 11:48:07 +00:00
|
|
|
for (auto var : state_->variables_) {
|
2016-04-18 11:57:06 +00:00
|
|
|
Node* shared_value = nullptr;
|
|
|
|
auto i = variable_merges_.find(var);
|
|
|
|
if (i != variable_merges_.end()) {
|
|
|
|
for (auto value : i->second) {
|
2017-09-25 09:45:55 +00:00
|
|
|
DCHECK_NOT_NULL(value);
|
2016-04-18 11:57:06 +00:00
|
|
|
if (value != shared_value) {
|
|
|
|
if (shared_value == nullptr) {
|
|
|
|
shared_value = value;
|
|
|
|
} else {
|
|
|
|
variable_phis_[var] = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto var : variable_phis_) {
|
2016-12-06 10:29:13 +00:00
|
|
|
CodeAssemblerVariable::Impl* var_impl = var.first;
|
2016-04-18 11:57:06 +00:00
|
|
|
auto i = variable_merges_.find(var_impl);
|
2017-05-24 09:04:59 +00:00
|
|
|
#if DEBUG
|
|
|
|
bool not_found = i == variable_merges_.end();
|
|
|
|
if (not_found || i->second.size() != merge_count_) {
|
|
|
|
std::stringstream str;
|
|
|
|
str << "A variable that has been marked as beeing merged at the label"
|
|
|
|
<< "\n# doesn't have a bound value along all of the paths that "
|
|
|
|
<< "\n# have been merged into the label up to this point."
|
|
|
|
<< "\n#"
|
|
|
|
<< "\n# This can happen in the following cases:"
|
|
|
|
<< "\n# - By explicitly marking it so in the label constructor"
|
|
|
|
<< "\n# - By having seen different bound values at branches"
|
|
|
|
<< "\n#"
|
|
|
|
<< "\n# Merge count: expected=" << merge_count_
|
|
|
|
<< " vs. found=" << (not_found ? 0 : i->second.size())
|
|
|
|
<< "\n# Variable: " << *var_impl
|
|
|
|
<< "\n# Current Block: " << *label_->block();
|
2017-12-18 16:19:23 +00:00
|
|
|
FATAL("%s", str.str().c_str());
|
2017-05-24 09:04:59 +00:00
|
|
|
}
|
|
|
|
#endif // DEBUG
|
2016-11-16 11:48:07 +00:00
|
|
|
Node* phi = state_->raw_assembler_->Phi(
|
2016-04-18 11:57:06 +00:00
|
|
|
var.first->rep_, static_cast<int>(merge_count_), &(i->second[0]));
|
|
|
|
variable_phis_[var_impl] = phi;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Bind all variables to a merge phi, the common value along all paths or
|
|
|
|
// null.
|
2016-11-16 11:48:07 +00:00
|
|
|
for (auto var : state_->variables_) {
|
2016-04-18 11:57:06 +00:00
|
|
|
auto i = variable_phis_.find(var);
|
|
|
|
if (i != variable_phis_.end()) {
|
|
|
|
var->value_ = i->second;
|
|
|
|
} else {
|
|
|
|
auto j = variable_merges_.find(var);
|
|
|
|
if (j != variable_merges_.end() && j->second.size() == merge_count_) {
|
|
|
|
var->value_ = j->second.back();
|
|
|
|
} else {
|
|
|
|
var->value_ = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bound_ = true;
|
|
|
|
}
|
|
|
|
|
2018-10-04 20:45:31 +00:00
|
|
|
void CodeAssemblerParameterizedLabelBase::AddInputs(std::vector<Node*> inputs) {
|
|
|
|
if (!phi_nodes_.empty()) {
|
|
|
|
DCHECK_EQ(inputs.size(), phi_nodes_.size());
|
|
|
|
for (size_t i = 0; i < inputs.size(); ++i) {
|
|
|
|
state_->raw_assembler_->AppendPhiInput(phi_nodes_[i], inputs[i]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
DCHECK_EQ(inputs.size(), phi_inputs_.size());
|
|
|
|
for (size_t i = 0; i < inputs.size(); ++i) {
|
|
|
|
phi_inputs_[i].push_back(inputs[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* CodeAssemblerParameterizedLabelBase::CreatePhi(
|
|
|
|
MachineRepresentation rep, const std::vector<Node*>& inputs) {
|
|
|
|
for (Node* input : inputs) {
|
|
|
|
// We use {nullptr} as a sentinel for an uninitialized value. We must not
|
|
|
|
// create phi nodes for these.
|
|
|
|
if (input == nullptr) return nullptr;
|
|
|
|
}
|
|
|
|
return state_->raw_assembler_->Phi(rep, static_cast<int>(inputs.size()),
|
|
|
|
&inputs.front());
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::vector<Node*>& CodeAssemblerParameterizedLabelBase::CreatePhis(
|
|
|
|
std::vector<MachineRepresentation> representations) {
|
|
|
|
DCHECK(is_used());
|
|
|
|
DCHECK(phi_nodes_.empty());
|
|
|
|
phi_nodes_.reserve(phi_inputs_.size());
|
|
|
|
DCHECK_EQ(representations.size(), phi_inputs_.size());
|
|
|
|
for (size_t i = 0; i < phi_inputs_.size(); ++i) {
|
|
|
|
phi_nodes_.push_back(CreatePhi(representations[i], phi_inputs_[i]));
|
|
|
|
}
|
|
|
|
return phi_nodes_;
|
|
|
|
}
|
|
|
|
|
2018-10-31 13:00:51 +00:00
|
|
|
void CodeAssemblerState::PushExceptionHandler(
|
|
|
|
CodeAssemblerExceptionHandlerLabel* label) {
|
|
|
|
exception_handler_labels_.push_back(label);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CodeAssemblerState::PopExceptionHandler() {
|
|
|
|
exception_handler_labels_.pop_back();
|
|
|
|
}
|
|
|
|
|
2018-11-13 14:01:35 +00:00
|
|
|
CodeAssemblerScopedExceptionHandler::CodeAssemblerScopedExceptionHandler(
|
|
|
|
CodeAssembler* assembler, CodeAssemblerExceptionHandlerLabel* label)
|
|
|
|
: has_handler_(label != nullptr),
|
|
|
|
assembler_(assembler),
|
|
|
|
compatibility_label_(nullptr),
|
|
|
|
exception_(nullptr) {
|
|
|
|
if (has_handler_) {
|
|
|
|
assembler_->state()->PushExceptionHandler(label);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CodeAssemblerScopedExceptionHandler::CodeAssemblerScopedExceptionHandler(
|
|
|
|
CodeAssembler* assembler, CodeAssemblerLabel* label,
|
|
|
|
TypedCodeAssemblerVariable<Object>* exception)
|
|
|
|
: has_handler_(label != nullptr),
|
|
|
|
assembler_(assembler),
|
|
|
|
compatibility_label_(label),
|
|
|
|
exception_(exception) {
|
|
|
|
if (has_handler_) {
|
|
|
|
label_ = base::make_unique<CodeAssemblerExceptionHandlerLabel>(
|
|
|
|
assembler, CodeAssemblerLabel::kDeferred);
|
|
|
|
assembler_->state()->PushExceptionHandler(label_.get());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CodeAssemblerScopedExceptionHandler::~CodeAssemblerScopedExceptionHandler() {
|
|
|
|
if (has_handler_) {
|
|
|
|
assembler_->state()->PopExceptionHandler();
|
|
|
|
}
|
|
|
|
if (label_ && label_->is_used()) {
|
|
|
|
CodeAssembler::Label skip(assembler_);
|
|
|
|
bool inside_block = assembler_->state()->InsideBlock();
|
|
|
|
if (inside_block) {
|
|
|
|
assembler_->Goto(&skip);
|
|
|
|
}
|
|
|
|
TNode<Object> e;
|
|
|
|
assembler_->Bind(label_.get(), &e);
|
|
|
|
*exception_ = e;
|
|
|
|
assembler_->Goto(compatibility_label_);
|
|
|
|
if (inside_block) {
|
|
|
|
assembler_->Bind(&skip);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-18 11:57:06 +00:00
|
|
|
} // namespace compiler
|
2017-08-03 14:27:11 +00:00
|
|
|
|
2018-11-27 00:48:42 +00:00
|
|
|
Address CheckObjectType(Object* value, Address raw_type, Address raw_location) {
|
2017-08-03 14:27:11 +00:00
|
|
|
#ifdef DEBUG
|
2018-11-03 00:13:22 +00:00
|
|
|
Smi type(raw_type);
|
2018-11-27 00:48:42 +00:00
|
|
|
String location = String::cast(ObjectPtr(raw_location));
|
2017-08-03 14:27:11 +00:00
|
|
|
const char* expected;
|
|
|
|
switch (static_cast<ObjectType>(type->value())) {
|
2018-11-03 00:13:22 +00:00
|
|
|
#define TYPE_CASE(Name) \
|
|
|
|
case ObjectType::k##Name: \
|
|
|
|
if (value->Is##Name()) return Smi::FromInt(0).ptr(); \
|
|
|
|
expected = #Name; \
|
2017-08-03 14:27:11 +00:00
|
|
|
break;
|
2018-11-03 00:13:22 +00:00
|
|
|
#define TYPE_STRUCT_CASE(NAME, Name, name) \
|
|
|
|
case ObjectType::k##Name: \
|
|
|
|
if (value->Is##Name()) return Smi::FromInt(0).ptr(); \
|
|
|
|
expected = #Name; \
|
2017-08-03 14:27:11 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
TYPE_CASE(Object)
|
|
|
|
OBJECT_TYPE_LIST(TYPE_CASE)
|
|
|
|
HEAP_OBJECT_TYPE_LIST(TYPE_CASE)
|
|
|
|
STRUCT_LIST(TYPE_STRUCT_CASE)
|
|
|
|
#undef TYPE_CASE
|
|
|
|
#undef TYPE_STRUCT_CASE
|
|
|
|
}
|
|
|
|
std::stringstream value_description;
|
2018-07-12 10:06:42 +00:00
|
|
|
value->Print(value_description);
|
2017-08-03 14:27:11 +00:00
|
|
|
V8_Fatal(__FILE__, __LINE__,
|
|
|
|
"Type cast failed in %s\n"
|
|
|
|
" Expected %s but found %s",
|
|
|
|
location->ToAsciiArray(), expected, value_description.str().c_str());
|
|
|
|
#else
|
|
|
|
UNREACHABLE();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2016-04-18 11:57:06 +00:00
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|