mirror of
https://github.com/corda/corda.git
synced 2025-01-21 03:55:00 +00:00
use unconditional branch to stack overflow thunk on ARM64
On ARM64, conditional branches to immediate offsets can span no more than 2^19 instructions. In the case of the stack overflow check, which wants to do a conditional branch from every non-leaf method to a handler, this can be a problem, especially when compiled code grows large as with a bootimage=true build against the OpenJDK class library. Therefore, we use an unconditional branch to reach the handler on this platform.
This commit is contained in:
parent
6b12c87e52
commit
df0085ada5
@ -95,7 +95,7 @@ void nextFrame(ArchitectureContext* con,
|
||||
|
||||
if ((*start >> 20) == (TargetBytesPerWord == 8 ? 0xf94 : 0xe59)) {
|
||||
// skip stack overflow check
|
||||
start += 3;
|
||||
start += TargetBytesPerWord == 8 ? 4 : 3;
|
||||
}
|
||||
|
||||
if (instruction <= start) {
|
||||
|
@ -106,17 +106,18 @@ void* updateOffset(vm::System* s, uint8_t* instruction, int64_t value)
|
||||
// conditional branch
|
||||
v = ((reinterpret_cast<uint8_t*>(value) - instruction) >> 2) << 5;
|
||||
mask = 0xFFFFE0;
|
||||
expect(s, bounded(5, 8, v));
|
||||
} else {
|
||||
// unconditional branch
|
||||
v = (reinterpret_cast<uint8_t*>(value) - instruction) >> 2;
|
||||
mask = 0x3FFFFFF;
|
||||
expect(s, bounded(0, 6, v));
|
||||
}
|
||||
} else {
|
||||
v = (reinterpret_cast<uint8_t*>(value) - (instruction + 8)) >> 2;
|
||||
mask = 0xFFFFFF;
|
||||
}
|
||||
|
||||
expect(s, bounded(0, 8, v));
|
||||
}
|
||||
|
||||
*p = (v & mask) | ((~mask) & *p);
|
||||
|
||||
|
@ -1483,12 +1483,17 @@ void branchRM(Context* c,
|
||||
assertT(c, size <= vm::TargetBytesPerWord);
|
||||
|
||||
if (a->low.index() == 31) {
|
||||
// stack overflow checks need to compare to the stack pointer, but
|
||||
// Stack overflow checks need to compare to the stack pointer, but
|
||||
// we can only encode that in the opposite operand order we're
|
||||
// given, so we need to reverse everything:
|
||||
// given, so we need to reverse everything. Also, we can't do a
|
||||
// conditional jump further than 2^19 instructions away, which can
|
||||
// cause trouble with large code, so we branch past an
|
||||
// unconditional branch which can jump further, which reverses the
|
||||
// logic again. Confused? Good.
|
||||
assertT(c, op == lir::JumpIfGreaterOrEqual);
|
||||
compareMR(c, size, b, size, a);
|
||||
branch(c, lir::JumpIfLess, target);
|
||||
append(c, bge(8));
|
||||
jumpC(c, vm::TargetBytesPerWord, target);
|
||||
} else {
|
||||
compareRM(c, size, a, size, b);
|
||||
branch(c, op, target);
|
||||
|
Loading…
Reference in New Issue
Block a user