Index: lib/Target/ARM/MCTargetDesc/ARMAsmBackend.cpp =================================================================== --- lib/Target/ARM/MCTargetDesc/ARMAsmBackend.cpp +++ lib/Target/ARM/MCTargetDesc/ARMAsmBackend.cpp @@ -487,6 +487,11 @@ return 0xffffff & ((Value - 8) >> 2); case ARM::fixup_t2_uncondbranch: { Value = Value - 4; + if (!isInt<25>(Value)) { + Ctx.reportError(Fixup.getLoc(), "Relocation out of range"); + return 0; + } + Value >>= 1; // Low bit is not encoded. uint32_t out = 0; @@ -506,6 +511,11 @@ } case ARM::fixup_t2_condbranch: { Value = Value - 4; + if (!isInt<21>(Value)) { + Ctx.reportError(Fixup.getLoc(), "Relocation out of range"); + return 0; + } + Value >>= 1; // Low bit is not encoded. uint64_t out = 0; Index: test/MC/ARM/thumb2-branch-ranges.s =================================================================== --- /dev/null +++ test/MC/ARM/thumb2-branch-ranges.s @@ -0,0 +1,92 @@ +@ RUN: not llvm-mc %s -triple thumbv7-linux-gnueabi -filetype=obj -o /dev/null 2>&1 | FileCheck %s + +// Thumb2 unconditional branch has a range of +- 16 Megabytes. The +// conditional branch has a range of +- 1 Megabyte. We should give +// an error message if we evaluate the expression at assembly +// time and it is out of range. + + .syntax unified + .thumb + b.w end + .space 0xfffffe +end: + b.w end2 + .space 0xfffffe + .global end2 +end2: + +// branch to arm function uses relocation + b.w end3 + .space 0x1000000 + .global end3 + .type end3, %function + .arm +end3: bx lr + .thumb + +// branch to thumb function is resolved at assembly time +// CHECK-NOT: error +// CHECK: [[@LINE+1]]:{{[0-9]}}: error: Relocation out of range + b.w end4 + .space 0x1000000 + .thumb_func +end4: + + beq.w end5 + .space 0xffffc +end5: + +// conditional branch to arm function uses relocation + beq.w end6 + .arm + .type end6, %function + .space 0x100000 +end6: bx lr + .thumb + +// conditional branch to thumb function resolved at assembly time +// CHECK-NOT: error +// CHECK: [[@LINE+1]]:{{[0-9]}}: error: Relocation out of range + beq.w end7 + .space 0x100000 +end7: + +start: + .space 0xfffffc + b.w start + + .arm + .global start2 + .type start2, %function +start2: + .space 0x1000000 + .thumb +// branch to arm function uses relocation + b.w start2 + +start3: + .space 0x1000000 +// branch to thumb function resolved at assembly time +// CHECK-NOT: error +// CHECK: [[@LINE+1]]:{{[0-9]}}: error: Relocation out of range + b.w start3 + +start4: + .space 0xffffc + b.w start4 + + .arm + .global start5 + .type start5, %function +start5: + .space 0x100000 + .thumb +// conditional branch to arm function uses relocation + beq.w start5 + +start6: + .space 0x100000 +// branch to thumb function resolved at assembly time +// CHECK-NOT: error +// CHECK: [[@LINE+1]]:{{[0-9]}}: error: Relocation out of range + beq.w start6