blob: c195e300e1d605c1d91a1d1d2e863427ec40d9f1 [file] [log] [blame]
// Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
// Class for intrinsifying functions.
#include "vm/assembler.h"
#include "vm/intrinsifier.h"
#include "vm/flags.h"
#include "vm/object.h"
#include "vm/symbols.h"
#include "vm/flow_graph.h"
#include "vm/flow_graph_compiler.h"
#include "vm/flow_graph_allocator.h"
#include "vm/flow_graph_builder.h"
#include "vm/il_printer.h"
#include "vm/intermediate_language.h"
#include "vm/parser.h"
namespace dart {
DEFINE_FLAG(bool, intrinsify, true, "Instrinsify when possible");
DECLARE_FLAG(bool, throw_on_javascript_int_overflow);
DECLARE_FLAG(bool, code_comments);
DECLARE_FLAG(bool, print_flow_graph);
DECLARE_FLAG(bool, print_flow_graph_optimized);
bool Intrinsifier::CanIntrinsify(const Function& function) {
if (!FLAG_intrinsify) return false;
if (function.IsClosureFunction()) return false;
// Can occur because of compile-all flag.
if (function.is_external()) return false;
return function.is_intrinsic();
}
void Intrinsifier::InitializeState() {
Isolate* isolate = Isolate::Current();
Library& lib = Library::Handle(isolate);
Class& cls = Class::Handle(isolate);
Function& func = Function::Handle(isolate);
String& str = String::Handle(isolate);
Error& error = Error::Handle(isolate);
#define SETUP_FUNCTION(class_name, function_name, destination, fp) \
if (strcmp(#class_name, "::") == 0) { \
str = String::New(#function_name); \
func = lib.LookupFunctionAllowPrivate(str); \
} else { \
str = String::New(#class_name); \
cls = lib.LookupClassAllowPrivate(str); \
ASSERT(!cls.IsNull()); \
error = cls.EnsureIsFinalized(isolate); \
if (!error.IsNull()) { \
OS::PrintErr("%s\n", error.ToErrorCString()); \
} \
ASSERT(error.IsNull()); \
if (#function_name[0] == '.') { \
str = String::New(#class_name#function_name); \
} else { \
str = String::New(#function_name); \
} \
func = cls.LookupFunctionAllowPrivate(str); \
} \
ASSERT(!func.IsNull()); \
func.set_is_intrinsic(true);
// Set up all core lib functions that can be intrisified.
lib = Library::CoreLibrary();
ASSERT(!lib.IsNull());
CORE_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
CORE_INTEGER_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
GRAPH_CORE_INTRINSICS_LIST(SETUP_FUNCTION);
// Set up all math lib functions that can be intrisified.
lib = Library::MathLibrary();
ASSERT(!lib.IsNull());
MATH_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
// Set up all dart:typed_data lib functions that can be intrisified.
lib = Library::TypedDataLibrary();
ASSERT(!lib.IsNull());
TYPED_DATA_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
GRAPH_TYPED_DATA_INTRINSICS_LIST(SETUP_FUNCTION);
// Setup all dart:profiler lib functions that can be intrinsified.
lib = Library::ProfilerLibrary();
ASSERT(!lib.IsNull());
PROFILER_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
#undef SETUP_FUNCTION
}
static void EmitCodeFor(FlowGraphCompiler* compiler,
FlowGraph* graph) {
// The FlowGraph here is constructed by the intrinsics builder methods, and
// is different from compiler->flow_graph(), the original method's flow graph.
compiler->assembler()->Comment("Graph intrinsic");
for (intptr_t i = 0; i < graph->reverse_postorder().length(); i++) {
BlockEntryInstr* block = graph->reverse_postorder()[i];
if (block->IsGraphEntry()) continue; // No code for graph entry needed.
if (block->HasParallelMove()) {
compiler->parallel_move_resolver()->EmitNativeCode(
block->parallel_move());
}
for (ForwardInstructionIterator it(block); !it.Done(); it.Advance()) {
Instruction* instr = it.Current();
if (FLAG_code_comments) compiler->EmitComment(instr);
if (instr->IsParallelMove()) {
compiler->parallel_move_resolver()->EmitNativeCode(
instr->AsParallelMove());
} else {
ASSERT(instr->locs() != NULL);
// Calls are not supported in intrinsics code.
ASSERT(!instr->locs()->always_calls());
instr->EmitNativeCode(compiler);
}
}
}
}
bool Intrinsifier::GraphIntrinsify(ParsedFunction* parsed_function,
FlowGraphCompiler* compiler) {
ZoneGrowableArray<const ICData*>* ic_data_array =
new ZoneGrowableArray<const ICData*>();
FlowGraphBuilder builder(parsed_function,
*ic_data_array,
NULL, // NULL = not inlining.
Isolate::kNoDeoptId); // No OSR id.
intptr_t block_id = builder.AllocateBlockId();
TargetEntryInstr* normal_entry =
new TargetEntryInstr(block_id,
CatchClauseNode::kInvalidTryIndex);
GraphEntryInstr* graph_entry = new GraphEntryInstr(
parsed_function, normal_entry, Isolate::kNoDeoptId); // No OSR id.
FlowGraph* graph = new FlowGraph(builder, graph_entry, block_id);
const Function& function = parsed_function->function();
switch (function.recognized_kind()) {
#define EMIT_CASE(test_class_name, test_function_name, enum_name, fp) \
case MethodRecognizer::k##enum_name: \
ASSERT(function.CheckSourceFingerprint(fp)); \
if (!Build_##enum_name(graph)) return false; \
break;
GRAPH_INTRINSICS_LIST(EMIT_CASE);
default:
return false;
#undef EMIT_CASE
}
if (FLAG_print_flow_graph || FLAG_print_flow_graph_optimized) {
OS::Print("Intrinsic graph before\n");
FlowGraphPrinter printer(*graph);
printer.PrintBlocks();
}
// Perform register allocation on the SSA graph.
FlowGraphAllocator allocator(*graph, true); // Intrinsic mode.
allocator.AllocateRegisters();
if (FLAG_print_flow_graph || FLAG_print_flow_graph_optimized) {
OS::Print("Intrinsic graph after\n");
FlowGraphPrinter printer(*graph);
printer.PrintBlocks();
}
EmitCodeFor(compiler, graph);
return true;
}
void Intrinsifier::Intrinsify(ParsedFunction* parsed_function,
FlowGraphCompiler* compiler) {
const Function& function = parsed_function->function();
if (!CanIntrinsify(function)) {
return;
}
ASSERT(!compiler->flow_graph().IsCompiledForOsr());
if (GraphIntrinsify(parsed_function, compiler)) {
return;
}
#define EMIT_CASE(test_class_name, test_function_name, enum_name, fp) \
case MethodRecognizer::k##enum_name: \
ASSERT(function.CheckSourceFingerprint(fp)); \
compiler->assembler()->Comment("Intrinsic"); \
enum_name(compiler->assembler()); \
break;
if (FLAG_throw_on_javascript_int_overflow && (Smi::kBits >= 32)) {
// Integer intrinsics are in the core library, but we don't want to
// intrinsify when Smi > 32 bits if we are looking for javascript integer
// overflow.
switch (function.recognized_kind()) {
ALL_INTRINSICS_NO_INTEGER_LIB_LIST(EMIT_CASE);
default:
break;
}
} else {
switch (function.recognized_kind()) {
ALL_INTRINSICS_LIST(EMIT_CASE);
default:
UNREACHABLE();
break;
}
}
#undef EMIT_INTRINSIC
}
class BlockBuilder : public ValueObject {
public:
BlockBuilder(FlowGraph* flow_graph, TargetEntryInstr* entry)
: flow_graph_(flow_graph), entry_(entry), current_(entry) { }
Definition* AddToInitialDefinitions(Definition* def) {
def->set_ssa_temp_index(flow_graph_->alloc_ssa_temp_index());
flow_graph_->AddToInitialDefinitions(def);
return def;
}
Definition* AddDefinition(Definition* def) {
def->set_ssa_temp_index(flow_graph_->alloc_ssa_temp_index());
current_ = current_->AppendInstruction(def);
return def;
}
Instruction* AddInstruction(Instruction* instr) {
current_ = current_->AppendInstruction(instr);
return instr;
}
void AddIntrinsicReturn(Value* value) {
ReturnInstr* instr = new ReturnInstr(TokenPos(), value);
AddInstruction(instr);
entry_->set_last_instruction(instr);
}
Definition* AddParameter(intptr_t index) {
intptr_t adjustment = Intrinsifier::ParameterSlotFromSp();
return AddToInitialDefinitions(
new ParameterInstr(adjustment + index,
flow_graph_->graph_entry(),
SPREG));
}
intptr_t TokenPos() {
return flow_graph_->parsed_function().function().token_pos();
}
private:
FlowGraph* flow_graph_;
BlockEntryInstr* entry_;
Instruction* current_;
};
static void PrepareIndexedOp(BlockBuilder* builder,
Definition* array,
Definition* index,
intptr_t length_offset) {
intptr_t token_pos = builder->TokenPos();
builder->AddInstruction(
new CheckSmiInstr(new Value(index),
Isolate::kNoDeoptId,
token_pos));
Definition* length = builder->AddDefinition(
new LoadFieldInstr(new Value(array),
length_offset,
Type::ZoneHandle(Type::SmiType()),
true)); // immutable
builder->AddInstruction(
new CheckArrayBoundInstr(new Value(length),
new Value(index),
Isolate::kNoDeoptId));
}
bool Intrinsifier::Build_ObjectArrayGetIndexed(FlowGraph* flow_graph) {
GraphEntryInstr* graph_entry = flow_graph->graph_entry();
TargetEntryInstr* normal_entry = graph_entry->normal_entry();
BlockBuilder builder(flow_graph, normal_entry);
Definition* index = builder.AddParameter(1);
Definition* array = builder.AddParameter(2);
PrepareIndexedOp(&builder, array, index, Array::length_offset());
Definition* result = builder.AddDefinition(
new LoadIndexedInstr(new Value(array),
new Value(index),
Instance::ElementSizeFor(kArrayCid), // index scale
kArrayCid,
Isolate::kNoDeoptId,
builder.TokenPos()));
builder.AddIntrinsicReturn(new Value(result));
return true;
}
bool Intrinsifier::Build_ImmutableArrayGetIndexed(FlowGraph* flow_graph) {
return Build_ObjectArrayGetIndexed(flow_graph);
}
bool Intrinsifier::Build_Uint8ArrayGetIndexed(FlowGraph* flow_graph) {
GraphEntryInstr* graph_entry = flow_graph->graph_entry();
TargetEntryInstr* normal_entry = graph_entry->normal_entry();
BlockBuilder builder(flow_graph, normal_entry);
Definition* index = builder.AddParameter(1);
Definition* array = builder.AddParameter(2);
PrepareIndexedOp(&builder, array, index, TypedData::length_offset());
Definition* result = builder.AddDefinition(
new LoadIndexedInstr(new Value(array),
new Value(index),
1, // index scale
kTypedDataUint8ArrayCid,
Isolate::kNoDeoptId,
builder.TokenPos()));
builder.AddIntrinsicReturn(new Value(result));
return true;
}
bool Intrinsifier::Build_ExternalUint8ArrayGetIndexed(FlowGraph* flow_graph) {
GraphEntryInstr* graph_entry = flow_graph->graph_entry();
TargetEntryInstr* normal_entry = graph_entry->normal_entry();
BlockBuilder builder(flow_graph, normal_entry);
Definition* index = builder.AddParameter(1);
Definition* array = builder.AddParameter(2);
PrepareIndexedOp(&builder, array, index, ExternalTypedData::length_offset());
Definition* elements = builder.AddDefinition(
new LoadUntaggedInstr(new Value(array),
ExternalTypedData::data_offset()));
Definition* result = builder.AddDefinition(
new LoadIndexedInstr(new Value(elements),
new Value(index),
1, // index scale
kExternalTypedDataUint8ArrayCid,
Isolate::kNoDeoptId,
builder.TokenPos()));
builder.AddIntrinsicReturn(new Value(result));
return true;
}
bool Intrinsifier::Build_Uint8ArraySetIndexed(FlowGraph* flow_graph) {
GraphEntryInstr* graph_entry = flow_graph->graph_entry();
TargetEntryInstr* normal_entry = graph_entry->normal_entry();
BlockBuilder builder(flow_graph, normal_entry);
Definition* value = builder.AddParameter(1);
Definition* index = builder.AddParameter(2);
Definition* array = builder.AddParameter(3);
PrepareIndexedOp(&builder, array, index, TypedData::length_offset());
builder.AddInstruction(
new CheckSmiInstr(new Value(value),
Isolate::kNoDeoptId,
builder.TokenPos()));
builder.AddInstruction(
new StoreIndexedInstr(new Value(array),
new Value(index),
new Value(value),
kNoStoreBarrier,
1, // index scale
kTypedDataUint8ArrayCid,
Isolate::kNoDeoptId,
builder.TokenPos()));
// Return null.
Definition* null_def = builder.AddDefinition(
new ConstantInstr(Object::ZoneHandle(Object::null())));
builder.AddIntrinsicReturn(new Value(null_def));
return true;
}
bool Intrinsifier::Build_ExternalUint8ArraySetIndexed(FlowGraph* flow_graph) {
GraphEntryInstr* graph_entry = flow_graph->graph_entry();
TargetEntryInstr* normal_entry = graph_entry->normal_entry();
BlockBuilder builder(flow_graph, normal_entry);
Definition* value = builder.AddParameter(1);
Definition* index = builder.AddParameter(2);
Definition* array = builder.AddParameter(3);
PrepareIndexedOp(&builder, array, index, ExternalTypedData::length_offset());
builder.AddInstruction(
new CheckSmiInstr(new Value(value),
Isolate::kNoDeoptId,
builder.TokenPos()));
Definition* elements = builder.AddDefinition(
new LoadUntaggedInstr(new Value(array),
ExternalTypedData::data_offset()));
builder.AddInstruction(
new StoreIndexedInstr(new Value(elements),
new Value(index),
new Value(value),
kNoStoreBarrier,
1, // index scale
kExternalTypedDataUint8ArrayCid,
Isolate::kNoDeoptId,
builder.TokenPos()));
// Return null.
Definition* null_def = builder.AddDefinition(
new ConstantInstr(Object::ZoneHandle(Object::null())));
builder.AddIntrinsicReturn(new Value(null_def));
return true;
}
bool Intrinsifier::Build_Float64ArraySetIndexed(FlowGraph* flow_graph) {
if (!FlowGraphCompiler::SupportsUnboxedDoubles()) return false;
GraphEntryInstr* graph_entry = flow_graph->graph_entry();
TargetEntryInstr* normal_entry = graph_entry->normal_entry();
BlockBuilder builder(flow_graph, normal_entry);
Definition* value = builder.AddParameter(1);
Definition* index = builder.AddParameter(2);
Definition* array = builder.AddParameter(3);
PrepareIndexedOp(&builder, array, index, TypedData::length_offset());
const ICData& value_check = ICData::ZoneHandle(ICData::New(
flow_graph->parsed_function().function(),
String::Handle(flow_graph->parsed_function().function().name()),
Object::empty_array(), // Dummy args. descr.
Isolate::kNoDeoptId,
1));
value_check.AddReceiverCheck(kDoubleCid,
flow_graph->parsed_function().function());
builder.AddInstruction(
new CheckClassInstr(new Value(value),
Isolate::kNoDeoptId,
value_check,
builder.TokenPos()));
Definition* double_value = builder.AddDefinition(
UnboxInstr::Create(kUnboxedDouble,
new Value(value),
Isolate::kNoDeoptId));
// Manually adjust reaching type because there is no type propagation
// when building intrinsics.
double_value->AsUnbox()->value()->SetReachingType(
ZoneCompileType::Wrap(CompileType::FromCid(kDoubleCid)));
builder.AddInstruction(
new StoreIndexedInstr(new Value(array),
new Value(index),
new Value(double_value),
kNoStoreBarrier,
8, // index scale
kTypedDataFloat64ArrayCid,
Isolate::kNoDeoptId,
builder.TokenPos()));
// Return null.
Definition* null_def = builder.AddDefinition(
new ConstantInstr(Object::ZoneHandle(Object::null())));
builder.AddIntrinsicReturn(new Value(null_def));
return true;
}
bool Intrinsifier::Build_Float64ArrayGetIndexed(FlowGraph* flow_graph) {
if (!FlowGraphCompiler::SupportsUnboxedDoubles()) return false;
GraphEntryInstr* graph_entry = flow_graph->graph_entry();
TargetEntryInstr* normal_entry = graph_entry->normal_entry();
BlockBuilder builder(flow_graph, normal_entry);
Definition* index = builder.AddParameter(1);
Definition* array = builder.AddParameter(2);
PrepareIndexedOp(&builder, array, index, TypedData::length_offset());
Definition* unboxed_value = builder.AddDefinition(
new LoadIndexedInstr(new Value(array),
new Value(index),
8, // index scale
kTypedDataFloat64ArrayCid,
Isolate::kNoDeoptId,
builder.TokenPos()));
Definition* result = builder.AddDefinition(
BoxInstr::Create(kUnboxedDouble, new Value(unboxed_value)));
builder.AddIntrinsicReturn(new Value(result));
return true;
}
static bool BuildLoadField(FlowGraph* flow_graph, intptr_t offset) {
GraphEntryInstr* graph_entry = flow_graph->graph_entry();
TargetEntryInstr* normal_entry = graph_entry->normal_entry();
BlockBuilder builder(flow_graph, normal_entry);
Definition* array = builder.AddParameter(1);
Definition* length = builder.AddDefinition(
new LoadFieldInstr(new Value(array),
offset,
Type::ZoneHandle(),
builder.TokenPos()));
builder.AddIntrinsicReturn(new Value(length));
return true;
}
bool Intrinsifier::Build_ObjectArrayLength(FlowGraph* flow_graph) {
return BuildLoadField(flow_graph, Array::length_offset());
}
bool Intrinsifier::Build_ImmutableArrayLength(FlowGraph* flow_graph) {
return BuildLoadField(flow_graph, Array::length_offset());
}
bool Intrinsifier::Build_GrowableArrayLength(FlowGraph* flow_graph) {
return BuildLoadField(flow_graph, GrowableObjectArray::length_offset());
}
bool Intrinsifier::Build_StringBaseLength(FlowGraph* flow_graph) {
return BuildLoadField(flow_graph, String::length_offset());
}
bool Intrinsifier::Build_TypedDataLength(FlowGraph* flow_graph) {
return BuildLoadField(flow_graph, TypedData::length_offset());
}
bool Intrinsifier::Build_GrowableArrayCapacity(FlowGraph* flow_graph) {
GraphEntryInstr* graph_entry = flow_graph->graph_entry();
TargetEntryInstr* normal_entry = graph_entry->normal_entry();
BlockBuilder builder(flow_graph, normal_entry);
Definition* array = builder.AddParameter(1);
Definition* backing_store = builder.AddDefinition(
new LoadFieldInstr(new Value(array),
GrowableObjectArray::data_offset(),
Type::ZoneHandle(),
builder.TokenPos()));
Definition* capacity = builder.AddDefinition(
new LoadFieldInstr(new Value(backing_store),
Array::length_offset(),
Type::ZoneHandle(),
builder.TokenPos()));
builder.AddIntrinsicReturn(new Value(capacity));
return true;
}
} // namespace dart