lin
2025-06-05 ed3dd9d3e7519a82bb871d5eedb24a2fa0c91f47
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
// Copyright 2017 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 pprof
 
import (
   "encoding/binary"
   "errors"
   "fmt"
   "os"
)
 
var (
   errBadELF    = errors.New("malformed ELF binary")
   errNoBuildID = errors.New("no NT_GNU_BUILD_ID found in ELF binary")
)
 
// elfBuildID returns the GNU build ID of the named ELF binary,
// without introducing a dependency on debug/elf and its dependencies.
func elfBuildID(file string) (string, error) {
   buf := make([]byte, 256)
   f, err := os.Open(file)
   if err != nil {
       return "", err
   }
   defer f.Close()
 
   if _, err := f.ReadAt(buf[:64], 0); err != nil {
       return "", err
   }
 
   // ELF file begins with \x7F E L F.
   if buf[0] != 0x7F || buf[1] != 'E' || buf[2] != 'L' || buf[3] != 'F' {
       return "", errBadELF
   }
 
   var byteOrder binary.ByteOrder
   switch buf[5] {
   default:
       return "", errBadELF
   case 1: // little-endian
       byteOrder = binary.LittleEndian
   case 2: // big-endian
       byteOrder = binary.BigEndian
   }
 
   var shnum int
   var shoff, shentsize int64
   switch buf[4] {
   default:
       return "", errBadELF
   case 1: // 32-bit file header
       shoff = int64(byteOrder.Uint32(buf[32:]))
       shentsize = int64(byteOrder.Uint16(buf[46:]))
       if shentsize != 40 {
           return "", errBadELF
       }
       shnum = int(byteOrder.Uint16(buf[48:]))
   case 2: // 64-bit file header
       shoff = int64(byteOrder.Uint64(buf[40:]))
       shentsize = int64(byteOrder.Uint16(buf[58:]))
       if shentsize != 64 {
           return "", errBadELF
       }
       shnum = int(byteOrder.Uint16(buf[60:]))
   }
 
   for i := 0; i < shnum; i++ {
       if _, err := f.ReadAt(buf[:shentsize], shoff+int64(i)*shentsize); err != nil {
           return "", err
       }
       if typ := byteOrder.Uint32(buf[4:]); typ != 7 { // SHT_NOTE
           continue
       }
       var off, size int64
       if shentsize == 40 {
           // 32-bit section header
           off = int64(byteOrder.Uint32(buf[16:]))
           size = int64(byteOrder.Uint32(buf[20:]))
       } else {
           // 64-bit section header
           off = int64(byteOrder.Uint64(buf[24:]))
           size = int64(byteOrder.Uint64(buf[32:]))
       }
       size += off
       for off < size {
           if _, err := f.ReadAt(buf[:16], off); err != nil { // room for header + name GNU\x00
               return "", err
           }
           nameSize := int(byteOrder.Uint32(buf[0:]))
           descSize := int(byteOrder.Uint32(buf[4:]))
           noteType := int(byteOrder.Uint32(buf[8:]))
           descOff := off + int64(12+(nameSize+3)&^3)
           off = descOff + int64((descSize+3)&^3)
           if nameSize != 4 || noteType != 3 || buf[12] != 'G' || buf[13] != 'N' || buf[14] != 'U' || buf[15] != '\x00' { // want name GNU\x00 type 3 (NT_GNU_BUILD_ID)
               continue
           }
           if descSize > len(buf) {
               return "", errBadELF
           }
           if _, err := f.ReadAt(buf[:descSize], descOff); err != nil {
               return "", err
           }
           return fmt.Sprintf("%x", buf[:descSize]), nil
       }
   }
   return "", errNoBuildID
}