mirror of
https://github.com/google/gops.git
synced 2024-11-24 08:22:25 +02:00
209 lines
4.7 KiB
Go
209 lines
4.7 KiB
Go
// Copyright 2013 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package obj
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"strings"
|
|
)
|
|
|
|
type Plist struct {
|
|
Firstpc *Prog
|
|
}
|
|
|
|
/*
|
|
* start a new Prog list.
|
|
*/
|
|
func Linknewplist(ctxt *Link) *Plist {
|
|
pl := new(Plist)
|
|
ctxt.Plists = append(ctxt.Plists, pl)
|
|
return pl
|
|
}
|
|
|
|
func Flushplist(ctxt *Link) {
|
|
flushplist(ctxt, ctxt.Debugasm == 0)
|
|
}
|
|
func FlushplistNoFree(ctxt *Link) {
|
|
flushplist(ctxt, false)
|
|
}
|
|
func flushplist(ctxt *Link, freeProgs bool) {
|
|
// Build list of symbols, and assign instructions to lists.
|
|
// Ignore ctxt->plist boundaries. There are no guarantees there,
|
|
// and the assemblers just use one big list.
|
|
var curtext *LSym
|
|
var etext *Prog
|
|
var text []*LSym
|
|
|
|
for _, pl := range ctxt.Plists {
|
|
var plink *Prog
|
|
for p := pl.Firstpc; p != nil; p = plink {
|
|
if ctxt.Debugasm != 0 && ctxt.Debugvlog != 0 {
|
|
fmt.Printf("obj: %v\n", p)
|
|
}
|
|
plink = p.Link
|
|
p.Link = nil
|
|
|
|
switch p.As {
|
|
case AEND:
|
|
continue
|
|
|
|
case ATYPE:
|
|
// Assume each TYPE instruction describes
|
|
// a different local variable or parameter,
|
|
// so no dedup.
|
|
// Using only the TYPE instructions means
|
|
// that we discard location information about local variables
|
|
// in C and assembly functions; that information is inferred
|
|
// from ordinary references, because there are no TYPE
|
|
// instructions there. Without the type information, gdb can't
|
|
// use the locations, so we don't bother to save them.
|
|
// If something else could use them, we could arrange to
|
|
// preserve them.
|
|
if curtext == nil {
|
|
continue
|
|
}
|
|
a := new(Auto)
|
|
a.Asym = p.From.Sym
|
|
a.Aoffset = int32(p.From.Offset)
|
|
a.Name = int16(p.From.Name)
|
|
a.Gotype = p.To.Sym
|
|
a.Link = curtext.Autom
|
|
curtext.Autom = a
|
|
continue
|
|
|
|
case ATEXT:
|
|
s := p.From.Sym
|
|
if s == nil {
|
|
// func _() { }
|
|
curtext = nil
|
|
|
|
continue
|
|
}
|
|
|
|
if s.Text != nil {
|
|
log.Fatalf("duplicate TEXT for %s", s.Name)
|
|
}
|
|
if s.OnList() {
|
|
log.Fatalf("symbol %s listed multiple times", s.Name)
|
|
}
|
|
s.Set(AttrOnList, true)
|
|
text = append(text, s)
|
|
flag := int(p.From3Offset())
|
|
if flag&DUPOK != 0 {
|
|
s.Set(AttrDuplicateOK, true)
|
|
}
|
|
if flag&NOSPLIT != 0 {
|
|
s.Set(AttrNoSplit, true)
|
|
}
|
|
if flag&REFLECTMETHOD != 0 {
|
|
s.Set(AttrReflectMethod, true)
|
|
}
|
|
s.Type = STEXT
|
|
s.Text = p
|
|
etext = p
|
|
curtext = s
|
|
continue
|
|
|
|
case AFUNCDATA:
|
|
// Rewrite reference to go_args_stackmap(SB) to the Go-provided declaration information.
|
|
if curtext == nil { // func _() {}
|
|
continue
|
|
}
|
|
if p.To.Sym.Name == "go_args_stackmap" {
|
|
if p.From.Type != TYPE_CONST || p.From.Offset != FUNCDATA_ArgsPointerMaps {
|
|
ctxt.Diag("FUNCDATA use of go_args_stackmap(SB) without FUNCDATA_ArgsPointerMaps")
|
|
}
|
|
p.To.Sym = Linklookup(ctxt, fmt.Sprintf("%s.args_stackmap", curtext.Name), int(curtext.Version))
|
|
}
|
|
|
|
}
|
|
|
|
if curtext == nil {
|
|
etext = nil
|
|
continue
|
|
}
|
|
etext.Link = p
|
|
etext = p
|
|
}
|
|
}
|
|
|
|
// Add reference to Go arguments for C or assembly functions without them.
|
|
for _, s := range text {
|
|
if !strings.HasPrefix(s.Name, "\"\".") {
|
|
continue
|
|
}
|
|
found := false
|
|
var p *Prog
|
|
for p = s.Text; p != nil; p = p.Link {
|
|
if p.As == AFUNCDATA && p.From.Type == TYPE_CONST && p.From.Offset == FUNCDATA_ArgsPointerMaps {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
|
|
if !found {
|
|
p = Appendp(ctxt, s.Text)
|
|
p.As = AFUNCDATA
|
|
p.From.Type = TYPE_CONST
|
|
p.From.Offset = FUNCDATA_ArgsPointerMaps
|
|
p.To.Type = TYPE_MEM
|
|
p.To.Name = NAME_EXTERN
|
|
p.To.Sym = Linklookup(ctxt, fmt.Sprintf("%s.args_stackmap", s.Name), int(s.Version))
|
|
}
|
|
}
|
|
|
|
// Turn functions into machine code images.
|
|
for _, s := range text {
|
|
mkfwd(s)
|
|
linkpatch(ctxt, s)
|
|
if ctxt.Flag_optimize {
|
|
ctxt.Arch.Follow(ctxt, s)
|
|
}
|
|
ctxt.Arch.Preprocess(ctxt, s)
|
|
ctxt.Arch.Assemble(ctxt, s)
|
|
fieldtrack(ctxt, s)
|
|
linkpcln(ctxt, s)
|
|
if freeProgs {
|
|
s.Text = nil
|
|
}
|
|
}
|
|
|
|
// Add to running list in ctxt.
|
|
ctxt.Text = append(ctxt.Text, text...)
|
|
ctxt.Data = append(ctxt.Data, gendwarf(ctxt, text)...)
|
|
ctxt.Plists = nil
|
|
ctxt.Curp = nil
|
|
if freeProgs {
|
|
ctxt.freeProgs()
|
|
}
|
|
}
|
|
|
|
func (ctxt *Link) Globl(s *LSym, size int64, flag int) {
|
|
if s.SeenGlobl() {
|
|
fmt.Printf("duplicate %v\n", s)
|
|
}
|
|
s.Set(AttrSeenGlobl, true)
|
|
if s.OnList() {
|
|
log.Fatalf("symbol %s listed multiple times", s.Name)
|
|
}
|
|
s.Set(AttrOnList, true)
|
|
ctxt.Data = append(ctxt.Data, s)
|
|
s.Size = size
|
|
if s.Type == 0 || s.Type == SXREF {
|
|
s.Type = SBSS
|
|
}
|
|
if flag&DUPOK != 0 {
|
|
s.Set(AttrDuplicateOK, true)
|
|
}
|
|
if flag&RODATA != 0 {
|
|
s.Type = SRODATA
|
|
} else if flag&NOPTR != 0 {
|
|
s.Type = SNOPTRBSS
|
|
} else if flag&TLSBSS != 0 {
|
|
s.Type = STLSBSS
|
|
}
|
|
}
|