fix native call argument marshalling regression

This commit is contained in:
Joel Dice 2007-12-19 17:02:32 -07:00
parent 9b08fe083c
commit 7dd81c803a
3 changed files with 93 additions and 64 deletions

View File

@ -28,7 +28,7 @@ src = src
classpath = classpath classpath = classpath
test = test test = test
input = $(test-build)/Threads.class input = $(test-build)/Enums.class
build-cxx = g++ build-cxx = g++
build-cc = gcc build-cc = gcc

View File

@ -3000,10 +3000,10 @@ finish(MyThread* t, Compiler* c, object method, Vector* objectPool,
if (false and if (false and
strcmp(reinterpret_cast<const char*> strcmp(reinterpret_cast<const char*>
(&byteArrayBody(t, className(t, methodClass(t, method)), 0)), (&byteArrayBody(t, className(t, methodClass(t, method)), 0)),
"java/util/Properties$Parser") == 0 and "java/lang/reflect/Method") == 0 and
strcmp(reinterpret_cast<const char*> strcmp(reinterpret_cast<const char*>
(&byteArrayBody(t, methodName(t, method), 0)), (&byteArrayBody(t, methodName(t, method), 0)),
"parse") == 0) "invoke") == 0)
{ {
asm("int3"); asm("int3");
} }
@ -3178,7 +3178,12 @@ invokeNative2(MyThread* t, object method)
} break; } break;
case POINTER_TYPE: { case POINTER_TYPE: {
args[argOffset++] = *sp ? reinterpret_cast<uintptr_t>(sp--) : 0; if (*sp) {
args[argOffset++] = reinterpret_cast<uintptr_t>(sp);
} else {
args[argOffset++] = 0;
}
-- sp;
} break; } break;
default: abort(t); default: abort(t);

View File

