2015-02-27 22:57:28 -05:00
|
|
|
// Inferno utils/5l/asm.c
|
|
|
|
|
// http://code.google.com/p/inferno-os/source/browse/utils/5l/asm.c
|
|
|
|
|
//
|
|
|
|
|
// Copyright © 1994-1999 Lucent Technologies Inc. All rights reserved.
|
|
|
|
|
// Portions Copyright © 1995-1997 C H Forsyth (forsyth@terzarima.net)
|
|
|
|
|
// Portions Copyright © 1997-1999 Vita Nuova Limited
|
|
|
|
|
// Portions Copyright © 2000-2007 Vita Nuova Holdings Limited (www.vitanuova.com)
|
|
|
|
|
// Portions Copyright © 2004,2006 Bruce Ellis
|
|
|
|
|
// Portions Copyright © 2005-2007 C H Forsyth (forsyth@terzarima.net)
|
|
|
|
|
// Revisions Copyright © 2000-2007 Lucent Technologies Inc. and others
|
|
|
|
|
// Portions Copyright © 2009 The Go Authors. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
|
//
|
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
|
//
|
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
|
// THE SOFTWARE.
|
|
|
|
|
|
2015-05-21 13:28:10 -04:00
|
|
|
package ppc64
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"cmd/internal/obj"
|
2015-05-21 13:28:10 -04:00
|
|
|
"cmd/link/internal/ld"
|
2015-03-01 22:02:13 -05:00
|
|
|
"encoding/binary"
|
2015-02-27 22:57:28 -05:00
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
)
|
|
|
|
|
|
2015-10-16 21:22:20 +13:00
|
|
|
func genplt() {
|
2015-02-27 22:57:28 -05:00
|
|
|
var s *ld.LSym
|
|
|
|
|
var stub *ld.LSym
|
|
|
|
|
var pprevtextp **ld.LSym
|
|
|
|
|
var r *ld.Reloc
|
|
|
|
|
var n string
|
|
|
|
|
var o1 uint32
|
|
|
|
|
var i int
|
|
|
|
|
|
|
|
|
|
// The ppc64 ABI PLT has similar concepts to other
|
|
|
|
|
// architectures, but is laid out quite differently. When we
|
|
|
|
|
// see an R_PPC64_REL24 relocation to a dynamic symbol
|
|
|
|
|
// (indicating that the call needs to go through the PLT), we
|
|
|
|
|
// generate up to three stubs and reserve a PLT slot.
|
|
|
|
|
//
|
|
|
|
|
// 1) The call site will be bl x; nop (where the relocation
|
|
|
|
|
// applies to the bl). We rewrite this to bl x_stub; ld
|
|
|
|
|
// r2,24(r1). The ld is necessary because x_stub will save
|
|
|
|
|
// r2 (the TOC pointer) at 24(r1) (the "TOC save slot").
|
|
|
|
|
//
|
|
|
|
|
// 2) We reserve space for a pointer in the .plt section (once
|
|
|
|
|
// per referenced dynamic function). .plt is a data
|
|
|
|
|
// section filled solely by the dynamic linker (more like
|
|
|
|
|
// .plt.got on other architectures). Initially, the
|
|
|
|
|
// dynamic linker will fill each slot with a pointer to the
|
|
|
|
|
// corresponding x@plt entry point.
|
|
|
|
|
//
|
|
|
|
|
// 3) We generate the "call stub" x_stub (once per dynamic
|
|
|
|
|
// function/object file pair). This saves the TOC in the
|
|
|
|
|
// TOC save slot, reads the function pointer from x's .plt
|
|
|
|
|
// slot and calls it like any other global entry point
|
|
|
|
|
// (including setting r12 to the function address).
|
|
|
|
|
//
|
|
|
|
|
// 4) We generate the "symbol resolver stub" x@plt (once per
|
|
|
|
|
// dynamic function). This is solely a branch to the glink
|
|
|
|
|
// resolver stub.
|
|
|
|
|
//
|
|
|
|
|
// 5) We generate the glink resolver stub (only once). This
|
|
|
|
|
// computes which symbol resolver stub we came through and
|
|
|
|
|
// invokes the dynamic resolver via a pointer provided by
|
|
|
|
|
// the dynamic linker. This will patch up the .plt slot to
|
|
|
|
|
// point directly at the function so future calls go
|
|
|
|
|
// straight from the call stub to the real function, and
|
|
|
|
|
// then call the function.
|
|
|
|
|
|
|
|
|
|
// NOTE: It's possible we could make ppc64 closer to other
|
|
|
|
|
// architectures: ppc64's .plt is like .plt.got on other
|
|
|
|
|
// platforms and ppc64's .glink is like .plt on other
|
|
|
|
|
// platforms.
|
|
|
|
|
|
|
|
|
|
// Find all R_PPC64_REL24 relocations that reference dynamic
|
|
|
|
|
// imports. Reserve PLT entries for these symbols and
|
|
|
|
|
// generate call stubs. The call stubs need to live in .text,
|
|
|
|
|
// which is why we need to do this pass this early.
|
|
|
|
|
//
|
|
|
|
|
// This assumes "case 1" from the ABI, where the caller needs
|
|
|
|
|
// us to save and restore the TOC pointer.
|
|
|
|
|
pprevtextp = &ld.Ctxt.Textp
|
|
|
|
|
|
2015-03-09 00:31:13 -04:00
|
|
|
for s = *pprevtextp; s != nil; pprevtextp, s = &s.Next, s.Next {
|
2015-02-27 22:57:28 -05:00
|
|
|
for i = range s.R {
|
|
|
|
|
r = &s.R[i]
|
2015-04-19 19:33:58 -07:00
|
|
|
if r.Type != 256+ld.R_PPC64_REL24 || r.Sym.Type != obj.SDYNIMPORT {
|
2015-02-27 22:57:28 -05:00
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Reserve PLT entry and generate symbol
|
|
|
|
|
// resolver
|
|
|
|
|
addpltsym(ld.Ctxt, r.Sym)
|
|
|
|
|
|
|
|
|
|
// Generate call stub
|
|
|
|
|
n = fmt.Sprintf("%s.%s", s.Name, r.Sym.Name)
|
|
|
|
|
|
|
|
|
|
stub = ld.Linklookup(ld.Ctxt, n, 0)
|
|
|
|
|
stub.Reachable = stub.Reachable || s.Reachable
|
|
|
|
|
if stub.Size == 0 {
|
|
|
|
|
// Need outer to resolve .TOC.
|
|
|
|
|
stub.Outer = s
|
|
|
|
|
|
|
|
|
|
// Link in to textp before s (we could
|
|
|
|
|
// do it after, but would have to skip
|
|
|
|
|
// the subsymbols)
|
|
|
|
|
*pprevtextp = stub
|
|
|
|
|
|
|
|
|
|
stub.Next = s
|
|
|
|
|
pprevtextp = &stub.Next
|
|
|
|
|
|
|
|
|
|
gencallstub(1, stub, r.Sym)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Update the relocation to use the call stub
|
|
|
|
|
r.Sym = stub
|
|
|
|
|
|
|
|
|
|
// Restore TOC after bl. The compiler put a
|
|
|
|
|
// nop here for us to overwrite.
|
|
|
|
|
o1 = 0xe8410018 // ld r2,24(r1)
|
2015-03-02 00:16:16 -05:00
|
|
|
ld.Ctxt.Arch.ByteOrder.PutUint32(s.P[r.Off+4:], o1)
|
2015-02-27 22:57:28 -05:00
|
|
|
}
|
|
|
|
|
}
|
2015-10-16 21:22:20 +13:00
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func genaddmoduledata() {
|
|
|
|
|
addmoduledata := ld.Linkrlookup(ld.Ctxt, "runtime.addmoduledata", 0)
|
|
|
|
|
if addmoduledata.Type == obj.STEXT {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
addmoduledata.Reachable = true
|
|
|
|
|
initfunc := ld.Linklookup(ld.Ctxt, "go.link.addmoduledata", 0)
|
|
|
|
|
initfunc.Type = obj.STEXT
|
|
|
|
|
initfunc.Local = true
|
|
|
|
|
initfunc.Reachable = true
|
|
|
|
|
o := func(op uint32) {
|
|
|
|
|
ld.Adduint32(ld.Ctxt, initfunc, op)
|
|
|
|
|
}
|
|
|
|
|
// addis r2, r12, .TOC.-func@ha
|
|
|
|
|
rel := ld.Addrel(initfunc)
|
|
|
|
|
rel.Off = int32(initfunc.Size)
|
|
|
|
|
rel.Siz = 8
|
|
|
|
|
rel.Sym = ld.Linklookup(ld.Ctxt, ".TOC.", 0)
|
|
|
|
|
rel.Type = obj.R_ADDRPOWER_PCREL
|
|
|
|
|
o(0x3c4c0000)
|
|
|
|
|
// addi r2, r2, .TOC.-func@l
|
|
|
|
|
o(0x38420000)
|
|
|
|
|
// mflr r31
|
|
|
|
|
o(0x7c0802a6)
|
|
|
|
|
// stdu r31, -32(r1)
|
|
|
|
|
o(0xf801ffe1)
|
|
|
|
|
// addis r3, r2, local.moduledata@got@ha
|
|
|
|
|
rel = ld.Addrel(initfunc)
|
|
|
|
|
rel.Off = int32(initfunc.Size)
|
|
|
|
|
rel.Siz = 8
|
|
|
|
|
rel.Sym = ld.Linklookup(ld.Ctxt, "local.moduledata", 0)
|
|
|
|
|
rel.Type = obj.R_ADDRPOWER_GOT
|
|
|
|
|
o(0x3c620000)
|
|
|
|
|
// ld r3, local.moduledata@got@l(r3)
|
|
|
|
|
o(0xe8630000)
|
|
|
|
|
// bl runtime.addmoduledata
|
|
|
|
|
rel = ld.Addrel(initfunc)
|
|
|
|
|
rel.Off = int32(initfunc.Size)
|
|
|
|
|
rel.Siz = 4
|
|
|
|
|
rel.Sym = addmoduledata
|
|
|
|
|
rel.Type = obj.R_CALLPOWER
|
|
|
|
|
o(0x48000001)
|
|
|
|
|
// nop
|
|
|
|
|
o(0x60000000)
|
|
|
|
|
// ld r31, 0(r1)
|
|
|
|
|
o(0xe8010000)
|
|
|
|
|
// mtlr r31
|
|
|
|
|
o(0x7c0803a6)
|
|
|
|
|
// addi r1,r1,32
|
|
|
|
|
o(0x38210020)
|
|
|
|
|
// blr
|
|
|
|
|
o(0x4e800020)
|
|
|
|
|
|
|
|
|
|
if ld.Ctxt.Etextp != nil {
|
|
|
|
|
ld.Ctxt.Etextp.Next = initfunc
|
|
|
|
|
} else {
|
|
|
|
|
ld.Ctxt.Textp = initfunc
|
|
|
|
|
}
|
|
|
|
|
ld.Ctxt.Etextp = initfunc
|
|
|
|
|
|
|
|
|
|
initarray_entry := ld.Linklookup(ld.Ctxt, "go.link.addmoduledatainit", 0)
|
|
|
|
|
initarray_entry.Reachable = true
|
|
|
|
|
initarray_entry.Local = true
|
|
|
|
|
initarray_entry.Type = obj.SINITARR
|
|
|
|
|
ld.Addaddr(ld.Ctxt, initarray_entry, initfunc)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func gentext() {
|
|
|
|
|
if ld.DynlinkingGo() {
|
|
|
|
|
genaddmoduledata()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ld.Linkmode == ld.LinkInternal {
|
|
|
|
|
genplt()
|
|
|
|
|
}
|
2015-02-27 22:57:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Construct a call stub in stub that calls symbol targ via its PLT
|
|
|
|
|
// entry.
|
|
|
|
|
func gencallstub(abicase int, stub *ld.LSym, targ *ld.LSym) {
|
|
|
|
|
if abicase != 1 {
|
|
|
|
|
// If we see R_PPC64_TOCSAVE or R_PPC64_REL24_NOTOC
|
|
|
|
|
// relocations, we'll need to implement cases 2 and 3.
|
|
|
|
|
log.Fatalf("gencallstub only implements case 1 calls")
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-02 12:35:15 -05:00
|
|
|
plt := ld.Linklookup(ld.Ctxt, ".plt", 0)
|
2015-02-27 22:57:28 -05:00
|
|
|
|
2015-04-19 19:33:58 -07:00
|
|
|
stub.Type = obj.STEXT
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
// Save TOC pointer in TOC save slot
|
|
|
|
|
ld.Adduint32(ld.Ctxt, stub, 0xf8410018) // std r2,24(r1)
|
|
|
|
|
|
|
|
|
|
// Load the function pointer from the PLT.
|
2015-03-02 12:35:15 -05:00
|
|
|
r := ld.Addrel(stub)
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
r.Off = int32(stub.Size)
|
|
|
|
|
r.Sym = plt
|
|
|
|
|
r.Add = int64(targ.Plt)
|
|
|
|
|
r.Siz = 2
|
2015-03-01 22:02:13 -05:00
|
|
|
if ld.Ctxt.Arch.ByteOrder == binary.BigEndian {
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Off += int32(r.Siz)
|
|
|
|
|
}
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_POWER_TOC
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_HA
|
|
|
|
|
ld.Adduint32(ld.Ctxt, stub, 0x3d820000) // addis r12,r2,targ@plt@toc@ha
|
|
|
|
|
r = ld.Addrel(stub)
|
|
|
|
|
r.Off = int32(stub.Size)
|
|
|
|
|
r.Sym = plt
|
|
|
|
|
r.Add = int64(targ.Plt)
|
|
|
|
|
r.Siz = 2
|
2015-03-01 22:02:13 -05:00
|
|
|
if ld.Ctxt.Arch.ByteOrder == binary.BigEndian {
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Off += int32(r.Siz)
|
|
|
|
|
}
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_POWER_TOC
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_LO
|
|
|
|
|
ld.Adduint32(ld.Ctxt, stub, 0xe98c0000) // ld r12,targ@plt@toc@l(r12)
|
|
|
|
|
|
|
|
|
|
// Jump to the loaded pointer
|
|
|
|
|
ld.Adduint32(ld.Ctxt, stub, 0x7d8903a6) // mtctr r12
|
|
|
|
|
ld.Adduint32(ld.Ctxt, stub, 0x4e800420) // bctr
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func adddynrela(rel *ld.LSym, s *ld.LSym, r *ld.Reloc) {
|
|
|
|
|
log.Fatalf("adddynrela not implemented")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func adddynrel(s *ld.LSym, r *ld.Reloc) {
|
2015-03-02 12:35:15 -05:00
|
|
|
targ := r.Sym
|
2015-02-27 22:57:28 -05:00
|
|
|
ld.Ctxt.Cursym = s
|
|
|
|
|
|
|
|
|
|
switch r.Type {
|
|
|
|
|
default:
|
|
|
|
|
if r.Type >= 256 {
|
|
|
|
|
ld.Diag("unexpected relocation type %d", r.Type)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Handle relocations found in ELF object files.
|
|
|
|
|
case 256 + ld.R_PPC64_REL24:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_CALLPOWER
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
// This is a local call, so the caller isn't setting
|
|
|
|
|
// up r12 and r2 is the same for the caller and
|
|
|
|
|
// callee. Hence, we need to go to the local entry
|
|
|
|
|
// point. (If we don't do this, the callee will try
|
|
|
|
|
// to use r12 to compute r2.)
|
|
|
|
|
r.Add += int64(r.Sym.Localentry) * 4
|
|
|
|
|
|
2015-04-19 19:33:58 -07:00
|
|
|
if targ.Type == obj.SDYNIMPORT {
|
2015-02-27 22:57:28 -05:00
|
|
|
// Should have been handled in elfsetupplt
|
|
|
|
|
ld.Diag("unexpected R_PPC64_REL24 for dyn import")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
2015-11-23 12:55:37 -08:00
|
|
|
case 256 + ld.R_PPC_REL32:
|
|
|
|
|
r.Type = obj.R_PCREL
|
|
|
|
|
r.Add += 4
|
|
|
|
|
|
|
|
|
|
if targ.Type == obj.SDYNIMPORT {
|
|
|
|
|
ld.Diag("unexpected R_PPC_REL32 for dyn import")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
2015-02-27 22:57:28 -05:00
|
|
|
case 256 + ld.R_PPC64_ADDR64:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_ADDR
|
|
|
|
|
if targ.Type == obj.SDYNIMPORT {
|
2015-02-27 22:57:28 -05:00
|
|
|
// These happen in .toc sections
|
2015-05-12 15:59:15 +12:00
|
|
|
ld.Adddynsym(ld.Ctxt, targ)
|
2015-02-27 22:57:28 -05:00
|
|
|
|
2015-03-02 12:35:15 -05:00
|
|
|
rela := ld.Linklookup(ld.Ctxt, ".rela", 0)
|
2015-02-27 22:57:28 -05:00
|
|
|
ld.Addaddrplus(ld.Ctxt, rela, s, int64(r.Off))
|
|
|
|
|
ld.Adduint64(ld.Ctxt, rela, ld.ELF64_R_INFO(uint32(targ.Dynid), ld.R_PPC64_ADDR64))
|
|
|
|
|
ld.Adduint64(ld.Ctxt, rela, uint64(r.Add))
|
|
|
|
|
r.Type = 256 // ignore during relocsym
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
case 256 + ld.R_PPC64_TOC16:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_POWER_TOC
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_LO | ld.RV_CHECK_OVERFLOW
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
case 256 + ld.R_PPC64_TOC16_LO:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_POWER_TOC
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_LO
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
case 256 + ld.R_PPC64_TOC16_HA:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_POWER_TOC
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_HA | ld.RV_CHECK_OVERFLOW
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
case 256 + ld.R_PPC64_TOC16_HI:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_POWER_TOC
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_HI | ld.RV_CHECK_OVERFLOW
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
case 256 + ld.R_PPC64_TOC16_DS:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_POWER_TOC
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_DS | ld.RV_CHECK_OVERFLOW
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
case 256 + ld.R_PPC64_TOC16_LO_DS:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_POWER_TOC
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_DS
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
case 256 + ld.R_PPC64_REL16_LO:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_PCREL
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_LO
|
|
|
|
|
r.Add += 2 // Compensate for relocation size of 2
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
case 256 + ld.R_PPC64_REL16_HI:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_PCREL
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_HI | ld.RV_CHECK_OVERFLOW
|
|
|
|
|
r.Add += 2
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
case 256 + ld.R_PPC64_REL16_HA:
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_PCREL
|
2015-02-27 22:57:28 -05:00
|
|
|
r.Variant = ld.RV_POWER_HA | ld.RV_CHECK_OVERFLOW
|
|
|
|
|
r.Add += 2
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Handle references to ELF symbols from our own object files.
|
2015-04-19 19:33:58 -07:00
|
|
|
if targ.Type != obj.SDYNIMPORT {
|
2015-02-27 22:57:28 -05:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO(austin): Translate our relocations to ELF
|
|
|
|
|
|
|
|
|
|
ld.Diag("unsupported relocation for dynamic symbol %s (type=%d stype=%d)", targ.Name, r.Type, targ.Type)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func elfreloc1(r *ld.Reloc, sectoff int64) int {
|
2015-09-04 08:30:15 +12:00
|
|
|
ld.Thearch.Vput(uint64(sectoff))
|
|
|
|
|
|
|
|
|
|
elfsym := r.Xsym.ElfsymForReloc()
|
|
|
|
|
switch r.Type {
|
|
|
|
|
default:
|
|
|
|
|
return -1
|
|
|
|
|
|
|
|
|
|
case obj.R_ADDR:
|
|
|
|
|
switch r.Siz {
|
|
|
|
|
case 4:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_ADDR32 | uint64(elfsym)<<32)
|
|
|
|
|
case 8:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_ADDR64 | uint64(elfsym)<<32)
|
|
|
|
|
default:
|
|
|
|
|
return -1
|
|
|
|
|
}
|
|
|
|
|
|
2015-10-16 21:15:18 +13:00
|
|
|
case obj.R_POWER_TLS:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_TLS | uint64(elfsym)<<32)
|
|
|
|
|
|
2015-09-04 08:30:15 +12:00
|
|
|
case obj.R_POWER_TLS_LE:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_TPREL16 | uint64(elfsym)<<32)
|
|
|
|
|
|
2015-10-16 21:15:18 +13:00
|
|
|
case obj.R_POWER_TLS_IE:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_GOT_TPREL16_HA | uint64(elfsym)<<32)
|
|
|
|
|
ld.Thearch.Vput(uint64(r.Xadd))
|
|
|
|
|
ld.Thearch.Vput(uint64(sectoff + 4))
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_GOT_TPREL16_LO_DS | uint64(elfsym)<<32)
|
|
|
|
|
|
2015-09-04 08:30:15 +12:00
|
|
|
case obj.R_ADDRPOWER:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_ADDR16_HA | uint64(elfsym)<<32)
|
|
|
|
|
ld.Thearch.Vput(uint64(r.Xadd))
|
|
|
|
|
ld.Thearch.Vput(uint64(sectoff + 4))
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_ADDR16_LO | uint64(elfsym)<<32)
|
|
|
|
|
|
|
|
|
|
case obj.R_ADDRPOWER_DS:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_ADDR16_HA | uint64(elfsym)<<32)
|
|
|
|
|
ld.Thearch.Vput(uint64(r.Xadd))
|
|
|
|
|
ld.Thearch.Vput(uint64(sectoff + 4))
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_ADDR16_LO_DS | uint64(elfsym)<<32)
|
|
|
|
|
|
2015-10-16 15:54:10 +13:00
|
|
|
case obj.R_ADDRPOWER_GOT:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_GOT16_HA | uint64(elfsym)<<32)
|
|
|
|
|
ld.Thearch.Vput(uint64(r.Xadd))
|
|
|
|
|
ld.Thearch.Vput(uint64(sectoff + 4))
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_GOT16_LO_DS | uint64(elfsym)<<32)
|
|
|
|
|
|
cmd/compile, cmd/link, runtime: on ppc64x, maintain the TOC pointer in R2 when compiling PIC
The PowerPC ISA does not have a PC-relative load instruction, which poses
obvious challenges when generating position-independent code. The way the ELFv2
ABI addresses this is to specify that r2 points to a per "module" (shared
library or executable) TOC pointer. Maintaining this pointer requires
cooperation between codegen and the system linker:
* Non-leaf functions leave space on the stack at r1+24 to save the TOC pointer.
* A call to a function that *might* have to go via a PLT stub must be followed
by a nop instruction that the system linker can replace with "ld r1, 24(r1)"
to restore the TOC pointer (only when dynamically linking Go code).
* When calling a function via a function pointer, the address of the function
must be in r12, and the first couple of instructions (the "global entry
point") of the called function use this to derive the address of the TOC
for the module it is in.
* When calling a function that is implemented in the same module, the system
linker adjusts the call to skip over the instructions mentioned above (the
"local entry point"), assuming that r2 is already correctly set.
So this changeset adds the global entry point instructions, sets the metadata so
the system linker knows where the local entry point is, inserts code to save the
TOC pointer at 24(r1), adds a nop after any call not known to be local and copes
with the odd non-local code transfer in the runtime (e.g. the stuff around
jmpdefer). It does not actually compile PIC yet.
Change-Id: I7522e22bdfd2f891745a900c60254fe9e372c854
Reviewed-on: https://go-review.googlesource.com/15967
Reviewed-by: Russ Cox <rsc@golang.org>
2015-10-16 15:42:09 +13:00
|
|
|
case obj.R_ADDRPOWER_PCREL:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_REL16_HA | uint64(elfsym)<<32)
|
|
|
|
|
ld.Thearch.Vput(uint64(r.Xadd))
|
|
|
|
|
ld.Thearch.Vput(uint64(sectoff + 4))
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_REL16_LO | uint64(elfsym)<<32)
|
|
|
|
|
r.Xadd += 4
|
|
|
|
|
|
2015-10-16 20:48:56 +13:00
|
|
|
case obj.R_ADDRPOWER_TOCREL:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_TOC16_HA | uint64(elfsym)<<32)
|
|
|
|
|
ld.Thearch.Vput(uint64(r.Xadd))
|
|
|
|
|
ld.Thearch.Vput(uint64(sectoff + 4))
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_TOC16_LO | uint64(elfsym)<<32)
|
|
|
|
|
|
|
|
|
|
case obj.R_ADDRPOWER_TOCREL_DS:
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_TOC16_HA | uint64(elfsym)<<32)
|
|
|
|
|
ld.Thearch.Vput(uint64(r.Xadd))
|
|
|
|
|
ld.Thearch.Vput(uint64(sectoff + 4))
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_TOC16_LO_DS | uint64(elfsym)<<32)
|
|
|
|
|
|
2015-09-04 08:30:15 +12:00
|
|
|
case obj.R_CALLPOWER:
|
|
|
|
|
if r.Siz != 4 {
|
|
|
|
|
return -1
|
|
|
|
|
}
|
|
|
|
|
ld.Thearch.Vput(ld.R_PPC64_REL24 | uint64(elfsym)<<32)
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
ld.Thearch.Vput(uint64(r.Xadd))
|
|
|
|
|
|
|
|
|
|
return 0
|
2015-02-27 22:57:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func elfsetupplt() {
|
2015-03-02 12:35:15 -05:00
|
|
|
plt := ld.Linklookup(ld.Ctxt, ".plt", 0)
|
2015-02-27 22:57:28 -05:00
|
|
|
if plt.Size == 0 {
|
|
|
|
|
// The dynamic linker stores the address of the
|
|
|
|
|
// dynamic resolver and the DSO identifier in the two
|
|
|
|
|
// doublewords at the beginning of the .plt section
|
|
|
|
|
// before the PLT array. Reserve space for these.
|
|
|
|
|
plt.Size = 16
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func machoreloc1(r *ld.Reloc, sectoff int64) int {
|
|
|
|
|
return -1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Return the value of .TOC. for symbol s
|
|
|
|
|
func symtoc(s *ld.LSym) int64 {
|
|
|
|
|
var toc *ld.LSym
|
|
|
|
|
|
|
|
|
|
if s.Outer != nil {
|
|
|
|
|
toc = ld.Linkrlookup(ld.Ctxt, ".TOC.", int(s.Outer.Version))
|
|
|
|
|
} else {
|
|
|
|
|
toc = ld.Linkrlookup(ld.Ctxt, ".TOC.", int(s.Version))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if toc == nil {
|
|
|
|
|
ld.Diag("TOC-relative relocation in object without .TOC.")
|
|
|
|
|
return 0
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return toc.Value
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-08 15:21:58 +12:00
|
|
|
func archrelocaddr(r *ld.Reloc, s *ld.LSym, val *int64) int {
|
|
|
|
|
var o1, o2 uint32
|
|
|
|
|
if ld.Ctxt.Arch.ByteOrder == binary.BigEndian {
|
|
|
|
|
o1 = uint32(*val >> 32)
|
|
|
|
|
o2 = uint32(*val)
|
|
|
|
|
} else {
|
|
|
|
|
o1 = uint32(*val)
|
|
|
|
|
o2 = uint32(*val >> 32)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// We are spreading a 31-bit address across two instructions, putting the
|
|
|
|
|
// high (adjusted) part in the low 16 bits of the first instruction and the
|
|
|
|
|
// low part in the low 16 bits of the second instruction, or, in the DS case,
|
|
|
|
|
// bits 15-2 (inclusive) of the address into bits 15-2 of the second
|
|
|
|
|
// instruction (it is an error in this case if the low 2 bits of the address
|
|
|
|
|
// are non-zero).
|
|
|
|
|
|
|
|
|
|
t := ld.Symaddr(r.Sym) + r.Add
|
|
|
|
|
if t < 0 || t >= 1<<31 {
|
|
|
|
|
ld.Ctxt.Diag("relocation for %s is too big (>=2G): %d", s.Name, ld.Symaddr(r.Sym))
|
|
|
|
|
}
|
|
|
|
|
if t&0x8000 != 0 {
|
|
|
|
|
t += 0x10000
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch r.Type {
|
|
|
|
|
case obj.R_ADDRPOWER:
|
|
|
|
|
o1 |= (uint32(t) >> 16) & 0xffff
|
|
|
|
|
o2 |= uint32(t) & 0xffff
|
|
|
|
|
|
|
|
|
|
case obj.R_ADDRPOWER_DS:
|
|
|
|
|
o1 |= (uint32(t) >> 16) & 0xffff
|
|
|
|
|
if t&3 != 0 {
|
|
|
|
|
ld.Ctxt.Diag("bad DS reloc for %s: %d", s.Name, ld.Symaddr(r.Sym))
|
|
|
|
|
}
|
|
|
|
|
o2 |= uint32(t) & 0xfffc
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
return -1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ld.Ctxt.Arch.ByteOrder == binary.BigEndian {
|
|
|
|
|
*val = int64(o1)<<32 | int64(o2)
|
|
|
|
|
} else {
|
|
|
|
|
*val = int64(o2)<<32 | int64(o1)
|
|
|
|
|
}
|
|
|
|
|
return 0
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-27 22:57:28 -05:00
|
|
|
func archreloc(r *ld.Reloc, s *ld.LSym, val *int64) int {
|
|
|
|
|
if ld.Linkmode == ld.LinkExternal {
|
2015-09-04 08:30:15 +12:00
|
|
|
switch r.Type {
|
|
|
|
|
default:
|
|
|
|
|
return -1
|
|
|
|
|
|
2015-10-16 21:15:18 +13:00
|
|
|
case obj.R_POWER_TLS, obj.R_POWER_TLS_LE, obj.R_POWER_TLS_IE:
|
2015-09-04 08:30:15 +12:00
|
|
|
r.Done = 0
|
|
|
|
|
// check Outer is nil, Type is TLSBSS?
|
|
|
|
|
r.Xadd = r.Add
|
|
|
|
|
r.Xsym = r.Sym
|
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
|
case obj.R_ADDRPOWER,
|
cmd/compile, cmd/link, runtime: on ppc64x, maintain the TOC pointer in R2 when compiling PIC
The PowerPC ISA does not have a PC-relative load instruction, which poses
obvious challenges when generating position-independent code. The way the ELFv2
ABI addresses this is to specify that r2 points to a per "module" (shared
library or executable) TOC pointer. Maintaining this pointer requires
cooperation between codegen and the system linker:
* Non-leaf functions leave space on the stack at r1+24 to save the TOC pointer.
* A call to a function that *might* have to go via a PLT stub must be followed
by a nop instruction that the system linker can replace with "ld r1, 24(r1)"
to restore the TOC pointer (only when dynamically linking Go code).
* When calling a function via a function pointer, the address of the function
must be in r12, and the first couple of instructions (the "global entry
point") of the called function use this to derive the address of the TOC
for the module it is in.
* When calling a function that is implemented in the same module, the system
linker adjusts the call to skip over the instructions mentioned above (the
"local entry point"), assuming that r2 is already correctly set.
So this changeset adds the global entry point instructions, sets the metadata so
the system linker knows where the local entry point is, inserts code to save the
TOC pointer at 24(r1), adds a nop after any call not known to be local and copes
with the odd non-local code transfer in the runtime (e.g. the stuff around
jmpdefer). It does not actually compile PIC yet.
Change-Id: I7522e22bdfd2f891745a900c60254fe9e372c854
Reviewed-on: https://go-review.googlesource.com/15967
Reviewed-by: Russ Cox <rsc@golang.org>
2015-10-16 15:42:09 +13:00
|
|
|
obj.R_ADDRPOWER_DS,
|
2015-10-16 20:48:56 +13:00
|
|
|
obj.R_ADDRPOWER_TOCREL,
|
|
|
|
|
obj.R_ADDRPOWER_TOCREL_DS,
|
2015-10-16 15:54:10 +13:00
|
|
|
obj.R_ADDRPOWER_GOT,
|
cmd/compile, cmd/link, runtime: on ppc64x, maintain the TOC pointer in R2 when compiling PIC
The PowerPC ISA does not have a PC-relative load instruction, which poses
obvious challenges when generating position-independent code. The way the ELFv2
ABI addresses this is to specify that r2 points to a per "module" (shared
library or executable) TOC pointer. Maintaining this pointer requires
cooperation between codegen and the system linker:
* Non-leaf functions leave space on the stack at r1+24 to save the TOC pointer.
* A call to a function that *might* have to go via a PLT stub must be followed
by a nop instruction that the system linker can replace with "ld r1, 24(r1)"
to restore the TOC pointer (only when dynamically linking Go code).
* When calling a function via a function pointer, the address of the function
must be in r12, and the first couple of instructions (the "global entry
point") of the called function use this to derive the address of the TOC
for the module it is in.
* When calling a function that is implemented in the same module, the system
linker adjusts the call to skip over the instructions mentioned above (the
"local entry point"), assuming that r2 is already correctly set.
So this changeset adds the global entry point instructions, sets the metadata so
the system linker knows where the local entry point is, inserts code to save the
TOC pointer at 24(r1), adds a nop after any call not known to be local and copes
with the odd non-local code transfer in the runtime (e.g. the stuff around
jmpdefer). It does not actually compile PIC yet.
Change-Id: I7522e22bdfd2f891745a900c60254fe9e372c854
Reviewed-on: https://go-review.googlesource.com/15967
Reviewed-by: Russ Cox <rsc@golang.org>
2015-10-16 15:42:09 +13:00
|
|
|
obj.R_ADDRPOWER_PCREL:
|
2015-09-04 08:30:15 +12:00
|
|
|
r.Done = 0
|
|
|
|
|
|
|
|
|
|
// set up addend for eventual relocation via outer symbol.
|
|
|
|
|
rs := r.Sym
|
|
|
|
|
r.Xadd = r.Add
|
|
|
|
|
for rs.Outer != nil {
|
|
|
|
|
r.Xadd += ld.Symaddr(rs) - ld.Symaddr(rs.Outer)
|
|
|
|
|
rs = rs.Outer
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if rs.Type != obj.SHOSTOBJ && rs.Type != obj.SDYNIMPORT && rs.Sect == nil {
|
|
|
|
|
ld.Diag("missing section for %s", rs.Name)
|
|
|
|
|
}
|
|
|
|
|
r.Xsym = rs
|
|
|
|
|
|
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
|
case obj.R_CALLPOWER:
|
|
|
|
|
r.Done = 0
|
|
|
|
|
r.Xsym = r.Sym
|
|
|
|
|
r.Xadd = r.Add
|
|
|
|
|
return 0
|
|
|
|
|
}
|
2015-02-27 22:57:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch r.Type {
|
2015-04-19 19:33:58 -07:00
|
|
|
case obj.R_CONST:
|
2015-02-27 22:57:28 -05:00
|
|
|
*val = r.Add
|
|
|
|
|
return 0
|
|
|
|
|
|
2015-04-19 19:33:58 -07:00
|
|
|
case obj.R_GOTOFF:
|
2015-02-27 22:57:28 -05:00
|
|
|
*val = ld.Symaddr(r.Sym) + r.Add - ld.Symaddr(ld.Linklookup(ld.Ctxt, ".got", 0))
|
|
|
|
|
return 0
|
|
|
|
|
|
2015-09-08 15:21:58 +12:00
|
|
|
case obj.R_ADDRPOWER, obj.R_ADDRPOWER_DS:
|
|
|
|
|
return archrelocaddr(r, s, val)
|
2015-02-27 22:57:28 -05:00
|
|
|
|
2015-04-19 19:33:58 -07:00
|
|
|
case obj.R_CALLPOWER:
|
2015-03-01 13:32:49 -05:00
|
|
|
// Bits 6 through 29 = (S + A - P) >> 2
|
2015-02-27 22:57:28 -05:00
|
|
|
|
2015-03-02 12:35:15 -05:00
|
|
|
t := ld.Symaddr(r.Sym) + r.Add - (s.Value + int64(r.Off))
|
2015-02-27 22:57:28 -05:00
|
|
|
if t&3 != 0 {
|
|
|
|
|
ld.Ctxt.Diag("relocation for %s+%d is not aligned: %d", r.Sym.Name, r.Off, t)
|
|
|
|
|
}
|
|
|
|
|
if int64(int32(t<<6)>>6) != t {
|
|
|
|
|
// TODO(austin) This can happen if text > 32M.
|
|
|
|
|
// Add a call trampoline to .text in that case.
|
|
|
|
|
ld.Ctxt.Diag("relocation for %s+%d is too big: %d", r.Sym.Name, r.Off, t)
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-31 13:51:37 +12:00
|
|
|
*val |= int64(uint32(t) &^ 0xfc000003)
|
2015-02-27 22:57:28 -05:00
|
|
|
return 0
|
|
|
|
|
|
2015-04-19 19:33:58 -07:00
|
|
|
case obj.R_POWER_TOC: // S + A - .TOC.
|
2015-02-27 22:57:28 -05:00
|
|
|
*val = ld.Symaddr(r.Sym) + r.Add - symtoc(s)
|
|
|
|
|
|
|
|
|
|
return 0
|
2015-10-30 10:13:13 +13:00
|
|
|
|
|
|
|
|
case obj.R_POWER_TLS_LE:
|
|
|
|
|
// The thread pointer points 0x7000 bytes after the start of the the
|
|
|
|
|
// thread local storage area as documented in section "3.7.2 TLS
|
|
|
|
|
// Runtime Handling" of "Power Architecture 64-Bit ELF V2 ABI
|
|
|
|
|
// Specification".
|
|
|
|
|
v := r.Sym.Value - 0x7000
|
|
|
|
|
if int64(int16(v)) != v {
|
|
|
|
|
ld.Diag("TLS offset out of range %d", v)
|
|
|
|
|
}
|
|
|
|
|
*val = (*val &^ 0xffff) | (v & 0xffff)
|
|
|
|
|
return 0
|
2015-02-27 22:57:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return -1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func archrelocvariant(r *ld.Reloc, s *ld.LSym, t int64) int64 {
|
|
|
|
|
switch r.Variant & ld.RV_TYPE_MASK {
|
|
|
|
|
default:
|
|
|
|
|
ld.Diag("unexpected relocation variant %d", r.Variant)
|
|
|
|
|
fallthrough
|
|
|
|
|
|
|
|
|
|
case ld.RV_NONE:
|
|
|
|
|
return t
|
|
|
|
|
|
|
|
|
|
case ld.RV_POWER_LO:
|
|
|
|
|
if r.Variant&ld.RV_CHECK_OVERFLOW != 0 {
|
|
|
|
|
// Whether to check for signed or unsigned
|
|
|
|
|
// overflow depends on the instruction
|
2015-03-02 12:35:15 -05:00
|
|
|
var o1 uint32
|
2015-03-01 22:02:13 -05:00
|
|
|
if ld.Ctxt.Arch.ByteOrder == binary.BigEndian {
|
2015-02-27 22:57:28 -05:00
|
|
|
o1 = ld.Be32(s.P[r.Off-2:])
|
|
|
|
|
} else {
|
|
|
|
|
o1 = ld.Le32(s.P[r.Off:])
|
|
|
|
|
}
|
|
|
|
|
switch o1 >> 26 {
|
|
|
|
|
case 24, // ori
|
|
|
|
|
26, // xori
|
|
|
|
|
28: // andi
|
|
|
|
|
if t>>16 != 0 {
|
|
|
|
|
goto overflow
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
if int64(int16(t)) != t {
|
|
|
|
|
goto overflow
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return int64(int16(t))
|
|
|
|
|
|
|
|
|
|
case ld.RV_POWER_HA:
|
|
|
|
|
t += 0x8000
|
|
|
|
|
fallthrough
|
|
|
|
|
|
|
|
|
|
// Fallthrough
|
|
|
|
|
case ld.RV_POWER_HI:
|
|
|
|
|
t >>= 16
|
|
|
|
|
|
|
|
|
|
if r.Variant&ld.RV_CHECK_OVERFLOW != 0 {
|
|
|
|
|
// Whether to check for signed or unsigned
|
|
|
|
|
// overflow depends on the instruction
|
2015-03-02 12:35:15 -05:00
|
|
|
var o1 uint32
|
2015-03-01 22:02:13 -05:00
|
|
|
if ld.Ctxt.Arch.ByteOrder == binary.BigEndian {
|
2015-02-27 22:57:28 -05:00
|
|
|
o1 = ld.Be32(s.P[r.Off-2:])
|
|
|
|
|
} else {
|
|
|
|
|
o1 = ld.Le32(s.P[r.Off:])
|
|
|
|
|
}
|
|
|
|
|
switch o1 >> 26 {
|
|
|
|
|
case 25, // oris
|
|
|
|
|
27, // xoris
|
|
|
|
|
29: // andis
|
|
|
|
|
if t>>16 != 0 {
|
|
|
|
|
goto overflow
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
if int64(int16(t)) != t {
|
|
|
|
|
goto overflow
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return int64(int16(t))
|
|
|
|
|
|
|
|
|
|
case ld.RV_POWER_DS:
|
2015-03-02 12:35:15 -05:00
|
|
|
var o1 uint32
|
2015-03-01 22:02:13 -05:00
|
|
|
if ld.Ctxt.Arch.ByteOrder == binary.BigEndian {
|
2015-02-27 22:57:28 -05:00
|
|
|
o1 = uint32(ld.Be16(s.P[r.Off:]))
|
|
|
|
|
} else {
|
|
|
|
|
o1 = uint32(ld.Le16(s.P[r.Off:]))
|
|
|
|
|
}
|
|
|
|
|
if t&3 != 0 {
|
|
|
|
|
ld.Diag("relocation for %s+%d is not aligned: %d", r.Sym.Name, r.Off, t)
|
|
|
|
|
}
|
|
|
|
|
if (r.Variant&ld.RV_CHECK_OVERFLOW != 0) && int64(int16(t)) != t {
|
|
|
|
|
goto overflow
|
|
|
|
|
}
|
|
|
|
|
return int64(o1)&0x3 | int64(int16(t))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
overflow:
|
|
|
|
|
ld.Diag("relocation for %s+%d is too big: %d", r.Sym.Name, r.Off, t)
|
|
|
|
|
return t
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func addpltsym(ctxt *ld.Link, s *ld.LSym) {
|
|
|
|
|
if s.Plt >= 0 {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2015-05-12 15:59:15 +12:00
|
|
|
ld.Adddynsym(ctxt, s)
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
if ld.Iself {
|
2015-03-02 12:35:15 -05:00
|
|
|
plt := ld.Linklookup(ctxt, ".plt", 0)
|
|
|
|
|
rela := ld.Linklookup(ctxt, ".rela.plt", 0)
|
2015-02-27 22:57:28 -05:00
|
|
|
if plt.Size == 0 {
|
|
|
|
|
elfsetupplt()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create the glink resolver if necessary
|
2015-03-02 12:35:15 -05:00
|
|
|
glink := ensureglinkresolver()
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
// Write symbol resolver stub (just a branch to the
|
|
|
|
|
// glink resolver stub)
|
2015-03-02 12:35:15 -05:00
|
|
|
r := ld.Addrel(glink)
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
r.Sym = glink
|
|
|
|
|
r.Off = int32(glink.Size)
|
|
|
|
|
r.Siz = 4
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_CALLPOWER
|
2015-02-27 22:57:28 -05:00
|
|
|
ld.Adduint32(ctxt, glink, 0x48000000) // b .glink
|
|
|
|
|
|
|
|
|
|
// In the ppc64 ABI, the dynamic linker is responsible
|
|
|
|
|
// for writing the entire PLT. We just need to
|
|
|
|
|
// reserve 8 bytes for each PLT entry and generate a
|
|
|
|
|
// JMP_SLOT dynamic relocation for it.
|
|
|
|
|
//
|
|
|
|
|
// TODO(austin): ABI v1 is different
|
|
|
|
|
s.Plt = int32(plt.Size)
|
|
|
|
|
|
|
|
|
|
plt.Size += 8
|
|
|
|
|
|
|
|
|
|
ld.Addaddrplus(ctxt, rela, plt, int64(s.Plt))
|
|
|
|
|
ld.Adduint64(ctxt, rela, ld.ELF64_R_INFO(uint32(s.Dynid), ld.R_PPC64_JMP_SLOT))
|
|
|
|
|
ld.Adduint64(ctxt, rela, 0)
|
|
|
|
|
} else {
|
|
|
|
|
ld.Diag("addpltsym: unsupported binary format")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Generate the glink resolver stub if necessary and return the .glink section
|
|
|
|
|
func ensureglinkresolver() *ld.LSym {
|
2015-03-02 12:35:15 -05:00
|
|
|
glink := ld.Linklookup(ld.Ctxt, ".glink", 0)
|
2015-02-27 22:57:28 -05:00
|
|
|
if glink.Size != 0 {
|
|
|
|
|
return glink
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// This is essentially the resolver from the ppc64 ELF ABI.
|
|
|
|
|
// At entry, r12 holds the address of the symbol resolver stub
|
|
|
|
|
// for the target routine and the argument registers hold the
|
|
|
|
|
// arguments for the target routine.
|
|
|
|
|
//
|
|
|
|
|
// This stub is PIC, so first get the PC of label 1 into r11.
|
|
|
|
|
// Other things will be relative to this.
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x7c0802a6) // mflr r0
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x429f0005) // bcl 20,31,1f
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x7d6802a6) // 1: mflr r11
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x7c0803a6) // mtlf r0
|
|
|
|
|
|
|
|
|
|
// Compute the .plt array index from the entry point address.
|
|
|
|
|
// Because this is PIC, everything is relative to label 1b (in
|
|
|
|
|
// r11):
|
|
|
|
|
// r0 = ((r12 - r11) - (res_0 - r11)) / 4 = (r12 - res_0) / 4
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x3800ffd0) // li r0,-(res_0-1b)=-48
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x7c006214) // add r0,r0,r12
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x7c0b0050) // sub r0,r0,r11
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x7800f082) // srdi r0,r0,2
|
|
|
|
|
|
|
|
|
|
// r11 = address of the first byte of the PLT
|
2015-03-02 12:35:15 -05:00
|
|
|
r := ld.Addrel(glink)
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
r.Off = int32(glink.Size)
|
|
|
|
|
r.Sym = ld.Linklookup(ld.Ctxt, ".plt", 0)
|
|
|
|
|
r.Siz = 8
|
2015-04-19 19:33:58 -07:00
|
|
|
r.Type = obj.R_ADDRPOWER
|
2015-02-27 22:57:28 -05:00
|
|
|
|
2015-08-31 13:51:37 +12:00
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x3d600000) // addis r11,0,.plt@ha
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x396b0000) // addi r11,r11,.plt@l
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
// Load r12 = dynamic resolver address and r11 = DSO
|
|
|
|
|
// identifier from the first two doublewords of the PLT.
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0xe98b0000) // ld r12,0(r11)
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0xe96b0008) // ld r11,8(r11)
|
|
|
|
|
|
|
|
|
|
// Jump to the dynamic resolver
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x7d8903a6) // mtctr r12
|
|
|
|
|
ld.Adduint32(ld.Ctxt, glink, 0x4e800420) // bctr
|
|
|
|
|
|
|
|
|
|
// The symbol resolvers must immediately follow.
|
|
|
|
|
// res_0:
|
|
|
|
|
|
|
|
|
|
// Add DT_PPC64_GLINK .dynamic entry, which points to 32 bytes
|
|
|
|
|
// before the first symbol resolver stub.
|
2015-03-02 12:35:15 -05:00
|
|
|
s := ld.Linklookup(ld.Ctxt, ".dynamic", 0)
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
ld.Elfwritedynentsymplus(s, ld.DT_PPC64_GLINK, glink, glink.Size-32)
|
|
|
|
|
|
|
|
|
|
return glink
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func asmb() {
|
|
|
|
|
if ld.Debug['v'] != 0 {
|
|
|
|
|
fmt.Fprintf(&ld.Bso, "%5.2f asmb\n", obj.Cputime())
|
|
|
|
|
}
|
2015-05-02 12:44:49 +10:00
|
|
|
ld.Bso.Flush()
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
if ld.Iself {
|
|
|
|
|
ld.Asmbelfsetup()
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-02 12:35:15 -05:00
|
|
|
sect := ld.Segtext.Sect
|
2015-02-27 22:57:28 -05:00
|
|
|
ld.Cseek(int64(sect.Vaddr - ld.Segtext.Vaddr + ld.Segtext.Fileoff))
|
|
|
|
|
ld.Codeblk(int64(sect.Vaddr), int64(sect.Length))
|
|
|
|
|
for sect = sect.Next; sect != nil; sect = sect.Next {
|
|
|
|
|
ld.Cseek(int64(sect.Vaddr - ld.Segtext.Vaddr + ld.Segtext.Fileoff))
|
|
|
|
|
ld.Datblk(int64(sect.Vaddr), int64(sect.Length))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ld.Segrodata.Filelen > 0 {
|
|
|
|
|
if ld.Debug['v'] != 0 {
|
|
|
|
|
fmt.Fprintf(&ld.Bso, "%5.2f rodatblk\n", obj.Cputime())
|
|
|
|
|
}
|
2015-05-02 12:44:49 +10:00
|
|
|
ld.Bso.Flush()
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
ld.Cseek(int64(ld.Segrodata.Fileoff))
|
|
|
|
|
ld.Datblk(int64(ld.Segrodata.Vaddr), int64(ld.Segrodata.Filelen))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ld.Debug['v'] != 0 {
|
|
|
|
|
fmt.Fprintf(&ld.Bso, "%5.2f datblk\n", obj.Cputime())
|
|
|
|
|
}
|
2015-05-02 12:44:49 +10:00
|
|
|
ld.Bso.Flush()
|
2015-02-27 22:57:28 -05:00
|
|
|
|
|
|
|
|
ld.Cseek(int64(ld.Segdata.Fileoff))
|
|
|
|
|
ld.Datblk(int64(ld.Segdata.Vaddr), int64(ld.Segdata.Filelen))
|
|
|
|
|
|
|
|
|
|
/* output symbol table */
|
|
|
|
|
ld.Symsize = 0
|
|
|
|
|
|
|
|
|
|
ld.Lcsize = 0
|
2015-03-02 12:35:15 -05:00
|
|
|
symo := uint32(0)
|
2015-02-27 22:57:28 -05:00
|
|
|
if ld.Debug['s'] == 0 {
|
|
|
|
|
// TODO: rationalize
|
|
|
|
|
if ld.Debug['v'] != 0 {
|
|
|
|
|
fmt.Fprintf(&ld.Bso, "%5.2f sym\n", obj.Cputime())
|
|
|
|
|
}
|
2015-05-02 12:44:49 +10:00
|
|
|
ld.Bso.Flush()
|
2015-02-27 22:57:28 -05:00
|
|
|
switch ld.HEADTYPE {
|
|
|
|
|
default:
|
|
|
|
|
if ld.Iself {
|
|
|
|
|
symo = uint32(ld.Segdata.Fileoff + ld.Segdata.Filelen)
|
|
|
|
|
symo = uint32(ld.Rnd(int64(symo), int64(ld.INITRND)))
|
|
|
|
|
}
|
|
|
|
|
|
2015-04-19 19:33:58 -07:00
|
|
|
case obj.Hplan9:
|
2015-02-27 22:57:28 -05:00
|
|
|
symo = uint32(ld.Segdata.Fileoff + ld.Segdata.Filelen)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ld.Cseek(int64(symo))
|
|
|
|
|
switch ld.HEADTYPE {
|
|
|
|
|
default:
|
|
|
|
|
if ld.Iself {
|
|
|
|
|
if ld.Debug['v'] != 0 {
|
|
|
|
|
fmt.Fprintf(&ld.Bso, "%5.2f elfsym\n", obj.Cputime())
|
|
|
|
|
}
|
|
|
|
|
ld.Asmelfsym()
|
|
|
|
|
ld.Cflush()
|
|
|
|
|
ld.Cwrite(ld.Elfstrdat)
|
|
|
|
|
|
|
|
|
|
if ld.Debug['v'] != 0 {
|
|
|
|
|
fmt.Fprintf(&ld.Bso, "%5.2f dwarf\n", obj.Cputime())
|
|
|
|
|
}
|
|
|
|
|
ld.Dwarfemitdebugsections()
|
|
|
|
|
|
|
|
|
|
if ld.Linkmode == ld.LinkExternal {
|
|
|
|
|
ld.Elfemitreloc()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-04-19 19:33:58 -07:00
|
|
|
case obj.Hplan9:
|
2015-02-27 22:57:28 -05:00
|
|
|
ld.Asmplan9sym()
|
|
|
|
|
ld.Cflush()
|
|
|
|
|
|
2015-03-02 12:35:15 -05:00
|
|
|
sym := ld.Linklookup(ld.Ctxt, "pclntab", 0)
|
2015-02-27 22:57:28 -05:00
|
|
|
if sym != nil {
|
|
|
|
|
ld.Lcsize = int32(len(sym.P))
|
2015-03-02 12:35:15 -05:00
|
|
|
for i := 0; int32(i) < ld.Lcsize; i++ {
|
2015-02-27 22:57:28 -05:00
|
|
|
ld.Cput(uint8(sym.P[i]))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ld.Cflush()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ld.Ctxt.Cursym = nil
|
|
|
|
|
if ld.Debug['v'] != 0 {
|
|
|
|
|
fmt.Fprintf(&ld.Bso, "%5.2f header\n", obj.Cputime())
|
|
|
|
|
}
|
2015-05-02 12:44:49 +10:00
|
|
|
ld.Bso.Flush()
|
2015-02-27 22:57:28 -05:00
|
|
|
ld.Cseek(0)
|
|
|
|
|
switch ld.HEADTYPE {
|
|
|
|
|
default:
|
2015-04-19 19:33:58 -07:00
|
|
|
case obj.Hplan9: /* plan 9 */
|
2015-02-27 22:57:28 -05:00
|
|
|
ld.Thearch.Lput(0x647) /* magic */
|
|
|
|
|
ld.Thearch.Lput(uint32(ld.Segtext.Filelen)) /* sizes */
|
|
|
|
|
ld.Thearch.Lput(uint32(ld.Segdata.Filelen))
|
|
|
|
|
ld.Thearch.Lput(uint32(ld.Segdata.Length - ld.Segdata.Filelen))
|
|
|
|
|
ld.Thearch.Lput(uint32(ld.Symsize)) /* nsyms */
|
|
|
|
|
ld.Thearch.Lput(uint32(ld.Entryvalue())) /* va of entry */
|
|
|
|
|
ld.Thearch.Lput(0)
|
|
|
|
|
ld.Thearch.Lput(uint32(ld.Lcsize))
|
|
|
|
|
|
2015-04-19 19:33:58 -07:00
|
|
|
case obj.Hlinux,
|
|
|
|
|
obj.Hfreebsd,
|
|
|
|
|
obj.Hnetbsd,
|
|
|
|
|
obj.Hopenbsd,
|
|
|
|
|
obj.Hnacl:
|
2015-02-27 22:57:28 -05:00
|
|
|
ld.Asmbelf(int64(symo))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ld.Cflush()
|
|
|
|
|
if ld.Debug['c'] != 0 {
|
|
|
|
|
fmt.Printf("textsize=%d\n", ld.Segtext.Filelen)
|
|
|
|
|
fmt.Printf("datsize=%d\n", ld.Segdata.Filelen)
|
|
|
|
|
fmt.Printf("bsssize=%d\n", ld.Segdata.Length-ld.Segdata.Filelen)
|
|
|
|
|
fmt.Printf("symsize=%d\n", ld.Symsize)
|
|
|
|
|
fmt.Printf("lcsize=%d\n", ld.Lcsize)
|
|
|
|
|
fmt.Printf("total=%d\n", ld.Segtext.Filelen+ld.Segdata.Length+uint64(ld.Symsize)+uint64(ld.Lcsize))
|
|
|
|
|
}
|
|
|
|
|
}
|