pkg/tools/elf/dwarf.go (97 lines of code) (raw):
// Licensed to Apache Software Foundation (ASF) under one or more contributor
// license agreements. See the NOTICE file distributed with
// this work for additional information regarding copyright
// ownership. Apache Software Foundation (ASF) licenses this file to you 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/dwarf"
"encoding/binary"
"fmt"
)
var ReaderLanguageGolang = 22
type DwarfReader struct {
elfByteOrder binary.ByteOrder
producer string
language int
functions map[string]*FunctionInfo
structures map[string]*StructureInfo
classes map[string]*ClassInfo
}
func (f *File) NewDwarfReader(attrNames ...string) (*DwarfReader, error) {
data, err := f.realFile.DWARF()
if err != nil {
return nil, err
}
reader := &DwarfReader{}
if err := reader.init(data, attrNames); err != nil {
return nil, err
}
return reader, nil
}
func (r *DwarfReader) GetFunction(name string) *FunctionInfo {
return r.functions[name]
}
func (r *DwarfReader) GetStructure(name string) *StructureInfo {
return r.structures[name]
}
func (r *DwarfReader) GetStructMemberOffset(structName, memberName string) (uint64, error) {
structure := r.GetStructure(structName)
if structure == nil {
return 0, fmt.Errorf("the struct not found: %s", structName)
}
field := structure.GetField(memberName)
if field == nil {
return 0, fmt.Errorf("the field not found, struct name: %s, member name: %s", structName, memberName)
}
return uint64(field.Offset), nil
}
func (r *DwarfReader) GetClassParentOffset(className, parentClassName string) (uint64, error) {
class := r.GetClass(className)
if class == nil {
return 0, fmt.Errorf("the class not found: %s", className)
}
offset := class.GetInheritanceOffset(parentClassName)
if offset >= 0 {
return uint64(offset), nil
}
return 0, fmt.Errorf("the parent class not found, class: %s, parent: %s", className, parentClassName)
}
func (r *DwarfReader) GetClassMemberOffset(className, memberName string) (uint64, error) {
class := r.GetClass(className)
if class == nil {
return 0, fmt.Errorf("the class not found: %s", className)
}
offset := class.GetMemberOffset(memberName)
if offset >= 0 {
return uint64(offset), nil
}
return 0, fmt.Errorf("the member not found, class: %s, member: %s", className, memberName)
}
func (r *DwarfReader) GetClass(name string) *ClassInfo {
return r.classes[name]
}
func (r *DwarfReader) init(data *dwarf.Data, readAttrNames []string) error {
r.functions = make(map[string]*FunctionInfo)
r.structures = make(map[string]*StructureInfo)
r.classes = make(map[string]*ClassInfo)
reader := data.Reader()
r.elfByteOrder = reader.ByteOrder()
for {
entry, err := reader.Next()
if err != nil {
return fmt.Errorf("read dwarf error: %v", err)
}
if entry == nil {
break
}
if err := r.processProducer(data, entry); err != nil {
return err
}
if err := r.processFunctions(readAttrNames, data, entry); err != nil {
return err
}
if err := r.processStructure(readAttrNames, data, entry); err != nil {
return err
}
if err := r.processClass(readAttrNames, data, entry); err != nil {
return err
}
}
return nil
}