Commit e511f153 authored by Josh Bleecher Snyder's avatar Josh Bleecher Snyder

debug/gosym: intern LineTable strings

This cuts the allocated space while executing

go tool objdump -S `go tool -n compile`

by over 10%.

It also speeds it up slightly:

name              old time/op       new time/op       delta
ObjdumpSCompiler        9.03s ± 1%        8.88s ± 1%  -1.59%  (p=0.000 n=20+20)

Updates #24725

Change-Id: Ic6ef8e273ede589334ab6e07099ac2e5bdf990c9
Reviewed-on: https://go-review.googlesource.com/106798
Run-TryBot: Josh Bleecher Snyder <josharian@gmail.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
parent c0547476
...@@ -9,6 +9,7 @@ ...@@ -9,6 +9,7 @@
package gosym package gosym
import ( import (
"bytes"
"encoding/binary" "encoding/binary"
"sync" "sync"
) )
...@@ -42,6 +43,7 @@ type LineTable struct { ...@@ -42,6 +43,7 @@ type LineTable struct {
filetab []byte filetab []byte
nfiletab uint32 nfiletab uint32
fileMap map[string]uint32 fileMap map[string]uint32
strings map[uint32]string // interned substrings of Data, keyed by offset
} }
// NOTE(rsc): This is wrong for GOARCH=arm, which uses a quantum of 4, // NOTE(rsc): This is wrong for GOARCH=arm, which uses a quantum of 4,
...@@ -120,7 +122,7 @@ func (t *LineTable) LineToPC(line int, maxpc uint64) uint64 { ...@@ -120,7 +122,7 @@ func (t *LineTable) LineToPC(line int, maxpc uint64) uint64 {
// Text must be the start address of the // Text must be the start address of the
// corresponding text segment. // corresponding text segment.
func NewLineTable(data []byte, text uint64) *LineTable { func NewLineTable(data []byte, text uint64) *LineTable {
return &LineTable{Data: data, PC: text, Line: 0} return &LineTable{Data: data, PC: text, Line: 0, strings: make(map[uint32]string)}
} }
// Go 1.2 symbol table format. // Go 1.2 symbol table format.
...@@ -266,11 +268,13 @@ func (t *LineTable) readvarint(pp *[]byte) uint32 { ...@@ -266,11 +268,13 @@ func (t *LineTable) readvarint(pp *[]byte) uint32 {
// string returns a Go string found at off. // string returns a Go string found at off.
func (t *LineTable) string(off uint32) string { func (t *LineTable) string(off uint32) string {
for i := off; ; i++ { if s, ok := t.strings[off]; ok {
if t.Data[i] == 0 { return s
return string(t.Data[off:i])
}
} }
i := bytes.IndexByte(t.Data[off:], 0)
s := string(t.Data[off : off+uint32(i)])
t.strings[off] = s
return s
} }
// step advances to the next pc, value pair in the encoded table. // step advances to the next pc, value pair in the encoded table.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment