liyujie
2025-08-28 d9927380ed7c8366f762049be9f3fee225860833
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
// 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
 
import (
   "sort"
   "strconv"
)
 
// DWARF debug info is split into a sequence of compilation units.
// Each unit has its own abbreviation table and address size.
 
type unit struct {
   base   Offset // byte offset of header within the aggregate info
   off    Offset // byte offset of data within the aggregate info
   data   []byte
   atable abbrevTable
   asize  int
   vers   int
   is64   bool // True for 64-bit DWARF format
}
 
// Implement the dataFormat interface.
 
func (u *unit) version() int {
   return u.vers
}
 
func (u *unit) dwarf64() (bool, bool) {
   return u.is64, true
}
 
func (u *unit) addrsize() int {
   return u.asize
}
 
func (d *Data) parseUnits() ([]unit, error) {
   // Count units.
   nunit := 0
   b := makeBuf(d, unknownFormat{}, "info", 0, d.info)
   for len(b.data) > 0 {
       len, _ := b.unitLength()
       if len != Offset(uint32(len)) {
           b.error("unit length overflow")
           break
       }
       b.skip(int(len))
       nunit++
   }
   if b.err != nil {
       return nil, b.err
   }
 
   // Again, this time writing them down.
   b = makeBuf(d, unknownFormat{}, "info", 0, d.info)
   units := make([]unit, nunit)
   for i := range units {
       u := &units[i]
       u.base = b.off
       var n Offset
       n, u.is64 = b.unitLength()
       dataOff := b.off
       vers := b.uint16()
       if vers != 2 && vers != 3 && vers != 4 {
           b.error("unsupported DWARF version " + strconv.Itoa(int(vers)))
           break
       }
       u.vers = int(vers)
       var abbrevOff uint64
       if u.is64 {
           abbrevOff = b.uint64()
       } else {
           abbrevOff = uint64(b.uint32())
       }
       atable, err := d.parseAbbrev(abbrevOff, u.vers)
       if err != nil {
           if b.err == nil {
               b.err = err
           }
           break
       }
       u.atable = atable
       u.asize = int(b.uint8())
       u.off = b.off
       u.data = b.bytes(int(n - (b.off - dataOff)))
   }
   if b.err != nil {
       return nil, b.err
   }
   return units, nil
}
 
// offsetToUnit returns the index of the unit containing offset off.
// It returns -1 if no unit contains this offset.
func (d *Data) offsetToUnit(off Offset) int {
   // Find the unit after off
   next := sort.Search(len(d.unit), func(i int) bool {
       return d.unit[i].off > off
   })
   if next == 0 {
       return -1
   }
   u := &d.unit[next-1]
   if u.off <= off && off < u.off+Offset(len(u.data)) {
       return next - 1
   }
   return -1
}