fix int-to-long conversions from memory on x86_32

The instruction for 32-bit-to-64-bit sign extension on x86_32 requires
that the input value be placed in EAX and the sign extension in EDX.
However, the compiler can get confused if the input value is in memory
addressed via one of those registers and doesn't know how to move it.
This patch works around that limitation by doing the move explicitly
in MemoryEvent::compile if necessary.
This commit is contained in:
Joel Dice 2013-01-26 17:15:15 -07:00
parent 1a44ec9eef
commit 70a7a50a49

View File

@ -2003,6 +2003,13 @@ class MemorySite: public Site {
} }
} }
bool conflicts(const SiteMask& mask) {
return (mask.typeMask & (1 << RegisterOperand)) != 0
and (((1 << base) & mask.registerMask) == 0
or (index != NoRegister
and ((1 << index) & mask.registerMask) == 0));
}
virtual bool match(Context* c, const SiteMask& mask) { virtual bool match(Context* c, const SiteMask& mask) {
assert(c, acquired); assert(c, acquired);
@ -4574,8 +4581,20 @@ class OperationEvent: public Event {
void void
appendOperation(Context* c, Operation op) appendOperation(Context* c, Operation op)
{ {
append append(c, new(c->zone) OperationEvent(c, op));
(c, new(c->zone) OperationEvent(c, op)); }
void
moveIfConflict(Context* c, Value* v, MemorySite* s)
{
if (v->reads) {
SiteMask mask(1 << RegisterOperand, ~0, AnyFrameIndex);
v->reads->intersect(&mask);
if (s->conflicts(mask)) {
maybeMove(c, v->reads, true, false);
removeSite(c, v, s);
}
}
} }
class MemoryEvent: public Event { class MemoryEvent: public Event {
@ -4626,22 +4645,24 @@ class MemoryEvent: public Event {
popRead(c, this, index); popRead(c, this, index);
} }
Site* site = memorySite MemorySite* site = memorySite
(c, baseRegister, displacement, indexRegister, scale); (c, baseRegister, displacement, indexRegister, scale);
Site* low; MemorySite* low;
if (result->nextWord != result) { if (result->nextWord != result) {
Site* high = site->copyHigh(c); MemorySite* high = static_cast<MemorySite*>(site->copyHigh(c));
low = site->copyLow(c); low = static_cast<MemorySite*>(site->copyLow(c));
result->nextWord->target = high; result->nextWord->target = high;
addSite(c, result->nextWord, high); addSite(c, result->nextWord, high);
moveIfConflict(c, result->nextWord, high);
} else { } else {
low = site; low = site;
} }
result->target = low; result->target = low;
addSite(c, result, low); addSite(c, result, low);
moveIfConflict(c, result, low);
} }
Value* base; Value* base;