Chromium Code Reviews| Index: runtime/vm/intermediate_language_arm.cc |
| =================================================================== |
| --- runtime/vm/intermediate_language_arm.cc (revision 36577) |
| +++ runtime/vm/intermediate_language_arm.cc (working copy) |
| @@ -1138,6 +1138,34 @@ |
| } |
| +static bool CanBeImmediateIndex(Value* value, |
| + intptr_t cid, |
| + bool is_external, |
| + bool is_load) { |
| + ConstantInstr* constant = value->definition()->AsConstant(); |
| + if ((constant == NULL) || !Assembler::IsSafeSmi(constant->value())) { |
| + return false; |
| + } |
| + const int64_t index = Smi::Cast(constant->value()).AsInt64Value(); |
| + const intptr_t scale = Instance::ElementSizeFor(cid); |
| + const int64_t offset = index * scale + |
| + (is_external ? 0 : (Instance::DataOffsetFor(cid) - kHeapObjectTag)); |
| + if (!Utils::IsAbsoluteUint(12, offset)) { |
| + return false; |
| + } |
| + int32_t offset_mask = 0; |
| + if (is_load) { |
| + return Address::CanHoldLoadOffset(Address::OperandSizeFor(cid), |
| + offset, |
| + &offset_mask); |
| + } else { |
| + return Address::CanHoldStoreOffset(Address::OperandSizeFor(cid), |
| + offset, |
| + &offset_mask); |
| + } |
| +} |
| + |
| + |
| LocationSummary* LoadIndexedInstr::MakeLocationSummary(Isolate* isolate, |
| bool opt) const { |
| const intptr_t kNumInputs = 2; |
| @@ -1145,13 +1173,11 @@ |
| LocationSummary* locs = new(isolate) LocationSummary( |
| isolate, kNumInputs, kNumTemps, LocationSummary::kNoCall); |
| locs->set_in(0, Location::RequiresRegister()); |
| - // The smi index is either untagged (element size == 1), or it is left smi |
| - // tagged (for all element sizes > 1). |
| - // TODO(regis): Revisit and see if the index can be immediate. |
| - if (index_scale() == 2 && IsExternal()) { |
| - locs->set_in(1, Location::RequiresRegister()); |
| + if (CanBeImmediateIndex(index(), class_id(), IsExternal(), true /*load*/)) { |
| + // CanBeImmediateIndex must return false for unsafe smis. |
| + locs->set_in(1, Location::Constant(index()->BoundConstant())); |
| } else { |
| - locs->set_in(1, Location::WritableRegister()); |
| + locs->set_in(1, Location::RequiresRegister()); |
| } |
| if ((representation() == kUnboxedDouble) || |
| (representation() == kUnboxedFloat32x4) || |
| @@ -1176,36 +1202,79 @@ |
| } |
| +static Address ElementAddressForIntIndex(bool is_external, |
| + intptr_t cid, |
| + intptr_t index_scale, |
| + Register array, |
| + intptr_t index) { |
| + const int64_t offset = static_cast<int64_t>(index) * index_scale + |
| + (is_external ? 0 : (Instance::DataOffsetFor(cid) - kHeapObjectTag)); |
| + ASSERT(Utils::IsInt(32, offset)); |
| + return Address(array, static_cast<int32_t>(offset)); |
| +} |
| + |
| + |
| +static Address ElementAddressForRegIndex(Assembler* assembler, |
| + bool is_load, |
| + bool is_external, |
| + intptr_t cid, |
| + intptr_t index_scale, |
| + Register array, |
| + Register index) { |
| + // Note that index is expected smi-tagged, (i.e, LSL 1) for all arrays. |
| + const intptr_t shift = Utils::ShiftForPowerOfTwo(index_scale) - kSmiTagShift; |
| + int32_t offset = |
| + is_external ? 0 : (Instance::DataOffsetFor(cid) - kHeapObjectTag); |
| + const OperandSize size = Address::OperandSizeFor(cid); |
| + const Register base = is_load ? IP : index; |
|
zra
2014/05/23 20:03:24
Maybe assert that array and index can't be IP.
regis
2014/05/23 20:21:35
Registers array and index are defined by the regis
|
| + if ((offset != 0) || |
| + (size == kSWord) || (size == kDWord) || (size == kRegList)) { |
| + if (shift < 0) { |
| + ASSERT(shift == -1); |
| + assembler->add(base, array, ShifterOperand(index, ASR, 1)); |
| + } else { |
| + assembler->add(base, array, ShifterOperand(index, LSL, shift)); |
| + } |
| + } else { |
| + if (shift < 0) { |
| + ASSERT(shift == -1); |
| + return Address(array, index, ASR, 1); |
| + } else { |
| + return Address(array, index, LSL, shift); |
| + } |
| + } |
| + int32_t offset_mask = 0; |
| + if ((is_load && !Address::CanHoldLoadOffset(size, |
| + offset, |
| + &offset_mask)) || |
| + (!is_load && !Address::CanHoldStoreOffset(size, |
| + offset, |
| + &offset_mask))) { |
| + assembler->AddImmediate(base, offset & ~offset_mask); |
| + offset = offset & offset_mask; |
| + } |
| + return Address(base, offset); |
| +} |
| + |
| + |
| void LoadIndexedInstr::EmitNativeCode(FlowGraphCompiler* compiler) { |
| + Register array = locs()->in(0).reg(); |
|
zra
2014/05/23 20:03:24
const
regis
2014/05/23 20:21:35
Done.
|
| + Location index = locs()->in(1); |
| + |
| + Address element_address(kNoRegister, 0); |
| + element_address = index.IsRegister() |
| + ? ElementAddressForRegIndex(compiler->assembler(), |
| + true, // Load. |
| + IsExternal(), class_id(), index_scale(), |
| + array, index.reg()) |
| + : ElementAddressForIntIndex(IsExternal(), class_id(), index_scale(), |
| + array, Smi::Cast(index.constant()).Value()); |
| + // Warning: element_address may use register IP as base. |
| + |
| if ((representation() == kUnboxedDouble) || |
| (representation() == kUnboxedFloat32x4) || |
| (representation() == kUnboxedInt32x4) || |
| (representation() == kUnboxedFloat64x2)) { |
| - const Register array = locs()->in(0).reg(); |
| - const Register idx = locs()->in(1).reg(); |
| - switch (index_scale()) { |
| - case 1: |
| - __ add(idx, array, ShifterOperand(idx, ASR, kSmiTagSize)); |
| - break; |
| - case 4: |
| - __ add(idx, array, ShifterOperand(idx, LSL, 1)); |
| - break; |
| - case 8: |
| - __ add(idx, array, ShifterOperand(idx, LSL, 2)); |
| - break; |
| - case 16: |
| - __ add(idx, array, ShifterOperand(idx, LSL, 3)); |
| - break; |
| - default: |
| - // Case 2 is not reachable: We don't have unboxed 16-bit sized loads. |
| - UNREACHABLE(); |
| - } |
| - if (!IsExternal()) { |
| - ASSERT(this->array()->definition()->representation() == kTagged); |
| - __ AddImmediate(idx, |
| - Instance::DataOffsetFor(class_id()) - kHeapObjectTag); |
| - } |
| - Address element_address(idx); |
| const QRegister result = locs()->out(0).fpu_reg(); |
| const DRegister dresult0 = EvenDRegisterOf(result); |
| switch (class_id()) { |
| @@ -1221,7 +1290,8 @@ |
| case kTypedDataFloat64x2ArrayCid: |
| case kTypedDataInt32x4ArrayCid: |
| case kTypedDataFloat32x4ArrayCid: |
| - __ vldmd(IA, idx, dresult0, 2); |
| + ASSERT(element_address.Equals(Address(IP))); |
| + __ vldmd(IA, IP, dresult0, 2); |
| break; |
| default: |
| UNREACHABLE(); |
| @@ -1229,44 +1299,6 @@ |
| return; |
| } |
| - const Register array = locs()->in(0).reg(); |
| - Location index = locs()->in(1); |
| - ASSERT(index.IsRegister()); // TODO(regis): Revisit. |
| - Address element_address(kNoRegister, 0); |
| - // Note that index is expected smi-tagged, (i.e, times 2) for all arrays |
| - // with index scale factor > 1. E.g., for Uint8Array and OneByteString the |
| - // index is expected to be untagged before accessing. |
| - ASSERT(kSmiTagShift == 1); |
| - const intptr_t offset = IsExternal() |
| - ? 0 |
| - : Instance::DataOffsetFor(class_id()) - kHeapObjectTag; |
| - switch (index_scale()) { |
| - case 1: { |
| - __ add(index.reg(), array, ShifterOperand(index.reg(), ASR, kSmiTagSize)); |
| - element_address = Address(index.reg(), offset); |
| - break; |
| - } |
| - case 2: { |
| - // No scaling needed, since index is a smi. |
| - if (!IsExternal()) { |
| - __ AddImmediate(index.reg(), index.reg(), |
| - Instance::DataOffsetFor(class_id()) - kHeapObjectTag); |
| - element_address = Address(array, index.reg(), LSL, 0); |
| - } else { |
| - element_address = Address(array, index.reg(), LSL, 0); |
| - } |
| - break; |
| - } |
| - case 4: { |
| - __ add(index.reg(), array, ShifterOperand(index.reg(), LSL, 1)); |
| - element_address = Address(index.reg(), offset); |
| - break; |
| - } |
| - // Cases 8 and 16 are only for unboxed values and are handled above. |
| - default: |
| - UNREACHABLE(); |
| - } |
| - |
| if (representation() == kUnboxedMint) { |
| ASSERT(locs()->out(0).IsPairLocation()); |
| PairLocation* result_pair = locs()->out(0).AsPairLocation(); |
| @@ -1278,16 +1310,16 @@ |
| __ ldr(result1, element_address); |
| // Sign extend into high word. |
| __ SignFill(result2, result1); |
| - break; |
| + break; |
| case kTypedDataUint32ArrayCid: |
| // Load low word. |
| __ ldr(result1, element_address); |
| // Zero high word. |
| __ eor(result2, result2, ShifterOperand(result2)); |
| - break; |
| + break; |
| default: |
| UNREACHABLE(); |
| - break; |
| + break; |
| } |
| return; |
| } |
| @@ -1390,10 +1422,12 @@ |
| LocationSummary* locs = new(isolate) LocationSummary( |
| isolate, kNumInputs, kNumTemps, LocationSummary::kNoCall); |
| locs->set_in(0, Location::RequiresRegister()); |
| - // The smi index is either untagged (element size == 1), or it is left smi |
| - // tagged (for all element sizes > 1). |
| - // TODO(regis): Revisit and see if the index can be immediate. |
| - locs->set_in(1, Location::WritableRegister()); |
| + if (CanBeImmediateIndex(index(), class_id(), IsExternal(), false /*store*/)) { |
| + // CanBeImmediateIndex must return false for unsafe smis. |
| + locs->set_in(1, Location::Constant(index()->BoundConstant())); |
| + } else { |
| + locs->set_in(1, Location::WritableRegister()); |
| + } |
| switch (class_id()) { |
| case kArrayCid: |
| locs->set_in(2, ShouldEmitStoreBarrier() |
| @@ -1442,98 +1476,17 @@ |
| void StoreIndexedInstr::EmitNativeCode(FlowGraphCompiler* compiler) { |
| - if ((class_id() == kTypedDataFloat32ArrayCid) || |
| - (class_id() == kTypedDataFloat64ArrayCid) || |
| - (class_id() == kTypedDataFloat32x4ArrayCid) || |
| - (class_id() == kTypedDataFloat64x2ArrayCid) || |
| - (class_id() == kTypedDataInt32x4ArrayCid)) { |
| - const Register array = locs()->in(0).reg(); |
| - const Register idx = locs()->in(1).reg(); |
| - Location value = locs()->in(2); |
| - switch (index_scale()) { |
| - case 1: |
| - __ add(idx, array, ShifterOperand(idx, ASR, kSmiTagSize)); |
| - break; |
| - case 4: |
| - __ add(idx, array, ShifterOperand(idx, LSL, 1)); |
| - break; |
| - case 8: |
| - __ add(idx, array, ShifterOperand(idx, LSL, 2)); |
| - break; |
| - case 16: |
| - __ add(idx, array, ShifterOperand(idx, LSL, 3)); |
| - break; |
| - default: |
| - // Case 2 is not reachable: We don't have unboxed 16-bit sized loads. |
| - UNREACHABLE(); |
| - } |
| - if (!IsExternal()) { |
| - ASSERT(this->array()->definition()->representation() == kTagged); |
| - __ AddImmediate(idx, |
| - Instance::DataOffsetFor(class_id()) - kHeapObjectTag); |
| - } |
| - switch (class_id()) { |
| - case kTypedDataFloat32ArrayCid: { |
| - const SRegister value_reg = |
| - EvenSRegisterOf(EvenDRegisterOf(value.fpu_reg())); |
| - __ StoreSToOffset(value_reg, idx, 0); |
| - break; |
| - } |
| - case kTypedDataFloat64ArrayCid: { |
| - const DRegister value_reg = EvenDRegisterOf(value.fpu_reg()); |
| - __ StoreDToOffset(value_reg, idx, 0); |
| - break; |
| - } |
| - case kTypedDataFloat64x2ArrayCid: |
| - case kTypedDataInt32x4ArrayCid: |
| - case kTypedDataFloat32x4ArrayCid: { |
| - const DRegister value_reg = EvenDRegisterOf(value.fpu_reg()); |
| - __ vstmd(IA, idx, value_reg, 2); |
| - break; |
| - } |
| - default: |
| - UNREACHABLE(); |
| - } |
| - return; |
| - } |
| - |
| - const Register array = locs()->in(0).reg(); |
| + Register array = locs()->in(0).reg(); |
|
zra
2014/05/23 20:03:24
const
regis
2014/05/23 20:21:35
Done.
|
| Location index = locs()->in(1); |
| Address element_address(kNoRegister, 0); |
| - ASSERT(index.IsRegister()); // TODO(regis): Revisit. |
| - // Note that index is expected smi-tagged, (i.e, times 2) for all arrays |
| - // with index scale factor > 1. E.g., for Uint8Array and OneByteString the |
| - // index is expected to be untagged before accessing. |
| - ASSERT(kSmiTagShift == 1); |
| - const intptr_t offset = IsExternal() |
| - ? 0 |
| - : Instance::DataOffsetFor(class_id()) - kHeapObjectTag; |
| - switch (index_scale()) { |
| - case 1: { |
| - __ add(index.reg(), array, ShifterOperand(index.reg(), ASR, kSmiTagSize)); |
| - element_address = Address(index.reg(), offset); |
| - break; |
| - } |
| - case 2: { |
| - // No scaling needed, since index is a smi. |
| - if (!IsExternal()) { |
| - __ AddImmediate(index.reg(), index.reg(), offset); |
| - element_address = Address(array, index.reg(), LSL, 0); |
| - } else { |
| - element_address = Address(array, index.reg(), LSL, 0); |
| - } |
| - break; |
| - } |
| - case 4: { |
| - __ add(index.reg(), array, ShifterOperand(index.reg(), LSL, 1)); |
| - element_address = Address(index.reg(), offset); |
| - break; |
| - } |
| - // Cases 8 and 16 are only for unboxed values and are handled above. |
| - default: |
| - UNREACHABLE(); |
| - } |
| + element_address = index.IsRegister() |
| + ? ElementAddressForRegIndex(compiler->assembler(), |
| + false, // Store. |
| + IsExternal(), class_id(), index_scale(), |
| + array, index.reg()) |
| + : ElementAddressForIntIndex(IsExternal(), class_id(), index_scale(), |
| + array, Smi::Cast(index.constant()).Value()); |
| switch (class_id()) { |
| case kArrayCid: |
| @@ -1612,6 +1565,25 @@ |
| } |
| break; |
| } |
| + case kTypedDataFloat32ArrayCid: { |
| + const SRegister value_reg = |
| + EvenSRegisterOf(EvenDRegisterOf(locs()->in(2).fpu_reg())); |
| + __ vstrs(value_reg, element_address); |
| + break; |
| + } |
| + case kTypedDataFloat64ArrayCid: { |
| + const DRegister value_reg = EvenDRegisterOf(locs()->in(2).fpu_reg()); |
| + __ vstrd(value_reg, element_address); |
| + break; |
| + } |
| + case kTypedDataFloat64x2ArrayCid: |
| + case kTypedDataInt32x4ArrayCid: |
| + case kTypedDataFloat32x4ArrayCid: { |
| + ASSERT(element_address.Equals(Address(index.reg()))); |
| + const DRegister value_reg = EvenDRegisterOf(locs()->in(2).fpu_reg()); |
| + __ vstmd(IA, index.reg(), value_reg, 2); |
| + break; |
| + } |
| default: |
| UNREACHABLE(); |
| } |