|
1 | 1 | # REQUIRES: riscv |
2 | | -# RUN: llvm-mc -filetype=obj -triple=riscv32 %s -o %t.32.o |
3 | | -# RUN: ld.lld -pie %t.32.o -o %t.32 |
4 | | -# RUN: ld.lld -pie %t.32.o -o %t.32-apply --apply-dynamic-relocs |
| 2 | +# RUN: llvm-mc -filetype=obj -triple=riscv32 %s -mattr=+relax -o %t.32.o |
| 3 | +# DEFINE: %{layout} = --section-start .rela.dyn=0x1000 -Ttext=0x2000 --section-start=.iplt=0x3000 |
| 4 | +# RUN: ld.lld -pie %{layout} %t.32.o -o %t.32 |
| 5 | +# RUN: ld.lld -pie %{layout} %t.32.o -o %t.32-apply --apply-dynamic-relocs |
5 | 6 | # RUN: llvm-readobj -r -x .got.plt %t.32 | FileCheck --check-prefixes=RELOC32,NO-APPLY-RELOC32 %s |
6 | 7 | # RUN: llvm-readobj -r -x .got.plt %t.32-apply | FileCheck --check-prefixes=RELOC32,APPLY-RELOC32 %s |
7 | 8 | # RUN: llvm-readelf -s %t.32 | FileCheck --check-prefix=SYM32 %s |
8 | 9 | # RUN: llvm-objdump -d --no-show-raw-insn %t.32 | FileCheck --check-prefix=DIS32 %s |
9 | 10 |
|
10 | | -# RUN: llvm-mc -filetype=obj -triple=riscv64 %s -o %t.64.o |
11 | | -# RUN: ld.lld -pie %t.64.o -o %t.64 |
12 | | -# RUN: ld.lld -pie %t.64.o -o %t.64-apply --apply-dynamic-relocs |
| 11 | +# RUN: llvm-mc -filetype=obj -triple=riscv64 %s -mattr=+relax -o %t.64.o |
| 12 | +# RUN: ld.lld -pie %{layout} %t.64.o -o %t.64 |
| 13 | +# RUN: ld.lld -pie %{layout} %t.64.o -o %t.64-apply --apply-dynamic-relocs |
13 | 14 | # RUN: llvm-readobj -r -x .got.plt %t.64 | FileCheck --check-prefixes=RELOC64,NO-APPLY-RELOC64 %s |
14 | 15 | # RUN: llvm-readobj -r -x .got.plt %t.64-apply | FileCheck --check-prefixes=RELOC64,APPLY-RELOC64 %s |
15 | 16 | # RUN: llvm-readelf -s %t.64 | FileCheck --check-prefix=SYM64 %s |
16 | 17 | # RUN: llvm-objdump -d --no-show-raw-insn %t.64 | FileCheck --check-prefix=DIS64 %s |
17 | 18 |
|
| 19 | +## ifunc0 has a direct relocation, so it gets canonicalized to the IPLT entry. |
| 20 | +## ifunc1 has only a GOT relocation, so its symbol remains in the original section. |
| 21 | +## ifunc2 has both direct and GOT relocations, so it gets canonicalized to the IPLT entry. |
| 22 | +## All IRELATIVE addends must be correctly adjusted after relaxation. |
| 23 | + |
18 | 24 | # RELOC32: .rela.dyn { |
19 | | -# RELOC32-NEXT: 0x3200 R_RISCV_IRELATIVE - 0x117C |
| 25 | +# RELOC32-NEXT: 0x50D8 R_RISCV_RELATIVE - 0x3020 |
| 26 | +# RELOC32-NEXT: 0x60DC R_RISCV_IRELATIVE - 0x2028 |
| 27 | +# RELOC32-NEXT: 0x60E0 R_RISCV_IRELATIVE - 0x202C |
| 28 | +# RELOC32-NEXT: 0x60E4 R_RISCV_IRELATIVE - 0x2030 |
20 | 29 | # RELOC32-NEXT: } |
21 | 30 | # RELOC32-LABEL: Hex dump of section '.got.plt': |
22 | | -# NO-APPLY-RELOC32: 0x00003200 00000000 |
23 | | -# APPLY-RELOC32: 0x00003200 7c110000 |
24 | | -# RELOC32-EMPTY: |
| 31 | +# NO-APPLY-RELOC32: 0x000060dc 00000000 00000000 00000000 |
| 32 | +# APPLY-RELOC32: 0x000060dc 28200000 2c200000 30200000 |
25 | 33 |
|
26 | | -# SYM32: 0001190 0 FUNC GLOBAL DEFAULT {{.*}} func |
| 34 | +# SYM32: {{0*}}3000 0 FUNC GLOBAL DEFAULT {{.*}} ifunc0 |
| 35 | +# SYM32-NEXT: {{0*}}202c 0 IFUNC GLOBAL DEFAULT {{.*}} ifunc1 |
| 36 | +# SYM32-NEXT: {{0*}}3020 0 FUNC GLOBAL DEFAULT {{.*}} ifunc2 |
27 | 37 |
|
28 | 38 | # DIS32: <_start>: |
29 | | -# DIS32-NEXT: 1180: auipc a0, 0x0 |
30 | | -# DIS32-NEXT: addi a0, a0, 0x10 |
| 39 | +# DIS32-NEXT: 2000: jal 0x2024 <func> |
| 40 | +# DIS32: <.L0>: |
| 41 | +# DIS32-NEXT: 2004: auipc a0, 0x1 |
| 42 | +# DIS32-NEXT: addi a0, a0, -0x4 |
| 43 | +# DIS32: <.L1>: |
| 44 | +# DIS32-NEXT: 200c: auipc a1, 0x4 |
| 45 | +# DIS32-NEXT: addi a1, a1, 0xd4 |
| 46 | +# DIS32: <.L2>: |
| 47 | +# DIS32-NEXT: 2014: auipc a2, 0x1 |
| 48 | +# DIS32-NEXT: addi a2, a2, 0xc |
| 49 | +# DIS32: <.L3>: |
| 50 | +# DIS32-NEXT: 201c: auipc a3, 0x3 |
| 51 | +# DIS32-NEXT: addi a3, a3, 0xbc |
31 | 52 | # DIS32: Disassembly of section .iplt: |
32 | | -# DIS32: <func>: |
33 | | -## 32-bit: &.got.plt[func]-. = 0x3200-0x1190 = 4096*2+0x70 |
34 | | -# DIS32-NEXT: 1190: auipc t3, 0x2 |
35 | | -# DIS32-NEXT: lw t3, 0x70(t3) |
36 | | -# DIS32-NEXT: jalr t1, t3 |
37 | | -# DIS32-NEXT: nop |
| 53 | +# DIS32: <ifunc0>: |
| 54 | +## 32-bit: &.got.plt[ifunc0]-. = 0x60dc-0x3000 = 4096*3+0xdc |
| 55 | +# DIS32-NEXT: 3000: auipc t3, 0x3 |
| 56 | +# DIS32-NEXT: lw t3, 0xdc(t3) |
38 | 57 |
|
39 | 58 | # RELOC64: .rela.dyn { |
40 | | -# RELOC64-NEXT: 0x3340 R_RISCV_IRELATIVE - 0x1260 |
| 59 | +# RELOC64-NEXT: 0x5150 R_RISCV_RELATIVE - 0x3020 |
| 60 | +# RELOC64-NEXT: 0x6158 R_RISCV_IRELATIVE - 0x2028 |
| 61 | +# RELOC64-NEXT: 0x6160 R_RISCV_IRELATIVE - 0x202C |
| 62 | +# RELOC64-NEXT: 0x6168 R_RISCV_IRELATIVE - 0x2030 |
41 | 63 | # RELOC64-NEXT: } |
42 | 64 | # RELOC64-LABEL: Hex dump of section '.got.plt': |
43 | | -# NO-APPLY-RELOC64: 0x00003340 00000000 00000000 |
44 | | -# APPLY-RELOC64: 0x00003340 60120000 00000000 |
45 | | -# RELOC64-EMPTY: |
| 65 | +# NO-APPLY-RELOC64: 0x00006158 00000000 00000000 00000000 00000000 |
| 66 | +# APPLY-RELOC64: 0x00006158 28200000 00000000 2c200000 00000000 |
46 | 67 |
|
47 | | -# SYM64: 000000000001270 0 FUNC GLOBAL DEFAULT {{.*}} func |
| 68 | +# SYM64: {{0*}}3000 0 FUNC GLOBAL DEFAULT {{.*}} ifunc0 |
| 69 | +# SYM64-NEXT: {{0*}}202c 0 IFUNC GLOBAL DEFAULT {{.*}} ifunc1 |
| 70 | +# SYM64-NEXT: {{0*}}3020 0 FUNC GLOBAL DEFAULT {{.*}} ifunc2 |
48 | 71 |
|
49 | 72 | # DIS64: <_start>: |
50 | | -# DIS64-NEXT: 1264: auipc a0, 0x0 |
51 | | -# DIS64-NEXT: addi a0, a0, 0xc |
| 73 | +# DIS64-NEXT: 2000: jal 0x2024 <func> |
| 74 | +# DIS64: <.L0>: |
| 75 | +# DIS64-NEXT: 2004: auipc a0, 0x1 |
| 76 | +# DIS64-NEXT: addi a0, a0, -0x4 |
| 77 | +# DIS64: <.L1>: |
| 78 | +# DIS64-NEXT: 200c: auipc a1, 0x4 |
| 79 | +# DIS64-NEXT: addi a1, a1, 0x154 |
| 80 | +# DIS64: <.L2>: |
| 81 | +# DIS64-NEXT: 2014: auipc a2, 0x1 |
| 82 | +# DIS64-NEXT: addi a2, a2, 0xc |
| 83 | +# DIS64: <.L3>: |
| 84 | +# DIS64-NEXT: 201c: auipc a3, 0x3 |
| 85 | +# DIS64-NEXT: addi a3, a3, 0x134 |
52 | 86 | # DIS64: Disassembly of section .iplt: |
53 | | -# DIS64: <func>: |
54 | | -## 64-bit: &.got.plt[func]-. = 0x3340-0x1270 = 4096*2+0xd0 |
55 | | -# DIS64-NEXT: 1270: auipc t3, 0x2 |
56 | | -# DIS64-NEXT: ld t3, 0xd0(t3) |
57 | | -# DIS64-NEXT: jalr t1, t3 |
58 | | -# DIS64-NEXT: nop |
| 87 | +# DIS64: <ifunc0>: |
| 88 | +## 64-bit: &.got.plt[ifunc0]-. = 0x6158-0x3000 = 4096*3+0x158 |
| 89 | +# DIS64-NEXT: 3000: auipc t3, 0x3 |
| 90 | +# DIS64-NEXT: ld t3, 0x158(t3) |
59 | 91 |
|
60 | 92 | .text |
| 93 | +.globl _start |
| 94 | +_start: |
| 95 | + call func |
| 96 | +.L0: |
| 97 | + auipc a0, %pcrel_hi(ifunc0) |
| 98 | + addi a0, a0, %pcrel_lo(.L0) |
| 99 | +.L1: |
| 100 | + auipc a1, %got_pcrel_hi(ifunc1) |
| 101 | + addi a1, a1, %pcrel_lo(.L1) |
| 102 | +.L2: |
| 103 | + auipc a2, %pcrel_hi(ifunc2) |
| 104 | + addi a2, a2, %pcrel_lo(.L2) |
| 105 | +.L3: |
| 106 | + auipc a3, %got_pcrel_hi(ifunc2) |
| 107 | + addi a3, a3, %pcrel_lo(.L3) |
| 108 | + |
61 | 109 | .globl func |
62 | | -.type func, @gnu_indirect_function |
63 | 110 | func: |
64 | 111 | ret |
65 | 112 |
|
66 | | -.globl _start |
67 | | -_start: |
68 | | -.L: |
69 | | - auipc a0, %pcrel_hi(func) |
70 | | - addi a0, a0, %pcrel_lo(.L) |
| 113 | +## Resolvers are after relaxed code, so their addresses shift due to relaxation. |
| 114 | +## The IRELATIVE addends must be adjusted accordingly. |
| 115 | +.globl ifunc0, ifunc1, ifunc2 |
| 116 | +.type ifunc0, @gnu_indirect_function |
| 117 | +.type ifunc1, @gnu_indirect_function |
| 118 | +.type ifunc2, @gnu_indirect_function |
| 119 | +ifunc0: |
| 120 | + ret |
| 121 | +ifunc1: |
| 122 | + ret |
| 123 | +ifunc2: |
| 124 | + ret |
0 commit comments