123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233 |
- // 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.
- package goobj
- import (
- "bytes"
- "github.com/twitchyliquid64/golang-asm/objabi"
- "encoding/binary"
- )
- // CUFileIndex is used to index the filenames that are stored in the
- // per-package/per-CU FileList.
- type CUFileIndex uint32
- // FuncInfo is serialized as a symbol (aux symbol). The symbol data is
- // the binary encoding of the struct below.
- //
- // TODO: make each pcdata a separate symbol?
- type FuncInfo struct {
- Args uint32
- Locals uint32
- FuncID objabi.FuncID
- Pcsp uint32
- Pcfile uint32
- Pcline uint32
- Pcinline uint32
- Pcdata []uint32
- PcdataEnd uint32
- Funcdataoff []uint32
- File []CUFileIndex
- InlTree []InlTreeNode
- }
- func (a *FuncInfo) Write(w *bytes.Buffer) {
- var b [4]byte
- writeUint32 := func(x uint32) {
- binary.LittleEndian.PutUint32(b[:], x)
- w.Write(b[:])
- }
- writeUint32(a.Args)
- writeUint32(a.Locals)
- writeUint32(uint32(a.FuncID))
- writeUint32(a.Pcsp)
- writeUint32(a.Pcfile)
- writeUint32(a.Pcline)
- writeUint32(a.Pcinline)
- writeUint32(uint32(len(a.Pcdata)))
- for _, x := range a.Pcdata {
- writeUint32(x)
- }
- writeUint32(a.PcdataEnd)
- writeUint32(uint32(len(a.Funcdataoff)))
- for _, x := range a.Funcdataoff {
- writeUint32(x)
- }
- writeUint32(uint32(len(a.File)))
- for _, f := range a.File {
- writeUint32(uint32(f))
- }
- writeUint32(uint32(len(a.InlTree)))
- for i := range a.InlTree {
- a.InlTree[i].Write(w)
- }
- }
- func (a *FuncInfo) Read(b []byte) {
- readUint32 := func() uint32 {
- x := binary.LittleEndian.Uint32(b)
- b = b[4:]
- return x
- }
- a.Args = readUint32()
- a.Locals = readUint32()
- a.FuncID = objabi.FuncID(readUint32())
- a.Pcsp = readUint32()
- a.Pcfile = readUint32()
- a.Pcline = readUint32()
- a.Pcinline = readUint32()
- pcdatalen := readUint32()
- a.Pcdata = make([]uint32, pcdatalen)
- for i := range a.Pcdata {
- a.Pcdata[i] = readUint32()
- }
- a.PcdataEnd = readUint32()
- funcdataofflen := readUint32()
- a.Funcdataoff = make([]uint32, funcdataofflen)
- for i := range a.Funcdataoff {
- a.Funcdataoff[i] = readUint32()
- }
- filelen := readUint32()
- a.File = make([]CUFileIndex, filelen)
- for i := range a.File {
- a.File[i] = CUFileIndex(readUint32())
- }
- inltreelen := readUint32()
- a.InlTree = make([]InlTreeNode, inltreelen)
- for i := range a.InlTree {
- b = a.InlTree[i].Read(b)
- }
- }
- // FuncInfoLengths is a cache containing a roadmap of offsets and
- // lengths for things within a serialized FuncInfo. Each length field
- // stores the number of items (e.g. files, inltree nodes, etc), and the
- // corresponding "off" field stores the byte offset of the start of
- // the items in question.
- type FuncInfoLengths struct {
- NumPcdata uint32
- PcdataOff uint32
- NumFuncdataoff uint32
- FuncdataoffOff uint32
- NumFile uint32
- FileOff uint32
- NumInlTree uint32
- InlTreeOff uint32
- Initialized bool
- }
- func (*FuncInfo) ReadFuncInfoLengths(b []byte) FuncInfoLengths {
- var result FuncInfoLengths
- const numpcdataOff = 28
- result.NumPcdata = binary.LittleEndian.Uint32(b[numpcdataOff:])
- result.PcdataOff = numpcdataOff + 4
- numfuncdataoffOff := result.PcdataOff + 4*(result.NumPcdata+1)
- result.NumFuncdataoff = binary.LittleEndian.Uint32(b[numfuncdataoffOff:])
- result.FuncdataoffOff = numfuncdataoffOff + 4
- numfileOff := result.FuncdataoffOff + 4*result.NumFuncdataoff
- result.NumFile = binary.LittleEndian.Uint32(b[numfileOff:])
- result.FileOff = numfileOff + 4
- numinltreeOff := result.FileOff + 4*result.NumFile
- result.NumInlTree = binary.LittleEndian.Uint32(b[numinltreeOff:])
- result.InlTreeOff = numinltreeOff + 4
- result.Initialized = true
- return result
- }
- func (*FuncInfo) ReadArgs(b []byte) uint32 { return binary.LittleEndian.Uint32(b) }
- func (*FuncInfo) ReadLocals(b []byte) uint32 { return binary.LittleEndian.Uint32(b[4:]) }
- func (*FuncInfo) ReadFuncID(b []byte) uint32 { return binary.LittleEndian.Uint32(b[8:]) }
- // return start and end offsets.
- func (*FuncInfo) ReadPcsp(b []byte) (uint32, uint32) {
- return binary.LittleEndian.Uint32(b[12:]), binary.LittleEndian.Uint32(b[16:])
- }
- // return start and end offsets.
- func (*FuncInfo) ReadPcfile(b []byte) (uint32, uint32) {
- return binary.LittleEndian.Uint32(b[16:]), binary.LittleEndian.Uint32(b[20:])
- }
- // return start and end offsets.
- func (*FuncInfo) ReadPcline(b []byte) (uint32, uint32) {
- return binary.LittleEndian.Uint32(b[20:]), binary.LittleEndian.Uint32(b[24:])
- }
- // return start and end offsets.
- func (*FuncInfo) ReadPcinline(b []byte, pcdataoffset uint32) (uint32, uint32) {
- return binary.LittleEndian.Uint32(b[24:]), binary.LittleEndian.Uint32(b[pcdataoffset:])
- }
- // return start and end offsets.
- func (*FuncInfo) ReadPcdata(b []byte, pcdataoffset uint32, k uint32) (uint32, uint32) {
- return binary.LittleEndian.Uint32(b[pcdataoffset+4*k:]), binary.LittleEndian.Uint32(b[pcdataoffset+4+4*k:])
- }
- func (*FuncInfo) ReadFuncdataoff(b []byte, funcdataofffoff uint32, k uint32) int64 {
- return int64(binary.LittleEndian.Uint32(b[funcdataofffoff+4*k:]))
- }
- func (*FuncInfo) ReadFile(b []byte, filesoff uint32, k uint32) CUFileIndex {
- return CUFileIndex(binary.LittleEndian.Uint32(b[filesoff+4*k:]))
- }
- func (*FuncInfo) ReadInlTree(b []byte, inltreeoff uint32, k uint32) InlTreeNode {
- const inlTreeNodeSize = 4 * 6
- var result InlTreeNode
- result.Read(b[inltreeoff+k*inlTreeNodeSize:])
- return result
- }
- // InlTreeNode is the serialized form of FileInfo.InlTree.
- type InlTreeNode struct {
- Parent int32
- File CUFileIndex
- Line int32
- Func SymRef
- ParentPC int32
- }
- func (inl *InlTreeNode) Write(w *bytes.Buffer) {
- var b [4]byte
- writeUint32 := func(x uint32) {
- binary.LittleEndian.PutUint32(b[:], x)
- w.Write(b[:])
- }
- writeUint32(uint32(inl.Parent))
- writeUint32(uint32(inl.File))
- writeUint32(uint32(inl.Line))
- writeUint32(inl.Func.PkgIdx)
- writeUint32(inl.Func.SymIdx)
- writeUint32(uint32(inl.ParentPC))
- }
- // Read an InlTreeNode from b, return the remaining bytes.
- func (inl *InlTreeNode) Read(b []byte) []byte {
- readUint32 := func() uint32 {
- x := binary.LittleEndian.Uint32(b)
- b = b[4:]
- return x
- }
- inl.Parent = int32(readUint32())
- inl.File = CUFileIndex(readUint32())
- inl.Line = int32(readUint32())
- inl.Func = SymRef{readUint32(), readUint32()}
- inl.ParentPC = int32(readUint32())
- return b
- }
|