2020-01-21 20:41:24 +00:00
|
|
|
package proc
|
|
|
|
|
|
|
|
type goroutineCache struct {
|
|
|
|
partialGCache map[int]*G
|
|
|
|
allGCache []*G
|
|
|
|
|
|
|
|
allgentryAddr, allglenAddr uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gcache *goroutineCache) init(bi *BinaryInfo) {
|
|
|
|
var err error
|
|
|
|
|
|
|
|
exeimage := bi.Images[0]
|
|
|
|
rdr := exeimage.DwarfReader()
|
|
|
|
|
|
|
|
gcache.allglenAddr, _ = rdr.AddrFor("runtime.allglen", exeimage.StaticBase)
|
|
|
|
|
|
|
|
rdr.Seek(0)
|
|
|
|
gcache.allgentryAddr, err = rdr.AddrFor("runtime.allgs", exeimage.StaticBase)
|
|
|
|
if err != nil {
|
|
|
|
// try old name (pre Go 1.6)
|
|
|
|
gcache.allgentryAddr, _ = rdr.AddrFor("runtime.allg", exeimage.StaticBase)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gcache *goroutineCache) getRuntimeAllg(bi *BinaryInfo, mem MemoryReadWriter) (uint64, uint64, error) {
|
|
|
|
if gcache.allglenAddr == 0 || gcache.allgentryAddr == 0 {
|
|
|
|
return 0, 0, ErrNoRuntimeAllG
|
|
|
|
}
|
2020-02-11 01:32:50 +00:00
|
|
|
allglen, err := readUintRaw(mem, uintptr(gcache.allglenAddr), int64(bi.Arch.PtrSize()))
|
2020-01-21 20:41:24 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, 0, err
|
|
|
|
}
|
2020-02-11 01:32:50 +00:00
|
|
|
allgptr, err := readUintRaw(mem, uintptr(gcache.allgentryAddr), int64(bi.Arch.PtrSize()))
|
2020-01-21 20:41:24 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, 0, err
|
|
|
|
}
|
|
|
|
return allgptr, allglen, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gcache *goroutineCache) addGoroutine(g *G) {
|
|
|
|
if gcache.partialGCache == nil {
|
|
|
|
gcache.partialGCache = make(map[int]*G)
|
|
|
|
}
|
|
|
|
gcache.partialGCache[g.ID] = g
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clear clears the cached contents of the cache for runtime.allgs.
|
|
|
|
func (gcache *goroutineCache) Clear() {
|
|
|
|
gcache.partialGCache = nil
|
|
|
|
gcache.allGCache = nil
|
|
|
|
}
|