//===-- DWARFDebugMacro.cpp -------------------------------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "DWARFDebugMacro.h" #include "SymbolFileDWARF.h" #include "lldb/Symbol/DebugMacros.h" #include "DWARFDataExtractor.h" using namespace lldb_private; DWARFDebugMacroHeader DWARFDebugMacroHeader::ParseHeader(const DWARFDataExtractor &debug_macro_data, lldb::offset_t *offset) { DWARFDebugMacroHeader header; // Skip over the version field in header. header.m_version = debug_macro_data.GetU16(offset); uint8_t flags = debug_macro_data.GetU8(offset); header.m_offset_is_64_bit = (flags & OFFSET_SIZE_MASK) != 0; if (flags & DEBUG_LINE_OFFSET_MASK) { if (header.m_offset_is_64_bit) header.m_debug_line_offset = debug_macro_data.GetU64(offset); else header.m_debug_line_offset = debug_macro_data.GetU32(offset); } // Skip over the operands table if it is present. if (flags & OPCODE_OPERANDS_TABLE_MASK) SkipOperandTable(debug_macro_data, offset); return header; } void DWARFDebugMacroHeader::SkipOperandTable( const DWARFDataExtractor &debug_macro_data, lldb::offset_t *offset) { uint8_t entry_count = debug_macro_data.GetU8(offset); for (uint8_t i = 0; i < entry_count; i++) { // Skip over the opcode number. debug_macro_data.GetU8(offset); uint64_t operand_count = debug_macro_data.GetULEB128(offset); for (uint64_t j = 0; j < operand_count; j++) { // Skip over the operand form debug_macro_data.GetU8(offset); } } } void DWARFDebugMacroEntry::ReadMacroEntries( const DWARFDataExtractor &debug_macro_data, const DWARFDataExtractor &debug_str_data, const bool offset_is_64_bit, lldb::offset_t *offset, SymbolFileDWARF *sym_file_dwarf, DebugMacrosSP &debug_macros_sp) { llvm::dwarf::MacroEntryType type = static_cast(debug_macro_data.GetU8(offset)); while (type != 0) { lldb::offset_t new_offset = 0, str_offset = 0; uint32_t line = 0; const char *macro_str = nullptr; uint32_t debug_line_file_idx = 0; switch (type) { case DW_MACRO_define: case DW_MACRO_undef: line = debug_macro_data.GetULEB128(offset); macro_str = debug_macro_data.GetCStr(offset); if (type == DW_MACRO_define) debug_macros_sp->AddMacroEntry( DebugMacroEntry::CreateDefineEntry(line, macro_str)); else debug_macros_sp->AddMacroEntry( DebugMacroEntry::CreateUndefEntry(line, macro_str)); break; case DW_MACRO_define_strp: case DW_MACRO_undef_strp: line = debug_macro_data.GetULEB128(offset); if (offset_is_64_bit) str_offset = debug_macro_data.GetU64(offset); else str_offset = debug_macro_data.GetU32(offset); macro_str = debug_str_data.GetCStr(&str_offset); if (type == DW_MACRO_define_strp) debug_macros_sp->AddMacroEntry( DebugMacroEntry::CreateDefineEntry(line, macro_str)); else debug_macros_sp->AddMacroEntry( DebugMacroEntry::CreateUndefEntry(line, macro_str)); break; case DW_MACRO_start_file: line = debug_macro_data.GetULEB128(offset); debug_line_file_idx = debug_macro_data.GetULEB128(offset); debug_macros_sp->AddMacroEntry( DebugMacroEntry::CreateStartFileEntry(line, debug_line_file_idx)); break; case DW_MACRO_end_file: // This operation has no operands. debug_macros_sp->AddMacroEntry(DebugMacroEntry::CreateEndFileEntry()); break; case DW_MACRO_import: if (offset_is_64_bit) new_offset = debug_macro_data.GetU64(offset); else new_offset = debug_macro_data.GetU32(offset); debug_macros_sp->AddMacroEntry(DebugMacroEntry::CreateIndirectEntry( sym_file_dwarf->ParseDebugMacros(&new_offset))); break; default: // TODO: Add support for other standard operations. // TODO: Provide mechanism to hook handling of non-standard/extension // operands. return; } type = static_cast( debug_macro_data.GetU8(offset)); } }