Skip to content
Snippets Groups Projects
loader.go 50.5 KiB
Newer Older
  • Learn to ignore specific revisions
  • // Copyright 2019 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.
    
    
    	"cmd/internal/goobj2"
    	"cmd/internal/obj"
    	"cmd/internal/objabi"
    	"cmd/internal/sys"
    	"cmd/link/internal/sym"
    	"fmt"
    	"log"
    	"os"
    	"sort"
    	"strconv"
    	"strings"
    )
    
    var _ = fmt.Print
    
    
    // Sym encapsulates a global symbol index, used to identify a specific
    // Go symbol. The 0-valued Sym is corresponds to an invalid symbol.
    type Sym int
    
    
    // Relocs encapsulates the set of relocations on a given symbol; an
    // instance of this type is returned by the Loader Relocs() method.
    type Relocs struct {
    	Count int // number of relocs
    
    	li int      // local index of symbol whose relocs we're examining
    	r  *oReader // object reader for containing package
    	l  *Loader  // loader
    
    	extIdx Sym // index of external symbol we're examining or 0
    
    }
    
    // Reloc contains the payload for a specific relocation.
    // TODO: replace this with sym.Reloc, once we change the
    // relocation target from "*sym.Symbol" to "loader.Sym" in sym.Reloc.
    type Reloc struct {
    	Off  int32            // offset to rewrite
    	Size uint8            // number of bytes to rewrite: 0, 1, 2, or 4
    	Type objabi.RelocType // the relocation type
    	Add  int64            // addend
    	Sym  Sym              // global index of symbol the reloc addresses
    }
    
    
    // oReader is a wrapper type of obj.Reader, along with some
    // extra information.
    // TODO: rename to objReader once the old one is gone?
    type oReader struct {
    	*goobj2.Reader
    	unit      *sym.CompilationUnit
    
    	version   int    // version of static symbol
    	flags     uint32 // read from object file
    
    	rcache    []Sym // cache mapping local PkgNone symbol to resolved Sym
    
    type objIdx struct {
    
    }
    
    type nameVer struct {
    	name string
    	v    int
    }
    
    
    type bitmap []uint32
    
    // set the i-th bit.
    
    func (bm bitmap) set(i Sym) {
    
    	n, r := uint(i)/32, uint(i)%32
    	bm[n] |= 1 << r
    }
    
    
    // unset the i-th bit.
    func (bm bitmap) unset(i Sym) {
    	n, r := uint(i)/32, uint(i)%32
    	bm[n] &^= (1 << r)
    }
    
    
    // whether the i-th bit is set.
    
    func (bm bitmap) has(i Sym) bool {
    
    	n, r := uint(i)/32, uint(i)%32
    	return bm[n]&(1<<r) != 0
    }
    
    
    // return current length of bitmap in bits.
    func (bm bitmap) len() int {
    	return len(bm) * 32
    }
    
    func makeBitmap(n int) bitmap {
    	return make(bitmap, (n+31)/32)
    }
    
    
    // growBitmap insures that the specified bitmap has enough capacity,
    // reallocating (doubling the size) if needed.
    func growBitmap(reqLen int, b bitmap) bitmap {
    	curLen := b.len()
    	if reqLen > curLen {
    
    		b = append(b, makeBitmap(reqLen+1-curLen)...)
    
    // A Loader loads new object files and resolves indexed symbol references.
    
    //
    // Notes on the layout of global symbol index space:
    //
    // - Go object files are read before host object files; each Go object
    //   read allocates a new chunk of global index space of size P + NP,
    //   where P is the number of package defined symbols in the object and
    //   NP is the number of non-package defined symbols.
    //
    // - In loader.LoadRefs(), the loader makes a sweep through all of the
    //   non-package references in each object file and allocates sym indices
    //   for any symbols that have not yet been defined (start of this space
    //   is marked by loader.extStart).
    //
    // - Host object file loading happens; the host object loader does a
    //   name/version lookup for each symbol it finds; this can wind up
    //   extending the external symbol index space range. The host object
    //   loader currently stores symbol payloads in sym.Symbol objects,
    //   which get handed off to the loader.
    //
    // - A given external symbol (Sym) either has a sym.Symbol acting as
    //   its backing store (this will continue to be the case until we
    //   finish rewriting the host object loader to work entirely with
    //   loader.Sym) or it has a "payload" backing store (represented by
    //   extSymPayload). Newly created external symbols (created by
    //   a call to AddExtSym or equivalent) start out in the "has payload"
    //   state, and continue until installSym is called for the sym
    //   index in question.
    //
    // - At some point (when the wayfront is pushed through all of the
    //   linker), all external symbols will be payload-based, and we can
    //   get rid of the loader.Syms array.
    //
    
    	start       map[*oReader]Sym // map from object file to its start index
    	objs        []objIdx         // sorted by start index (i.e. objIdx.i)
    	max         Sym              // current max index
    	extStart    Sym              // from this index on, the symbols are externally defined
    	builtinSyms []Sym            // global index of builtin symbols
    
    	ocache      int              // index (into 'objs') of most recent lookup
    
    	symsByName    [2]map[string]Sym // map symbol name to index, two maps are for ABI0 and ABIInternal
    	extStaticSyms map[nameVer]Sym   // externally defined static symbols, keyed by name
    	overwrite     map[Sym]Sym       // overwrite[i]=j if symbol j overwrites symbol i
    
    	payloads []extSymPayload // contents of linker-materialized external syms
    
    	values   []int64         // symbol values, indexed by global sym index
    
    	itablink map[Sym]struct{} // itablink[j] defined if j is go.itablink.*
    
    
    	objByPkg map[string]*oReader // map package path to its Go object reader
    
    
    	Syms     []*sym.Symbol // indexed symbols. XXX we still make sym.Symbol for now.
    	symBatch []sym.Symbol  // batch of symbols.
    
    	anonVersion int // most recently assigned ext static sym pseudo-version
    
    
    	// Bitmaps and other side structures used to store data used to store
    	// symbol flags/attributes; these are to be accessed via the
    	// corresponding loader "AttrXXX" and "SetAttrXXX" methods. Please
    	// visit the comments on these methods for more details on the
    	// semantics / interpretation of the specific flags or attribute.
    	attrReachable        bitmap // reachable symbols, indexed by global index
    	attrOnList           bitmap // "on list" symbols, indexed by global index
    	attrVisibilityHidden bitmap // hidden symbols, indexed by ext sym index
    	attrDuplicateOK      bitmap // dupOK symbols, indexed by ext sym index
    	attrShared           bitmap // shared symbols, indexed by ext sym index
    	attrExternal         bitmap // external symbols, indexed by ext sym index
    
    	attrTopFrame         map[Sym]struct{} // top frame symbols
    	attrSpecial          map[Sym]struct{} // "special" frame symbols
    	attrCgoExportDynamic map[Sym]struct{} // "cgo_export_dynamic" symbols
    	attrCgoExportStatic  map[Sym]struct{} // "cgo_export_static" symbols
    
    
    	// Used to implement field tracking; created during deadcode if
    	// field tracking is enabled. Reachparent[K] contains the index of
    	// the symbol that triggered the marking of symbol K as live.
    	Reachparent []Sym
    
    
    	relocBatch []sym.Reloc // for bulk allocation of relocations
    
    
    	flags uint32
    
    	strictDupMsgs int // number of strict-dup warning/errors, when FlagStrictDups is enabled
    
    // extSymPayload holds the payload (data + relocations) for linker-synthesized
    
    // external symbols (note that symbol value is stored in a separate slice).
    
    type extSymPayload struct {
    	name   string // TODO: would this be better as offset into str table?
    	size   int64
    	ver    int
    	kind   sym.SymKind
    	relocs []Reloc
    	data   []byte
    }
    
    
    const (
    	// Loader.flags
    	FlagStrictDups = 1 << iota
    )
    
    func NewLoader(flags uint32) *Loader {
    
    		start:         make(map[*oReader]Sym),
    
    		symsByName:    [2]map[string]Sym{make(map[string]Sym), make(map[string]Sym)},
    		objByPkg:      make(map[string]*oReader),
    		overwrite:     make(map[Sym]Sym),
    		itablink:      make(map[Sym]struct{}),
    		extStaticSyms: make(map[nameVer]Sym),
    
    		builtinSyms:   make([]Sym, nbuiltin),
    
    	}
    }
    
    // Return the start index in the global index space for a given object file.
    
    func (l *Loader) startIndex(r *oReader) Sym {
    
    	return l.start[r]
    }
    
    // Add object file r, return the start index.
    
    func (l *Loader) addObj(pkg string, r *oReader) Sym {
    
    	if _, ok := l.start[r]; ok {
    		panic("already added")
    	}
    
    	pkg = objabi.PathToPrefix(pkg) // the object file contains escaped package path
    
    	if _, ok := l.objByPkg[pkg]; !ok {
    		l.objByPkg[pkg] = r
    	}
    
    	n := r.NSym() + r.NNonpkgdef()
    	i := l.max + 1
    	l.start[r] = i
    
    	l.objs = append(l.objs, objIdx{r, i, i + Sym(n) - 1})
    
    	return i
    }
    
    // Add a symbol with a given index, return if it is added.
    
    func (l *Loader) AddSym(name string, ver int, i Sym, r *oReader, dupok bool, typ sym.SymKind) bool {
    
    	if l.extStart != 0 {
    		panic("AddSym called after AddExtSym is called")
    	}
    
    	if ver == r.version {
    		// Static symbol. Add its global index but don't
    		// add to name lookup table, as it cannot be
    		// referenced by name.
    		return true
    	}
    
    	if oldi, ok := l.symsByName[ver][name]; ok {
    
    			if l.flags&FlagStrictDups != 0 {
    				l.checkdup(name, i, r, oldi)
    			}
    
    		oldr, li := l.toLocal(oldi)
    		oldsym := goobj2.Sym{}
    		oldsym.Read(oldr.Reader, oldr.SymOff(li))
    		if oldsym.Dupok() {
    			return false
    		}
    
    		overwrite := r.DataSize(int(i-l.startIndex(r))) != 0
    
    		if overwrite {
    			// new symbol overwrites old symbol.
    			oldtyp := sym.AbiSymKindToSymKind[objabi.SymKind(oldsym.Type)]
    
    			if !oldtyp.IsData() && r.DataSize(li) == 0 {
    
    				log.Fatalf("duplicated definition of symbol " + name)
    			}
    			l.overwrite[oldi] = i
    		} else {
    			// old symbol overwrites new symbol.
    			if typ != sym.SDATA && typ != sym.SNOPTRDATA && typ != sym.SBSS && typ != sym.SNOPTRBSS { // only allow overwriting data symbol
    				log.Fatalf("duplicated definition of symbol " + name)
    			}
    			l.overwrite[i] = oldi
    
    	l.symsByName[ver][name] = i
    
    // newExtSym creates a new external sym with the specified
    // name/version.
    func (l *Loader) newExtSym(name string, ver int) Sym {
    	l.max++
    	i := l.max
    	if l.extStart == 0 {
    		l.extStart = i
    	}
    	l.growSyms(int(i))
    	pi := i - l.extStart
    	l.payloads[pi].name = name
    	l.payloads[pi].ver = ver
    	return i
    }
    
    
    // Add an external symbol (without index). Return the index of newly added
    // symbol, or 0 if not added.
    
    func (l *Loader) AddExtSym(name string, ver int) Sym {
    
    	i := l.Lookup(name, ver)
    	if i != 0 {
    		return 0
    	}
    	i = l.newExtSym(name, ver)
    	static := ver >= sym.SymVerStatic || ver < 0
    
    		l.extStaticSyms[nameVer{name, ver}] = i
    
    	return i
    }
    
    // LookupOrCreateSym looks up the symbol with the specified name/version,
    // returning its Sym index if found. If the lookup fails, a new external
    // Sym will be created, entered into the lookup tables, and returned.
    func (l *Loader) LookupOrCreateSym(name string, ver int) Sym {
    	i := l.Lookup(name, ver)
    	if i != 0 {
    		return i
    	}
    	i = l.newExtSym(name, ver)
    	static := ver >= sym.SymVerStatic || ver < 0
    
    	if static {
    		l.extStaticSyms[nameVer{name, ver}] = i
    	} else {
    		l.symsByName[ver][name] = i
    	}
    
    func (l *Loader) IsExternal(i Sym) bool {
    
    	return l.extStart != 0 && i >= l.extStart
    }
    
    
    // getPayload returns a pointer to the extSymPayload struct for an
    // external symbol if the symbol has a payload, or nil if the
    // data for the sym is being stored in a sym.Symbol. Will panic if
    // the symbol in question is bogus (zero or not an external sym).
    func (l *Loader) getPayload(i Sym) *extSymPayload {
    	if l.extStart == 0 || i < l.extStart {
    		panic(fmt.Sprintf("bogus symbol index %d in getPayload", i))
    	}
    	if l.Syms[i] != nil {
    		return nil
    	}
    	pi := i - l.extStart
    	return &l.payloads[pi]
    }
    
    
    // Ensure Syms slice has enough space, as well as growing the
    // 'payloads' slice.
    
    func (l *Loader) growSyms(i int) {
    	n := len(l.Syms)
    	if n > i {
    		return
    	}
    	l.Syms = append(l.Syms, make([]*sym.Symbol, i+1-n)...)
    
    	l.payloads = append(l.payloads, make([]extSymPayload, i+1-n)...)
    
    	l.growAttrBitmaps(int(i) + 1)
    
    // Convert a local index to a global index.
    
    func (l *Loader) toGlobal(r *oReader, i int) Sym {
    	g := l.startIndex(r) + Sym(i)
    
    	if ov, ok := l.overwrite[g]; ok {
    		return ov
    	}
    	return g
    
    // Convert a global index to a local index.
    
    func (l *Loader) toLocal(i Sym) (*oReader, int) {
    
    	if ov, ok := l.overwrite[i]; ok {
    		i = ov
    	}
    
    		return nil, int(i - l.extStart)
    
    	oc := l.ocache
    	if oc != 0 && i >= l.objs[oc].i && i <= l.objs[oc].e {
    		return l.objs[oc].r, int(i - l.objs[oc].i)
    	}
    
    	// Search for the local object holding index i.
    	// Below k is the first one that has its start index > i,
    	// so k-1 is the one we want.
    	k := sort.Search(len(l.objs), func(k int) bool {
    		return l.objs[k].i > i
    	})
    
    	return l.objs[k-1].r, int(i - l.objs[k-1].i)
    
    // rcacheGet checks for a valid entry for 's' in the readers cache,
    // where 's' is a local PkgIdxNone ref or def, or zero if
    // the cache is empty or doesn't contain a value for 's'.
    func (or *oReader) rcacheGet(symIdx uint32) Sym {
    	if len(or.rcache) > 0 {
    		return or.rcache[symIdx]
    	}
    	return 0
    }
    
    // rcacheSet installs a new entry in the oReader's PkgNone
    // resolver cache for the specified PkgIdxNone ref or def,
    // allocating a new cache if needed.
    func (or *oReader) rcacheSet(symIdx uint32, gsym Sym) {
    	if len(or.rcache) == 0 {
    		or.rcache = make([]Sym, or.NNonpkgdef()+or.NNonpkgref())
    	}
    	or.rcache[symIdx] = gsym
    }
    
    
    // Resolve a local symbol reference. Return global index.
    
    func (l *Loader) resolve(r *oReader, s goobj2.SymRef) Sym {
    
    	var rr *oReader
    	switch p := s.PkgIdx; p {
    	case goobj2.PkgIdxInvalid:
    		if s.SymIdx != 0 {
    			panic("bad sym ref")
    		}
    		return 0
    	case goobj2.PkgIdxNone:
    
    		// Check for cached version first
    		if cached := r.rcacheGet(s.SymIdx); cached != 0 {
    			return cached
    		}
    
    		// Resolve by name
    		i := int(s.SymIdx) + r.NSym()
    		osym := goobj2.Sym{}
    		osym.Read(r.Reader, r.SymOff(i))
    		name := strings.Replace(osym.Name, "\"\".", r.pkgprefix, -1)
    		v := abiToVer(osym.ABI, r.version)
    
    		gsym := l.Lookup(name, v)
    		// Add to cache, then return.
    		r.rcacheSet(s.SymIdx, gsym)
    		return gsym
    
    		return l.builtinSyms[s.SymIdx]
    
    	case goobj2.PkgIdxSelf:
    		rr = r
    	default:
    		pkg := r.Pkg(int(p))
    
    		var ok bool
    		rr, ok = l.objByPkg[pkg]
    		if !ok {
    			log.Fatalf("reference of nonexisted package %s, from %v", pkg, r.unit.Lib)
    		}
    
    	return l.toGlobal(rr, int(s.SymIdx))
    
    // Look up a symbol by name, return global index, or 0 if not found.
    // This is more like Syms.ROLookup than Lookup -- it doesn't create
    // new symbol.
    
    func (l *Loader) Lookup(name string, ver int) Sym {
    
    	if ver >= sym.SymVerStatic || ver < 0 {
    
    		return l.extStaticSyms[nameVer{name, ver}]
    	}
    	return l.symsByName[ver][name]
    
    // Returns whether i is a dup of another symbol, and i is not
    // "primary", i.e. Lookup i by name will not return i.
    func (l *Loader) IsDup(i Sym) bool {
    	if _, ok := l.overwrite[i]; ok {
    		return true
    	}
    
    		return false
    	}
    	r, li := l.toLocal(i)
    	osym := goobj2.Sym{}
    	osym.Read(r.Reader, r.SymOff(li))
    	if !osym.Dupok() {
    		return false
    	}
    	if osym.Name == "" {
    
    		return false // Unnamed aux symbol cannot be dup.
    	}
    	if osym.ABI == goobj2.SymABIstatic {
    		return false // Static symbol cannot be dup.
    
    	}
    	name := strings.Replace(osym.Name, "\"\".", r.pkgprefix, -1)
    	ver := abiToVer(osym.ABI, r.version)
    
    	return l.symsByName[ver][name] != i
    
    // Check that duplicate symbols have same contents.
    func (l *Loader) checkdup(name string, i Sym, r *oReader, dup Sym) {
    	li := int(i - l.startIndex(r))
    	p := r.Data(li)
    	if strings.HasPrefix(name, "go.info.") {
    		p, _ = patchDWARFName1(p, r)
    	}
    	rdup, ldup := l.toLocal(dup)
    	pdup := rdup.Data(ldup)
    	if strings.HasPrefix(name, "go.info.") {
    		pdup, _ = patchDWARFName1(pdup, rdup)
    	}
    	if bytes.Equal(p, pdup) {
    		return
    	}
    	reason := "same length but different contents"
    	if len(p) != len(pdup) {
    		reason = fmt.Sprintf("new length %d != old length %d", len(p), len(pdup))
    	}
    	fmt.Fprintf(os.Stderr, "cmd/link: while reading object for '%v': duplicate symbol '%s', previous def at '%v', with mismatched payload: %s\n", r.unit.Lib, name, rdup.unit.Lib, reason)
    
    	// For the moment, whitelist DWARF subprogram DIEs for
    	// auto-generated wrapper functions. What seems to happen
    	// here is that we get different line numbers on formal
    	// params; I am guessing that the pos is being inherited
    	// from the spot where the wrapper is needed.
    	whitelist := strings.HasPrefix(name, "go.info.go.interface") ||
    		strings.HasPrefix(name, "go.info.go.builtin") ||
    		strings.HasPrefix(name, "go.debuglines")
    	if !whitelist {
    		l.strictDupMsgs++
    	}
    }
    
    func (l *Loader) NStrictDupMsgs() int { return l.strictDupMsgs }
    
    
    // Number of total symbols.
    func (l *Loader) NSym() int {
    	return int(l.max + 1)
    }
    
    
    // Number of defined Go symbols.
    func (l *Loader) NDef() int {
    	return int(l.extStart)
    }
    
    
    // Returns the raw (unpatched) name of the i-th symbol.
    func (l *Loader) RawSymName(i Sym) string {
    
    		if s := l.Syms[i]; s != nil {
    			return s.Name
    		}
    
    	osym := goobj2.Sym{}
    	osym.Read(r.Reader, r.SymOff(li))
    	return osym.Name
    }
    
    // Returns the (patched) name of the i-th symbol.
    func (l *Loader) SymName(i Sym) string {
    
    		if s := l.Syms[i]; s != nil {
    			return s.Name // external name should already be patched?
    		}
    
    	osym := goobj2.Sym{}
    	osym.Read(r.Reader, r.SymOff(li))
    	return strings.Replace(osym.Name, "\"\".", r.pkgprefix, -1)
    }
    
    // Returns the type of the i-th symbol.
    func (l *Loader) SymType(i Sym) sym.SymKind {
    
    		if s := l.Syms[i]; s != nil {
    			return s.Type
    		}
    
    		pp := l.getPayload(i)
    		if pp != nil {
    			return pp.kind
    		}
    
    	osym := goobj2.Sym{}
    	osym.Read(r.Reader, r.SymOff(li))
    	return sym.AbiSymKindToSymKind[objabi.SymKind(osym.Type)]
    }
    
    
    // Returns the attributes of the i-th symbol.
    func (l *Loader) SymAttr(i Sym) uint8 {
    
    		// TODO: do something? External symbols have different representation of attributes. For now, ReflectMethod is the only thing matters and it cannot be set by external symbol.
    
    	osym := goobj2.Sym{}
    	osym.Read(r.Reader, r.SymOff(li))
    	return osym.Flag
    }
    
    
    // AttrReachable returns true for symbols that are transitively
    // referenced from the entry points. Unreachable symbols are not
    // written to the output.
    func (l *Loader) AttrReachable(i Sym) bool {
    	return l.attrReachable.has(i)
    }
    
    // SetAttrReachable sets the reachability property for a symbol (see
    // AttrReachable).
    func (l *Loader) SetAttrReachable(i Sym, v bool) {
    	if v {
    		l.attrReachable.set(i)
    	} else {
    		l.attrReachable.unset(i)
    	}
    }
    
    // AttrOnList returns true for symbols that are on some list (such as
    // the list of all text symbols, or one of the lists of data symbols)
    // and is consulted to avoid bugs where a symbol is put on a list
    // twice.
    func (l *Loader) AttrOnList(i Sym) bool {
    	return l.attrOnList.has(i)
    }
    
    // SetAttrOnList sets the "on list" property for a symbol (see
    // AttrOnList).
    func (l *Loader) SetAttrOnList(i Sym, v bool) {
    	if v {
    		l.attrOnList.set(i)
    	} else {
    		l.attrOnList.unset(i)
    	}
    }
    
    // AttrVisibilityHidden symbols returns true for ELF symbols with
    // visibility set to STV_HIDDEN. They become local symbols in
    // the final executable. Only relevant when internally linking
    // on an ELF platform.
    func (l *Loader) AttrVisibilityHidden(i Sym) bool {
    	if i < l.extStart {
    		return false
    	}
    	return l.attrVisibilityHidden.has(i - l.extStart)
    }
    
    // SetAttrVisibilityHidden sets the "hidden visibility" property for a
    // symbol (see AttrVisibilityHidden).
    func (l *Loader) SetAttrVisibilityHidden(i Sym, v bool) {
    	if i < l.extStart {
    		panic("tried to set visibility attr on non-external symbol")
    	}
    	if v {
    		l.attrVisibilityHidden.set(i - l.extStart)
    	} else {
    		l.attrVisibilityHidden.unset(i - l.extStart)
    	}
    }
    
    // AttrDuplicateOK returns true for a symbol that can be present in
    // multiple object files.
    func (l *Loader) AttrDuplicateOK(i Sym) bool {
    	if i < l.extStart {
    		// TODO: if this path winds up being taken frequently, it
    		// might make more sense to copy the flag value out of the object
    		// into a larger bitmap during preload.
    		r, li := l.toLocal(i)
    		osym := goobj2.Sym{}
    		osym.Read(r.Reader, r.SymOff(li))
    		return osym.Dupok()
    	}
    	return l.attrDuplicateOK.has(i - l.extStart)
    }
    
    // SetAttrDuplicateOK sets the "duplicate OK" property for an external
    // symbol (see AttrDuplicateOK).
    func (l *Loader) SetAttrDuplicateOK(i Sym, v bool) {
    	if i < l.extStart {
    		panic("tried to set dupok attr on non-external symbol")
    	}
    	if v {
    		l.attrDuplicateOK.set(i - l.extStart)
    	} else {
    		l.attrDuplicateOK.unset(i - l.extStart)
    	}
    }
    
    // AttrShared returns true for symbols compiled with the -shared option.
    func (l *Loader) AttrShared(i Sym) bool {
    	if i < l.extStart {
    		// TODO: if this path winds up being taken frequently, it
    		// might make more sense to copy the flag value out of the
    		// object into a larger bitmap during preload.
    		r, _ := l.toLocal(i)
    		return (r.Flags() & goobj2.ObjFlagShared) != 0
    	}
    	return l.attrShared.has(i - l.extStart)
    }
    
    // SetAttrShared sets the "shared" property for an external
    // symbol (see AttrShared).
    func (l *Loader) SetAttrShared(i Sym, v bool) {
    	if i < l.extStart {
    		panic("tried to set shared attr on non-external symbol")
    	}
    	if v {
    		l.attrShared.set(i - l.extStart)
    	} else {
    		l.attrShared.unset(i - l.extStart)
    	}
    }
    
    // AttrExternal returns true for function symbols loaded from host
    // object files.
    func (l *Loader) AttrExternal(i Sym) bool {
    	if i < l.extStart {
    		return false
    	}
    	return l.attrExternal.has(i - l.extStart)
    }
    
    // SetAttrExternal sets the "external" property for an host object
    // symbol (see AttrExternal).
    func (l *Loader) SetAttrExternal(i Sym, v bool) {
    	if i < l.extStart {
    		panic("tried to set external attr on non-external symbol")
    	}
    	if v {
    		l.attrExternal.set(i - l.extStart)
    	} else {
    		l.attrExternal.unset(i - l.extStart)
    	}
    }
    
    // AttrTopFrame returns true for a function symbol that is an entry
    // point, meaning that unwinders should stop when they hit this
    // function.
    func (l *Loader) AttrTopFrame(i Sym) bool {
    	_, ok := l.attrTopFrame[i]
    	return ok
    }
    
    // SetAttrTopFrame sets the "top frame" property for a symbol (see
    // AttrTopFrame).
    func (l *Loader) SetAttrTopFrame(i Sym, v bool) {
    	if v {
    		l.attrTopFrame[i] = struct{}{}
    	} else {
    		delete(l.attrTopFrame, i)
    	}
    }
    
    // AttrSpecial returns true for a symbols that do not have their
    // address (i.e. Value) computed by the usual mechanism of
    // data.go:dodata() & data.go:address().
    func (l *Loader) AttrSpecial(i Sym) bool {
    	_, ok := l.attrSpecial[i]
    	return ok
    }
    
    // SetAttrSpecial sets the "special" property for a symbol (see
    // AttrSpecial).
    func (l *Loader) SetAttrSpecial(i Sym, v bool) {
    	if v {
    		l.attrSpecial[i] = struct{}{}
    	} else {
    		delete(l.attrSpecial, i)
    	}
    }
    
    // AttrCgoExportDynamic returns true for a symbol that has been
    // specially marked via the "cgo_export_dynamic" compiler directive
    // written by cgo (in response to //export directives in the source).
    func (l *Loader) AttrCgoExportDynamic(i Sym) bool {
    	_, ok := l.attrCgoExportDynamic[i]
    	return ok
    }
    
    // SetAttrCgoExportDynamic sets the "cgo_export_dynamic" for a symbol
    // (see AttrCgoExportDynamic).
    func (l *Loader) SetAttrCgoExportDynamic(i Sym, v bool) {
    	if v {
    		l.attrCgoExportDynamic[i] = struct{}{}
    	} else {
    		delete(l.attrCgoExportDynamic, i)
    	}
    }
    
    // AttrCgoExportStatic returns true for a symbol that has been
    // specially marked via the "cgo_export_static" directive
    // written by cgo.
    func (l *Loader) AttrCgoExportStatic(i Sym) bool {
    	_, ok := l.attrCgoExportStatic[i]
    	return ok
    }
    
    // SetAttrCgoExportStatic sets the "cgo_export_dynamic" for a symbol
    // (see AttrCgoExportStatic).
    func (l *Loader) SetAttrCgoExportStatic(i Sym, v bool) {
    	if v {
    		l.attrCgoExportStatic[i] = struct{}{}
    	} else {
    		delete(l.attrCgoExportStatic, i)
    	}
    }
    
    
    // Returns whether the i-th symbol has ReflectMethod attribute set.
    func (l *Loader) IsReflectMethod(i Sym) bool {
    	return l.SymAttr(i)&goobj2.SymFlagReflectMethod != 0
    }
    
    
    // Returns whether this is a Go type symbol.
    func (l *Loader) IsGoType(i Sym) bool {
    	return l.SymAttr(i)&goobj2.SymFlagGoType != 0
    }
    
    
    // Returns whether this is a "go.itablink.*" symbol.
    func (l *Loader) IsItabLink(i Sym) bool {
    	if _, ok := l.itablink[i]; ok {
    		return true
    	}
    	return false
    }
    
    
    // growValues grows the slice used to store symbol values.
    func (l *Loader) growValues(reqLen int) {
    	curLen := len(l.values)
    	if reqLen > curLen {
    		l.values = append(l.values, make([]int64, reqLen+1-curLen)...)
    	}
    }
    
    // SymValue returns the value of the i-th symbol. i is global index.
    func (l *Loader) SymValue(i Sym) int64 {
    	return l.values[i]
    }
    
    // SetSymValue sets the value of the i-th symbol. i is global index.
    func (l *Loader) SetSymValue(i Sym, val int64) {
    	l.values[i] = val
    }
    
    
    // Returns the symbol content of the i-th symbol. i is global index.
    func (l *Loader) Data(i Sym) []byte {
    
    		if s := l.Syms[i]; s != nil {
    			return s.P
    		}
    
    		pp := l.getPayload(i)
    		if pp != nil {
    			return pp.data
    		}
    
    // Returns the number of aux symbols given a global index.
    func (l *Loader) NAux(i Sym) int {
    
    	return r.NAux(li)
    }
    
    // Returns the referred symbol of the j-th aux symbol of the i-th
    // symbol.
    func (l *Loader) AuxSym(i Sym, j int) Sym {
    
    	a := goobj2.Aux{}
    	a.Read(r.Reader, r.AuxOff(li, j))
    
    // ReadAuxSyms reads the aux symbol ids for the specified symbol into the
    // slice passed as a parameter. If the slice capacity is not large enough, a new
    // larger slice will be allocated. Final slice is returned.
    func (l *Loader) ReadAuxSyms(symIdx Sym, dst []Sym) []Sym {
    
    	if l.IsExternal(symIdx) {
    
    		return dst[:0]
    	}
    	naux := l.NAux(symIdx)
    	if naux == 0 {
    		return dst[:0]
    	}
    
    	if cap(dst) < naux {
    		dst = make([]Sym, naux)
    	}
    	dst = dst[:0]
    
    	r, li := l.toLocal(symIdx)
    	for i := 0; i < naux; i++ {
    		a := goobj2.Aux{}
    		a.Read(r.Reader, r.AuxOff(li, i))
    		dst = append(dst, l.resolve(r, a.Sym))
    	}
    
    	return dst
    }
    
    
    // OuterSym gets the outer symbol for host object loaded symbols.
    func (l *Loader) OuterSym(i Sym) Sym {
    	sym := l.Syms[i]
    	if sym != nil && sym.Outer != nil {
    		outer := sym.Outer
    		return l.Lookup(outer.Name, int(outer.Version))
    	}
    	return 0
    }
    
    
    // SubSym gets the subsymbol for host object loaded symbols.
    
    func (l *Loader) SubSym(i Sym) Sym {
    	sym := l.Syms[i]
    	if sym != nil && sym.Sub != nil {
    		sub := sym.Sub
    		return l.Lookup(sub.Name, int(sub.Version))
    	}
    	return 0
    }
    
    
    // Initialize Reachable bitmap and its siblings for running deadcode pass.
    
    func (l *Loader) InitReachable() {
    
    	l.growAttrBitmaps(l.NSym() + 1)
    
    // Insure that reachable bitmap and its siblings have enough size.
    func (l *Loader) growAttrBitmaps(reqLen int) {
    	if reqLen > l.attrReachable.len() {
    		// These are indexed by global symbol
    		l.attrReachable = growBitmap(reqLen, l.attrReachable)
    		l.attrOnList = growBitmap(reqLen, l.attrReachable)
    	}
    	// These are indexed by external symbol offset (e.g. i - l.extStart)
    	if l.extStart == 0 {
    		return
    	}
    	extReqLen := reqLen - int(l.extStart)
    	if extReqLen > l.attrVisibilityHidden.len() {
    		l.attrVisibilityHidden = growBitmap(extReqLen, l.attrVisibilityHidden)
    		l.attrDuplicateOK = growBitmap(extReqLen, l.attrDuplicateOK)
    		l.attrShared = growBitmap(extReqLen, l.attrShared)
    		l.attrExternal = growBitmap(extReqLen, l.attrExternal)
    
    // At method returns the j-th reloc for a global symbol.
    func (relocs *Relocs) At(j int) Reloc {
    
    	if s := relocs.l.Syms[relocs.extIdx]; s != nil {
    		rel := s.R[j]
    
    		return Reloc{
    			Off:  rel.Off,
    			Size: rel.Siz,
    			Type: rel.Type,
    			Add:  rel.Add,
    			Sym:  relocs.l.Lookup(rel.Sym.Name, int(rel.Sym.Version)),
    		}
    	}
    
    	if relocs.extIdx != 0 {
    		pp := relocs.l.getPayload(relocs.extIdx)
    		return pp.relocs[j]
    	}
    
    	rel := goobj2.Reloc{}
    	rel.Read(relocs.r.Reader, relocs.r.RelocOff(relocs.li, j))
    
    	target := relocs.l.resolve(relocs.r, rel.Sym)
    
    	return Reloc{
    		Off:  rel.Off,
    		Size: rel.Siz,
    		Type: objabi.RelocType(rel.Type),
    		Add:  rel.Add,
    		Sym:  target,
    	}
    }
    
    
    // ReadAll method reads all relocations for a symbol into the
    // specified slice. If the slice capacity is not large enough, a new
    // larger slice will be allocated. Final slice is returned.
    func (relocs *Relocs) ReadAll(dst []Reloc) []Reloc {
    	if relocs.Count == 0 {