// Copyright 2022 Google Inc. All rights reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package elf import ( "debug/elf" "encoding/binary" "encoding/hex" "errors" "fmt" "io" "os" ) const gnuBuildID = "GNU\x00" // Identifier extracts the elf build ID from an elf file. If allowMissing is true it returns // an empty identifier if the file exists but the build ID note does not. func Identifier(filename string, allowMissing bool) (string, error) { f, err := os.Open(filename) if err != nil { return "", fmt.Errorf("failed to open %s: %w", filename, err) } defer f.Close() return elfIdentifierFromReaderAt(f, filename, allowMissing) } // elfIdentifierFromReaderAt extracts the elf build ID from a ReaderAt. If allowMissing is true it // returns an empty identifier if the file exists but the build ID note does not. func elfIdentifierFromReaderAt(r io.ReaderAt, filename string, allowMissing bool) (string, error) { f, err := elf.NewFile(r) if err != nil { if allowMissing { if errors.Is(err, io.EOF) || errors.Is(err, io.ErrUnexpectedEOF) { return "", nil } if _, ok := err.(*elf.FormatError); ok { // The file was not an elf file. return "", nil } } return "", fmt.Errorf("failed to parse elf file %s: %w", filename, err) } defer f.Close() buildIDNote := f.Section(".note.gnu.build-id") if buildIDNote == nil { if allowMissing { return "", nil } return "", fmt.Errorf("failed to find .note.gnu.build-id in %s", filename) } buildIDs, err := readNote(buildIDNote.Open(), f.ByteOrder) if err != nil { return "", fmt.Errorf("failed to read .note.gnu.build-id: %w", err) } for name, desc := range buildIDs { if name == gnuBuildID { return hex.EncodeToString(desc), nil } } return "", nil } // readNote reads the contents of a note section, returning it as a map from name to descriptor. func readNote(note io.Reader, byteOrder binary.ByteOrder) (map[string][]byte, error) { var noteHeader struct { Namesz uint32 Descsz uint32 Type uint32 } notes := make(map[string][]byte) for { err := binary.Read(note, byteOrder, ¬eHeader) if err != nil { if err == io.EOF { return notes, nil } return nil, fmt.Errorf("failed to read note header: %w", err) } nameBuf := make([]byte, align4(noteHeader.Namesz)) err = binary.Read(note, byteOrder, &nameBuf) if err != nil { return nil, fmt.Errorf("failed to read note name: %w", err) } name := string(nameBuf[:noteHeader.Namesz]) descBuf := make([]byte, align4(noteHeader.Descsz)) err = binary.Read(note, byteOrder, &descBuf) if err != nil { return nil, fmt.Errorf("failed to read note desc: %w", err) } notes[name] = descBuf[:noteHeader.Descsz] } } // align4 rounds the input up to the next multiple of 4. func align4(i uint32) uint32 { return (i + 3) &^ 3 }