Skip to content

Commit 988efd5

Browse files
committed
cmd/link: don't use label symbol for absolute address relocations on ARM64 PE
On ARM64 PE, when external linking, the PE relocation does not have an explicit addend, and instead has the addend encoded in the instruction or data. An instruction (e.g. ADRP, ADD) has limited width for the addend, so when the addend is large we use a label symbol, which points to the middle of the original target symbol, and a smaller addend. But for an absolute address relocation in the data section, we have the full width to encode the addend and we should not use the label symbol. Also, since we do not adjust the addend in the data, using the label symbol will actually make it point to the wrong address. E.g for an R_ADDR relocation targeting x+0x123456, we should emit 0x123456 in the data with an IMAGE_REL_ARM64_ADDR64 relocation pointing to x, whereas the current code emits 0x123456 in the data with an IMAGE_REL_ARM64_ADDR64 relocation pointing to the label symbol x+1MB, so it will actually be resolved to x+0x223456. This CL fixes this. Fixes #47557. Change-Id: I64e02b56f1d792f8c20ca61b78623ef5c3e34d7e Reviewed-on: https://go-review.googlesource.com/c/go/+/360895 Trust: Cherry Mui <[email protected]> Run-TryBot: Cherry Mui <[email protected]> TryBot-Result: Go Bot <[email protected]> Reviewed-by: Than McIntosh <[email protected]>
1 parent 5af93a2 commit 988efd5

File tree

2 files changed

+19
-1
lines changed

2 files changed

+19
-1
lines changed

src/cmd/link/internal/arm64/asm.go

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -602,7 +602,7 @@ func pereloc1(arch *sys.Arch, out *ld.OutBuf, ldr *loader.Loader, s loader.Sym,
602602
rs := r.Xsym
603603
rt := r.Type
604604

605-
if r.Xadd != signext21(r.Xadd) {
605+
if rt == objabi.R_ADDRARM64 && r.Xadd != signext21(r.Xadd) {
606606
// If the relocation target would overflow the addend, then target
607607
// a linker-manufactured label symbol with a smaller addend instead.
608608
label := ldr.Lookup(offsetLabelName(ldr, rs, r.Xadd/peRelocLimit*peRelocLimit), ldr.SymVersion(rs))

src/cmd/link/link_test.go

Lines changed: 18 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -997,13 +997,31 @@ package main
997997
998998
var x = [1<<25]byte{1<<23: 23, 1<<24: 24}
999999
1000+
var addr = [...]*byte{
1001+
&x[1<<23-1],
1002+
&x[1<<23],
1003+
&x[1<<23+1],
1004+
&x[1<<24-1],
1005+
&x[1<<24],
1006+
&x[1<<24+1],
1007+
}
1008+
10001009
func main() {
1010+
// check relocations in instructions
10011011
check(x[1<<23-1], 0)
10021012
check(x[1<<23], 23)
10031013
check(x[1<<23+1], 0)
10041014
check(x[1<<24-1], 0)
10051015
check(x[1<<24], 24)
10061016
check(x[1<<24+1], 0)
1017+
1018+
// check absolute address relocations in data
1019+
check(*addr[0], 0)
1020+
check(*addr[1], 23)
1021+
check(*addr[2], 0)
1022+
check(*addr[3], 0)
1023+
check(*addr[4], 24)
1024+
check(*addr[5], 0)
10071025
}
10081026
10091027
func check(x, y byte) {

0 commit comments

Comments
 (0)