2011-01-12 07:53:41 +08:00
|
|
|
@ RUN: llvm-mc %s -triple=armv7-linux-gnueabi | FileCheck -check-prefix=ASM %s
|
2017-06-21 14:42:56 +08:00
|
|
|
@ RUN: llvm-mc %s -triple=armv7-linux-gnueabi -filetype=obj -o %t.o
|
2017-06-22 08:44:05 +08:00
|
|
|
@ RUN: llvm-objdump -d -r %t.o -triple=armv7-linux-gnueabi | FileCheck -check-prefix=OBJ %s
|
|
|
|
@ RUN: llvm-mc %s -triple=thumbv7-linux-gnueabi -filetype=obj -o %t.o
|
|
|
|
@ RUN: llvm-objdump -d -r %t.o -triple=thumbv7-linux-gnueabi | FileCheck -check-prefix=THUMB %s
|
|
|
|
|
2011-01-12 07:53:41 +08:00
|
|
|
.syntax unified
|
|
|
|
.text
|
|
|
|
.globl barf
|
|
|
|
.align 2
|
|
|
|
.type barf,%function
|
|
|
|
barf: @ @barf
|
2017-12-05 01:18:51 +08:00
|
|
|
@ %bb.0: @ %entry
|
2011-01-12 07:53:41 +08:00
|
|
|
movw r0, :lower16:GOT-(.LPC0_2+8)
|
This fixes one divergence between LLVM and binutils for ARM in the
text section.
Assume the following bit of annotated assembly:
.section .data.rel.ro,"aw",%progbits
.align 2
.LAlpha:
.long startval(GOTOFF)
.text
.align 2
.type main,%function
.align 4
main: ;;; assume "main" starts at offset 0x20
0x0 push {r11, lr}
0x4 movw r0, :lower16:(.LAlpha-(.LBeta+8))
;;; ==> (.AddrOf(.LAlpha) - ((.AddrOf(.LBeta) - .AddrOf(".")) + 8)
;;; ==> (??? - ((16-4) + 8) = -20
0x8 movt r0, :upper16:(.LAlpha-(.LBeta+8))
;;; ==> (.AddrOf(.LAlpha) - ((.AddrOf(.LBeta) - .AddrOf(".")) + 8)
;;; ==> (??? - ((16-8) + 8) = -16
0xc ... blah
.LBeta:
0x10 add r0, pc, r0
0x14 ... blah
.LGamma:
0x18 add r1, pc, r1
Above snippet results in the following relocs in the .o file for the
first pair of movw/movt instructions
00000024 R_ARM_MOVW_PREL_NC .LAlpha
00000028 R_ARM_MOVT_PREL .LAlpha
And the encoded instructions in the .o file for main: must be
00000020 <main>:
20: e92d4800 push {fp, lr}
24: e30f0fec movw r0, #65516 ; 0xffec i.e. -20
28: e34f0ff0 movt r0, #65520 ; 0xfff0 i.e. -16
However, llc (prior to this commit) generates the following sequence
00000020 <main>:
20: e92d4800 push {fp, lr}
24: e30f0fec movw r0, #65516 ; 0xffec - i.e. -20
28: e34f0fff movt r0, #65535 ; 0xffff - i.e. -1
What has to happen in the ArmAsmBackend is that if the relocation is PC
relative, the 16 bits encoded as part of movw and movt must be both addends,
not addresses. It makes sense to encode addresses by right shifting the value
by 16, but the result is incorrect for PIC.
i.e., the right shift by 16 for movt is ONLY valid for the NON-PCRel case.
This change agrees with what GNU as does, and makes the PIC code run.
MC/ARM/elf-movt.s covers this case.
llvm-svn: 131674
2011-05-20 04:55:25 +08:00
|
|
|
movt r0, :upper16:GOT-(.LPC0_2+8)
|
2011-01-12 07:53:41 +08:00
|
|
|
.LPC0_2:
|
[ARM, ELF] Don't shift movt relocation offsets
For ELF, a movw+movt pair is handled as two separate relocations.
If an offset should be applied to the symbol address, this offset is
stored as an immediate in the instruction (as opposed to stored as an
offset in the relocation itself).
Even though the actual value stored in the movt immediate after linking
is the top half of the value, we need to store the unshifted offset
prior to linking. When the relocation is made during linking, the offset
gets added to the target symbol value, and the upper half of the value
is stored in the instruction.
This makes sure that movw+movt with offset symbols get properly
handled, in case the offset addition in the lower half should be
carried over to the upper half.
This makes the output from the additions to the test case match
the output from GNU binutils.
For COFF and MachO, the movw/movt relocations are handled as a pair,
and the overflow from the lower half gets carried over to the movt,
so they should keep the shifted offset just as before.
Differential Revision: https://reviews.llvm.org/D35242
llvm-svn: 307713
2017-07-12 05:07:10 +08:00
|
|
|
movw r0, :lower16:extern_symbol+1234
|
|
|
|
movt r0, :upper16:extern_symbol+1234
|
2017-07-12 07:18:25 +08:00
|
|
|
|
|
|
|
movw r0, :lower16:(foo - bar + 1234)
|
|
|
|
movt r0, :upper16:(foo - bar + 1234)
|
|
|
|
foo:
|
|
|
|
bar:
|
|
|
|
|
2011-01-13 15:58:56 +08:00
|
|
|
@ ASM: movw r0, :lower16:(GOT-(.LPC0_2+8))
|
This fixes one divergence between LLVM and binutils for ARM in the
text section.
Assume the following bit of annotated assembly:
.section .data.rel.ro,"aw",%progbits
.align 2
.LAlpha:
.long startval(GOTOFF)
.text
.align 2
.type main,%function
.align 4
main: ;;; assume "main" starts at offset 0x20
0x0 push {r11, lr}
0x4 movw r0, :lower16:(.LAlpha-(.LBeta+8))
;;; ==> (.AddrOf(.LAlpha) - ((.AddrOf(.LBeta) - .AddrOf(".")) + 8)
;;; ==> (??? - ((16-4) + 8) = -20
0x8 movt r0, :upper16:(.LAlpha-(.LBeta+8))
;;; ==> (.AddrOf(.LAlpha) - ((.AddrOf(.LBeta) - .AddrOf(".")) + 8)
;;; ==> (??? - ((16-8) + 8) = -16
0xc ... blah
.LBeta:
0x10 add r0, pc, r0
0x14 ... blah
.LGamma:
0x18 add r1, pc, r1
Above snippet results in the following relocs in the .o file for the
first pair of movw/movt instructions
00000024 R_ARM_MOVW_PREL_NC .LAlpha
00000028 R_ARM_MOVT_PREL .LAlpha
And the encoded instructions in the .o file for main: must be
00000020 <main>:
20: e92d4800 push {fp, lr}
24: e30f0fec movw r0, #65516 ; 0xffec i.e. -20
28: e34f0ff0 movt r0, #65520 ; 0xfff0 i.e. -16
However, llc (prior to this commit) generates the following sequence
00000020 <main>:
20: e92d4800 push {fp, lr}
24: e30f0fec movw r0, #65516 ; 0xffec - i.e. -20
28: e34f0fff movt r0, #65535 ; 0xffff - i.e. -1
What has to happen in the ArmAsmBackend is that if the relocation is PC
relative, the 16 bits encoded as part of movw and movt must be both addends,
not addresses. It makes sense to encode addresses by right shifting the value
by 16, but the result is incorrect for PIC.
i.e., the right shift by 16 for movt is ONLY valid for the NON-PCRel case.
This change agrees with what GNU as does, and makes the PIC code run.
MC/ARM/elf-movt.s covers this case.
llvm-svn: 131674
2011-05-20 04:55:25 +08:00
|
|
|
@ ASM-NEXT: movt r0, :upper16:(GOT-(.LPC0_2+8))
|
[ARM, ELF] Don't shift movt relocation offsets
For ELF, a movw+movt pair is handled as two separate relocations.
If an offset should be applied to the symbol address, this offset is
stored as an immediate in the instruction (as opposed to stored as an
offset in the relocation itself).
Even though the actual value stored in the movt immediate after linking
is the top half of the value, we need to store the unshifted offset
prior to linking. When the relocation is made during linking, the offset
gets added to the target symbol value, and the upper half of the value
is stored in the instruction.
This makes sure that movw+movt with offset symbols get properly
handled, in case the offset addition in the lower half should be
carried over to the upper half.
This makes the output from the additions to the test case match
the output from GNU binutils.
For COFF and MachO, the movw/movt relocations are handled as a pair,
and the overflow from the lower half gets carried over to the movt,
so they should keep the shifted offset just as before.
Differential Revision: https://reviews.llvm.org/D35242
llvm-svn: 307713
2017-07-12 05:07:10 +08:00
|
|
|
@ ASM: movw r0, :lower16:(extern_symbol+1234)
|
2017-07-13 00:14:00 +08:00
|
|
|
@ ASM-NEXT: movt r0, :upper16:(extern_symbol+1234)
|
2017-07-12 07:18:25 +08:00
|
|
|
@ ASM: movw r0, :lower16:((foo-bar)+1234)
|
|
|
|
@ ASM-NEXT: movt r0, :upper16:((foo-bar)+1234)
|
2011-01-12 07:53:41 +08:00
|
|
|
|
2017-06-21 14:42:56 +08:00
|
|
|
@OBJ: Disassembly of section .text:
|
2019-05-01 18:40:48 +08:00
|
|
|
@OBJ-EMPTY:
|
2017-06-21 14:42:56 +08:00
|
|
|
@OBJ-NEXT: barf:
|
|
|
|
@OBJ-NEXT: 0: f0 0f 0f e3 movw r0, #65520
|
|
|
|
@OBJ-NEXT: 00000000: R_ARM_MOVW_PREL_NC GOT
|
|
|
|
@OBJ-NEXT: 4: f4 0f 4f e3 movt r0, #65524
|
|
|
|
@OBJ-NEXT: 00000004: R_ARM_MOVT_PREL GOT
|
[ARM, ELF] Don't shift movt relocation offsets
For ELF, a movw+movt pair is handled as two separate relocations.
If an offset should be applied to the symbol address, this offset is
stored as an immediate in the instruction (as opposed to stored as an
offset in the relocation itself).
Even though the actual value stored in the movt immediate after linking
is the top half of the value, we need to store the unshifted offset
prior to linking. When the relocation is made during linking, the offset
gets added to the target symbol value, and the upper half of the value
is stored in the instruction.
This makes sure that movw+movt with offset symbols get properly
handled, in case the offset addition in the lower half should be
carried over to the upper half.
This makes the output from the additions to the test case match
the output from GNU binutils.
For COFF and MachO, the movw/movt relocations are handled as a pair,
and the overflow from the lower half gets carried over to the movt,
so they should keep the shifted offset just as before.
Differential Revision: https://reviews.llvm.org/D35242
llvm-svn: 307713
2017-07-12 05:07:10 +08:00
|
|
|
@OBJ-NEXT: 8: d2 04 00 e3 movw r0, #1234
|
|
|
|
@OBJ-NEXT: 00000008: R_ARM_MOVW_ABS_NC extern_symbol
|
|
|
|
@OBJ-NEXT: c: d2 04 40 e3 movt r0, #1234
|
|
|
|
@OBJ-NEXT: 0000000c: R_ARM_MOVT_ABS extern_symbol
|
2017-07-12 07:18:25 +08:00
|
|
|
@OBJ-NEXT: 10: d2 04 00 e3 movw r0, #1234
|
|
|
|
@OBJ-NEXT: 14: 00 00 40 e3 movt r0, #0
|
2017-06-22 08:44:05 +08:00
|
|
|
|
|
|
|
@THUMB: Disassembly of section .text:
|
2019-05-01 18:40:48 +08:00
|
|
|
@THUMB-EMPTY:
|
2017-06-22 08:44:05 +08:00
|
|
|
@THUMB-NEXT: barf:
|
|
|
|
@THUMB-NEXT: 0: 4f f6 f0 70 movw r0, #65520
|
|
|
|
@THUMB-NEXT: 00000000: R_ARM_THM_MOVW_PREL_NC GOT
|
|
|
|
@THUMB-NEXT: 4: cf f6 f4 70 movt r0, #65524
|
|
|
|
@THUMB-NEXT: 00000004: R_ARM_THM_MOVT_PREL GOT
|
[ARM, ELF] Don't shift movt relocation offsets
For ELF, a movw+movt pair is handled as two separate relocations.
If an offset should be applied to the symbol address, this offset is
stored as an immediate in the instruction (as opposed to stored as an
offset in the relocation itself).
Even though the actual value stored in the movt immediate after linking
is the top half of the value, we need to store the unshifted offset
prior to linking. When the relocation is made during linking, the offset
gets added to the target symbol value, and the upper half of the value
is stored in the instruction.
This makes sure that movw+movt with offset symbols get properly
handled, in case the offset addition in the lower half should be
carried over to the upper half.
This makes the output from the additions to the test case match
the output from GNU binutils.
For COFF and MachO, the movw/movt relocations are handled as a pair,
and the overflow from the lower half gets carried over to the movt,
so they should keep the shifted offset just as before.
Differential Revision: https://reviews.llvm.org/D35242
llvm-svn: 307713
2017-07-12 05:07:10 +08:00
|
|
|
@THUMB-NEXT: 8: 40 f2 d2 40 movw r0, #1234
|
|
|
|
@THUMB-NEXT: 00000008: R_ARM_THM_MOVW_ABS_NC extern_symbol
|
|
|
|
@THUMB-NEXT: c: c0 f2 d2 40 movt r0, #1234
|
|
|
|
@THUMB-NEXT: 0000000c: R_ARM_THM_MOVT_ABS extern_symbol
|
2017-07-12 07:18:25 +08:00
|
|
|
@THUMB-NEXT: 10: 40 f2 d2 40 movw r0, #1234
|
|
|
|
@THUMB-NEXT: 14: c0 f2 00 00 movt r0, #0
|