1944b828aSGreg Clayton //===-- DynamicLoaderDarwinKernel.cpp -----------------------------*- C++ -*-===// 2d4bfbc9aSGreg Clayton // 3d4bfbc9aSGreg Clayton // The LLVM Compiler Infrastructure 4d4bfbc9aSGreg Clayton // 5d4bfbc9aSGreg Clayton // This file is distributed under the University of Illinois Open Source 6d4bfbc9aSGreg Clayton // License. See LICENSE.TXT for details. 7d4bfbc9aSGreg Clayton // 8d4bfbc9aSGreg Clayton //===----------------------------------------------------------------------===// 9d4bfbc9aSGreg Clayton 10d4bfbc9aSGreg Clayton #include "lldb/Breakpoint/StoppointCallbackContext.h" 11d4bfbc9aSGreg Clayton #include "lldb/Core/DataBuffer.h" 12d4bfbc9aSGreg Clayton #include "lldb/Core/DataBufferHeap.h" 13d4bfbc9aSGreg Clayton #include "lldb/Core/Debugger.h" 14d4bfbc9aSGreg Clayton #include "lldb/Core/Log.h" 15d4bfbc9aSGreg Clayton #include "lldb/Core/Module.h" 161f746071SGreg Clayton #include "lldb/Core/ModuleSpec.h" 17d4bfbc9aSGreg Clayton #include "lldb/Core/PluginManager.h" 181f746071SGreg Clayton #include "lldb/Core/Section.h" 19d4bfbc9aSGreg Clayton #include "lldb/Core/State.h" 2068b3607fSJason Molenda #include "lldb/Host/Symbols.h" 21d4bfbc9aSGreg Clayton #include "lldb/Symbol/ObjectFile.h" 22d4bfbc9aSGreg Clayton #include "lldb/Target/RegisterContext.h" 2368b3607fSJason Molenda #include "lldb/Target/StackFrame.h" 24d4bfbc9aSGreg Clayton #include "lldb/Target/Target.h" 25d4bfbc9aSGreg Clayton #include "lldb/Target/Thread.h" 26d4bfbc9aSGreg Clayton #include "lldb/Target/ThreadPlanRunToAddress.h" 2768b3607fSJason Molenda 28d4bfbc9aSGreg Clayton 29944b828aSGreg Clayton #include "DynamicLoaderDarwinKernel.h" 30d4bfbc9aSGreg Clayton 31d4bfbc9aSGreg Clayton //#define ENABLE_DEBUG_PRINTF // COMMENT THIS LINE OUT PRIOR TO CHECKIN 32d4bfbc9aSGreg Clayton #ifdef ENABLE_DEBUG_PRINTF 33d4bfbc9aSGreg Clayton #include <stdio.h> 34d4bfbc9aSGreg Clayton #define DEBUG_PRINTF(fmt, ...) printf(fmt, ## __VA_ARGS__) 35d4bfbc9aSGreg Clayton #else 36d4bfbc9aSGreg Clayton #define DEBUG_PRINTF(fmt, ...) 37d4bfbc9aSGreg Clayton #endif 38d4bfbc9aSGreg Clayton 39d4bfbc9aSGreg Clayton using namespace lldb; 40d4bfbc9aSGreg Clayton using namespace lldb_private; 41d4bfbc9aSGreg Clayton 42e8cd0c98SGreg Clayton static PropertyDefinition 43e8cd0c98SGreg Clayton g_properties[] = 44e8cd0c98SGreg Clayton { 4566763eedSGreg Clayton { "load-kexts" , OptionValue::eTypeBoolean, true, true, NULL, NULL, "Automatically loads kext images when attaching to a kernel." }, 46e8cd0c98SGreg Clayton { NULL , OptionValue::eTypeInvalid, false, 0 , NULL, NULL, NULL } 47e8cd0c98SGreg Clayton }; 48e8cd0c98SGreg Clayton 49e8cd0c98SGreg Clayton enum { 5066763eedSGreg Clayton ePropertyLoadKexts 51e8cd0c98SGreg Clayton }; 52e8cd0c98SGreg Clayton 53e8cd0c98SGreg Clayton class DynamicLoaderDarwinKernelProperties : public Properties 54e8cd0c98SGreg Clayton { 55e8cd0c98SGreg Clayton public: 56e8cd0c98SGreg Clayton 57e8cd0c98SGreg Clayton static ConstString & 58e8cd0c98SGreg Clayton GetSettingName () 59e8cd0c98SGreg Clayton { 60468ea4ebSGreg Clayton static ConstString g_setting_name("darwin-kernel"); 61e8cd0c98SGreg Clayton return g_setting_name; 62e8cd0c98SGreg Clayton } 63e8cd0c98SGreg Clayton 64e8cd0c98SGreg Clayton DynamicLoaderDarwinKernelProperties() : 65e8cd0c98SGreg Clayton Properties () 66e8cd0c98SGreg Clayton { 67e8cd0c98SGreg Clayton m_collection_sp.reset (new OptionValueProperties(GetSettingName())); 68e8cd0c98SGreg Clayton m_collection_sp->Initialize(g_properties); 69e8cd0c98SGreg Clayton } 70e8cd0c98SGreg Clayton 71e8cd0c98SGreg Clayton virtual 72e8cd0c98SGreg Clayton ~DynamicLoaderDarwinKernelProperties() 73e8cd0c98SGreg Clayton { 74e8cd0c98SGreg Clayton } 75e8cd0c98SGreg Clayton 76e8cd0c98SGreg Clayton bool 7766763eedSGreg Clayton GetLoadKexts() const 78e8cd0c98SGreg Clayton { 7966763eedSGreg Clayton const uint32_t idx = ePropertyLoadKexts; 80e8cd0c98SGreg Clayton return m_collection_sp->GetPropertyAtIndexAsBoolean (NULL, idx, g_properties[idx].default_uint_value != 0); 81e8cd0c98SGreg Clayton } 82e8cd0c98SGreg Clayton 83e8cd0c98SGreg Clayton }; 84e8cd0c98SGreg Clayton 85e8cd0c98SGreg Clayton typedef STD_SHARED_PTR(DynamicLoaderDarwinKernelProperties) DynamicLoaderDarwinKernelPropertiesSP; 86e8cd0c98SGreg Clayton 87e8cd0c98SGreg Clayton static const DynamicLoaderDarwinKernelPropertiesSP & 88e8cd0c98SGreg Clayton GetGlobalProperties() 89e8cd0c98SGreg Clayton { 90e8cd0c98SGreg Clayton static DynamicLoaderDarwinKernelPropertiesSP g_settings_sp; 91e8cd0c98SGreg Clayton if (!g_settings_sp) 92e8cd0c98SGreg Clayton g_settings_sp.reset (new DynamicLoaderDarwinKernelProperties ()); 93e8cd0c98SGreg Clayton return g_settings_sp; 94e8cd0c98SGreg Clayton } 95e8cd0c98SGreg Clayton 96d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 97d4bfbc9aSGreg Clayton // Create an instance of this class. This function is filled into 98d4bfbc9aSGreg Clayton // the plugin info class that gets handed out by the plugin factory and 99d4bfbc9aSGreg Clayton // allows the lldb to instantiate an instance of this class. 100d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 101d4bfbc9aSGreg Clayton DynamicLoader * 102944b828aSGreg Clayton DynamicLoaderDarwinKernel::CreateInstance (Process* process, bool force) 103d4bfbc9aSGreg Clayton { 104d4bfbc9aSGreg Clayton bool create = force; 105d4bfbc9aSGreg Clayton if (!create) 106d4bfbc9aSGreg Clayton { 107d4bfbc9aSGreg Clayton Module* exe_module = process->GetTarget().GetExecutableModulePointer(); 108d4bfbc9aSGreg Clayton if (exe_module) 109d4bfbc9aSGreg Clayton { 110d4bfbc9aSGreg Clayton ObjectFile *object_file = exe_module->GetObjectFile(); 111d4bfbc9aSGreg Clayton if (object_file) 112d4bfbc9aSGreg Clayton { 11349bce8ecSSean Callanan create = (object_file->GetStrata() == ObjectFile::eStrataKernel); 114d4bfbc9aSGreg Clayton } 115d4bfbc9aSGreg Clayton } 116d4bfbc9aSGreg Clayton 117d4bfbc9aSGreg Clayton if (create) 118d4bfbc9aSGreg Clayton { 119d4bfbc9aSGreg Clayton const llvm::Triple &triple_ref = process->GetTarget().GetArchitecture().GetTriple(); 12070512317SGreg Clayton switch (triple_ref.getOS()) 12170512317SGreg Clayton { 12270512317SGreg Clayton case llvm::Triple::Darwin: 12370512317SGreg Clayton case llvm::Triple::MacOSX: 12470512317SGreg Clayton case llvm::Triple::IOS: 12570512317SGreg Clayton create = triple_ref.getVendor() == llvm::Triple::Apple; 12670512317SGreg Clayton break; 12770512317SGreg Clayton default: 12870512317SGreg Clayton create = false; 12970512317SGreg Clayton break; 13070512317SGreg Clayton } 131d4bfbc9aSGreg Clayton } 132d4bfbc9aSGreg Clayton } 133d4bfbc9aSGreg Clayton 134d4bfbc9aSGreg Clayton if (create) 13590539456SSean Callanan { 13690539456SSean Callanan process->SetCanJIT(false); 137944b828aSGreg Clayton return new DynamicLoaderDarwinKernel (process); 13890539456SSean Callanan } 139d4bfbc9aSGreg Clayton return NULL; 140d4bfbc9aSGreg Clayton } 141d4bfbc9aSGreg Clayton 142d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 143d4bfbc9aSGreg Clayton // Constructor 144d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 145944b828aSGreg Clayton DynamicLoaderDarwinKernel::DynamicLoaderDarwinKernel (Process* process) : 146d4bfbc9aSGreg Clayton DynamicLoader(process), 147d4bfbc9aSGreg Clayton m_kernel(), 148d4bfbc9aSGreg Clayton m_kext_summary_header_ptr_addr (), 149d4bfbc9aSGreg Clayton m_kext_summary_header_addr (), 150d4bfbc9aSGreg Clayton m_kext_summary_header (), 151d4bfbc9aSGreg Clayton m_kext_summaries(), 152a08823fdSDaniel Dunbar m_mutex(Mutex::eMutexTypeRecursive), 153a08823fdSDaniel Dunbar m_break_id (LLDB_INVALID_BREAK_ID) 154d4bfbc9aSGreg Clayton { 155d4bfbc9aSGreg Clayton } 156d4bfbc9aSGreg Clayton 157d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 158d4bfbc9aSGreg Clayton // Destructor 159d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 160944b828aSGreg Clayton DynamicLoaderDarwinKernel::~DynamicLoaderDarwinKernel() 161d4bfbc9aSGreg Clayton { 162d4bfbc9aSGreg Clayton Clear(true); 163d4bfbc9aSGreg Clayton } 164d4bfbc9aSGreg Clayton 165d4bfbc9aSGreg Clayton void 166944b828aSGreg Clayton DynamicLoaderDarwinKernel::UpdateIfNeeded() 167d4bfbc9aSGreg Clayton { 168d4bfbc9aSGreg Clayton LoadKernelModuleIfNeeded(); 169d4bfbc9aSGreg Clayton SetNotificationBreakpointIfNeeded (); 170d4bfbc9aSGreg Clayton } 171d4bfbc9aSGreg Clayton //------------------------------------------------------------------ 172d4bfbc9aSGreg Clayton /// Called after attaching a process. 173d4bfbc9aSGreg Clayton /// 174d4bfbc9aSGreg Clayton /// Allow DynamicLoader plug-ins to execute some code after 175d4bfbc9aSGreg Clayton /// attaching to a process. 176d4bfbc9aSGreg Clayton //------------------------------------------------------------------ 177d4bfbc9aSGreg Clayton void 178944b828aSGreg Clayton DynamicLoaderDarwinKernel::DidAttach () 179d4bfbc9aSGreg Clayton { 180d4bfbc9aSGreg Clayton PrivateInitialize(m_process); 181d4bfbc9aSGreg Clayton UpdateIfNeeded(); 182d4bfbc9aSGreg Clayton } 183d4bfbc9aSGreg Clayton 184d4bfbc9aSGreg Clayton //------------------------------------------------------------------ 185d4bfbc9aSGreg Clayton /// Called after attaching a process. 186d4bfbc9aSGreg Clayton /// 187d4bfbc9aSGreg Clayton /// Allow DynamicLoader plug-ins to execute some code after 188d4bfbc9aSGreg Clayton /// attaching to a process. 189d4bfbc9aSGreg Clayton //------------------------------------------------------------------ 190d4bfbc9aSGreg Clayton void 191944b828aSGreg Clayton DynamicLoaderDarwinKernel::DidLaunch () 192d4bfbc9aSGreg Clayton { 193d4bfbc9aSGreg Clayton PrivateInitialize(m_process); 194d4bfbc9aSGreg Clayton UpdateIfNeeded(); 195d4bfbc9aSGreg Clayton } 196d4bfbc9aSGreg Clayton 197d4bfbc9aSGreg Clayton 198d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 199d4bfbc9aSGreg Clayton // Clear out the state of this class. 200d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 201d4bfbc9aSGreg Clayton void 202944b828aSGreg Clayton DynamicLoaderDarwinKernel::Clear (bool clear_process) 203d4bfbc9aSGreg Clayton { 204d4bfbc9aSGreg Clayton Mutex::Locker locker(m_mutex); 205d4bfbc9aSGreg Clayton 206d4bfbc9aSGreg Clayton if (m_process->IsAlive() && LLDB_BREAK_ID_IS_VALID(m_break_id)) 207d4bfbc9aSGreg Clayton m_process->ClearBreakpointSiteByID(m_break_id); 208d4bfbc9aSGreg Clayton 209d4bfbc9aSGreg Clayton if (clear_process) 210d4bfbc9aSGreg Clayton m_process = NULL; 211d4bfbc9aSGreg Clayton m_kernel.Clear(false); 212d4bfbc9aSGreg Clayton m_kext_summary_header_ptr_addr.Clear(); 213d4bfbc9aSGreg Clayton m_kext_summary_header_addr.Clear(); 214d4bfbc9aSGreg Clayton m_kext_summaries.clear(); 215d4bfbc9aSGreg Clayton m_break_id = LLDB_INVALID_BREAK_ID; 216d4bfbc9aSGreg Clayton } 217d4bfbc9aSGreg Clayton 218d4bfbc9aSGreg Clayton 219c859e2d5SGreg Clayton bool 2202af282a1SGreg Clayton DynamicLoaderDarwinKernel::OSKextLoadedKextSummary::LoadImageAtFileAddress (Process *process) 2212af282a1SGreg Clayton { 2222af282a1SGreg Clayton if (IsLoaded()) 2232af282a1SGreg Clayton return true; 2242af282a1SGreg Clayton 2252af282a1SGreg Clayton if (module_sp) 2262af282a1SGreg Clayton { 2272af282a1SGreg Clayton bool changed = false; 2282af282a1SGreg Clayton if (module_sp->SetLoadAddress (process->GetTarget(), 0, changed)) 2292af282a1SGreg Clayton load_process_stop_id = process->GetStopID(); 2302af282a1SGreg Clayton } 2312af282a1SGreg Clayton return false; 2322af282a1SGreg Clayton } 2332af282a1SGreg Clayton 2342af282a1SGreg Clayton bool 235c859e2d5SGreg Clayton DynamicLoaderDarwinKernel::OSKextLoadedKextSummary::LoadImageUsingMemoryModule (Process *process) 236c859e2d5SGreg Clayton { 237c859e2d5SGreg Clayton if (IsLoaded()) 238c859e2d5SGreg Clayton return true; 239c859e2d5SGreg Clayton 240c859e2d5SGreg Clayton bool uuid_is_valid = uuid.IsValid(); 24168b3607fSJason Molenda bool memory_module_is_kernel = false; 242c859e2d5SGreg Clayton 243c859e2d5SGreg Clayton Target &target = process->GetTarget(); 244c859e2d5SGreg Clayton ModuleSP memory_module_sp; 24587a04b24SJason Molenda 24687a04b24SJason Molenda // If this is a kext and the user asked us to ignore kexts, don't try to load it. 24766763eedSGreg Clayton if (kernel_image == false && GetGlobalProperties()->GetLoadKexts() == false) 24887a04b24SJason Molenda { 24987a04b24SJason Molenda return false; 25087a04b24SJason Molenda } 25187a04b24SJason Molenda 25287a04b24SJason Molenda // Use the memory module as the module if we have one 253c859e2d5SGreg Clayton if (address != LLDB_INVALID_ADDRESS) 254c859e2d5SGreg Clayton { 255c859e2d5SGreg Clayton FileSpec file_spec; 256c859e2d5SGreg Clayton if (module_sp) 257c859e2d5SGreg Clayton file_spec = module_sp->GetFileSpec(); 258c859e2d5SGreg Clayton else 259c859e2d5SGreg Clayton file_spec.SetFile (name, false); 260c859e2d5SGreg Clayton 261c859e2d5SGreg Clayton memory_module_sp = process->ReadModuleFromMemory (file_spec, address, false, false); 262c859e2d5SGreg Clayton if (memory_module_sp && !uuid_is_valid) 263c859e2d5SGreg Clayton { 264c859e2d5SGreg Clayton uuid = memory_module_sp->GetUUID(); 265c859e2d5SGreg Clayton uuid_is_valid = uuid.IsValid(); 266c859e2d5SGreg Clayton } 267*c56bd083SJason Molenda if (memory_module_sp 268*c56bd083SJason Molenda && memory_module_sp->GetObjectFile() 26968b3607fSJason Molenda && memory_module_sp->GetObjectFile()->GetType() == ObjectFile::eTypeExecutable 27068b3607fSJason Molenda && memory_module_sp->GetObjectFile()->GetStrata() == ObjectFile::eStrataKernel) 27168b3607fSJason Molenda { 27268b3607fSJason Molenda memory_module_is_kernel = true; 27353667f5dSJason Molenda if (memory_module_sp->GetArchitecture().IsValid()) 27453667f5dSJason Molenda { 27553667f5dSJason Molenda target.SetArchitecture(memory_module_sp->GetArchitecture()); 27653667f5dSJason Molenda } 27768b3607fSJason Molenda } 278c859e2d5SGreg Clayton } 279c859e2d5SGreg Clayton 280c859e2d5SGreg Clayton if (!module_sp) 281c859e2d5SGreg Clayton { 282c859e2d5SGreg Clayton if (uuid_is_valid) 283c859e2d5SGreg Clayton { 284c859e2d5SGreg Clayton ModuleList &target_images = target.GetImages(); 285c859e2d5SGreg Clayton module_sp = target_images.FindModule(uuid); 286c859e2d5SGreg Clayton 287c859e2d5SGreg Clayton if (!module_sp) 288b9a01b39SGreg Clayton { 2892af282a1SGreg Clayton ModuleSpec module_spec; 290b9a01b39SGreg Clayton module_spec.GetUUID() = uuid; 29153667f5dSJason Molenda module_spec.GetArchitecture() = target.GetArchitecture(); 292bb860bd2SJason Molenda 293bb860bd2SJason Molenda // For the kernel, we really do need an on-disk file copy of the 294bb860bd2SJason Molenda // binary. 295bb860bd2SJason Molenda bool force_symbols_search = false; 296bb860bd2SJason Molenda if (memory_module_is_kernel) 297bb860bd2SJason Molenda { 298bb860bd2SJason Molenda force_symbols_search = true; 299bb860bd2SJason Molenda } 300bb860bd2SJason Molenda 301bb860bd2SJason Molenda if (Symbols::DownloadObjectAndSymbolFile (module_spec, force_symbols_search)) 302bb860bd2SJason Molenda { 303bb860bd2SJason Molenda if (module_spec.GetFileSpec().Exists()) 304bb860bd2SJason Molenda { 305bb860bd2SJason Molenda module_sp.reset(new Module (module_spec.GetFileSpec(), target.GetArchitecture())); 306bb860bd2SJason Molenda if (module_sp.get() && module_sp->MatchesModuleSpec (module_spec)) 307bb860bd2SJason Molenda { 308bb860bd2SJason Molenda ModuleList loaded_module_list; 309bb860bd2SJason Molenda loaded_module_list.Append (module_sp); 310bb860bd2SJason Molenda target.ModulesDidLoad (loaded_module_list); 311bb860bd2SJason Molenda } 312bb860bd2SJason Molenda } 313bb860bd2SJason Molenda } 314bb860bd2SJason Molenda 315bb860bd2SJason Molenda // Ask the Target to find this file on the local system, if possible. 316bb860bd2SJason Molenda // This will search in the list of currently-loaded files, look in the 317bb860bd2SJason Molenda // standard search paths on the system, and on a Mac it will try calling 318bb860bd2SJason Molenda // the DebugSymbols framework with the UUID to find the binary via its 319bb860bd2SJason Molenda // search methods. 320bb860bd2SJason Molenda if (!module_sp) 321bb860bd2SJason Molenda { 322b9a01b39SGreg Clayton module_sp = target.GetSharedModule (module_spec); 323b9a01b39SGreg Clayton } 324c859e2d5SGreg Clayton } 325c859e2d5SGreg Clayton } 326bb860bd2SJason Molenda } 327c859e2d5SGreg Clayton 328c859e2d5SGreg Clayton 329bb860bd2SJason Molenda if (memory_module_sp && module_sp) 330c859e2d5SGreg Clayton { 331c859e2d5SGreg Clayton if (module_sp->GetUUID() == memory_module_sp->GetUUID()) 332c859e2d5SGreg Clayton { 333bb860bd2SJason Molenda target.GetImages().Append(module_sp); 334bb860bd2SJason Molenda if (memory_module_is_kernel && target.GetExecutableModulePointer() != module_sp.get()) 335bb860bd2SJason Molenda { 336bb860bd2SJason Molenda target.SetExecutableModule (module_sp, false); 337bb860bd2SJason Molenda } 338bb860bd2SJason Molenda 339c859e2d5SGreg Clayton ObjectFile *ondisk_object_file = module_sp->GetObjectFile(); 340c859e2d5SGreg Clayton ObjectFile *memory_object_file = memory_module_sp->GetObjectFile(); 341c859e2d5SGreg Clayton if (memory_object_file && ondisk_object_file) 342c859e2d5SGreg Clayton { 343c859e2d5SGreg Clayton SectionList *ondisk_section_list = ondisk_object_file->GetSectionList (); 344c859e2d5SGreg Clayton SectionList *memory_section_list = memory_object_file->GetSectionList (); 345c859e2d5SGreg Clayton if (memory_section_list && ondisk_section_list) 346c859e2d5SGreg Clayton { 347bae2f2f8SGreg Clayton const uint32_t num_ondisk_sections = ondisk_section_list->GetSize(); 348c859e2d5SGreg Clayton // There may be CTF sections in the memory image so we can't 349c859e2d5SGreg Clayton // always just compare the number of sections (which are actually 350c859e2d5SGreg Clayton // segments in mach-o parlance) 351c859e2d5SGreg Clayton uint32_t sect_idx = 0; 352bae2f2f8SGreg Clayton 35353667f5dSJason Molenda // Use the memory_module's addresses for each section to set the 35453667f5dSJason Molenda // file module's load address as appropriate. We don't want to use 35553667f5dSJason Molenda // a single slide value for the entire kext - different segments may 35653667f5dSJason Molenda // be slid different amounts by the kext loader. 357bae2f2f8SGreg Clayton 358bae2f2f8SGreg Clayton uint32_t num_sections_loaded = 0; 359bae2f2f8SGreg Clayton for (sect_idx=0; sect_idx<num_ondisk_sections; ++sect_idx) 360c859e2d5SGreg Clayton { 3617820bd1eSGreg Clayton SectionSP ondisk_section_sp(ondisk_section_list->GetSectionAtIndex(sect_idx)); 3627820bd1eSGreg Clayton if (ondisk_section_sp) 363c859e2d5SGreg Clayton { 3647820bd1eSGreg Clayton const Section *memory_section = memory_section_list->FindSectionByName(ondisk_section_sp->GetName()).get(); 365bae2f2f8SGreg Clayton if (memory_section) 366c859e2d5SGreg Clayton { 3677820bd1eSGreg Clayton target.GetSectionLoadList().SetSectionLoadAddress (ondisk_section_sp, memory_section->GetFileAddress()); 368bae2f2f8SGreg Clayton ++num_sections_loaded; 369c859e2d5SGreg Clayton } 370bae2f2f8SGreg Clayton } 371bae2f2f8SGreg Clayton } 372bae2f2f8SGreg Clayton if (num_sections_loaded > 0) 373c859e2d5SGreg Clayton load_process_stop_id = process->GetStopID(); 374bae2f2f8SGreg Clayton else 375bae2f2f8SGreg Clayton module_sp.reset(); // No sections were loaded 376c859e2d5SGreg Clayton } 377c859e2d5SGreg Clayton else 378c859e2d5SGreg Clayton module_sp.reset(); // One or both section lists 379c859e2d5SGreg Clayton } 380c859e2d5SGreg Clayton else 381c859e2d5SGreg Clayton module_sp.reset(); // One or both object files missing 382c859e2d5SGreg Clayton } 383c859e2d5SGreg Clayton else 384c859e2d5SGreg Clayton module_sp.reset(); // UUID mismatch 385c859e2d5SGreg Clayton } 386c859e2d5SGreg Clayton 387c859e2d5SGreg Clayton bool is_loaded = IsLoaded(); 388c859e2d5SGreg Clayton 389c859e2d5SGreg Clayton if (so_address.IsValid()) 390c859e2d5SGreg Clayton { 391c859e2d5SGreg Clayton if (is_loaded) 392c859e2d5SGreg Clayton so_address.SetLoadAddress (address, &target); 393c859e2d5SGreg Clayton else 394c859e2d5SGreg Clayton target.GetImages().ResolveFileAddress (address, so_address); 395c859e2d5SGreg Clayton 396c859e2d5SGreg Clayton } 39768b3607fSJason Molenda 39868b3607fSJason Molenda if (is_loaded && module_sp && memory_module_is_kernel) 39968b3607fSJason Molenda { 40068b3607fSJason Molenda Stream *s = &target.GetDebugger().GetOutputStream(); 40168b3607fSJason Molenda if (s) 40268b3607fSJason Molenda { 40368b3607fSJason Molenda char uuidbuf[64]; 40468b3607fSJason Molenda s->Printf ("Kernel UUID: %s\n", module_sp->GetUUID().GetAsCString(uuidbuf, sizeof (uuidbuf))); 40568b3607fSJason Molenda s->Printf ("Load Address: 0x%llx\n", address); 406743e4396SJason Molenda if (module_sp->GetFileSpec().GetDirectory().IsEmpty()) 407743e4396SJason Molenda { 408743e4396SJason Molenda s->Printf ("Loaded kernel file %s\n", module_sp->GetFileSpec().GetFilename().AsCString()); 409743e4396SJason Molenda } 410743e4396SJason Molenda else 411743e4396SJason Molenda { 41268b3607fSJason Molenda s->Printf ("Loaded kernel file %s/%s\n", 41368b3607fSJason Molenda module_sp->GetFileSpec().GetDirectory().AsCString(), 41468b3607fSJason Molenda module_sp->GetFileSpec().GetFilename().AsCString()); 415743e4396SJason Molenda } 41668b3607fSJason Molenda s->Flush (); 41768b3607fSJason Molenda } 41868b3607fSJason Molenda } 419c859e2d5SGreg Clayton return is_loaded; 420c859e2d5SGreg Clayton } 421c859e2d5SGreg Clayton 4221f746071SGreg Clayton uint32_t 4231f746071SGreg Clayton DynamicLoaderDarwinKernel::OSKextLoadedKextSummary::GetAddressByteSize () 4241f746071SGreg Clayton { 4251f746071SGreg Clayton if (module_sp) 4261f746071SGreg Clayton return module_sp->GetArchitecture().GetAddressByteSize(); 4271f746071SGreg Clayton return 0; 4281f746071SGreg Clayton } 4291f746071SGreg Clayton 4301f746071SGreg Clayton lldb::ByteOrder 4311f746071SGreg Clayton DynamicLoaderDarwinKernel::OSKextLoadedKextSummary::GetByteOrder() 4321f746071SGreg Clayton { 4331f746071SGreg Clayton if (module_sp) 4341f746071SGreg Clayton return module_sp->GetArchitecture().GetByteOrder(); 4351f746071SGreg Clayton return lldb::endian::InlHostByteOrder(); 4361f746071SGreg Clayton } 4371f746071SGreg Clayton 4381f746071SGreg Clayton lldb_private::ArchSpec 4391f746071SGreg Clayton DynamicLoaderDarwinKernel::OSKextLoadedKextSummary::GetArchitecture () const 4401f746071SGreg Clayton { 4411f746071SGreg Clayton if (module_sp) 4421f746071SGreg Clayton return module_sp->GetArchitecture(); 4431f746071SGreg Clayton return lldb_private::ArchSpec (); 4441f746071SGreg Clayton } 4451f746071SGreg Clayton 4461f746071SGreg Clayton 447d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 448d4bfbc9aSGreg Clayton // Load the kernel module and initialize the "m_kernel" member. Return 449d4bfbc9aSGreg Clayton // true _only_ if the kernel is loaded the first time through (subsequent 450d4bfbc9aSGreg Clayton // calls to this function should return false after the kernel has been 451d4bfbc9aSGreg Clayton // already loaded). 452d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 453d4bfbc9aSGreg Clayton void 454944b828aSGreg Clayton DynamicLoaderDarwinKernel::LoadKernelModuleIfNeeded() 455d4bfbc9aSGreg Clayton { 456d4bfbc9aSGreg Clayton if (!m_kext_summary_header_ptr_addr.IsValid()) 457d4bfbc9aSGreg Clayton { 458d4bfbc9aSGreg Clayton m_kernel.Clear(false); 459d4bfbc9aSGreg Clayton m_kernel.module_sp = m_process->GetTarget().GetExecutableModule(); 46087a04b24SJason Molenda m_kernel.kernel_image = true; 4614bd4e7e3SJason Molenda 4624bd4e7e3SJason Molenda ConstString kernel_name("mach_kernel"); 4634bd4e7e3SJason Molenda if (m_kernel.module_sp.get() 4644bd4e7e3SJason Molenda && m_kernel.module_sp->GetObjectFile() 4654bd4e7e3SJason Molenda && !m_kernel.module_sp->GetObjectFile()->GetFileSpec().GetFilename().IsEmpty()) 4664bd4e7e3SJason Molenda { 4674bd4e7e3SJason Molenda kernel_name = m_kernel.module_sp->GetObjectFile()->GetFileSpec().GetFilename(); 4684bd4e7e3SJason Molenda } 46931a6961cSJason Molenda strncpy (m_kernel.name, kernel_name.AsCString(), sizeof(m_kernel.name)); 47031a6961cSJason Molenda m_kernel.name[sizeof (m_kernel.name) - 1] = '\0'; 4714bd4e7e3SJason Molenda 472c859e2d5SGreg Clayton if (m_kernel.address == LLDB_INVALID_ADDRESS) 473d4bfbc9aSGreg Clayton { 474c859e2d5SGreg Clayton m_kernel.address = m_process->GetImageInfoAddress (); 475c859e2d5SGreg Clayton if (m_kernel.address == LLDB_INVALID_ADDRESS && m_kernel.module_sp) 476c859e2d5SGreg Clayton { 477c859e2d5SGreg Clayton // We didn't get a hint from the process, so we will 478c859e2d5SGreg Clayton // try the kernel at the address that it exists at in 479c859e2d5SGreg Clayton // the file if we have one 480c859e2d5SGreg Clayton ObjectFile *kernel_object_file = m_kernel.module_sp->GetObjectFile(); 481c859e2d5SGreg Clayton if (kernel_object_file) 4824bd4e7e3SJason Molenda { 4834bd4e7e3SJason Molenda addr_t load_address = kernel_object_file->GetHeaderAddress().GetLoadAddress(&m_process->GetTarget()); 4844bd4e7e3SJason Molenda addr_t file_address = kernel_object_file->GetHeaderAddress().GetFileAddress(); 4854bd4e7e3SJason Molenda if (load_address != LLDB_INVALID_ADDRESS && load_address != 0) 4864bd4e7e3SJason Molenda { 4874bd4e7e3SJason Molenda m_kernel.address = load_address; 4884bd4e7e3SJason Molenda if (load_address != file_address) 4894bd4e7e3SJason Molenda { 4904bd4e7e3SJason Molenda // Don't accidentally relocate the kernel to the File address -- 4914bd4e7e3SJason Molenda // the Load address has already been set to its actual in-memory address. 4924bd4e7e3SJason Molenda // Mark it as IsLoaded. 4934bd4e7e3SJason Molenda m_kernel.load_process_stop_id = m_process->GetStopID(); 4944bd4e7e3SJason Molenda } 4954bd4e7e3SJason Molenda } 4964bd4e7e3SJason Molenda else 4974bd4e7e3SJason Molenda { 4984bd4e7e3SJason Molenda m_kernel.address = file_address; 4994bd4e7e3SJason Molenda } 5004bd4e7e3SJason Molenda } 501c859e2d5SGreg Clayton } 502c859e2d5SGreg Clayton } 503c859e2d5SGreg Clayton 504c859e2d5SGreg Clayton if (m_kernel.address != LLDB_INVALID_ADDRESS) 5052af282a1SGreg Clayton { 5062af282a1SGreg Clayton if (!m_kernel.LoadImageUsingMemoryModule (m_process)) 5072af282a1SGreg Clayton { 5082af282a1SGreg Clayton m_kernel.LoadImageAtFileAddress (m_process); 5092af282a1SGreg Clayton } 5102af282a1SGreg Clayton } 511c859e2d5SGreg Clayton 51228eb5711SJim Ingham if (m_kernel.IsLoaded() && m_kernel.module_sp) 513c859e2d5SGreg Clayton { 514d4bfbc9aSGreg Clayton static ConstString kext_summary_symbol ("gLoadedKextSummaries"); 515c859e2d5SGreg Clayton const Symbol *symbol = m_kernel.module_sp->FindFirstSymbolWithNameAndType (kext_summary_symbol, eSymbolTypeData); 516d4bfbc9aSGreg Clayton if (symbol) 517c859e2d5SGreg Clayton { 518e7612134SGreg Clayton m_kext_summary_header_ptr_addr = symbol->GetAddress(); 519d4bfbc9aSGreg Clayton // Update all image infos 520d4bfbc9aSGreg Clayton ReadAllKextSummaries (); 521d4bfbc9aSGreg Clayton } 522d4bfbc9aSGreg Clayton } 523d4bfbc9aSGreg Clayton else 524d4bfbc9aSGreg Clayton { 525d4bfbc9aSGreg Clayton m_kernel.Clear(false); 526d4bfbc9aSGreg Clayton } 527d4bfbc9aSGreg Clayton } 528d4bfbc9aSGreg Clayton } 529d4bfbc9aSGreg Clayton 530d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 531d4bfbc9aSGreg Clayton // Static callback function that gets called when our DYLD notification 532d4bfbc9aSGreg Clayton // breakpoint gets hit. We update all of our image infos and then 533d4bfbc9aSGreg Clayton // let our super class DynamicLoader class decide if we should stop 534d4bfbc9aSGreg Clayton // or not (based on global preference). 535d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 536d4bfbc9aSGreg Clayton bool 537944b828aSGreg Clayton DynamicLoaderDarwinKernel::BreakpointHitCallback (void *baton, 538d4bfbc9aSGreg Clayton StoppointCallbackContext *context, 539d4bfbc9aSGreg Clayton user_id_t break_id, 540d4bfbc9aSGreg Clayton user_id_t break_loc_id) 541d4bfbc9aSGreg Clayton { 542944b828aSGreg Clayton return static_cast<DynamicLoaderDarwinKernel*>(baton)->BreakpointHit (context, break_id, break_loc_id); 543d4bfbc9aSGreg Clayton } 544d4bfbc9aSGreg Clayton 545d4bfbc9aSGreg Clayton bool 546944b828aSGreg Clayton DynamicLoaderDarwinKernel::BreakpointHit (StoppointCallbackContext *context, 547d4bfbc9aSGreg Clayton user_id_t break_id, 548d4bfbc9aSGreg Clayton user_id_t break_loc_id) 549d4bfbc9aSGreg Clayton { 550d4bfbc9aSGreg Clayton LogSP log(GetLogIfAnyCategoriesSet (LIBLLDB_LOG_DYNAMIC_LOADER)); 551d4bfbc9aSGreg Clayton if (log) 552944b828aSGreg Clayton log->Printf ("DynamicLoaderDarwinKernel::BreakpointHit (...)\n"); 553d4bfbc9aSGreg Clayton 554d4bfbc9aSGreg Clayton ReadAllKextSummaries (); 555d4bfbc9aSGreg Clayton 556d4bfbc9aSGreg Clayton if (log) 557d4bfbc9aSGreg Clayton PutToLog(log.get()); 558d4bfbc9aSGreg Clayton 559d4bfbc9aSGreg Clayton return GetStopWhenImagesChange(); 560d4bfbc9aSGreg Clayton } 561d4bfbc9aSGreg Clayton 562d4bfbc9aSGreg Clayton 563d4bfbc9aSGreg Clayton bool 564944b828aSGreg Clayton DynamicLoaderDarwinKernel::ReadKextSummaryHeader () 565d4bfbc9aSGreg Clayton { 566d4bfbc9aSGreg Clayton Mutex::Locker locker(m_mutex); 567d4bfbc9aSGreg Clayton 568d4bfbc9aSGreg Clayton // the all image infos is already valid for this process stop ID 569d4bfbc9aSGreg Clayton 570d4bfbc9aSGreg Clayton m_kext_summaries.clear(); 571d4bfbc9aSGreg Clayton if (m_kext_summary_header_ptr_addr.IsValid()) 572d4bfbc9aSGreg Clayton { 573d4bfbc9aSGreg Clayton const uint32_t addr_size = m_kernel.GetAddressByteSize (); 574d4bfbc9aSGreg Clayton const ByteOrder byte_order = m_kernel.GetByteOrder(); 575d4bfbc9aSGreg Clayton Error error; 576d4bfbc9aSGreg Clayton // Read enough bytes for a "OSKextLoadedKextSummaryHeader" structure 577d4bfbc9aSGreg Clayton // which is currenty 4 uint32_t and a pointer. 578d4bfbc9aSGreg Clayton uint8_t buf[24]; 579d4bfbc9aSGreg Clayton DataExtractor data (buf, sizeof(buf), byte_order, addr_size); 580d4bfbc9aSGreg Clayton const size_t count = 4 * sizeof(uint32_t) + addr_size; 581d4bfbc9aSGreg Clayton const bool prefer_file_cache = false; 582d4bfbc9aSGreg Clayton if (m_process->GetTarget().ReadPointerFromMemory (m_kext_summary_header_ptr_addr, 583d4bfbc9aSGreg Clayton prefer_file_cache, 584d4bfbc9aSGreg Clayton error, 585d4bfbc9aSGreg Clayton m_kext_summary_header_addr)) 586d4bfbc9aSGreg Clayton { 587d4bfbc9aSGreg Clayton // We got a valid address for our kext summary header and make sure it isn't NULL 588d4bfbc9aSGreg Clayton if (m_kext_summary_header_addr.IsValid() && 589d4bfbc9aSGreg Clayton m_kext_summary_header_addr.GetFileAddress() != 0) 590d4bfbc9aSGreg Clayton { 591d4bfbc9aSGreg Clayton const size_t bytes_read = m_process->GetTarget().ReadMemory (m_kext_summary_header_addr, prefer_file_cache, buf, count, error); 592d4bfbc9aSGreg Clayton if (bytes_read == count) 593d4bfbc9aSGreg Clayton { 594d4bfbc9aSGreg Clayton uint32_t offset = 0; 595d4bfbc9aSGreg Clayton m_kext_summary_header.version = data.GetU32(&offset); 596d4bfbc9aSGreg Clayton if (m_kext_summary_header.version >= 2) 597d4bfbc9aSGreg Clayton { 598d4bfbc9aSGreg Clayton m_kext_summary_header.entry_size = data.GetU32(&offset); 599d4bfbc9aSGreg Clayton } 600d4bfbc9aSGreg Clayton else 601d4bfbc9aSGreg Clayton { 602d4bfbc9aSGreg Clayton // Versions less than 2 didn't have an entry size, it was hard coded 603d4bfbc9aSGreg Clayton m_kext_summary_header.entry_size = KERNEL_MODULE_ENTRY_SIZE_VERSION_1; 604d4bfbc9aSGreg Clayton } 605d4bfbc9aSGreg Clayton m_kext_summary_header.entry_count = data.GetU32(&offset); 606d4bfbc9aSGreg Clayton return true; 607d4bfbc9aSGreg Clayton } 608d4bfbc9aSGreg Clayton } 609d4bfbc9aSGreg Clayton } 610d4bfbc9aSGreg Clayton } 611d4bfbc9aSGreg Clayton m_kext_summary_header_addr.Clear(); 612d4bfbc9aSGreg Clayton return false; 613d4bfbc9aSGreg Clayton } 614d4bfbc9aSGreg Clayton 615d4bfbc9aSGreg Clayton 616d4bfbc9aSGreg Clayton bool 617944b828aSGreg Clayton DynamicLoaderDarwinKernel::ParseKextSummaries (const Address &kext_summary_addr, 618d4bfbc9aSGreg Clayton uint32_t count) 619d4bfbc9aSGreg Clayton { 620d4bfbc9aSGreg Clayton OSKextLoadedKextSummary::collection kext_summaries; 621d4bfbc9aSGreg Clayton LogSP log(GetLogIfAnyCategoriesSet (LIBLLDB_LOG_DYNAMIC_LOADER)); 622d4bfbc9aSGreg Clayton if (log) 623fd54b368SJason Molenda log->Printf ("Adding %d modules.\n", count); 624d4bfbc9aSGreg Clayton 625d4bfbc9aSGreg Clayton Mutex::Locker locker(m_mutex); 626d4bfbc9aSGreg Clayton 627d4bfbc9aSGreg Clayton if (!ReadKextSummaries (kext_summary_addr, count, kext_summaries)) 628d4bfbc9aSGreg Clayton return false; 629d4bfbc9aSGreg Clayton 630d4bfbc9aSGreg Clayton Stream *s = &m_process->GetTarget().GetDebugger().GetOutputStream(); 63168b3607fSJason Molenda if (s) 63268b3607fSJason Molenda s->Printf ("Loading %d kext modules ", count); 633d4bfbc9aSGreg Clayton for (uint32_t i = 0; i < count; i++) 634d4bfbc9aSGreg Clayton { 6352af282a1SGreg Clayton if (!kext_summaries[i].LoadImageUsingMemoryModule (m_process)) 6362af282a1SGreg Clayton kext_summaries[i].LoadImageAtFileAddress (m_process); 637d4bfbc9aSGreg Clayton 638d4bfbc9aSGreg Clayton if (s) 63968b3607fSJason Molenda s->Printf ("."); 640d4bfbc9aSGreg Clayton 641d4bfbc9aSGreg Clayton if (log) 642d4bfbc9aSGreg Clayton kext_summaries[i].PutToLog (log.get()); 643d4bfbc9aSGreg Clayton } 64468b3607fSJason Molenda if (s) 64568b3607fSJason Molenda { 64668b3607fSJason Molenda s->Printf (" done.\n"); 64768b3607fSJason Molenda s->Flush (); 64868b3607fSJason Molenda } 64968b3607fSJason Molenda 650d4bfbc9aSGreg Clayton bool return_value = AddModulesUsingImageInfos (kext_summaries); 651d4bfbc9aSGreg Clayton return return_value; 652d4bfbc9aSGreg Clayton } 653d4bfbc9aSGreg Clayton 654d4bfbc9aSGreg Clayton // Adds the modules in image_infos to m_kext_summaries. 655d4bfbc9aSGreg Clayton // NB don't call this passing in m_kext_summaries. 656d4bfbc9aSGreg Clayton 657d4bfbc9aSGreg Clayton bool 658944b828aSGreg Clayton DynamicLoaderDarwinKernel::AddModulesUsingImageInfos (OSKextLoadedKextSummary::collection &image_infos) 659d4bfbc9aSGreg Clayton { 660d4bfbc9aSGreg Clayton // Now add these images to the main list. 661d4bfbc9aSGreg Clayton ModuleList loaded_module_list; 662d4bfbc9aSGreg Clayton 663d4bfbc9aSGreg Clayton for (uint32_t idx = 0; idx < image_infos.size(); ++idx) 664d4bfbc9aSGreg Clayton { 665c859e2d5SGreg Clayton OSKextLoadedKextSummary &image_info = image_infos[idx]; 666c859e2d5SGreg Clayton m_kext_summaries.push_back(image_info); 667d4bfbc9aSGreg Clayton 668c859e2d5SGreg Clayton if (image_info.module_sp && m_process->GetStopID() == image_info.load_process_stop_id) 669d4bfbc9aSGreg Clayton loaded_module_list.AppendIfNeeded (image_infos[idx].module_sp); 670d4bfbc9aSGreg Clayton } 671d4bfbc9aSGreg Clayton 672d4bfbc9aSGreg Clayton if (loaded_module_list.GetSize() > 0) 673d4bfbc9aSGreg Clayton { 674d4bfbc9aSGreg Clayton m_process->GetTarget().ModulesDidLoad (loaded_module_list); 675d4bfbc9aSGreg Clayton } 676d4bfbc9aSGreg Clayton return true; 677d4bfbc9aSGreg Clayton } 678d4bfbc9aSGreg Clayton 679d4bfbc9aSGreg Clayton 680d4bfbc9aSGreg Clayton uint32_t 681944b828aSGreg Clayton DynamicLoaderDarwinKernel::ReadKextSummaries (const Address &kext_summary_addr, 682d4bfbc9aSGreg Clayton uint32_t image_infos_count, 683d4bfbc9aSGreg Clayton OSKextLoadedKextSummary::collection &image_infos) 684d4bfbc9aSGreg Clayton { 685d4bfbc9aSGreg Clayton const ByteOrder endian = m_kernel.GetByteOrder(); 686d4bfbc9aSGreg Clayton const uint32_t addr_size = m_kernel.GetAddressByteSize(); 687d4bfbc9aSGreg Clayton 688d4bfbc9aSGreg Clayton image_infos.resize(image_infos_count); 689d4bfbc9aSGreg Clayton const size_t count = image_infos.size() * m_kext_summary_header.entry_size; 690d4bfbc9aSGreg Clayton DataBufferHeap data(count, 0); 691d4bfbc9aSGreg Clayton Error error; 692d4bfbc9aSGreg Clayton 693d4bfbc9aSGreg Clayton const bool prefer_file_cache = false; 694d4bfbc9aSGreg Clayton const size_t bytes_read = m_process->GetTarget().ReadMemory (kext_summary_addr, 695d4bfbc9aSGreg Clayton prefer_file_cache, 696d4bfbc9aSGreg Clayton data.GetBytes(), 697d4bfbc9aSGreg Clayton data.GetByteSize(), 698d4bfbc9aSGreg Clayton error); 699d4bfbc9aSGreg Clayton if (bytes_read == count) 700d4bfbc9aSGreg Clayton { 701d4bfbc9aSGreg Clayton 702d4bfbc9aSGreg Clayton DataExtractor extractor (data.GetBytes(), data.GetByteSize(), endian, addr_size); 703d4bfbc9aSGreg Clayton uint32_t i=0; 704d4bfbc9aSGreg Clayton for (uint32_t kext_summary_offset = 0; 705d4bfbc9aSGreg Clayton i < image_infos.size() && extractor.ValidOffsetForDataOfSize(kext_summary_offset, m_kext_summary_header.entry_size); 706d4bfbc9aSGreg Clayton ++i, kext_summary_offset += m_kext_summary_header.entry_size) 707d4bfbc9aSGreg Clayton { 708d4bfbc9aSGreg Clayton uint32_t offset = kext_summary_offset; 709d4bfbc9aSGreg Clayton const void *name_data = extractor.GetData(&offset, KERNEL_MODULE_MAX_NAME); 710d4bfbc9aSGreg Clayton if (name_data == NULL) 711d4bfbc9aSGreg Clayton break; 712d4bfbc9aSGreg Clayton memcpy (image_infos[i].name, name_data, KERNEL_MODULE_MAX_NAME); 713d4bfbc9aSGreg Clayton image_infos[i].uuid.SetBytes(extractor.GetData (&offset, 16)); 714d4bfbc9aSGreg Clayton image_infos[i].address = extractor.GetU64(&offset); 715d4bfbc9aSGreg Clayton if (!image_infos[i].so_address.SetLoadAddress (image_infos[i].address, &m_process->GetTarget())) 716d4bfbc9aSGreg Clayton m_process->GetTarget().GetImages().ResolveFileAddress (image_infos[i].address, image_infos[i].so_address); 717d4bfbc9aSGreg Clayton image_infos[i].size = extractor.GetU64(&offset); 718d4bfbc9aSGreg Clayton image_infos[i].version = extractor.GetU64(&offset); 719d4bfbc9aSGreg Clayton image_infos[i].load_tag = extractor.GetU32(&offset); 720d4bfbc9aSGreg Clayton image_infos[i].flags = extractor.GetU32(&offset); 721d4bfbc9aSGreg Clayton if ((offset - kext_summary_offset) < m_kext_summary_header.entry_size) 722d4bfbc9aSGreg Clayton { 723d4bfbc9aSGreg Clayton image_infos[i].reference_list = extractor.GetU64(&offset); 724d4bfbc9aSGreg Clayton } 725d4bfbc9aSGreg Clayton else 726d4bfbc9aSGreg Clayton { 727d4bfbc9aSGreg Clayton image_infos[i].reference_list = 0; 728d4bfbc9aSGreg Clayton } 729bae2f2f8SGreg Clayton // printf ("[%3u] %*.*s: address=0x%16.16llx, size=0x%16.16llx, version=0x%16.16llx, load_tag=0x%8.8x, flags=0x%8.8x\n", 730bae2f2f8SGreg Clayton // i, 731bae2f2f8SGreg Clayton // KERNEL_MODULE_MAX_NAME, KERNEL_MODULE_MAX_NAME, (char *)name_data, 732bae2f2f8SGreg Clayton // image_infos[i].address, 733bae2f2f8SGreg Clayton // image_infos[i].size, 734bae2f2f8SGreg Clayton // image_infos[i].version, 735bae2f2f8SGreg Clayton // image_infos[i].load_tag, 736bae2f2f8SGreg Clayton // image_infos[i].flags); 737d4bfbc9aSGreg Clayton } 738d4bfbc9aSGreg Clayton if (i < image_infos.size()) 739d4bfbc9aSGreg Clayton image_infos.resize(i); 740d4bfbc9aSGreg Clayton } 741d4bfbc9aSGreg Clayton else 742d4bfbc9aSGreg Clayton { 743d4bfbc9aSGreg Clayton image_infos.clear(); 744d4bfbc9aSGreg Clayton } 745d4bfbc9aSGreg Clayton return image_infos.size(); 746d4bfbc9aSGreg Clayton } 747d4bfbc9aSGreg Clayton 748d4bfbc9aSGreg Clayton bool 749944b828aSGreg Clayton DynamicLoaderDarwinKernel::ReadAllKextSummaries () 750d4bfbc9aSGreg Clayton { 751d4bfbc9aSGreg Clayton LogSP log(GetLogIfAnyCategoriesSet (LIBLLDB_LOG_DYNAMIC_LOADER)); 752d4bfbc9aSGreg Clayton 753d4bfbc9aSGreg Clayton Mutex::Locker locker(m_mutex); 754d4bfbc9aSGreg Clayton 755d4bfbc9aSGreg Clayton if (ReadKextSummaryHeader ()) 756d4bfbc9aSGreg Clayton { 757d4bfbc9aSGreg Clayton if (m_kext_summary_header.entry_count > 0 && m_kext_summary_header_addr.IsValid()) 758d4bfbc9aSGreg Clayton { 759d4bfbc9aSGreg Clayton Address summary_addr (m_kext_summary_header_addr); 760d4bfbc9aSGreg Clayton summary_addr.Slide(m_kext_summary_header.GetSize()); 761d4bfbc9aSGreg Clayton if (!ParseKextSummaries (summary_addr, m_kext_summary_header.entry_count)) 762d4bfbc9aSGreg Clayton { 763d4bfbc9aSGreg Clayton m_kext_summaries.clear(); 764d4bfbc9aSGreg Clayton } 765d4bfbc9aSGreg Clayton return true; 766d4bfbc9aSGreg Clayton } 767d4bfbc9aSGreg Clayton } 768d4bfbc9aSGreg Clayton return false; 769d4bfbc9aSGreg Clayton } 770d4bfbc9aSGreg Clayton 771d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 772d4bfbc9aSGreg Clayton // Dump an image info structure to the file handle provided. 773d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 774d4bfbc9aSGreg Clayton void 775944b828aSGreg Clayton DynamicLoaderDarwinKernel::OSKextLoadedKextSummary::PutToLog (Log *log) const 776d4bfbc9aSGreg Clayton { 777d4bfbc9aSGreg Clayton if (log == NULL) 778d4bfbc9aSGreg Clayton return; 779d4bfbc9aSGreg Clayton const uint8_t *u = (uint8_t *)uuid.GetBytes(); 780d4bfbc9aSGreg Clayton 781d4bfbc9aSGreg Clayton if (address == LLDB_INVALID_ADDRESS) 782d4bfbc9aSGreg Clayton { 783d4bfbc9aSGreg Clayton if (u) 784d4bfbc9aSGreg Clayton { 785d4bfbc9aSGreg Clayton log->Printf("\tuuid=%2.2X%2.2X%2.2X%2.2X-%2.2X%2.2X-%2.2X%2.2X-%2.2X%2.2X-%2.2X%2.2X%2.2X%2.2X%2.2X%2.2X name=\"%s\" (UNLOADED)", 786d4bfbc9aSGreg Clayton u[ 0], u[ 1], u[ 2], u[ 3], 787d4bfbc9aSGreg Clayton u[ 4], u[ 5], u[ 6], u[ 7], 788d4bfbc9aSGreg Clayton u[ 8], u[ 9], u[10], u[11], 789d4bfbc9aSGreg Clayton u[12], u[13], u[14], u[15], 790d4bfbc9aSGreg Clayton name); 791d4bfbc9aSGreg Clayton } 792d4bfbc9aSGreg Clayton else 793d4bfbc9aSGreg Clayton log->Printf("\tname=\"%s\" (UNLOADED)", name); 794d4bfbc9aSGreg Clayton } 795d4bfbc9aSGreg Clayton else 796d4bfbc9aSGreg Clayton { 797d4bfbc9aSGreg Clayton if (u) 798d4bfbc9aSGreg Clayton { 799d4bfbc9aSGreg Clayton log->Printf("\taddr=0x%16.16llx size=0x%16.16llx version=0x%16.16llx load-tag=0x%8.8x flags=0x%8.8x ref-list=0x%16.16llx uuid=%2.2X%2.2X%2.2X%2.2X-%2.2X%2.2X-%2.2X%2.2X-%2.2X%2.2X-%2.2X%2.2X%2.2X%2.2X%2.2X%2.2X name=\"%s\"", 800d4bfbc9aSGreg Clayton address, size, version, load_tag, flags, reference_list, 801d4bfbc9aSGreg Clayton u[ 0], u[ 1], u[ 2], u[ 3], u[ 4], u[ 5], u[ 6], u[ 7], 802d4bfbc9aSGreg Clayton u[ 8], u[ 9], u[10], u[11], u[12], u[13], u[14], u[15], 803d4bfbc9aSGreg Clayton name); 804d4bfbc9aSGreg Clayton } 805d4bfbc9aSGreg Clayton else 806d4bfbc9aSGreg Clayton { 807d4bfbc9aSGreg Clayton log->Printf("\t[0x%16.16llx - 0x%16.16llx) version=0x%16.16llx load-tag=0x%8.8x flags=0x%8.8x ref-list=0x%16.16llx name=\"%s\"", 808d4bfbc9aSGreg Clayton address, address+size, version, load_tag, flags, reference_list, 809d4bfbc9aSGreg Clayton name); 810d4bfbc9aSGreg Clayton } 811d4bfbc9aSGreg Clayton } 812d4bfbc9aSGreg Clayton } 813d4bfbc9aSGreg Clayton 814d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 815d4bfbc9aSGreg Clayton // Dump the _dyld_all_image_infos members and all current image infos 816d4bfbc9aSGreg Clayton // that we have parsed to the file handle provided. 817d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 818d4bfbc9aSGreg Clayton void 819944b828aSGreg Clayton DynamicLoaderDarwinKernel::PutToLog(Log *log) const 820d4bfbc9aSGreg Clayton { 821d4bfbc9aSGreg Clayton if (log == NULL) 822d4bfbc9aSGreg Clayton return; 823d4bfbc9aSGreg Clayton 824d4bfbc9aSGreg Clayton Mutex::Locker locker(m_mutex); 825d4bfbc9aSGreg Clayton log->Printf("gLoadedKextSummaries = 0x%16.16llx { version=%u, entry_size=%u, entry_count=%u }", 826d4bfbc9aSGreg Clayton m_kext_summary_header_addr.GetFileAddress(), 827d4bfbc9aSGreg Clayton m_kext_summary_header.version, 828d4bfbc9aSGreg Clayton m_kext_summary_header.entry_size, 829d4bfbc9aSGreg Clayton m_kext_summary_header.entry_count); 830d4bfbc9aSGreg Clayton 831d4bfbc9aSGreg Clayton size_t i; 832d4bfbc9aSGreg Clayton const size_t count = m_kext_summaries.size(); 833d4bfbc9aSGreg Clayton if (count > 0) 834d4bfbc9aSGreg Clayton { 835d4bfbc9aSGreg Clayton log->PutCString("Loaded:"); 836d4bfbc9aSGreg Clayton for (i = 0; i<count; i++) 837d4bfbc9aSGreg Clayton m_kext_summaries[i].PutToLog(log); 838d4bfbc9aSGreg Clayton } 839d4bfbc9aSGreg Clayton } 840d4bfbc9aSGreg Clayton 841d4bfbc9aSGreg Clayton void 842944b828aSGreg Clayton DynamicLoaderDarwinKernel::PrivateInitialize(Process *process) 843d4bfbc9aSGreg Clayton { 844944b828aSGreg Clayton DEBUG_PRINTF("DynamicLoaderDarwinKernel::%s() process state = %s\n", __FUNCTION__, StateAsCString(m_process->GetState())); 845d4bfbc9aSGreg Clayton Clear(true); 846d4bfbc9aSGreg Clayton m_process = process; 847d4bfbc9aSGreg Clayton } 848d4bfbc9aSGreg Clayton 849d4bfbc9aSGreg Clayton void 850944b828aSGreg Clayton DynamicLoaderDarwinKernel::SetNotificationBreakpointIfNeeded () 851d4bfbc9aSGreg Clayton { 852c859e2d5SGreg Clayton if (m_break_id == LLDB_INVALID_BREAK_ID && m_kernel.module_sp) 853d4bfbc9aSGreg Clayton { 854944b828aSGreg Clayton DEBUG_PRINTF("DynamicLoaderDarwinKernel::%s() process state = %s\n", __FUNCTION__, StateAsCString(m_process->GetState())); 855d4bfbc9aSGreg Clayton 856d4bfbc9aSGreg Clayton 857a8558b62SJim Ingham const bool internal_bp = true; 858d4bfbc9aSGreg Clayton const LazyBool skip_prologue = eLazyBoolNo; 859969795f1SJim Ingham FileSpecList module_spec_list; 860969795f1SJim Ingham module_spec_list.Append (m_kernel.module_sp->GetFileSpec()); 861969795f1SJim Ingham Breakpoint *bp = m_process->GetTarget().CreateBreakpoint (&module_spec_list, 86287df91b8SJim Ingham NULL, 863d4bfbc9aSGreg Clayton "OSKextLoadedKextSummariesUpdated", 864d4bfbc9aSGreg Clayton eFunctionNameTypeFull, 865a8558b62SJim Ingham skip_prologue, 866a8558b62SJim Ingham internal_bp).get(); 867d4bfbc9aSGreg Clayton 868944b828aSGreg Clayton bp->SetCallback (DynamicLoaderDarwinKernel::BreakpointHitCallback, this, true); 869d4bfbc9aSGreg Clayton m_break_id = bp->GetID(); 870d4bfbc9aSGreg Clayton } 871d4bfbc9aSGreg Clayton } 872d4bfbc9aSGreg Clayton 873d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 874d4bfbc9aSGreg Clayton // Member function that gets called when the process state changes. 875d4bfbc9aSGreg Clayton //---------------------------------------------------------------------- 876d4bfbc9aSGreg Clayton void 877944b828aSGreg Clayton DynamicLoaderDarwinKernel::PrivateProcessStateChanged (Process *process, StateType state) 878d4bfbc9aSGreg Clayton { 879944b828aSGreg Clayton DEBUG_PRINTF("DynamicLoaderDarwinKernel::%s(%s)\n", __FUNCTION__, StateAsCString(state)); 880d4bfbc9aSGreg Clayton switch (state) 881d4bfbc9aSGreg Clayton { 882d4bfbc9aSGreg Clayton case eStateConnected: 883d4bfbc9aSGreg Clayton case eStateAttaching: 884d4bfbc9aSGreg Clayton case eStateLaunching: 885d4bfbc9aSGreg Clayton case eStateInvalid: 886d4bfbc9aSGreg Clayton case eStateUnloaded: 887d4bfbc9aSGreg Clayton case eStateExited: 888d4bfbc9aSGreg Clayton case eStateDetached: 889d4bfbc9aSGreg Clayton Clear(false); 890d4bfbc9aSGreg Clayton break; 891d4bfbc9aSGreg Clayton 892d4bfbc9aSGreg Clayton case eStateStopped: 893d4bfbc9aSGreg Clayton UpdateIfNeeded(); 894d4bfbc9aSGreg Clayton break; 895d4bfbc9aSGreg Clayton 896d4bfbc9aSGreg Clayton case eStateRunning: 897d4bfbc9aSGreg Clayton case eStateStepping: 898d4bfbc9aSGreg Clayton case eStateCrashed: 899d4bfbc9aSGreg Clayton case eStateSuspended: 900d4bfbc9aSGreg Clayton break; 901d4bfbc9aSGreg Clayton 902d4bfbc9aSGreg Clayton default: 903d4bfbc9aSGreg Clayton break; 904d4bfbc9aSGreg Clayton } 905d4bfbc9aSGreg Clayton } 906d4bfbc9aSGreg Clayton 907d4bfbc9aSGreg Clayton ThreadPlanSP 908944b828aSGreg Clayton DynamicLoaderDarwinKernel::GetStepThroughTrampolinePlan (Thread &thread, bool stop_others) 909d4bfbc9aSGreg Clayton { 910d4bfbc9aSGreg Clayton ThreadPlanSP thread_plan_sp; 911d4bfbc9aSGreg Clayton LogSP log(GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP)); 912d4bfbc9aSGreg Clayton if (log) 913d4bfbc9aSGreg Clayton log->Printf ("Could not find symbol for step through."); 914d4bfbc9aSGreg Clayton return thread_plan_sp; 915d4bfbc9aSGreg Clayton } 916d4bfbc9aSGreg Clayton 917d4bfbc9aSGreg Clayton Error 918944b828aSGreg Clayton DynamicLoaderDarwinKernel::CanLoadImage () 919d4bfbc9aSGreg Clayton { 920d4bfbc9aSGreg Clayton Error error; 921d4bfbc9aSGreg Clayton error.SetErrorString("always unsafe to load or unload shared libraries in the darwin kernel"); 922d4bfbc9aSGreg Clayton return error; 923d4bfbc9aSGreg Clayton } 924d4bfbc9aSGreg Clayton 925d4bfbc9aSGreg Clayton void 926944b828aSGreg Clayton DynamicLoaderDarwinKernel::Initialize() 927d4bfbc9aSGreg Clayton { 928d4bfbc9aSGreg Clayton PluginManager::RegisterPlugin (GetPluginNameStatic(), 929d4bfbc9aSGreg Clayton GetPluginDescriptionStatic(), 930e8cd0c98SGreg Clayton CreateInstance, 931e8cd0c98SGreg Clayton DebuggerInitialize); 932d4bfbc9aSGreg Clayton } 933d4bfbc9aSGreg Clayton 934d4bfbc9aSGreg Clayton void 935944b828aSGreg Clayton DynamicLoaderDarwinKernel::Terminate() 936d4bfbc9aSGreg Clayton { 937d4bfbc9aSGreg Clayton PluginManager::UnregisterPlugin (CreateInstance); 938d4bfbc9aSGreg Clayton } 939d4bfbc9aSGreg Clayton 940e8cd0c98SGreg Clayton void 941e8cd0c98SGreg Clayton DynamicLoaderDarwinKernel::DebuggerInitialize (lldb_private::Debugger &debugger) 942e8cd0c98SGreg Clayton { 943e8cd0c98SGreg Clayton if (!PluginManager::GetSettingForDynamicLoaderPlugin (debugger, DynamicLoaderDarwinKernelProperties::GetSettingName())) 944e8cd0c98SGreg Clayton { 945e8cd0c98SGreg Clayton const bool is_global_setting = true; 946e8cd0c98SGreg Clayton PluginManager::CreateSettingForDynamicLoaderPlugin (debugger, 947e8cd0c98SGreg Clayton GetGlobalProperties()->GetValueProperties(), 948e8cd0c98SGreg Clayton ConstString ("Properties for the DynamicLoaderDarwinKernel plug-in."), 949e8cd0c98SGreg Clayton is_global_setting); 950e8cd0c98SGreg Clayton } 951e8cd0c98SGreg Clayton } 952d4bfbc9aSGreg Clayton 953d4bfbc9aSGreg Clayton const char * 954944b828aSGreg Clayton DynamicLoaderDarwinKernel::GetPluginNameStatic() 955d4bfbc9aSGreg Clayton { 956468ea4ebSGreg Clayton return "dynamic-loader.darwin-kernel"; 957d4bfbc9aSGreg Clayton } 958d4bfbc9aSGreg Clayton 959d4bfbc9aSGreg Clayton const char * 960944b828aSGreg Clayton DynamicLoaderDarwinKernel::GetPluginDescriptionStatic() 961d4bfbc9aSGreg Clayton { 962d4bfbc9aSGreg Clayton return "Dynamic loader plug-in that watches for shared library loads/unloads in the MacOSX kernel."; 963d4bfbc9aSGreg Clayton } 964d4bfbc9aSGreg Clayton 965d4bfbc9aSGreg Clayton 966d4bfbc9aSGreg Clayton //------------------------------------------------------------------ 967d4bfbc9aSGreg Clayton // PluginInterface protocol 968d4bfbc9aSGreg Clayton //------------------------------------------------------------------ 969d4bfbc9aSGreg Clayton const char * 970944b828aSGreg Clayton DynamicLoaderDarwinKernel::GetPluginName() 971d4bfbc9aSGreg Clayton { 972944b828aSGreg Clayton return "DynamicLoaderDarwinKernel"; 973d4bfbc9aSGreg Clayton } 974d4bfbc9aSGreg Clayton 975d4bfbc9aSGreg Clayton const char * 976944b828aSGreg Clayton DynamicLoaderDarwinKernel::GetShortPluginName() 977d4bfbc9aSGreg Clayton { 978d4bfbc9aSGreg Clayton return GetPluginNameStatic(); 979d4bfbc9aSGreg Clayton } 980d4bfbc9aSGreg Clayton 981d4bfbc9aSGreg Clayton uint32_t 982944b828aSGreg Clayton DynamicLoaderDarwinKernel::GetPluginVersion() 983d4bfbc9aSGreg Clayton { 984d4bfbc9aSGreg Clayton return 1; 985d4bfbc9aSGreg Clayton } 986d4bfbc9aSGreg Clayton 9871f746071SGreg Clayton lldb::ByteOrder 9881f746071SGreg Clayton DynamicLoaderDarwinKernel::GetByteOrderFromMagic (uint32_t magic) 9891f746071SGreg Clayton { 9901f746071SGreg Clayton switch (magic) 9911f746071SGreg Clayton { 9921f746071SGreg Clayton case llvm::MachO::HeaderMagic32: 9931f746071SGreg Clayton case llvm::MachO::HeaderMagic64: 9941f746071SGreg Clayton return lldb::endian::InlHostByteOrder(); 9951f746071SGreg Clayton 9961f746071SGreg Clayton case llvm::MachO::HeaderMagic32Swapped: 9971f746071SGreg Clayton case llvm::MachO::HeaderMagic64Swapped: 9981f746071SGreg Clayton if (lldb::endian::InlHostByteOrder() == lldb::eByteOrderBig) 9991f746071SGreg Clayton return lldb::eByteOrderLittle; 10001f746071SGreg Clayton else 10011f746071SGreg Clayton return lldb::eByteOrderBig; 10021f746071SGreg Clayton 10031f746071SGreg Clayton default: 10041f746071SGreg Clayton break; 10051f746071SGreg Clayton } 10061f746071SGreg Clayton return lldb::eByteOrderInvalid; 10071f746071SGreg Clayton } 10081f746071SGreg Clayton 1009