2015-12-11 12:26:16 +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.
|
|
|
|
|
2015-12-17 14:52:27 +00:00
|
|
|
#include "src/compiler/wasm-compiler.h"
|
|
|
|
|
|
|
|
#include "src/isolate-inl.h"
|
|
|
|
|
2016-03-11 13:18:18 +00:00
|
|
|
#include "src/base/platform/elapsed-timer.h"
|
2015-12-17 14:52:27 +00:00
|
|
|
#include "src/base/platform/platform.h"
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
#include "src/compiler/access-builder.h"
|
|
|
|
#include "src/compiler/common-operator.h"
|
|
|
|
#include "src/compiler/diamond.h"
|
|
|
|
#include "src/compiler/graph-visualizer.h"
|
2016-04-26 08:38:24 +00:00
|
|
|
#include "src/compiler/graph.h"
|
2015-12-11 12:26:16 +00:00
|
|
|
#include "src/compiler/instruction-selector.h"
|
2016-02-04 09:40:55 +00:00
|
|
|
#include "src/compiler/int64-lowering.h"
|
2015-12-11 12:26:16 +00:00
|
|
|
#include "src/compiler/js-generic-lowering.h"
|
|
|
|
#include "src/compiler/js-graph.h"
|
|
|
|
#include "src/compiler/js-operator.h"
|
|
|
|
#include "src/compiler/linkage.h"
|
|
|
|
#include "src/compiler/machine-operator.h"
|
|
|
|
#include "src/compiler/node-matchers.h"
|
|
|
|
#include "src/compiler/pipeline.h"
|
|
|
|
#include "src/compiler/source-position.h"
|
2016-04-26 08:38:24 +00:00
|
|
|
#include "src/compiler/zone-pool.h"
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
#include "src/code-factory.h"
|
|
|
|
#include "src/code-stubs.h"
|
2016-01-29 11:21:34 +00:00
|
|
|
#include "src/factory.h"
|
|
|
|
#include "src/log-inl.h"
|
|
|
|
#include "src/profiler/cpu-profiler.h"
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
#include "src/wasm/ast-decoder.h"
|
|
|
|
#include "src/wasm/wasm-module.h"
|
|
|
|
#include "src/wasm/wasm-opcodes.h"
|
|
|
|
|
|
|
|
// TODO(titzer): pull WASM_64 up to a common header.
|
|
|
|
#if !V8_TARGET_ARCH_32_BIT || V8_TARGET_ARCH_X64
|
|
|
|
#define WASM_64 1
|
|
|
|
#else
|
|
|
|
#define WASM_64 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
namespace compiler {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
const Operator* UnsupportedOpcode(wasm::WasmOpcode opcode) {
|
|
|
|
V8_Fatal(__FILE__, __LINE__, "Unsupported opcode #%d:%s", opcode,
|
|
|
|
wasm::WasmOpcodes::OpcodeName(opcode));
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MergeControlToEnd(JSGraph* jsgraph, Node* node) {
|
|
|
|
Graph* g = jsgraph->graph();
|
|
|
|
if (g->end()) {
|
|
|
|
NodeProperties::MergeControlToEnd(g, jsgraph->common(), node);
|
|
|
|
} else {
|
|
|
|
g->SetEnd(g->NewNode(jsgraph->common()->End(1), node));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
// A helper that handles building graph fragments for trapping.
|
|
|
|
// To avoid generating a ton of redundant code that just calls the runtime
|
|
|
|
// to trap, we generate a per-trap-reason block of code that all trap sites
|
|
|
|
// in this function will branch to.
|
|
|
|
class WasmTrapHelper : public ZoneObject {
|
|
|
|
public:
|
|
|
|
explicit WasmTrapHelper(WasmGraphBuilder* builder)
|
|
|
|
: builder_(builder),
|
|
|
|
jsgraph_(builder->jsgraph()),
|
2016-05-02 08:47:05 +00:00
|
|
|
graph_(builder->jsgraph() ? builder->jsgraph()->graph() : nullptr) {}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
// Make the current control path trap to unreachable.
|
2016-05-02 08:47:05 +00:00
|
|
|
void Unreachable(wasm::WasmCodePosition position) {
|
|
|
|
ConnectTrap(wasm::kTrapUnreachable, position);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-01-28 16:56:54 +00:00
|
|
|
// Always trap with the given reason.
|
2016-05-02 08:47:05 +00:00
|
|
|
void TrapAlways(wasm::TrapReason reason, wasm::WasmCodePosition position) {
|
|
|
|
ConnectTrap(reason, position);
|
|
|
|
}
|
2016-01-28 16:56:54 +00:00
|
|
|
|
2015-12-11 12:26:16 +00:00
|
|
|
// Add a check that traps if {node} is equal to {val}.
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* TrapIfEq32(wasm::TrapReason reason, Node* node, int32_t val,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-11 12:26:16 +00:00
|
|
|
Int32Matcher m(node);
|
|
|
|
if (m.HasValue() && !m.Is(val)) return graph()->start();
|
|
|
|
if (val == 0) {
|
2016-05-02 08:47:05 +00:00
|
|
|
AddTrapIfFalse(reason, node, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
} else {
|
|
|
|
AddTrapIfTrue(reason,
|
|
|
|
graph()->NewNode(jsgraph()->machine()->Word32Equal(), node,
|
2016-05-02 08:47:05 +00:00
|
|
|
jsgraph()->Int32Constant(val)),
|
|
|
|
position);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
return builder_->Control();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add a check that traps if {node} is zero.
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* ZeroCheck32(wasm::TrapReason reason, Node* node,
|
|
|
|
wasm::WasmCodePosition position) {
|
|
|
|
return TrapIfEq32(reason, node, 0, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add a check that traps if {node} is equal to {val}.
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* TrapIfEq64(wasm::TrapReason reason, Node* node, int64_t val,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-11 12:26:16 +00:00
|
|
|
Int64Matcher m(node);
|
|
|
|
if (m.HasValue() && !m.Is(val)) return graph()->start();
|
2016-05-02 08:47:05 +00:00
|
|
|
AddTrapIfTrue(reason, graph()->NewNode(jsgraph()->machine()->Word64Equal(),
|
|
|
|
node, jsgraph()->Int64Constant(val)),
|
|
|
|
position);
|
2015-12-11 12:26:16 +00:00
|
|
|
return builder_->Control();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add a check that traps if {node} is zero.
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* ZeroCheck64(wasm::TrapReason reason, Node* node,
|
|
|
|
wasm::WasmCodePosition position) {
|
|
|
|
return TrapIfEq64(reason, node, 0, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add a trap if {cond} is true.
|
2016-05-02 08:47:05 +00:00
|
|
|
void AddTrapIfTrue(wasm::TrapReason reason, Node* cond,
|
|
|
|
wasm::WasmCodePosition position) {
|
|
|
|
AddTrapIf(reason, cond, true, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add a trap if {cond} is false.
|
2016-05-02 08:47:05 +00:00
|
|
|
void AddTrapIfFalse(wasm::TrapReason reason, Node* cond,
|
|
|
|
wasm::WasmCodePosition position) {
|
|
|
|
AddTrapIf(reason, cond, false, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add a trap if {cond} is true or false according to {iftrue}.
|
2016-05-02 08:47:05 +00:00
|
|
|
void AddTrapIf(wasm::TrapReason reason, Node* cond, bool iftrue,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-11 12:26:16 +00:00
|
|
|
Node** effect_ptr = builder_->effect_;
|
|
|
|
Node** control_ptr = builder_->control_;
|
|
|
|
Node* before = *effect_ptr;
|
|
|
|
BranchHint hint = iftrue ? BranchHint::kFalse : BranchHint::kTrue;
|
|
|
|
Node* branch = graph()->NewNode(common()->Branch(hint), cond, *control_ptr);
|
|
|
|
Node* if_true = graph()->NewNode(common()->IfTrue(), branch);
|
|
|
|
Node* if_false = graph()->NewNode(common()->IfFalse(), branch);
|
|
|
|
|
|
|
|
*control_ptr = iftrue ? if_true : if_false;
|
2016-05-02 08:47:05 +00:00
|
|
|
ConnectTrap(reason, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
*control_ptr = iftrue ? if_false : if_true;
|
|
|
|
*effect_ptr = before;
|
|
|
|
}
|
|
|
|
|
2016-01-28 16:56:54 +00:00
|
|
|
Node* GetTrapValue(wasm::FunctionSig* sig) {
|
|
|
|
if (sig->return_count() > 0) {
|
|
|
|
switch (sig->GetReturn()) {
|
|
|
|
case wasm::kAstI32:
|
|
|
|
return jsgraph()->Int32Constant(0xdeadbeef);
|
|
|
|
case wasm::kAstI64:
|
|
|
|
return jsgraph()->Int64Constant(0xdeadbeefdeadbeef);
|
|
|
|
case wasm::kAstF32:
|
|
|
|
return jsgraph()->Float32Constant(bit_cast<float>(0xdeadbeef));
|
|
|
|
case wasm::kAstF64:
|
|
|
|
return jsgraph()->Float64Constant(
|
|
|
|
bit_cast<double>(0xdeadbeefdeadbeef));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return jsgraph()->Int32Constant(0xdeadbeef);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-11 12:26:16 +00:00
|
|
|
private:
|
|
|
|
WasmGraphBuilder* builder_;
|
|
|
|
JSGraph* jsgraph_;
|
|
|
|
Graph* graph_;
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* trap_merge_ = nullptr;
|
|
|
|
Node* trap_effect_;
|
|
|
|
Node* trap_reason_;
|
|
|
|
Node* trap_position_;
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
JSGraph* jsgraph() { return jsgraph_; }
|
|
|
|
Graph* graph() { return jsgraph_->graph(); }
|
|
|
|
CommonOperatorBuilder* common() { return jsgraph()->common(); }
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
void ConnectTrap(wasm::TrapReason reason, wasm::WasmCodePosition position) {
|
|
|
|
DCHECK(position != wasm::kNoCodePosition);
|
|
|
|
Node* reason_node = builder_->Int32Constant(
|
|
|
|
wasm::WasmOpcodes::TrapReasonToMessageId(reason));
|
|
|
|
Node* position_node = builder_->Int32Constant(position);
|
|
|
|
if (trap_merge_ == nullptr) {
|
|
|
|
// Create trap code for the first time.
|
|
|
|
return BuildTrapCode(reason_node, position_node);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
// Connect the current control and effect to the existing trap code.
|
2016-05-02 08:47:05 +00:00
|
|
|
builder_->AppendToMerge(trap_merge_, builder_->Control());
|
|
|
|
builder_->AppendToPhi(trap_effect_, builder_->Effect());
|
|
|
|
builder_->AppendToPhi(trap_reason_, reason_node);
|
|
|
|
builder_->AppendToPhi(trap_position_, position_node);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
void BuildTrapCode(Node* reason_node, Node* position_node) {
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* end;
|
|
|
|
Node** control_ptr = builder_->control_;
|
|
|
|
Node** effect_ptr = builder_->effect_;
|
|
|
|
wasm::ModuleEnv* module = builder_->module_;
|
2016-05-02 08:47:05 +00:00
|
|
|
DCHECK(trap_merge_ == NULL);
|
|
|
|
*control_ptr = trap_merge_ =
|
2015-12-11 12:26:16 +00:00
|
|
|
graph()->NewNode(common()->Merge(1), *control_ptr);
|
2016-05-02 08:47:05 +00:00
|
|
|
*effect_ptr = trap_effect_ =
|
2015-12-11 12:26:16 +00:00
|
|
|
graph()->NewNode(common()->EffectPhi(1), *effect_ptr, *control_ptr);
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_reason_ =
|
|
|
|
graph()->NewNode(common()->Phi(MachineRepresentation::kWord32, 1),
|
|
|
|
reason_node, *control_ptr);
|
|
|
|
trap_position_ =
|
|
|
|
graph()->NewNode(common()->Phi(MachineRepresentation::kWord32, 1),
|
|
|
|
position_node, *control_ptr);
|
|
|
|
|
|
|
|
Node* trap_reason_smi = builder_->BuildChangeInt32ToSmi(trap_reason_);
|
|
|
|
Node* trap_position_smi = builder_->BuildChangeInt32ToSmi(trap_position_);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-01-27 11:04:02 +00:00
|
|
|
if (module && !module->instance->context.is_null()) {
|
2015-12-11 12:26:16 +00:00
|
|
|
// Use the module context to call the runtime to throw an exception.
|
2016-04-20 14:51:35 +00:00
|
|
|
Runtime::FunctionId f = Runtime::kThrowWasmError;
|
2015-12-11 12:26:16 +00:00
|
|
|
const Runtime::Function* fun = Runtime::FunctionForId(f);
|
|
|
|
CallDescriptor* desc = Linkage::GetRuntimeCallDescriptor(
|
|
|
|
jsgraph()->zone(), f, fun->nargs, Operator::kNoProperties,
|
|
|
|
CallDescriptor::kNoFlags);
|
2016-05-12 11:58:27 +00:00
|
|
|
// CEntryStubConstant nodes have to be created and cached in the main
|
|
|
|
// thread. At the moment this is only done for CEntryStubConstant(1).
|
|
|
|
DCHECK_EQ(1, fun->result_size);
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* inputs[] = {
|
|
|
|
jsgraph()->CEntryStubConstant(fun->result_size), // C entry
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_reason_smi, // message id
|
|
|
|
trap_position_smi, // byte position
|
2015-12-11 12:26:16 +00:00
|
|
|
jsgraph()->ExternalConstant(
|
2016-05-12 11:58:27 +00:00
|
|
|
ExternalReference(f, jsgraph()->isolate())), // ref
|
|
|
|
jsgraph()->Int32Constant(fun->nargs), // arity
|
|
|
|
builder_->HeapConstant(module->instance->context), // context
|
2015-12-11 12:26:16 +00:00
|
|
|
*effect_ptr,
|
|
|
|
*control_ptr};
|
|
|
|
|
|
|
|
Node* node = graph()->NewNode(
|
|
|
|
common()->Call(desc), static_cast<int>(arraysize(inputs)), inputs);
|
|
|
|
*control_ptr = node;
|
|
|
|
*effect_ptr = node;
|
|
|
|
}
|
|
|
|
if (false) {
|
|
|
|
// End the control flow with a throw
|
|
|
|
Node* thrw =
|
|
|
|
graph()->NewNode(common()->Throw(), jsgraph()->ZeroConstant(),
|
|
|
|
*effect_ptr, *control_ptr);
|
|
|
|
end = thrw;
|
|
|
|
} else {
|
|
|
|
// End the control flow with returning 0xdeadbeef
|
2016-01-28 16:56:54 +00:00
|
|
|
Node* ret_value = GetTrapValue(builder_->GetFunctionSignature());
|
2015-12-11 14:54:07 +00:00
|
|
|
end = graph()->NewNode(jsgraph()->common()->Return(), ret_value,
|
|
|
|
*effect_ptr, *control_ptr);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MergeControlToEnd(jsgraph(), end);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-04-26 12:46:03 +00:00
|
|
|
WasmGraphBuilder::WasmGraphBuilder(
|
|
|
|
Zone* zone, JSGraph* jsgraph, wasm::FunctionSig* function_signature,
|
|
|
|
compiler::SourcePositionTable* source_position_table)
|
2015-12-11 12:26:16 +00:00
|
|
|
: zone_(zone),
|
|
|
|
jsgraph_(jsgraph),
|
|
|
|
module_(nullptr),
|
|
|
|
mem_buffer_(nullptr),
|
|
|
|
mem_size_(nullptr),
|
|
|
|
function_table_(nullptr),
|
|
|
|
control_(nullptr),
|
|
|
|
effect_(nullptr),
|
|
|
|
cur_buffer_(def_buffer_),
|
|
|
|
cur_bufsize_(kDefaultBufferSize),
|
2015-12-11 14:54:07 +00:00
|
|
|
trap_(new (zone) WasmTrapHelper(this)),
|
2016-04-26 12:46:03 +00:00
|
|
|
function_signature_(function_signature),
|
|
|
|
source_position_table_(source_position_table) {
|
2015-12-11 12:26:16 +00:00
|
|
|
DCHECK_NOT_NULL(jsgraph_);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Error() { return jsgraph()->Dead(); }
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Start(unsigned params) {
|
|
|
|
Node* start = graph()->NewNode(jsgraph()->common()->Start(params));
|
|
|
|
graph()->SetStart(start);
|
|
|
|
return start;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Param(unsigned index, wasm::LocalType type) {
|
|
|
|
return graph()->NewNode(jsgraph()->common()->Parameter(index),
|
|
|
|
graph()->start());
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Loop(Node* entry) {
|
|
|
|
return graph()->NewNode(jsgraph()->common()->Loop(1), entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Terminate(Node* effect, Node* control) {
|
|
|
|
Node* terminate =
|
|
|
|
graph()->NewNode(jsgraph()->common()->Terminate(), effect, control);
|
|
|
|
MergeControlToEnd(jsgraph(), terminate);
|
|
|
|
return terminate;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned WasmGraphBuilder::InputCount(Node* node) {
|
|
|
|
return static_cast<unsigned>(node->InputCount());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool WasmGraphBuilder::IsPhiWithMerge(Node* phi, Node* merge) {
|
|
|
|
return phi && IrOpcode::IsPhiOpcode(phi->opcode()) &&
|
|
|
|
NodeProperties::GetControlInput(phi) == merge;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WasmGraphBuilder::AppendToMerge(Node* merge, Node* from) {
|
|
|
|
DCHECK(IrOpcode::IsMergeOpcode(merge->opcode()));
|
|
|
|
merge->AppendInput(jsgraph()->zone(), from);
|
|
|
|
int new_size = merge->InputCount();
|
|
|
|
NodeProperties::ChangeOp(
|
|
|
|
merge, jsgraph()->common()->ResizeMergeOrPhi(merge->op(), new_size));
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
void WasmGraphBuilder::AppendToPhi(Node* phi, Node* from) {
|
2015-12-11 12:26:16 +00:00
|
|
|
DCHECK(IrOpcode::IsPhiOpcode(phi->opcode()));
|
|
|
|
int new_size = phi->InputCount();
|
|
|
|
phi->InsertInput(jsgraph()->zone(), phi->InputCount() - 1, from);
|
|
|
|
NodeProperties::ChangeOp(
|
|
|
|
phi, jsgraph()->common()->ResizeMergeOrPhi(phi->op(), new_size));
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Merge(unsigned count, Node** controls) {
|
|
|
|
return graph()->NewNode(jsgraph()->common()->Merge(count), count, controls);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Phi(wasm::LocalType type, unsigned count, Node** vals,
|
|
|
|
Node* control) {
|
|
|
|
DCHECK(IrOpcode::IsMergeOpcode(control->opcode()));
|
2016-03-29 12:50:36 +00:00
|
|
|
Node** buf = Realloc(vals, count, count + 1);
|
2015-12-11 12:26:16 +00:00
|
|
|
buf[count] = control;
|
|
|
|
return graph()->NewNode(jsgraph()->common()->Phi(type, count), count + 1,
|
|
|
|
buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::EffectPhi(unsigned count, Node** effects,
|
|
|
|
Node* control) {
|
|
|
|
DCHECK(IrOpcode::IsMergeOpcode(control->opcode()));
|
2016-03-29 12:50:36 +00:00
|
|
|
Node** buf = Realloc(effects, count, count + 1);
|
2015-12-11 12:26:16 +00:00
|
|
|
buf[count] = control;
|
|
|
|
return graph()->NewNode(jsgraph()->common()->EffectPhi(count), count + 1,
|
|
|
|
buf);
|
|
|
|
}
|
|
|
|
|
2016-04-20 14:51:35 +00:00
|
|
|
Node* WasmGraphBuilder::NumberConstant(int32_t value) {
|
|
|
|
return jsgraph()->Constant(value);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Int32Constant(int32_t value) {
|
|
|
|
return jsgraph()->Int32Constant(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Int64Constant(int64_t value) {
|
|
|
|
return jsgraph()->Int64Constant(value);
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::Binop(wasm::WasmOpcode opcode, Node* left, Node* right,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-11 12:26:16 +00:00
|
|
|
const Operator* op;
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
switch (opcode) {
|
|
|
|
case wasm::kExprI32Add:
|
|
|
|
op = m->Int32Add();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32Sub:
|
|
|
|
op = m->Int32Sub();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32Mul:
|
|
|
|
op = m->Int32Mul();
|
|
|
|
break;
|
2016-03-30 11:37:24 +00:00
|
|
|
case wasm::kExprI32DivS:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI32DivS(left, right, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprI32DivU:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI32DivU(left, right, position);
|
2016-03-30 11:37:24 +00:00
|
|
|
case wasm::kExprI32RemS:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI32RemS(left, right, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprI32RemU:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI32RemU(left, right, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprI32And:
|
|
|
|
op = m->Word32And();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32Ior:
|
|
|
|
op = m->Word32Or();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32Xor:
|
|
|
|
op = m->Word32Xor();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32Shl:
|
|
|
|
op = m->Word32Shl();
|
2016-03-30 12:38:47 +00:00
|
|
|
right = MaskShiftCount32(right);
|
2015-12-11 12:26:16 +00:00
|
|
|
break;
|
|
|
|
case wasm::kExprI32ShrU:
|
|
|
|
op = m->Word32Shr();
|
2016-03-30 12:38:47 +00:00
|
|
|
right = MaskShiftCount32(right);
|
2015-12-11 12:26:16 +00:00
|
|
|
break;
|
|
|
|
case wasm::kExprI32ShrS:
|
|
|
|
op = m->Word32Sar();
|
2016-03-30 12:38:47 +00:00
|
|
|
right = MaskShiftCount32(right);
|
2015-12-11 12:26:16 +00:00
|
|
|
break;
|
2016-03-03 05:10:15 +00:00
|
|
|
case wasm::kExprI32Ror:
|
|
|
|
op = m->Word32Ror();
|
2016-03-30 12:38:47 +00:00
|
|
|
right = MaskShiftCount32(right);
|
2016-03-03 05:10:15 +00:00
|
|
|
break;
|
|
|
|
case wasm::kExprI32Rol:
|
2016-03-30 12:38:47 +00:00
|
|
|
right = MaskShiftCount32(right);
|
2016-03-03 05:10:15 +00:00
|
|
|
return BuildI32Rol(left, right);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprI32Eq:
|
|
|
|
op = m->Word32Equal();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32Ne:
|
|
|
|
return Invert(Binop(wasm::kExprI32Eq, left, right));
|
|
|
|
case wasm::kExprI32LtS:
|
|
|
|
op = m->Int32LessThan();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32LeS:
|
|
|
|
op = m->Int32LessThanOrEqual();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32LtU:
|
|
|
|
op = m->Uint32LessThan();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32LeU:
|
|
|
|
op = m->Uint32LessThanOrEqual();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32GtS:
|
|
|
|
op = m->Int32LessThan();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32GeS:
|
|
|
|
op = m->Int32LessThanOrEqual();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32GtU:
|
|
|
|
op = m->Uint32LessThan();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32GeU:
|
|
|
|
op = m->Uint32LessThanOrEqual();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
2016-02-04 09:40:55 +00:00
|
|
|
case wasm::kExprI64And:
|
|
|
|
op = m->Word64And();
|
|
|
|
break;
|
2016-03-14 15:33:15 +00:00
|
|
|
case wasm::kExprI64Add:
|
|
|
|
op = m->Int64Add();
|
|
|
|
break;
|
2016-03-16 10:56:29 +00:00
|
|
|
case wasm::kExprI64Sub:
|
|
|
|
op = m->Int64Sub();
|
|
|
|
break;
|
2016-03-30 10:39:04 +00:00
|
|
|
case wasm::kExprI64Mul:
|
|
|
|
op = m->Int64Mul();
|
|
|
|
break;
|
2016-03-15 06:51:53 +00:00
|
|
|
case wasm::kExprI64DivS:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI64DivS(left, right, position);
|
2016-03-15 06:51:53 +00:00
|
|
|
case wasm::kExprI64DivU:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI64DivU(left, right, position);
|
2016-03-15 06:51:53 +00:00
|
|
|
case wasm::kExprI64RemS:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI64RemS(left, right, position);
|
2016-03-15 06:51:53 +00:00
|
|
|
case wasm::kExprI64RemU:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI64RemU(left, right, position);
|
2016-02-23 16:30:27 +00:00
|
|
|
case wasm::kExprI64Ior:
|
|
|
|
op = m->Word64Or();
|
|
|
|
break;
|
2016-02-24 09:51:30 +00:00
|
|
|
case wasm::kExprI64Xor:
|
|
|
|
op = m->Word64Xor();
|
|
|
|
break;
|
2016-03-07 15:17:54 +00:00
|
|
|
case wasm::kExprI64Shl:
|
|
|
|
op = m->Word64Shl();
|
2016-03-30 12:38:47 +00:00
|
|
|
right = MaskShiftCount64(right);
|
2016-03-07 15:17:54 +00:00
|
|
|
break;
|
2016-03-09 16:37:29 +00:00
|
|
|
case wasm::kExprI64ShrU:
|
|
|
|
op = m->Word64Shr();
|
2016-03-30 12:38:47 +00:00
|
|
|
right = MaskShiftCount64(right);
|
2016-03-09 16:37:29 +00:00
|
|
|
break;
|
|
|
|
case wasm::kExprI64ShrS:
|
|
|
|
op = m->Word64Sar();
|
2016-03-30 12:38:47 +00:00
|
|
|
right = MaskShiftCount64(right);
|
2016-03-09 16:37:29 +00:00
|
|
|
break;
|
2016-02-24 12:09:19 +00:00
|
|
|
case wasm::kExprI64Eq:
|
|
|
|
op = m->Word64Equal();
|
|
|
|
break;
|
2016-02-25 10:12:35 +00:00
|
|
|
case wasm::kExprI64Ne:
|
|
|
|
return Invert(Binop(wasm::kExprI64Eq, left, right));
|
2016-02-25 12:14:35 +00:00
|
|
|
case wasm::kExprI64LtS:
|
|
|
|
op = m->Int64LessThan();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI64LeS:
|
|
|
|
op = m->Int64LessThanOrEqual();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI64LtU:
|
|
|
|
op = m->Uint64LessThan();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI64LeU:
|
|
|
|
op = m->Uint64LessThanOrEqual();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI64GtS:
|
|
|
|
op = m->Int64LessThan();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
|
|
|
case wasm::kExprI64GeS:
|
|
|
|
op = m->Int64LessThanOrEqual();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
|
|
|
case wasm::kExprI64GtU:
|
|
|
|
op = m->Uint64LessThan();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
|
|
|
case wasm::kExprI64GeU:
|
|
|
|
op = m->Uint64LessThanOrEqual();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
2016-03-03 05:10:15 +00:00
|
|
|
case wasm::kExprI64Ror:
|
|
|
|
op = m->Word64Ror();
|
2016-03-30 12:38:47 +00:00
|
|
|
right = MaskShiftCount64(right);
|
2016-03-03 05:10:15 +00:00
|
|
|
break;
|
|
|
|
case wasm::kExprI64Rol:
|
|
|
|
return BuildI64Rol(left, right);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprF32CopySign:
|
|
|
|
return BuildF32CopySign(left, right);
|
|
|
|
case wasm::kExprF64CopySign:
|
|
|
|
return BuildF64CopySign(left, right);
|
|
|
|
case wasm::kExprF32Add:
|
|
|
|
op = m->Float32Add();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32Sub:
|
|
|
|
op = m->Float32Sub();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32Mul:
|
|
|
|
op = m->Float32Mul();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32Div:
|
|
|
|
op = m->Float32Div();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32Eq:
|
|
|
|
op = m->Float32Equal();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32Ne:
|
|
|
|
return Invert(Binop(wasm::kExprF32Eq, left, right));
|
|
|
|
case wasm::kExprF32Lt:
|
|
|
|
op = m->Float32LessThan();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32Ge:
|
|
|
|
op = m->Float32LessThanOrEqual();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32Gt:
|
|
|
|
op = m->Float32LessThan();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32Le:
|
|
|
|
op = m->Float32LessThanOrEqual();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Add:
|
|
|
|
op = m->Float64Add();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Sub:
|
|
|
|
op = m->Float64Sub();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Mul:
|
|
|
|
op = m->Float64Mul();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Div:
|
|
|
|
op = m->Float64Div();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Eq:
|
|
|
|
op = m->Float64Equal();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Ne:
|
|
|
|
return Invert(Binop(wasm::kExprF64Eq, left, right));
|
|
|
|
case wasm::kExprF64Lt:
|
|
|
|
op = m->Float64LessThan();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Le:
|
|
|
|
op = m->Float64LessThanOrEqual();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Gt:
|
|
|
|
op = m->Float64LessThan();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Ge:
|
|
|
|
op = m->Float64LessThanOrEqual();
|
|
|
|
std::swap(left, right);
|
|
|
|
break;
|
2015-12-11 16:39:54 +00:00
|
|
|
case wasm::kExprF32Min:
|
|
|
|
return BuildF32Min(left, right);
|
|
|
|
case wasm::kExprF64Min:
|
|
|
|
return BuildF64Min(left, right);
|
|
|
|
case wasm::kExprF32Max:
|
|
|
|
return BuildF32Max(left, right);
|
|
|
|
case wasm::kExprF64Max:
|
|
|
|
return BuildF64Max(left, right);
|
2016-05-10 17:58:03 +00:00
|
|
|
case wasm::kExprF64Pow:
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildF64Pow(left, right);
|
2016-05-10 17:58:03 +00:00
|
|
|
case wasm::kExprF64Atan2:
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildF64Atan2(left, right);
|
2016-05-10 17:58:03 +00:00
|
|
|
case wasm::kExprF64Mod:
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildF64Mod(left, right);
|
2016-05-10 17:58:03 +00:00
|
|
|
case wasm::kExprI32AsmjsDivS:
|
|
|
|
return BuildI32AsmjsDivS(left, right);
|
|
|
|
case wasm::kExprI32AsmjsDivU:
|
|
|
|
return BuildI32AsmjsDivU(left, right);
|
|
|
|
case wasm::kExprI32AsmjsRemS:
|
|
|
|
return BuildI32AsmjsRemS(left, right);
|
|
|
|
case wasm::kExprI32AsmjsRemU:
|
|
|
|
return BuildI32AsmjsRemU(left, right);
|
2016-05-11 11:50:30 +00:00
|
|
|
case wasm::kExprI32AsmjsStoreMem8:
|
|
|
|
return BuildAsmjsStoreMem(MachineType::Int8(), left, right);
|
|
|
|
case wasm::kExprI32AsmjsStoreMem16:
|
|
|
|
return BuildAsmjsStoreMem(MachineType::Int16(), left, right);
|
|
|
|
case wasm::kExprI32AsmjsStoreMem:
|
|
|
|
return BuildAsmjsStoreMem(MachineType::Int32(), left, right);
|
|
|
|
case wasm::kExprF32AsmjsStoreMem:
|
|
|
|
return BuildAsmjsStoreMem(MachineType::Float32(), left, right);
|
|
|
|
case wasm::kExprF64AsmjsStoreMem:
|
|
|
|
return BuildAsmjsStoreMem(MachineType::Float64(), left, right);
|
2015-12-11 12:26:16 +00:00
|
|
|
default:
|
|
|
|
op = UnsupportedOpcode(opcode);
|
|
|
|
}
|
|
|
|
return graph()->NewNode(op, left, right);
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::Unop(wasm::WasmOpcode opcode, Node* input,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-11 12:26:16 +00:00
|
|
|
const Operator* op;
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
switch (opcode) {
|
2016-02-28 19:01:27 +00:00
|
|
|
case wasm::kExprI32Eqz:
|
2015-12-11 12:26:16 +00:00
|
|
|
op = m->Word32Equal();
|
|
|
|
return graph()->NewNode(op, input, jsgraph()->Int32Constant(0));
|
|
|
|
case wasm::kExprF32Abs:
|
|
|
|
op = m->Float32Abs();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32Neg:
|
2015-12-16 11:44:46 +00:00
|
|
|
return BuildF32Neg(input);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprF32Sqrt:
|
|
|
|
op = m->Float32Sqrt();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Abs:
|
|
|
|
op = m->Float64Abs();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64Neg:
|
2015-12-16 11:44:46 +00:00
|
|
|
return BuildF64Neg(input);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprF64Sqrt:
|
|
|
|
op = m->Float64Sqrt();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32SConvertF64:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI32SConvertF64(input, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprI32UConvertF64:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI32UConvertF64(input, position);
|
2016-05-11 09:31:32 +00:00
|
|
|
case wasm::kExprI32AsmjsSConvertF64:
|
|
|
|
return BuildI32AsmjsSConvertF64(input);
|
|
|
|
case wasm::kExprI32AsmjsUConvertF64:
|
|
|
|
return BuildI32AsmjsUConvertF64(input);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprF32ConvertF64:
|
|
|
|
op = m->TruncateFloat64ToFloat32();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64SConvertI32:
|
|
|
|
op = m->ChangeInt32ToFloat64();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64UConvertI32:
|
|
|
|
op = m->ChangeUint32ToFloat64();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32SConvertI32:
|
[turbofan] Add the RoundInt32ToFloat32 operator to turbofan.
The new operator converts an int32 input to float32. If the input cannot
be represented exactly in float32, the value is rounded using the
round-ties-even rounding mode (the default rounding mode).
I provide implementations of the new operator for x64, ia32, arm, arm64,
mips, mips64, ppc, and ppc64.
R=titzer@chromium.org, v8-arm-ports@googlegroups.com, v8-mips-ports@googlegroups.com, v8-ppc-ports@googlegroups.com
Review URL: https://codereview.chromium.org/1589363002
Cr-Commit-Position: refs/heads/master@{#33347}
2016-01-16 13:11:40 +00:00
|
|
|
op = m->RoundInt32ToFloat32();
|
2015-12-11 12:26:16 +00:00
|
|
|
break;
|
|
|
|
case wasm::kExprF32UConvertI32:
|
2016-02-06 18:07:39 +00:00
|
|
|
op = m->RoundUint32ToFloat32();
|
2015-12-11 12:26:16 +00:00
|
|
|
break;
|
|
|
|
case wasm::kExprI32SConvertF32:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI32SConvertF32(input, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprI32UConvertF32:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI32UConvertF32(input, position);
|
2016-05-11 09:31:32 +00:00
|
|
|
case wasm::kExprI32AsmjsSConvertF32:
|
|
|
|
return BuildI32AsmjsSConvertF32(input);
|
|
|
|
case wasm::kExprI32AsmjsUConvertF32:
|
|
|
|
return BuildI32AsmjsUConvertF32(input);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprF64ConvertF32:
|
|
|
|
op = m->ChangeFloat32ToFloat64();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32ReinterpretI32:
|
|
|
|
op = m->BitcastInt32ToFloat32();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32ReinterpretF32:
|
|
|
|
op = m->BitcastFloat32ToInt32();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32Clz:
|
|
|
|
op = m->Word32Clz();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI32Ctz: {
|
|
|
|
if (m->Word32Ctz().IsSupported()) {
|
|
|
|
op = m->Word32Ctz().op();
|
|
|
|
break;
|
2016-02-16 11:14:01 +00:00
|
|
|
} else if (m->Word32ReverseBits().IsSupported()) {
|
|
|
|
Node* reversed = graph()->NewNode(m->Word32ReverseBits().op(), input);
|
|
|
|
Node* result = graph()->NewNode(m->Word32Clz(), reversed);
|
|
|
|
return result;
|
2015-12-11 12:26:16 +00:00
|
|
|
} else {
|
|
|
|
return BuildI32Ctz(input);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case wasm::kExprI32Popcnt: {
|
|
|
|
if (m->Word32Popcnt().IsSupported()) {
|
|
|
|
op = m->Word32Popcnt().op();
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
return BuildI32Popcnt(input);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case wasm::kExprF32Floor: {
|
2016-02-02 12:26:38 +00:00
|
|
|
if (!m->Float32RoundDown().IsSupported()) return BuildF32Floor(input);
|
|
|
|
op = m->Float32RoundDown().op();
|
|
|
|
break;
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
case wasm::kExprF32Ceil: {
|
2016-02-02 12:26:38 +00:00
|
|
|
if (!m->Float32RoundUp().IsSupported()) return BuildF32Ceil(input);
|
|
|
|
op = m->Float32RoundUp().op();
|
|
|
|
break;
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
case wasm::kExprF32Trunc: {
|
2016-02-02 12:26:38 +00:00
|
|
|
if (!m->Float32RoundTruncate().IsSupported()) return BuildF32Trunc(input);
|
|
|
|
op = m->Float32RoundTruncate().op();
|
2016-01-20 15:43:03 +00:00
|
|
|
break;
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
case wasm::kExprF32NearestInt: {
|
2016-02-02 12:26:38 +00:00
|
|
|
if (!m->Float32RoundTiesEven().IsSupported())
|
|
|
|
return BuildF32NearestInt(input);
|
|
|
|
op = m->Float32RoundTiesEven().op();
|
|
|
|
break;
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
case wasm::kExprF64Floor: {
|
2016-02-02 12:26:38 +00:00
|
|
|
if (!m->Float64RoundDown().IsSupported()) return BuildF64Floor(input);
|
|
|
|
op = m->Float64RoundDown().op();
|
|
|
|
break;
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
case wasm::kExprF64Ceil: {
|
2016-02-02 12:26:38 +00:00
|
|
|
if (!m->Float64RoundUp().IsSupported()) return BuildF64Ceil(input);
|
|
|
|
op = m->Float64RoundUp().op();
|
|
|
|
break;
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
case wasm::kExprF64Trunc: {
|
2016-02-02 10:57:58 +00:00
|
|
|
if (!m->Float64RoundTruncate().IsSupported()) return BuildF64Trunc(input);
|
|
|
|
op = m->Float64RoundTruncate().op();
|
|
|
|
break;
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
case wasm::kExprF64NearestInt: {
|
2016-02-02 12:26:38 +00:00
|
|
|
if (!m->Float64RoundTiesEven().IsSupported())
|
|
|
|
return BuildF64NearestInt(input);
|
|
|
|
op = m->Float64RoundTiesEven().op();
|
|
|
|
break;
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
2016-03-03 01:22:01 +00:00
|
|
|
case wasm::kExprF64Acos: {
|
|
|
|
return BuildF64Acos(input);
|
|
|
|
}
|
|
|
|
case wasm::kExprF64Asin: {
|
|
|
|
return BuildF64Asin(input);
|
|
|
|
}
|
|
|
|
case wasm::kExprF64Atan: {
|
|
|
|
return BuildF64Atan(input);
|
|
|
|
}
|
|
|
|
case wasm::kExprF64Cos: {
|
|
|
|
return BuildF64Cos(input);
|
|
|
|
}
|
|
|
|
case wasm::kExprF64Sin: {
|
|
|
|
return BuildF64Sin(input);
|
|
|
|
}
|
|
|
|
case wasm::kExprF64Tan: {
|
|
|
|
return BuildF64Tan(input);
|
|
|
|
}
|
|
|
|
case wasm::kExprF64Exp: {
|
|
|
|
return BuildF64Exp(input);
|
|
|
|
}
|
|
|
|
case wasm::kExprF64Log: {
|
|
|
|
return BuildF64Log(input);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprI32ConvertI64:
|
|
|
|
op = m->TruncateInt64ToInt32();
|
|
|
|
break;
|
2016-03-09 16:20:59 +00:00
|
|
|
case wasm::kExprI64SConvertI32:
|
|
|
|
op = m->ChangeInt32ToInt64();
|
|
|
|
break;
|
|
|
|
case wasm::kExprI64UConvertI32:
|
|
|
|
op = m->ChangeUint32ToUint64();
|
|
|
|
break;
|
2016-03-15 12:45:43 +00:00
|
|
|
case wasm::kExprF64ReinterpretI64:
|
|
|
|
op = m->BitcastInt64ToFloat64();
|
|
|
|
break;
|
2016-03-15 12:17:09 +00:00
|
|
|
case wasm::kExprI64ReinterpretF64:
|
|
|
|
op = m->BitcastFloat64ToInt64();
|
|
|
|
break;
|
2016-03-16 11:02:28 +00:00
|
|
|
case wasm::kExprI64Clz:
|
|
|
|
op = m->Word64Clz();
|
|
|
|
break;
|
2016-03-16 12:14:44 +00:00
|
|
|
case wasm::kExprI64Ctz: {
|
|
|
|
if (m->Word64Ctz().IsSupported()) {
|
|
|
|
op = m->Word64Ctz().op();
|
|
|
|
break;
|
|
|
|
} else if (m->Is32() && m->Word32Ctz().IsSupported()) {
|
|
|
|
op = m->Word64CtzPlaceholder();
|
|
|
|
break;
|
|
|
|
} else if (m->Word64ReverseBits().IsSupported()) {
|
|
|
|
Node* reversed = graph()->NewNode(m->Word64ReverseBits().op(), input);
|
|
|
|
Node* result = graph()->NewNode(m->Word64Clz(), reversed);
|
|
|
|
return result;
|
|
|
|
} else {
|
|
|
|
return BuildI64Ctz(input);
|
|
|
|
}
|
|
|
|
}
|
2016-03-15 10:41:55 +00:00
|
|
|
case wasm::kExprI64Popcnt: {
|
|
|
|
if (m->Word64Popcnt().IsSupported()) {
|
|
|
|
op = m->Word64Popcnt().op();
|
|
|
|
} else if (m->Is32() && m->Word32Popcnt().IsSupported()) {
|
|
|
|
op = m->Word64PopcntPlaceholder();
|
|
|
|
} else {
|
|
|
|
return BuildI64Popcnt(input);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2016-03-14 17:55:14 +00:00
|
|
|
case wasm::kExprI64Eqz:
|
|
|
|
op = m->Word64Equal();
|
|
|
|
return graph()->NewNode(op, input, jsgraph()->Int64Constant(0));
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kExprF32SConvertI64:
|
2016-03-08 12:41:25 +00:00
|
|
|
if (m->Is32()) {
|
2016-03-04 09:50:48 +00:00
|
|
|
return BuildF32SConvertI64(input);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
op = m->RoundInt64ToFloat32();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF32UConvertI64:
|
2016-03-08 12:41:25 +00:00
|
|
|
if (m->Is32()) {
|
2016-03-04 09:50:48 +00:00
|
|
|
return BuildF32UConvertI64(input);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
op = m->RoundUint64ToFloat32();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64SConvertI64:
|
2016-03-08 12:41:25 +00:00
|
|
|
if (m->Is32()) {
|
2016-03-04 09:50:48 +00:00
|
|
|
return BuildF64SConvertI64(input);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
op = m->RoundInt64ToFloat64();
|
|
|
|
break;
|
|
|
|
case wasm::kExprF64UConvertI64:
|
2016-03-08 12:41:25 +00:00
|
|
|
if (m->Is32()) {
|
2016-03-04 09:50:48 +00:00
|
|
|
return BuildF64UConvertI64(input);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
op = m->RoundUint64ToFloat64();
|
|
|
|
break;
|
2016-05-11 11:50:30 +00:00
|
|
|
case wasm::kExprI64SConvertF32:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI64SConvertF32(input, position);
|
2016-05-11 11:50:30 +00:00
|
|
|
case wasm::kExprI64SConvertF64:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI64SConvertF64(input, position);
|
2016-05-11 11:50:30 +00:00
|
|
|
case wasm::kExprI64UConvertF32:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI64UConvertF32(input, position);
|
2016-05-11 11:50:30 +00:00
|
|
|
case wasm::kExprI64UConvertF64:
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildI64UConvertF64(input, position);
|
2016-05-11 11:50:30 +00:00
|
|
|
case wasm::kExprI32AsmjsLoadMem8S:
|
|
|
|
return BuildAsmjsLoadMem(MachineType::Int8(), input);
|
|
|
|
case wasm::kExprI32AsmjsLoadMem8U:
|
|
|
|
return BuildAsmjsLoadMem(MachineType::Uint8(), input);
|
|
|
|
case wasm::kExprI32AsmjsLoadMem16S:
|
|
|
|
return BuildAsmjsLoadMem(MachineType::Int16(), input);
|
|
|
|
case wasm::kExprI32AsmjsLoadMem16U:
|
|
|
|
return BuildAsmjsLoadMem(MachineType::Uint16(), input);
|
|
|
|
case wasm::kExprI32AsmjsLoadMem:
|
|
|
|
return BuildAsmjsLoadMem(MachineType::Int32(), input);
|
|
|
|
case wasm::kExprF32AsmjsLoadMem:
|
|
|
|
return BuildAsmjsLoadMem(MachineType::Float32(), input);
|
|
|
|
case wasm::kExprF64AsmjsLoadMem:
|
|
|
|
return BuildAsmjsLoadMem(MachineType::Float64(), input);
|
2015-12-11 12:26:16 +00:00
|
|
|
default:
|
|
|
|
op = UnsupportedOpcode(opcode);
|
|
|
|
}
|
|
|
|
return graph()->NewNode(op, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Float32Constant(float value) {
|
|
|
|
return jsgraph()->Float32Constant(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Float64Constant(double value) {
|
|
|
|
return jsgraph()->Float64Constant(value);
|
|
|
|
}
|
|
|
|
|
2016-05-12 11:58:27 +00:00
|
|
|
Node* WasmGraphBuilder::HeapConstant(Handle<HeapObject> value) {
|
|
|
|
return jsgraph()->HeapConstant(value);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Branch(Node* cond, Node** true_node,
|
|
|
|
Node** false_node) {
|
|
|
|
DCHECK_NOT_NULL(cond);
|
|
|
|
DCHECK_NOT_NULL(*control_);
|
|
|
|
Node* branch =
|
|
|
|
graph()->NewNode(jsgraph()->common()->Branch(), cond, *control_);
|
|
|
|
*true_node = graph()->NewNode(jsgraph()->common()->IfTrue(), branch);
|
|
|
|
*false_node = graph()->NewNode(jsgraph()->common()->IfFalse(), branch);
|
|
|
|
return branch;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Switch(unsigned count, Node* key) {
|
|
|
|
return graph()->NewNode(jsgraph()->common()->Switch(count), key, *control_);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::IfValue(int32_t value, Node* sw) {
|
|
|
|
DCHECK_EQ(IrOpcode::kSwitch, sw->opcode());
|
|
|
|
return graph()->NewNode(jsgraph()->common()->IfValue(value), sw);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::IfDefault(Node* sw) {
|
|
|
|
DCHECK_EQ(IrOpcode::kSwitch, sw->opcode());
|
|
|
|
return graph()->NewNode(jsgraph()->common()->IfDefault(), sw);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Return(unsigned count, Node** vals) {
|
|
|
|
DCHECK_NOT_NULL(*control_);
|
|
|
|
DCHECK_NOT_NULL(*effect_);
|
|
|
|
|
|
|
|
if (count == 0) {
|
|
|
|
// Handle a return of void.
|
|
|
|
vals[0] = jsgraph()->Int32Constant(0);
|
|
|
|
count = 1;
|
|
|
|
}
|
|
|
|
|
2016-03-29 12:50:36 +00:00
|
|
|
Node** buf = Realloc(vals, count, count + 2);
|
2015-12-11 12:26:16 +00:00
|
|
|
buf[count] = *effect_;
|
|
|
|
buf[count + 1] = *control_;
|
|
|
|
Node* ret = graph()->NewNode(jsgraph()->common()->Return(), count + 2, vals);
|
|
|
|
|
|
|
|
MergeControlToEnd(jsgraph(), ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::ReturnVoid() { return Return(0, Buffer(0)); }
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::Unreachable(wasm::WasmCodePosition position) {
|
|
|
|
trap_->Unreachable(position);
|
2015-12-11 12:26:16 +00:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-03-30 12:38:47 +00:00
|
|
|
Node* WasmGraphBuilder::MaskShiftCount32(Node* node) {
|
|
|
|
static const int32_t kMask32 = 0x1f;
|
|
|
|
if (!jsgraph()->machine()->Word32ShiftIsSafe()) {
|
|
|
|
// Shifts by constants are so common we pattern-match them here.
|
|
|
|
Int32Matcher match(node);
|
|
|
|
if (match.HasValue()) {
|
|
|
|
int32_t masked = (match.Value() & kMask32);
|
|
|
|
if (match.Value() != masked) node = jsgraph()->Int32Constant(masked);
|
|
|
|
} else {
|
|
|
|
node = graph()->NewNode(jsgraph()->machine()->Word32And(), node,
|
|
|
|
jsgraph()->Int32Constant(kMask32));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::MaskShiftCount64(Node* node) {
|
|
|
|
static const int64_t kMask64 = 0x3f;
|
|
|
|
if (!jsgraph()->machine()->Word32ShiftIsSafe()) {
|
|
|
|
// Shifts by constants are so common we pattern-match them here.
|
|
|
|
Int64Matcher match(node);
|
|
|
|
if (match.HasValue()) {
|
|
|
|
int64_t masked = (match.Value() & kMask64);
|
|
|
|
if (match.Value() != masked) node = jsgraph()->Int64Constant(masked);
|
|
|
|
} else {
|
|
|
|
node = graph()->NewNode(jsgraph()->machine()->Word64And(), node,
|
|
|
|
jsgraph()->Int64Constant(kMask64));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return node;
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2015-12-16 11:44:46 +00:00
|
|
|
Node* WasmGraphBuilder::BuildF32Neg(Node* input) {
|
|
|
|
Node* result =
|
|
|
|
Unop(wasm::kExprF32ReinterpretI32,
|
|
|
|
Binop(wasm::kExprI32Xor, Unop(wasm::kExprI32ReinterpretF32, input),
|
|
|
|
jsgraph()->Int32Constant(0x80000000)));
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Neg(Node* input) {
|
|
|
|
#if WASM_64
|
|
|
|
Node* result =
|
|
|
|
Unop(wasm::kExprF64ReinterpretI64,
|
|
|
|
Binop(wasm::kExprI64Xor, Unop(wasm::kExprI64ReinterpretF64, input),
|
|
|
|
jsgraph()->Int64Constant(0x8000000000000000)));
|
|
|
|
|
|
|
|
return result;
|
|
|
|
#else
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
|
|
|
|
Node* old_high_word = graph()->NewNode(m->Float64ExtractHighWord32(), input);
|
|
|
|
Node* new_high_word = Binop(wasm::kExprI32Xor, old_high_word,
|
|
|
|
jsgraph()->Int32Constant(0x80000000));
|
|
|
|
|
|
|
|
return graph()->NewNode(m->Float64InsertHighWord32(), input, new_high_word);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* WasmGraphBuilder::BuildF32CopySign(Node* left, Node* right) {
|
|
|
|
Node* result = Unop(
|
|
|
|
wasm::kExprF32ReinterpretI32,
|
|
|
|
Binop(wasm::kExprI32Ior,
|
|
|
|
Binop(wasm::kExprI32And, Unop(wasm::kExprI32ReinterpretF32, left),
|
|
|
|
jsgraph()->Int32Constant(0x7fffffff)),
|
|
|
|
Binop(wasm::kExprI32And, Unop(wasm::kExprI32ReinterpretF32, right),
|
|
|
|
jsgraph()->Int32Constant(0x80000000))));
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64CopySign(Node* left, Node* right) {
|
|
|
|
#if WASM_64
|
|
|
|
Node* result = Unop(
|
|
|
|
wasm::kExprF64ReinterpretI64,
|
|
|
|
Binop(wasm::kExprI64Ior,
|
|
|
|
Binop(wasm::kExprI64And, Unop(wasm::kExprI64ReinterpretF64, left),
|
|
|
|
jsgraph()->Int64Constant(0x7fffffffffffffff)),
|
|
|
|
Binop(wasm::kExprI64And, Unop(wasm::kExprI64ReinterpretF64, right),
|
|
|
|
jsgraph()->Int64Constant(0x8000000000000000))));
|
|
|
|
|
|
|
|
return result;
|
|
|
|
#else
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
|
|
|
|
Node* high_word_left = graph()->NewNode(m->Float64ExtractHighWord32(), left);
|
|
|
|
Node* high_word_right =
|
|
|
|
graph()->NewNode(m->Float64ExtractHighWord32(), right);
|
|
|
|
|
|
|
|
Node* new_high_word =
|
|
|
|
Binop(wasm::kExprI32Ior, Binop(wasm::kExprI32And, high_word_left,
|
|
|
|
jsgraph()->Int32Constant(0x7fffffff)),
|
|
|
|
Binop(wasm::kExprI32And, high_word_right,
|
|
|
|
jsgraph()->Int32Constant(0x80000000)));
|
|
|
|
|
|
|
|
return graph()->NewNode(m->Float64InsertHighWord32(), left, new_high_word);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-12-11 16:39:54 +00:00
|
|
|
Node* WasmGraphBuilder::BuildF32Min(Node* left, Node* right) {
|
|
|
|
Diamond left_le_right(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF32Le, left, right));
|
|
|
|
|
|
|
|
Diamond right_lt_left(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF32Lt, right, left));
|
|
|
|
|
|
|
|
Diamond left_is_not_nan(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF32Eq, left, left));
|
|
|
|
|
|
|
|
return left_le_right.Phi(
|
|
|
|
wasm::kAstF32, left,
|
2016-02-05 14:59:40 +00:00
|
|
|
right_lt_left.Phi(
|
|
|
|
wasm::kAstF32, right,
|
|
|
|
left_is_not_nan.Phi(
|
|
|
|
wasm::kAstF32,
|
|
|
|
Binop(wasm::kExprF32Mul, right, Float32Constant(1.0)),
|
|
|
|
Binop(wasm::kExprF32Mul, left, Float32Constant(1.0)))));
|
2015-12-11 16:39:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF32Max(Node* left, Node* right) {
|
|
|
|
Diamond left_ge_right(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF32Ge, left, right));
|
|
|
|
|
|
|
|
Diamond right_gt_left(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF32Gt, right, left));
|
|
|
|
|
|
|
|
Diamond left_is_not_nan(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF32Eq, left, left));
|
|
|
|
|
|
|
|
return left_ge_right.Phi(
|
|
|
|
wasm::kAstF32, left,
|
2016-02-05 14:59:40 +00:00
|
|
|
right_gt_left.Phi(
|
|
|
|
wasm::kAstF32, right,
|
|
|
|
left_is_not_nan.Phi(
|
|
|
|
wasm::kAstF32,
|
|
|
|
Binop(wasm::kExprF32Mul, right, Float32Constant(1.0)),
|
|
|
|
Binop(wasm::kExprF32Mul, left, Float32Constant(1.0)))));
|
2015-12-11 16:39:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Min(Node* left, Node* right) {
|
|
|
|
Diamond left_le_right(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF64Le, left, right));
|
|
|
|
|
|
|
|
Diamond right_lt_left(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF64Lt, right, left));
|
|
|
|
|
|
|
|
Diamond left_is_not_nan(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF64Eq, left, left));
|
|
|
|
|
|
|
|
return left_le_right.Phi(
|
|
|
|
wasm::kAstF64, left,
|
2016-02-05 14:59:40 +00:00
|
|
|
right_lt_left.Phi(
|
|
|
|
wasm::kAstF64, right,
|
|
|
|
left_is_not_nan.Phi(
|
|
|
|
wasm::kAstF64,
|
|
|
|
Binop(wasm::kExprF64Mul, right, Float64Constant(1.0)),
|
|
|
|
Binop(wasm::kExprF64Mul, left, Float64Constant(1.0)))));
|
2015-12-11 16:39:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Max(Node* left, Node* right) {
|
|
|
|
Diamond left_ge_right(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF64Ge, left, right));
|
|
|
|
|
|
|
|
Diamond right_gt_left(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF64Lt, right, left));
|
|
|
|
|
|
|
|
Diamond left_is_not_nan(graph(), jsgraph()->common(),
|
|
|
|
Binop(wasm::kExprF64Eq, left, left));
|
|
|
|
|
|
|
|
return left_ge_right.Phi(
|
|
|
|
wasm::kAstF64, left,
|
2016-02-05 14:59:40 +00:00
|
|
|
right_gt_left.Phi(
|
|
|
|
wasm::kAstF64, right,
|
|
|
|
left_is_not_nan.Phi(
|
|
|
|
wasm::kAstF64,
|
|
|
|
Binop(wasm::kExprF64Mul, right, Float64Constant(1.0)),
|
|
|
|
Binop(wasm::kExprF64Mul, left, Float64Constant(1.0)))));
|
2015-12-11 16:39:54 +00:00
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32SConvertF32(Node* input,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-21 10:52:48 +00:00
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// Truncation of the input value is needed for the overflow check later.
|
|
|
|
Node* trunc = Unop(wasm::kExprF32Trunc, input);
|
[turbofan] Add the TruncateFloat32ToInt32 operator to turbofan.
The new operator converts a float32 input to int32 through truncation.
I provide implementations of the new operator for x64, ia32, arm,
arm64, mips, mips64, and x87. @v8-ppc-ports, can you please take care
of the ppc implementation?
R=titzer@chromium.org, v8-arm-ports@googlegroups.com, v8-mips-ports@googlegroups.com, weiliang.lin@intel.com
Review URL: https://codereview.chromium.org/1583323004
Cr-Commit-Position: refs/heads/master@{#33346}
2016-01-16 11:40:47 +00:00
|
|
|
Node* result = graph()->NewNode(m->TruncateFloat32ToInt32(), trunc);
|
2015-12-21 10:52:48 +00:00
|
|
|
|
|
|
|
// Convert the result back to f64. If we end up at a different value than the
|
|
|
|
// truncated input value, then there has been an overflow and we trap.
|
[turbofan] Add the TruncateFloat32ToInt32 operator to turbofan.
The new operator converts a float32 input to int32 through truncation.
I provide implementations of the new operator for x64, ia32, arm,
arm64, mips, mips64, and x87. @v8-ppc-ports, can you please take care
of the ppc implementation?
R=titzer@chromium.org, v8-arm-ports@googlegroups.com, v8-mips-ports@googlegroups.com, weiliang.lin@intel.com
Review URL: https://codereview.chromium.org/1583323004
Cr-Commit-Position: refs/heads/master@{#33346}
2016-01-16 11:40:47 +00:00
|
|
|
Node* check = Unop(wasm::kExprF32SConvertI32, result);
|
|
|
|
Node* overflow = Binop(wasm::kExprF32Ne, trunc, check);
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->AddTrapIfTrue(wasm::kTrapFloatUnrepresentable, overflow, position);
|
2015-12-21 10:52:48 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32SConvertF64(Node* input,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-21 10:52:48 +00:00
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// Truncation of the input value is needed for the overflow check later.
|
|
|
|
Node* trunc = Unop(wasm::kExprF64Trunc, input);
|
|
|
|
Node* result = graph()->NewNode(m->ChangeFloat64ToInt32(), trunc);
|
|
|
|
|
|
|
|
// Convert the result back to f64. If we end up at a different value than the
|
|
|
|
// truncated input value, then there has been an overflow and we trap.
|
|
|
|
Node* check = Unop(wasm::kExprF64SConvertI32, result);
|
|
|
|
Node* overflow = Binop(wasm::kExprF64Ne, trunc, check);
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->AddTrapIfTrue(wasm::kTrapFloatUnrepresentable, overflow, position);
|
2015-12-21 10:52:48 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32UConvertF32(Node* input,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-21 10:52:48 +00:00
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// Truncation of the input value is needed for the overflow check later.
|
|
|
|
Node* trunc = Unop(wasm::kExprF32Trunc, input);
|
2016-02-06 18:08:54 +00:00
|
|
|
Node* result = graph()->NewNode(m->TruncateFloat32ToUint32(), trunc);
|
2015-12-21 10:52:48 +00:00
|
|
|
|
2016-02-06 18:08:54 +00:00
|
|
|
// Convert the result back to f32. If we end up at a different value than the
|
2015-12-21 10:52:48 +00:00
|
|
|
// truncated input value, then there has been an overflow and we trap.
|
2016-02-06 18:08:54 +00:00
|
|
|
Node* check = Unop(wasm::kExprF32UConvertI32, result);
|
|
|
|
Node* overflow = Binop(wasm::kExprF32Ne, trunc, check);
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->AddTrapIfTrue(wasm::kTrapFloatUnrepresentable, overflow, position);
|
2015-12-21 10:52:48 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32UConvertF64(Node* input,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-21 10:52:48 +00:00
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// Truncation of the input value is needed for the overflow check later.
|
|
|
|
Node* trunc = Unop(wasm::kExprF64Trunc, input);
|
2016-03-31 18:38:16 +00:00
|
|
|
Node* result = graph()->NewNode(m->TruncateFloat64ToUint32(), trunc);
|
2015-12-21 10:52:48 +00:00
|
|
|
|
|
|
|
// Convert the result back to f64. If we end up at a different value than the
|
|
|
|
// truncated input value, then there has been an overflow and we trap.
|
|
|
|
Node* check = Unop(wasm::kExprF64UConvertI32, result);
|
|
|
|
Node* overflow = Binop(wasm::kExprF64Ne, trunc, check);
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->AddTrapIfTrue(wasm::kTrapFloatUnrepresentable, overflow, position);
|
2015-12-21 10:52:48 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2016-05-11 09:31:32 +00:00
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI32AsmjsSConvertF32(Node* input) {
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// asm.js must use the wacky JS semantics.
|
|
|
|
input = graph()->NewNode(m->ChangeFloat32ToFloat64(), input);
|
|
|
|
return graph()->NewNode(m->TruncateFloat64ToWord32(), input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI32AsmjsSConvertF64(Node* input) {
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// asm.js must use the wacky JS semantics.
|
|
|
|
return graph()->NewNode(m->TruncateFloat64ToWord32(), input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI32AsmjsUConvertF32(Node* input) {
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// asm.js must use the wacky JS semantics.
|
|
|
|
input = graph()->NewNode(m->ChangeFloat32ToFloat64(), input);
|
|
|
|
return graph()->NewNode(m->TruncateFloat64ToWord32(), input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI32AsmjsUConvertF64(Node* input) {
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// asm.js must use the wacky JS semantics.
|
|
|
|
return graph()->NewNode(m->TruncateFloat64ToWord32(), input);
|
|
|
|
}
|
2015-12-21 10:52:48 +00:00
|
|
|
|
2016-04-21 07:45:36 +00:00
|
|
|
Node* WasmGraphBuilder::BuildBitCountingCall(Node* input, ExternalReference ref,
|
|
|
|
MachineRepresentation input_type) {
|
|
|
|
Node* stack_slot_param =
|
|
|
|
graph()->NewNode(jsgraph()->machine()->StackSlot(input_type));
|
2015-12-21 10:52:48 +00:00
|
|
|
|
2016-04-21 07:45:36 +00:00
|
|
|
const Operator* store_op = jsgraph()->machine()->Store(
|
|
|
|
StoreRepresentation(input_type, kNoWriteBarrier));
|
|
|
|
*effect_ =
|
|
|
|
graph()->NewNode(store_op, stack_slot_param, jsgraph()->Int32Constant(0),
|
|
|
|
input, *effect_, *control_);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-04-21 07:45:36 +00:00
|
|
|
MachineSignature::Builder sig_builder(jsgraph()->zone(), 1, 1);
|
|
|
|
sig_builder.AddReturn(MachineType::Int32());
|
|
|
|
sig_builder.AddParam(MachineType::Pointer());
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-04-21 07:45:36 +00:00
|
|
|
Node* function = graph()->NewNode(jsgraph()->common()->ExternalConstant(ref));
|
|
|
|
Node* args[] = {function, stack_slot_param};
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-04-21 07:45:36 +00:00
|
|
|
return BuildCCall(sig_builder.Build(), args);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
2016-04-21 07:45:36 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32Ctz(Node* input) {
|
|
|
|
return BuildBitCountingCall(
|
|
|
|
input, ExternalReference::wasm_word32_ctz(jsgraph()->isolate()),
|
|
|
|
MachineRepresentation::kWord32);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI64Ctz(Node* input) {
|
2016-04-21 07:45:36 +00:00
|
|
|
return Unop(wasm::kExprI64UConvertI32,
|
|
|
|
BuildBitCountingCall(input, ExternalReference::wasm_word64_ctz(
|
|
|
|
jsgraph()->isolate()),
|
|
|
|
MachineRepresentation::kWord64));
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI32Popcnt(Node* input) {
|
2016-04-21 07:45:36 +00:00
|
|
|
return BuildBitCountingCall(
|
|
|
|
input, ExternalReference::wasm_word32_popcnt(jsgraph()->isolate()),
|
|
|
|
MachineRepresentation::kWord32);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI64Popcnt(Node* input) {
|
2016-04-21 07:45:36 +00:00
|
|
|
return Unop(wasm::kExprI64UConvertI32,
|
|
|
|
BuildBitCountingCall(input, ExternalReference::wasm_word64_popcnt(
|
|
|
|
jsgraph()->isolate()),
|
|
|
|
MachineRepresentation::kWord64));
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
2016-01-20 15:43:03 +00:00
|
|
|
Node* WasmGraphBuilder::BuildF32Trunc(Node* input) {
|
2016-02-02 12:26:38 +00:00
|
|
|
MachineType type = MachineType::Float32();
|
|
|
|
ExternalReference ref =
|
2016-03-04 09:50:48 +00:00
|
|
|
ExternalReference::wasm_f32_trunc(jsgraph()->isolate());
|
|
|
|
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
2016-02-02 12:26:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF32Floor(Node* input) {
|
|
|
|
MachineType type = MachineType::Float32();
|
|
|
|
ExternalReference ref =
|
2016-03-04 09:50:48 +00:00
|
|
|
ExternalReference::wasm_f32_floor(jsgraph()->isolate());
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
2016-02-02 12:26:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF32Ceil(Node* input) {
|
|
|
|
MachineType type = MachineType::Float32();
|
|
|
|
ExternalReference ref =
|
2016-03-04 09:50:48 +00:00
|
|
|
ExternalReference::wasm_f32_ceil(jsgraph()->isolate());
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
2016-02-02 12:26:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF32NearestInt(Node* input) {
|
|
|
|
MachineType type = MachineType::Float32();
|
|
|
|
ExternalReference ref =
|
2016-03-04 09:50:48 +00:00
|
|
|
ExternalReference::wasm_f32_nearest_int(jsgraph()->isolate());
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
2016-01-20 15:43:03 +00:00
|
|
|
}
|
|
|
|
|
2016-02-02 10:57:58 +00:00
|
|
|
Node* WasmGraphBuilder::BuildF64Trunc(Node* input) {
|
2016-02-02 12:26:38 +00:00
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
2016-03-04 09:50:48 +00:00
|
|
|
ExternalReference::wasm_f64_trunc(jsgraph()->isolate());
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
2016-02-02 12:26:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Floor(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
2016-03-04 09:50:48 +00:00
|
|
|
ExternalReference::wasm_f64_floor(jsgraph()->isolate());
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
2016-02-02 12:26:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Ceil(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
2016-03-04 09:50:48 +00:00
|
|
|
ExternalReference::wasm_f64_ceil(jsgraph()->isolate());
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
2016-02-02 12:26:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64NearestInt(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
2016-03-04 09:50:48 +00:00
|
|
|
ExternalReference::wasm_f64_nearest_int(jsgraph()->isolate());
|
2016-03-03 01:22:01 +00:00
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Acos(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_acos_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Asin(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_asin_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Atan(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_atan_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Cos(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_cos_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Sin(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_sin_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Tan(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_tan_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Exp(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_exp_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Log(Node* input) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_log_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildF64Atan2(Node* left, Node* right) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_atan2_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, left, right);
|
2016-02-02 12:26:38 +00:00
|
|
|
}
|
|
|
|
|
2016-03-03 01:22:01 +00:00
|
|
|
Node* WasmGraphBuilder::BuildF64Pow(Node* left, Node* right) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_pow_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, left, right);
|
|
|
|
}
|
2016-02-02 10:57:58 +00:00
|
|
|
|
2016-03-03 01:22:01 +00:00
|
|
|
Node* WasmGraphBuilder::BuildF64Mod(Node* left, Node* right) {
|
|
|
|
MachineType type = MachineType::Float64();
|
|
|
|
ExternalReference ref =
|
|
|
|
ExternalReference::f64_mod_wrapper_function(jsgraph()->isolate());
|
|
|
|
return BuildCFuncInstruction(ref, type, left, right);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildCFuncInstruction(ExternalReference ref,
|
|
|
|
MachineType type, Node* input0,
|
|
|
|
Node* input1) {
|
|
|
|
// We do truncation by calling a C function which calculates the result.
|
|
|
|
// The input is passed to the C function as a double*'s to avoid double
|
|
|
|
// parameters. For this we reserve slots on the stack, store the parameters
|
|
|
|
// in those slots, pass pointers to the slot to the C function,
|
|
|
|
// and after calling the C function we collect the return value from
|
|
|
|
// the stack slot.
|
|
|
|
|
|
|
|
Node* stack_slot_param0 =
|
2016-02-02 12:26:38 +00:00
|
|
|
graph()->NewNode(jsgraph()->machine()->StackSlot(type.representation()));
|
2016-02-02 10:57:58 +00:00
|
|
|
|
2016-03-03 01:22:01 +00:00
|
|
|
const Operator* store_op0 = jsgraph()->machine()->Store(
|
2016-02-02 12:26:38 +00:00
|
|
|
StoreRepresentation(type.representation(), kNoWriteBarrier));
|
2016-03-03 01:22:01 +00:00
|
|
|
*effect_ = graph()->NewNode(store_op0, stack_slot_param0,
|
|
|
|
jsgraph()->Int32Constant(0), input0, *effect_,
|
|
|
|
*control_);
|
2016-02-02 10:57:58 +00:00
|
|
|
|
2016-02-02 12:26:38 +00:00
|
|
|
Node* function = graph()->NewNode(jsgraph()->common()->ExternalConstant(ref));
|
2016-03-29 12:50:36 +00:00
|
|
|
Node** args = Buffer(5);
|
2016-03-03 01:22:01 +00:00
|
|
|
args[0] = function;
|
|
|
|
args[1] = stack_slot_param0;
|
|
|
|
int input_count = 1;
|
|
|
|
|
|
|
|
if (input1 != nullptr) {
|
|
|
|
Node* stack_slot_param1 = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->StackSlot(type.representation()));
|
|
|
|
const Operator* store_op1 = jsgraph()->machine()->Store(
|
|
|
|
StoreRepresentation(type.representation(), kNoWriteBarrier));
|
|
|
|
*effect_ = graph()->NewNode(store_op1, stack_slot_param1,
|
|
|
|
jsgraph()->Int32Constant(0), input1, *effect_,
|
|
|
|
*control_);
|
|
|
|
args[2] = stack_slot_param1;
|
|
|
|
++input_count;
|
|
|
|
}
|
2016-02-02 10:57:58 +00:00
|
|
|
|
2016-03-03 01:22:01 +00:00
|
|
|
Signature<MachineType>::Builder sig_builder(jsgraph()->zone(), 0,
|
|
|
|
input_count);
|
|
|
|
sig_builder.AddParam(MachineType::Pointer());
|
|
|
|
if (input1 != nullptr) {
|
|
|
|
sig_builder.AddParam(MachineType::Pointer());
|
|
|
|
}
|
2016-02-02 10:57:58 +00:00
|
|
|
BuildCCall(sig_builder.Build(), args);
|
|
|
|
|
2016-02-02 12:26:38 +00:00
|
|
|
const Operator* load_op = jsgraph()->machine()->Load(type);
|
2016-02-02 10:57:58 +00:00
|
|
|
|
|
|
|
Node* load =
|
2016-03-03 01:22:01 +00:00
|
|
|
graph()->NewNode(load_op, stack_slot_param0, jsgraph()->Int32Constant(0),
|
2016-02-02 10:57:58 +00:00
|
|
|
*effect_, *control_);
|
|
|
|
*effect_ = load;
|
|
|
|
return load;
|
|
|
|
}
|
|
|
|
|
2016-03-04 09:50:48 +00:00
|
|
|
Node* WasmGraphBuilder::BuildF32SConvertI64(Node* input) {
|
Fix conversion to float32, typing issue, split apart asm-wasm tests.
Add missing conversions from other types to f32 in fround.
Restrict fround() to only float, double, signed, unsigned (no unions / intish).
Restrict Bitwise operations to intish, particularly |0, when not applied to a foreign function.
Adding more exhaustive tests of stdlib Math, move to a separate file.
Adding tests of interesting values for the stdlib asm.js functions.
BUG= https://bugs.chromium.org/p/v8/issues/detail?id=4203
TEST=test-asm-validator,asm-wasm
R=titzer@chromium.org,rossberg@chromium.org
LOG=N
Review URL: https://codereview.chromium.org/1804243003
Cr-Commit-Position: refs/heads/master@{#34967}
2016-03-21 20:33:52 +00:00
|
|
|
// TODO(titzer/bradnelson): Check handlng of asm.js case.
|
2016-03-14 10:13:49 +00:00
|
|
|
return BuildIntToFloatConversionInstruction(
|
2016-03-04 09:50:48 +00:00
|
|
|
input, ExternalReference::wasm_int64_to_float32(jsgraph()->isolate()),
|
|
|
|
MachineRepresentation::kWord64, MachineType::Float32());
|
|
|
|
}
|
|
|
|
Node* WasmGraphBuilder::BuildF32UConvertI64(Node* input) {
|
Fix conversion to float32, typing issue, split apart asm-wasm tests.
Add missing conversions from other types to f32 in fround.
Restrict fround() to only float, double, signed, unsigned (no unions / intish).
Restrict Bitwise operations to intish, particularly |0, when not applied to a foreign function.
Adding more exhaustive tests of stdlib Math, move to a separate file.
Adding tests of interesting values for the stdlib asm.js functions.
BUG= https://bugs.chromium.org/p/v8/issues/detail?id=4203
TEST=test-asm-validator,asm-wasm
R=titzer@chromium.org,rossberg@chromium.org
LOG=N
Review URL: https://codereview.chromium.org/1804243003
Cr-Commit-Position: refs/heads/master@{#34967}
2016-03-21 20:33:52 +00:00
|
|
|
// TODO(titzer/bradnelson): Check handlng of asm.js case.
|
2016-03-14 10:13:49 +00:00
|
|
|
return BuildIntToFloatConversionInstruction(
|
2016-03-04 09:50:48 +00:00
|
|
|
input, ExternalReference::wasm_uint64_to_float32(jsgraph()->isolate()),
|
|
|
|
MachineRepresentation::kWord64, MachineType::Float32());
|
|
|
|
}
|
|
|
|
Node* WasmGraphBuilder::BuildF64SConvertI64(Node* input) {
|
2016-03-14 10:13:49 +00:00
|
|
|
return BuildIntToFloatConversionInstruction(
|
2016-03-04 09:50:48 +00:00
|
|
|
input, ExternalReference::wasm_int64_to_float64(jsgraph()->isolate()),
|
|
|
|
MachineRepresentation::kWord64, MachineType::Float64());
|
|
|
|
}
|
|
|
|
Node* WasmGraphBuilder::BuildF64UConvertI64(Node* input) {
|
2016-03-14 10:13:49 +00:00
|
|
|
return BuildIntToFloatConversionInstruction(
|
2016-03-04 09:50:48 +00:00
|
|
|
input, ExternalReference::wasm_uint64_to_float64(jsgraph()->isolate()),
|
|
|
|
MachineRepresentation::kWord64, MachineType::Float64());
|
|
|
|
}
|
2016-03-14 10:13:49 +00:00
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildIntToFloatConversionInstruction(
|
2016-03-04 09:50:48 +00:00
|
|
|
Node* input, ExternalReference ref,
|
|
|
|
MachineRepresentation parameter_representation,
|
|
|
|
const MachineType result_type) {
|
|
|
|
Node* stack_slot_param = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->StackSlot(parameter_representation));
|
|
|
|
Node* stack_slot_result = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->StackSlot(result_type.representation()));
|
|
|
|
const Operator* store_op = jsgraph()->machine()->Store(
|
|
|
|
StoreRepresentation(parameter_representation, kNoWriteBarrier));
|
|
|
|
*effect_ =
|
|
|
|
graph()->NewNode(store_op, stack_slot_param, jsgraph()->Int32Constant(0),
|
|
|
|
input, *effect_, *control_);
|
|
|
|
MachineSignature::Builder sig_builder(jsgraph()->zone(), 0, 2);
|
|
|
|
sig_builder.AddParam(MachineType::Pointer());
|
|
|
|
sig_builder.AddParam(MachineType::Pointer());
|
|
|
|
Node* function = graph()->NewNode(jsgraph()->common()->ExternalConstant(ref));
|
|
|
|
Node* args[] = {function, stack_slot_param, stack_slot_result};
|
|
|
|
BuildCCall(sig_builder.Build(), args);
|
|
|
|
const Operator* load_op = jsgraph()->machine()->Load(result_type);
|
|
|
|
Node* load =
|
|
|
|
graph()->NewNode(load_op, stack_slot_result, jsgraph()->Int32Constant(0),
|
|
|
|
*effect_, *control_);
|
2016-03-14 10:13:49 +00:00
|
|
|
*effect_ = load;
|
|
|
|
return load;
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI64SConvertF32(Node* input,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-14 10:13:49 +00:00
|
|
|
if (jsgraph()->machine()->Is32()) {
|
|
|
|
return BuildFloatToIntConversionInstruction(
|
|
|
|
input, ExternalReference::wasm_float32_to_int64(jsgraph()->isolate()),
|
2016-05-02 08:47:05 +00:00
|
|
|
MachineRepresentation::kFloat32, MachineType::Int64(), position);
|
2016-03-14 10:13:49 +00:00
|
|
|
} else {
|
|
|
|
Node* trunc = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->TryTruncateFloat32ToInt64(), input);
|
|
|
|
Node* result = graph()->NewNode(jsgraph()->common()->Projection(0), trunc);
|
|
|
|
Node* overflow =
|
|
|
|
graph()->NewNode(jsgraph()->common()->Projection(1), trunc);
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->ZeroCheck64(wasm::kTrapFloatUnrepresentable, overflow, position);
|
2016-03-14 10:13:49 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI64UConvertF32(Node* input,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-14 10:13:49 +00:00
|
|
|
if (jsgraph()->machine()->Is32()) {
|
|
|
|
return BuildFloatToIntConversionInstruction(
|
|
|
|
input, ExternalReference::wasm_float32_to_uint64(jsgraph()->isolate()),
|
2016-05-02 08:47:05 +00:00
|
|
|
MachineRepresentation::kFloat32, MachineType::Int64(), position);
|
2016-03-14 10:13:49 +00:00
|
|
|
} else {
|
|
|
|
Node* trunc = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->TryTruncateFloat32ToUint64(), input);
|
|
|
|
Node* result = graph()->NewNode(jsgraph()->common()->Projection(0), trunc);
|
|
|
|
Node* overflow =
|
|
|
|
graph()->NewNode(jsgraph()->common()->Projection(1), trunc);
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->ZeroCheck64(wasm::kTrapFloatUnrepresentable, overflow, position);
|
2016-03-14 10:13:49 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI64SConvertF64(Node* input,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-14 10:13:49 +00:00
|
|
|
if (jsgraph()->machine()->Is32()) {
|
|
|
|
return BuildFloatToIntConversionInstruction(
|
|
|
|
input, ExternalReference::wasm_float64_to_int64(jsgraph()->isolate()),
|
2016-05-02 08:47:05 +00:00
|
|
|
MachineRepresentation::kFloat64, MachineType::Int64(), position);
|
2016-03-14 10:13:49 +00:00
|
|
|
} else {
|
|
|
|
Node* trunc = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->TryTruncateFloat64ToInt64(), input);
|
|
|
|
Node* result = graph()->NewNode(jsgraph()->common()->Projection(0), trunc);
|
|
|
|
Node* overflow =
|
|
|
|
graph()->NewNode(jsgraph()->common()->Projection(1), trunc);
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->ZeroCheck64(wasm::kTrapFloatUnrepresentable, overflow, position);
|
2016-03-14 10:13:49 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI64UConvertF64(Node* input,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-14 10:13:49 +00:00
|
|
|
if (jsgraph()->machine()->Is32()) {
|
|
|
|
return BuildFloatToIntConversionInstruction(
|
|
|
|
input, ExternalReference::wasm_float64_to_uint64(jsgraph()->isolate()),
|
2016-05-02 08:47:05 +00:00
|
|
|
MachineRepresentation::kFloat64, MachineType::Int64(), position);
|
2016-03-14 10:13:49 +00:00
|
|
|
} else {
|
|
|
|
Node* trunc = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->TryTruncateFloat64ToUint64(), input);
|
|
|
|
Node* result = graph()->NewNode(jsgraph()->common()->Projection(0), trunc);
|
|
|
|
Node* overflow =
|
|
|
|
graph()->NewNode(jsgraph()->common()->Projection(1), trunc);
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->ZeroCheck64(wasm::kTrapFloatUnrepresentable, overflow, position);
|
2016-03-14 10:13:49 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildFloatToIntConversionInstruction(
|
|
|
|
Node* input, ExternalReference ref,
|
|
|
|
MachineRepresentation parameter_representation,
|
2016-05-02 08:47:05 +00:00
|
|
|
const MachineType result_type, wasm::WasmCodePosition position) {
|
2016-03-14 10:13:49 +00:00
|
|
|
Node* stack_slot_param = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->StackSlot(parameter_representation));
|
|
|
|
Node* stack_slot_result = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->StackSlot(result_type.representation()));
|
|
|
|
const Operator* store_op = jsgraph()->machine()->Store(
|
|
|
|
StoreRepresentation(parameter_representation, kNoWriteBarrier));
|
|
|
|
*effect_ =
|
|
|
|
graph()->NewNode(store_op, stack_slot_param, jsgraph()->Int32Constant(0),
|
|
|
|
input, *effect_, *control_);
|
|
|
|
MachineSignature::Builder sig_builder(jsgraph()->zone(), 1, 2);
|
|
|
|
sig_builder.AddReturn(MachineType::Int32());
|
|
|
|
sig_builder.AddParam(MachineType::Pointer());
|
|
|
|
sig_builder.AddParam(MachineType::Pointer());
|
|
|
|
Node* function = graph()->NewNode(jsgraph()->common()->ExternalConstant(ref));
|
|
|
|
Node* args[] = {function, stack_slot_param, stack_slot_result};
|
2016-04-05 17:28:45 +00:00
|
|
|
trap_->ZeroCheck32(wasm::kTrapFloatUnrepresentable,
|
2016-05-02 08:47:05 +00:00
|
|
|
BuildCCall(sig_builder.Build(), args), position);
|
2016-03-14 10:13:49 +00:00
|
|
|
const Operator* load_op = jsgraph()->machine()->Load(result_type);
|
|
|
|
Node* load =
|
|
|
|
graph()->NewNode(load_op, stack_slot_result, jsgraph()->Int32Constant(0),
|
|
|
|
*effect_, *control_);
|
2016-03-04 09:50:48 +00:00
|
|
|
*effect_ = load;
|
2016-03-15 06:51:53 +00:00
|
|
|
return load;
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32DivS(Node* left, Node* right,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-30 11:37:24 +00:00
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->ZeroCheck32(wasm::kTrapDivByZero, right, position);
|
2016-03-30 11:37:24 +00:00
|
|
|
Node* before = *control_;
|
|
|
|
Node* denom_is_m1;
|
|
|
|
Node* denom_is_not_m1;
|
|
|
|
Branch(
|
|
|
|
graph()->NewNode(m->Word32Equal(), right, jsgraph()->Int32Constant(-1)),
|
|
|
|
&denom_is_m1, &denom_is_not_m1);
|
|
|
|
*control_ = denom_is_m1;
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->TrapIfEq32(wasm::kTrapDivUnrepresentable, left, kMinInt, position);
|
2016-03-30 11:37:24 +00:00
|
|
|
if (*control_ != denom_is_m1) {
|
|
|
|
*control_ = graph()->NewNode(jsgraph()->common()->Merge(2), denom_is_not_m1,
|
|
|
|
*control_);
|
|
|
|
} else {
|
|
|
|
*control_ = before;
|
|
|
|
}
|
|
|
|
return graph()->NewNode(m->Int32Div(), left, right, *control_);
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32RemS(Node* left, Node* right,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-30 11:37:24 +00:00
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->ZeroCheck32(wasm::kTrapRemByZero, right, position);
|
2016-03-30 11:37:24 +00:00
|
|
|
|
|
|
|
Diamond d(
|
|
|
|
graph(), jsgraph()->common(),
|
|
|
|
graph()->NewNode(m->Word32Equal(), right, jsgraph()->Int32Constant(-1)),
|
|
|
|
BranchHint::kFalse);
|
|
|
|
d.Chain(*control_);
|
|
|
|
|
|
|
|
return d.Phi(MachineRepresentation::kWord32, jsgraph()->Int32Constant(0),
|
|
|
|
graph()->NewNode(m->Int32Mod(), left, right, d.if_false));
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32DivU(Node* left, Node* right,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-30 11:37:24 +00:00
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
2016-05-02 08:47:05 +00:00
|
|
|
return graph()->NewNode(
|
|
|
|
m->Uint32Div(), left, right,
|
|
|
|
trap_->ZeroCheck32(wasm::kTrapDivByZero, right, position));
|
2016-03-30 11:37:24 +00:00
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32RemU(Node* left, Node* right,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-30 11:37:24 +00:00
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
2016-05-02 08:47:05 +00:00
|
|
|
return graph()->NewNode(
|
|
|
|
m->Uint32Mod(), left, right,
|
|
|
|
trap_->ZeroCheck32(wasm::kTrapRemByZero, right, position));
|
2016-03-30 11:37:24 +00:00
|
|
|
}
|
|
|
|
|
2016-05-10 17:58:03 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32AsmjsDivS(Node* left, Node* right) {
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// asm.js semantics return 0 on divide or mod by zero.
|
|
|
|
if (m->Int32DivIsSafe()) {
|
|
|
|
// The hardware instruction does the right thing (e.g. arm).
|
|
|
|
return graph()->NewNode(m->Int32Div(), left, right, graph()->start());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check denominator for zero.
|
|
|
|
Diamond z(
|
|
|
|
graph(), jsgraph()->common(),
|
|
|
|
graph()->NewNode(m->Word32Equal(), right, jsgraph()->Int32Constant(0)),
|
|
|
|
BranchHint::kFalse);
|
|
|
|
|
|
|
|
// Check numerator for -1. (avoid minint / -1 case).
|
|
|
|
Diamond n(
|
|
|
|
graph(), jsgraph()->common(),
|
|
|
|
graph()->NewNode(m->Word32Equal(), right, jsgraph()->Int32Constant(-1)),
|
|
|
|
BranchHint::kFalse);
|
|
|
|
|
|
|
|
Node* div = graph()->NewNode(m->Int32Div(), left, right, z.if_false);
|
|
|
|
Node* neg =
|
|
|
|
graph()->NewNode(m->Int32Sub(), jsgraph()->Int32Constant(0), left);
|
|
|
|
|
|
|
|
return n.Phi(
|
|
|
|
MachineRepresentation::kWord32, neg,
|
|
|
|
z.Phi(MachineRepresentation::kWord32, jsgraph()->Int32Constant(0), div));
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI32AsmjsRemS(Node* left, Node* right) {
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// asm.js semantics return 0 on divide or mod by zero.
|
|
|
|
// Explicit check for x % 0.
|
|
|
|
Diamond z(
|
|
|
|
graph(), jsgraph()->common(),
|
|
|
|
graph()->NewNode(m->Word32Equal(), right, jsgraph()->Int32Constant(0)),
|
|
|
|
BranchHint::kFalse);
|
|
|
|
|
|
|
|
// Explicit check for x % -1.
|
|
|
|
Diamond d(
|
|
|
|
graph(), jsgraph()->common(),
|
|
|
|
graph()->NewNode(m->Word32Equal(), right, jsgraph()->Int32Constant(-1)),
|
|
|
|
BranchHint::kFalse);
|
|
|
|
d.Chain(z.if_false);
|
|
|
|
|
|
|
|
return z.Phi(
|
|
|
|
MachineRepresentation::kWord32, jsgraph()->Int32Constant(0),
|
|
|
|
d.Phi(MachineRepresentation::kWord32, jsgraph()->Int32Constant(0),
|
|
|
|
graph()->NewNode(m->Int32Mod(), left, right, d.if_false)));
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI32AsmjsDivU(Node* left, Node* right) {
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// asm.js semantics return 0 on divide or mod by zero.
|
|
|
|
if (m->Uint32DivIsSafe()) {
|
|
|
|
// The hardware instruction does the right thing (e.g. arm).
|
|
|
|
return graph()->NewNode(m->Uint32Div(), left, right, graph()->start());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Explicit check for x % 0.
|
|
|
|
Diamond z(
|
|
|
|
graph(), jsgraph()->common(),
|
|
|
|
graph()->NewNode(m->Word32Equal(), right, jsgraph()->Int32Constant(0)),
|
|
|
|
BranchHint::kFalse);
|
|
|
|
|
|
|
|
return z.Phi(MachineRepresentation::kWord32, jsgraph()->Int32Constant(0),
|
|
|
|
graph()->NewNode(jsgraph()->machine()->Uint32Div(), left, right,
|
|
|
|
z.if_false));
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI32AsmjsRemU(Node* left, Node* right) {
|
|
|
|
MachineOperatorBuilder* m = jsgraph()->machine();
|
|
|
|
// asm.js semantics return 0 on divide or mod by zero.
|
|
|
|
// Explicit check for x % 0.
|
|
|
|
Diamond z(
|
|
|
|
graph(), jsgraph()->common(),
|
|
|
|
graph()->NewNode(m->Word32Equal(), right, jsgraph()->Int32Constant(0)),
|
|
|
|
BranchHint::kFalse);
|
|
|
|
|
|
|
|
Node* rem = graph()->NewNode(jsgraph()->machine()->Uint32Mod(), left, right,
|
|
|
|
z.if_false);
|
|
|
|
return z.Phi(MachineRepresentation::kWord32, jsgraph()->Int32Constant(0),
|
|
|
|
rem);
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI64DivS(Node* left, Node* right,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-15 06:51:53 +00:00
|
|
|
if (jsgraph()->machine()->Is32()) {
|
|
|
|
return BuildDiv64Call(
|
|
|
|
left, right, ExternalReference::wasm_int64_div(jsgraph()->isolate()),
|
2016-05-02 08:47:05 +00:00
|
|
|
MachineType::Int64(), wasm::kTrapDivByZero, position);
|
2016-03-15 06:51:53 +00:00
|
|
|
}
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->ZeroCheck64(wasm::kTrapDivByZero, right, position);
|
2016-03-15 06:51:53 +00:00
|
|
|
Node* before = *control_;
|
|
|
|
Node* denom_is_m1;
|
|
|
|
Node* denom_is_not_m1;
|
|
|
|
Branch(graph()->NewNode(jsgraph()->machine()->Word64Equal(), right,
|
|
|
|
jsgraph()->Int64Constant(-1)),
|
|
|
|
&denom_is_m1, &denom_is_not_m1);
|
|
|
|
*control_ = denom_is_m1;
|
2016-04-05 17:28:45 +00:00
|
|
|
trap_->TrapIfEq64(wasm::kTrapDivUnrepresentable, left,
|
2016-05-02 08:47:05 +00:00
|
|
|
std::numeric_limits<int64_t>::min(), position);
|
2016-03-15 06:51:53 +00:00
|
|
|
if (*control_ != denom_is_m1) {
|
|
|
|
*control_ = graph()->NewNode(jsgraph()->common()->Merge(2), denom_is_not_m1,
|
|
|
|
*control_);
|
|
|
|
} else {
|
|
|
|
*control_ = before;
|
|
|
|
}
|
|
|
|
return graph()->NewNode(jsgraph()->machine()->Int64Div(), left, right,
|
|
|
|
*control_);
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI64RemS(Node* left, Node* right,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-15 06:51:53 +00:00
|
|
|
if (jsgraph()->machine()->Is32()) {
|
|
|
|
return BuildDiv64Call(
|
|
|
|
left, right, ExternalReference::wasm_int64_mod(jsgraph()->isolate()),
|
2016-05-02 08:47:05 +00:00
|
|
|
MachineType::Int64(), wasm::kTrapRemByZero, position);
|
2016-03-15 06:51:53 +00:00
|
|
|
}
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->ZeroCheck64(wasm::kTrapRemByZero, right, position);
|
2016-03-15 06:51:53 +00:00
|
|
|
Diamond d(jsgraph()->graph(), jsgraph()->common(),
|
|
|
|
graph()->NewNode(jsgraph()->machine()->Word64Equal(), right,
|
|
|
|
jsgraph()->Int64Constant(-1)));
|
|
|
|
|
|
|
|
Node* rem = graph()->NewNode(jsgraph()->machine()->Int64Mod(), left, right,
|
|
|
|
d.if_false);
|
|
|
|
|
|
|
|
return d.Phi(MachineRepresentation::kWord64, jsgraph()->Int64Constant(0),
|
|
|
|
rem);
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI64DivU(Node* left, Node* right,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-15 06:51:53 +00:00
|
|
|
if (jsgraph()->machine()->Is32()) {
|
|
|
|
return BuildDiv64Call(
|
|
|
|
left, right, ExternalReference::wasm_uint64_div(jsgraph()->isolate()),
|
2016-05-02 08:47:05 +00:00
|
|
|
MachineType::Int64(), wasm::kTrapDivByZero, position);
|
2016-03-15 06:51:53 +00:00
|
|
|
}
|
2016-05-02 08:47:05 +00:00
|
|
|
return graph()->NewNode(
|
|
|
|
jsgraph()->machine()->Uint64Div(), left, right,
|
|
|
|
trap_->ZeroCheck64(wasm::kTrapDivByZero, right, position));
|
2016-03-15 06:51:53 +00:00
|
|
|
}
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI64RemU(Node* left, Node* right,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-15 06:51:53 +00:00
|
|
|
if (jsgraph()->machine()->Is32()) {
|
|
|
|
return BuildDiv64Call(
|
|
|
|
left, right, ExternalReference::wasm_uint64_mod(jsgraph()->isolate()),
|
2016-05-02 08:47:05 +00:00
|
|
|
MachineType::Int64(), wasm::kTrapRemByZero, position);
|
2016-03-15 06:51:53 +00:00
|
|
|
}
|
2016-05-02 08:47:05 +00:00
|
|
|
return graph()->NewNode(
|
|
|
|
jsgraph()->machine()->Uint64Mod(), left, right,
|
|
|
|
trap_->ZeroCheck64(wasm::kTrapRemByZero, right, position));
|
2016-03-15 06:51:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildDiv64Call(Node* left, Node* right,
|
|
|
|
ExternalReference ref,
|
2016-05-02 08:47:05 +00:00
|
|
|
MachineType result_type, int trap_zero,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-03-15 06:51:53 +00:00
|
|
|
Node* stack_slot_dst = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->StackSlot(MachineRepresentation::kWord64));
|
|
|
|
Node* stack_slot_src = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->StackSlot(MachineRepresentation::kWord64));
|
|
|
|
|
|
|
|
const Operator* store_op = jsgraph()->machine()->Store(
|
|
|
|
StoreRepresentation(MachineRepresentation::kWord64, kNoWriteBarrier));
|
|
|
|
*effect_ =
|
|
|
|
graph()->NewNode(store_op, stack_slot_dst, jsgraph()->Int32Constant(0),
|
|
|
|
left, *effect_, *control_);
|
|
|
|
*effect_ =
|
|
|
|
graph()->NewNode(store_op, stack_slot_src, jsgraph()->Int32Constant(0),
|
|
|
|
right, *effect_, *control_);
|
|
|
|
|
|
|
|
MachineSignature::Builder sig_builder(jsgraph()->zone(), 1, 2);
|
|
|
|
sig_builder.AddReturn(MachineType::Int32());
|
|
|
|
sig_builder.AddParam(MachineType::Pointer());
|
|
|
|
sig_builder.AddParam(MachineType::Pointer());
|
|
|
|
|
|
|
|
Node* function = graph()->NewNode(jsgraph()->common()->ExternalConstant(ref));
|
|
|
|
Node* args[] = {function, stack_slot_dst, stack_slot_src};
|
|
|
|
|
|
|
|
Node* call = BuildCCall(sig_builder.Build(), args);
|
|
|
|
|
|
|
|
// TODO(wasm): This can get simpler if we have a specialized runtime call to
|
|
|
|
// throw WASM exceptions by trap code instead of by string.
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->ZeroCheck32(static_cast<wasm::TrapReason>(trap_zero), call, position);
|
|
|
|
trap_->TrapIfEq32(wasm::kTrapDivUnrepresentable, call, -1, position);
|
2016-03-15 06:51:53 +00:00
|
|
|
const Operator* load_op = jsgraph()->machine()->Load(result_type);
|
|
|
|
Node* load =
|
|
|
|
graph()->NewNode(load_op, stack_slot_dst, jsgraph()->Int32Constant(0),
|
|
|
|
*effect_, *control_);
|
|
|
|
*effect_ = load;
|
2016-03-04 09:50:48 +00:00
|
|
|
return load;
|
|
|
|
}
|
|
|
|
|
2016-02-02 10:57:58 +00:00
|
|
|
Node* WasmGraphBuilder::BuildCCall(MachineSignature* sig, Node** args) {
|
|
|
|
const size_t params = sig->parameter_count();
|
|
|
|
const size_t extra = 2; // effect and control inputs.
|
|
|
|
const size_t count = 1 + params + extra;
|
|
|
|
|
|
|
|
// Reallocate the buffer to make space for extra inputs.
|
2016-03-29 12:50:36 +00:00
|
|
|
args = Realloc(args, 1 + params, count);
|
2016-02-02 10:57:58 +00:00
|
|
|
|
|
|
|
// Add effect and control inputs.
|
|
|
|
args[params + 1] = *effect_;
|
|
|
|
args[params + 2] = *control_;
|
|
|
|
|
|
|
|
CallDescriptor* desc =
|
|
|
|
Linkage::GetSimplifiedCDescriptor(jsgraph()->zone(), sig);
|
|
|
|
|
|
|
|
const Operator* op = jsgraph()->common()->Call(desc);
|
|
|
|
Node* call = graph()->NewNode(op, static_cast<int>(count), args);
|
|
|
|
*effect_ = call;
|
|
|
|
return call;
|
|
|
|
}
|
2016-01-20 15:43:03 +00:00
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::BuildWasmCall(wasm::FunctionSig* sig, Node** args,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-11 12:26:16 +00:00
|
|
|
const size_t params = sig->parameter_count();
|
|
|
|
const size_t extra = 2; // effect and control inputs.
|
|
|
|
const size_t count = 1 + params + extra;
|
|
|
|
|
|
|
|
// Reallocate the buffer to make space for extra inputs.
|
2016-03-29 12:50:36 +00:00
|
|
|
args = Realloc(args, 1 + params, count);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
// Add effect and control inputs.
|
|
|
|
args[params + 1] = *effect_;
|
|
|
|
args[params + 2] = *control_;
|
|
|
|
|
2016-02-18 15:18:41 +00:00
|
|
|
CallDescriptor* descriptor =
|
2016-02-23 15:33:06 +00:00
|
|
|
wasm::ModuleEnv::GetWasmCallDescriptor(jsgraph()->zone(), sig);
|
2016-02-18 15:18:41 +00:00
|
|
|
const Operator* op = jsgraph()->common()->Call(descriptor);
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* call = graph()->NewNode(op, static_cast<int>(count), args);
|
2016-05-02 08:47:05 +00:00
|
|
|
SetSourcePosition(call, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
*effect_ = call;
|
|
|
|
return call;
|
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::CallDirect(uint32_t index, Node** args,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-11 12:26:16 +00:00
|
|
|
DCHECK_NULL(args[0]);
|
|
|
|
|
|
|
|
// Add code object as constant.
|
2016-05-12 11:58:27 +00:00
|
|
|
args[0] = HeapConstant(module_->GetFunctionCode(index));
|
2015-12-11 12:26:16 +00:00
|
|
|
wasm::FunctionSig* sig = module_->GetFunctionSignature(index);
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildWasmCall(sig, args, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::CallImport(uint32_t index, Node** args,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-02-19 14:58:25 +00:00
|
|
|
DCHECK_NULL(args[0]);
|
|
|
|
|
|
|
|
// Add code object as constant.
|
2016-05-12 11:58:27 +00:00
|
|
|
args[0] = HeapConstant(module_->GetImportCode(index));
|
2016-02-22 00:46:52 +00:00
|
|
|
wasm::FunctionSig* sig = module_->GetImportSignature(index);
|
2016-02-19 14:58:25 +00:00
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildWasmCall(sig, args, position);
|
2016-02-19 14:58:25 +00:00
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* WasmGraphBuilder::CallIndirect(uint32_t index, Node** args,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-11 12:26:16 +00:00
|
|
|
DCHECK_NOT_NULL(args[0]);
|
2016-01-28 16:56:54 +00:00
|
|
|
DCHECK(module_ && module_->instance);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
MachineOperatorBuilder* machine = jsgraph()->machine();
|
|
|
|
|
|
|
|
// Compute the code object by loading it from the function table.
|
|
|
|
Node* key = args[0];
|
|
|
|
|
|
|
|
// Bounds check the index.
|
|
|
|
int table_size = static_cast<int>(module_->FunctionTableSize());
|
2016-01-28 16:56:54 +00:00
|
|
|
if (table_size > 0) {
|
|
|
|
// Bounds check against the table size.
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* size = Int32Constant(static_cast<int>(table_size));
|
|
|
|
Node* in_bounds = graph()->NewNode(machine->Uint32LessThan(), key, size);
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->AddTrapIfFalse(wasm::kTrapFuncInvalid, in_bounds, position);
|
2016-01-28 16:56:54 +00:00
|
|
|
} else {
|
|
|
|
// No function table. Generate a trap and return a constant.
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->AddTrapIfFalse(wasm::kTrapFuncInvalid, Int32Constant(0), position);
|
2016-01-28 16:56:54 +00:00
|
|
|
return trap_->GetTrapValue(module_->GetSignature(index));
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
2016-01-28 16:56:54 +00:00
|
|
|
Node* table = FunctionTable();
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
// Load signature from the table and check.
|
|
|
|
// The table is a FixedArray; signatures are encoded as SMIs.
|
|
|
|
// [sig1, sig2, sig3, ...., code1, code2, code3 ...]
|
|
|
|
ElementAccess access = AccessBuilder::ForFixedArrayElement();
|
|
|
|
const int fixed_offset = access.header_size - access.tag();
|
|
|
|
{
|
|
|
|
Node* load_sig = graph()->NewNode(
|
|
|
|
machine->Load(MachineType::AnyTagged()), table,
|
|
|
|
graph()->NewNode(machine->Int32Add(),
|
|
|
|
graph()->NewNode(machine->Word32Shl(), key,
|
|
|
|
Int32Constant(kPointerSizeLog2)),
|
|
|
|
Int32Constant(fixed_offset)),
|
|
|
|
*effect_, *control_);
|
|
|
|
Node* sig_match = graph()->NewNode(machine->WordEqual(), load_sig,
|
|
|
|
jsgraph()->SmiConstant(index));
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->AddTrapIfFalse(wasm::kTrapFuncSigMismatch, sig_match, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Load code object from the table.
|
|
|
|
int offset = fixed_offset + kPointerSize * table_size;
|
|
|
|
Node* load_code = graph()->NewNode(
|
|
|
|
machine->Load(MachineType::AnyTagged()), table,
|
|
|
|
graph()->NewNode(machine->Int32Add(),
|
|
|
|
graph()->NewNode(machine->Word32Shl(), key,
|
|
|
|
Int32Constant(kPointerSizeLog2)),
|
|
|
|
Int32Constant(offset)),
|
|
|
|
*effect_, *control_);
|
|
|
|
|
|
|
|
args[0] = load_code;
|
|
|
|
wasm::FunctionSig* sig = module_->GetSignature(index);
|
2016-05-02 08:47:05 +00:00
|
|
|
return BuildWasmCall(sig, args, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
2016-04-20 11:43:46 +00:00
|
|
|
Node* WasmGraphBuilder::BuildI32Rol(Node* left, Node* right) {
|
|
|
|
// Implement Rol by Ror since TurboFan does not have Rol opcode.
|
|
|
|
// TODO(weiliang): support Word32Rol opcode in TurboFan.
|
|
|
|
Int32Matcher m(right);
|
|
|
|
if (m.HasValue()) {
|
|
|
|
return Binop(wasm::kExprI32Ror, left,
|
|
|
|
jsgraph()->Int32Constant(32 - m.Value()));
|
|
|
|
} else {
|
|
|
|
return Binop(wasm::kExprI32Ror, left,
|
|
|
|
Binop(wasm::kExprI32Sub, jsgraph()->Int32Constant(32), right));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildI64Rol(Node* left, Node* right) {
|
|
|
|
// Implement Rol by Ror since TurboFan does not have Rol opcode.
|
|
|
|
// TODO(weiliang): support Word64Rol opcode in TurboFan.
|
|
|
|
Int64Matcher m(right);
|
|
|
|
if (m.HasValue()) {
|
|
|
|
return Binop(wasm::kExprI64Ror, left,
|
|
|
|
jsgraph()->Int64Constant(64 - m.Value()));
|
|
|
|
} else {
|
|
|
|
return Binop(wasm::kExprI64Ror, left,
|
|
|
|
Binop(wasm::kExprI64Sub, jsgraph()->Int64Constant(64), right));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::Invert(Node* node) {
|
|
|
|
return Unop(wasm::kExprI32Eqz, node);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildChangeInt32ToTagged(Node* value) {
|
|
|
|
MachineOperatorBuilder* machine = jsgraph()->machine();
|
|
|
|
CommonOperatorBuilder* common = jsgraph()->common();
|
|
|
|
|
|
|
|
if (machine->Is64()) {
|
|
|
|
return BuildChangeInt32ToSmi(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* add = graph()->NewNode(machine->Int32AddWithOverflow(), value, value);
|
|
|
|
|
|
|
|
Node* ovf = graph()->NewNode(common->Projection(1), add);
|
|
|
|
Node* branch = graph()->NewNode(common->Branch(BranchHint::kFalse), ovf,
|
|
|
|
graph()->start());
|
|
|
|
|
|
|
|
Node* if_true = graph()->NewNode(common->IfTrue(), branch);
|
|
|
|
Node* vtrue = BuildAllocateHeapNumberWithValue(
|
|
|
|
graph()->NewNode(machine->ChangeInt32ToFloat64(), value), if_true);
|
|
|
|
|
|
|
|
Node* if_false = graph()->NewNode(common->IfFalse(), branch);
|
|
|
|
Node* vfalse = graph()->NewNode(common->Projection(0), add);
|
|
|
|
|
|
|
|
Node* merge = graph()->NewNode(common->Merge(2), if_true, if_false);
|
|
|
|
Node* phi = graph()->NewNode(common->Phi(MachineRepresentation::kTagged, 2),
|
|
|
|
vtrue, vfalse, merge);
|
|
|
|
return phi;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildChangeFloat64ToTagged(Node* value) {
|
|
|
|
MachineOperatorBuilder* machine = jsgraph()->machine();
|
|
|
|
CommonOperatorBuilder* common = jsgraph()->common();
|
|
|
|
|
2016-04-24 11:39:31 +00:00
|
|
|
Node* value32 = graph()->NewNode(machine->RoundFloat64ToInt32(), value);
|
2016-04-20 11:43:46 +00:00
|
|
|
Node* check_same = graph()->NewNode(
|
|
|
|
machine->Float64Equal(), value,
|
|
|
|
graph()->NewNode(machine->ChangeInt32ToFloat64(), value32));
|
|
|
|
Node* branch_same =
|
|
|
|
graph()->NewNode(common->Branch(), check_same, graph()->start());
|
|
|
|
|
|
|
|
Node* if_smi = graph()->NewNode(common->IfTrue(), branch_same);
|
|
|
|
Node* vsmi;
|
|
|
|
Node* if_box = graph()->NewNode(common->IfFalse(), branch_same);
|
|
|
|
Node* vbox;
|
|
|
|
|
|
|
|
// We only need to check for -0 if the {value} can potentially contain -0.
|
|
|
|
Node* check_zero = graph()->NewNode(machine->Word32Equal(), value32,
|
|
|
|
jsgraph()->Int32Constant(0));
|
|
|
|
Node* branch_zero =
|
|
|
|
graph()->NewNode(common->Branch(BranchHint::kFalse), check_zero, if_smi);
|
|
|
|
|
|
|
|
Node* if_zero = graph()->NewNode(common->IfTrue(), branch_zero);
|
|
|
|
Node* if_notzero = graph()->NewNode(common->IfFalse(), branch_zero);
|
|
|
|
|
|
|
|
// In case of 0, we need to check the high bits for the IEEE -0 pattern.
|
|
|
|
Node* check_negative = graph()->NewNode(
|
|
|
|
machine->Int32LessThan(),
|
|
|
|
graph()->NewNode(machine->Float64ExtractHighWord32(), value),
|
|
|
|
jsgraph()->Int32Constant(0));
|
|
|
|
Node* branch_negative = graph()->NewNode(common->Branch(BranchHint::kFalse),
|
|
|
|
check_negative, if_zero);
|
|
|
|
|
|
|
|
Node* if_negative = graph()->NewNode(common->IfTrue(), branch_negative);
|
|
|
|
Node* if_notnegative = graph()->NewNode(common->IfFalse(), branch_negative);
|
|
|
|
|
|
|
|
// We need to create a box for negative 0.
|
|
|
|
if_smi = graph()->NewNode(common->Merge(2), if_notzero, if_notnegative);
|
|
|
|
if_box = graph()->NewNode(common->Merge(2), if_box, if_negative);
|
|
|
|
|
|
|
|
// On 64-bit machines we can just wrap the 32-bit integer in a smi, for 32-bit
|
|
|
|
// machines we need to deal with potential overflow and fallback to boxing.
|
|
|
|
if (machine->Is64()) {
|
|
|
|
vsmi = BuildChangeInt32ToSmi(value32);
|
|
|
|
} else {
|
|
|
|
Node* smi_tag =
|
|
|
|
graph()->NewNode(machine->Int32AddWithOverflow(), value32, value32);
|
|
|
|
|
|
|
|
Node* check_ovf = graph()->NewNode(common->Projection(1), smi_tag);
|
|
|
|
Node* branch_ovf =
|
|
|
|
graph()->NewNode(common->Branch(BranchHint::kFalse), check_ovf, if_smi);
|
|
|
|
|
|
|
|
Node* if_ovf = graph()->NewNode(common->IfTrue(), branch_ovf);
|
|
|
|
if_box = graph()->NewNode(common->Merge(2), if_ovf, if_box);
|
|
|
|
|
|
|
|
if_smi = graph()->NewNode(common->IfFalse(), branch_ovf);
|
|
|
|
vsmi = graph()->NewNode(common->Projection(0), smi_tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate the box for the {value}.
|
|
|
|
vbox = BuildAllocateHeapNumberWithValue(value, if_box);
|
|
|
|
|
|
|
|
Node* control = graph()->NewNode(common->Merge(2), if_smi, if_box);
|
|
|
|
value = graph()->NewNode(common->Phi(MachineRepresentation::kTagged, 2), vsmi,
|
|
|
|
vbox, control);
|
|
|
|
return value;
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
Node* WasmGraphBuilder::ToJS(Node* node, Node* context, wasm::LocalType type) {
|
|
|
|
switch (type) {
|
|
|
|
case wasm::kAstI32:
|
2016-04-20 11:43:46 +00:00
|
|
|
return BuildChangeInt32ToTagged(node);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kAstI64:
|
|
|
|
// TODO(titzer): i64->JS has no good solution right now. Using lower 32
|
|
|
|
// bits.
|
2016-04-20 14:44:15 +00:00
|
|
|
if (jsgraph()->machine()->Is64()) {
|
|
|
|
// On 32 bit platforms we do not have to do the truncation because the
|
|
|
|
// node we get in as a parameter only contains the low word anyways.
|
|
|
|
node = graph()->NewNode(jsgraph()->machine()->TruncateInt64ToInt32(),
|
|
|
|
node);
|
|
|
|
}
|
2016-04-20 11:43:46 +00:00
|
|
|
return BuildChangeInt32ToTagged(node);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kAstF32:
|
|
|
|
node = graph()->NewNode(jsgraph()->machine()->ChangeFloat32ToFloat64(),
|
|
|
|
node);
|
2016-04-20 11:43:46 +00:00
|
|
|
return BuildChangeFloat64ToTagged(node);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kAstF64:
|
2016-04-20 11:43:46 +00:00
|
|
|
return BuildChangeFloat64ToTagged(node);
|
2015-12-11 12:26:16 +00:00
|
|
|
case wasm::kAstStmt:
|
|
|
|
return jsgraph()->UndefinedConstant();
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-14 10:33:33 +00:00
|
|
|
Node* WasmGraphBuilder::BuildJavaScriptToNumber(Node* node, Node* context,
|
|
|
|
Node* effect, Node* control) {
|
|
|
|
Callable callable = CodeFactory::ToNumber(jsgraph()->isolate());
|
|
|
|
CallDescriptor* desc = Linkage::GetStubCallDescriptor(
|
|
|
|
jsgraph()->isolate(), jsgraph()->zone(), callable.descriptor(), 0,
|
2016-04-18 11:10:48 +00:00
|
|
|
CallDescriptor::kNoFlags, Operator::kNoProperties);
|
2016-04-14 10:33:33 +00:00
|
|
|
Node* stub_code = jsgraph()->HeapConstant(callable.code());
|
|
|
|
|
2016-04-18 11:10:48 +00:00
|
|
|
Node* result = graph()->NewNode(jsgraph()->common()->Call(desc), stub_code,
|
|
|
|
node, context, effect, control);
|
2016-04-14 10:33:33 +00:00
|
|
|
|
|
|
|
*control_ = result;
|
|
|
|
*effect_ = result;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-04-20 11:43:46 +00:00
|
|
|
bool CanCover(Node* value, IrOpcode::Value opcode) {
|
|
|
|
if (value->opcode() != opcode) return false;
|
|
|
|
bool first = true;
|
|
|
|
for (Edge const edge : value->use_edges()) {
|
|
|
|
if (NodeProperties::IsControlEdge(edge)) continue;
|
|
|
|
if (NodeProperties::IsEffectEdge(edge)) continue;
|
|
|
|
DCHECK(NodeProperties::IsValueEdge(edge));
|
|
|
|
if (!first) return false;
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildChangeTaggedToFloat64(Node* value) {
|
|
|
|
MachineOperatorBuilder* machine = jsgraph()->machine();
|
|
|
|
CommonOperatorBuilder* common = jsgraph()->common();
|
|
|
|
|
|
|
|
if (CanCover(value, IrOpcode::kJSToNumber)) {
|
|
|
|
// ChangeTaggedToFloat64(JSToNumber(x)) =>
|
|
|
|
// if IsSmi(x) then ChangeSmiToFloat64(x)
|
|
|
|
// else let y = JSToNumber(x) in
|
|
|
|
// if IsSmi(y) then ChangeSmiToFloat64(y)
|
|
|
|
// else BuildLoadHeapNumberValue(y)
|
|
|
|
Node* object = NodeProperties::GetValueInput(value, 0);
|
|
|
|
Node* context = NodeProperties::GetContextInput(value);
|
|
|
|
Node* frame_state = NodeProperties::GetFrameStateInput(value, 0);
|
|
|
|
Node* effect = NodeProperties::GetEffectInput(value);
|
|
|
|
Node* control = NodeProperties::GetControlInput(value);
|
|
|
|
|
|
|
|
const Operator* merge_op = common->Merge(2);
|
|
|
|
const Operator* ephi_op = common->EffectPhi(2);
|
|
|
|
const Operator* phi_op = common->Phi(MachineRepresentation::kFloat64, 2);
|
|
|
|
|
|
|
|
Node* check1 = BuildTestNotSmi(object);
|
|
|
|
Node* branch1 =
|
|
|
|
graph()->NewNode(common->Branch(BranchHint::kFalse), check1, control);
|
|
|
|
|
|
|
|
Node* if_true1 = graph()->NewNode(common->IfTrue(), branch1);
|
|
|
|
Node* vtrue1 = graph()->NewNode(value->op(), object, context, frame_state,
|
|
|
|
effect, if_true1);
|
|
|
|
Node* etrue1 = vtrue1;
|
|
|
|
|
|
|
|
Node* check2 = BuildTestNotSmi(vtrue1);
|
|
|
|
Node* branch2 = graph()->NewNode(common->Branch(), check2, if_true1);
|
|
|
|
|
|
|
|
Node* if_true2 = graph()->NewNode(common->IfTrue(), branch2);
|
|
|
|
Node* vtrue2 = BuildLoadHeapNumberValue(vtrue1, if_true2);
|
|
|
|
|
|
|
|
Node* if_false2 = graph()->NewNode(common->IfFalse(), branch2);
|
|
|
|
Node* vfalse2 = BuildChangeSmiToFloat64(vtrue1);
|
|
|
|
|
|
|
|
if_true1 = graph()->NewNode(merge_op, if_true2, if_false2);
|
|
|
|
vtrue1 = graph()->NewNode(phi_op, vtrue2, vfalse2, if_true1);
|
|
|
|
|
|
|
|
Node* if_false1 = graph()->NewNode(common->IfFalse(), branch1);
|
|
|
|
Node* vfalse1 = BuildChangeSmiToFloat64(object);
|
|
|
|
Node* efalse1 = effect;
|
|
|
|
|
|
|
|
Node* merge1 = graph()->NewNode(merge_op, if_true1, if_false1);
|
|
|
|
Node* ephi1 = graph()->NewNode(ephi_op, etrue1, efalse1, merge1);
|
|
|
|
Node* phi1 = graph()->NewNode(phi_op, vtrue1, vfalse1, merge1);
|
|
|
|
|
|
|
|
// Wire the new diamond into the graph, {JSToNumber} can still throw.
|
|
|
|
NodeProperties::ReplaceUses(value, phi1, ephi1, etrue1, etrue1);
|
|
|
|
|
|
|
|
// TODO(mstarzinger): This iteration cuts out the IfSuccess projection from
|
|
|
|
// the node and places it inside the diamond. Come up with a helper method!
|
|
|
|
for (Node* use : etrue1->uses()) {
|
|
|
|
if (use->opcode() == IrOpcode::kIfSuccess) {
|
|
|
|
use->ReplaceUses(merge1);
|
|
|
|
NodeProperties::ReplaceControlInput(branch2, use);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return phi1;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* check = BuildTestNotSmi(value);
|
|
|
|
Node* branch = graph()->NewNode(common->Branch(BranchHint::kFalse), check,
|
|
|
|
graph()->start());
|
|
|
|
|
|
|
|
Node* if_not_smi = graph()->NewNode(common->IfTrue(), branch);
|
|
|
|
|
|
|
|
Node* vnot_smi;
|
|
|
|
Node* check_undefined = graph()->NewNode(machine->WordEqual(), value,
|
|
|
|
jsgraph()->UndefinedConstant());
|
|
|
|
Node* branch_undefined = graph()->NewNode(common->Branch(BranchHint::kFalse),
|
|
|
|
check_undefined, if_not_smi);
|
|
|
|
|
|
|
|
Node* if_undefined = graph()->NewNode(common->IfTrue(), branch_undefined);
|
|
|
|
Node* vundefined =
|
|
|
|
jsgraph()->Float64Constant(std::numeric_limits<double>::quiet_NaN());
|
|
|
|
|
|
|
|
Node* if_not_undefined =
|
|
|
|
graph()->NewNode(common->IfFalse(), branch_undefined);
|
|
|
|
Node* vheap_number = BuildLoadHeapNumberValue(value, if_not_undefined);
|
|
|
|
|
|
|
|
if_not_smi =
|
|
|
|
graph()->NewNode(common->Merge(2), if_undefined, if_not_undefined);
|
|
|
|
vnot_smi = graph()->NewNode(common->Phi(MachineRepresentation::kFloat64, 2),
|
|
|
|
vundefined, vheap_number, if_not_smi);
|
|
|
|
|
|
|
|
Node* if_smi = graph()->NewNode(common->IfFalse(), branch);
|
|
|
|
Node* vfrom_smi = BuildChangeSmiToFloat64(value);
|
|
|
|
|
|
|
|
Node* merge = graph()->NewNode(common->Merge(2), if_not_smi, if_smi);
|
|
|
|
Node* phi = graph()->NewNode(common->Phi(MachineRepresentation::kFloat64, 2),
|
|
|
|
vnot_smi, vfrom_smi, merge);
|
|
|
|
|
|
|
|
return phi;
|
|
|
|
}
|
|
|
|
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* WasmGraphBuilder::FromJS(Node* node, Node* context,
|
|
|
|
wasm::LocalType type) {
|
|
|
|
// Do a JavaScript ToNumber.
|
2016-04-14 10:33:33 +00:00
|
|
|
Node* num = BuildJavaScriptToNumber(node, context, *effect_, *control_);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
// Change representation.
|
|
|
|
SimplifiedOperatorBuilder simplified(jsgraph()->zone());
|
2016-04-20 11:43:46 +00:00
|
|
|
num = BuildChangeTaggedToFloat64(num);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case wasm::kAstI32: {
|
2016-04-24 11:39:31 +00:00
|
|
|
num = graph()->NewNode(jsgraph()->machine()->TruncateFloat64ToWord32(),
|
2015-12-11 12:26:16 +00:00
|
|
|
num);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case wasm::kAstI64:
|
|
|
|
// TODO(titzer): JS->i64 has no good solution right now. Using 32 bits.
|
2016-04-24 11:39:31 +00:00
|
|
|
num = graph()->NewNode(jsgraph()->machine()->TruncateFloat64ToWord32(),
|
2015-12-11 12:26:16 +00:00
|
|
|
num);
|
2016-04-20 14:44:15 +00:00
|
|
|
if (jsgraph()->machine()->Is64()) {
|
|
|
|
// We cannot change an int32 to an int64 on a 32 bit platform. Instead
|
|
|
|
// we will split the parameter node later.
|
|
|
|
num = graph()->NewNode(jsgraph()->machine()->ChangeInt32ToInt64(), num);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
break;
|
|
|
|
case wasm::kAstF32:
|
|
|
|
num = graph()->NewNode(jsgraph()->machine()->TruncateFloat64ToFloat32(),
|
|
|
|
num);
|
|
|
|
break;
|
|
|
|
case wasm::kAstF64:
|
|
|
|
break;
|
|
|
|
case wasm::kAstStmt:
|
|
|
|
num = jsgraph()->Int32Constant(0);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return num;
|
|
|
|
}
|
|
|
|
|
2016-04-20 11:43:46 +00:00
|
|
|
Node* WasmGraphBuilder::BuildChangeInt32ToSmi(Node* value) {
|
|
|
|
if (jsgraph()->machine()->Is64()) {
|
|
|
|
value = graph()->NewNode(jsgraph()->machine()->ChangeInt32ToInt64(), value);
|
2016-03-03 05:10:15 +00:00
|
|
|
}
|
2016-04-20 11:43:46 +00:00
|
|
|
return graph()->NewNode(jsgraph()->machine()->WordShl(), value,
|
|
|
|
BuildSmiShiftBitsConstant());
|
2016-03-03 05:10:15 +00:00
|
|
|
}
|
|
|
|
|
2016-04-20 11:43:46 +00:00
|
|
|
Node* WasmGraphBuilder::BuildChangeSmiToInt32(Node* value) {
|
|
|
|
value = graph()->NewNode(jsgraph()->machine()->WordSar(), value,
|
|
|
|
BuildSmiShiftBitsConstant());
|
|
|
|
if (jsgraph()->machine()->Is64()) {
|
|
|
|
value =
|
|
|
|
graph()->NewNode(jsgraph()->machine()->TruncateInt64ToInt32(), value);
|
2016-03-03 05:10:15 +00:00
|
|
|
}
|
2016-04-20 11:43:46 +00:00
|
|
|
return value;
|
2016-03-03 05:10:15 +00:00
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-04-20 11:43:46 +00:00
|
|
|
Node* WasmGraphBuilder::BuildChangeSmiToFloat64(Node* value) {
|
|
|
|
return graph()->NewNode(jsgraph()->machine()->ChangeInt32ToFloat64(),
|
|
|
|
BuildChangeSmiToInt32(value));
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
2016-04-20 11:43:46 +00:00
|
|
|
Node* WasmGraphBuilder::BuildTestNotSmi(Node* value) {
|
|
|
|
STATIC_ASSERT(kSmiTag == 0);
|
|
|
|
STATIC_ASSERT(kSmiTagMask == 1);
|
|
|
|
return graph()->NewNode(jsgraph()->machine()->WordAnd(), value,
|
|
|
|
jsgraph()->IntPtrConstant(kSmiTagMask));
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildSmiShiftBitsConstant() {
|
|
|
|
return jsgraph()->IntPtrConstant(kSmiShiftSize + kSmiTagSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildAllocateHeapNumberWithValue(Node* value,
|
|
|
|
Node* control) {
|
|
|
|
MachineOperatorBuilder* machine = jsgraph()->machine();
|
|
|
|
CommonOperatorBuilder* common = jsgraph()->common();
|
|
|
|
// The AllocateHeapNumberStub does not use the context, so we can safely pass
|
|
|
|
// in Smi zero here.
|
|
|
|
Callable callable = CodeFactory::AllocateHeapNumber(jsgraph()->isolate());
|
|
|
|
Node* target = jsgraph()->HeapConstant(callable.code());
|
|
|
|
Node* context = jsgraph()->NoContextConstant();
|
|
|
|
Node* effect = graph()->NewNode(common->BeginRegion(), graph()->start());
|
|
|
|
if (!allocate_heap_number_operator_.is_set()) {
|
|
|
|
CallDescriptor* descriptor = Linkage::GetStubCallDescriptor(
|
|
|
|
jsgraph()->isolate(), jsgraph()->zone(), callable.descriptor(), 0,
|
|
|
|
CallDescriptor::kNoFlags, Operator::kNoThrow);
|
|
|
|
allocate_heap_number_operator_.set(common->Call(descriptor));
|
|
|
|
}
|
|
|
|
Node* heap_number = graph()->NewNode(allocate_heap_number_operator_.get(),
|
|
|
|
target, context, effect, control);
|
|
|
|
Node* store =
|
|
|
|
graph()->NewNode(machine->Store(StoreRepresentation(
|
|
|
|
MachineRepresentation::kFloat64, kNoWriteBarrier)),
|
|
|
|
heap_number, BuildHeapNumberValueIndexConstant(), value,
|
|
|
|
heap_number, control);
|
|
|
|
return graph()->NewNode(common->FinishRegion(), heap_number, store);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildLoadHeapNumberValue(Node* value, Node* control) {
|
|
|
|
return graph()->NewNode(jsgraph()->machine()->Load(MachineType::Float64()),
|
|
|
|
value, BuildHeapNumberValueIndexConstant(),
|
|
|
|
graph()->start(), control);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildHeapNumberValueIndexConstant() {
|
|
|
|
return jsgraph()->IntPtrConstant(HeapNumber::kValueOffset - kHeapObjectTag);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
void WasmGraphBuilder::BuildJSToWasmWrapper(Handle<Code> wasm_code,
|
|
|
|
wasm::FunctionSig* sig) {
|
2016-04-20 14:44:15 +00:00
|
|
|
int wasm_count = static_cast<int>(sig->parameter_count());
|
|
|
|
int param_count;
|
|
|
|
if (jsgraph()->machine()->Is64()) {
|
|
|
|
param_count = static_cast<int>(sig->parameter_count());
|
|
|
|
} else {
|
|
|
|
param_count = Int64Lowering::GetParameterCountAfterLowering(sig);
|
|
|
|
}
|
|
|
|
int count = param_count + 3;
|
2015-12-11 12:26:16 +00:00
|
|
|
Node** args = Buffer(count);
|
|
|
|
|
|
|
|
// Build the start and the JS parameter nodes.
|
2016-04-20 14:44:15 +00:00
|
|
|
Node* start = Start(param_count + 5);
|
2015-12-11 12:26:16 +00:00
|
|
|
*control_ = start;
|
|
|
|
*effect_ = start;
|
2016-03-24 13:23:54 +00:00
|
|
|
// Create the context parameter
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* context = graph()->NewNode(
|
2016-03-24 13:23:54 +00:00
|
|
|
jsgraph()->common()->Parameter(
|
2016-04-20 14:44:15 +00:00
|
|
|
Linkage::GetJSCallContextParamIndex(wasm_count + 1), "%context"),
|
2016-03-24 13:23:54 +00:00
|
|
|
graph()->start());
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
int pos = 0;
|
2016-05-12 11:58:27 +00:00
|
|
|
args[pos++] = HeapConstant(wasm_code);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
// Convert JS parameters to WASM numbers.
|
2016-04-20 14:44:15 +00:00
|
|
|
for (int i = 0; i < wasm_count; i++) {
|
2016-03-24 13:23:54 +00:00
|
|
|
Node* param =
|
|
|
|
graph()->NewNode(jsgraph()->common()->Parameter(i + 1), start);
|
2016-04-20 14:44:15 +00:00
|
|
|
Node* wasm_param = FromJS(param, context, sig->GetParam(i));
|
|
|
|
args[pos++] = wasm_param;
|
|
|
|
if (jsgraph()->machine()->Is32() && sig->GetParam(i) == wasm::kAstI64) {
|
|
|
|
// We make up the high word with SAR to get the proper sign extension.
|
|
|
|
args[pos++] = graph()->NewNode(jsgraph()->machine()->Word32Sar(),
|
|
|
|
wasm_param, jsgraph()->Int32Constant(31));
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
args[pos++] = *effect_;
|
|
|
|
args[pos++] = *control_;
|
|
|
|
|
|
|
|
// Call the WASM code.
|
2016-02-23 15:33:06 +00:00
|
|
|
CallDescriptor* desc =
|
|
|
|
wasm::ModuleEnv::GetWasmCallDescriptor(jsgraph()->zone(), sig);
|
2016-04-20 14:44:15 +00:00
|
|
|
if (jsgraph()->machine()->Is32()) {
|
|
|
|
desc = wasm::ModuleEnv::GetI32WasmCallDescriptor(jsgraph()->zone(), desc);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* call = graph()->NewNode(jsgraph()->common()->Call(desc), count, args);
|
2016-04-20 14:44:15 +00:00
|
|
|
Node* retval = call;
|
|
|
|
if (jsgraph()->machine()->Is32() && sig->return_count() > 0 &&
|
|
|
|
sig->GetReturn(0) == wasm::kAstI64) {
|
|
|
|
// The return values comes as two values, we pick the low word.
|
|
|
|
retval = graph()->NewNode(jsgraph()->common()->Projection(0), retval);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* jsval =
|
2016-04-20 14:44:15 +00:00
|
|
|
ToJS(retval, context,
|
2015-12-11 12:26:16 +00:00
|
|
|
sig->return_count() == 0 ? wasm::kAstStmt : sig->GetReturn());
|
|
|
|
Node* ret =
|
|
|
|
graph()->NewNode(jsgraph()->common()->Return(), jsval, call, start);
|
|
|
|
|
|
|
|
MergeControlToEnd(jsgraph(), ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WasmGraphBuilder::BuildWasmToJSWrapper(Handle<JSFunction> function,
|
|
|
|
wasm::FunctionSig* sig) {
|
|
|
|
int js_count = function->shared()->internal_formal_parameter_count();
|
|
|
|
int wasm_count = static_cast<int>(sig->parameter_count());
|
2016-04-20 14:44:15 +00:00
|
|
|
int param_count;
|
|
|
|
if (jsgraph()->machine()->Is64()) {
|
|
|
|
param_count = wasm_count;
|
|
|
|
} else {
|
|
|
|
param_count = Int64Lowering::GetParameterCountAfterLowering(sig);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
// Build the start and the parameter nodes.
|
|
|
|
Isolate* isolate = jsgraph()->isolate();
|
|
|
|
CallDescriptor* desc;
|
2016-04-20 14:44:15 +00:00
|
|
|
Node* start = Start(param_count + 3);
|
2015-12-11 12:26:16 +00:00
|
|
|
*effect_ = start;
|
|
|
|
*control_ = start;
|
|
|
|
// JS context is the last parameter.
|
2016-05-12 11:58:27 +00:00
|
|
|
Node* context = HeapConstant(Handle<Context>(function->context(), isolate));
|
2015-12-11 12:26:16 +00:00
|
|
|
Node** args = Buffer(wasm_count + 7);
|
|
|
|
|
|
|
|
bool arg_count_before_args = false;
|
|
|
|
bool add_new_target_undefined = false;
|
|
|
|
|
|
|
|
int pos = 0;
|
|
|
|
if (js_count == wasm_count) {
|
|
|
|
// exact arity match, just call the function directly.
|
|
|
|
desc = Linkage::GetJSCallDescriptor(graph()->zone(), false, wasm_count + 1,
|
|
|
|
CallDescriptor::kNoFlags);
|
|
|
|
arg_count_before_args = false;
|
|
|
|
add_new_target_undefined = true;
|
|
|
|
} else {
|
|
|
|
// Use the Call builtin.
|
|
|
|
Callable callable = CodeFactory::Call(isolate);
|
|
|
|
args[pos++] = jsgraph()->HeapConstant(callable.code());
|
|
|
|
desc = Linkage::GetStubCallDescriptor(isolate, graph()->zone(),
|
|
|
|
callable.descriptor(), wasm_count + 1,
|
|
|
|
CallDescriptor::kNoFlags);
|
|
|
|
arg_count_before_args = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
args[pos++] = jsgraph()->Constant(function); // JS function.
|
|
|
|
if (arg_count_before_args) {
|
|
|
|
args[pos++] = jsgraph()->Int32Constant(wasm_count); // argument count
|
|
|
|
}
|
2016-01-14 09:48:45 +00:00
|
|
|
// JS receiver.
|
|
|
|
Handle<Object> global(function->context()->global_object(), isolate);
|
|
|
|
args[pos++] = jsgraph()->Constant(global);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
// Convert WASM numbers to JS values.
|
2016-04-20 14:44:15 +00:00
|
|
|
int param_index = 0;
|
2015-12-11 12:26:16 +00:00
|
|
|
for (int i = 0; i < wasm_count; i++) {
|
2016-04-20 14:44:15 +00:00
|
|
|
Node* param =
|
|
|
|
graph()->NewNode(jsgraph()->common()->Parameter(param_index++), start);
|
2015-12-11 12:26:16 +00:00
|
|
|
args[pos++] = ToJS(param, context, sig->GetParam(i));
|
2016-04-20 14:44:15 +00:00
|
|
|
if (jsgraph()->machine()->Is32() && sig->GetParam(i) == wasm::kAstI64) {
|
|
|
|
// On 32 bit platforms we have to skip the high word of int64 parameters.
|
|
|
|
param_index++;
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (add_new_target_undefined) {
|
|
|
|
args[pos++] = jsgraph()->UndefinedConstant(); // new target
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!arg_count_before_args) {
|
|
|
|
args[pos++] = jsgraph()->Int32Constant(wasm_count); // argument count
|
|
|
|
}
|
|
|
|
args[pos++] = context;
|
|
|
|
args[pos++] = *effect_;
|
|
|
|
args[pos++] = *control_;
|
|
|
|
|
|
|
|
Node* call = graph()->NewNode(jsgraph()->common()->Call(desc), pos, args);
|
|
|
|
|
|
|
|
// Convert the return value back.
|
2016-04-20 14:44:15 +00:00
|
|
|
Node* ret;
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* val =
|
|
|
|
FromJS(call, context,
|
|
|
|
sig->return_count() == 0 ? wasm::kAstStmt : sig->GetReturn());
|
2016-04-20 14:44:15 +00:00
|
|
|
if (jsgraph()->machine()->Is32() && sig->return_count() > 0 &&
|
|
|
|
sig->GetReturn() == wasm::kAstI64) {
|
|
|
|
ret = graph()->NewNode(jsgraph()->common()->Return(), val,
|
|
|
|
graph()->NewNode(jsgraph()->machine()->Word32Sar(),
|
|
|
|
val, jsgraph()->Int32Constant(31)),
|
|
|
|
call, start);
|
|
|
|
} else {
|
|
|
|
ret = graph()->NewNode(jsgraph()->common()->Return(), val, call, start);
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
MergeControlToEnd(jsgraph(), ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::MemBuffer(uint32_t offset) {
|
2016-01-27 11:04:02 +00:00
|
|
|
DCHECK(module_ && module_->instance);
|
2015-12-11 12:26:16 +00:00
|
|
|
if (offset == 0) {
|
2016-01-27 11:04:02 +00:00
|
|
|
if (!mem_buffer_) {
|
2016-04-12 09:05:05 +00:00
|
|
|
mem_buffer_ = jsgraph()->RelocatableIntPtrConstant(
|
|
|
|
reinterpret_cast<uintptr_t>(module_->instance->mem_start),
|
|
|
|
RelocInfo::WASM_MEMORY_REFERENCE);
|
2016-01-27 11:04:02 +00:00
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
return mem_buffer_;
|
|
|
|
} else {
|
2016-04-12 09:05:05 +00:00
|
|
|
return jsgraph()->RelocatableIntPtrConstant(
|
|
|
|
reinterpret_cast<uintptr_t>(module_->instance->mem_start + offset),
|
|
|
|
RelocInfo::WASM_MEMORY_REFERENCE);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::MemSize(uint32_t offset) {
|
2016-01-27 11:04:02 +00:00
|
|
|
DCHECK(module_ && module_->instance);
|
|
|
|
uint32_t size = static_cast<uint32_t>(module_->instance->mem_size);
|
2015-12-11 12:26:16 +00:00
|
|
|
if (offset == 0) {
|
|
|
|
if (!mem_size_) mem_size_ = jsgraph()->Int32Constant(size);
|
|
|
|
return mem_size_;
|
|
|
|
} else {
|
|
|
|
return jsgraph()->Int32Constant(size + offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::FunctionTable() {
|
2016-01-27 11:04:02 +00:00
|
|
|
DCHECK(module_ && module_->instance &&
|
|
|
|
!module_->instance->function_table.is_null());
|
2015-12-11 12:26:16 +00:00
|
|
|
if (!function_table_) {
|
2016-05-12 11:58:27 +00:00
|
|
|
function_table_ = HeapConstant(module_->instance->function_table);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
return function_table_;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::LoadGlobal(uint32_t index) {
|
2016-01-27 11:04:02 +00:00
|
|
|
DCHECK(module_ && module_->instance && module_->instance->globals_start);
|
2015-12-11 12:26:16 +00:00
|
|
|
MachineType mem_type = module_->GetGlobalType(index);
|
|
|
|
Node* addr = jsgraph()->IntPtrConstant(
|
2016-01-27 11:04:02 +00:00
|
|
|
reinterpret_cast<uintptr_t>(module_->instance->globals_start +
|
2016-02-28 19:04:10 +00:00
|
|
|
module_->module->globals[index].offset));
|
2015-12-11 12:26:16 +00:00
|
|
|
const Operator* op = jsgraph()->machine()->Load(mem_type);
|
|
|
|
Node* node = graph()->NewNode(op, addr, jsgraph()->Int32Constant(0), *effect_,
|
|
|
|
*control_);
|
|
|
|
*effect_ = node;
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::StoreGlobal(uint32_t index, Node* val) {
|
2016-01-27 11:04:02 +00:00
|
|
|
DCHECK(module_ && module_->instance && module_->instance->globals_start);
|
2015-12-11 12:26:16 +00:00
|
|
|
MachineType mem_type = module_->GetGlobalType(index);
|
|
|
|
Node* addr = jsgraph()->IntPtrConstant(
|
2016-01-27 11:04:02 +00:00
|
|
|
reinterpret_cast<uintptr_t>(module_->instance->globals_start +
|
2016-02-28 19:04:10 +00:00
|
|
|
module_->module->globals[index].offset));
|
2015-12-11 12:26:16 +00:00
|
|
|
const Operator* op = jsgraph()->machine()->Store(
|
2015-12-11 15:34:00 +00:00
|
|
|
StoreRepresentation(mem_type.representation(), kNoWriteBarrier));
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* node = graph()->NewNode(op, addr, jsgraph()->Int32Constant(0), val,
|
|
|
|
*effect_, *control_);
|
|
|
|
*effect_ = node;
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WasmGraphBuilder::BoundsCheckMem(MachineType memtype, Node* index,
|
2016-05-02 08:47:05 +00:00
|
|
|
uint32_t offset,
|
|
|
|
wasm::WasmCodePosition position) {
|
2016-01-27 11:04:02 +00:00
|
|
|
DCHECK(module_ && module_->instance);
|
|
|
|
size_t size = module_->instance->mem_size;
|
2015-12-11 12:26:16 +00:00
|
|
|
byte memsize = wasm::WasmOpcodes::MemSize(memtype);
|
2016-05-09 08:52:05 +00:00
|
|
|
|
2016-01-27 11:04:02 +00:00
|
|
|
if (offset >= size || (static_cast<uint64_t>(offset) + memsize) > size) {
|
2016-05-09 08:52:05 +00:00
|
|
|
// The access will always throw (unless memory is grown).
|
|
|
|
Node* cond = jsgraph()->Int32Constant(0);
|
|
|
|
trap_->AddTrapIfFalse(wasm::kTrapMemOutOfBounds, cond, position);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check against the effective size.
|
|
|
|
size_t effective_size = size - offset - memsize;
|
|
|
|
CHECK(effective_size <= kMaxUInt32);
|
|
|
|
|
|
|
|
Uint32Matcher m(index);
|
|
|
|
if (m.HasValue()) {
|
|
|
|
uint32_t value = m.Value();
|
|
|
|
if (value <= effective_size) {
|
|
|
|
// The bounds check will always succeed.
|
|
|
|
return;
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
2016-05-09 08:52:05 +00:00
|
|
|
Node* cond = graph()->NewNode(
|
|
|
|
jsgraph()->machine()->Uint32LessThanOrEqual(), index,
|
|
|
|
jsgraph()->Int32Constant(static_cast<uint32_t>(effective_size)));
|
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
trap_->AddTrapIfFalse(wasm::kTrapMemOutOfBounds, cond, position);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::LoadMem(wasm::LocalType type, MachineType memtype,
|
2016-05-02 08:47:05 +00:00
|
|
|
Node* index, uint32_t offset,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* load;
|
2016-05-11 11:50:30 +00:00
|
|
|
// WASM semantics throw on OOB. Introduce explicit bounds check.
|
|
|
|
BoundsCheckMem(memtype, index, offset, position);
|
|
|
|
load = graph()->NewNode(jsgraph()->machine()->Load(memtype),
|
|
|
|
MemBuffer(offset), index, *effect_, *control_);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
*effect_ = load;
|
|
|
|
|
|
|
|
if (type == wasm::kAstI64 &&
|
|
|
|
ElementSizeLog2Of(memtype.representation()) < 3) {
|
|
|
|
// TODO(titzer): TF zeroes the upper bits of 64-bit loads for subword sizes.
|
|
|
|
if (memtype.IsSigned()) {
|
|
|
|
// sign extend
|
|
|
|
load = graph()->NewNode(jsgraph()->machine()->ChangeInt32ToInt64(), load);
|
|
|
|
} else {
|
|
|
|
// zero extend
|
|
|
|
load =
|
|
|
|
graph()->NewNode(jsgraph()->machine()->ChangeUint32ToUint64(), load);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return load;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::StoreMem(MachineType memtype, Node* index,
|
2016-05-02 08:47:05 +00:00
|
|
|
uint32_t offset, Node* val,
|
|
|
|
wasm::WasmCodePosition position) {
|
2015-12-11 12:26:16 +00:00
|
|
|
Node* store;
|
2016-05-11 11:50:30 +00:00
|
|
|
// WASM semantics throw on OOB. Introduce explicit bounds check.
|
|
|
|
BoundsCheckMem(memtype, index, offset, position);
|
|
|
|
StoreRepresentation rep(memtype.representation(), kNoWriteBarrier);
|
|
|
|
store = graph()->NewNode(jsgraph()->machine()->Store(rep), MemBuffer(offset),
|
|
|
|
index, val, *effect_, *control_);
|
2015-12-11 12:26:16 +00:00
|
|
|
*effect_ = store;
|
|
|
|
return store;
|
|
|
|
}
|
|
|
|
|
2016-05-11 11:50:30 +00:00
|
|
|
Node* WasmGraphBuilder::BuildAsmjsLoadMem(MachineType type, Node* index) {
|
|
|
|
// TODO(turbofan): fold bounds checks for constant asm.js loads.
|
|
|
|
// asm.js semantics use CheckedLoad (i.e. OOB reads return 0ish).
|
|
|
|
const Operator* op = jsgraph()->machine()->CheckedLoad(type);
|
|
|
|
Node* load = graph()->NewNode(op, MemBuffer(0), index, MemSize(0), *effect_,
|
|
|
|
*control_);
|
|
|
|
*effect_ = load;
|
|
|
|
return load;
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::BuildAsmjsStoreMem(MachineType type, Node* index,
|
|
|
|
Node* val) {
|
|
|
|
// TODO(turbofan): fold bounds checks for constant asm.js stores.
|
|
|
|
// asm.js semantics use CheckedStore (i.e. ignore OOB writes).
|
|
|
|
const Operator* op =
|
|
|
|
jsgraph()->machine()->CheckedStore(type.representation());
|
|
|
|
Node* store = graph()->NewNode(op, MemBuffer(0), index, MemSize(0), val,
|
|
|
|
*effect_, *control_);
|
|
|
|
*effect_ = store;
|
|
|
|
return val;
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
void WasmGraphBuilder::PrintDebugName(Node* node) {
|
|
|
|
PrintF("#%d:%s", node->id(), node->op()->mnemonic());
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* WasmGraphBuilder::String(const char* string) {
|
|
|
|
return jsgraph()->Constant(
|
|
|
|
jsgraph()->isolate()->factory()->NewStringFromAsciiChecked(string));
|
|
|
|
}
|
|
|
|
|
|
|
|
Graph* WasmGraphBuilder::graph() { return jsgraph()->graph(); }
|
|
|
|
|
2016-02-04 09:40:55 +00:00
|
|
|
void WasmGraphBuilder::Int64LoweringForTesting() {
|
2016-03-08 12:41:25 +00:00
|
|
|
if (jsgraph()->machine()->Is32()) {
|
2016-02-18 15:18:41 +00:00
|
|
|
Int64Lowering r(jsgraph()->graph(), jsgraph()->machine(),
|
|
|
|
jsgraph()->common(), jsgraph()->zone(),
|
|
|
|
function_signature_);
|
|
|
|
r.LowerGraph();
|
|
|
|
}
|
2016-02-04 09:40:55 +00:00
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-05-02 08:47:05 +00:00
|
|
|
void WasmGraphBuilder::SetSourcePosition(Node* node,
|
|
|
|
wasm::WasmCodePosition position) {
|
|
|
|
DCHECK_NE(position, wasm::kNoCodePosition);
|
2016-04-26 12:46:03 +00:00
|
|
|
compiler::SourcePosition pos(position);
|
|
|
|
if (source_position_table_)
|
|
|
|
source_position_table_->SetSourcePosition(node, pos);
|
|
|
|
}
|
|
|
|
|
2016-01-29 11:21:34 +00:00
|
|
|
static void RecordFunctionCompilation(Logger::LogEventsAndTags tag,
|
|
|
|
CompilationInfo* info,
|
|
|
|
const char* message, uint32_t index,
|
2016-03-09 18:55:27 +00:00
|
|
|
wasm::WasmName func_name) {
|
2016-01-29 11:21:34 +00:00
|
|
|
Isolate* isolate = info->isolate();
|
|
|
|
if (isolate->logger()->is_logging_code_events() ||
|
|
|
|
isolate->cpu_profiler()->is_profiling()) {
|
|
|
|
ScopedVector<char> buffer(128);
|
2016-04-26 14:47:01 +00:00
|
|
|
SNPrintF(buffer, "%s#%d:%.*s", message, index, func_name.length(),
|
|
|
|
func_name.start());
|
2016-01-29 11:21:34 +00:00
|
|
|
Handle<String> name_str =
|
|
|
|
isolate->factory()->NewStringFromAsciiChecked(buffer.start());
|
|
|
|
Handle<String> script_str =
|
|
|
|
isolate->factory()->NewStringFromAsciiChecked("(WASM)");
|
|
|
|
Handle<Code> code = info->code();
|
|
|
|
Handle<SharedFunctionInfo> shared =
|
|
|
|
isolate->factory()->NewSharedFunctionInfo(name_str, code, false);
|
2016-02-26 11:04:04 +00:00
|
|
|
PROFILE(isolate, CodeCreateEvent(tag, AbstractCode::cast(*code), *shared,
|
|
|
|
info, *script_str, 0, 0));
|
2016-01-29 11:21:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-17 14:52:27 +00:00
|
|
|
Handle<JSFunction> CompileJSToWasmWrapper(
|
|
|
|
Isolate* isolate, wasm::ModuleEnv* module, Handle<String> name,
|
|
|
|
Handle<Code> wasm_code, Handle<JSObject> module_object, uint32_t index) {
|
2016-02-28 19:04:10 +00:00
|
|
|
wasm::WasmFunction* func = &module->module->functions[index];
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
// Create the JSFunction object.
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
Handle<SharedFunctionInfo> shared =
|
|
|
|
isolate->factory()->NewSharedFunctionInfo(name, wasm_code, false);
|
|
|
|
int params = static_cast<int>(func->sig->parameter_count());
|
|
|
|
shared->set_length(params);
|
2016-03-24 13:23:54 +00:00
|
|
|
shared->set_internal_formal_parameter_count(params);
|
2015-12-17 14:52:27 +00:00
|
|
|
Handle<JSFunction> function = isolate->factory()->NewFunction(
|
|
|
|
isolate->wasm_function_map(), name, MaybeHandle<Code>());
|
|
|
|
function->SetInternalField(0, *module_object);
|
2015-12-11 12:26:16 +00:00
|
|
|
function->set_shared(*shared);
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
// Create the Graph
|
|
|
|
//----------------------------------------------------------------------------
|
2016-04-01 10:00:30 +00:00
|
|
|
Zone zone(isolate->allocator());
|
2015-12-11 12:26:16 +00:00
|
|
|
Graph graph(&zone);
|
|
|
|
CommonOperatorBuilder common(&zone);
|
|
|
|
MachineOperatorBuilder machine(&zone);
|
2016-04-14 10:33:33 +00:00
|
|
|
JSGraph jsgraph(isolate, &graph, &common, nullptr, nullptr, &machine);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
Node* control = nullptr;
|
|
|
|
Node* effect = nullptr;
|
|
|
|
|
2015-12-11 14:54:07 +00:00
|
|
|
WasmGraphBuilder builder(&zone, &jsgraph, func->sig);
|
2015-12-11 12:26:16 +00:00
|
|
|
builder.set_control_ptr(&control);
|
|
|
|
builder.set_effect_ptr(&effect);
|
|
|
|
builder.set_module(module);
|
|
|
|
builder.BuildJSToWasmWrapper(wasm_code, func->sig);
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
// Run the compilation pipeline.
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
{
|
|
|
|
if (FLAG_trace_turbo_graph) { // Simple textual RPO.
|
|
|
|
OFStream os(stdout);
|
|
|
|
os << "-- Graph after change lowering -- " << std::endl;
|
|
|
|
os << AsRPO(graph);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Schedule and compile to machine code.
|
|
|
|
int params = static_cast<int>(
|
|
|
|
module->GetFunctionSignature(index)->parameter_count());
|
|
|
|
CallDescriptor* incoming = Linkage::GetJSCallDescriptor(
|
|
|
|
&zone, false, params + 1, CallDescriptor::kNoFlags);
|
2016-03-04 04:45:22 +00:00
|
|
|
Code::Flags flags = Code::ComputeFlags(Code::JS_TO_WASM_FUNCTION);
|
2016-02-18 16:51:43 +00:00
|
|
|
bool debugging =
|
|
|
|
#if DEBUG
|
|
|
|
true;
|
|
|
|
#else
|
|
|
|
FLAG_print_opt_code || FLAG_trace_turbo || FLAG_trace_turbo_graph;
|
|
|
|
#endif
|
2016-04-26 14:36:35 +00:00
|
|
|
Vector<const char> func_name = ArrayVector("js-to-wasm");
|
2016-02-18 16:51:43 +00:00
|
|
|
|
|
|
|
static unsigned id = 0;
|
|
|
|
Vector<char> buffer;
|
|
|
|
if (debugging) {
|
|
|
|
buffer = Vector<char>::New(128);
|
2016-04-26 14:36:35 +00:00
|
|
|
int chars = SNPrintF(buffer, "js-to-wasm#%d", id);
|
|
|
|
func_name = Vector<const char>::cast(buffer.SubVector(0, chars));
|
2016-02-18 16:51:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CompilationInfo info(func_name, isolate, &zone, flags);
|
2015-12-11 12:26:16 +00:00
|
|
|
Handle<Code> code =
|
2016-04-26 12:46:03 +00:00
|
|
|
Pipeline::GenerateCodeForTesting(&info, incoming, &graph);
|
2016-03-29 11:41:19 +00:00
|
|
|
#ifdef ENABLE_DISASSEMBLER
|
|
|
|
if (FLAG_print_opt_code && !code.is_null()) {
|
|
|
|
OFStream os(stdout);
|
|
|
|
code->Disassemble(buffer.start(), os);
|
|
|
|
}
|
|
|
|
#endif
|
2016-02-18 16:51:43 +00:00
|
|
|
if (debugging) {
|
|
|
|
buffer.Dispose();
|
|
|
|
}
|
|
|
|
|
2016-03-09 18:55:27 +00:00
|
|
|
RecordFunctionCompilation(
|
|
|
|
Logger::FUNCTION_TAG, &info, "js-to-wasm", index,
|
|
|
|
module->module->GetName(func->name_offset, func->name_length));
|
2015-12-11 12:26:16 +00:00
|
|
|
// Set the JSFunction's machine code.
|
|
|
|
function->set_code(*code);
|
|
|
|
}
|
|
|
|
return function;
|
|
|
|
}
|
|
|
|
|
|
|
|
Handle<Code> CompileWasmToJSWrapper(Isolate* isolate, wasm::ModuleEnv* module,
|
|
|
|
Handle<JSFunction> function,
|
2016-03-08 19:55:36 +00:00
|
|
|
wasm::FunctionSig* sig,
|
2016-03-09 18:55:27 +00:00
|
|
|
wasm::WasmName module_name,
|
|
|
|
wasm::WasmName function_name) {
|
2015-12-11 12:26:16 +00:00
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
// Create the Graph
|
|
|
|
//----------------------------------------------------------------------------
|
2016-04-01 10:00:30 +00:00
|
|
|
Zone zone(isolate->allocator());
|
2015-12-11 12:26:16 +00:00
|
|
|
Graph graph(&zone);
|
|
|
|
CommonOperatorBuilder common(&zone);
|
|
|
|
MachineOperatorBuilder machine(&zone);
|
2016-04-14 10:33:33 +00:00
|
|
|
JSGraph jsgraph(isolate, &graph, &common, nullptr, nullptr, &machine);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
Node* control = nullptr;
|
|
|
|
Node* effect = nullptr;
|
|
|
|
|
2016-02-19 14:58:25 +00:00
|
|
|
WasmGraphBuilder builder(&zone, &jsgraph, sig);
|
2015-12-11 12:26:16 +00:00
|
|
|
builder.set_control_ptr(&control);
|
|
|
|
builder.set_effect_ptr(&effect);
|
|
|
|
builder.set_module(module);
|
2016-02-19 14:58:25 +00:00
|
|
|
builder.BuildWasmToJSWrapper(function, sig);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
|
|
|
Handle<Code> code = Handle<Code>::null();
|
|
|
|
{
|
|
|
|
if (FLAG_trace_turbo_graph) { // Simple textual RPO.
|
|
|
|
OFStream os(stdout);
|
|
|
|
os << "-- Graph after change lowering -- " << std::endl;
|
|
|
|
os << AsRPO(graph);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Schedule and compile to machine code.
|
2016-02-23 15:33:06 +00:00
|
|
|
CallDescriptor* incoming =
|
|
|
|
wasm::ModuleEnv::GetWasmCallDescriptor(&zone, sig);
|
2016-04-20 14:44:15 +00:00
|
|
|
if (machine.Is32()) {
|
|
|
|
incoming = wasm::ModuleEnv::GetI32WasmCallDescriptor(&zone, incoming);
|
|
|
|
}
|
2016-03-04 04:45:22 +00:00
|
|
|
Code::Flags flags = Code::ComputeFlags(Code::WASM_TO_JS_FUNCTION);
|
2016-02-18 16:51:43 +00:00
|
|
|
bool debugging =
|
|
|
|
#if DEBUG
|
|
|
|
true;
|
|
|
|
#else
|
|
|
|
FLAG_print_opt_code || FLAG_trace_turbo || FLAG_trace_turbo_graph;
|
|
|
|
#endif
|
2016-04-26 14:36:35 +00:00
|
|
|
Vector<const char> func_name = ArrayVector("wasm-to-js");
|
2016-02-18 16:51:43 +00:00
|
|
|
static unsigned id = 0;
|
|
|
|
Vector<char> buffer;
|
|
|
|
if (debugging) {
|
|
|
|
buffer = Vector<char>::New(128);
|
2016-04-26 14:36:35 +00:00
|
|
|
int chars = SNPrintF(buffer, "wasm-to-js#%d", id);
|
|
|
|
func_name = Vector<const char>::cast(buffer.SubVector(0, chars));
|
2016-02-18 16:51:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CompilationInfo info(func_name, isolate, &zone, flags);
|
2015-12-11 12:26:16 +00:00
|
|
|
code = Pipeline::GenerateCodeForTesting(&info, incoming, &graph, nullptr);
|
2016-03-29 11:41:19 +00:00
|
|
|
#ifdef ENABLE_DISASSEMBLER
|
|
|
|
if (FLAG_print_opt_code && !code.is_null()) {
|
|
|
|
OFStream os(stdout);
|
|
|
|
code->Disassemble(buffer.start(), os);
|
|
|
|
}
|
|
|
|
#endif
|
2016-02-18 16:51:43 +00:00
|
|
|
if (debugging) {
|
|
|
|
buffer.Dispose();
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-02-19 14:58:25 +00:00
|
|
|
RecordFunctionCompilation(Logger::FUNCTION_TAG, &info, "wasm-to-js", 0,
|
2016-03-09 18:55:27 +00:00
|
|
|
module_name);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
|
|
|
return code;
|
|
|
|
}
|
|
|
|
|
2016-04-26 12:46:03 +00:00
|
|
|
std::pair<JSGraph*, SourcePositionTable*> BuildGraphForWasmFunction(
|
2016-05-12 11:58:27 +00:00
|
|
|
JSGraph* jsgraph, wasm::ErrorThrower* thrower, Isolate* isolate,
|
2016-04-28 16:39:53 +00:00
|
|
|
wasm::ModuleEnv*& module_env, const wasm::WasmFunction* function,
|
2016-04-26 12:46:03 +00:00
|
|
|
double* decode_ms) {
|
2016-03-11 13:18:18 +00:00
|
|
|
base::ElapsedTimer decode_timer;
|
|
|
|
if (FLAG_trace_wasm_decode_time) {
|
|
|
|
decode_timer.Start();
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
// Create a TF graph during decoding.
|
2016-05-12 11:58:27 +00:00
|
|
|
Graph* graph = jsgraph->graph();
|
|
|
|
CommonOperatorBuilder* common = jsgraph->common();
|
|
|
|
MachineOperatorBuilder* machine = jsgraph->machine();
|
2016-04-26 12:46:03 +00:00
|
|
|
SourcePositionTable* source_position_table =
|
2016-05-12 11:58:27 +00:00
|
|
|
new (jsgraph->zone()) SourcePositionTable(graph);
|
|
|
|
WasmGraphBuilder builder(jsgraph->zone(), jsgraph, function->sig,
|
|
|
|
source_position_table);
|
2016-03-07 21:04:07 +00:00
|
|
|
wasm::FunctionBody body = {
|
2016-04-28 16:39:53 +00:00
|
|
|
module_env, function->sig, module_env->module->module_start,
|
|
|
|
module_env->module->module_start + function->code_start_offset,
|
|
|
|
module_env->module->module_start + function->code_end_offset};
|
2016-04-01 10:00:30 +00:00
|
|
|
wasm::TreeResult result =
|
|
|
|
wasm::BuildTFGraph(isolate->allocator(), &builder, body);
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-04-26 08:38:24 +00:00
|
|
|
if (machine->Is32()) {
|
2016-05-12 11:58:27 +00:00
|
|
|
Int64Lowering r(graph, machine, common, jsgraph->zone(), function->sig);
|
2016-04-20 14:44:15 +00:00
|
|
|
r.LowerGraph();
|
|
|
|
}
|
|
|
|
|
2015-12-11 12:26:16 +00:00
|
|
|
if (result.failed()) {
|
|
|
|
if (FLAG_trace_wasm_compiler) {
|
|
|
|
OFStream os(stdout);
|
|
|
|
os << "Compilation failed: " << result << std::endl;
|
|
|
|
}
|
|
|
|
// Add the function as another context for the exception
|
2016-01-29 11:21:34 +00:00
|
|
|
ScopedVector<char> buffer(128);
|
2016-04-28 16:39:53 +00:00
|
|
|
wasm::WasmName name = module_env->module->GetName(function->name_offset,
|
|
|
|
function->name_length);
|
2016-03-09 18:55:27 +00:00
|
|
|
SNPrintF(buffer, "Compiling WASM function #%d:%.*s failed:",
|
2016-04-28 16:39:53 +00:00
|
|
|
function->func_index, name.length(), name.start());
|
|
|
|
thrower->Failed(buffer.start(), result);
|
2016-04-26 12:46:03 +00:00
|
|
|
return std::make_pair(nullptr, nullptr);
|
2015-12-11 12:26:16 +00:00
|
|
|
}
|
2016-04-28 16:39:53 +00:00
|
|
|
int index = static_cast<int>(function->func_index);
|
2016-03-17 08:21:50 +00:00
|
|
|
if (index >= FLAG_trace_wasm_ast_start && index < FLAG_trace_wasm_ast_end) {
|
2016-04-01 10:00:30 +00:00
|
|
|
PrintAst(isolate->allocator(), body);
|
2016-03-17 08:21:50 +00:00
|
|
|
}
|
2016-03-11 13:18:18 +00:00
|
|
|
if (FLAG_trace_wasm_decode_time) {
|
2016-04-26 08:38:24 +00:00
|
|
|
*decode_ms = decode_timer.Elapsed().InMillisecondsF();
|
|
|
|
}
|
2016-04-26 12:46:03 +00:00
|
|
|
return std::make_pair(jsgraph, source_position_table);
|
2016-04-26 08:38:24 +00:00
|
|
|
}
|
|
|
|
|
2016-05-03 11:12:41 +00:00
|
|
|
class WasmCompilationUnit {
|
|
|
|
public:
|
|
|
|
WasmCompilationUnit(wasm::ErrorThrower* thrower, Isolate* isolate,
|
|
|
|
wasm::ModuleEnv* module_env,
|
|
|
|
const wasm::WasmFunction* function, uint32_t index)
|
|
|
|
: thrower_(thrower),
|
|
|
|
isolate_(isolate),
|
|
|
|
module_env_(module_env),
|
|
|
|
function_(function),
|
2016-05-12 11:58:27 +00:00
|
|
|
graph_zone_(new Zone(isolate->allocator())),
|
|
|
|
jsgraph_(new (graph_zone()) JSGraph(
|
|
|
|
isolate, new (graph_zone()) Graph(graph_zone()),
|
|
|
|
new (graph_zone()) CommonOperatorBuilder(graph_zone()), nullptr,
|
|
|
|
nullptr,
|
|
|
|
new (graph_zone()) MachineOperatorBuilder(
|
|
|
|
graph_zone(), MachineType::PointerRepresentation(),
|
|
|
|
InstructionSelector::SupportedMachineOperatorFlags()))),
|
2016-05-03 11:12:41 +00:00
|
|
|
compilation_zone_(isolate->allocator()),
|
|
|
|
info_(function->name_length != 0
|
|
|
|
? module_env->module->GetNameOrNull(function->name_offset,
|
|
|
|
function->name_length)
|
|
|
|
: ArrayVector("wasm"),
|
|
|
|
isolate, &compilation_zone_,
|
|
|
|
Code::ComputeFlags(Code::WASM_FUNCTION)),
|
|
|
|
job_(),
|
|
|
|
index_(index),
|
2016-05-12 11:58:27 +00:00
|
|
|
ok_(true) {
|
|
|
|
// Create and cache this node in the main thread.
|
|
|
|
jsgraph_->CEntryStubConstant(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
Zone* graph_zone() { return graph_zone_.get(); }
|
2016-04-26 08:38:24 +00:00
|
|
|
|
2016-05-03 11:12:41 +00:00
|
|
|
void ExecuteCompilation() {
|
2016-05-12 11:58:27 +00:00
|
|
|
// TODO(ahaas): The counters are not thread-safe at the moment.
|
|
|
|
// HistogramTimerScope wasm_compile_function_time_scope(
|
|
|
|
// isolate_->counters()->wasm_compile_function_time());
|
2016-05-03 11:12:41 +00:00
|
|
|
if (FLAG_trace_wasm_compiler) {
|
|
|
|
OFStream os(stdout);
|
|
|
|
os << "Compiling WASM function "
|
|
|
|
<< wasm::WasmFunctionName(function_, module_env_) << std::endl;
|
|
|
|
os << std::endl;
|
|
|
|
}
|
2016-04-26 08:38:24 +00:00
|
|
|
|
2016-05-03 11:12:41 +00:00
|
|
|
double decode_ms = 0;
|
|
|
|
size_t node_count = 0;
|
2016-03-11 13:18:18 +00:00
|
|
|
|
2016-05-12 11:58:27 +00:00
|
|
|
base::SmartPointer<Zone> graph_zone(graph_zone_.Detach());
|
2016-05-03 11:12:41 +00:00
|
|
|
std::pair<JSGraph*, SourcePositionTable*> graph_result =
|
2016-05-12 11:58:27 +00:00
|
|
|
BuildGraphForWasmFunction(jsgraph_, thrower_, isolate_, module_env_,
|
2016-05-03 11:12:41 +00:00
|
|
|
function_, &decode_ms);
|
|
|
|
JSGraph* jsgraph = graph_result.first;
|
|
|
|
SourcePositionTable* source_positions = graph_result.second;
|
|
|
|
|
|
|
|
if (jsgraph == nullptr) {
|
|
|
|
ok_ = false;
|
|
|
|
return;
|
2016-04-26 14:36:35 +00:00
|
|
|
}
|
2016-04-26 09:44:38 +00:00
|
|
|
|
2016-05-03 11:12:41 +00:00
|
|
|
base::ElapsedTimer pipeline_timer;
|
|
|
|
if (FLAG_trace_wasm_decode_time) {
|
|
|
|
node_count = jsgraph->graph()->NodeCount();
|
|
|
|
pipeline_timer.Start();
|
|
|
|
}
|
2016-04-29 12:24:58 +00:00
|
|
|
|
2016-05-03 11:12:41 +00:00
|
|
|
// Run the compiler pipeline to generate machine code.
|
|
|
|
CallDescriptor* descriptor = wasm::ModuleEnv::GetWasmCallDescriptor(
|
|
|
|
&compilation_zone_, function_->sig);
|
|
|
|
if (jsgraph->machine()->Is32()) {
|
|
|
|
descriptor =
|
|
|
|
module_env_->GetI32WasmCallDescriptor(&compilation_zone_, descriptor);
|
|
|
|
}
|
|
|
|
job_.Reset(Pipeline::NewWasmCompilationJob(&info_, jsgraph->graph(),
|
|
|
|
descriptor, source_positions));
|
|
|
|
ok_ = job_->OptimizeGraph() == CompilationJob::SUCCEEDED;
|
|
|
|
// TODO(bradnelson): Improve histogram handling of size_t.
|
2016-05-12 11:58:27 +00:00
|
|
|
// TODO(ahaas): The counters are not thread-safe at the moment.
|
|
|
|
// isolate_->counters()->wasm_compile_function_peak_memory_bytes()->AddSample(
|
|
|
|
// static_cast<int>(jsgraph->graph()->zone()->allocation_size()));
|
2016-05-03 11:12:41 +00:00
|
|
|
|
|
|
|
if (FLAG_trace_wasm_decode_time) {
|
|
|
|
double pipeline_ms = pipeline_timer.Elapsed().InMillisecondsF();
|
|
|
|
PrintF(
|
|
|
|
"wasm-compilation phase 1 ok: %d bytes, %0.3f ms decode, %zu nodes, "
|
|
|
|
"%0.3f ms pipeline\n",
|
|
|
|
static_cast<int>(function_->code_end_offset -
|
|
|
|
function_->code_start_offset),
|
|
|
|
decode_ms, node_count, pipeline_ms);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Handle<Code> FinishCompilation() {
|
|
|
|
if (!ok_) {
|
|
|
|
return Handle<Code>::null();
|
|
|
|
}
|
|
|
|
if (job_->GenerateCode() != CompilationJob::SUCCEEDED) {
|
|
|
|
return Handle<Code>::null();
|
|
|
|
}
|
|
|
|
base::ElapsedTimer compile_timer;
|
|
|
|
if (FLAG_trace_wasm_decode_time) {
|
|
|
|
compile_timer.Start();
|
|
|
|
}
|
|
|
|
Handle<Code> code = info_.code();
|
|
|
|
DCHECK(!code.is_null());
|
2016-04-29 12:24:58 +00:00
|
|
|
DCHECK(code->deoptimization_data() == nullptr ||
|
|
|
|
code->deoptimization_data()->length() == 0);
|
|
|
|
Handle<FixedArray> deopt_data =
|
2016-05-03 11:12:41 +00:00
|
|
|
isolate_->factory()->NewFixedArray(2, TENURED);
|
|
|
|
if (!module_env_->instance->js_object.is_null()) {
|
|
|
|
deopt_data->set(0, *module_env_->instance->js_object);
|
2016-04-29 12:24:58 +00:00
|
|
|
}
|
2016-05-06 09:07:30 +00:00
|
|
|
deopt_data->set(1, Smi::FromInt(function_->func_index));
|
2016-04-29 12:24:58 +00:00
|
|
|
deopt_data->set_length(2);
|
|
|
|
code->set_deoptimization_data(*deopt_data);
|
|
|
|
|
2016-04-28 16:39:53 +00:00
|
|
|
RecordFunctionCompilation(
|
2016-05-03 11:12:41 +00:00
|
|
|
Logger::FUNCTION_TAG, &info_, "WASM_function", function_->func_index,
|
|
|
|
module_env_->module->GetName(function_->name_offset,
|
|
|
|
function_->name_length));
|
|
|
|
|
|
|
|
if (FLAG_trace_wasm_decode_time) {
|
|
|
|
double compile_ms = compile_timer.Elapsed().InMillisecondsF();
|
|
|
|
PrintF("wasm-code-generation ok: %d bytes, %0.3f ms code generation\n",
|
|
|
|
static_cast<int>(function_->code_end_offset -
|
|
|
|
function_->code_start_offset),
|
|
|
|
compile_ms);
|
2016-04-28 16:39:53 +00:00
|
|
|
}
|
|
|
|
|
2016-05-03 11:12:41 +00:00
|
|
|
return code;
|
2016-04-28 16:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
wasm::ErrorThrower* thrower_;
|
|
|
|
Isolate* isolate_;
|
|
|
|
wasm::ModuleEnv* module_env_;
|
|
|
|
const wasm::WasmFunction* function_;
|
2016-05-12 11:58:27 +00:00
|
|
|
// The graph zone is deallocated at the end of ExecuteCompilation.
|
|
|
|
base::SmartPointer<Zone> graph_zone_;
|
|
|
|
JSGraph* jsgraph_;
|
2016-05-03 11:12:41 +00:00
|
|
|
Zone compilation_zone_;
|
|
|
|
CompilationInfo info_;
|
|
|
|
base::SmartPointer<CompilationJob> job_;
|
|
|
|
uint32_t index_;
|
|
|
|
bool ok_;
|
2016-04-28 16:39:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
WasmCompilationUnit* CreateWasmCompilationUnit(
|
|
|
|
wasm::ErrorThrower* thrower, Isolate* isolate, wasm::ModuleEnv* module_env,
|
2016-05-03 11:12:41 +00:00
|
|
|
const wasm::WasmFunction* function, uint32_t index) {
|
|
|
|
return new WasmCompilationUnit(thrower, isolate, module_env, function, index);
|
2016-04-28 16:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ExecuteCompilation(WasmCompilationUnit* unit) {
|
|
|
|
unit->ExecuteCompilation();
|
|
|
|
}
|
|
|
|
|
2016-05-03 11:12:41 +00:00
|
|
|
uint32_t GetIndexOfWasmCompilationUnit(WasmCompilationUnit* unit) {
|
|
|
|
return unit->index_;
|
|
|
|
}
|
|
|
|
|
2016-04-28 16:39:53 +00:00
|
|
|
Handle<Code> FinishCompilation(WasmCompilationUnit* unit) {
|
|
|
|
Handle<Code> result = unit->FinishCompilation();
|
|
|
|
delete unit;
|
|
|
|
return result;
|
|
|
|
}
|
2015-12-11 12:26:16 +00:00
|
|
|
|
2016-05-03 11:12:41 +00:00
|
|
|
// Helper function to compile a single function.
|
|
|
|
Handle<Code> CompileWasmFunction(wasm::ErrorThrower* thrower, Isolate* isolate,
|
|
|
|
wasm::ModuleEnv* module_env,
|
|
|
|
const wasm::WasmFunction* function) {
|
|
|
|
WasmCompilationUnit* unit =
|
|
|
|
CreateWasmCompilationUnit(thrower, isolate, module_env, function, 0);
|
|
|
|
ExecuteCompilation(unit);
|
|
|
|
return FinishCompilation(unit);
|
|
|
|
}
|
|
|
|
|
2015-12-11 12:26:16 +00:00
|
|
|
} // namespace compiler
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|