Merge pull request #200 from dicej/stack-signal

fix a few stack walking and signal handling bugs
This commit is contained in:
Joshua Warner 2014-03-14 10:45:18 -06:00
commit 56ec7fc1f9
7 changed files with 36 additions and 16 deletions

View File

@ -88,7 +88,7 @@ virtual unsigned alignFrameSize(unsigned sizeInWords) = 0;
virtual void nextFrame(void* start, unsigned size, unsigned footprint, virtual void nextFrame(void* start, unsigned size, unsigned footprint,
void* link, bool mostRecent, void* link, bool mostRecent,
unsigned targetParameterFootprint, void** ip, int targetParameterFootprint, void** ip,
void** stack) = 0; void** stack) = 0;
virtual void* frameIp(void* stack) = 0; virtual void* frameIp(void* stack) = 0;
virtual unsigned frameHeaderSize() = 0; virtual unsigned frameHeaderSize() = 0;
@ -138,4 +138,4 @@ virtual void release() = 0;
} // namespace codegen } // namespace codegen
} // namespace avian } // namespace avian
#endif // AVIAN_CODEGEN_ARCHITECTURE_H #endif // AVIAN_CODEGEN_ARCHITECTURE_H

View File

@ -83,7 +83,7 @@ argumentFootprint(unsigned footprint)
void void
nextFrame(ArchitectureContext* con, uint32_t* start, unsigned size UNUSED, nextFrame(ArchitectureContext* con, uint32_t* start, unsigned size UNUSED,
unsigned footprint, void* link, bool, unsigned footprint, void* link, bool,
unsigned targetParameterFootprint UNUSED, void** ip, void** stack) int targetParameterFootprint UNUSED, void** ip, void** stack)
{ {
assert(con, *ip >= start); assert(con, *ip >= start);
assert(con, *ip <= start + (size / TargetBytesPerWord)); assert(con, *ip <= start + (size / TargetBytesPerWord));
@ -113,13 +113,13 @@ nextFrame(ArchitectureContext* con, uint32_t* start, unsigned size UNUSED,
return; return;
} }
if (TailCalls) { if (TailCalls and targetParameterFootprint >= 0) {
if (argumentFootprint(targetParameterFootprint) > StackAlignmentInWords) { if (argumentFootprint(targetParameterFootprint) > StackAlignmentInWords) {
offset += argumentFootprint(targetParameterFootprint) offset += argumentFootprint(targetParameterFootprint)
- StackAlignmentInWords; - StackAlignmentInWords;
} }
// check for post-non-tail-call stack adjustment of the form "add // check for post-non-tail-call stack adjustment of the form "sub
// sp, sp, #offset": // sp, sp, #offset":
if ((*instruction >> 12) == 0xe24dd) { if ((*instruction >> 12) == 0xe24dd) {
unsigned value = *instruction & 0xff; unsigned value = *instruction & 0xff;
@ -292,7 +292,7 @@ class MyArchitecture: public Architecture {
virtual void nextFrame(void* start, unsigned size, unsigned footprint, virtual void nextFrame(void* start, unsigned size, unsigned footprint,
void* link, bool mostRecent, void* link, bool mostRecent,
unsigned targetParameterFootprint, void** ip, int targetParameterFootprint, void** ip,
void** stack) void** stack)
{ {
arm::nextFrame(&con, static_cast<uint32_t*>(start), size, footprint, link, arm::nextFrame(&con, static_cast<uint32_t*>(start), size, footprint, link,

View File

@ -191,7 +191,7 @@ argumentFootprint(unsigned footprint)
void void
nextFrame(ArchitectureContext* c UNUSED, int32_t* start, unsigned size, nextFrame(ArchitectureContext* c UNUSED, int32_t* start, unsigned size,
unsigned footprint, void* link, bool, unsigned footprint, void* link, bool,
unsigned targetParameterFootprint, void** ip, void** stack) int targetParameterFootprint, void** ip, void** stack)
{ {
assert(c, *ip >= start); assert(c, *ip >= start);
assert(c, *ip <= start + (size / BytesPerWord)); assert(c, *ip <= start + (size / BytesPerWord));
@ -214,7 +214,7 @@ nextFrame(ArchitectureContext* c UNUSED, int32_t* start, unsigned size,
unsigned offset = footprint; unsigned offset = footprint;
if (TailCalls) { if (TailCalls and targetParameterFootprint >= 0) {
if (argumentFootprint(targetParameterFootprint) > StackAlignmentInWords) { if (argumentFootprint(targetParameterFootprint) > StackAlignmentInWords) {
offset += argumentFootprint(targetParameterFootprint) offset += argumentFootprint(targetParameterFootprint)
- StackAlignmentInWords; - StackAlignmentInWords;
@ -391,7 +391,7 @@ class MyArchitecture: public Architecture {
virtual void nextFrame(void* start, unsigned size, unsigned footprint, virtual void nextFrame(void* start, unsigned size, unsigned footprint,
void* link, bool mostRecent, void* link, bool mostRecent,
unsigned targetParameterFootprint, void** ip, int targetParameterFootprint, void** ip,
void** stack) void** stack)
{ {
powerpc::nextFrame(&c, static_cast<int32_t*>(start), size, footprint, link, powerpc::nextFrame(&c, static_cast<int32_t*>(start), size, footprint, link,

View File

@ -72,7 +72,7 @@ read4(uint8_t* p)
void void
nextFrame(ArchitectureContext* c UNUSED, uint8_t* start, unsigned size UNUSED, nextFrame(ArchitectureContext* c UNUSED, uint8_t* start, unsigned size UNUSED,
unsigned footprint, void*, bool mostRecent, unsigned footprint, void*, bool mostRecent,
unsigned targetParameterFootprint, void** ip, void** stack) int targetParameterFootprint, void** ip, void** stack)
{ {
assert(c, *ip >= start); assert(c, *ip >= start);
assert(c, *ip <= start + size); assert(c, *ip <= start + size);
@ -114,13 +114,13 @@ nextFrame(ArchitectureContext* c UNUSED, uint8_t* start, unsigned size UNUSED,
unsigned offset = footprint + FrameHeaderSize - (mostRecent ? 1 : 0); unsigned offset = footprint + FrameHeaderSize - (mostRecent ? 1 : 0);
if (TailCalls) { if (TailCalls and targetParameterFootprint >= 0) {
if (argumentFootprint(targetParameterFootprint) > StackAlignmentInWords) { if (argumentFootprint(targetParameterFootprint) > StackAlignmentInWords) {
offset += argumentFootprint(targetParameterFootprint) offset += argumentFootprint(targetParameterFootprint)
- StackAlignmentInWords; - StackAlignmentInWords;
} }
// check for post-non-tail-call stack adjustment of the form "add // check for post-non-tail-call stack adjustment of the form "sub
// $offset,%rsp": // $offset,%rsp":
if (TargetBytesPerWord == 4) { if (TargetBytesPerWord == 4) {
if ((*instruction == 0x83 or *instruction == 0x81) if ((*instruction == 0x83 or *instruction == 0x81)
@ -141,6 +141,14 @@ nextFrame(ArchitectureContext* c UNUSED, uint8_t* start, unsigned size UNUSED,
// todo: check for and handle tail calls // todo: check for and handle tail calls
} }
if (UseFramePointer and not mostRecent) {
assert(c, static_cast<void***>(*stack)[-1] + 1
== static_cast<void**>(*stack) + offset);
assert(c, static_cast<void***>(*stack)[-1][1]
== static_cast<void**>(*stack)[offset]);
}
*ip = static_cast<void**>(*stack)[offset]; *ip = static_cast<void**>(*stack)[offset];
*stack = static_cast<void**>(*stack) + offset; *stack = static_cast<void**>(*stack) + offset;
@ -369,7 +377,7 @@ class MyArchitecture: public Architecture {
virtual void nextFrame(void* start, unsigned size, unsigned footprint, virtual void nextFrame(void* start, unsigned size, unsigned footprint,
void* link, bool mostRecent, void* link, bool mostRecent,
unsigned targetParameterFootprint, void** ip, int targetParameterFootprint, void** ip,
void** stack) void** stack)
{ {
x86::nextFrame(&c, static_cast<uint8_t*>(start), size, footprint, x86::nextFrame(&c, static_cast<uint8_t*>(start), size, footprint,

View File

@ -4330,9 +4330,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned initialIp,
Compiler::VoidType, Compiler::VoidType,
2, c->register_(t->arch->thread()), target); 2, c->register_(t->arch->thread()), target);
if (ip == codeLength(t, code)) { c->nullaryOp(lir::Trap);
c->nullaryOp(lir::Trap);
}
} goto next; } goto next;
case bipush: case bipush:

View File

@ -140,6 +140,8 @@ void handleSignal(int signal, siginfo_t*, void* context)
pthread_sigmask(SIG_UNBLOCK, &set, 0); pthread_sigmask(SIG_UNBLOCK, &set, 0);
vmJump(ip, frame, stack, thread, 0, 0); vmJump(ip, frame, stack, thread, 0, 0);
} else {
crash();
} }
} break; } break;

View File

@ -1,5 +1,17 @@
public class StackOverflow { public class StackOverflow {
private static int add(int[] numbers, int offset, int length) {
if (length == 0) {
return 0;
} else {
return numbers[offset] + add(numbers, offset + 1, length - 1);
}
}
private static int add(int ... numbers) {
return add(numbers, 0, numbers.length);
}
private static int test1() { private static int test1() {
add(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
return test1() + 1; return test1() + 1;
} }