mirror of
https://github.com/golang/go
synced 2024-11-05 15:46:11 -07:00
[dev.link] cmd/link: convert reloc pass to using the loader when internal linking
Only enabled for AMD64 when internal linking for now. Change-Id: I2aa9ee47c0f7413ea7bbcdd31b8317c14220bba3 Reviewed-on: https://go-review.googlesource.com/c/go/+/230302 Run-TryBot: Cherry Zhang <cherryyz@google.com> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Than McIntosh <thanm@google.com>
This commit is contained in:
parent
1adae7fe76
commit
76c6cce116
@ -841,7 +841,7 @@ func asmb2(ctxt *ld.Link) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func tlsIEtoLE(s *sym.Symbol, off, size int) {
|
func tlsIEtoLE(P []byte, off, size int) {
|
||||||
// Transform the PC-relative instruction into a constant load.
|
// Transform the PC-relative instruction into a constant load.
|
||||||
// That is,
|
// That is,
|
||||||
//
|
//
|
||||||
@ -852,7 +852,7 @@ func tlsIEtoLE(s *sym.Symbol, off, size int) {
|
|||||||
if off < 3 {
|
if off < 3 {
|
||||||
log.Fatal("R_X86_64_GOTTPOFF reloc not preceded by MOVQ or ADDQ instruction")
|
log.Fatal("R_X86_64_GOTTPOFF reloc not preceded by MOVQ or ADDQ instruction")
|
||||||
}
|
}
|
||||||
op := s.P[off-3 : off]
|
op := P[off-3 : off]
|
||||||
reg := op[2] >> 3
|
reg := op[2] >> 3
|
||||||
|
|
||||||
if op[1] == 0x8b || reg == 4 {
|
if op[1] == 0x8b || reg == 4 {
|
||||||
|
@ -271,7 +271,6 @@ func relocsym(target *Target, ldr *loader.Loader, err *ErrorReporter, syms *Arch
|
|||||||
log.Fatalf("unexpected R_TLS_LE relocation for %v", target.HeadType)
|
log.Fatalf("unexpected R_TLS_LE relocation for %v", target.HeadType)
|
||||||
}
|
}
|
||||||
case objabi.R_TLS_IE:
|
case objabi.R_TLS_IE:
|
||||||
panic("not implemented")
|
|
||||||
//if target.IsExternal() && target.IsElf() {
|
//if target.IsExternal() && target.IsElf() {
|
||||||
// r.Done = false
|
// r.Done = false
|
||||||
// if r.Sym == nil {
|
// if r.Sym == nil {
|
||||||
@ -285,17 +284,17 @@ func relocsym(target *Target, ldr *loader.Loader, err *ErrorReporter, syms *Arch
|
|||||||
// }
|
// }
|
||||||
// break
|
// break
|
||||||
//}
|
//}
|
||||||
//if target.IsPIE() && target.IsElf() {
|
if target.IsPIE() && target.IsElf() {
|
||||||
// // We are linking the final executable, so we
|
// We are linking the final executable, so we
|
||||||
// // can optimize any TLS IE relocation to LE.
|
// can optimize any TLS IE relocation to LE.
|
||||||
// if thearch.TLSIEtoLE == nil {
|
if thearch.TLSIEtoLE == nil {
|
||||||
// log.Fatalf("internal linking of TLS IE not supported on %v", target.Arch.Family)
|
log.Fatalf("internal linking of TLS IE not supported on %v", target.Arch.Family)
|
||||||
// }
|
}
|
||||||
// thearch.TLSIEtoLE(ldr, s, int(off), int(siz))
|
thearch.TLSIEtoLE(P, int(off), int(siz))
|
||||||
// o = int64(syms.Tlsoffset)
|
o = int64(syms.Tlsoffset)
|
||||||
//} else {
|
} else {
|
||||||
// log.Fatalf("cannot handle R_TLS_IE (sym %s) when linking internally", ldr.SymName(s))
|
log.Fatalf("cannot handle R_TLS_IE (sym %s) when linking internally", ldr.SymName(s))
|
||||||
//}
|
}
|
||||||
case objabi.R_ADDR:
|
case objabi.R_ADDR:
|
||||||
//if target.IsExternal() && r.Sym.Type != sym.SCONST {
|
//if target.IsExternal() && r.Sym.Type != sym.SCONST {
|
||||||
// r.Done = false
|
// r.Done = false
|
||||||
@ -562,22 +561,22 @@ func (ctxt *Link) reloc() {
|
|||||||
wg.Add(3)
|
wg.Add(3)
|
||||||
go func() {
|
go func() {
|
||||||
if !ctxt.IsWasm() { // On Wasm, text relocations are applied in Asmb2.
|
if !ctxt.IsWasm() { // On Wasm, text relocations are applied in Asmb2.
|
||||||
for _, s := range ctxt.Textp {
|
for _, s := range ctxt.Textp2 {
|
||||||
relocsym2(target, ldr, reporter, syms, s)
|
relocsym(target, ldr, reporter, syms, s, ldr.OutData(s))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
wg.Done()
|
wg.Done()
|
||||||
}()
|
}()
|
||||||
go func() {
|
go func() {
|
||||||
for _, s := range ctxt.datap {
|
for _, s := range ctxt.datap2 {
|
||||||
relocsym2(target, ldr, reporter, syms, s)
|
relocsym(target, ldr, reporter, syms, s, ldr.OutData(s))
|
||||||
}
|
}
|
||||||
wg.Done()
|
wg.Done()
|
||||||
}()
|
}()
|
||||||
go func() {
|
go func() {
|
||||||
for _, si := range dwarfp {
|
for _, si := range dwarfp2 {
|
||||||
for _, s := range si.syms {
|
for _, s := range si.syms {
|
||||||
relocsym2(target, ldr, reporter, syms, s)
|
relocsym(target, ldr, reporter, syms, s, ldr.OutData(s))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
wg.Done()
|
wg.Done()
|
||||||
|
@ -11,6 +11,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
"strings"
|
"strings"
|
||||||
|
"sync"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Temporary dumping around for sym.Symbol version of helper
|
// Temporary dumping around for sym.Symbol version of helper
|
||||||
@ -199,7 +200,7 @@ func relocsym2(target *Target, ldr *loader.Loader, err *ErrorReporter, syms *Arc
|
|||||||
if thearch.TLSIEtoLE == nil {
|
if thearch.TLSIEtoLE == nil {
|
||||||
log.Fatalf("internal linking of TLS IE not supported on %v", target.Arch.Family)
|
log.Fatalf("internal linking of TLS IE not supported on %v", target.Arch.Family)
|
||||||
}
|
}
|
||||||
thearch.TLSIEtoLE(s, int(off), int(r.Siz))
|
thearch.TLSIEtoLE(s.P, int(off), int(r.Siz))
|
||||||
o = int64(syms.Tlsoffset)
|
o = int64(syms.Tlsoffset)
|
||||||
// TODO: o += r.Add when !target.IsAmd64()?
|
// TODO: o += r.Add when !target.IsAmd64()?
|
||||||
// Why do we treat r.Add differently on AMD64?
|
// Why do we treat r.Add differently on AMD64?
|
||||||
@ -483,3 +484,35 @@ func relocsym2(target *Target, ldr *loader.Loader, err *ErrorReporter, syms *Arc
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (ctxt *Link) reloc2() {
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
target := &ctxt.Target
|
||||||
|
ldr := ctxt.loader
|
||||||
|
reporter := &ctxt.ErrorReporter
|
||||||
|
syms := &ctxt.ArchSyms
|
||||||
|
wg.Add(3)
|
||||||
|
go func() {
|
||||||
|
if !ctxt.IsWasm() { // On Wasm, text relocations are applied in Asmb2.
|
||||||
|
for _, s := range ctxt.Textp {
|
||||||
|
relocsym2(target, ldr, reporter, syms, s)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
wg.Done()
|
||||||
|
}()
|
||||||
|
go func() {
|
||||||
|
for _, s := range ctxt.datap {
|
||||||
|
relocsym2(target, ldr, reporter, syms, s)
|
||||||
|
}
|
||||||
|
wg.Done()
|
||||||
|
}()
|
||||||
|
go func() {
|
||||||
|
for _, si := range dwarfp {
|
||||||
|
for _, s := range si.syms {
|
||||||
|
relocsym2(target, ldr, reporter, syms, s)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
wg.Done()
|
||||||
|
}()
|
||||||
|
wg.Wait()
|
||||||
|
}
|
||||||
|
@ -282,7 +282,7 @@ type Arch struct {
|
|||||||
// This is possible when a TLS IE relocation refers to a local
|
// This is possible when a TLS IE relocation refers to a local
|
||||||
// symbol in an executable, which is typical when internally
|
// symbol in an executable, which is typical when internally
|
||||||
// linking PIE binaries.
|
// linking PIE binaries.
|
||||||
TLSIEtoLE func(s *sym.Symbol, off, size int)
|
TLSIEtoLE func(P []byte, off, size int)
|
||||||
|
|
||||||
// optional override for assignAddress
|
// optional override for assignAddress
|
||||||
AssignAddress func(ldr *loader.Loader, sect *sym.Section, n int, s loader.Sym, va uint64, isTramp bool) (*sym.Section, int, uint64)
|
AssignAddress func(ldr *loader.Loader, sect *sym.Section, n int, s loader.Sym, va uint64, isTramp bool) (*sym.Section, int, uint64)
|
||||||
@ -2824,10 +2824,9 @@ func addToTextp(ctxt *Link) {
|
|||||||
ctxt.Textp = textp
|
ctxt.Textp = textp
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ctxt *Link) loadlibfull(symGroupType []sym.SymKind) {
|
func (ctxt *Link) loadlibfull(symGroupType []sym.SymKind, needReloc bool) {
|
||||||
|
|
||||||
// Load full symbol contents, resolve indexed references.
|
// Load full symbol contents, resolve indexed references.
|
||||||
ctxt.loader.LoadFull(ctxt.Arch, ctxt.Syms)
|
ctxt.loader.LoadFull(ctxt.Arch, ctxt.Syms, needReloc)
|
||||||
|
|
||||||
// Convert ctxt.Moduledata2 to ctxt.Moduledata, etc
|
// Convert ctxt.Moduledata2 to ctxt.Moduledata, etc
|
||||||
if ctxt.Moduledata2 != 0 {
|
if ctxt.Moduledata2 != 0 {
|
||||||
|
@ -325,10 +325,22 @@ func Main(arch *sys.Arch, theArch Arch) {
|
|||||||
bench.Start("Asmb")
|
bench.Start("Asmb")
|
||||||
ctxt.loader.InitOutData()
|
ctxt.loader.InitOutData()
|
||||||
thearch.Asmb(ctxt, ctxt.loader)
|
thearch.Asmb(ctxt, ctxt.loader)
|
||||||
bench.Start("loadlibfull")
|
|
||||||
ctxt.loadlibfull(symGroupType) // XXX do it here for now
|
newreloc := ctxt.IsInternal() && ctxt.IsAMD64()
|
||||||
bench.Start("reloc")
|
if newreloc {
|
||||||
ctxt.reloc()
|
bench.Start("reloc")
|
||||||
|
ctxt.reloc()
|
||||||
|
bench.Start("loadlibfull")
|
||||||
|
// We don't need relocations at this point.
|
||||||
|
// An exception is Windows, see pe.go:addPEBaseRelocSym
|
||||||
|
needReloc := ctxt.IsWindows()
|
||||||
|
ctxt.loadlibfull(symGroupType, needReloc) // XXX do it here for now
|
||||||
|
} else {
|
||||||
|
bench.Start("loadlibfull")
|
||||||
|
ctxt.loadlibfull(symGroupType, true) // XXX do it here for now
|
||||||
|
bench.Start("reloc")
|
||||||
|
ctxt.reloc2()
|
||||||
|
}
|
||||||
bench.Start("Asmb2")
|
bench.Start("Asmb2")
|
||||||
thearch.Asmb2(ctxt)
|
thearch.Asmb2(ctxt)
|
||||||
|
|
||||||
|
@ -2016,7 +2016,7 @@ func (l *Loader) preprocess(arch *sys.Arch, s Sym, name string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Load full contents.
|
// Load full contents.
|
||||||
func (l *Loader) LoadFull(arch *sys.Arch, syms *sym.Symbols) {
|
func (l *Loader) LoadFull(arch *sys.Arch, syms *sym.Symbols, needReloc bool) {
|
||||||
// create all Symbols first.
|
// create all Symbols first.
|
||||||
l.growSyms(l.NSym())
|
l.growSyms(l.NSym())
|
||||||
l.growSects(l.NSym())
|
l.growSects(l.NSym())
|
||||||
@ -2049,7 +2049,9 @@ func (l *Loader) LoadFull(arch *sys.Arch, syms *sym.Symbols) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// allocate a single large slab of relocations for all live symbols
|
// allocate a single large slab of relocations for all live symbols
|
||||||
l.relocBatch = make([]sym.Reloc, nr)
|
if needReloc {
|
||||||
|
l.relocBatch = make([]sym.Reloc, nr)
|
||||||
|
}
|
||||||
|
|
||||||
// convert payload-based external symbols into sym.Symbol-based
|
// convert payload-based external symbols into sym.Symbol-based
|
||||||
for _, i := range toConvert {
|
for _, i := range toConvert {
|
||||||
@ -2062,11 +2064,13 @@ func (l *Loader) LoadFull(arch *sys.Arch, syms *sym.Symbols) {
|
|||||||
s.Size = pp.size
|
s.Size = pp.size
|
||||||
|
|
||||||
// Copy relocations
|
// Copy relocations
|
||||||
batch := l.relocBatch
|
if needReloc {
|
||||||
s.R = batch[:len(pp.relocs):len(pp.relocs)]
|
batch := l.relocBatch
|
||||||
l.relocBatch = batch[len(pp.relocs):]
|
s.R = batch[:len(pp.relocs):len(pp.relocs)]
|
||||||
relocs := l.Relocs(i)
|
l.relocBatch = batch[len(pp.relocs):]
|
||||||
l.convertRelocations(i, &relocs, s, false)
|
relocs := l.Relocs(i)
|
||||||
|
l.convertRelocations(i, &relocs, s, false)
|
||||||
|
}
|
||||||
|
|
||||||
// Copy data
|
// Copy data
|
||||||
s.P = pp.data
|
s.P = pp.data
|
||||||
@ -2077,7 +2081,7 @@ func (l *Loader) LoadFull(arch *sys.Arch, syms *sym.Symbols) {
|
|||||||
|
|
||||||
// load contents of defined symbols
|
// load contents of defined symbols
|
||||||
for _, o := range l.objs[1:] {
|
for _, o := range l.objs[1:] {
|
||||||
loadObjFull(l, o.r)
|
loadObjFull(l, o.r, needReloc)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Note: resolution of ABI aliases is now also handled in
|
// Note: resolution of ABI aliases is now also handled in
|
||||||
@ -2598,7 +2602,7 @@ func (l *Loader) CreateStaticSym(name string) Sym {
|
|||||||
return l.newExtSym(name, l.anonVersion)
|
return l.newExtSym(name, l.anonVersion)
|
||||||
}
|
}
|
||||||
|
|
||||||
func loadObjFull(l *Loader, r *oReader) {
|
func loadObjFull(l *Loader, r *oReader, needReloc bool) {
|
||||||
for i, n := 0, r.NSym()+r.NNonpkgdef(); i < n; i++ {
|
for i, n := 0, r.NSym()+r.NNonpkgdef(); i < n; i++ {
|
||||||
// A symbol may be a dup or overwritten. In this case, its
|
// A symbol may be a dup or overwritten. In this case, its
|
||||||
// content will actually be provided by a different object
|
// content will actually be provided by a different object
|
||||||
@ -2623,11 +2627,13 @@ func loadObjFull(l *Loader, r *oReader) {
|
|||||||
s.P = l.OutData(gi)
|
s.P = l.OutData(gi)
|
||||||
|
|
||||||
// Relocs
|
// Relocs
|
||||||
relocs := l.relocs(r, i)
|
if needReloc {
|
||||||
batch := l.relocBatch
|
relocs := l.relocs(r, i)
|
||||||
s.R = batch[:relocs.Count():relocs.Count()]
|
batch := l.relocBatch
|
||||||
l.relocBatch = batch[relocs.Count():]
|
s.R = batch[:relocs.Count():relocs.Count()]
|
||||||
l.convertRelocations(gi, &relocs, s, false)
|
l.relocBatch = batch[relocs.Count():]
|
||||||
|
l.convertRelocations(gi, &relocs, s, false)
|
||||||
|
}
|
||||||
|
|
||||||
// Aux symbol info
|
// Aux symbol info
|
||||||
auxs := r.Auxs(i)
|
auxs := r.Auxs(i)
|
||||||
|
Loading…
Reference in New Issue
Block a user