| Index: runtime/vm/flow_graph_compiler_mips.cc
|
| diff --git a/runtime/vm/flow_graph_compiler_mips.cc b/runtime/vm/flow_graph_compiler_mips.cc
|
| deleted file mode 100644
|
| index 5002660e7cfbad847d6974047a67b02e292fc239..0000000000000000000000000000000000000000
|
| --- a/runtime/vm/flow_graph_compiler_mips.cc
|
| +++ /dev/null
|
| @@ -1,1851 +0,0 @@
|
| -// Copyright (c) 2013, 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.
|
| -
|
| -#include "vm/globals.h" // Needed here to get TARGET_ARCH_MIPS.
|
| -#if defined(TARGET_ARCH_MIPS)
|
| -
|
| -#include "vm/flow_graph_compiler.h"
|
| -
|
| -#include "vm/ast_printer.h"
|
| -#include "vm/compiler.h"
|
| -#include "vm/dart_entry.h"
|
| -#include "vm/deopt_instructions.h"
|
| -#include "vm/il_printer.h"
|
| -#include "vm/instructions.h"
|
| -#include "vm/locations.h"
|
| -#include "vm/object_store.h"
|
| -#include "vm/parser.h"
|
| -#include "vm/stack_frame.h"
|
| -#include "vm/stub_code.h"
|
| -#include "vm/symbols.h"
|
| -
|
| -namespace dart {
|
| -
|
| -DEFINE_FLAG(bool, trap_on_deoptimization, false, "Trap on deoptimization.");
|
| -
|
| -
|
| -FlowGraphCompiler::~FlowGraphCompiler() {
|
| - // BlockInfos are zone-allocated, so their destructors are not called.
|
| - // Verify the labels explicitly here.
|
| - for (int i = 0; i < block_info_.length(); ++i) {
|
| - ASSERT(!block_info_[i]->jump_label()->IsLinked());
|
| - }
|
| -}
|
| -
|
| -
|
| -bool FlowGraphCompiler::SupportsUnboxedDoubles() {
|
| - return true;
|
| -}
|
| -
|
| -
|
| -bool FlowGraphCompiler::SupportsUnboxedMints() {
|
| - return true;
|
| -}
|
| -
|
| -
|
| -bool FlowGraphCompiler::SupportsUnboxedSimd128() {
|
| - return false;
|
| -}
|
| -
|
| -
|
| -bool FlowGraphCompiler::SupportsHardwareDivision() {
|
| - return true;
|
| -}
|
| -
|
| -
|
| -bool FlowGraphCompiler::CanConvertUnboxedMintToDouble() {
|
| - // TODO(johnmccutchan): Investigate possibility on MIPS once
|
| - // mints are implemented there.
|
| - return false;
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EnterIntrinsicMode() {
|
| - ASSERT(!intrinsic_mode());
|
| - intrinsic_mode_ = true;
|
| - assembler()->set_constant_pool_allowed(false);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::ExitIntrinsicMode() {
|
| - ASSERT(intrinsic_mode());
|
| - intrinsic_mode_ = false;
|
| - assembler()->set_constant_pool_allowed(true);
|
| -}
|
| -
|
| -
|
| -RawTypedData* CompilerDeoptInfo::CreateDeoptInfo(FlowGraphCompiler* compiler,
|
| - DeoptInfoBuilder* builder,
|
| - const Array& deopt_table) {
|
| - if (deopt_env_ == NULL) {
|
| - ++builder->current_info_number_;
|
| - return TypedData::null();
|
| - }
|
| -
|
| - intptr_t stack_height = compiler->StackSize();
|
| - AllocateIncomingParametersRecursive(deopt_env_, &stack_height);
|
| -
|
| - intptr_t slot_ix = 0;
|
| - Environment* current = deopt_env_;
|
| -
|
| - // Emit all kMaterializeObject instructions describing objects to be
|
| - // materialized on the deoptimization as a prefix to the deoptimization info.
|
| - EmitMaterializations(deopt_env_, builder);
|
| -
|
| - // The real frame starts here.
|
| - builder->MarkFrameStart();
|
| -
|
| - Zone* zone = compiler->zone();
|
| -
|
| - builder->AddPp(current->function(), slot_ix++);
|
| - builder->AddPcMarker(Function::ZoneHandle(zone), slot_ix++);
|
| - builder->AddCallerFp(slot_ix++);
|
| - builder->AddReturnAddress(current->function(), deopt_id(), slot_ix++);
|
| -
|
| -
|
| - // Emit all values that are needed for materialization as a part of the
|
| - // expression stack for the bottom-most frame. This guarantees that GC
|
| - // will be able to find them during materialization.
|
| - slot_ix = builder->EmitMaterializationArguments(slot_ix);
|
| -
|
| - // For the innermost environment, set outgoing arguments and the locals.
|
| - for (intptr_t i = current->Length() - 1;
|
| - i >= current->fixed_parameter_count(); i--) {
|
| - builder->AddCopy(current->ValueAt(i), current->LocationAt(i), slot_ix++);
|
| - }
|
| -
|
| - Environment* previous = current;
|
| - current = current->outer();
|
| - while (current != NULL) {
|
| - builder->AddPp(current->function(), slot_ix++);
|
| - builder->AddPcMarker(previous->function(), slot_ix++);
|
| - builder->AddCallerFp(slot_ix++);
|
| -
|
| - // For any outer environment the deopt id is that of the call instruction
|
| - // which is recorded in the outer environment.
|
| - builder->AddReturnAddress(current->function(),
|
| - Thread::ToDeoptAfter(current->deopt_id()),
|
| - slot_ix++);
|
| -
|
| - // The values of outgoing arguments can be changed from the inlined call so
|
| - // we must read them from the previous environment.
|
| - for (intptr_t i = previous->fixed_parameter_count() - 1; i >= 0; i--) {
|
| - builder->AddCopy(previous->ValueAt(i), previous->LocationAt(i),
|
| - slot_ix++);
|
| - }
|
| -
|
| - // Set the locals, note that outgoing arguments are not in the environment.
|
| - for (intptr_t i = current->Length() - 1;
|
| - i >= current->fixed_parameter_count(); i--) {
|
| - builder->AddCopy(current->ValueAt(i), current->LocationAt(i), slot_ix++);
|
| - }
|
| -
|
| - // Iterate on the outer environment.
|
| - previous = current;
|
| - current = current->outer();
|
| - }
|
| - // The previous pointer is now the outermost environment.
|
| - ASSERT(previous != NULL);
|
| -
|
| - // Set slots for the outermost environment.
|
| - builder->AddCallerPp(slot_ix++);
|
| - builder->AddPcMarker(previous->function(), slot_ix++);
|
| - builder->AddCallerFp(slot_ix++);
|
| - builder->AddCallerPc(slot_ix++);
|
| -
|
| - // For the outermost environment, set the incoming arguments.
|
| - for (intptr_t i = previous->fixed_parameter_count() - 1; i >= 0; i--) {
|
| - builder->AddCopy(previous->ValueAt(i), previous->LocationAt(i), slot_ix++);
|
| - }
|
| -
|
| - return builder->CreateDeoptInfo(deopt_table);
|
| -}
|
| -
|
| -
|
| -void CompilerDeoptInfoWithStub::GenerateCode(FlowGraphCompiler* compiler,
|
| - intptr_t stub_ix) {
|
| - // Calls do not need stubs, they share a deoptimization trampoline.
|
| - ASSERT(reason() != ICData::kDeoptAtCall);
|
| - Assembler* assembler = compiler->assembler();
|
| -#define __ assembler->
|
| - __ Comment("%s", Name());
|
| - __ Bind(entry_label());
|
| - if (FLAG_trap_on_deoptimization) {
|
| - __ break_(0);
|
| - }
|
| -
|
| - ASSERT(deopt_env() != NULL);
|
| - __ Push(CODE_REG);
|
| - __ BranchLink(*StubCode::Deoptimize_entry());
|
| - set_pc_offset(assembler->CodeSize());
|
| -#undef __
|
| -}
|
| -
|
| -
|
| -#define __ assembler()->
|
| -
|
| -
|
| -// Fall through if bool_register contains null.
|
| -void FlowGraphCompiler::GenerateBoolToJump(Register bool_register,
|
| - Label* is_true,
|
| - Label* is_false) {
|
| - __ Comment("BoolToJump");
|
| - Label fall_through;
|
| - __ BranchEqual(bool_register, Object::null_object(), &fall_through);
|
| - __ BranchEqual(bool_register, Bool::True(), is_true);
|
| - __ b(is_false);
|
| - __ Bind(&fall_through);
|
| -}
|
| -
|
| -
|
| -// A0: instance (must be preserved).
|
| -// A1: instantiator type arguments (if used).
|
| -// A2: function type arguments (if used).
|
| -// Clobbers A3.
|
| -RawSubtypeTestCache* FlowGraphCompiler::GenerateCallSubtypeTestStub(
|
| - TypeTestStubKind test_kind,
|
| - Register instance_reg,
|
| - Register instantiator_type_arguments_reg,
|
| - Register function_type_arguments_reg,
|
| - Register temp_reg,
|
| - Label* is_instance_lbl,
|
| - Label* is_not_instance_lbl) {
|
| - __ Comment("CallSubtypeTestStub");
|
| - ASSERT(instance_reg == A0);
|
| - ASSERT(temp_reg == kNoRegister); // Unused on MIPS.
|
| - const SubtypeTestCache& type_test_cache =
|
| - SubtypeTestCache::ZoneHandle(zone(), SubtypeTestCache::New());
|
| - __ LoadUniqueObject(A3, type_test_cache);
|
| - if (test_kind == kTestTypeOneArg) {
|
| - ASSERT(instantiator_type_arguments_reg == kNoRegister);
|
| - ASSERT(function_type_arguments_reg == kNoRegister);
|
| - __ BranchLink(*StubCode::Subtype1TestCache_entry());
|
| - } else if (test_kind == kTestTypeTwoArgs) {
|
| - ASSERT(instantiator_type_arguments_reg == kNoRegister);
|
| - ASSERT(function_type_arguments_reg == kNoRegister);
|
| - __ BranchLink(*StubCode::Subtype2TestCache_entry());
|
| - } else if (test_kind == kTestTypeFourArgs) {
|
| - ASSERT(instantiator_type_arguments_reg == A1);
|
| - ASSERT(function_type_arguments_reg == A2);
|
| - __ BranchLink(*StubCode::Subtype4TestCache_entry());
|
| - } else {
|
| - UNREACHABLE();
|
| - }
|
| - // Result is in V0: null -> not found, otherwise Bool::True or Bool::False.
|
| - GenerateBoolToJump(V0, is_instance_lbl, is_not_instance_lbl);
|
| - return type_test_cache.raw();
|
| -}
|
| -
|
| -
|
| -// Jumps to labels 'is_instance' or 'is_not_instance' respectively, if
|
| -// type test is conclusive, otherwise fallthrough if a type test could not
|
| -// be completed.
|
| -// A0: instance being type checked (preserved).
|
| -// Clobbers T0.
|
| -RawSubtypeTestCache*
|
| -FlowGraphCompiler::GenerateInstantiatedTypeWithArgumentsTest(
|
| - TokenPosition token_pos,
|
| - const AbstractType& type,
|
| - Label* is_instance_lbl,
|
| - Label* is_not_instance_lbl) {
|
| - __ Comment("InstantiatedTypeWithArgumentsTest");
|
| - ASSERT(type.IsInstantiated());
|
| - const Class& type_class = Class::ZoneHandle(zone(), type.type_class());
|
| - ASSERT(type.IsFunctionType() || (type_class.NumTypeArguments() > 0));
|
| - const Register kInstanceReg = A0;
|
| - Error& bound_error = Error::Handle(zone());
|
| - const Type& int_type = Type::Handle(zone(), Type::IntType());
|
| - const bool smi_is_ok =
|
| - int_type.IsSubtypeOf(type, &bound_error, NULL, Heap::kOld);
|
| - // Malformed type should have been handled at graph construction time.
|
| - ASSERT(smi_is_ok || bound_error.IsNull());
|
| - __ andi(CMPRES1, kInstanceReg, Immediate(kSmiTagMask));
|
| - if (smi_is_ok) {
|
| - __ beq(CMPRES1, ZR, is_instance_lbl);
|
| - } else {
|
| - __ beq(CMPRES1, ZR, is_not_instance_lbl);
|
| - }
|
| - // A function type test requires checking the function signature.
|
| - if (!type.IsFunctionType()) {
|
| - const intptr_t num_type_args = type_class.NumTypeArguments();
|
| - const intptr_t num_type_params = type_class.NumTypeParameters();
|
| - const intptr_t from_index = num_type_args - num_type_params;
|
| - const TypeArguments& type_arguments =
|
| - TypeArguments::ZoneHandle(zone(), type.arguments());
|
| - const bool is_raw_type = type_arguments.IsNull() ||
|
| - type_arguments.IsRaw(from_index, num_type_params);
|
| - if (is_raw_type) {
|
| - const Register kClassIdReg = T0;
|
| - // dynamic type argument, check only classes.
|
| - __ LoadClassId(kClassIdReg, kInstanceReg);
|
| - __ BranchEqual(kClassIdReg, Immediate(type_class.id()), is_instance_lbl);
|
| - // List is a very common case.
|
| - if (IsListClass(type_class)) {
|
| - GenerateListTypeCheck(kClassIdReg, is_instance_lbl);
|
| - }
|
| - return GenerateSubtype1TestCacheLookup(
|
| - token_pos, type_class, is_instance_lbl, is_not_instance_lbl);
|
| - }
|
| - // If one type argument only, check if type argument is Object or dynamic.
|
| - if (type_arguments.Length() == 1) {
|
| - const AbstractType& tp_argument =
|
| - AbstractType::ZoneHandle(zone(), type_arguments.TypeAt(0));
|
| - ASSERT(!tp_argument.IsMalformed());
|
| - if (tp_argument.IsType()) {
|
| - ASSERT(tp_argument.HasResolvedTypeClass());
|
| - // Check if type argument is dynamic or Object.
|
| - const Type& object_type = Type::Handle(zone(), Type::ObjectType());
|
| - if (object_type.IsSubtypeOf(tp_argument, NULL, NULL, Heap::kOld)) {
|
| - // Instance class test only necessary.
|
| - return GenerateSubtype1TestCacheLookup(
|
| - token_pos, type_class, is_instance_lbl, is_not_instance_lbl);
|
| - }
|
| - }
|
| - }
|
| - }
|
| - // Regular subtype test cache involving instance's type arguments.
|
| - const Register kInstantiatorTypeArgumentsReg = kNoRegister;
|
| - const Register kFunctionTypeArgumentsReg = kNoRegister;
|
| - const Register kTempReg = kNoRegister;
|
| - // A0: instance (must be preserved).
|
| - return GenerateCallSubtypeTestStub(kTestTypeTwoArgs, kInstanceReg,
|
| - kInstantiatorTypeArgumentsReg,
|
| - kFunctionTypeArgumentsReg, kTempReg,
|
| - is_instance_lbl, is_not_instance_lbl);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::CheckClassIds(Register class_id_reg,
|
| - const GrowableArray<intptr_t>& class_ids,
|
| - Label* is_equal_lbl,
|
| - Label* is_not_equal_lbl) {
|
| - __ Comment("CheckClassIds");
|
| - for (intptr_t i = 0; i < class_ids.length(); i++) {
|
| - __ BranchEqual(class_id_reg, Immediate(class_ids[i]), is_equal_lbl);
|
| - }
|
| - __ b(is_not_equal_lbl);
|
| -}
|
| -
|
| -
|
| -// Testing against an instantiated type with no arguments, without
|
| -// SubtypeTestCache.
|
| -// A0: instance being type checked (preserved).
|
| -// Clobbers: T0, T1, T2
|
| -// Returns true if there is a fallthrough.
|
| -bool FlowGraphCompiler::GenerateInstantiatedTypeNoArgumentsTest(
|
| - TokenPosition token_pos,
|
| - const AbstractType& type,
|
| - Label* is_instance_lbl,
|
| - Label* is_not_instance_lbl) {
|
| - __ Comment("InstantiatedTypeNoArgumentsTest");
|
| - ASSERT(type.IsInstantiated());
|
| - if (type.IsFunctionType()) {
|
| - // Fallthrough.
|
| - return true;
|
| - }
|
| - const Class& type_class = Class::Handle(zone(), type.type_class());
|
| - ASSERT(type_class.NumTypeArguments() == 0);
|
| -
|
| - const Register kInstanceReg = A0;
|
| - __ andi(T0, A0, Immediate(kSmiTagMask));
|
| - // If instance is Smi, check directly.
|
| - const Class& smi_class = Class::Handle(zone(), Smi::Class());
|
| - if (smi_class.IsSubtypeOf(Object::null_type_arguments(), type_class,
|
| - Object::null_type_arguments(), NULL, NULL,
|
| - Heap::kOld)) {
|
| - __ beq(T0, ZR, is_instance_lbl);
|
| - } else {
|
| - __ beq(T0, ZR, is_not_instance_lbl);
|
| - }
|
| - const Register kClassIdReg = T0;
|
| - __ LoadClassId(kClassIdReg, kInstanceReg);
|
| - // See ClassFinalizer::ResolveSuperTypeAndInterfaces for list of restricted
|
| - // interfaces.
|
| - // Bool interface can be implemented only by core class Bool.
|
| - if (type.IsBoolType()) {
|
| - __ BranchEqual(kClassIdReg, Immediate(kBoolCid), is_instance_lbl);
|
| - __ b(is_not_instance_lbl);
|
| - return false;
|
| - }
|
| - // Custom checking for numbers (Smi, Mint, Bigint and Double).
|
| - // Note that instance is not Smi (checked above).
|
| - if (type.IsNumberType() || type.IsIntType() || type.IsDoubleType()) {
|
| - GenerateNumberTypeCheck(kClassIdReg, type, is_instance_lbl,
|
| - is_not_instance_lbl);
|
| - return false;
|
| - }
|
| - if (type.IsStringType()) {
|
| - GenerateStringTypeCheck(kClassIdReg, is_instance_lbl, is_not_instance_lbl);
|
| - return false;
|
| - }
|
| - if (type.IsDartFunctionType()) {
|
| - // Check if instance is a closure.
|
| - __ BranchEqual(kClassIdReg, Immediate(kClosureCid), is_instance_lbl);
|
| - return true; // Fall through
|
| - }
|
| - // Compare if the classes are equal.
|
| - if (!type_class.is_abstract()) {
|
| - __ BranchEqual(kClassIdReg, Immediate(type_class.id()), is_instance_lbl);
|
| - }
|
| - // Otherwise fallthrough.
|
| - return true;
|
| -}
|
| -
|
| -
|
| -// Uses SubtypeTestCache to store instance class and result.
|
| -// A0: instance to test.
|
| -// Clobbers A1-A3, T0-T3.
|
| -// Immediate class test already done.
|
| -// TODO(srdjan): Implement a quicker subtype check, as type test
|
| -// arrays can grow too high, but they may be useful when optimizing
|
| -// code (type-feedback).
|
| -RawSubtypeTestCache* FlowGraphCompiler::GenerateSubtype1TestCacheLookup(
|
| - TokenPosition token_pos,
|
| - const Class& type_class,
|
| - Label* is_instance_lbl,
|
| - Label* is_not_instance_lbl) {
|
| - __ Comment("Subtype1TestCacheLookup");
|
| - const Register kInstanceReg = A0;
|
| - __ LoadClass(T0, kInstanceReg);
|
| - // T0: instance class.
|
| - // Check immediate superclass equality.
|
| - __ lw(T0, FieldAddress(T0, Class::super_type_offset()));
|
| - __ lw(T0, FieldAddress(T0, Type::type_class_id_offset()));
|
| - __ BranchEqual(T0, Immediate(Smi::RawValue(type_class.id())),
|
| - is_instance_lbl);
|
| -
|
| - const Register kInstantiatorTypeArgumentsReg = kNoRegister;
|
| - const Register kFunctionTypeArgumentsReg = kNoRegister;
|
| - const Register kTempReg = kNoRegister;
|
| - return GenerateCallSubtypeTestStub(kTestTypeOneArg, kInstanceReg,
|
| - kInstantiatorTypeArgumentsReg,
|
| - kFunctionTypeArgumentsReg, kTempReg,
|
| - is_instance_lbl, is_not_instance_lbl);
|
| -}
|
| -
|
| -
|
| -// Generates inlined check if 'type' is a type parameter or type itself
|
| -// A0: instance (preserved).
|
| -RawSubtypeTestCache* FlowGraphCompiler::GenerateUninstantiatedTypeTest(
|
| - TokenPosition token_pos,
|
| - const AbstractType& type,
|
| - Label* is_instance_lbl,
|
| - Label* is_not_instance_lbl) {
|
| - __ Comment("UninstantiatedTypeTest");
|
| - ASSERT(!type.IsInstantiated());
|
| - // Skip check if destination is a dynamic type.
|
| - if (type.IsTypeParameter()) {
|
| - const TypeParameter& type_param = TypeParameter::Cast(type);
|
| - __ lw(A1, Address(SP, 1 * kWordSize)); // Get instantiator type args.
|
| - __ lw(A2, Address(SP, 0 * kWordSize)); // Get function type args.
|
| - // A1: instantiator type arguments.
|
| - // A2: function type arguments.
|
| - const Register kTypeArgumentsReg =
|
| - type_param.IsClassTypeParameter() ? A1 : A2;
|
| - // Check if type arguments are null, i.e. equivalent to vector of dynamic.
|
| - __ LoadObject(T7, Object::null_object());
|
| - __ beq(kTypeArgumentsReg, T7, is_instance_lbl);
|
| - __ lw(T2, FieldAddress(kTypeArgumentsReg,
|
| - TypeArguments::type_at_offset(type_param.index())));
|
| - // T2: concrete type of type.
|
| - // Check if type argument is dynamic.
|
| - __ BranchEqual(T2, Object::dynamic_type(), is_instance_lbl);
|
| - __ BranchEqual(T2, Type::ZoneHandle(zone(), Type::ObjectType()),
|
| - is_instance_lbl);
|
| - // TODO(regis): Optimize void type as well once allowed as type argument.
|
| -
|
| - // For Smi check quickly against int and num interfaces.
|
| - Label not_smi;
|
| - __ andi(CMPRES1, A0, Immediate(kSmiTagMask));
|
| - __ bne(CMPRES1, ZR, ¬_smi); // Value is Smi?
|
| - __ BranchEqual(T2, Type::ZoneHandle(zone(), Type::IntType()),
|
| - is_instance_lbl);
|
| - __ BranchEqual(T2, Type::ZoneHandle(zone(), Type::Number()),
|
| - is_instance_lbl);
|
| - // Smi must be handled in runtime.
|
| - Label fall_through;
|
| - __ b(&fall_through);
|
| -
|
| - __ Bind(¬_smi);
|
| - // A0: instance.
|
| - // A1: instantiator type arguments.
|
| - // A2: function type arguments.
|
| - const Register kInstanceReg = A0;
|
| - const Register kInstantiatorTypeArgumentsReg = A1;
|
| - const Register kFunctionTypeArgumentsReg = A2;
|
| - const Register kTempReg = kNoRegister;
|
| - const SubtypeTestCache& type_test_cache = SubtypeTestCache::ZoneHandle(
|
| - zone(), GenerateCallSubtypeTestStub(
|
| - kTestTypeFourArgs, kInstanceReg,
|
| - kInstantiatorTypeArgumentsReg, kFunctionTypeArgumentsReg,
|
| - kTempReg, is_instance_lbl, is_not_instance_lbl));
|
| - __ Bind(&fall_through);
|
| - return type_test_cache.raw();
|
| - }
|
| - if (type.IsType()) {
|
| - const Register kInstanceReg = A0;
|
| - const Register kInstantiatorTypeArgumentsReg = A1;
|
| - const Register kFunctionTypeArgumentsReg = A2;
|
| - __ andi(CMPRES1, kInstanceReg, Immediate(kSmiTagMask));
|
| - __ beq(CMPRES1, ZR, is_not_instance_lbl); // Is instance Smi?
|
| - __ lw(kInstantiatorTypeArgumentsReg, Address(SP, 1 * kWordSize));
|
| - __ lw(kFunctionTypeArgumentsReg, Address(SP, 0 * kWordSize));
|
| - // Uninstantiated type class is known at compile time, but the type
|
| - // arguments are determined at runtime by the instantiator.
|
| - const Register kTempReg = kNoRegister;
|
| - return GenerateCallSubtypeTestStub(kTestTypeFourArgs, kInstanceReg,
|
| - kInstantiatorTypeArgumentsReg,
|
| - kFunctionTypeArgumentsReg, kTempReg,
|
| - is_instance_lbl, is_not_instance_lbl);
|
| - }
|
| - return SubtypeTestCache::null();
|
| -}
|
| -
|
| -
|
| -// Inputs:
|
| -// - A0: instance being type checked (preserved).
|
| -// - A1: optional instantiator type arguments (preserved).
|
| -// - A2: optional function type arguments (preserved).
|
| -// Returns:
|
| -// - preserved instance in A0, optional instantiator type arguments in A1, and
|
| -// optional function type arguments in A2.
|
| -// Clobbers: T0, T1, T2
|
| -// Note that this inlined code must be followed by the runtime_call code, as it
|
| -// may fall through to it. Otherwise, this inline code will jump to the label
|
| -// is_instance or to the label is_not_instance.
|
| -RawSubtypeTestCache* FlowGraphCompiler::GenerateInlineInstanceof(
|
| - TokenPosition token_pos,
|
| - const AbstractType& type,
|
| - Label* is_instance_lbl,
|
| - Label* is_not_instance_lbl) {
|
| - __ Comment("InlineInstanceof");
|
| - if (type.IsInstantiated()) {
|
| - const Class& type_class = Class::ZoneHandle(zone(), type.type_class());
|
| - // A class equality check is only applicable with a dst type (not a
|
| - // function type) of a non-parameterized class or with a raw dst type of
|
| - // a parameterized class.
|
| - if (type.IsFunctionType() || (type_class.NumTypeArguments() > 0)) {
|
| - return GenerateInstantiatedTypeWithArgumentsTest(
|
| - token_pos, type, is_instance_lbl, is_not_instance_lbl);
|
| - // Fall through to runtime call.
|
| - }
|
| - const bool has_fall_through = GenerateInstantiatedTypeNoArgumentsTest(
|
| - token_pos, type, is_instance_lbl, is_not_instance_lbl);
|
| - if (has_fall_through) {
|
| - // If test non-conclusive so far, try the inlined type-test cache.
|
| - // 'type' is known at compile time.
|
| - return GenerateSubtype1TestCacheLookup(
|
| - token_pos, type_class, is_instance_lbl, is_not_instance_lbl);
|
| - } else {
|
| - return SubtypeTestCache::null();
|
| - }
|
| - }
|
| - return GenerateUninstantiatedTypeTest(token_pos, type, is_instance_lbl,
|
| - is_not_instance_lbl);
|
| -}
|
| -
|
| -
|
| -// If instanceof type test cannot be performed successfully at compile time and
|
| -// therefore eliminated, optimize it by adding inlined tests for:
|
| -// - NULL -> return type == Null (type is not Object or dynamic).
|
| -// - Smi -> compile time subtype check (only if dst class is not parameterized).
|
| -// - Class equality (only if class is not parameterized).
|
| -// Inputs:
|
| -// - A0: object.
|
| -// - A1: instantiator type arguments or raw_null.
|
| -// - A2: function type arguments or raw_null.
|
| -// Returns:
|
| -// - true or false in V0.
|
| -void FlowGraphCompiler::GenerateInstanceOf(TokenPosition token_pos,
|
| - intptr_t deopt_id,
|
| - const AbstractType& type,
|
| - LocationSummary* locs) {
|
| - ASSERT(type.IsFinalized() && !type.IsMalformed() && !type.IsMalbounded());
|
| - ASSERT(!type.IsObjectType() && !type.IsDynamicType() && !type.IsVoidType());
|
| -
|
| - // Preserve instantiator type arguments (A1) and function type arguments (A2).
|
| - __ addiu(SP, SP, Immediate(-2 * kWordSize));
|
| - __ sw(A1, Address(SP, 1 * kWordSize));
|
| - __ sw(A2, Address(SP, 0 * kWordSize));
|
| -
|
| - Label is_instance, is_not_instance;
|
| - // If type is instantiated and non-parameterized, we can inline code
|
| - // checking whether the tested instance is a Smi.
|
| - if (type.IsInstantiated()) {
|
| - // A null object is only an instance of Null, Object, and dynamic.
|
| - // Object and dynamic have already been checked above (if the type is
|
| - // instantiated). So we can return false here if the instance is null,
|
| - // unless the type is Null (and if the type is instantiated).
|
| - // We can only inline this null check if the type is instantiated at compile
|
| - // time, since an uninstantiated type at compile time could be Null, Object,
|
| - // or dynamic at run time.
|
| - __ BranchEqual(A0, Object::null_object(),
|
| - type.IsNullType() ? &is_instance : &is_not_instance);
|
| - }
|
| -
|
| - // Generate inline instanceof test.
|
| - SubtypeTestCache& test_cache = SubtypeTestCache::ZoneHandle(zone());
|
| - test_cache =
|
| - GenerateInlineInstanceof(token_pos, type, &is_instance, &is_not_instance);
|
| -
|
| - // test_cache is null if there is no fall-through.
|
| - Label done;
|
| - if (!test_cache.IsNull()) {
|
| - // Generate runtime call.
|
| - __ lw(A1, Address(SP, 1 * kWordSize)); // Get instantiator type args.
|
| - __ lw(A2, Address(SP, 0 * kWordSize)); // Get function type args.
|
| - __ addiu(SP, SP, Immediate(-6 * kWordSize));
|
| - __ LoadObject(TMP, Object::null_object());
|
| - __ sw(TMP, Address(SP, 5 * kWordSize)); // Make room for the result.
|
| - __ sw(A0, Address(SP, 4 * kWordSize)); // Push the instance.
|
| - __ LoadObject(TMP, type);
|
| - __ sw(TMP, Address(SP, 3 * kWordSize)); // Push the type.
|
| - __ sw(A1, Address(SP, 2 * kWordSize)); // Push instantiator type args.
|
| - __ sw(A2, Address(SP, 1 * kWordSize)); // Push function type args.
|
| - __ LoadUniqueObject(A0, test_cache);
|
| - __ sw(A0, Address(SP, 0 * kWordSize));
|
| - GenerateRuntimeCall(token_pos, deopt_id, kInstanceofRuntimeEntry, 5, locs);
|
| - // Pop the parameters supplied to the runtime entry. The result of the
|
| - // instanceof runtime call will be left as the result of the operation.
|
| - __ lw(V0, Address(SP, 5 * kWordSize));
|
| - __ b(&done);
|
| - __ delay_slot()->addiu(SP, SP, Immediate(6 * kWordSize));
|
| - }
|
| - __ Bind(&is_not_instance);
|
| - __ LoadObject(V0, Bool::Get(false));
|
| - __ b(&done);
|
| -
|
| - __ Bind(&is_instance);
|
| - __ LoadObject(V0, Bool::Get(true));
|
| - __ Bind(&done);
|
| - // Remove instantiator type arguments and function type arguments.
|
| - __ Drop(2);
|
| -}
|
| -
|
| -
|
| -// Optimize assignable type check by adding inlined tests for:
|
| -// - NULL -> return NULL.
|
| -// - Smi -> compile time subtype check (only if dst class is not parameterized).
|
| -// - Class equality (only if class is not parameterized).
|
| -// Inputs:
|
| -// - A0: instance being type checked.
|
| -// - A1: instantiator type arguments or raw_null.
|
| -// - A2: function type arguments or raw_null.
|
| -// Returns:
|
| -// - object in A0 for successful assignable check (or throws TypeError).
|
| -// Clobbers: T0, T1, T2
|
| -// Performance notes: positive checks must be quick, negative checks can be slow
|
| -// as they throw an exception.
|
| -void FlowGraphCompiler::GenerateAssertAssignable(TokenPosition token_pos,
|
| - intptr_t deopt_id,
|
| - const AbstractType& dst_type,
|
| - const String& dst_name,
|
| - LocationSummary* locs) {
|
| - __ Comment("AssertAssignable");
|
| - ASSERT(!token_pos.IsClassifying());
|
| - ASSERT(!dst_type.IsNull());
|
| - ASSERT(dst_type.IsFinalized());
|
| - // Assignable check is skipped in FlowGraphBuilder, not here.
|
| - ASSERT(dst_type.IsMalformedOrMalbounded() ||
|
| - (!dst_type.IsDynamicType() && !dst_type.IsObjectType() &&
|
| - !dst_type.IsVoidType()));
|
| -
|
| - // Preserve instantiator type arguments (A1) and function type arguments (A2).
|
| - __ addiu(SP, SP, Immediate(-2 * kWordSize));
|
| - __ sw(A1, Address(SP, 1 * kWordSize));
|
| - __ sw(A2, Address(SP, 0 * kWordSize));
|
| -
|
| - // A null object is always assignable and is returned as result.
|
| - Label is_assignable, runtime_call;
|
| -
|
| - __ BranchEqual(A0, Object::null_object(), &is_assignable);
|
| -
|
| - // Generate throw new TypeError() if the type is malformed or malbounded.
|
| - if (dst_type.IsMalformedOrMalbounded()) {
|
| - __ addiu(SP, SP, Immediate(-4 * kWordSize));
|
| - __ LoadObject(TMP, Object::null_object());
|
| - __ sw(TMP, Address(SP, 3 * kWordSize)); // Make room for the result.
|
| - __ sw(A0, Address(SP, 2 * kWordSize)); // Push the source object.
|
| - __ LoadObject(TMP, dst_name);
|
| - __ sw(TMP, Address(SP, 1 * kWordSize)); // Push the destination name.
|
| - __ LoadObject(TMP, dst_type);
|
| - __ sw(TMP, Address(SP, 0 * kWordSize)); // Push the destination type.
|
| -
|
| - GenerateRuntimeCall(token_pos, deopt_id, kBadTypeErrorRuntimeEntry, 3,
|
| - locs);
|
| - // We should never return here.
|
| - __ break_(0);
|
| -
|
| - __ Bind(&is_assignable); // For a null object.
|
| - __ lw(A1, Address(SP, 1 * kWordSize)); // Restore instantiator type args.
|
| - __ lw(A2, Address(SP, 0 * kWordSize)); // Restore function type args.
|
| - __ addiu(SP, SP, Immediate(2 * kWordSize));
|
| - return;
|
| - }
|
| -
|
| - // Generate inline type check, linking to runtime call if not assignable.
|
| - SubtypeTestCache& test_cache = SubtypeTestCache::ZoneHandle(zone());
|
| - test_cache = GenerateInlineInstanceof(token_pos, dst_type, &is_assignable,
|
| - &runtime_call);
|
| -
|
| - __ Bind(&runtime_call);
|
| - __ lw(A1, Address(SP, 1 * kWordSize)); // Load instantiator type args.
|
| - __ lw(A2, Address(SP, 0 * kWordSize)); // Load function type args.
|
| -
|
| - __ addiu(SP, SP, Immediate(-7 * kWordSize));
|
| - __ LoadObject(TMP, Object::null_object());
|
| - __ sw(TMP, Address(SP, 6 * kWordSize)); // Make room for the result.
|
| - __ sw(A0, Address(SP, 5 * kWordSize)); // Push the source object.
|
| - __ LoadObject(TMP, dst_type);
|
| - __ sw(TMP, Address(SP, 4 * kWordSize)); // Push the type of the destination.
|
| - __ sw(A1, Address(SP, 3 * kWordSize)); // Push instantiator type args.
|
| - __ sw(A2, Address(SP, 2 * kWordSize)); // Push function type args.
|
| - __ LoadObject(TMP, dst_name);
|
| - __ sw(TMP, Address(SP, 1 * kWordSize)); // Push the name of the destination.
|
| - __ LoadUniqueObject(T0, test_cache);
|
| - __ sw(T0, Address(SP, 0 * kWordSize));
|
| -
|
| - GenerateRuntimeCall(token_pos, deopt_id, kTypeCheckRuntimeEntry, 6, locs);
|
| - // Pop the parameters supplied to the runtime entry. The result of the
|
| - // type check runtime call is the checked value.
|
| - __ lw(A0, Address(SP, 6 * kWordSize));
|
| - __ addiu(SP, SP, Immediate(7 * kWordSize));
|
| -
|
| - __ Bind(&is_assignable);
|
| - __ lw(A1, Address(SP, 1 * kWordSize)); // Restore instantiator type args.
|
| - __ lw(A2, Address(SP, 0 * kWordSize)); // Restore function type args.
|
| - __ addiu(SP, SP, Immediate(2 * kWordSize));
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitInstructionEpilogue(Instruction* instr) {
|
| - if (is_optimizing()) return;
|
| - Definition* defn = instr->AsDefinition();
|
| - if ((defn != NULL) && defn->HasTemp()) {
|
| - __ Push(defn->locs()->out(0).reg());
|
| - }
|
| -}
|
| -
|
| -
|
| -// Input parameters:
|
| -// S4: arguments descriptor array.
|
| -void FlowGraphCompiler::CopyParameters() {
|
| - __ Comment("Copy parameters");
|
| - const Function& function = parsed_function().function();
|
| - LocalScope* scope = parsed_function().node_sequence()->scope();
|
| - const int num_fixed_params = function.num_fixed_parameters();
|
| - const int num_opt_pos_params = function.NumOptionalPositionalParameters();
|
| - const int num_opt_named_params = function.NumOptionalNamedParameters();
|
| - const int num_params =
|
| - num_fixed_params + num_opt_pos_params + num_opt_named_params;
|
| - ASSERT(function.NumParameters() == num_params);
|
| - ASSERT(parsed_function().first_parameter_index() == kFirstLocalSlotFromFp);
|
| -
|
| - // Check that min_num_pos_args <= num_pos_args <= max_num_pos_args,
|
| - // where num_pos_args is the number of positional arguments passed in.
|
| - const int min_num_pos_args = num_fixed_params;
|
| - const int max_num_pos_args = num_fixed_params + num_opt_pos_params;
|
| -
|
| - __ lw(T2, FieldAddress(S4, ArgumentsDescriptor::positional_count_offset()));
|
| - // Check that min_num_pos_args <= num_pos_args.
|
| - Label wrong_num_arguments;
|
| - __ BranchSignedLess(T2, Immediate(Smi::RawValue(min_num_pos_args)),
|
| - &wrong_num_arguments);
|
| -
|
| - // Check that num_pos_args <= max_num_pos_args.
|
| - __ BranchSignedGreater(T2, Immediate(Smi::RawValue(max_num_pos_args)),
|
| - &wrong_num_arguments);
|
| -
|
| - // Copy positional arguments.
|
| - // Argument i passed at fp[kParamEndSlotFromFp + num_args - i] is copied
|
| - // to fp[kFirstLocalSlotFromFp - i].
|
| -
|
| - __ lw(T1, FieldAddress(S4, ArgumentsDescriptor::count_offset()));
|
| - // Since T1 and T2 are Smi, use sll 1 instead of sll 2.
|
| - // Let T1 point to the last passed positional argument, i.e. to
|
| - // fp[kParamEndSlotFromFp + num_args - (num_pos_args - 1)].
|
| - __ subu(T1, T1, T2);
|
| - __ sll(T1, T1, 1);
|
| - __ addu(T1, FP, T1);
|
| - __ AddImmediate(T1, (kParamEndSlotFromFp + 1) * kWordSize);
|
| -
|
| - // Let T0 point to the last copied positional argument, i.e. to
|
| - // fp[kFirstLocalSlotFromFp - (num_pos_args - 1)].
|
| - __ AddImmediate(T0, FP, (kFirstLocalSlotFromFp + 1) * kWordSize);
|
| - __ sll(T2, T2, 1); // T2 is a Smi.
|
| -
|
| - __ Comment("Argument Copy Loop");
|
| - Label loop, loop_exit;
|
| - __ blez(T2, &loop_exit);
|
| - __ delay_slot()->subu(T0, T0, T2);
|
| - __ Bind(&loop);
|
| - __ addu(T4, T1, T2);
|
| - __ lw(T3, Address(T4, -kWordSize));
|
| - __ addiu(T2, T2, Immediate(-kWordSize));
|
| - __ addu(T5, T0, T2);
|
| - __ bgtz(T2, &loop);
|
| - __ delay_slot()->sw(T3, Address(T5));
|
| - __ Bind(&loop_exit);
|
| -
|
| - // Copy or initialize optional named arguments.
|
| - Label all_arguments_processed;
|
| -#ifdef DEBUG
|
| - const bool check_correct_named_args = true;
|
| -#else
|
| - const bool check_correct_named_args = function.IsClosureFunction();
|
| -#endif
|
| - if (num_opt_named_params > 0) {
|
| - __ Comment("There are named parameters");
|
| - // Start by alphabetically sorting the names of the optional parameters.
|
| - LocalVariable** opt_param = new LocalVariable*[num_opt_named_params];
|
| - int* opt_param_position = new int[num_opt_named_params];
|
| - for (int pos = num_fixed_params; pos < num_params; pos++) {
|
| - LocalVariable* parameter = scope->VariableAt(pos);
|
| - const String& opt_param_name = parameter->name();
|
| - int i = pos - num_fixed_params;
|
| - while (--i >= 0) {
|
| - LocalVariable* param_i = opt_param[i];
|
| - const intptr_t result = opt_param_name.CompareTo(param_i->name());
|
| - ASSERT(result != 0);
|
| - if (result > 0) break;
|
| - opt_param[i + 1] = opt_param[i];
|
| - opt_param_position[i + 1] = opt_param_position[i];
|
| - }
|
| - opt_param[i + 1] = parameter;
|
| - opt_param_position[i + 1] = pos;
|
| - }
|
| - // Generate code handling each optional parameter in alphabetical order.
|
| - __ lw(T1, FieldAddress(S4, ArgumentsDescriptor::count_offset()));
|
| - // Let T1 point to the first passed argument, i.e. to
|
| - // fp[kParamEndSlotFromFp + num_args - 0]; num_args (T1) is Smi.
|
| - __ sll(T3, T1, 1);
|
| - __ addu(T1, FP, T3);
|
| - __ AddImmediate(T1, kParamEndSlotFromFp * kWordSize);
|
| - // Let T0 point to the entry of the first named argument.
|
| - __ AddImmediate(T0, S4, ArgumentsDescriptor::first_named_entry_offset() -
|
| - kHeapObjectTag);
|
| - for (int i = 0; i < num_opt_named_params; i++) {
|
| - Label load_default_value, assign_optional_parameter;
|
| - const int param_pos = opt_param_position[i];
|
| - // Check if this named parameter was passed in.
|
| - // Load T3 with the name of the argument.
|
| - __ lw(T3, Address(T0, ArgumentsDescriptor::name_offset()));
|
| - ASSERT(opt_param[i]->name().IsSymbol());
|
| - __ BranchNotEqual(T3, opt_param[i]->name(), &load_default_value);
|
| -
|
| - // Load T3 with passed-in argument at provided arg_pos, i.e. at
|
| - // fp[kParamEndSlotFromFp + num_args - arg_pos].
|
| - __ lw(T3, Address(T0, ArgumentsDescriptor::position_offset()));
|
| - // T3 is arg_pos as Smi.
|
| - // Point to next named entry.
|
| - __ AddImmediate(T0, ArgumentsDescriptor::named_entry_size());
|
| - __ subu(T3, ZR, T3);
|
| - __ sll(T3, T3, 1);
|
| - __ addu(T3, T1, T3);
|
| - __ b(&assign_optional_parameter);
|
| - __ delay_slot()->lw(T3, Address(T3));
|
| -
|
| - __ Bind(&load_default_value);
|
| - // Load T3 with default argument.
|
| - const Instance& value = parsed_function().DefaultParameterValueAt(
|
| - param_pos - num_fixed_params);
|
| - __ LoadObject(T3, value);
|
| - __ Bind(&assign_optional_parameter);
|
| - // Assign T3 to fp[kFirstLocalSlotFromFp - param_pos].
|
| - // We do not use the final allocation index of the variable here, i.e.
|
| - // scope->VariableAt(i)->index(), because captured variables still need
|
| - // to be copied to the context that is not yet allocated.
|
| - const intptr_t computed_param_pos = kFirstLocalSlotFromFp - param_pos;
|
| - __ sw(T3, Address(FP, computed_param_pos * kWordSize));
|
| - }
|
| - delete[] opt_param;
|
| - delete[] opt_param_position;
|
| - if (check_correct_named_args) {
|
| - // Check that T0 now points to the null terminator in the arguments
|
| - // descriptor.
|
| - __ lw(T3, Address(T0));
|
| - __ BranchEqual(T3, Object::null_object(), &all_arguments_processed);
|
| - }
|
| - } else {
|
| - ASSERT(num_opt_pos_params > 0);
|
| - __ Comment("There are optional positional parameters");
|
| - __ lw(T2, FieldAddress(S4, ArgumentsDescriptor::positional_count_offset()));
|
| - __ SmiUntag(T2);
|
| - for (int i = 0; i < num_opt_pos_params; i++) {
|
| - Label next_parameter;
|
| - // Handle this optional positional parameter only if k or fewer positional
|
| - // arguments have been passed, where k is param_pos, the position of this
|
| - // optional parameter in the formal parameter list.
|
| - const int param_pos = num_fixed_params + i;
|
| - __ BranchSignedGreater(T2, Immediate(param_pos), &next_parameter);
|
| - // Load T3 with default argument.
|
| - const Object& value = parsed_function().DefaultParameterValueAt(i);
|
| - __ LoadObject(T3, value);
|
| - // Assign T3 to fp[kFirstLocalSlotFromFp - param_pos].
|
| - // We do not use the final allocation index of the variable here, i.e.
|
| - // scope->VariableAt(i)->index(), because captured variables still need
|
| - // to be copied to the context that is not yet allocated.
|
| - const intptr_t computed_param_pos = kFirstLocalSlotFromFp - param_pos;
|
| - __ sw(T3, Address(FP, computed_param_pos * kWordSize));
|
| - __ Bind(&next_parameter);
|
| - }
|
| - if (check_correct_named_args) {
|
| - __ lw(T1, FieldAddress(S4, ArgumentsDescriptor::count_offset()));
|
| - __ SmiUntag(T1);
|
| - // Check that T2 equals T1, i.e. no named arguments passed.
|
| - __ beq(T2, T1, &all_arguments_processed);
|
| - }
|
| - }
|
| -
|
| - __ Bind(&wrong_num_arguments);
|
| - if (function.IsClosureFunction()) {
|
| - __ LeaveDartFrame(kKeepCalleePP); // Arguments are still on the stack.
|
| - __ Branch(*StubCode::CallClosureNoSuchMethod_entry());
|
| - // The noSuchMethod call may return to the caller, but not here.
|
| - } else if (check_correct_named_args) {
|
| - __ Stop("Wrong arguments");
|
| - }
|
| -
|
| - __ Bind(&all_arguments_processed);
|
| - // Nullify originally passed arguments only after they have been copied and
|
| - // checked, otherwise noSuchMethod would not see their original values.
|
| - // This step can be skipped in case we decide that formal parameters are
|
| - // implicitly final, since garbage collecting the unmodified value is not
|
| - // an issue anymore.
|
| -
|
| - // S4 : arguments descriptor array.
|
| - __ lw(T2, FieldAddress(S4, ArgumentsDescriptor::count_offset()));
|
| - __ sll(T2, T2, 1); // T2 is a Smi.
|
| -
|
| - __ Comment("Null arguments loop");
|
| - Label null_args_loop, null_args_loop_exit;
|
| - __ blez(T2, &null_args_loop_exit);
|
| - __ delay_slot()->addiu(T1, FP,
|
| - Immediate((kParamEndSlotFromFp + 1) * kWordSize));
|
| - __ Bind(&null_args_loop);
|
| - __ addiu(T2, T2, Immediate(-kWordSize));
|
| - __ addu(T3, T1, T2);
|
| - __ LoadObject(T5, Object::null_object());
|
| - __ bgtz(T2, &null_args_loop);
|
| - __ delay_slot()->sw(T5, Address(T3));
|
| - __ Bind(&null_args_loop_exit);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::GenerateInlinedGetter(intptr_t offset) {
|
| - // RA: return address.
|
| - // SP: receiver.
|
| - // Sequence node has one return node, its input is load field node.
|
| - __ Comment("Inlined Getter");
|
| - __ lw(V0, Address(SP, 0 * kWordSize));
|
| - __ LoadFieldFromOffset(V0, V0, offset);
|
| - __ Ret();
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::GenerateInlinedSetter(intptr_t offset) {
|
| - // RA: return address.
|
| - // SP+1: receiver.
|
| - // SP+0: value.
|
| - // Sequence node has one store node and one return NULL node.
|
| - __ Comment("Inlined Setter");
|
| - __ lw(T0, Address(SP, 1 * kWordSize)); // Receiver.
|
| - __ lw(T1, Address(SP, 0 * kWordSize)); // Value.
|
| - __ StoreIntoObjectOffset(T0, offset, T1);
|
| - __ LoadObject(V0, Object::null_object());
|
| - __ Ret();
|
| -}
|
| -
|
| -
|
| -static const Register new_pp = T7;
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitFrameEntry() {
|
| - const Function& function = parsed_function().function();
|
| - if (CanOptimizeFunction() && function.IsOptimizable() &&
|
| - (!is_optimizing() || may_reoptimize())) {
|
| - __ Comment("Invocation Count Check");
|
| - const Register function_reg = T0;
|
| -
|
| - // Temporarily setup pool pointer for this dart function.
|
| - __ LoadPoolPointer(new_pp);
|
| - // Load function object from object pool.
|
| - __ LoadFunctionFromCalleePool(function_reg, function, new_pp);
|
| -
|
| - __ lw(T1, FieldAddress(function_reg, Function::usage_counter_offset()));
|
| - // Reoptimization of an optimized function is triggered by counting in
|
| - // IC stubs, but not at the entry of the function.
|
| - if (!is_optimizing()) {
|
| - __ addiu(T1, T1, Immediate(1));
|
| - __ sw(T1, FieldAddress(function_reg, Function::usage_counter_offset()));
|
| - }
|
| -
|
| - // Skip Branch if T1 is less than the threshold.
|
| - Label dont_branch;
|
| - __ BranchSignedLess(T1, Immediate(GetOptimizationThreshold()),
|
| - &dont_branch);
|
| -
|
| - ASSERT(function_reg == T0);
|
| - __ Branch(*StubCode::OptimizeFunction_entry(), new_pp);
|
| -
|
| - __ Bind(&dont_branch);
|
| - }
|
| - __ Comment("Enter frame");
|
| - if (flow_graph().IsCompiledForOsr()) {
|
| - intptr_t extra_slots = StackSize() - flow_graph().num_stack_locals() -
|
| - flow_graph().num_copied_params();
|
| - ASSERT(extra_slots >= 0);
|
| - __ EnterOsrFrame(extra_slots * kWordSize);
|
| - } else {
|
| - ASSERT(StackSize() >= 0);
|
| - __ EnterDartFrame(StackSize() * kWordSize);
|
| - }
|
| -}
|
| -
|
| -
|
| -// Input parameters:
|
| -// RA: return address.
|
| -// SP: address of last argument.
|
| -// FP: caller's frame pointer.
|
| -// PP: caller's pool pointer.
|
| -// S5: ic-data.
|
| -// S4: arguments descriptor array.
|
| -void FlowGraphCompiler::CompileGraph() {
|
| - InitCompiler();
|
| - const Function& function = parsed_function().function();
|
| -
|
| -#ifdef DART_PRECOMPILER
|
| - if (function.IsDynamicFunction()) {
|
| - __ MonomorphicCheckedEntry();
|
| - }
|
| -#endif // DART_PRECOMPILER
|
| -
|
| - if (TryIntrinsify()) {
|
| - // Skip regular code generation.
|
| - return;
|
| - }
|
| -
|
| - EmitFrameEntry();
|
| - ASSERT(assembler()->constant_pool_allowed());
|
| -
|
| - const int num_fixed_params = function.num_fixed_parameters();
|
| - const int num_copied_params = parsed_function().num_copied_params();
|
| - const int num_locals = parsed_function().num_stack_locals();
|
| -
|
| - // We check the number of passed arguments when we have to copy them due to
|
| - // the presence of optional parameters.
|
| - // No such checking code is generated if only fixed parameters are declared,
|
| - // unless we are in debug mode or unless we are compiling a closure.
|
| - if (num_copied_params == 0) {
|
| - const bool check_arguments =
|
| - function.IsClosureFunction() && !flow_graph().IsCompiledForOsr();
|
| - if (check_arguments) {
|
| - __ Comment("Check argument count");
|
| - // Check that exactly num_fixed arguments are passed in.
|
| - Label correct_num_arguments, wrong_num_arguments;
|
| - __ lw(T0, FieldAddress(S4, ArgumentsDescriptor::count_offset()));
|
| - __ BranchNotEqual(T0, Immediate(Smi::RawValue(num_fixed_params)),
|
| - &wrong_num_arguments);
|
| -
|
| - __ lw(T1,
|
| - FieldAddress(S4, ArgumentsDescriptor::positional_count_offset()));
|
| - __ beq(T0, T1, &correct_num_arguments);
|
| - __ Bind(&wrong_num_arguments);
|
| - __ LeaveDartFrame(kKeepCalleePP); // Arguments are still on the stack.
|
| - __ Branch(*StubCode::CallClosureNoSuchMethod_entry());
|
| - // The noSuchMethod call may return to the caller, but not here.
|
| - __ Bind(&correct_num_arguments);
|
| - }
|
| - } else if (!flow_graph().IsCompiledForOsr()) {
|
| - CopyParameters();
|
| - }
|
| -
|
| - if (function.IsClosureFunction() && !flow_graph().IsCompiledForOsr()) {
|
| - // Load context from the closure object (first argument).
|
| - LocalScope* scope = parsed_function().node_sequence()->scope();
|
| - LocalVariable* closure_parameter = scope->VariableAt(0);
|
| - __ lw(CTX, Address(FP, closure_parameter->index() * kWordSize));
|
| - __ lw(CTX, FieldAddress(CTX, Closure::context_offset()));
|
| - }
|
| -
|
| - // In unoptimized code, initialize (non-argument) stack allocated slots to
|
| - // null.
|
| - if (!is_optimizing()) {
|
| - ASSERT(num_locals > 0); // There is always at least context_var.
|
| - __ Comment("Initialize spill slots");
|
| - const intptr_t slot_base = parsed_function().first_stack_local_index();
|
| - const intptr_t context_index =
|
| - parsed_function().current_context_var()->index();
|
| - if (num_locals > 1) {
|
| - __ LoadObject(V0, Object::null_object());
|
| - }
|
| - for (intptr_t i = 0; i < num_locals; ++i) {
|
| - // Subtract index i (locals lie at lower addresses than FP).
|
| - if (((slot_base - i) == context_index)) {
|
| - if (function.IsClosureFunction()) {
|
| - __ sw(CTX, Address(FP, (slot_base - i) * kWordSize));
|
| - } else {
|
| - __ LoadObject(V1, Object::empty_context());
|
| - __ sw(V1, Address(FP, (slot_base - i) * kWordSize));
|
| - }
|
| - } else {
|
| - ASSERT(num_locals > 1);
|
| - __ sw(V0, Address(FP, (slot_base - i) * kWordSize));
|
| - }
|
| - }
|
| - }
|
| -
|
| - // Check for a passed type argument vector if the function is generic.
|
| - if (FLAG_reify_generic_functions && function.IsGeneric()) {
|
| - __ Comment("Check passed-in type args");
|
| - Label store_type_args, ok;
|
| - __ lw(T1, FieldAddress(S4, ArgumentsDescriptor::type_args_len_offset()));
|
| - if (is_optimizing()) {
|
| - // Initialize type_args to null if none passed in.
|
| - __ LoadObject(T0, Object::null_object());
|
| - __ BranchEqual(T1, Immediate(0), &store_type_args);
|
| - } else {
|
| - __ BranchEqual(T1, Immediate(0), &ok); // Already initialized to null.
|
| - }
|
| - // TODO(regis): Verify that type_args_len is correct.
|
| - // Load the passed type args vector in T0 from
|
| - // fp[kParamEndSlotFromFp + num_args + 1]; num_args (T1) is Smi.
|
| - __ lw(T1, FieldAddress(S4, ArgumentsDescriptor::count_offset()));
|
| - __ sll(T1, T1, 1);
|
| - __ addu(T1, FP, T1);
|
| - __ lw(T0, Address(T1, (kParamEndSlotFromFp + 1) * kWordSize));
|
| - // Store T0 into the stack slot reserved for the function type arguments.
|
| - // If the function type arguments variable is captured, a copy will happen
|
| - // after the context is allocated.
|
| - const intptr_t slot_base = parsed_function().first_stack_local_index();
|
| - ASSERT(parsed_function().function_type_arguments()->is_captured() ||
|
| - parsed_function().function_type_arguments()->index() == slot_base);
|
| - __ Bind(&store_type_args);
|
| - __ sw(T0, Address(FP, slot_base * kWordSize));
|
| - __ Bind(&ok);
|
| - }
|
| -
|
| - // TODO(regis): Verify that no vector is passed if not generic, unless already
|
| - // checked during resolution.
|
| -
|
| - EndCodeSourceRange(TokenPosition::kDartCodePrologue);
|
| - VisitBlocks();
|
| -
|
| - __ break_(0);
|
| - GenerateDeferredCode();
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::GenerateCall(TokenPosition token_pos,
|
| - const StubEntry& stub_entry,
|
| - RawPcDescriptors::Kind kind,
|
| - LocationSummary* locs) {
|
| - __ BranchLink(stub_entry);
|
| - EmitCallsiteMetaData(token_pos, Thread::kNoDeoptId, kind, locs);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::GeneratePatchableCall(TokenPosition token_pos,
|
| - const StubEntry& stub_entry,
|
| - RawPcDescriptors::Kind kind,
|
| - LocationSummary* locs) {
|
| - __ BranchLinkPatchable(stub_entry);
|
| - EmitCallsiteMetaData(token_pos, Thread::kNoDeoptId, kind, locs);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::GenerateDartCall(intptr_t deopt_id,
|
| - TokenPosition token_pos,
|
| - const StubEntry& stub_entry,
|
| - RawPcDescriptors::Kind kind,
|
| - LocationSummary* locs) {
|
| - __ BranchLinkPatchable(stub_entry);
|
| - EmitCallsiteMetaData(token_pos, deopt_id, kind, locs);
|
| - // Marks either the continuation point in unoptimized code or the
|
| - // deoptimization point in optimized code, after call.
|
| - const intptr_t deopt_id_after = Thread::ToDeoptAfter(deopt_id);
|
| - if (is_optimizing()) {
|
| - AddDeoptIndexAtCall(deopt_id_after);
|
| - } else {
|
| - // Add deoptimization continuation point after the call and before the
|
| - // arguments are removed.
|
| - AddCurrentDescriptor(RawPcDescriptors::kDeopt, deopt_id_after, token_pos);
|
| - }
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::GenerateStaticDartCall(intptr_t deopt_id,
|
| - TokenPosition token_pos,
|
| - const StubEntry& stub_entry,
|
| - RawPcDescriptors::Kind kind,
|
| - LocationSummary* locs,
|
| - const Function& target) {
|
| - // Call sites to the same target can share object pool entries. These
|
| - // call sites are never patched for breakpoints: the function is deoptimized
|
| - // and the unoptimized code with IC calls for static calls is patched instead.
|
| - ASSERT(is_optimizing());
|
| - __ BranchLinkWithEquivalence(stub_entry, target);
|
| -
|
| - EmitCallsiteMetaData(token_pos, deopt_id, kind, locs);
|
| - // Marks either the continuation point in unoptimized code or the
|
| - // deoptimization point in optimized code, after call.
|
| - const intptr_t deopt_id_after = Thread::ToDeoptAfter(deopt_id);
|
| - if (is_optimizing()) {
|
| - AddDeoptIndexAtCall(deopt_id_after);
|
| - } else {
|
| - // Add deoptimization continuation point after the call and before the
|
| - // arguments are removed.
|
| - AddCurrentDescriptor(RawPcDescriptors::kDeopt, deopt_id_after, token_pos);
|
| - }
|
| - AddStaticCallTarget(target);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::GenerateRuntimeCall(TokenPosition token_pos,
|
| - intptr_t deopt_id,
|
| - const RuntimeEntry& entry,
|
| - intptr_t argument_count,
|
| - LocationSummary* locs) {
|
| - __ CallRuntime(entry, argument_count);
|
| - EmitCallsiteMetaData(token_pos, deopt_id, RawPcDescriptors::kOther, locs);
|
| - if (deopt_id != Thread::kNoDeoptId) {
|
| - // Marks either the continuation point in unoptimized code or the
|
| - // deoptimization point in optimized code, after call.
|
| - const intptr_t deopt_id_after = Thread::ToDeoptAfter(deopt_id);
|
| - if (is_optimizing()) {
|
| - AddDeoptIndexAtCall(deopt_id_after);
|
| - } else {
|
| - // Add deoptimization continuation point after the call and before the
|
| - // arguments are removed.
|
| - AddCurrentDescriptor(RawPcDescriptors::kDeopt, deopt_id_after, token_pos);
|
| - }
|
| - }
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitEdgeCounter(intptr_t edge_id) {
|
| - // We do not check for overflow when incrementing the edge counter. The
|
| - // function should normally be optimized long before the counter can
|
| - // overflow; and though we do not reset the counters when we optimize or
|
| - // deoptimize, there is a bound on the number of
|
| - // optimization/deoptimization cycles we will attempt.
|
| - ASSERT(!edge_counters_array_.IsNull());
|
| - __ Comment("Edge counter");
|
| - __ LoadObject(T0, edge_counters_array_);
|
| - __ LoadFieldFromOffset(T1, T0, Array::element_offset(edge_id));
|
| - __ AddImmediate(T1, T1, Smi::RawValue(1));
|
| - __ StoreFieldToOffset(T1, T0, Array::element_offset(edge_id));
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitOptimizedInstanceCall(const StubEntry& stub_entry,
|
| - const ICData& ic_data,
|
| - intptr_t argument_count,
|
| - intptr_t deopt_id,
|
| - TokenPosition token_pos,
|
| - LocationSummary* locs) {
|
| - ASSERT(Array::Handle(zone(), ic_data.arguments_descriptor()).Length() > 0);
|
| - // Each ICData propagated from unoptimized to optimized code contains the
|
| - // function that corresponds to the Dart function of that IC call. Due
|
| - // to inlining in optimized code, that function may not correspond to the
|
| - // top-level function (parsed_function().function()) which could be
|
| - // reoptimized and which counter needs to be incremented.
|
| - // Pass the function explicitly, it is used in IC stub.
|
| - __ Comment("OptimizedInstanceCall");
|
| - __ LoadObject(T0, parsed_function().function());
|
| - __ LoadUniqueObject(S5, ic_data);
|
| - GenerateDartCall(deopt_id, token_pos, stub_entry, RawPcDescriptors::kIcCall,
|
| - locs);
|
| - __ Drop(argument_count);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitInstanceCall(const StubEntry& stub_entry,
|
| - const ICData& ic_data,
|
| - intptr_t argument_count,
|
| - intptr_t deopt_id,
|
| - TokenPosition token_pos,
|
| - LocationSummary* locs) {
|
| - ASSERT(Array::Handle(zone(), ic_data.arguments_descriptor()).Length() > 0);
|
| - __ Comment("InstanceCall");
|
| - __ LoadUniqueObject(S5, ic_data);
|
| - GenerateDartCall(deopt_id, token_pos, stub_entry, RawPcDescriptors::kIcCall,
|
| - locs);
|
| - __ Comment("InstanceCall return");
|
| - __ Drop(argument_count);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitMegamorphicInstanceCall(
|
| - const String& name,
|
| - const Array& arguments_descriptor,
|
| - intptr_t argument_count,
|
| - intptr_t deopt_id,
|
| - TokenPosition token_pos,
|
| - LocationSummary* locs,
|
| - intptr_t try_index,
|
| - intptr_t slow_path_argument_count) {
|
| - ASSERT(!arguments_descriptor.IsNull() && (arguments_descriptor.Length() > 0));
|
| - const MegamorphicCache& cache = MegamorphicCache::ZoneHandle(
|
| - zone(),
|
| - MegamorphicCacheTable::Lookup(isolate(), name, arguments_descriptor));
|
| -
|
| - __ Comment("MegamorphicCall");
|
| - // Load receiver into T0,
|
| - __ lw(T0, Address(SP, (argument_count - 1) * kWordSize));
|
| - __ LoadObject(S5, cache);
|
| - __ lw(T9, Address(THR, Thread::megamorphic_call_checked_entry_offset()));
|
| - __ jalr(T9);
|
| -
|
| - RecordSafepoint(locs, slow_path_argument_count);
|
| - const intptr_t deopt_id_after = Thread::ToDeoptAfter(deopt_id);
|
| - if (FLAG_precompiled_mode) {
|
| - // Megamorphic calls may occur in slow path stubs.
|
| - // If valid use try_index argument.
|
| - if (try_index == CatchClauseNode::kInvalidTryIndex) {
|
| - try_index = CurrentTryIndex();
|
| - }
|
| - AddDescriptor(RawPcDescriptors::kOther, assembler()->CodeSize(),
|
| - Thread::kNoDeoptId, token_pos, try_index);
|
| - } else if (is_optimizing()) {
|
| - AddCurrentDescriptor(RawPcDescriptors::kOther, Thread::kNoDeoptId,
|
| - token_pos);
|
| - AddDeoptIndexAtCall(deopt_id_after);
|
| - } else {
|
| - AddCurrentDescriptor(RawPcDescriptors::kOther, Thread::kNoDeoptId,
|
| - token_pos);
|
| - // Add deoptimization continuation point after the call and before the
|
| - // arguments are removed.
|
| - AddCurrentDescriptor(RawPcDescriptors::kDeopt, deopt_id_after, token_pos);
|
| - }
|
| - EmitCatchEntryState(pending_deoptimization_env_, try_index);
|
| - __ Drop(argument_count);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitSwitchableInstanceCall(const ICData& ic_data,
|
| - intptr_t argument_count,
|
| - intptr_t deopt_id,
|
| - TokenPosition token_pos,
|
| - LocationSummary* locs) {
|
| - ASSERT(ic_data.NumArgsTested() == 1);
|
| - const Code& initial_stub =
|
| - Code::ZoneHandle(StubCode::ICCallThroughFunction_entry()->code());
|
| -
|
| - __ Comment("SwitchableCall");
|
| - __ lw(T0, Address(SP, (argument_count - 1) * kWordSize));
|
| - __ LoadUniqueObject(CODE_REG, initial_stub);
|
| - __ lw(T9, FieldAddress(CODE_REG, Code::checked_entry_point_offset()));
|
| - __ LoadUniqueObject(S5, ic_data);
|
| - __ jalr(T9);
|
| -
|
| - EmitCallsiteMetaData(token_pos, Thread::kNoDeoptId, RawPcDescriptors::kOther,
|
| - locs);
|
| - const intptr_t deopt_id_after = Thread::ToDeoptAfter(deopt_id);
|
| - if (is_optimizing()) {
|
| - AddDeoptIndexAtCall(deopt_id_after);
|
| - } else {
|
| - // Add deoptimization continuation point after the call and before the
|
| - // arguments are removed.
|
| - AddCurrentDescriptor(RawPcDescriptors::kDeopt, deopt_id_after, token_pos);
|
| - }
|
| - __ Drop(argument_count);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitUnoptimizedStaticCall(intptr_t argument_count,
|
| - intptr_t deopt_id,
|
| - TokenPosition token_pos,
|
| - LocationSummary* locs,
|
| - const ICData& ic_data) {
|
| - const StubEntry* stub_entry =
|
| - StubCode::UnoptimizedStaticCallEntry(ic_data.NumArgsTested());
|
| - __ LoadObject(S5, ic_data);
|
| - GenerateDartCall(deopt_id, token_pos, *stub_entry,
|
| - RawPcDescriptors::kUnoptStaticCall, locs);
|
| - __ Drop(argument_count);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitOptimizedStaticCall(
|
| - const Function& function,
|
| - const Array& arguments_descriptor,
|
| - intptr_t argument_count,
|
| - intptr_t deopt_id,
|
| - TokenPosition token_pos,
|
| - LocationSummary* locs) {
|
| - __ Comment("StaticCall");
|
| - ASSERT(!function.IsClosureFunction());
|
| - if (function.HasOptionalParameters() ||
|
| - (FLAG_reify_generic_functions && function.IsGeneric())) {
|
| - __ LoadObject(S4, arguments_descriptor);
|
| - } else {
|
| - __ LoadImmediate(S4, 0); // GC safe smi zero because of stub.
|
| - }
|
| - // Do not use the code from the function, but let the code be patched so that
|
| - // we can record the outgoing edges to other code.
|
| - GenerateStaticDartCall(deopt_id, token_pos,
|
| - *StubCode::CallStaticFunction_entry(),
|
| - RawPcDescriptors::kOther, locs, function);
|
| - __ Drop(argument_count);
|
| -}
|
| -
|
| -
|
| -Condition FlowGraphCompiler::EmitEqualityRegConstCompare(
|
| - Register reg,
|
| - const Object& obj,
|
| - bool needs_number_check,
|
| - TokenPosition token_pos,
|
| - intptr_t deopt_id) {
|
| - __ Comment("EqualityRegConstCompare");
|
| - ASSERT(!needs_number_check ||
|
| - (!obj.IsMint() && !obj.IsDouble() && !obj.IsBigint()));
|
| - if (needs_number_check) {
|
| - ASSERT(!obj.IsMint() && !obj.IsDouble() && !obj.IsBigint());
|
| - __ addiu(SP, SP, Immediate(-2 * kWordSize));
|
| - __ sw(reg, Address(SP, 1 * kWordSize));
|
| - __ LoadObject(TMP, obj);
|
| - __ sw(TMP, Address(SP, 0 * kWordSize));
|
| - if (is_optimizing()) {
|
| - __ BranchLinkPatchable(
|
| - *StubCode::OptimizedIdenticalWithNumberCheck_entry());
|
| - } else {
|
| - __ BranchLinkPatchable(
|
| - *StubCode::UnoptimizedIdenticalWithNumberCheck_entry());
|
| - }
|
| - AddCurrentDescriptor(RawPcDescriptors::kRuntimeCall, deopt_id, token_pos);
|
| - __ Comment("EqualityRegConstCompare return");
|
| - // Stub returns result in CMPRES1 (if it is 0, then reg and obj are equal).
|
| - __ lw(reg, Address(SP, 1 * kWordSize)); // Restore 'reg'.
|
| - __ addiu(SP, SP, Immediate(2 * kWordSize)); // Discard constant.
|
| - return Condition(CMPRES1, ZR, EQ);
|
| - } else {
|
| - int16_t imm = 0;
|
| - const Register obj_reg = __ LoadConditionOperand(CMPRES1, obj, &imm);
|
| - return Condition(reg, obj_reg, EQ, imm);
|
| - }
|
| -}
|
| -
|
| -
|
| -Condition FlowGraphCompiler::EmitEqualityRegRegCompare(Register left,
|
| - Register right,
|
| - bool needs_number_check,
|
| - TokenPosition token_pos,
|
| - intptr_t deopt_id) {
|
| - __ Comment("EqualityRegRegCompare");
|
| - if (needs_number_check) {
|
| - __ addiu(SP, SP, Immediate(-2 * kWordSize));
|
| - __ sw(left, Address(SP, 1 * kWordSize));
|
| - __ sw(right, Address(SP, 0 * kWordSize));
|
| - if (is_optimizing()) {
|
| - __ BranchLinkPatchable(
|
| - *StubCode::OptimizedIdenticalWithNumberCheck_entry());
|
| - } else {
|
| - __ BranchLinkPatchable(
|
| - *StubCode::UnoptimizedIdenticalWithNumberCheck_entry());
|
| - }
|
| - if (token_pos.IsReal()) {
|
| - AddCurrentDescriptor(RawPcDescriptors::kRuntimeCall, Thread::kNoDeoptId,
|
| - token_pos);
|
| - }
|
| - __ Comment("EqualityRegRegCompare return");
|
| - // Stub returns result in CMPRES1 (if it is 0, then left and right are
|
| - // equal).
|
| - __ lw(right, Address(SP, 0 * kWordSize));
|
| - __ lw(left, Address(SP, 1 * kWordSize));
|
| - __ addiu(SP, SP, Immediate(2 * kWordSize));
|
| - return Condition(CMPRES1, ZR, EQ);
|
| - } else {
|
| - return Condition(left, right, EQ);
|
| - }
|
| -}
|
| -
|
| -
|
| -// This function must be in sync with FlowGraphCompiler::RecordSafepoint and
|
| -// FlowGraphCompiler::SlowPathEnvironmentFor.
|
| -void FlowGraphCompiler::SaveLiveRegisters(LocationSummary* locs) {
|
| -#if defined(DEBUG)
|
| - locs->CheckWritableInputs();
|
| - ClobberDeadTempRegisters(locs);
|
| -#endif
|
| -
|
| - __ Comment("SaveLiveRegisters");
|
| - // TODO(vegorov): consider saving only caller save (volatile) registers.
|
| - const intptr_t fpu_regs_count = locs->live_registers()->FpuRegisterCount();
|
| - if (fpu_regs_count > 0) {
|
| - __ AddImmediate(SP, -(fpu_regs_count * kFpuRegisterSize));
|
| - // Store fpu registers with the lowest register number at the lowest
|
| - // address.
|
| - intptr_t offset = 0;
|
| - for (intptr_t i = 0; i < kNumberOfFpuRegisters; ++i) {
|
| - DRegister fpu_reg = static_cast<DRegister>(i);
|
| - if (locs->live_registers()->ContainsFpuRegister(fpu_reg)) {
|
| - __ StoreDToOffset(fpu_reg, SP, offset);
|
| - offset += kFpuRegisterSize;
|
| - }
|
| - }
|
| - ASSERT(offset == (fpu_regs_count * kFpuRegisterSize));
|
| - }
|
| -
|
| - // The order in which the registers are pushed must match the order
|
| - // in which the registers are encoded in the safe point's stack map.
|
| - const intptr_t cpu_registers = locs->live_registers()->cpu_registers();
|
| - ASSERT((cpu_registers & ~kAllCpuRegistersList) == 0);
|
| - const int register_count = Utils::CountOneBits(cpu_registers);
|
| - if (register_count > 0) {
|
| - __ addiu(SP, SP, Immediate(-register_count * kWordSize));
|
| - intptr_t offset = register_count * kWordSize;
|
| - for (int i = kNumberOfCpuRegisters - 1; i >= 0; --i) {
|
| - Register r = static_cast<Register>(i);
|
| - if (locs->live_registers()->ContainsRegister(r)) {
|
| - offset -= kWordSize;
|
| - __ sw(r, Address(SP, offset));
|
| - }
|
| - }
|
| - ASSERT(offset == 0);
|
| - }
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::RestoreLiveRegisters(LocationSummary* locs) {
|
| - __ Comment("RestoreLiveRegisters");
|
| - const intptr_t cpu_registers = locs->live_registers()->cpu_registers();
|
| - ASSERT((cpu_registers & ~kAllCpuRegistersList) == 0);
|
| - const int register_count = Utils::CountOneBits(cpu_registers);
|
| - if (register_count > 0) {
|
| - intptr_t offset = register_count * kWordSize;
|
| - for (int i = kNumberOfCpuRegisters - 1; i >= 0; --i) {
|
| - Register r = static_cast<Register>(i);
|
| - if (locs->live_registers()->ContainsRegister(r)) {
|
| - offset -= kWordSize;
|
| - __ lw(r, Address(SP, offset));
|
| - }
|
| - }
|
| - ASSERT(offset == 0);
|
| - __ addiu(SP, SP, Immediate(register_count * kWordSize));
|
| - }
|
| -
|
| - const intptr_t fpu_regs_count = locs->live_registers()->FpuRegisterCount();
|
| - if (fpu_regs_count > 0) {
|
| - // Fpu registers have the lowest register number at the lowest address.
|
| - intptr_t offset = 0;
|
| - for (intptr_t i = 0; i < kNumberOfFpuRegisters; ++i) {
|
| - DRegister fpu_reg = static_cast<DRegister>(i);
|
| - if (locs->live_registers()->ContainsFpuRegister(fpu_reg)) {
|
| - __ LoadDFromOffset(fpu_reg, SP, offset);
|
| - offset += kFpuRegisterSize;
|
| - }
|
| - }
|
| - ASSERT(offset == (fpu_regs_count * kFpuRegisterSize));
|
| - __ AddImmediate(SP, offset);
|
| - }
|
| -}
|
| -
|
| -
|
| -#if defined(DEBUG)
|
| -void FlowGraphCompiler::ClobberDeadTempRegisters(LocationSummary* locs) {
|
| - // Clobber temporaries that have not been manually preserved.
|
| - for (intptr_t i = 0; i < locs->temp_count(); ++i) {
|
| - Location tmp = locs->temp(i);
|
| - // TODO(zerny): clobber non-live temporary FPU registers.
|
| - if (tmp.IsRegister() &&
|
| - !locs->live_registers()->ContainsRegister(tmp.reg())) {
|
| - __ LoadImmediate(tmp.reg(), 0xf7);
|
| - }
|
| - }
|
| -}
|
| -#endif
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitTestAndCallLoadReceiver(
|
| - intptr_t argument_count,
|
| - const Array& arguments_descriptor) {
|
| - __ Comment("EmitTestAndCall");
|
| - // Load receiver into T0.
|
| - __ LoadFromOffset(T0, SP, (argument_count - 1) * kWordSize);
|
| - __ LoadObject(S4, arguments_descriptor);
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitTestAndCallSmiBranch(Label* label, bool if_smi) {
|
| - __ andi(CMPRES1, T0, Immediate(kSmiTagMask));
|
| - if (if_smi) {
|
| - // Jump if receiver is Smi.
|
| - __ beq(CMPRES1, ZR, label);
|
| - } else {
|
| - // Jump if receiver is not Smi.
|
| - __ bne(CMPRES1, ZR, label);
|
| - }
|
| -}
|
| -
|
| -
|
| -void FlowGraphCompiler::EmitTestAndCallLoadCid() {
|
| - __ LoadClassId(T2, T0);
|
| -}
|
| -
|
| -
|
| -int FlowGraphCompiler::EmitTestAndCallCheckCid(Label* next_label,
|
| - const CidRange& range,
|
| - int bias) {
|
| - intptr_t cid_start = range.cid_start;
|
| - if (range.IsSingleCid()) {
|
| - __ BranchNotEqual(T2, Immediate(cid_start - bias), next_label);
|
| - } else {
|
| - __ AddImmediate(T2, T2, bias - cid_start);
|
| - bias = cid_start;
|
| - // TODO(erikcorry): We should use sltiu instead of the temporary TMP if
|
| - // the range is small enough.
|
| - __ LoadImmediate(TMP, range.Extent());
|
| - // Reverse comparison so we get 1 if biased cid > tmp ie cid is out of
|
| - // range.
|
| - __ sltu(TMP, TMP, T2);
|
| - __ bne(TMP, ZR, next_label);
|
| - }
|
| - return bias;
|
| -}
|
| -
|
| -
|
| -#undef __
|
| -#define __ compiler_->assembler()->
|
| -
|
| -
|
| -void ParallelMoveResolver::EmitMove(int index) {
|
| - MoveOperands* move = moves_[index];
|
| - const Location source = move->src();
|
| - const Location destination = move->dest();
|
| - __ Comment("ParallelMoveResolver::EmitMove");
|
| -
|
| - if (source.IsRegister()) {
|
| - if (destination.IsRegister()) {
|
| - __ mov(destination.reg(), source.reg());
|
| - } else {
|
| - ASSERT(destination.IsStackSlot());
|
| - const intptr_t dest_offset = destination.ToStackSlotOffset();
|
| - __ StoreToOffset(source.reg(), destination.base_reg(), dest_offset);
|
| - }
|
| - } else if (source.IsStackSlot()) {
|
| - if (destination.IsRegister()) {
|
| - const intptr_t source_offset = source.ToStackSlotOffset();
|
| - __ LoadFromOffset(destination.reg(), source.base_reg(), source_offset);
|
| - } else {
|
| - ASSERT(destination.IsStackSlot());
|
| - const intptr_t source_offset = source.ToStackSlotOffset();
|
| - const intptr_t dest_offset = destination.ToStackSlotOffset();
|
| - ScratchRegisterScope tmp(this, kNoRegister);
|
| - __ LoadFromOffset(tmp.reg(), source.base_reg(), source_offset);
|
| - __ StoreToOffset(tmp.reg(), destination.base_reg(), dest_offset);
|
| - }
|
| - } else if (source.IsFpuRegister()) {
|
| - if (destination.IsFpuRegister()) {
|
| - DRegister dst = destination.fpu_reg();
|
| - DRegister src = source.fpu_reg();
|
| - __ movd(dst, src);
|
| - } else {
|
| - ASSERT(destination.IsDoubleStackSlot());
|
| - const intptr_t dest_offset = destination.ToStackSlotOffset();
|
| - DRegister src = source.fpu_reg();
|
| - __ StoreDToOffset(src, destination.base_reg(), dest_offset);
|
| - }
|
| - } else if (source.IsDoubleStackSlot()) {
|
| - if (destination.IsFpuRegister()) {
|
| - const intptr_t source_offset = source.ToStackSlotOffset();
|
| - DRegister dst = destination.fpu_reg();
|
| - __ LoadDFromOffset(dst, source.base_reg(), source_offset);
|
| - } else {
|
| - ASSERT(destination.IsDoubleStackSlot());
|
| - const intptr_t source_offset = source.ToStackSlotOffset();
|
| - const intptr_t dest_offset = destination.ToStackSlotOffset();
|
| - __ LoadDFromOffset(DTMP, source.base_reg(), source_offset);
|
| - __ StoreDToOffset(DTMP, destination.base_reg(), dest_offset);
|
| - }
|
| - } else {
|
| - ASSERT(source.IsConstant());
|
| - const Object& constant = source.constant();
|
| - if (destination.IsRegister()) {
|
| - if (constant.IsSmi() &&
|
| - (source.constant_instruction()->representation() == kUnboxedInt32)) {
|
| - __ LoadImmediate(destination.reg(), Smi::Cast(constant).Value());
|
| - } else {
|
| - __ LoadObject(destination.reg(), constant);
|
| - }
|
| - } else if (destination.IsFpuRegister()) {
|
| - __ LoadObject(TMP, constant);
|
| - __ LoadDFromOffset(destination.fpu_reg(), TMP,
|
| - Double::value_offset() - kHeapObjectTag);
|
| - } else if (destination.IsDoubleStackSlot()) {
|
| - const intptr_t dest_offset = destination.ToStackSlotOffset();
|
| - __ LoadObject(TMP, constant);
|
| - __ LoadDFromOffset(DTMP, TMP, Double::value_offset() - kHeapObjectTag);
|
| - __ StoreDToOffset(DTMP, destination.base_reg(), dest_offset);
|
| - } else {
|
| - ASSERT(destination.IsStackSlot());
|
| - const intptr_t dest_offset = destination.ToStackSlotOffset();
|
| - ScratchRegisterScope tmp(this, kNoRegister);
|
| - if (constant.IsSmi() &&
|
| - (source.constant_instruction()->representation() == kUnboxedInt32)) {
|
| - __ LoadImmediate(tmp.reg(), Smi::Cast(constant).Value());
|
| - } else {
|
| - __ LoadObject(tmp.reg(), constant);
|
| - }
|
| - __ StoreToOffset(tmp.reg(), destination.base_reg(), dest_offset);
|
| - }
|
| - }
|
| -
|
| - move->Eliminate();
|
| -}
|
| -
|
| -
|
| -void ParallelMoveResolver::EmitSwap(int index) {
|
| - MoveOperands* move = moves_[index];
|
| - const Location source = move->src();
|
| - const Location destination = move->dest();
|
| -
|
| - if (source.IsRegister() && destination.IsRegister()) {
|
| - ASSERT(source.reg() != TMP);
|
| - ASSERT(destination.reg() != TMP);
|
| - __ mov(TMP, source.reg());
|
| - __ mov(source.reg(), destination.reg());
|
| - __ mov(destination.reg(), TMP);
|
| - } else if (source.IsRegister() && destination.IsStackSlot()) {
|
| - Exchange(source.reg(), destination.base_reg(),
|
| - destination.ToStackSlotOffset());
|
| - } else if (source.IsStackSlot() && destination.IsRegister()) {
|
| - Exchange(destination.reg(), source.base_reg(), source.ToStackSlotOffset());
|
| - } else if (source.IsStackSlot() && destination.IsStackSlot()) {
|
| - Exchange(source.base_reg(), source.ToStackSlotOffset(),
|
| - destination.base_reg(), destination.ToStackSlotOffset());
|
| - } else if (source.IsFpuRegister() && destination.IsFpuRegister()) {
|
| - DRegister dst = destination.fpu_reg();
|
| - DRegister src = source.fpu_reg();
|
| - __ movd(DTMP, src);
|
| - __ movd(src, dst);
|
| - __ movd(dst, DTMP);
|
| - } else if (source.IsFpuRegister() || destination.IsFpuRegister()) {
|
| - ASSERT(destination.IsDoubleStackSlot() || source.IsDoubleStackSlot());
|
| - DRegister reg =
|
| - source.IsFpuRegister() ? source.fpu_reg() : destination.fpu_reg();
|
| - Register base_reg =
|
| - source.IsFpuRegister() ? destination.base_reg() : source.base_reg();
|
| - const intptr_t slot_offset = source.IsFpuRegister()
|
| - ? destination.ToStackSlotOffset()
|
| - : source.ToStackSlotOffset();
|
| - __ LoadDFromOffset(DTMP, base_reg, slot_offset);
|
| - __ StoreDToOffset(reg, base_reg, slot_offset);
|
| - __ movd(reg, DTMP);
|
| - } else if (source.IsDoubleStackSlot() && destination.IsDoubleStackSlot()) {
|
| - const intptr_t source_offset = source.ToStackSlotOffset();
|
| - const intptr_t dest_offset = destination.ToStackSlotOffset();
|
| -
|
| - ScratchFpuRegisterScope ensure_scratch(this, DTMP);
|
| - DRegister scratch = ensure_scratch.reg();
|
| - __ LoadDFromOffset(DTMP, source.base_reg(), source_offset);
|
| - __ LoadDFromOffset(scratch, destination.base_reg(), dest_offset);
|
| - __ StoreDToOffset(DTMP, destination.base_reg(), dest_offset);
|
| - __ StoreDToOffset(scratch, source.base_reg(), source_offset);
|
| - } else {
|
| - UNREACHABLE();
|
| - }
|
| -
|
| - // The swap of source and destination has executed a move from source to
|
| - // destination.
|
| - move->Eliminate();
|
| -
|
| - // Any unperformed (including pending) move with a source of either
|
| - // this move's source or destination needs to have their source
|
| - // changed to reflect the state of affairs after the swap.
|
| - for (int i = 0; i < moves_.length(); ++i) {
|
| - const MoveOperands& other_move = *moves_[i];
|
| - if (other_move.Blocks(source)) {
|
| - moves_[i]->set_src(destination);
|
| - } else if (other_move.Blocks(destination)) {
|
| - moves_[i]->set_src(source);
|
| - }
|
| - }
|
| -}
|
| -
|
| -
|
| -void ParallelMoveResolver::MoveMemoryToMemory(const Address& dst,
|
| - const Address& src) {
|
| - __ Comment("ParallelMoveResolver::MoveMemoryToMemory");
|
| - __ lw(TMP, src);
|
| - __ sw(TMP, dst);
|
| -}
|
| -
|
| -
|
| -void ParallelMoveResolver::StoreObject(const Address& dst, const Object& obj) {
|
| - __ Comment("ParallelMoveResolver::StoreObject");
|
| - __ LoadObject(TMP, obj);
|
| - __ sw(TMP, dst);
|
| -}
|
| -
|
| -
|
| -// Do not call or implement this function. Instead, use the form below that
|
| -// uses an offset from the frame pointer instead of an Address.
|
| -void ParallelMoveResolver::Exchange(Register reg, const Address& mem) {
|
| - UNREACHABLE();
|
| -}
|
| -
|
| -
|
| -// Do not call or implement this function. Instead, use the form below that
|
| -// uses offsets from the frame pointer instead of Addresses.
|
| -void ParallelMoveResolver::Exchange(const Address& mem1, const Address& mem2) {
|
| - UNREACHABLE();
|
| -}
|
| -
|
| -
|
| -void ParallelMoveResolver::Exchange(Register reg,
|
| - Register base_reg,
|
| - intptr_t stack_offset) {
|
| - ScratchRegisterScope tmp(this, reg);
|
| - __ mov(tmp.reg(), reg);
|
| - __ LoadFromOffset(reg, base_reg, stack_offset);
|
| - __ StoreToOffset(tmp.reg(), base_reg, stack_offset);
|
| -}
|
| -
|
| -
|
| -void ParallelMoveResolver::Exchange(Register base_reg1,
|
| - intptr_t stack_offset1,
|
| - Register base_reg2,
|
| - intptr_t stack_offset2) {
|
| - ScratchRegisterScope tmp1(this, kNoRegister);
|
| - ScratchRegisterScope tmp2(this, tmp1.reg());
|
| - __ LoadFromOffset(tmp1.reg(), base_reg1, stack_offset1);
|
| - __ LoadFromOffset(tmp2.reg(), base_reg2, stack_offset2);
|
| - __ StoreToOffset(tmp1.reg(), base_reg1, stack_offset2);
|
| - __ StoreToOffset(tmp2.reg(), base_reg2, stack_offset1);
|
| -}
|
| -
|
| -
|
| -void ParallelMoveResolver::SpillScratch(Register reg) {
|
| - __ Comment("ParallelMoveResolver::SpillScratch");
|
| - __ Push(reg);
|
| -}
|
| -
|
| -
|
| -void ParallelMoveResolver::RestoreScratch(Register reg) {
|
| - __ Comment("ParallelMoveResolver::RestoreScratch");
|
| - __ Pop(reg);
|
| -}
|
| -
|
| -
|
| -void ParallelMoveResolver::SpillFpuScratch(FpuRegister reg) {
|
| - __ Comment("ParallelMoveResolver::SpillFpuScratch");
|
| - __ AddImmediate(SP, -kDoubleSize);
|
| - __ StoreDToOffset(reg, SP, 0);
|
| -}
|
| -
|
| -
|
| -void ParallelMoveResolver::RestoreFpuScratch(FpuRegister reg) {
|
| - __ Comment("ParallelMoveResolver::RestoreFpuScratch");
|
| - __ LoadDFromOffset(reg, SP, 0);
|
| - __ AddImmediate(SP, kDoubleSize);
|
| -}
|
| -
|
| -
|
| -#undef __
|
| -
|
| -
|
| -} // namespace dart
|
| -
|
| -#endif // defined TARGET_ARCH_MIPS
|
|
|