/* * Copyright (c) 2017 Facebook, Inc. * * 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. */ #include #include #include #include #if LLVM_MAJOR_VERSION >= 15 #include #endif #include #include #include #include #include #include #include #include #include #include #if LLVM_MAJOR_VERSION >= 15 #include #endif #if LLVM_MAJOR_VERSION >= 14 #include #else #include #endif #include "bcc_debug.h" namespace ebpf { // ld_pseudo can only be disassembled properly // in llvm 6.0, so having this workaround now // until disto llvm versions catch up #define WORKAROUND_FOR_LD_PSEUDO using std::get; using std::map; using std::string; using std::tuple; using std::vector; using namespace llvm; using DWARFLineTable = DWARFDebugLine::LineTable; void SourceDebugger::adjustInstSize(uint64_t &Size, uint8_t byte0, uint8_t byte1) { #ifdef WORKAROUND_FOR_LD_PSEUDO bool isLittleEndian = mod_->getDataLayout().isLittleEndian(); if (byte0 == 0x18 && ((isLittleEndian && (byte1 & 0xf) == 0x1) || (!isLittleEndian && (byte1 & 0xf0) == 0x10))) Size = 16; #endif } vector SourceDebugger::buildLineCache() { vector LineCache; size_t FileBufSize = mod_src_.size(); for (uint32_t start = 0, end = start; end < FileBufSize; end++) if (mod_src_[end] == '\n' || end == FileBufSize - 1 || (mod_src_[end] == '\r' && mod_src_[end + 1] == '\n')) { // Not including the endline LineCache.push_back(string(mod_src_.substr(start, end - start))); if (mod_src_[end] == '\r') end++; start = end + 1; } return LineCache; } void SourceDebugger::dumpSrcLine(const vector &LineCache, const string &FileName, uint32_t Line, uint32_t &CurrentSrcLine, llvm::raw_ostream &os) { if (Line != 0 && Line != CurrentSrcLine && Line < LineCache.size() && FileName == mod_->getSourceFileName()) { os << "; " << StringRef(LineCache[Line - 1]).ltrim() << format( " // Line" "%4" PRIu64 "\n", Line); CurrentSrcLine = Line; } } void SourceDebugger::getDebugSections( StringMap> &DebugSections) { for (auto section : sections_) { if (strncmp(section.first.c_str(), ".debug", 6) == 0) { StringRef SecData(reinterpret_cast(get<0>(section.second)), get<1>(section.second)); DebugSections[section.first.substr(1)] = MemoryBuffer::getMemBufferCopy(SecData); } } } void SourceDebugger::dump() { string Error; string TripleStr(mod_->getTargetTriple()); Triple TheTriple(TripleStr); const Target *T = TargetRegistry::lookupTarget(TripleStr, Error); if (!T) { errs() << "Debug Error: cannot get target\n"; return; } std::unique_ptr MRI(T->createMCRegInfo(TripleStr)); if (!MRI) { errs() << "Debug Error: cannot get register info\n"; return; } #if LLVM_MAJOR_VERSION >= 10 MCTargetOptions MCOptions; std::unique_ptr MAI(T->createMCAsmInfo(*MRI, TripleStr, MCOptions)); #else std::unique_ptr MAI(T->createMCAsmInfo(*MRI, TripleStr)); #endif if (!MAI) { errs() << "Debug Error: cannot get assembly info\n"; return; } std::unique_ptr STI( T->createMCSubtargetInfo(TripleStr, "", "")); MCObjectFileInfo MOFI; #if LLVM_MAJOR_VERSION >= 13 MCContext Ctx(TheTriple, MAI.get(), MRI.get(), STI.get(), nullptr); Ctx.setObjectFileInfo(&MOFI); MOFI.initMCObjectFileInfo(Ctx, false, false); #else MCContext Ctx(MAI.get(), MRI.get(), &MOFI, nullptr); MOFI.InitMCObjectFileInfo(TheTriple, false, Ctx, false); #endif std::unique_ptr MCII(T->createMCInstrInfo()); MCInstPrinter *IP = T->createMCInstPrinter(TheTriple, 0, *MAI, *MCII, *MRI); if (!IP) { errs() << "Debug Error: unable to create instruction printer\n"; return; } std::unique_ptr DisAsm( T->createMCDisassembler(*STI, Ctx)); if (!DisAsm) { errs() << "Debug Error: no disassembler\n"; return; } // Set up the dwarf debug context StringMap> DebugSections; getDebugSections(DebugSections); std::unique_ptr DwarfCtx = DWARFContext::create(DebugSections, 8); if (!DwarfCtx) { errs() << "Debug Error: dwarf context creation failed\n"; return; } // bcc has only one compilation unit // getCompileUnitAtIndex() was gone in llvm 8.0 (https://reviews.llvm.org/D49741) #if LLVM_MAJOR_VERSION >= 8 DWARFCompileUnit *CU = cast(DwarfCtx->getUnitAtIndex(0)); #else DWARFCompileUnit *CU = DwarfCtx->getCompileUnitAtIndex(0); #endif if (!CU) { errs() << "Debug Error: dwarf context failed to get compile unit\n"; return; } const DWARFLineTable *LineTable = DwarfCtx->getLineTableForUnit(CU); if (!LineTable) { errs() << "Debug Error: dwarf context failed to get line table\n"; return; } // Build LineCache for later source code printing vector LineCache = buildLineCache(); // Start to disassemble with source code annotation section by section prog_func_info_.for_each_func([&](std::string func_name, FuncInfo &info) { MCDisassembler::DecodeStatus S; MCInst Inst; uint64_t Size; uint8_t *FuncStart = info.start_; uint64_t FuncSize = info.size_; #if LLVM_MAJOR_VERSION >= 9 auto section = sections_.find(info.section_); if (section == sections_.end()) { errs() << "Debug Error: no section entry for section " << info.section_ << '\n'; return; } unsigned SectionID = get<2>(section->second); #endif ArrayRef Data(FuncStart, FuncSize); uint32_t CurrentSrcLine = 0; errs() << "Disassembly of function " << func_name << "\n"; string src_dbg_str; llvm::raw_string_ostream os(src_dbg_str); for (uint64_t Index = 0; Index < FuncSize; Index += Size) { #if LLVM_MAJOR_VERSION >= 10 S = DisAsm->getInstruction(Inst, Size, Data.slice(Index), Index, nulls()); #else S = DisAsm->getInstruction(Inst, Size, Data.slice(Index), Index, nulls(), nulls()); #endif if (S != MCDisassembler::Success) { os << "Debug Error: disassembler failed: " << std::to_string(S) << '\n'; break; } else { DILineInfo LineInfo; LineTable->getFileLineInfoForAddress( #if LLVM_MAJOR_VERSION >= 9 {(uint64_t)FuncStart + Index, SectionID}, #else (uint64_t)FuncStart + Index, #endif CU->getCompilationDir(), DILineInfoSpecifier::FileLineInfoKind::AbsoluteFilePath, LineInfo); adjustInstSize(Size, Data[Index], Data[Index + 1]); dumpSrcLine(LineCache, LineInfo.FileName, LineInfo.Line, CurrentSrcLine, os); os << format("%4" PRIu64 ":", Index >> 3) << '\t'; dumpBytes(Data.slice(Index, Size), os); #if LLVM_MAJOR_VERSION >= 10 IP->printInst(&Inst, 0, "", *STI, os); #else IP->printInst(&Inst, os, "", *STI); #endif os << '\n'; } } os.flush(); errs() << src_dbg_str << '\n'; src_dbg_fmap_[func_name] = src_dbg_str; }); } } // namespace ebpf