1061da546Spatrick //===-- ObjectFilePECOFF.h --------------------------------------*- C++ -*-===// 2061da546Spatrick // 3061da546Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4061da546Spatrick // See https://llvm.org/LICENSE.txt for license information. 5061da546Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6061da546Spatrick // 7061da546Spatrick //===----------------------------------------------------------------------===// 8061da546Spatrick 9dda28197Spatrick #ifndef LLDB_SOURCE_PLUGINS_OBJECTFILE_PECOFF_OBJECTFILEPECOFF_H 10dda28197Spatrick #define LLDB_SOURCE_PLUGINS_OBJECTFILE_PECOFF_OBJECTFILEPECOFF_H 11061da546Spatrick 12*f6aab3d8Srobert #include <optional> 13061da546Spatrick #include <vector> 14061da546Spatrick 15061da546Spatrick #include "lldb/Symbol/ObjectFile.h" 16dda28197Spatrick #include "llvm/Object/COFF.h" 17061da546Spatrick 18061da546Spatrick class ObjectFilePECOFF : public lldb_private::ObjectFile { 19061da546Spatrick public: 20061da546Spatrick enum MachineType { 21061da546Spatrick MachineUnknown = 0x0, 22061da546Spatrick MachineAm33 = 0x1d3, 23061da546Spatrick MachineAmd64 = 0x8664, 24061da546Spatrick MachineArm = 0x1c0, 25061da546Spatrick MachineArmNt = 0x1c4, 26061da546Spatrick MachineArm64 = 0xaa64, 27061da546Spatrick MachineEbc = 0xebc, 28061da546Spatrick MachineX86 = 0x14c, 29061da546Spatrick MachineIA64 = 0x200, 30061da546Spatrick MachineM32R = 0x9041, 31061da546Spatrick MachineMips16 = 0x266, 32061da546Spatrick MachineMipsFpu = 0x366, 33061da546Spatrick MachineMipsFpu16 = 0x466, 34061da546Spatrick MachinePowerPc = 0x1f0, 35061da546Spatrick MachinePowerPcfp = 0x1f1, 36061da546Spatrick MachineR4000 = 0x166, 37061da546Spatrick MachineSh3 = 0x1a2, 38061da546Spatrick MachineSh3dsp = 0x1a3, 39061da546Spatrick MachineSh4 = 0x1a6, 40061da546Spatrick MachineSh5 = 0x1a8, 41061da546Spatrick MachineThumb = 0x1c2, 42061da546Spatrick MachineWcemIpsv2 = 0x169 43061da546Spatrick }; 44061da546Spatrick 45*f6aab3d8Srobert ObjectFilePECOFF(const lldb::ModuleSP &module_sp, lldb::DataBufferSP data_sp, 46061da546Spatrick lldb::offset_t data_offset, 47061da546Spatrick const lldb_private::FileSpec *file, 48061da546Spatrick lldb::offset_t file_offset, lldb::offset_t length); 49061da546Spatrick 50061da546Spatrick ObjectFilePECOFF(const lldb::ModuleSP &module_sp, 51*f6aab3d8Srobert lldb::WritableDataBufferSP header_data_sp, 52061da546Spatrick const lldb::ProcessSP &process_sp, lldb::addr_t header_addr); 53061da546Spatrick 54061da546Spatrick ~ObjectFilePECOFF() override; 55061da546Spatrick 56061da546Spatrick // Static Functions 57061da546Spatrick static void Initialize(); 58061da546Spatrick 59*f6aab3d8Srobert static void DebuggerInitialize(lldb_private::Debugger &debugger); 60*f6aab3d8Srobert 61061da546Spatrick static void Terminate(); 62061da546Spatrick GetPluginNameStatic()63*f6aab3d8Srobert static llvm::StringRef GetPluginNameStatic() { return "pe-coff"; } 64061da546Spatrick 65*f6aab3d8Srobert static llvm::StringRef GetPluginDescriptionStatic(); 66061da546Spatrick 67061da546Spatrick static ObjectFile * 68*f6aab3d8Srobert CreateInstance(const lldb::ModuleSP &module_sp, lldb::DataBufferSP data_sp, 69061da546Spatrick lldb::offset_t data_offset, const lldb_private::FileSpec *file, 70061da546Spatrick lldb::offset_t offset, lldb::offset_t length); 71061da546Spatrick 72061da546Spatrick static lldb_private::ObjectFile *CreateMemoryInstance( 73*f6aab3d8Srobert const lldb::ModuleSP &module_sp, lldb::WritableDataBufferSP data_sp, 74061da546Spatrick const lldb::ProcessSP &process_sp, lldb::addr_t header_addr); 75061da546Spatrick 76061da546Spatrick static size_t GetModuleSpecifications(const lldb_private::FileSpec &file, 77061da546Spatrick lldb::DataBufferSP &data_sp, 78061da546Spatrick lldb::offset_t data_offset, 79061da546Spatrick lldb::offset_t file_offset, 80061da546Spatrick lldb::offset_t length, 81061da546Spatrick lldb_private::ModuleSpecList &specs); 82061da546Spatrick 83061da546Spatrick static bool SaveCore(const lldb::ProcessSP &process_sp, 84061da546Spatrick const lldb_private::FileSpec &outfile, 85be691f3bSpatrick lldb::SaveCoreStyle &core_style, 86061da546Spatrick lldb_private::Status &error); 87061da546Spatrick 88*f6aab3d8Srobert static bool MagicBytesMatch(lldb::DataBufferSP data_sp); 89061da546Spatrick 90061da546Spatrick static lldb::SymbolType MapSymbolType(uint16_t coff_symbol_type); 91061da546Spatrick 92061da546Spatrick // LLVM RTTI support 93061da546Spatrick static char ID; isA(const void * ClassID)94061da546Spatrick bool isA(const void *ClassID) const override { 95061da546Spatrick return ClassID == &ID || ObjectFile::isA(ClassID); 96061da546Spatrick } classof(const ObjectFile * obj)97061da546Spatrick static bool classof(const ObjectFile *obj) { return obj->isA(&ID); } 98061da546Spatrick 99061da546Spatrick bool ParseHeader() override; 100061da546Spatrick 101061da546Spatrick bool SetLoadAddress(lldb_private::Target &target, lldb::addr_t value, 102061da546Spatrick bool value_is_offset) override; 103061da546Spatrick 104061da546Spatrick lldb::ByteOrder GetByteOrder() const override; 105061da546Spatrick 106061da546Spatrick bool IsExecutable() const override; 107061da546Spatrick 108061da546Spatrick uint32_t GetAddressByteSize() const override; 109061da546Spatrick 110061da546Spatrick // virtual lldb_private::AddressClass 111061da546Spatrick // GetAddressClass (lldb::addr_t file_addr); 112061da546Spatrick 113*f6aab3d8Srobert void ParseSymtab(lldb_private::Symtab &symtab) override; 114061da546Spatrick 115061da546Spatrick bool IsStripped() override; 116061da546Spatrick 117061da546Spatrick void CreateSections(lldb_private::SectionList &unified_section_list) override; 118061da546Spatrick 119061da546Spatrick void Dump(lldb_private::Stream *s) override; 120061da546Spatrick 121061da546Spatrick lldb_private::ArchSpec GetArchitecture() override; 122061da546Spatrick 123061da546Spatrick lldb_private::UUID GetUUID() override; 124061da546Spatrick 125*f6aab3d8Srobert /// Return the contents of the .gnu_debuglink section, if the object file 126*f6aab3d8Srobert /// contains it. 127*f6aab3d8Srobert std::optional<lldb_private::FileSpec> GetDebugLink(); 128*f6aab3d8Srobert 129061da546Spatrick uint32_t GetDependentModules(lldb_private::FileSpecList &files) override; 130061da546Spatrick 131061da546Spatrick lldb_private::Address GetEntryPointAddress() override; 132061da546Spatrick 133061da546Spatrick lldb_private::Address GetBaseAddress() override; 134061da546Spatrick 135061da546Spatrick ObjectFile::Type CalculateType() override; 136061da546Spatrick 137061da546Spatrick ObjectFile::Strata CalculateStrata() override; 138061da546Spatrick 139061da546Spatrick // PluginInterface protocol GetPluginName()140*f6aab3d8Srobert llvm::StringRef GetPluginName() override { return GetPluginNameStatic(); } 141061da546Spatrick 142061da546Spatrick bool IsWindowsSubsystem(); 143061da546Spatrick 144061da546Spatrick uint32_t GetRVA(const lldb_private::Address &addr) const; 145061da546Spatrick lldb_private::Address GetAddress(uint32_t rva); 146061da546Spatrick lldb::addr_t GetFileAddress(uint32_t rva) const; 147061da546Spatrick 148061da546Spatrick lldb_private::DataExtractor ReadImageData(uint32_t offset, size_t size); 149061da546Spatrick lldb_private::DataExtractor ReadImageDataByRVA(uint32_t rva, size_t size); 150061da546Spatrick 151061da546Spatrick std::unique_ptr<lldb_private::CallFrameInfo> CreateCallFrameInfo() override; 152061da546Spatrick 153061da546Spatrick protected: 154061da546Spatrick bool NeedsEndianSwap() const; 155061da546Spatrick 156061da546Spatrick typedef struct dos_header { // DOS .EXE header 157*f6aab3d8Srobert uint16_t e_magic = 0; // Magic number 158*f6aab3d8Srobert uint16_t e_cblp = 0; // Bytes on last page of file 159*f6aab3d8Srobert uint16_t e_cp = 0; // Pages in file 160*f6aab3d8Srobert uint16_t e_crlc = 0; // Relocations 161*f6aab3d8Srobert uint16_t e_cparhdr = 0; // Size of header in paragraphs 162*f6aab3d8Srobert uint16_t e_minalloc = 0; // Minimum extra paragraphs needed 163*f6aab3d8Srobert uint16_t e_maxalloc = 0; // Maximum extra paragraphs needed 164*f6aab3d8Srobert uint16_t e_ss = 0; // Initial (relative) SS value 165*f6aab3d8Srobert uint16_t e_sp = 0; // Initial SP value 166*f6aab3d8Srobert uint16_t e_csum = 0; // Checksum 167*f6aab3d8Srobert uint16_t e_ip = 0; // Initial IP value 168*f6aab3d8Srobert uint16_t e_cs = 0; // Initial (relative) CS value 169*f6aab3d8Srobert uint16_t e_lfarlc = 0; // File address of relocation table 170*f6aab3d8Srobert uint16_t e_ovno = 0; // Overlay number 171061da546Spatrick uint16_t e_res[4]; // Reserved words 172*f6aab3d8Srobert uint16_t e_oemid = 0; // OEM identifier (for e_oeminfo) 173*f6aab3d8Srobert uint16_t e_oeminfo = 0; // OEM information; e_oemid specific 174*f6aab3d8Srobert uint16_t e_res2[10] = {}; // Reserved words 175*f6aab3d8Srobert uint32_t e_lfanew = 0; // File address of new exe header 176061da546Spatrick } dos_header_t; 177061da546Spatrick 178061da546Spatrick typedef struct coff_header { 179*f6aab3d8Srobert uint16_t machine = 0; 180*f6aab3d8Srobert uint16_t nsects = 0; 181*f6aab3d8Srobert uint32_t modtime = 0; 182*f6aab3d8Srobert uint32_t symoff = 0; 183*f6aab3d8Srobert uint32_t nsyms = 0; 184*f6aab3d8Srobert uint16_t hdrsize = 0; 185*f6aab3d8Srobert uint16_t flags = 0; 186061da546Spatrick } coff_header_t; 187061da546Spatrick 188061da546Spatrick typedef struct data_directory { 189*f6aab3d8Srobert uint32_t vmaddr = 0; 190*f6aab3d8Srobert uint32_t vmsize = 0; 191061da546Spatrick } data_directory_t; 192061da546Spatrick 193061da546Spatrick typedef struct coff_opt_header { 194061da546Spatrick uint16_t magic = 0; 195061da546Spatrick uint8_t major_linker_version = 0; 196061da546Spatrick uint8_t minor_linker_version = 0; 197061da546Spatrick uint32_t code_size = 0; 198061da546Spatrick uint32_t data_size = 0; 199061da546Spatrick uint32_t bss_size = 0; 200061da546Spatrick uint32_t entry = 0; 201061da546Spatrick uint32_t code_offset = 0; 202061da546Spatrick uint32_t data_offset = 0; 203061da546Spatrick 204061da546Spatrick uint64_t image_base = 0; 205061da546Spatrick uint32_t sect_alignment = 0; 206061da546Spatrick uint32_t file_alignment = 0; 207061da546Spatrick uint16_t major_os_system_version = 0; 208061da546Spatrick uint16_t minor_os_system_version = 0; 209061da546Spatrick uint16_t major_image_version = 0; 210061da546Spatrick uint16_t minor_image_version = 0; 211061da546Spatrick uint16_t major_subsystem_version = 0; 212061da546Spatrick uint16_t minor_subsystem_version = 0; 213061da546Spatrick uint32_t reserved1 = 0; 214061da546Spatrick uint32_t image_size = 0; 215061da546Spatrick uint32_t header_size = 0; 216061da546Spatrick uint32_t checksum = 0; 217061da546Spatrick uint16_t subsystem = 0; 218061da546Spatrick uint16_t dll_flags = 0; 219061da546Spatrick uint64_t stack_reserve_size = 0; 220061da546Spatrick uint64_t stack_commit_size = 0; 221061da546Spatrick uint64_t heap_reserve_size = 0; 222061da546Spatrick uint64_t heap_commit_size = 0; 223061da546Spatrick uint32_t loader_flags = 0; 224061da546Spatrick // uint32_t num_data_dir_entries; 225061da546Spatrick std::vector<data_directory> 226061da546Spatrick data_dirs; // will contain num_data_dir_entries entries 227061da546Spatrick } coff_opt_header_t; 228061da546Spatrick 229061da546Spatrick typedef struct section_header { 230*f6aab3d8Srobert char name[8] = {}; 231*f6aab3d8Srobert uint32_t vmsize = 0; // Virtual Size 232*f6aab3d8Srobert uint32_t vmaddr = 0; // Virtual Addr 233*f6aab3d8Srobert uint32_t size = 0; // File size 234*f6aab3d8Srobert uint32_t offset = 0; // File offset 235*f6aab3d8Srobert uint32_t reloff = 0; // Offset to relocations 236*f6aab3d8Srobert uint32_t lineoff = 0; // Offset to line table entries 237*f6aab3d8Srobert uint16_t nreloc = 0; // Number of relocation entries 238*f6aab3d8Srobert uint16_t nline = 0; // Number of line table entries 239*f6aab3d8Srobert uint32_t flags = 0; 240061da546Spatrick } section_header_t; 241061da546Spatrick 242061da546Spatrick static bool ParseDOSHeader(lldb_private::DataExtractor &data, 243061da546Spatrick dos_header_t &dos_header); 244061da546Spatrick static bool ParseCOFFHeader(lldb_private::DataExtractor &data, 245061da546Spatrick lldb::offset_t *offset_ptr, 246061da546Spatrick coff_header_t &coff_header); 247061da546Spatrick bool ParseCOFFOptionalHeader(lldb::offset_t *offset_ptr); 248061da546Spatrick bool ParseSectionHeaders(uint32_t offset); 249061da546Spatrick 250061da546Spatrick uint32_t ParseDependentModules(); 251061da546Spatrick 252061da546Spatrick static void DumpDOSHeader(lldb_private::Stream *s, 253061da546Spatrick const dos_header_t &header); 254061da546Spatrick static void DumpCOFFHeader(lldb_private::Stream *s, 255061da546Spatrick const coff_header_t &header); 256061da546Spatrick static void DumpOptCOFFHeader(lldb_private::Stream *s, 257061da546Spatrick const coff_opt_header_t &header); 258061da546Spatrick void DumpSectionHeaders(lldb_private::Stream *s); 259061da546Spatrick void DumpSectionHeader(lldb_private::Stream *s, const section_header_t &sh); 260061da546Spatrick void DumpDependentModules(lldb_private::Stream *s); 261061da546Spatrick 262061da546Spatrick llvm::StringRef GetSectionName(const section_header_t §); 263061da546Spatrick static lldb::SectionType GetSectionType(llvm::StringRef sect_name, 264061da546Spatrick const section_header_t §); 265061da546Spatrick 266061da546Spatrick typedef std::vector<section_header_t> SectionHeaderColl; 267061da546Spatrick typedef SectionHeaderColl::iterator SectionHeaderCollIter; 268061da546Spatrick typedef SectionHeaderColl::const_iterator SectionHeaderCollConstIter; 269061da546Spatrick 270061da546Spatrick private: 271061da546Spatrick bool CreateBinary(); 272*f6aab3d8Srobert typedef std::vector<std::pair<uint32_t, uint32_t>> rva_symbol_list_t; 273*f6aab3d8Srobert void AppendFromCOFFSymbolTable(lldb_private::SectionList *sect_list, 274*f6aab3d8Srobert lldb_private::Symtab &symtab, 275*f6aab3d8Srobert const rva_symbol_list_t &sorted_exports); 276*f6aab3d8Srobert rva_symbol_list_t AppendFromExportTable(lldb_private::SectionList *sect_list, 277*f6aab3d8Srobert lldb_private::Symtab &symtab); 278061da546Spatrick 279061da546Spatrick dos_header_t m_dos_header; 280061da546Spatrick coff_header_t m_coff_header; 281061da546Spatrick coff_opt_header_t m_coff_header_opt; 282061da546Spatrick SectionHeaderColl m_sect_headers; 283061da546Spatrick lldb::addr_t m_image_base; 284061da546Spatrick lldb_private::Address m_entry_point_address; 285*f6aab3d8Srobert std::optional<lldb_private::FileSpecList> m_deps_filespec; 286dda28197Spatrick std::unique_ptr<llvm::object::COFFObjectFile> m_binary; 287061da546Spatrick lldb_private::UUID m_uuid; 288061da546Spatrick }; 289061da546Spatrick 290dda28197Spatrick #endif // LLDB_SOURCE_PLUGINS_OBJECTFILE_PECOFF_OBJECTFILEPECOFF_H 291