@ -50,6 +50,7 @@ class RegisterOperand;
class MemoryOperand; class MemoryOperand;
class CodePromise; class CodePromise;
class MyPromise; class MyPromise;
class RegisterReference;
inline bool inline bool
isInt8(intptr_t v) isInt8(intptr_t v)
@ -127,10 +128,9 @@ class MyStack: public Stack {
class RegisterData { class RegisterData {
public: public:
RegisterData(): reserved(false), high(NoRegister) { } RegisterData(): reserved(false) { }
bool reserved; bool reserved;
Register high;
}; };
class Context { class Context {
@ -304,6 +304,9 @@ absolute(Context* c, MyPromise* v);
RegisterOperand* RegisterOperand*
register_(Context* c, RegisterReference*, SelectionType = DefaultSelection); register_(Context* c, RegisterReference*, SelectionType = DefaultSelection);
RegisterOperand*
register_(Context* c, Register, SelectionType = DefaultSelection);
MemoryOperand* MemoryOperand*
memory(Context* c, MyOperand* base, int displacement, memory(Context* c, MyOperand* base, int displacement,
MyOperand* index, unsigned scale, SelectionType = DefaultSelection); MyOperand* index, unsigned scale, SelectionType = DefaultSelection);
@ -403,23 +406,27 @@ release(Context* c, Register v)
fprintf(stderr, "release %d\n", v); fprintf(stderr, "release %d\n", v);
} }
c->registers[v].reserved = false; c->registers[v].reserved = false;
if (c->registers[v].high != NoRegister) {
release(c, c->registers[v].high);
}
} }
class RegisterReference { class RegisterReference {
public: public:
RegisterReference(Register v = NoRegister): value_(v) { } RegisterReference(Register v = NoRegister): value_(v), high_(NoRegister) { }
void acquire(Context* c) { void acquireValue(Context* c) {
value_ = ::acquire(c); value_ = ::acquire(c);
} }
void acquireHigh(Context* c) {
high_ = ::acquire(c);
}
void release(Context* c) { void release(Context* c) {
::release(c, value_); ::release(c, value_);
value_ = NoRegister; value_ = NoRegister;
if (high_ != NoRegister) {
::release(c, high_);
}
} }
Register value(Context* c UNUSED) { Register value(Context* c UNUSED) {
@ -427,7 +434,13 @@ class RegisterReference {
return value_; return value_;
} }
Register high(Context* c UNUSED) {
assert(c, high_ != NoRegister);
return high_;
}
Register value_; Register value_;
Register high_;
}; };
class RegisterOperand: public MyOperand { class RegisterOperand: public MyOperand {
@ -440,6 +453,10 @@ class RegisterOperand: public MyOperand {
return reference->value(c); return reference->value(c);
} }
Register high(Context* c) {
return reference->high(c);
}
virtual unsigned footprint(Context*) { virtual unsigned footprint(Context*) {
return (selection == S8Selection ? 8 : BytesPerWord); return (selection == S8Selection ? 8 : BytesPerWord);
} }
@ -453,11 +470,11 @@ class RegisterOperand: public MyOperand {
return this; return this;
} else { } else {
if (selection == S8Selection and BytesPerWord == 4 if (selection == S8Selection and BytesPerWord == 4
and c->registers[value(c)].high == NoRegister) and reference->high_ == NoRegister)
{ {
c->registers[value(c)].high = ::acquire(c); reference->acquireHigh(c);
} }
return register_(c, value(c), selection); return register_(c, reference, selection);
} }
} }
@ -632,15 +649,20 @@ absolute(Context* c, MyPromise* v)
} }
RegisterOperand* RegisterOperand*
register_(Context* c, Register v, SelectionType selection) register_(Context* c, RegisterReference* r, SelectionType selection)
{ {
assert(c, BytesPerWord != 4 or selection != S8Selection);
RegisterReference* r = new (c->zone.allocate(sizeof(RegisterReference)))
RegisterReference(v);
return new (c->zone.allocate(sizeof(RegisterOperand))) return new (c->zone.allocate(sizeof(RegisterOperand)))
RegisterOperand(r, selection); RegisterOperand(r, selection);
} }
RegisterOperand*
register_(Context* c, Register v, SelectionType selection)
{
RegisterReference* r = new (c->zone.allocate(sizeof(RegisterReference)))
RegisterReference(v);
return register_(c, r, selection);
}
MemoryOperand* MemoryOperand*
memory(Context* c, MyOperand* base, int displacement, memory(Context* c, MyOperand* base, int displacement,
MyOperand* index, unsigned scale, SelectionType selection) MyOperand* index, unsigned scale, SelectionType selection)
@ -743,7 +765,7 @@ class AcquireEvent: public Event {
{ } { }
virtual void run(Context* c) { virtual void run(Context* c) {
operand->reference->acquire(c); operand->reference->acquireValue(c);
} }
RegisterOperand* operand; RegisterOperand* operand;
@ -1043,25 +1065,24 @@ void
RegisterOperand::apply(Context* c, Operation operation) RegisterOperand::apply(Context* c, Operation operation)
{ {
assert(c, selection == DefaultSelection); assert(c, selection == DefaultSelection);
assert(c, value != NoRegister);
switch (operation) { switch (operation) {
case call: case call:
c->code.append(0xff); c->code.append(0xff);
c->code.append(0xd0 | value); c->code.append(0xd0 | value(c));
break; break;
case jmp: case jmp:
c->code.append(0xff); c->code.append(0xff);
c->code.append(0xe0 | value); c->code.append(0xe0 | value(c));
break; break;
case pop: case pop:
c->code.append(0x58 | value); c->code.append(0x58 | value(c));
break; break;
case push: case push:
c->code.append(0x50 | value); c->code.append(0x50 | value(c));
break; break;
default: abort(c); default: abort(c);
@ -1079,21 +1100,21 @@ RegisterOperand::accept(Context* c, Operation operation,
case add: case add:
rex(c); rex(c);
c->code.append(0x01); c->code.append(0x01);
c->code.append(0xc0 | (operand->value << 3) | value); c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
break; break;
case cmp: case cmp:
rex(c); rex(c);
c->code.append(0x39); c->code.append(0x39);
c->code.append(0xc0 | (operand->value << 3) | value); c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
break; break;
case mov: case mov:
if (value != operand->value or selection != operand->selection) { if (value(c) != operand->value(c) or selection != operand->selection) {
if (operand->selection == DefaultSelection) { if (operand->selection == DefaultSelection) {
rex(c); rex(c);
c->code.append(0x89); c->code.append(0x89);
c->code.append(0xc0 | (operand->value << 3) | value); c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
} else { } else {
switch (operand->selection) { switch (operand->selection) {
case S1Selection: case S1Selection:
@ -1115,7 +1136,7 @@ RegisterOperand::accept(Context* c, Operation operation,
default: abort(c); default: abort(c);
} }
c->code.append(0xc0 | (operand->value << 3) | value); c->code.append(0xc0 | (operand->value(c) << 3) | value(c));
} }
} }
break; break;
@ -1124,7 +1145,7 @@ RegisterOperand::accept(Context* c, Operation operation,
rex(c); rex(c);
c->code.append(0x0f); c->code.append(0x0f);
c->code.append(0xaf); c->code.append(0xaf);
c->code.append(0xc0 | (value << 3) | operand->value); c->code.append(0xc0 | (value(c) << 3) | operand->value(c));
break; break;
default: abort(c); default: abort(c);
@ -1144,11 +1165,11 @@ RegisterOperand::accept(Context* c, Operation operation,
rex(c); rex(c);
if (isInt8(operand->value)) { if (isInt8(operand->value)) {
c->code.append(0x83); c->code.append(0x83);
c->code.append(0xc0 | value); c->code.append(0xc0 | value(c));
c->code.append(operand->value); c->code.append(operand->value);
} else if (isInt32(operand->value)) { } else if (isInt32(operand->value)) {
c->code.append(0x81); c->code.append(0x81);
c->code.append(0xc0 | value); c->code.append(0xc0 | value(c));
c->code.append4(operand->value); c->code.append4(operand->value);
} else { } else {
abort(c); abort(c);
@ -1161,13 +1182,13 @@ RegisterOperand::accept(Context* c, Operation operation,
rex(c); rex(c);
if (isInt8(operand->value)) { if (isInt8(operand->value)) {
c->code.append(0x83); c->code.append(0x83);
c->code.append(0xe0 | value); c->code.append(0xe0 | value(c));
c->code.append(operand->value); c->code.append(operand->value);
} else { } else {
assert(c, isInt32(operand->value)); assert(c, isInt32(operand->value));
c->code.append(0x81); c->code.append(0x81);
c->code.append(0xe0 | value); c->code.append(0xe0 | value(c));
c->code.append(operand->value); c->code.append(operand->value);
} }
} break; } break;
@ -1177,13 +1198,13 @@ RegisterOperand::accept(Context* c, Operation operation,
rex(c); rex(c);
c->code.append(0x83); c->code.append(0x83);
c->code.append(0xf8 | value); c->code.append(0xf8 | value(c));
c->code.append(operand->value); c->code.append(operand->value);
} break; } break;
case mov: { case mov: {
rex(c); rex(c);
c->code.append(0xb8 | value); c->code.append(0xb8 | value(c));
c->code.appendAddress(operand->value); c->code.appendAddress(operand->value);
} break; } break;
@ -1192,12 +1213,12 @@ RegisterOperand::accept(Context* c, Operation operation,
rex(c); rex(c);
if (operand->value == 1) { if (operand->value == 1) {
c->code.append(0xd1); c->code.append(0xd1);
c->code.append(0xe0 | value); c->code.append(0xe0 | value(c));
} else { } else {
assert(c, isInt8(operand->value)); assert(c, isInt8(operand->value));
c->code.append(0xc1); c->code.append(0xc1);
c->code.append(0xe0 | value); c->code.append(0xe0 | value(c));
c->code.append(operand->value); c->code.append(operand->value);
} }
} }
@ -1208,11 +1229,11 @@ RegisterOperand::accept(Context* c, Operation operation,
rex(c); rex(c);
if (isInt8(operand->value)) { if (isInt8(operand->value)) {
c->code.append(0x83); c->code.append(0x83);
c->code.append(0xe8 | value); c->code.append(0xe8 | value(c));
c->code.append(operand->value); c->code.append(operand->value);
} else if (isInt32(operand->value)) { } else if (isInt32(operand->value)) {
c->code.append(0x81); c->code.append(0x81);
c->code.append(0xe8 | value); c->code.append(0xe8 | value(c));
c->code.append4(operand->value); c->code.append4(operand->value);
} else { } else {
abort(c); abort(c);
@ -1256,39 +1277,39 @@ RegisterOperand::accept(Context* c, Operation operation,
switch (operation) { switch (operation) {
case cmp: { case cmp: {
encode(c, 0x3b, value, operand, true); encode(c, 0x3b, value(c), operand, true);
} break; } break;
case mov: { case mov: {
if (operand->selection == DefaultSelection) { if (operand->selection == DefaultSelection) {
encode(c, 0x8b, value, operand, true); encode(c, 0x8b, value(c), operand, true);
} else { } else {
switch (operand->selection) { switch (operand->selection) {
case S1Selection: case S1Selection:
encode2(c, 0x0fbe, value, operand, true); encode2(c, 0x0fbe, value(c), operand, true);
break; break;
case S2Selection: case S2Selection:
encode2(c, 0x0fbf, value, operand, true); encode2(c, 0x0fbf, value(c), operand, true);
break; break;
case Z2Selection: case Z2Selection:
encode2(c, 0x0fb7, value, operand, true); encode2(c, 0x0fb7, value(c), operand, true);
break; break;
case S4Selection: case S4Selection:
encode(c, 0x63, value, operand, true); encode(c, 0x63, value(c), operand, true);
break; break;
case S8Selection: case S8Selection:
assert(c, selection == S8Selection); assert(c, selection == S8Selection);
register_(c, value)->accept register_(c, value(c))->accept
(c, mov, memory (c, mov, memory
(c, operand->base, operand->displacement, (c, operand->base, operand->displacement,
operand->index, operand->scale)); operand->index, operand->scale));
register_(c, c->registers[value].high)->accept register_(c, high(c))->accept
(c, mov, memory (c, mov, memory
(c, operand->base, operand->displacement + BytesPerWord, (c, operand->base, operand->displacement + BytesPerWord,
operand->index, operand->scale)); operand->index, operand->scale));
@ -1432,8 +1453,9 @@ AddressOperand::asRegister(Context* c)
RegisterOperand* tmp = temporary(c); RegisterOperand* tmp = temporary(c);
tmp->accept(c, mov, immediate(c, v)); tmp->accept(c, mov, immediate(c, v));
Register r = tmp->value(c);
tmp->release(c); tmp->release(c);
return tmp->value; return r;
} }
void void
@ -1478,8 +1500,9 @@ AbsoluteOperand::asRegister(Context* c)
{ {
RegisterOperand* tmp = temporary(c); RegisterOperand* tmp = temporary(c);
tmp->accept(c, mov, this); tmp->accept(c, mov, this);
Register v = tmp->value(c);
tmp->release(c); tmp->release(c);
return tmp->value; return v;
} }
void void
@ -1509,8 +1532,9 @@ MemoryOperand::asRegister(Context* c)
{ {
RegisterOperand* tmp = temporary(c); RegisterOperand* tmp = temporary(c);
tmp->accept(c, mov, this); tmp->accept(c, mov, this);
Register v = tmp->value(c);
tmp->release(c); tmp->release(c);
return tmp->value; return v;
} }
void void
@ -1559,11 +1583,11 @@ MemoryOperand::accept(Context* c, Operation operation,
switch (operation) { switch (operation) {
case and_: { case and_: {
encode(c, 0x21, operand->value, this, true); encode(c, 0x21, operand->value(c), this, true);
} break; } break;
case add: { case add: {
encode(c, 0x01, operand->value, this, true); encode(c, 0x01, operand->value(c), this, true);
} break; } break;
case div: { case div: {
@ -1575,7 +1599,7 @@ MemoryOperand::accept(Context* c, Operation operation,
c->code.append(0x99); c->code.append(0x99);
rex(c); rex(c);
c->code.append(0xf7); c->code.append(0xf7);
c->code.append(0xf8 | operand->value); c->code.append(0xf8 | operand->value(c));
accept(c, mov, ax); accept(c, mov, ax);
@ -1585,24 +1609,24 @@ MemoryOperand::accept(Context* c, Operation operation,
case mov: { case mov: {
if (selection == DefaultSelection) { if (selection == DefaultSelection) {
encode(c, 0x89, operand->value, this, true); encode(c, 0x89, operand->value(c), this, true);
} else { } else {
switch (selection) { switch (selection) {
case S1Selection: case S1Selection:
if (operand->value > rbx) { if (operand->value(c) > rbx) {
c->code.append(0x40); c->code.append(0x40);
} }
encode(c, 0x88, operand->value, this, false); encode(c, 0x88, operand->value(c), this, false);
break; break;
case S2Selection: case S2Selection:
case Z2Selection: case Z2Selection:
c->code.append(0x66); c->code.append(0x66);
encode(c, 0x89, operand->value, this, false); encode(c, 0x89, operand->value(c), this, false);
break; break;
case S4Selection: case S4Selection:
encode(c, 0x89, operand->value, this, false); encode(c, 0x89, operand->value(c), this, false);
break; break;
default: abort(c); default: abort(c);
@ -1621,7 +1645,7 @@ MemoryOperand::accept(Context* c, Operation operation,
} break; } break;
case or_: { case or_: {
encode(c, 0x09, operand->value, this, true); encode(c, 0x09, operand->value(c), this, true);
} break; } break;
case rem: { case rem: {
@ -1633,7 +1657,7 @@ MemoryOperand::accept(Context* c, Operation operation,
c->code.append(0x99); c->code.append(0x99);
rex(c); rex(c);
c->code.append(0xf7); c->code.append(0xf7);
c->code.append(0xf8 | operand->value); c->code.append(0xf8 | operand->value(c));
accept(c, mov, dx); accept(c, mov, dx);
@ -1663,11 +1687,11 @@ MemoryOperand::accept(Context* c, Operation operation,
} break; } break;
case sub: { case sub: {
encode(c, 0x29, operand->value, this, true); encode(c, 0x29, operand->value(c), this, true);
} break; } break;
case xor_: { case xor_: {
encode(c, 0x31, operand->value, this, true); encode(c, 0x31, operand->value(c), this, true);
} break; } break;
default: abort(c); default: abort(c);