| // Copyright 2009 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 dwarf provides access to DWARF debugging information loaded from | |
| executable files, as defined in the DWARF 2.0 Standard at | |
| http://dwarfstd.org/doc/dwarf-2.0.0.pdf. | |
| # Security | |
| This package is not designed to be hardened against adversarial inputs, and is | |
| outside the scope of https://go.dev/security/policy. In particular, only basic | |
| validation is done when parsing object files. As such, care should be taken when | |
| parsing untrusted inputs, as parsing malformed files may consume significant | |
| resources, or cause panics. | |
| */ | |
| package dwarf | |
| import ( | |
| "encoding/binary" | |
| "errors" | |
| ) | |
| // Data represents the DWARF debugging information | |
| // loaded from an executable file (for example, an ELF or Mach-O executable). | |
| type Data struct { | |
| // raw data | |
| abbrev []byte | |
| aranges []byte | |
| frame []byte | |
| info []byte | |
| line []byte | |
| pubnames []byte | |
| ranges []byte | |
| str []byte | |
| // New sections added in DWARF 5. | |
| addr []byte | |
| lineStr []byte | |
| strOffsets []byte | |
| rngLists []byte | |
| // parsed data | |
| abbrevCache map[uint64]abbrevTable | |
| bigEndian bool | |
| order binary.ByteOrder | |
| typeCache map[Offset]Type | |
| typeSigs map[uint64]*typeUnit | |
| unit []unit | |
| } | |
| var errSegmentSelector = errors.New("non-zero segment_selector size not supported") | |
| // New returns a new [Data] object initialized from the given parameters. | |
| // Rather than calling this function directly, clients should typically use | |
| // the DWARF method of the File type of the appropriate package [debug/elf], | |
| // [debug/macho], or [debug/pe]. | |
| // | |
| // The []byte arguments are the data from the corresponding debug section | |
| // in the object file; for example, for an ELF object, abbrev is the contents of | |
| // the ".debug_abbrev" section. | |
| func New(abbrev, aranges, frame, info, line, pubnames, ranges, str []byte) (*Data, error) { | |
| d := &Data{ | |
| abbrev: abbrev, | |
| aranges: aranges, | |
| frame: frame, | |
| info: info, | |
| line: line, | |
| pubnames: pubnames, | |
| ranges: ranges, | |
| str: str, | |
| abbrevCache: make(map[uint64]abbrevTable), | |
| typeCache: make(map[Offset]Type), | |
| typeSigs: make(map[uint64]*typeUnit), | |
| } | |
| // Sniff .debug_info to figure out byte order. | |
| // 32-bit DWARF: 4 byte length, 2 byte version. | |
| // 64-bit DWARf: 4 bytes of 0xff, 8 byte length, 2 byte version. | |
| if len(d.info) < 6 { | |
| return nil, DecodeError{"info", Offset(len(d.info)), "too short"} | |
| } | |
| offset := 4 | |
| if d.info[0] == 0xff && d.info[1] == 0xff && d.info[2] == 0xff && d.info[3] == 0xff { | |
| if len(d.info) < 14 { | |
| return nil, DecodeError{"info", Offset(len(d.info)), "too short"} | |
| } | |
| offset = 12 | |
| } | |
| // Fetch the version, a tiny 16-bit number (1, 2, 3, 4, 5). | |
| x, y := d.info[offset], d.info[offset+1] | |
| switch { | |
| case x == 0 && y == 0: | |
| return nil, DecodeError{"info", 4, "unsupported version 0"} | |
| case x == 0: | |
| d.bigEndian = true | |
| d.order = binary.BigEndian | |
| case y == 0: | |
| d.bigEndian = false | |
| d.order = binary.LittleEndian | |
| default: | |
| return nil, DecodeError{"info", 4, "cannot determine byte order"} | |
| } | |
| u, err := d.parseUnits() | |
| if err != nil { | |
| return nil, err | |
| } | |
| d.unit = u | |
| return d, nil | |
| } | |
| // AddTypes will add one .debug_types section to the DWARF data. A | |
| // typical object with DWARF version 4 debug info will have multiple | |
| // .debug_types sections. The name is used for error reporting only, | |
| // and serves to distinguish one .debug_types section from another. | |
| func (d *Data) AddTypes(name string, types []byte) error { | |
| return d.parseTypes(name, types) | |
| } | |
| // AddSection adds another DWARF section by name. The name should be a | |
| // DWARF section name such as ".debug_addr", ".debug_str_offsets", and | |
| // so forth. This approach is used for new DWARF sections added in | |
| // DWARF 5 and later. | |
| func (d *Data) AddSection(name string, contents []byte) error { | |
| var err error | |
| switch name { | |
| case ".debug_addr": | |
| d.addr = contents | |
| case ".debug_line_str": | |
| d.lineStr = contents | |
| case ".debug_str_offsets": | |
| d.strOffsets = contents | |
| case ".debug_rnglists": | |
| d.rngLists = contents | |
| } | |
| // Just ignore names that we don't yet support. | |
| return err | |
| } | |