]> Cypherpunks repositories - gostls13.git/commitdiff
[release-branch.go1.17] cmd/link: don't use label symbol for absolute address relocat...
authorCherry Mui <cherryyz@google.com>
Tue, 2 Nov 2021 22:30:08 +0000 (18:30 -0400)
committerCarlos Amedee <carlos@golang.org>
Wed, 10 Nov 2021 19:35:55 +0000 (19:35 +0000)
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 #49479

Change-Id: I64e02b56f1d792f8c20ca61b78623ef5c3e34d7e
Reviewed-on: https://go-review.googlesource.com/c/go/+/360895
Trust: Cherry Mui <cherryyz@google.com>
Run-TryBot: Cherry Mui <cherryyz@google.com>
TryBot-Result: Go Bot <gobot@golang.org>
Reviewed-by: Than McIntosh <thanm@google.com>
(cherry picked from commit 988efd58197205060ace508d29984fbab6eb3840)
Reviewed-on: https://go-review.googlesource.com/c/go/+/363014
Run-TryBot: Carlos Amedee <carlos@golang.org>
Reviewed-by: Cherry Mui <cherryyz@google.com>
src/cmd/link/internal/arm64/asm.go
src/cmd/link/link_test.go

index c10bdc4120a601676efdede7bb6859d1b13229fe..734ab3a37937e93da6b64ae18fa394d73c53e622 100644 (file)
@@ -602,7 +602,7 @@ func pereloc1(arch *sys.Arch, out *ld.OutBuf, ldr *loader.Loader, s loader.Sym,
        rs := r.Xsym
        rt := r.Type
 
-       if r.Xadd != signext21(r.Xadd) {
+       if rt == objabi.R_ADDRARM64 && r.Xadd != signext21(r.Xadd) {
                // If the relocation target would overflow the addend, then target
                // a linker-manufactured label symbol with a smaller addend instead.
                label := ldr.Lookup(offsetLabelName(ldr, rs, r.Xadd/peRelocLimit*peRelocLimit), ldr.SymVersion(rs))
index 7230054bedd2fa3149e4990a2e7d61eafbad31ad..a9b597b3236f037cec740c43a8819ee9fb34bb33 100644 (file)
@@ -993,13 +993,31 @@ package main
 
 var x = [1<<25]byte{1<<23: 23, 1<<24: 24}
 
+var addr = [...]*byte{
+       &x[1<<23-1],
+       &x[1<<23],
+       &x[1<<23+1],
+       &x[1<<24-1],
+       &x[1<<24],
+       &x[1<<24+1],
+}
+
 func main() {
+       // check relocations in instructions
        check(x[1<<23-1], 0)
        check(x[1<<23], 23)
        check(x[1<<23+1], 0)
        check(x[1<<24-1], 0)
        check(x[1<<24], 24)
        check(x[1<<24+1], 0)
+
+       // check absolute address relocations in data
+       check(*addr[0], 0)
+       check(*addr[1], 23)
+       check(*addr[2], 0)
+       check(*addr[3], 0)
+       check(*addr[4], 24)
+       check(*addr[5], 0)
 }
 
 func check(x, y byte) {