Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(750)

Unified Diff: runtime/vm/intermediate_language_ia32.cc

Issue 552303005: Fix StoreIndexedInstr input representation requirements for Int32/Uint32 arrays. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: fix typo Created 6 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: runtime/vm/intermediate_language_ia32.cc
diff --git a/runtime/vm/intermediate_language_ia32.cc b/runtime/vm/intermediate_language_ia32.cc
index 19ffac37acb9e35d70d7659d7e1301b0391c83cc..12ebd078fafab6e3064a60469bb225fd19782059 100644
--- a/runtime/vm/intermediate_language_ia32.cc
+++ b/runtime/vm/intermediate_language_ia32.cc
@@ -161,10 +161,16 @@ void ConstantInstr::EmitNativeCode(FlowGraphCompiler* compiler) {
LocationSummary* UnboxedConstantInstr::MakeLocationSummary(Isolate* isolate,
bool opt) const {
const intptr_t kNumInputs = 0;
- const intptr_t kNumTemps = (constant_address() == 0) ? 1 : 0;
+ const intptr_t kNumTemps =
+ (constant_address() == 0) && (representation() != kUnboxedInt32) ? 1 : 0;
LocationSummary* locs = new(isolate) LocationSummary(
isolate, kNumInputs, kNumTemps, LocationSummary::kNoCall);
- locs->set_out(0, Location::RequiresFpuRegister());
+ if (representation() == kUnboxedDouble) {
+ locs->set_out(0, Location::RequiresFpuRegister());
+ } else {
+ ASSERT(representation() == kUnboxedInt32);
+ locs->set_out(0, Location::RequiresRegister());
+ }
if (kNumTemps == 1) {
locs->set_temp(0, Location::RequiresRegister());
}
@@ -1221,8 +1227,9 @@ Representation StoreIndexedInstr::RequiredInputRepresentation(
case kTypedDataUint16ArrayCid:
return kTagged;
case kTypedDataInt32ArrayCid:
+ return kUnboxedInt32;
case kTypedDataUint32ArrayCid:
- return value()->IsSmiValue() ? kTagged : kUnboxedMint;
+ return kUnboxedUint32;
case kTypedDataFloat32ArrayCid:
case kTypedDataFloat64ArrayCid:
return kUnboxedDouble;
@@ -1279,16 +1286,7 @@ LocationSummary* StoreIndexedInstr::MakeLocationSummary(Isolate* isolate,
break;
case kTypedDataInt32ArrayCid:
case kTypedDataUint32ArrayCid:
- // For smis, use a writable register because the value must be untagged
- // before storing. Mints are stored in registers pairs.
- if (value()->IsSmiValue()) {
- locs->set_in(2, Location::WritableRegister());
- } else {
- // We only move the lower 32-bits so we don't care where the high bits
- // are located.
- locs->set_in(2, Location::Pair(Location::RequiresRegister(),
- Location::Any()));
- }
+ locs->set_in(2, Location::RequiresRegister());
break;
case kTypedDataFloat32ArrayCid:
case kTypedDataFloat64ArrayCid:
@@ -1389,17 +1387,7 @@ void StoreIndexedInstr::EmitNativeCode(FlowGraphCompiler* compiler) {
}
case kTypedDataInt32ArrayCid:
case kTypedDataUint32ArrayCid:
- if (value()->IsSmiValue()) {
- ASSERT(RequiredInputRepresentation(2) == kTagged);
- Register value = locs()->in(2).reg();
- __ SmiUntag(value);
- __ movl(element_address, value);
- } else {
- ASSERT(RequiredInputRepresentation(2) == kUnboxedMint);
- PairLocation* value_pair = locs()->in(2).AsPairLocation();
- Register value1 = value_pair->At(0).reg();
- __ movl(element_address, value1);
- }
+ __ movl(element_address, locs()->in(2).reg());
break;
case kTypedDataFloat32ArrayCid:
__ movss(element_address, locs()->in(2).fpu_reg());
@@ -6394,54 +6382,7 @@ void UnaryUint32OpInstr::EmitNativeCode(FlowGraphCompiler* compiler) {
}
-LocationSummary* BoxUint32Instr::MakeLocationSummary(Isolate* isolate,
- bool opt) const {
- const intptr_t kNumInputs = 1;
- const intptr_t kNumTemps = 0;
- LocationSummary* summary = new(isolate) LocationSummary(
- isolate,
- kNumInputs,
- kNumTemps,
- ValueFitsSmi() ? LocationSummary::kNoCall
- : LocationSummary::kCallOnSlowPath);
- summary->set_in(0, Location::RequiresRegister());
- summary->set_out(0, ValueFitsSmi() ? Location::SameAsFirstInput()
- : Location::RequiresRegister());
- return summary;
-}
-
-
-void BoxUint32Instr::EmitNativeCode(FlowGraphCompiler* compiler) {
- Register value = locs()->in(0).reg();
- Register out = locs()->out(0).reg();
-
- Label not_smi, done;
-
- if (ValueFitsSmi()) {
- ASSERT(value == out);
- __ SmiTag(value);
- } else {
- ASSERT(value != out);
- __ cmpl(value, Immediate(kSmiMax));
- __ j(ABOVE, &not_smi);
- // Smi.
- __ movl(out, value);
- __ SmiTag(out);
- __ jmp(&done);
- __ Bind(&not_smi);
- // Allocate a mint.
- BoxAllocationSlowPath::Allocate(
- compiler, this, compiler->mint_class(), out, kNoRegister);
- // Copy low word into mint.
- __ movl(FieldAddress(out, Mint::value_offset()), value);
- // Zero high word.
- __ movl(FieldAddress(out, Mint::value_offset() + kWordSize), Immediate(0));
- __ Bind(&done);
- }
-}
-
-
-LocationSummary* BoxInt32Instr::MakeLocationSummary(Isolate* isolate,
+LocationSummary* BoxIntNInstr::MakeLocationSummary(Isolate* isolate,
bool opt) const {
const intptr_t kNumInputs = 1;
const intptr_t kNumTemps = 0;
@@ -6449,25 +6390,32 @@ LocationSummary* BoxInt32Instr::MakeLocationSummary(Isolate* isolate,
isolate, kNumInputs, kNumTemps,
ValueFitsSmi() ? LocationSummary::kNoCall
: LocationSummary::kCallOnSlowPath);
- summary->set_in(0, ValueFitsSmi() ? Location::RequiresRegister()
- : Location::WritableRegister());
+ const bool needs_writable_input = ValueFitsSmi() ||
+ (from_representation() == kUnboxedUint32);
+ summary->set_in(0, needs_writable_input ? Location::RequiresRegister()
+ : Location::WritableRegister());
summary->set_out(0, ValueFitsSmi() ? Location::SameAsFirstInput()
: Location::RequiresRegister());
return summary;
}
-void BoxInt32Instr::EmitNativeCode(FlowGraphCompiler* compiler) {
- Register value = locs()->in(0).reg();
- Register out = locs()->out(0).reg();
+void BoxIntNInstr::EmitNativeCode(FlowGraphCompiler* compiler) {
+ const Register value = locs()->in(0).reg();
+ const Register out = locs()->out(0).reg();
- if (out != value) {
- __ movl(out, value);
- }
- __ shll(out, Immediate(1));
+ __ MoveRegister(out, value);
+ __ shll(out, Immediate(kSmiTagSize));
if (!ValueFitsSmi()) {
Label done;
- __ j(NO_OVERFLOW, &done);
+ ASSERT(value != out);
+ if (from_representation() == kUnboxedInt32) {
+ __ j(NO_OVERFLOW, &done);
+ } else {
+ __ testl(value, Immediate(0xC0000000));
+ __ j(ZERO, &done);
+ }
+
// Allocate a mint.
// Value input is writable register and has to be manually preserved
// on the slow path.
@@ -6475,126 +6423,102 @@ void BoxInt32Instr::EmitNativeCode(FlowGraphCompiler* compiler) {
BoxAllocationSlowPath::Allocate(
compiler, this, compiler->mint_class(), out, kNoRegister);
__ movl(FieldAddress(out, Mint::value_offset()), value);
- __ sarl(value, Immediate(31)); // Sign extend.
- __ movl(FieldAddress(out, Mint::value_offset() + kWordSize), value);
+ if (from_representation() == kUnboxedInt32) {
+ __ sarl(value, Immediate(31)); // Sign extend.
+ __ movl(FieldAddress(out, Mint::value_offset() + kWordSize), value);
+ } else {
+ __ movl(FieldAddress(out, Mint::value_offset() + kWordSize),
+ Immediate(0));
+ }
__ Bind(&done);
}
}
-LocationSummary* UnboxUint32Instr::MakeLocationSummary(Isolate* isolate,
+LocationSummary* UnboxIntNInstr::MakeLocationSummary(Isolate* isolate,
bool opt) const {
const intptr_t value_cid = value()->Type()->ToCid();
const intptr_t kNumInputs = 1;
- const intptr_t kNumTemps =
- ((value_cid == kMintCid) || (value_cid == kSmiCid)) ? 0 : 1;
- LocationSummary* summary = new(isolate) LocationSummary(
- isolate, kNumInputs, kNumTemps, LocationSummary::kNoCall);
- summary->set_in(0, Location::RequiresRegister());
- if (kNumTemps > 0) {
- summary->set_temp(0, Location::RequiresRegister());
- }
- summary->set_out(0, Location::SameAsFirstInput());
- return summary;
-}
+ intptr_t kNumTemps = 0;
-
-void UnboxUint32Instr::EmitNativeCode(FlowGraphCompiler* compiler) {
- const intptr_t value_cid = value()->Type()->ToCid();
- const Register value = locs()->in(0).reg();
- ASSERT(value == locs()->out(0).reg());
-
- // TODO(johnmccutchan): Emit better code for constant inputs.
- if (value_cid == kMintCid) {
- __ movl(value, FieldAddress(value, Mint::value_offset()));
- } else if (value_cid == kSmiCid) {
- __ SmiUntag(value);
- } else {
- Register temp = locs()->temp(0).reg();
- Label* deopt = compiler->AddDeoptStub(deopt_id_,
- ICData::kDeoptUnboxInteger);
- Label is_smi, done;
- __ testl(value, Immediate(kSmiTagMask));
- __ j(ZERO, &is_smi);
- __ CompareClassId(value, kMintCid, temp);
- __ j(NOT_EQUAL, deopt);
- __ movl(value, FieldAddress(value, Mint::value_offset()));
- __ jmp(&done);
- __ Bind(&is_smi);
- __ SmiUntag(value);
- __ Bind(&done);
+ if (CanDeoptimize()) {
+ if ((value_cid != kSmiCid) &&
+ (value_cid != kMintCid) &&
+ !is_truncating()) {
+ kNumTemps = 2;
+ } else {
+ kNumTemps = 1;
+ }
}
-}
-
-LocationSummary* UnboxInt32Instr::MakeLocationSummary(Isolate* isolate,
- bool opt) const {
- const intptr_t value_cid = value()->Type()->ToCid();
- const intptr_t kNumInputs = 1;
- const intptr_t kNumTemps = CanDeoptimize() ? 1 : 0;
LocationSummary* summary = new(isolate) LocationSummary(
isolate, kNumInputs, kNumTemps, LocationSummary::kNoCall);
summary->set_in(0, Location::RequiresRegister());
- if (kNumTemps > 0) {
- summary->set_temp(0, Location::RequiresRegister());
+ for (int i = 0; i < kNumTemps; i++) {
+ summary->set_temp(i, Location::RequiresRegister());
}
- summary->set_out(0, (value_cid == kSmiCid) ? Location::SameAsFirstInput()
- : Location::RequiresRegister());
+ summary->set_out(0, ((value_cid == kSmiCid) || (value_cid != kMintCid)) ?
+ Location::SameAsFirstInput() : Location::RequiresRegister());
return summary;
}
static void LoadInt32FromMint(FlowGraphCompiler* compiler,
- Register mint,
Register result,
+ const Address& lo,
+ const Address& hi,
Register temp,
Label* deopt) {
- __ movl(result, FieldAddress(mint, Mint::value_offset()));
+ __ movl(result, lo);
if (deopt != NULL) {
+ ASSERT(temp != result);
__ movl(temp, result);
__ sarl(temp, Immediate(31));
- __ cmpl(temp, FieldAddress(mint, Mint::value_offset() + kWordSize));
+ __ cmpl(temp, hi);
__ j(NOT_EQUAL, deopt);
}
}
-void UnboxInt32Instr::EmitNativeCode(FlowGraphCompiler* compiler) {
+void UnboxIntNInstr::EmitNativeCode(FlowGraphCompiler* compiler) {
const intptr_t value_cid = value()->Type()->ToCid();
- const Register value = locs()->in(0).reg();
+ Register value = locs()->in(0).reg();
const Register result = locs()->out(0).reg();
+ const Register temp = CanDeoptimize() ? locs()->temp(0).reg() : kNoRegister;
+ Label* deopt = CanDeoptimize() ?
+ compiler->AddDeoptStub(deopt_id_, ICData::kDeoptUnboxInteger) : NULL;
+ Label* out_of_range = !is_truncating() ? deopt : NULL;
- // TODO(johnmccutchan): Emit better code for constant inputs.
- if (value_cid == kMintCid) {
- Register temp = CanDeoptimize() ? locs()->temp(0).reg() : kNoRegister;
- Label* deopt = CanDeoptimize() ?
- compiler->AddDeoptStub(deopt_id_, ICData::kDeoptUnboxInteger) : NULL;
+ const intptr_t lo_offset = Mint::value_offset();
+ const intptr_t hi_offset = Mint::value_offset() + kWordSize;
+
+ if (value_cid == kSmiCid) {
+ ASSERT(value == result);
+ __ SmiUntag(value);
+ } else if (value_cid == kMintCid) {
+ ASSERT((value != result) || (out_of_range == NULL));
LoadInt32FromMint(compiler,
- value,
result,
+ FieldAddress(value, lo_offset),
+ FieldAddress(value, hi_offset),
temp,
- deopt);
- } else if (value_cid == kSmiCid) {
- ASSERT(value == result);
- __ SmiUntag(value);
+ out_of_range);
} else {
- Register temp = locs()->temp(0).reg();
- Label* deopt = compiler->AddDeoptStub(deopt_id_,
- ICData::kDeoptUnboxInteger);
- Label is_smi, done;
- __ testl(value, Immediate(kSmiTagMask));
- __ j(ZERO, &is_smi);
- __ CompareClassId(value, kMintCid, temp);
+ ASSERT(value == result);
+ Label done;
+ __ SmiUntagOrCheckClass(value, kMintCid, temp, &done);
__ j(NOT_EQUAL, deopt);
+ if (out_of_range != NULL) {
+ Register value_temp = locs()->temp(1).reg();
+ __ movl(value_temp, value);
+ value = value_temp;
+ }
LoadInt32FromMint(compiler,
- value,
result,
+ Address(value, TIMES_2, lo_offset),
+ Address(value, TIMES_2, hi_offset),
temp,
- deopt);
- __ movl(value, FieldAddress(value, Mint::value_offset()));
- __ jmp(&done);
- __ Bind(&is_smi);
- __ SmiUntag(value);
+ out_of_range);
__ Bind(&done);
}
}

Powered by Google App Engine
This is Rietveld 408576698