• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1// Copyright 2012 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
5package dwarf
6
7import (
8	"fmt"
9	"strconv"
10)
11
12// Parse the type units stored in a DWARF4 .debug_types section. Each
13// type unit defines a single primary type and an 8-byte signature.
14// Other sections may then use formRefSig8 to refer to the type.
15
16// The typeUnit format is a single type with a signature. It holds
17// the same data as a compilation unit.
18type typeUnit struct {
19	unit
20	toff  Offset // Offset to signature type within data.
21	name  string // Name of .debug_type section.
22	cache Type   // Cache the type, nil to start.
23}
24
25// Parse a .debug_types section.
26func (d *Data) parseTypes(name string, types []byte) error {
27	b := makeBuf(d, unknownFormat{}, name, 0, types)
28	for len(b.data) > 0 {
29		base := b.off
30		n, dwarf64 := b.unitLength()
31		if n != Offset(uint32(n)) {
32			b.error("type unit length overflow")
33			return b.err
34		}
35		hdroff := b.off
36		vers := int(b.uint16())
37		if vers != 4 {
38			b.error("unsupported DWARF version " + strconv.Itoa(vers))
39			return b.err
40		}
41		var ao uint64
42		if !dwarf64 {
43			ao = uint64(b.uint32())
44		} else {
45			ao = b.uint64()
46		}
47		atable, err := d.parseAbbrev(ao, vers)
48		if err != nil {
49			return err
50		}
51		asize := b.uint8()
52		sig := b.uint64()
53
54		var toff uint32
55		if !dwarf64 {
56			toff = b.uint32()
57		} else {
58			to64 := b.uint64()
59			if to64 != uint64(uint32(to64)) {
60				b.error("type unit type offset overflow")
61				return b.err
62			}
63			toff = uint32(to64)
64		}
65
66		boff := b.off
67		d.typeSigs[sig] = &typeUnit{
68			unit: unit{
69				base:   base,
70				off:    boff,
71				data:   b.bytes(int(n - (b.off - hdroff))),
72				atable: atable,
73				asize:  int(asize),
74				vers:   vers,
75				is64:   dwarf64,
76			},
77			toff: Offset(toff),
78			name: name,
79		}
80		if b.err != nil {
81			return b.err
82		}
83	}
84	return nil
85}
86
87// Return the type for a type signature.
88func (d *Data) sigToType(sig uint64) (Type, error) {
89	tu := d.typeSigs[sig]
90	if tu == nil {
91		return nil, fmt.Errorf("no type unit with signature %v", sig)
92	}
93	if tu.cache != nil {
94		return tu.cache, nil
95	}
96
97	b := makeBuf(d, tu, tu.name, tu.off, tu.data)
98	r := &typeUnitReader{d: d, tu: tu, b: b}
99	t, err := d.readType(tu.name, r, tu.toff, make(map[Offset]Type), nil)
100	if err != nil {
101		return nil, err
102	}
103
104	tu.cache = t
105	return t, nil
106}
107
108// typeUnitReader is a typeReader for a tagTypeUnit.
109type typeUnitReader struct {
110	d   *Data
111	tu  *typeUnit
112	b   buf
113	err error
114}
115
116// Seek to a new position in the type unit.
117func (tur *typeUnitReader) Seek(off Offset) {
118	tur.err = nil
119	doff := off - tur.tu.off
120	if doff < 0 || doff >= Offset(len(tur.tu.data)) {
121		tur.err = fmt.Errorf("%s: offset %d out of range; max %d", tur.tu.name, doff, len(tur.tu.data))
122		return
123	}
124	tur.b = makeBuf(tur.d, tur.tu, tur.tu.name, off, tur.tu.data[doff:])
125}
126
127// AddressSize returns the size in bytes of addresses in the current type unit.
128func (tur *typeUnitReader) AddressSize() int {
129	return tur.tu.unit.asize
130}
131
132// Next reads the next [Entry] from the type unit.
133func (tur *typeUnitReader) Next() (*Entry, error) {
134	if tur.err != nil {
135		return nil, tur.err
136	}
137	if len(tur.tu.data) == 0 {
138		return nil, nil
139	}
140	e := tur.b.entry(nil, tur.tu.atable, tur.tu.base, tur.tu.vers)
141	if tur.b.err != nil {
142		tur.err = tur.b.err
143		return nil, tur.err
144	}
145	return e, nil
146}
147
148// clone returns a new reader for the type unit.
149func (tur *typeUnitReader) clone() typeReader {
150	return &typeUnitReader{
151		d:  tur.d,
152		tu: tur.tu,
153		b:  makeBuf(tur.d, tur.tu, tur.tu.name, tur.tu.off, tur.tu.data),
154	}
155}
156
157// offset returns the current offset.
158func (tur *typeUnitReader) offset() Offset {
159	return tur.b.off
160}
161