// Copyright 2018 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. // Usage: // // checkdwarf <exe> <suffix> // // Opens <exe>, which must be an executable or a library and checks that // there is an entry in .debug_info whose name ends in <suffix> package main import ( "debug/dwarf" "debug/elf" "debug/macho" "debug/pe" "fmt" "os" "strings" ) func usage() { fmt.Fprintf(os.Stderr, "checkdwarf executable-or-library DIE-suffix\n") } type dwarfer interface { DWARF() (*dwarf.Data, error) } func openElf(path string) dwarfer { exe, err := elf.Open(path) if err != nil { return nil } return exe } func openMacho(path string) dwarfer { exe, err := macho.Open(path) if err != nil { return nil } return exe } func openPE(path string) dwarfer { exe, err := pe.Open(path) if err != nil { return nil } return exe } func main() { if len(os.Args) != 3 { usage() } exePath := os.Args[1] dieSuffix := os.Args[2] var exe dwarfer for _, openfn := range []func(string) dwarfer{openMacho, openPE, openElf} { exe = openfn(exePath) if exe != nil { break } } if exe == nil { fmt.Fprintf(os.Stderr, "could not open %s\n", exePath) os.Exit(1) } data, err := exe.DWARF() if err != nil { fmt.Fprintf(os.Stderr, "%s: error opening DWARF: %v\n", exePath, err) os.Exit(1) } rdr := data.Reader() for { e, err := rdr.Next() if err != nil { fmt.Fprintf(os.Stderr, "%s: error reading DWARF: %v\n", exePath, err) os.Exit(1) } if e == nil { break } name, hasname := e.Val(dwarf.AttrName).(string) if !hasname { continue } if strings.HasSuffix(name, dieSuffix) { // found os.Exit(0) } } fmt.Fprintf(os.Stderr, "%s: no entry with a name ending in %q was found\n", exePath, dieSuffix) os.Exit(1) }