| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1 | //===-- DynamicLoaderDarwinKernel.cpp -----------------------------*- C++ -*-===// | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
|  | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
|  | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 |  | 
| Daniel Malea | 93a6430 | 2012-12-05 00:20:57 +0000 | [diff] [blame] | 10 | #include "lldb/lldb-python.h" | 
|  | 11 |  | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 12 | #include "lldb/Breakpoint/StoppointCallbackContext.h" | 
|  | 13 | #include "lldb/Core/DataBuffer.h" | 
|  | 14 | #include "lldb/Core/DataBufferHeap.h" | 
| Greg Clayton | 07e66e3 | 2011-07-20 03:41:06 +0000 | [diff] [blame] | 15 | #include "lldb/Core/Debugger.h" | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 16 | #include "lldb/Core/Log.h" | 
|  | 17 | #include "lldb/Core/Module.h" | 
| Greg Clayton | 1f74607 | 2012-08-29 21:13:06 +0000 | [diff] [blame] | 18 | #include "lldb/Core/ModuleSpec.h" | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 19 | #include "lldb/Core/PluginManager.h" | 
| Greg Clayton | 1f74607 | 2012-08-29 21:13:06 +0000 | [diff] [blame] | 20 | #include "lldb/Core/Section.h" | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 21 | #include "lldb/Core/State.h" | 
| Jason Molenda | 68b3607 | 2012-10-02 03:49:41 +0000 | [diff] [blame] | 22 | #include "lldb/Host/Symbols.h" | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 23 | #include "lldb/Symbol/ObjectFile.h" | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 24 | #include "lldb/Target/RegisterContext.h" | 
| Jason Molenda | 68b3607 | 2012-10-02 03:49:41 +0000 | [diff] [blame] | 25 | #include "lldb/Target/StackFrame.h" | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 26 | #include "lldb/Target/Target.h" | 
|  | 27 | #include "lldb/Target/Thread.h" | 
|  | 28 | #include "lldb/Target/ThreadPlanRunToAddress.h" | 
| Jason Molenda | 68b3607 | 2012-10-02 03:49:41 +0000 | [diff] [blame] | 29 |  | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 30 |  | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 31 | #include "DynamicLoaderDarwinKernel.h" | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 32 |  | 
|  | 33 | //#define ENABLE_DEBUG_PRINTF // COMMENT THIS LINE OUT PRIOR TO CHECKIN | 
|  | 34 | #ifdef ENABLE_DEBUG_PRINTF | 
|  | 35 | #include <stdio.h> | 
|  | 36 | #define DEBUG_PRINTF(fmt, ...) printf(fmt, ## __VA_ARGS__) | 
|  | 37 | #else | 
|  | 38 | #define DEBUG_PRINTF(fmt, ...) | 
|  | 39 | #endif | 
|  | 40 |  | 
|  | 41 | using namespace lldb; | 
|  | 42 | using namespace lldb_private; | 
|  | 43 |  | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 44 | // Progressively greater amounts of scanning we will allow | 
|  | 45 | // For some targets very early in startup, we can't do any random reads of memory or we can crash the device | 
|  | 46 | // so a setting is needed that can completely disable the KASLR scans. | 
|  | 47 |  | 
|  | 48 | enum KASLRScanType | 
|  | 49 | { | 
|  | 50 | eKASLRScanNone = 0,         // No reading into the inferior at all | 
|  | 51 | eKASLRScanLowgloAddresses,  // Check one word of memory for a possible kernel addr, then see if a kernel is there | 
| Jason Molenda | 6396922 | 2013-01-30 04:48:16 +0000 | [diff] [blame] | 52 | eKASLRScanNearPC,           // Scan backwards from the current $pc looking for kernel; checking at 96 locations total | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 53 | eKASLRScanExhaustiveScan    // Scan through the entire possible kernel address range looking for a kernel | 
|  | 54 | }; | 
|  | 55 |  | 
|  | 56 | OptionEnumValueElement | 
|  | 57 | g_kaslr_kernel_scan_enum_values[] = | 
|  | 58 | { | 
|  | 59 | { eKASLRScanNone,            "none",            "Do not read memory looking for a Darwin kernel when attaching." }, | 
|  | 60 | { eKASLRScanLowgloAddresses, "basic",           "Check for the Darwin kernel's load addr in the lowglo page (boot-args=debug) only." }, | 
|  | 61 | { eKASLRScanNearPC,          "fast-scan",       "Scan near the pc value on attach to find the Darwin kernel's load address."}, | 
|  | 62 | { eKASLRScanExhaustiveScan,  "exhaustive-scan", "Scan through the entire potential address range of Darwin kernel (only on 32-bit targets)."}, | 
|  | 63 | { 0, NULL, NULL } | 
|  | 64 | }; | 
|  | 65 |  | 
| Greg Clayton | e8cd0c9 | 2012-10-19 18:02:49 +0000 | [diff] [blame] | 66 | static PropertyDefinition | 
|  | 67 | g_properties[] = | 
|  | 68 | { | 
| Greg Clayton | 66763ee | 2012-10-19 20:53:18 +0000 | [diff] [blame] | 69 | { "load-kexts" , OptionValue::eTypeBoolean, true, true, NULL, NULL, "Automatically loads kext images when attaching to a kernel." }, | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 70 | { "scan-type",   OptionValue::eTypeEnum,    true, eKASLRScanNearPC, NULL, g_kaslr_kernel_scan_enum_values, "Control how many reads lldb will make while searching for a Darwin kernel on attach." }, | 
| Greg Clayton | 66763ee | 2012-10-19 20:53:18 +0000 | [diff] [blame] | 71 | {  NULL        , OptionValue::eTypeInvalid, false, 0  , NULL, NULL, NULL  } | 
| Greg Clayton | e8cd0c9 | 2012-10-19 18:02:49 +0000 | [diff] [blame] | 72 | }; | 
|  | 73 |  | 
|  | 74 | enum { | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 75 | ePropertyLoadKexts, | 
|  | 76 | ePropertyScanType | 
| Greg Clayton | e8cd0c9 | 2012-10-19 18:02:49 +0000 | [diff] [blame] | 77 | }; | 
|  | 78 |  | 
|  | 79 | class DynamicLoaderDarwinKernelProperties : public Properties | 
|  | 80 | { | 
|  | 81 | public: | 
|  | 82 |  | 
|  | 83 | static ConstString & | 
|  | 84 | GetSettingName () | 
|  | 85 | { | 
| Greg Clayton | 468ea4e | 2012-10-19 18:14:47 +0000 | [diff] [blame] | 86 | static ConstString g_setting_name("darwin-kernel"); | 
| Greg Clayton | e8cd0c9 | 2012-10-19 18:02:49 +0000 | [diff] [blame] | 87 | return g_setting_name; | 
|  | 88 | } | 
|  | 89 |  | 
|  | 90 | DynamicLoaderDarwinKernelProperties() : | 
|  | 91 | Properties () | 
|  | 92 | { | 
|  | 93 | m_collection_sp.reset (new OptionValueProperties(GetSettingName())); | 
|  | 94 | m_collection_sp->Initialize(g_properties); | 
|  | 95 | } | 
|  | 96 |  | 
|  | 97 | virtual | 
|  | 98 | ~DynamicLoaderDarwinKernelProperties() | 
|  | 99 | { | 
|  | 100 | } | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 101 |  | 
| Greg Clayton | e8cd0c9 | 2012-10-19 18:02:49 +0000 | [diff] [blame] | 102 | bool | 
| Greg Clayton | 66763ee | 2012-10-19 20:53:18 +0000 | [diff] [blame] | 103 | GetLoadKexts() const | 
| Greg Clayton | e8cd0c9 | 2012-10-19 18:02:49 +0000 | [diff] [blame] | 104 | { | 
| Greg Clayton | 66763ee | 2012-10-19 20:53:18 +0000 | [diff] [blame] | 105 | const uint32_t idx = ePropertyLoadKexts; | 
| Greg Clayton | e8cd0c9 | 2012-10-19 18:02:49 +0000 | [diff] [blame] | 106 | return m_collection_sp->GetPropertyAtIndexAsBoolean (NULL, idx, g_properties[idx].default_uint_value != 0); | 
|  | 107 | } | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 108 |  | 
|  | 109 | KASLRScanType | 
|  | 110 | GetScanType() const | 
|  | 111 | { | 
|  | 112 | const uint32_t idx = ePropertyScanType; | 
| Jason Molenda | 6396922 | 2013-01-30 04:48:16 +0000 | [diff] [blame] | 113 | return (KASLRScanType) m_collection_sp->GetPropertyAtIndexAsEnumeration (NULL, idx, g_properties[idx].default_uint_value); | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 114 | } | 
|  | 115 |  | 
|  | 116 |  | 
| Greg Clayton | e8cd0c9 | 2012-10-19 18:02:49 +0000 | [diff] [blame] | 117 | }; | 
|  | 118 |  | 
|  | 119 | typedef STD_SHARED_PTR(DynamicLoaderDarwinKernelProperties) DynamicLoaderDarwinKernelPropertiesSP; | 
|  | 120 |  | 
|  | 121 | static const DynamicLoaderDarwinKernelPropertiesSP & | 
|  | 122 | GetGlobalProperties() | 
|  | 123 | { | 
|  | 124 | static DynamicLoaderDarwinKernelPropertiesSP g_settings_sp; | 
|  | 125 | if (!g_settings_sp) | 
|  | 126 | g_settings_sp.reset (new DynamicLoaderDarwinKernelProperties ()); | 
|  | 127 | return g_settings_sp; | 
|  | 128 | } | 
|  | 129 |  | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 130 | //---------------------------------------------------------------------- | 
|  | 131 | // Create an instance of this class. This function is filled into | 
|  | 132 | // the plugin info class that gets handed out by the plugin factory and | 
|  | 133 | // allows the lldb to instantiate an instance of this class. | 
|  | 134 | //---------------------------------------------------------------------- | 
|  | 135 | DynamicLoader * | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 136 | DynamicLoaderDarwinKernel::CreateInstance (Process* process, bool force) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 137 | { | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 138 | if (!force) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 139 | { | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 140 | // If the user provided an executable binary and it is not a kernel, | 
|  | 141 | // this plugin should not create an instance. | 
| Greg Clayton | aa149cb | 2011-08-11 02:48:45 +0000 | [diff] [blame] | 142 | Module* exe_module = process->GetTarget().GetExecutableModulePointer(); | 
| Greg Clayton | df0b7d5 | 2011-07-08 04:11:42 +0000 | [diff] [blame] | 143 | if (exe_module) | 
|  | 144 | { | 
|  | 145 | ObjectFile *object_file = exe_module->GetObjectFile(); | 
|  | 146 | if (object_file) | 
|  | 147 | { | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 148 | if (object_file->GetStrata() != ObjectFile::eStrataKernel) | 
|  | 149 | { | 
|  | 150 | return NULL; | 
|  | 151 | } | 
| Greg Clayton | df0b7d5 | 2011-07-08 04:11:42 +0000 | [diff] [blame] | 152 | } | 
|  | 153 | } | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 154 |  | 
|  | 155 | // If the target's architecture does not look like an Apple environment, | 
|  | 156 | // this plugin should not create an instance. | 
|  | 157 | const llvm::Triple &triple_ref = process->GetTarget().GetArchitecture().GetTriple(); | 
|  | 158 | switch (triple_ref.getOS()) | 
| Greg Clayton | df0b7d5 | 2011-07-08 04:11:42 +0000 | [diff] [blame] | 159 | { | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 160 | case llvm::Triple::Darwin: | 
|  | 161 | case llvm::Triple::MacOSX: | 
|  | 162 | case llvm::Triple::IOS: | 
|  | 163 | if (triple_ref.getVendor() != llvm::Triple::Apple) | 
|  | 164 | { | 
|  | 165 | return NULL; | 
|  | 166 | } | 
|  | 167 | break; | 
|  | 168 | // If we have triple like armv7-unknown-unknown, we should try looking for a Darwin kernel. | 
|  | 169 | case llvm::Triple::UnknownOS: | 
|  | 170 | break; | 
|  | 171 | default: | 
|  | 172 | return NULL; | 
|  | 173 | break; | 
|  | 174 | } | 
|  | 175 | } | 
|  | 176 |  | 
|  | 177 | // At this point if there is an ExecutableModule, it is a kernel and the Target is some variant of an Apple system. | 
|  | 178 | // If the Process hasn't provided the kernel load address, we need to look around in memory to find it. | 
|  | 179 |  | 
|  | 180 | addr_t kernel_load_address = process->GetImageInfoAddress(); | 
|  | 181 | if (kernel_load_address == LLDB_INVALID_ADDRESS) | 
|  | 182 | { | 
|  | 183 | kernel_load_address = SearchForKernelAtSameLoadAddr (process); | 
|  | 184 | if (kernel_load_address == LLDB_INVALID_ADDRESS) | 
|  | 185 | { | 
|  | 186 | kernel_load_address = SearchForKernelWithDebugHints (process); | 
|  | 187 | if (kernel_load_address == LLDB_INVALID_ADDRESS) | 
| Greg Clayton | 7051231 | 2012-05-08 01:45:38 +0000 | [diff] [blame] | 188 | { | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 189 | kernel_load_address = SearchForKernelNearPC (process); | 
|  | 190 | if (kernel_load_address == LLDB_INVALID_ADDRESS) | 
|  | 191 | { | 
|  | 192 | kernel_load_address = SearchForKernelViaExhaustiveSearch (process); | 
|  | 193 | } | 
| Greg Clayton | 7051231 | 2012-05-08 01:45:38 +0000 | [diff] [blame] | 194 | } | 
| Greg Clayton | df0b7d5 | 2011-07-08 04:11:42 +0000 | [diff] [blame] | 195 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 196 | } | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 197 |  | 
|  | 198 | if (kernel_load_address != LLDB_INVALID_ADDRESS) | 
| Sean Callanan | 9053945 | 2011-09-20 23:01:51 +0000 | [diff] [blame] | 199 | { | 
|  | 200 | process->SetCanJIT(false); | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 201 | return new DynamicLoaderDarwinKernel (process, kernel_load_address); | 
| Sean Callanan | 9053945 | 2011-09-20 23:01:51 +0000 | [diff] [blame] | 202 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 203 | return NULL; | 
|  | 204 | } | 
|  | 205 |  | 
|  | 206 | //---------------------------------------------------------------------- | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 207 | // Check if the kernel binary is loaded in memory without a slide. | 
|  | 208 | // First verify that the ExecutableModule is a kernel before we proceed. | 
|  | 209 | // Returns the address of the kernel if one was found, else LLDB_INVALID_ADDRESS. | 
|  | 210 | //---------------------------------------------------------------------- | 
|  | 211 | lldb::addr_t | 
|  | 212 | DynamicLoaderDarwinKernel::SearchForKernelAtSameLoadAddr (Process *process) | 
|  | 213 | { | 
|  | 214 | Module *exe_module = process->GetTarget().GetExecutableModulePointer(); | 
|  | 215 | if (exe_module == NULL) | 
|  | 216 | return LLDB_INVALID_ADDRESS; | 
|  | 217 |  | 
|  | 218 | ObjectFile *exe_objfile = exe_module->GetObjectFile(); | 
|  | 219 | if (exe_objfile == NULL) | 
|  | 220 | return LLDB_INVALID_ADDRESS; | 
|  | 221 |  | 
|  | 222 | if (exe_objfile->GetType() != ObjectFile::eTypeExecutable || exe_objfile->GetStrata() != ObjectFile::eStrataKernel) | 
|  | 223 | return LLDB_INVALID_ADDRESS; | 
|  | 224 |  | 
|  | 225 | if (!exe_objfile->GetHeaderAddress().IsValid()) | 
|  | 226 | return LLDB_INVALID_ADDRESS; | 
|  | 227 |  | 
|  | 228 | if (CheckForKernelImageAtAddress (exe_objfile->GetHeaderAddress().GetFileAddress(), process) == exe_module->GetUUID()) | 
|  | 229 | return exe_objfile->GetHeaderAddress().GetFileAddress(); | 
|  | 230 |  | 
|  | 231 | return LLDB_INVALID_ADDRESS; | 
|  | 232 | } | 
|  | 233 |  | 
|  | 234 | //---------------------------------------------------------------------- | 
|  | 235 | // If the debug flag is included in the boot-args nvram setting, the kernel's load address | 
|  | 236 | // will be noted in the lowglo page at a fixed address | 
|  | 237 | // Returns the address of the kernel if one was found, else LLDB_INVALID_ADDRESS. | 
|  | 238 | //---------------------------------------------------------------------- | 
|  | 239 | lldb::addr_t | 
|  | 240 | DynamicLoaderDarwinKernel::SearchForKernelWithDebugHints (Process *process) | 
|  | 241 | { | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 242 | if (GetGlobalProperties()->GetScanType() == eKASLRScanNone) | 
|  | 243 | return LLDB_INVALID_ADDRESS; | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 244 |  | 
|  | 245 | Error read_err; | 
|  | 246 | addr_t addr = LLDB_INVALID_ADDRESS; | 
|  | 247 | if (process->GetTarget().GetArchitecture().GetAddressByteSize() == 8) | 
|  | 248 | { | 
|  | 249 | addr = process->ReadUnsignedIntegerFromMemory (0xffffff8000002010ULL, 8, LLDB_INVALID_ADDRESS, read_err); | 
|  | 250 | } | 
|  | 251 | else | 
|  | 252 | { | 
|  | 253 | addr = process->ReadUnsignedIntegerFromMemory (0xffff0110, 4, LLDB_INVALID_ADDRESS, read_err); | 
|  | 254 | } | 
|  | 255 |  | 
|  | 256 | if (addr == 0) | 
|  | 257 | addr = LLDB_INVALID_ADDRESS; | 
|  | 258 |  | 
|  | 259 | if (addr != LLDB_INVALID_ADDRESS) | 
|  | 260 | { | 
|  | 261 | if (CheckForKernelImageAtAddress (addr, process).IsValid()) | 
|  | 262 | return addr; | 
|  | 263 | } | 
|  | 264 |  | 
|  | 265 | return LLDB_INVALID_ADDRESS; | 
|  | 266 | } | 
|  | 267 |  | 
|  | 268 | //---------------------------------------------------------------------- | 
|  | 269 | // If the kernel is currently executing when lldb attaches, and we don't have | 
|  | 270 | // a better way of finding the kernel's load address, try searching backwards | 
|  | 271 | // from the current pc value looking for the kernel's Mach header in memory. | 
|  | 272 | // Returns the address of the kernel if one was found, else LLDB_INVALID_ADDRESS. | 
|  | 273 | //---------------------------------------------------------------------- | 
|  | 274 | lldb::addr_t | 
|  | 275 | DynamicLoaderDarwinKernel::SearchForKernelNearPC (Process *process) | 
|  | 276 | { | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 277 | if (GetGlobalProperties()->GetScanType() == eKASLRScanNone | 
|  | 278 | || GetGlobalProperties()->GetScanType() == eKASLRScanLowgloAddresses) | 
|  | 279 | { | 
|  | 280 | return LLDB_INVALID_ADDRESS; | 
|  | 281 | } | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 282 |  | 
|  | 283 | ThreadSP thread = process->GetThreadList().GetSelectedThread (); | 
|  | 284 | if (thread.get() == NULL) | 
|  | 285 | return LLDB_INVALID_ADDRESS; | 
|  | 286 | addr_t pc = thread->GetRegisterContext ()->GetPC(LLDB_INVALID_ADDRESS); | 
|  | 287 |  | 
|  | 288 | if (pc == LLDB_INVALID_ADDRESS) | 
|  | 289 | return LLDB_INVALID_ADDRESS; | 
|  | 290 |  | 
|  | 291 | addr_t kernel_range_low, kernel_range_high; | 
|  | 292 | if (process->GetTarget().GetArchitecture().GetAddressByteSize() == 8) | 
|  | 293 | { | 
|  | 294 | kernel_range_low = 1ULL << 63; | 
|  | 295 | kernel_range_high = UINT64_MAX; | 
|  | 296 | } | 
|  | 297 | else | 
|  | 298 | { | 
|  | 299 | kernel_range_low = 1ULL << 31; | 
|  | 300 | kernel_range_high = UINT32_MAX; | 
|  | 301 | } | 
|  | 302 |  | 
|  | 303 | // Outside the normal kernel address range, this is probably userland code running right now | 
|  | 304 | if (pc < kernel_range_low) | 
|  | 305 | LLDB_INVALID_ADDRESS; | 
|  | 306 |  | 
|  | 307 | // The kernel will load at at one megabyte boundary (0x100000), or at that boundary plus | 
|  | 308 | // an offset of one page (0x1000) or two, depending on the device. | 
|  | 309 |  | 
|  | 310 | // Round the current pc down to the nearest one megabyte boundary - the place where we will start searching. | 
|  | 311 | addr_t addr = pc & ~0xfffff; | 
|  | 312 |  | 
|  | 313 | int i = 0; | 
|  | 314 | while (i < 32 && pc >= kernel_range_low) | 
|  | 315 | { | 
|  | 316 | if (CheckForKernelImageAtAddress (addr, process).IsValid()) | 
|  | 317 | return addr; | 
|  | 318 | if (CheckForKernelImageAtAddress (addr + 0x1000, process).IsValid()) | 
|  | 319 | return addr + 0x1000; | 
|  | 320 | if (CheckForKernelImageAtAddress (addr + 0x2000, process).IsValid()) | 
|  | 321 | return addr + 0x2000; | 
|  | 322 | i++; | 
|  | 323 | addr -= 0x100000; | 
|  | 324 | } | 
|  | 325 |  | 
|  | 326 | return LLDB_INVALID_ADDRESS; | 
|  | 327 | } | 
|  | 328 |  | 
|  | 329 | //---------------------------------------------------------------------- | 
|  | 330 | // Scan through the valid address range for a kernel binary. | 
|  | 331 | // This is uselessly slow in 64-bit environments so we don't even try it. | 
|  | 332 | // This scan is not enabled by default even for 32-bit targets. | 
|  | 333 | // Returns the address of the kernel if one was found, else LLDB_INVALID_ADDRESS. | 
|  | 334 | //---------------------------------------------------------------------- | 
|  | 335 | lldb::addr_t | 
|  | 336 | DynamicLoaderDarwinKernel::SearchForKernelViaExhaustiveSearch (Process *process) | 
|  | 337 | { | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 338 | if (GetGlobalProperties()->GetScanType() != eKASLRScanExhaustiveScan) | 
|  | 339 | { | 
|  | 340 | return LLDB_INVALID_ADDRESS; | 
|  | 341 | } | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 342 |  | 
|  | 343 | addr_t kernel_range_low, kernel_range_high; | 
|  | 344 | if (process->GetTarget().GetArchitecture().GetAddressByteSize() == 8) | 
|  | 345 | { | 
|  | 346 | kernel_range_low = 1ULL << 63; | 
|  | 347 | kernel_range_high = UINT64_MAX; | 
|  | 348 | } | 
|  | 349 | else | 
|  | 350 | { | 
|  | 351 | kernel_range_low = 1ULL << 31; | 
|  | 352 | kernel_range_high = UINT32_MAX; | 
|  | 353 | } | 
|  | 354 |  | 
|  | 355 | // Stepping through memory at one-megabyte resolution looking for a kernel | 
|  | 356 | // rarely works (fast enough) with a 64-bit address space -- for now, let's | 
|  | 357 | // not even bother.  We may be attaching to something which *isn't* a kernel | 
|  | 358 | // and we don't want to spin for minutes on-end looking for a kernel. | 
|  | 359 | if (process->GetTarget().GetArchitecture().GetAddressByteSize() == 8) | 
|  | 360 | return LLDB_INVALID_ADDRESS; | 
|  | 361 |  | 
|  | 362 | addr_t addr = kernel_range_low; | 
|  | 363 |  | 
|  | 364 | while (addr >= kernel_range_low && addr < kernel_range_high) | 
|  | 365 | { | 
|  | 366 | if (CheckForKernelImageAtAddress (addr, process).IsValid()) | 
|  | 367 | return addr; | 
|  | 368 | if (CheckForKernelImageAtAddress (addr + 0x1000, process).IsValid()) | 
|  | 369 | return addr + 0x1000; | 
|  | 370 | if (CheckForKernelImageAtAddress (addr + 0x2000, process).IsValid()) | 
|  | 371 | return addr + 0x2000; | 
|  | 372 | addr += 0x100000; | 
|  | 373 | } | 
|  | 374 | return LLDB_INVALID_ADDRESS; | 
|  | 375 | } | 
|  | 376 |  | 
|  | 377 | //---------------------------------------------------------------------- | 
|  | 378 | // Given an address in memory, look to see if there is a kernel image at that | 
|  | 379 | // address. | 
|  | 380 | // Returns a UUID; if a kernel was not found at that address, UUID.IsValid() will be false. | 
|  | 381 | //---------------------------------------------------------------------- | 
|  | 382 | lldb_private::UUID | 
|  | 383 | DynamicLoaderDarwinKernel::CheckForKernelImageAtAddress (lldb::addr_t addr, Process *process) | 
|  | 384 | { | 
|  | 385 | if (addr == LLDB_INVALID_ADDRESS) | 
|  | 386 | return UUID(); | 
|  | 387 |  | 
|  | 388 | // First try a quick test -- read the first 4 bytes and see if there is a valid Mach-O magic field there | 
|  | 389 | // (the first field of the mach_header/mach_header_64 struct). | 
|  | 390 |  | 
|  | 391 | Error read_error; | 
|  | 392 | uint64_t result = process->ReadUnsignedIntegerFromMemory (addr, 4, LLDB_INVALID_ADDRESS, read_error); | 
|  | 393 | if (result != llvm::MachO::HeaderMagic64 | 
|  | 394 | && result != llvm::MachO::HeaderMagic32 | 
|  | 395 | && result != llvm::MachO::HeaderMagic32Swapped | 
|  | 396 | && result != llvm::MachO::HeaderMagic64Swapped) | 
|  | 397 | { | 
|  | 398 | return UUID(); | 
|  | 399 | } | 
|  | 400 |  | 
|  | 401 | // Read the mach header and see whether it looks like a kernel | 
|  | 402 | llvm::MachO::mach_header header; | 
|  | 403 | if (process->DoReadMemory (addr, &header, sizeof(header), read_error) != sizeof(header)) | 
|  | 404 | return UUID(); | 
|  | 405 |  | 
|  | 406 | if (header.magic == llvm::MachO::HeaderMagic32Swapped || | 
|  | 407 | header.magic == llvm::MachO::HeaderMagic64Swapped) | 
|  | 408 | { | 
|  | 409 | header.magic        = llvm::ByteSwap_32(header.magic); | 
|  | 410 | header.cputype      = llvm::ByteSwap_32(header.cputype); | 
|  | 411 | header.cpusubtype   = llvm::ByteSwap_32(header.cpusubtype); | 
|  | 412 | header.filetype     = llvm::ByteSwap_32(header.filetype); | 
|  | 413 | header.ncmds        = llvm::ByteSwap_32(header.ncmds); | 
|  | 414 | header.sizeofcmds   = llvm::ByteSwap_32(header.sizeofcmds); | 
|  | 415 | header.flags        = llvm::ByteSwap_32(header.flags); | 
|  | 416 | } | 
|  | 417 |  | 
|  | 418 | // A kernel is an executable which does not have the dynamic link object flag set. | 
|  | 419 | if (header.filetype == llvm::MachO::HeaderFileTypeExecutable | 
|  | 420 | && (header.flags & llvm::MachO::HeaderFlagBitIsDynamicLinkObject) == 0) | 
|  | 421 | { | 
|  | 422 | // Create a full module to get the UUID | 
| Greg Clayton | 39f7ee8 | 2013-02-01 21:38:35 +0000 | [diff] [blame] | 423 | ModuleSP memory_module_sp = process->ReadModuleFromMemory (FileSpec ("temp_mach_kernel", false), addr); | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 424 | if (!memory_module_sp.get()) | 
|  | 425 | return UUID(); | 
|  | 426 |  | 
|  | 427 | ObjectFile *exe_objfile = memory_module_sp->GetObjectFile(); | 
|  | 428 | if (exe_objfile == NULL) | 
|  | 429 | return UUID(); | 
|  | 430 |  | 
|  | 431 | if (exe_objfile->GetType() == ObjectFile::eTypeExecutable && exe_objfile->GetStrata() == ObjectFile::eStrataKernel) | 
|  | 432 | { | 
|  | 433 | return memory_module_sp->GetUUID(); | 
|  | 434 | } | 
|  | 435 | } | 
|  | 436 |  | 
|  | 437 | return UUID(); | 
|  | 438 | } | 
|  | 439 |  | 
|  | 440 | //---------------------------------------------------------------------- | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 441 | // Constructor | 
|  | 442 | //---------------------------------------------------------------------- | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 443 | DynamicLoaderDarwinKernel::DynamicLoaderDarwinKernel (Process* process, lldb::addr_t kernel_addr) : | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 444 | DynamicLoader(process), | 
| Jason Molenda | 6ba6d3d | 2013-01-30 04:39:32 +0000 | [diff] [blame] | 445 | m_kernel_load_address (kernel_addr), | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 446 | m_kernel(), | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 447 | m_kext_summary_header_ptr_addr (), | 
| Greg Clayton | 0d9fc76 | 2011-07-08 03:21:57 +0000 | [diff] [blame] | 448 | m_kext_summary_header_addr (), | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 449 | m_kext_summary_header (), | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 450 | m_known_kexts (), | 
| Daniel Dunbar | a08823f | 2011-10-31 22:50:49 +0000 | [diff] [blame] | 451 | m_mutex(Mutex::eMutexTypeRecursive), | 
|  | 452 | m_break_id (LLDB_INVALID_BREAK_ID) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 453 | { | 
|  | 454 | } | 
|  | 455 |  | 
|  | 456 | //---------------------------------------------------------------------- | 
|  | 457 | // Destructor | 
|  | 458 | //---------------------------------------------------------------------- | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 459 | DynamicLoaderDarwinKernel::~DynamicLoaderDarwinKernel() | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 460 | { | 
|  | 461 | Clear(true); | 
|  | 462 | } | 
|  | 463 |  | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 464 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 465 | DynamicLoaderDarwinKernel::UpdateIfNeeded() | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 466 | { | 
|  | 467 | LoadKernelModuleIfNeeded(); | 
|  | 468 | SetNotificationBreakpointIfNeeded (); | 
|  | 469 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 470 | //------------------------------------------------------------------ | 
|  | 471 | /// Called after attaching a process. | 
|  | 472 | /// | 
|  | 473 | /// Allow DynamicLoader plug-ins to execute some code after | 
|  | 474 | /// attaching to a process. | 
|  | 475 | //------------------------------------------------------------------ | 
|  | 476 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 477 | DynamicLoaderDarwinKernel::DidAttach () | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 478 | { | 
|  | 479 | PrivateInitialize(m_process); | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 480 | UpdateIfNeeded(); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 481 | } | 
|  | 482 |  | 
|  | 483 | //------------------------------------------------------------------ | 
|  | 484 | /// Called after attaching a process. | 
|  | 485 | /// | 
|  | 486 | /// Allow DynamicLoader plug-ins to execute some code after | 
|  | 487 | /// attaching to a process. | 
|  | 488 | //------------------------------------------------------------------ | 
|  | 489 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 490 | DynamicLoaderDarwinKernel::DidLaunch () | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 491 | { | 
|  | 492 | PrivateInitialize(m_process); | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 493 | UpdateIfNeeded(); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 494 | } | 
|  | 495 |  | 
|  | 496 |  | 
|  | 497 | //---------------------------------------------------------------------- | 
|  | 498 | // Clear out the state of this class. | 
|  | 499 | //---------------------------------------------------------------------- | 
|  | 500 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 501 | DynamicLoaderDarwinKernel::Clear (bool clear_process) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 502 | { | 
|  | 503 | Mutex::Locker locker(m_mutex); | 
|  | 504 |  | 
|  | 505 | if (m_process->IsAlive() && LLDB_BREAK_ID_IS_VALID(m_break_id)) | 
|  | 506 | m_process->ClearBreakpointSiteByID(m_break_id); | 
|  | 507 |  | 
|  | 508 | if (clear_process) | 
|  | 509 | m_process = NULL; | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 510 | m_kernel.Clear(); | 
|  | 511 | m_known_kexts.clear(); | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 512 | m_kext_summary_header_ptr_addr.Clear(); | 
| Greg Clayton | 0d9fc76 | 2011-07-08 03:21:57 +0000 | [diff] [blame] | 513 | m_kext_summary_header_addr.Clear(); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 514 | m_break_id = LLDB_INVALID_BREAK_ID; | 
|  | 515 | } | 
|  | 516 |  | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 517 |  | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 518 | bool | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 519 | DynamicLoaderDarwinKernel::KextImageInfo::LoadImageAtFileAddress (Process *process) | 
| Greg Clayton | 2af282a | 2012-03-21 04:25:00 +0000 | [diff] [blame] | 520 | { | 
|  | 521 | if (IsLoaded()) | 
|  | 522 | return true; | 
|  | 523 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 524 | if (m_module_sp) | 
| Greg Clayton | 2af282a | 2012-03-21 04:25:00 +0000 | [diff] [blame] | 525 | { | 
|  | 526 | bool changed = false; | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 527 | if (m_module_sp->SetLoadAddress (process->GetTarget(), 0, changed)) | 
|  | 528 | m_load_process_stop_id = process->GetStopID(); | 
| Greg Clayton | 2af282a | 2012-03-21 04:25:00 +0000 | [diff] [blame] | 529 | } | 
|  | 530 | return false; | 
|  | 531 | } | 
|  | 532 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 533 | void | 
|  | 534 | DynamicLoaderDarwinKernel::KextImageInfo::SetModule (ModuleSP module_sp) | 
|  | 535 | { | 
|  | 536 | m_module_sp = module_sp; | 
|  | 537 | if (module_sp.get() && module_sp->GetObjectFile()) | 
|  | 538 | { | 
|  | 539 | if (module_sp->GetObjectFile()->GetType() == ObjectFile::eTypeExecutable | 
|  | 540 | && module_sp->GetObjectFile()->GetStrata() == ObjectFile::eStrataKernel) | 
|  | 541 | { | 
|  | 542 | m_kernel_image = true; | 
|  | 543 | } | 
|  | 544 | else | 
|  | 545 | { | 
|  | 546 | m_kernel_image = false; | 
|  | 547 | } | 
|  | 548 | } | 
|  | 549 | } | 
|  | 550 |  | 
|  | 551 | ModuleSP | 
|  | 552 | DynamicLoaderDarwinKernel::KextImageInfo::GetModule () | 
|  | 553 | { | 
|  | 554 | return m_module_sp; | 
|  | 555 | } | 
|  | 556 |  | 
|  | 557 | void | 
|  | 558 | DynamicLoaderDarwinKernel::KextImageInfo::SetLoadAddress (addr_t load_addr) | 
|  | 559 | { | 
|  | 560 | m_load_address = load_addr; | 
|  | 561 | } | 
|  | 562 |  | 
|  | 563 | addr_t | 
|  | 564 | DynamicLoaderDarwinKernel::KextImageInfo::GetLoadAddress () const | 
|  | 565 | { | 
|  | 566 | return m_load_address; | 
|  | 567 | } | 
|  | 568 |  | 
|  | 569 | uint64_t | 
|  | 570 | DynamicLoaderDarwinKernel::KextImageInfo::GetSize () const | 
|  | 571 | { | 
|  | 572 | return m_size; | 
|  | 573 | } | 
|  | 574 |  | 
|  | 575 | void | 
|  | 576 | DynamicLoaderDarwinKernel::KextImageInfo::SetSize (uint64_t size) | 
|  | 577 | { | 
|  | 578 | m_size = size; | 
|  | 579 | } | 
|  | 580 |  | 
|  | 581 | uint32_t | 
|  | 582 | DynamicLoaderDarwinKernel::KextImageInfo::GetProcessStopId () const | 
|  | 583 | { | 
|  | 584 | return m_load_process_stop_id; | 
|  | 585 | } | 
|  | 586 |  | 
|  | 587 | void | 
|  | 588 | DynamicLoaderDarwinKernel::KextImageInfo::SetProcessStopId (uint32_t stop_id) | 
|  | 589 | { | 
|  | 590 | m_load_process_stop_id = stop_id; | 
|  | 591 | } | 
|  | 592 |  | 
| Greg Clayton | 2af282a | 2012-03-21 04:25:00 +0000 | [diff] [blame] | 593 | bool | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 594 | DynamicLoaderDarwinKernel::KextImageInfo::operator== (const KextImageInfo &rhs) | 
|  | 595 | { | 
|  | 596 | if (m_uuid.IsValid() || rhs.GetUUID().IsValid()) | 
|  | 597 | { | 
|  | 598 | if (m_uuid == rhs.GetUUID()) | 
|  | 599 | { | 
|  | 600 | return true; | 
|  | 601 | } | 
|  | 602 | return false; | 
|  | 603 | } | 
|  | 604 |  | 
|  | 605 | if (m_name == rhs.GetName() && m_load_address == rhs.GetLoadAddress()) | 
|  | 606 | return true; | 
|  | 607 |  | 
|  | 608 | return false; | 
|  | 609 | } | 
|  | 610 |  | 
|  | 611 | void | 
|  | 612 | DynamicLoaderDarwinKernel::KextImageInfo::SetName (const char *name) | 
|  | 613 | { | 
|  | 614 | m_name = name; | 
|  | 615 | } | 
|  | 616 |  | 
|  | 617 | std::string | 
|  | 618 | DynamicLoaderDarwinKernel::KextImageInfo::GetName () const | 
|  | 619 | { | 
|  | 620 | return m_name; | 
|  | 621 | } | 
|  | 622 |  | 
|  | 623 | void | 
|  | 624 | DynamicLoaderDarwinKernel::KextImageInfo::SetUUID (const UUID &uuid) | 
|  | 625 | { | 
|  | 626 | m_uuid = uuid; | 
|  | 627 | } | 
|  | 628 |  | 
|  | 629 | UUID | 
|  | 630 | DynamicLoaderDarwinKernel::KextImageInfo::GetUUID () const | 
|  | 631 | { | 
|  | 632 | return m_uuid; | 
|  | 633 | } | 
|  | 634 |  | 
|  | 635 | // Given the m_load_address from the kext summaries, and a UUID, try to create an in-memory | 
|  | 636 | // Module at that address.  Require that the MemoryModule have a matching UUID and detect | 
|  | 637 | // if this MemoryModule is a kernel or a kext. | 
|  | 638 | // | 
|  | 639 | // Returns true if m_memory_module_sp is now set to a valid Module. | 
|  | 640 |  | 
|  | 641 | bool | 
|  | 642 | DynamicLoaderDarwinKernel::KextImageInfo::ReadMemoryModule (Process *process) | 
|  | 643 | { | 
|  | 644 | if (m_memory_module_sp.get() != NULL) | 
|  | 645 | return true; | 
|  | 646 | if (m_load_address == LLDB_INVALID_ADDRESS) | 
|  | 647 | return false; | 
|  | 648 |  | 
|  | 649 | FileSpec file_spec; | 
|  | 650 | file_spec.SetFile (m_name.c_str(), false); | 
|  | 651 |  | 
|  | 652 | ModuleSP memory_module_sp = process->ReadModuleFromMemory (file_spec, m_load_address); | 
|  | 653 |  | 
|  | 654 | if (memory_module_sp.get() == NULL) | 
|  | 655 | return false; | 
|  | 656 |  | 
|  | 657 | bool is_kernel = false; | 
|  | 658 | if (memory_module_sp->GetObjectFile()) | 
|  | 659 | { | 
|  | 660 | if (memory_module_sp->GetObjectFile()->GetType() == ObjectFile::eTypeExecutable | 
|  | 661 | && memory_module_sp->GetObjectFile()->GetStrata() == ObjectFile::eStrataKernel) | 
|  | 662 | { | 
|  | 663 | is_kernel = true; | 
|  | 664 | } | 
|  | 665 | else if (memory_module_sp->GetObjectFile()->GetType() == ObjectFile::eTypeSharedLibrary) | 
|  | 666 | { | 
|  | 667 | is_kernel = false; | 
|  | 668 | } | 
|  | 669 | } | 
|  | 670 |  | 
| Jason Molenda | 38e70d1 | 2013-02-27 03:07:49 +0000 | [diff] [blame] | 671 | // If this is a kext, and the kernel specified what UUID we should find at this | 
|  | 672 | // load address, require that the memory module have a matching UUID or something | 
|  | 673 | // has gone wrong and we should discard it. | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 674 | if (m_uuid.IsValid()) | 
|  | 675 | { | 
|  | 676 | if (m_uuid != memory_module_sp->GetUUID()) | 
|  | 677 | { | 
|  | 678 | return false; | 
|  | 679 | } | 
|  | 680 | } | 
|  | 681 |  | 
|  | 682 | // If the in-memory Module has a UUID, let's use that. | 
|  | 683 | if (!m_uuid.IsValid() && memory_module_sp->GetUUID().IsValid()) | 
|  | 684 | { | 
|  | 685 | m_uuid = memory_module_sp->GetUUID(); | 
|  | 686 | } | 
|  | 687 |  | 
|  | 688 | m_memory_module_sp = memory_module_sp; | 
|  | 689 | m_kernel_image = is_kernel; | 
|  | 690 | if (is_kernel) | 
|  | 691 | { | 
| Jason Molenda | 38e70d1 | 2013-02-27 03:07:49 +0000 | [diff] [blame] | 692 | if (memory_module_sp->GetArchitecture().IsValid()) | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 693 | { | 
|  | 694 | process->GetTarget().SetArchitecture(memory_module_sp->GetArchitecture()); | 
|  | 695 | } | 
| Jason Molenda | 38e70d1 | 2013-02-27 03:07:49 +0000 | [diff] [blame] | 696 | if (m_uuid.IsValid()) | 
|  | 697 | { | 
|  | 698 | Module* exe_module = process->GetTarget().GetExecutableModulePointer(); | 
|  | 699 | if (exe_module && exe_module->GetUUID().IsValid()) | 
|  | 700 | { | 
|  | 701 | if (m_uuid != exe_module->GetUUID()) | 
|  | 702 | { | 
|  | 703 | Stream *s = &process->GetTarget().GetDebugger().GetOutputStream(); | 
|  | 704 | if (s) | 
|  | 705 | { | 
|  | 706 | char memory_module_uuidbuf[64]; | 
|  | 707 | char exe_module_uuidbuf[64]; | 
|  | 708 | s->Printf ("warning: Host-side kernel file has Mach-O UUID of %s but remote kernel has a UUID of %s -- a mismatched kernel file will result in a poor debugger experience.\n", | 
|  | 709 | exe_module->GetUUID().GetAsCString(exe_module_uuidbuf, sizeof (exe_module_uuidbuf)), | 
|  | 710 | m_uuid.GetAsCString(memory_module_uuidbuf, sizeof (memory_module_uuidbuf))); | 
|  | 711 | s->Flush (); | 
|  | 712 | } | 
|  | 713 | } | 
|  | 714 | } | 
|  | 715 | } | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 716 | } | 
|  | 717 |  | 
|  | 718 | return true; | 
|  | 719 | } | 
|  | 720 |  | 
|  | 721 | bool | 
|  | 722 | DynamicLoaderDarwinKernel::KextImageInfo::IsKernel () const | 
|  | 723 | { | 
|  | 724 | return m_kernel_image == true; | 
|  | 725 | } | 
|  | 726 |  | 
|  | 727 | void | 
|  | 728 | DynamicLoaderDarwinKernel::KextImageInfo::SetIsKernel (bool is_kernel) | 
|  | 729 | { | 
|  | 730 | m_kernel_image = is_kernel; | 
|  | 731 | } | 
|  | 732 |  | 
|  | 733 | bool | 
|  | 734 | DynamicLoaderDarwinKernel::KextImageInfo::LoadImageUsingMemoryModule (Process *process) | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 735 | { | 
|  | 736 | if (IsLoaded()) | 
|  | 737 | return true; | 
|  | 738 |  | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 739 |  | 
|  | 740 | Target &target = process->GetTarget(); | 
| Jason Molenda | 87a04b2 | 2012-10-19 03:40:45 +0000 | [diff] [blame] | 741 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 742 | // If we don't have / can't create a memory module for this kext, don't try to load it - we won't | 
|  | 743 | // have the correct segment load addresses. | 
|  | 744 | if (!ReadMemoryModule (process)) | 
| Jason Molenda | 87a04b2 | 2012-10-19 03:40:45 +0000 | [diff] [blame] | 745 | { | 
|  | 746 | return false; | 
|  | 747 | } | 
|  | 748 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 749 | bool uuid_is_valid = m_uuid.IsValid(); | 
|  | 750 |  | 
| Jason Molenda | e575e7b | 2013-02-19 06:11:13 +0000 | [diff] [blame] | 751 | if (IsKernel() && uuid_is_valid && m_memory_module_sp.get()) | 
|  | 752 | { | 
|  | 753 | Stream *s = &target.GetDebugger().GetOutputStream(); | 
|  | 754 | if (s) | 
|  | 755 | { | 
|  | 756 | char uuidbuf[64]; | 
|  | 757 | s->Printf ("Kernel UUID: %s\n", m_memory_module_sp->GetUUID().GetAsCString(uuidbuf, sizeof (uuidbuf))); | 
|  | 758 | s->Printf ("Load Address: 0x%" PRIx64 "\n", m_load_address); | 
|  | 759 | } | 
|  | 760 | } | 
|  | 761 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 762 | if (!m_module_sp) | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 763 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 764 | // See if the kext has already been loaded into the target, probably by the user doing target modules add. | 
|  | 765 | const ModuleList &target_images = target.GetImages(); | 
|  | 766 | m_module_sp = target_images.FindModule(m_uuid); | 
|  | 767 |  | 
|  | 768 | // Search for the kext on the local filesystem via the UUID | 
|  | 769 | if (!m_module_sp && uuid_is_valid) | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 770 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 771 | ModuleSpec module_spec; | 
|  | 772 | module_spec.GetUUID() = m_uuid; | 
|  | 773 | module_spec.GetArchitecture() = target.GetArchitecture(); | 
| Jason Molenda | 53667f5 | 2012-10-06 02:02:26 +0000 | [diff] [blame] | 774 |  | 
| Jason Molenda | d76fb6e | 2013-02-19 07:16:22 +0000 | [diff] [blame] | 775 | // For the kernel, we really do need an on-disk file copy of the binary to do anything useful. | 
|  | 776 | // This will force a clal to | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 777 | if (IsKernel()) | 
| Greg Clayton | b9a01b3 | 2012-02-26 05:51:37 +0000 | [diff] [blame] | 778 | { | 
| Jason Molenda | d76fb6e | 2013-02-19 07:16:22 +0000 | [diff] [blame] | 779 | if (Symbols::DownloadObjectAndSymbolFile (module_spec, true)) | 
| Jason Molenda | bb860bd | 2012-10-09 01:17:11 +0000 | [diff] [blame] | 780 | { | 
| Jason Molenda | d76fb6e | 2013-02-19 07:16:22 +0000 | [diff] [blame] | 781 | if (module_spec.GetFileSpec().Exists()) | 
| Jason Molenda | bb860bd | 2012-10-09 01:17:11 +0000 | [diff] [blame] | 782 | { | 
| Jason Molenda | d76fb6e | 2013-02-19 07:16:22 +0000 | [diff] [blame] | 783 | m_module_sp.reset(new Module (module_spec.GetFileSpec(), target.GetArchitecture())); | 
|  | 784 | if (m_module_sp.get() && m_module_sp->MatchesModuleSpec (module_spec)) | 
|  | 785 | { | 
|  | 786 | ModuleList loaded_module_list; | 
|  | 787 | loaded_module_list.Append (m_module_sp); | 
|  | 788 | target.ModulesDidLoad (loaded_module_list); | 
|  | 789 | } | 
| Jason Molenda | bb860bd | 2012-10-09 01:17:11 +0000 | [diff] [blame] | 790 | } | 
|  | 791 | } | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 792 | } | 
| Jason Molenda | d76fb6e | 2013-02-19 07:16:22 +0000 | [diff] [blame] | 793 |  | 
|  | 794 | // Ask the Target to find this file on the local system, if possible. | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 795 | // This will search in the list of currently-loaded files, look in the | 
|  | 796 | // standard search paths on the system, and on a Mac it will try calling | 
|  | 797 | // the DebugSymbols framework with the UUID to find the binary via its | 
|  | 798 | // search methods. | 
|  | 799 | if (!m_module_sp) | 
|  | 800 | { | 
|  | 801 | m_module_sp = target.GetSharedModule (module_spec); | 
|  | 802 | } | 
| Jason Molenda | e575e7b | 2013-02-19 06:11:13 +0000 | [diff] [blame] | 803 |  | 
| Jason Molenda | d76fb6e | 2013-02-19 07:16:22 +0000 | [diff] [blame] | 804 | if (IsKernel() && !m_module_sp) | 
| Jason Molenda | e575e7b | 2013-02-19 06:11:13 +0000 | [diff] [blame] | 805 | { | 
|  | 806 | Stream *s = &target.GetDebugger().GetOutputStream(); | 
|  | 807 | if (s) | 
|  | 808 | { | 
| Jason Molenda | 56c2328 | 2013-02-19 06:39:56 +0000 | [diff] [blame] | 809 | s->Printf ("WARNING: Unable to locate kernel binary on this system.\n"); | 
| Jason Molenda | e575e7b | 2013-02-19 06:11:13 +0000 | [diff] [blame] | 810 | } | 
|  | 811 | } | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 812 | } | 
| Jason Molenda | 65d57a3 | 2012-12-01 06:13:29 +0000 | [diff] [blame] | 813 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 814 | // If we managed to find a module, append it to the target's list of images. | 
|  | 815 | // If we also have a memory module, require that they have matching UUIDs | 
|  | 816 | if (m_module_sp) | 
|  | 817 | { | 
|  | 818 | bool uuid_match_ok = true; | 
|  | 819 | if (m_memory_module_sp) | 
|  | 820 | { | 
|  | 821 | if (m_module_sp->GetUUID() != m_memory_module_sp->GetUUID()) | 
| Jason Molenda | 65d57a3 | 2012-12-01 06:13:29 +0000 | [diff] [blame] | 822 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 823 | uuid_match_ok = false; | 
|  | 824 | } | 
|  | 825 | } | 
|  | 826 | if (uuid_match_ok) | 
|  | 827 | { | 
| Jason Molenda | a4d3e1d | 2013-02-19 07:41:13 +0000 | [diff] [blame] | 828 | target.GetImages().AppendIfNeeded(m_module_sp); | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 829 | if (IsKernel() && target.GetExecutableModulePointer() != m_module_sp.get()) | 
|  | 830 | { | 
|  | 831 | target.SetExecutableModule (m_module_sp, false); | 
| Jason Molenda | 65d57a3 | 2012-12-01 06:13:29 +0000 | [diff] [blame] | 832 | } | 
| Greg Clayton | b9a01b3 | 2012-02-26 05:51:37 +0000 | [diff] [blame] | 833 | } | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 834 | } | 
|  | 835 | } | 
|  | 836 |  | 
| Jason Molenda | 56c2328 | 2013-02-19 06:39:56 +0000 | [diff] [blame] | 837 | if (!m_module_sp && !IsKernel() && m_uuid.IsValid() && !m_name.empty()) | 
|  | 838 | { | 
|  | 839 | Stream *s = &target.GetDebugger().GetOutputStream(); | 
|  | 840 | if (s) | 
|  | 841 | { | 
|  | 842 | char uuidbuf[64]; | 
|  | 843 | s->Printf ("warning: Can't find binary/dSYM for %s (%s)\n", | 
|  | 844 | m_name.c_str(), m_uuid.GetAsCString(uuidbuf, sizeof (uuidbuf))); | 
|  | 845 | } | 
|  | 846 | } | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 847 |  | 
| Greg Clayton | 6740853 | 2012-12-11 01:20:51 +0000 | [diff] [blame] | 848 | static ConstString g_section_name_LINKEDIT ("__LINKEDIT"); | 
|  | 849 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 850 | if (m_memory_module_sp && m_module_sp) | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 851 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 852 | if (m_module_sp->GetUUID() == m_memory_module_sp->GetUUID()) | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 853 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 854 | ObjectFile *ondisk_object_file = m_module_sp->GetObjectFile(); | 
|  | 855 | ObjectFile *memory_object_file = m_memory_module_sp->GetObjectFile(); | 
| Greg Clayton | 6740853 | 2012-12-11 01:20:51 +0000 | [diff] [blame] | 856 |  | 
| Jason Molenda | bb860bd | 2012-10-09 01:17:11 +0000 | [diff] [blame] | 857 | if (memory_object_file && ondisk_object_file) | 
|  | 858 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 859 | // The memory_module for kexts may have an invalid __LINKEDIT seg; skip it. | 
|  | 860 | const bool ignore_linkedit = !IsKernel (); | 
| Greg Clayton | 6740853 | 2012-12-11 01:20:51 +0000 | [diff] [blame] | 861 |  | 
| Jason Molenda | bb860bd | 2012-10-09 01:17:11 +0000 | [diff] [blame] | 862 | SectionList *ondisk_section_list = ondisk_object_file->GetSectionList (); | 
|  | 863 | SectionList *memory_section_list = memory_object_file->GetSectionList (); | 
|  | 864 | if (memory_section_list && ondisk_section_list) | 
|  | 865 | { | 
|  | 866 | const uint32_t num_ondisk_sections = ondisk_section_list->GetSize(); | 
|  | 867 | // There may be CTF sections in the memory image so we can't | 
|  | 868 | // always just compare the number of sections (which are actually | 
|  | 869 | // segments in mach-o parlance) | 
|  | 870 | uint32_t sect_idx = 0; | 
|  | 871 |  | 
|  | 872 | // Use the memory_module's addresses for each section to set the | 
|  | 873 | // file module's load address as appropriate.  We don't want to use | 
|  | 874 | // a single slide value for the entire kext - different segments may | 
|  | 875 | // be slid different amounts by the kext loader. | 
|  | 876 |  | 
|  | 877 | uint32_t num_sections_loaded = 0; | 
|  | 878 | for (sect_idx=0; sect_idx<num_ondisk_sections; ++sect_idx) | 
|  | 879 | { | 
|  | 880 | SectionSP ondisk_section_sp(ondisk_section_list->GetSectionAtIndex(sect_idx)); | 
|  | 881 | if (ondisk_section_sp) | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 882 | { | 
| Greg Clayton | 6740853 | 2012-12-11 01:20:51 +0000 | [diff] [blame] | 883 | // Don't ever load __LINKEDIT as it may or may not be actually | 
|  | 884 | // mapped into memory and there is no current way to tell. | 
|  | 885 | // I filed rdar://problem/12851706 to track being able to tell | 
|  | 886 | // if the __LINKEDIT is actually mapped, but until then, we need | 
|  | 887 | // to not load the __LINKEDIT | 
|  | 888 | if (ignore_linkedit && ondisk_section_sp->GetName() == g_section_name_LINKEDIT) | 
|  | 889 | continue; | 
|  | 890 |  | 
| Jason Molenda | bb860bd | 2012-10-09 01:17:11 +0000 | [diff] [blame] | 891 | const Section *memory_section = memory_section_list->FindSectionByName(ondisk_section_sp->GetName()).get(); | 
|  | 892 | if (memory_section) | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 893 | { | 
| Jason Molenda | bb860bd | 2012-10-09 01:17:11 +0000 | [diff] [blame] | 894 | target.GetSectionLoadList().SetSectionLoadAddress (ondisk_section_sp, memory_section->GetFileAddress()); | 
|  | 895 | ++num_sections_loaded; | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 896 | } | 
|  | 897 | } | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 898 | } | 
| Jason Molenda | bb860bd | 2012-10-09 01:17:11 +0000 | [diff] [blame] | 899 | if (num_sections_loaded > 0) | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 900 | m_load_process_stop_id = process->GetStopID(); | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 901 | else | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 902 | m_module_sp.reset(); // No sections were loaded | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 903 | } | 
|  | 904 | else | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 905 | m_module_sp.reset(); // One or both section lists | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 906 | } | 
|  | 907 | else | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 908 | m_module_sp.reset(); // One or both object files missing | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 909 | } | 
| Jason Molenda | bb860bd | 2012-10-09 01:17:11 +0000 | [diff] [blame] | 910 | else | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 911 | m_module_sp.reset(); // UUID mismatch | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 912 | } | 
| Jason Molenda | bb860bd | 2012-10-09 01:17:11 +0000 | [diff] [blame] | 913 |  | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 914 | bool is_loaded = IsLoaded(); | 
|  | 915 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 916 | if (is_loaded && m_module_sp && IsKernel()) | 
| Jason Molenda | 68b3607 | 2012-10-02 03:49:41 +0000 | [diff] [blame] | 917 | { | 
|  | 918 | Stream *s = &target.GetDebugger().GetOutputStream(); | 
|  | 919 | if (s) | 
|  | 920 | { | 
| Jason Molenda | 732238c | 2013-03-01 00:06:37 +0000 | [diff] [blame] | 921 | ObjectFile *kernel_object_file = m_module_sp->GetObjectFile(); | 
|  | 922 | if (kernel_object_file) | 
|  | 923 | { | 
|  | 924 | addr_t file_address = kernel_object_file->GetHeaderAddress().GetFileAddress(); | 
|  | 925 | if (m_load_address != LLDB_INVALID_ADDRESS && file_address != LLDB_INVALID_ADDRESS) | 
|  | 926 | { | 
|  | 927 | s->Printf ("Kernel slid 0x%llx in memory.\n", m_load_address - file_address); | 
|  | 928 | } | 
|  | 929 | } | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 930 | if (m_module_sp->GetFileSpec().GetDirectory().IsEmpty()) | 
| Jason Molenda | 743e4396 | 2012-10-02 22:23:42 +0000 | [diff] [blame] | 931 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 932 | s->Printf ("Loaded kernel file %s\n", m_module_sp->GetFileSpec().GetFilename().AsCString()); | 
| Jason Molenda | 743e4396 | 2012-10-02 22:23:42 +0000 | [diff] [blame] | 933 | } | 
|  | 934 | else | 
|  | 935 | { | 
|  | 936 | s->Printf ("Loaded kernel file %s/%s\n", | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 937 | m_module_sp->GetFileSpec().GetDirectory().AsCString(), | 
|  | 938 | m_module_sp->GetFileSpec().GetFilename().AsCString()); | 
| Jason Molenda | 743e4396 | 2012-10-02 22:23:42 +0000 | [diff] [blame] | 939 | } | 
| Jason Molenda | 68b3607 | 2012-10-02 03:49:41 +0000 | [diff] [blame] | 940 | s->Flush (); | 
|  | 941 | } | 
|  | 942 | } | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 943 | return is_loaded; | 
|  | 944 | } | 
|  | 945 |  | 
| Greg Clayton | 1f74607 | 2012-08-29 21:13:06 +0000 | [diff] [blame] | 946 | uint32_t | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 947 | DynamicLoaderDarwinKernel::KextImageInfo::GetAddressByteSize () | 
| Greg Clayton | 1f74607 | 2012-08-29 21:13:06 +0000 | [diff] [blame] | 948 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 949 | if (m_memory_module_sp) | 
|  | 950 | return m_memory_module_sp->GetArchitecture().GetAddressByteSize(); | 
|  | 951 | if (m_module_sp) | 
|  | 952 | return m_module_sp->GetArchitecture().GetAddressByteSize(); | 
| Greg Clayton | 1f74607 | 2012-08-29 21:13:06 +0000 | [diff] [blame] | 953 | return 0; | 
|  | 954 | } | 
|  | 955 |  | 
|  | 956 | lldb::ByteOrder | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 957 | DynamicLoaderDarwinKernel::KextImageInfo::GetByteOrder() | 
| Greg Clayton | 1f74607 | 2012-08-29 21:13:06 +0000 | [diff] [blame] | 958 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 959 | if (m_memory_module_sp) | 
|  | 960 | return m_memory_module_sp->GetArchitecture().GetByteOrder(); | 
|  | 961 | if (m_module_sp) | 
|  | 962 | return m_module_sp->GetArchitecture().GetByteOrder(); | 
| Greg Clayton | 1f74607 | 2012-08-29 21:13:06 +0000 | [diff] [blame] | 963 | return lldb::endian::InlHostByteOrder(); | 
|  | 964 | } | 
|  | 965 |  | 
|  | 966 | lldb_private::ArchSpec | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 967 | DynamicLoaderDarwinKernel::KextImageInfo::GetArchitecture () const | 
| Greg Clayton | 1f74607 | 2012-08-29 21:13:06 +0000 | [diff] [blame] | 968 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 969 | if (m_memory_module_sp) | 
|  | 970 | return m_memory_module_sp->GetArchitecture(); | 
|  | 971 | if (m_module_sp) | 
|  | 972 | return m_module_sp->GetArchitecture(); | 
| Greg Clayton | 1f74607 | 2012-08-29 21:13:06 +0000 | [diff] [blame] | 973 | return lldb_private::ArchSpec (); | 
|  | 974 | } | 
|  | 975 |  | 
|  | 976 |  | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 977 | //---------------------------------------------------------------------- | 
|  | 978 | // Load the kernel module and initialize the "m_kernel" member. Return | 
|  | 979 | // true _only_ if the kernel is loaded the first time through (subsequent | 
|  | 980 | // calls to this function should return false after the kernel has been | 
|  | 981 | // already loaded). | 
|  | 982 | //---------------------------------------------------------------------- | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 983 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 984 | DynamicLoaderDarwinKernel::LoadKernelModuleIfNeeded() | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 985 | { | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 986 | if (!m_kext_summary_header_ptr_addr.IsValid()) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 987 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 988 | m_kernel.Clear(); | 
|  | 989 | m_kernel.SetModule (m_process->GetTarget().GetExecutableModule()); | 
|  | 990 | m_kernel.SetIsKernel(true); | 
| Jason Molenda | 4bd4e7e | 2012-09-29 04:02:01 +0000 | [diff] [blame] | 991 |  | 
|  | 992 | ConstString kernel_name("mach_kernel"); | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 993 | if (m_kernel.GetModule().get() | 
|  | 994 | && m_kernel.GetModule()->GetObjectFile() | 
|  | 995 | && !m_kernel.GetModule()->GetObjectFile()->GetFileSpec().GetFilename().IsEmpty()) | 
| Jason Molenda | 4bd4e7e | 2012-09-29 04:02:01 +0000 | [diff] [blame] | 996 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 997 | kernel_name = m_kernel.GetModule()->GetObjectFile()->GetFileSpec().GetFilename(); | 
| Jason Molenda | 4bd4e7e | 2012-09-29 04:02:01 +0000 | [diff] [blame] | 998 | } | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 999 | m_kernel.SetName (kernel_name.AsCString()); | 
| Jason Molenda | 4bd4e7e | 2012-09-29 04:02:01 +0000 | [diff] [blame] | 1000 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1001 | if (m_kernel.GetLoadAddress() == LLDB_INVALID_ADDRESS) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1002 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1003 | m_kernel.SetLoadAddress(m_kernel_load_address); | 
|  | 1004 | if (m_kernel.GetLoadAddress() == LLDB_INVALID_ADDRESS && m_kernel.GetModule()) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1005 | { | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 1006 | // We didn't get a hint from the process, so we will | 
|  | 1007 | // try the kernel at the address that it exists at in | 
|  | 1008 | // the file if we have one | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1009 | ObjectFile *kernel_object_file = m_kernel.GetModule()->GetObjectFile(); | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 1010 | if (kernel_object_file) | 
| Jason Molenda | 4bd4e7e | 2012-09-29 04:02:01 +0000 | [diff] [blame] | 1011 | { | 
|  | 1012 | addr_t load_address = kernel_object_file->GetHeaderAddress().GetLoadAddress(&m_process->GetTarget()); | 
|  | 1013 | addr_t file_address = kernel_object_file->GetHeaderAddress().GetFileAddress(); | 
|  | 1014 | if (load_address != LLDB_INVALID_ADDRESS && load_address != 0) | 
|  | 1015 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1016 | m_kernel.SetLoadAddress (load_address); | 
| Jason Molenda | 4bd4e7e | 2012-09-29 04:02:01 +0000 | [diff] [blame] | 1017 | if (load_address != file_address) | 
|  | 1018 | { | 
|  | 1019 | // Don't accidentally relocate the kernel to the File address -- | 
|  | 1020 | // the Load address has already been set to its actual in-memory address. | 
|  | 1021 | // Mark it as IsLoaded. | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1022 | m_kernel.SetProcessStopId (m_process->GetStopID()); | 
| Jason Molenda | 4bd4e7e | 2012-09-29 04:02:01 +0000 | [diff] [blame] | 1023 | } | 
|  | 1024 | } | 
|  | 1025 | else | 
|  | 1026 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1027 | m_kernel.SetLoadAddress(file_address); | 
| Jason Molenda | 4bd4e7e | 2012-09-29 04:02:01 +0000 | [diff] [blame] | 1028 | } | 
|  | 1029 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1030 | } | 
|  | 1031 | } | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 1032 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1033 | if (m_kernel.GetLoadAddress() != LLDB_INVALID_ADDRESS) | 
| Greg Clayton | 2af282a | 2012-03-21 04:25:00 +0000 | [diff] [blame] | 1034 | { | 
|  | 1035 | if (!m_kernel.LoadImageUsingMemoryModule (m_process)) | 
|  | 1036 | { | 
|  | 1037 | m_kernel.LoadImageAtFileAddress (m_process); | 
|  | 1038 | } | 
|  | 1039 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1040 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1041 | if (m_kernel.IsLoaded() && m_kernel.GetModule()) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1042 | { | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 1043 | static ConstString kext_summary_symbol ("gLoadedKextSummaries"); | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1044 | const Symbol *symbol = m_kernel.GetModule()->FindFirstSymbolWithNameAndType (kext_summary_symbol, eSymbolTypeData); | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 1045 | if (symbol) | 
|  | 1046 | { | 
| Greg Clayton | e761213 | 2012-03-07 21:03:09 +0000 | [diff] [blame] | 1047 | m_kext_summary_header_ptr_addr = symbol->GetAddress(); | 
| Greg Clayton | c859e2d | 2012-02-13 23:10:39 +0000 | [diff] [blame] | 1048 | // Update all image infos | 
|  | 1049 | ReadAllKextSummaries (); | 
|  | 1050 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1051 | } | 
|  | 1052 | else | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1053 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1054 | m_kernel.Clear(); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1055 | } | 
|  | 1056 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1057 | } | 
|  | 1058 |  | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1059 | //---------------------------------------------------------------------- | 
|  | 1060 | // Static callback function that gets called when our DYLD notification | 
|  | 1061 | // breakpoint gets hit. We update all of our image infos and then | 
|  | 1062 | // let our super class DynamicLoader class decide if we should stop | 
|  | 1063 | // or not (based on global preference). | 
|  | 1064 | //---------------------------------------------------------------------- | 
|  | 1065 | bool | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1066 | DynamicLoaderDarwinKernel::BreakpointHitCallback (void *baton, | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1067 | StoppointCallbackContext *context, | 
|  | 1068 | user_id_t break_id, | 
|  | 1069 | user_id_t break_loc_id) | 
| Greg Clayton | 0d9fc76 | 2011-07-08 03:21:57 +0000 | [diff] [blame] | 1070 | { | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1071 | return static_cast<DynamicLoaderDarwinKernel*>(baton)->BreakpointHit (context, break_id, break_loc_id); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1072 | } | 
|  | 1073 |  | 
|  | 1074 | bool | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1075 | DynamicLoaderDarwinKernel::BreakpointHit (StoppointCallbackContext *context, | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1076 | user_id_t break_id, | 
|  | 1077 | user_id_t break_loc_id) | 
|  | 1078 | { | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 1079 | LogSP log(GetLogIfAnyCategoriesSet (LIBLLDB_LOG_DYNAMIC_LOADER)); | 
|  | 1080 | if (log) | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1081 | log->Printf ("DynamicLoaderDarwinKernel::BreakpointHit (...)\n"); | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 1082 |  | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1083 | ReadAllKextSummaries (); | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 1084 |  | 
|  | 1085 | if (log) | 
|  | 1086 | PutToLog(log.get()); | 
|  | 1087 |  | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1088 | return GetStopWhenImagesChange(); | 
|  | 1089 | } | 
|  | 1090 |  | 
|  | 1091 |  | 
|  | 1092 | bool | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1093 | DynamicLoaderDarwinKernel::ReadKextSummaryHeader () | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1094 | { | 
|  | 1095 | Mutex::Locker locker(m_mutex); | 
|  | 1096 |  | 
|  | 1097 | // the all image infos is already valid for this process stop ID | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1098 |  | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 1099 | if (m_kext_summary_header_ptr_addr.IsValid()) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1100 | { | 
|  | 1101 | const uint32_t addr_size = m_kernel.GetAddressByteSize (); | 
|  | 1102 | const ByteOrder byte_order = m_kernel.GetByteOrder(); | 
|  | 1103 | Error error; | 
|  | 1104 | // Read enough bytes for a "OSKextLoadedKextSummaryHeader" structure | 
|  | 1105 | // which is currenty 4 uint32_t and a pointer. | 
|  | 1106 | uint8_t buf[24]; | 
|  | 1107 | DataExtractor data (buf, sizeof(buf), byte_order, addr_size); | 
|  | 1108 | const size_t count = 4 * sizeof(uint32_t) + addr_size; | 
| Greg Clayton | 0d9fc76 | 2011-07-08 03:21:57 +0000 | [diff] [blame] | 1109 | const bool prefer_file_cache = false; | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 1110 | if (m_process->GetTarget().ReadPointerFromMemory (m_kext_summary_header_ptr_addr, | 
|  | 1111 | prefer_file_cache, | 
|  | 1112 | error, | 
|  | 1113 | m_kext_summary_header_addr)) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1114 | { | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 1115 | // We got a valid address for our kext summary header and make sure it isn't NULL | 
|  | 1116 | if (m_kext_summary_header_addr.IsValid() && | 
|  | 1117 | m_kext_summary_header_addr.GetFileAddress() != 0) | 
|  | 1118 | { | 
|  | 1119 | const size_t bytes_read = m_process->GetTarget().ReadMemory (m_kext_summary_header_addr, prefer_file_cache, buf, count, error); | 
|  | 1120 | if (bytes_read == count) | 
|  | 1121 | { | 
| Greg Clayton | c7bece56 | 2013-01-25 18:06:21 +0000 | [diff] [blame] | 1122 | lldb::offset_t offset = 0; | 
| Greg Clayton | a63d08c | 2011-07-19 03:57:15 +0000 | [diff] [blame] | 1123 | m_kext_summary_header.version = data.GetU32(&offset); | 
|  | 1124 | if (m_kext_summary_header.version >= 2) | 
|  | 1125 | { | 
|  | 1126 | m_kext_summary_header.entry_size = data.GetU32(&offset); | 
|  | 1127 | } | 
|  | 1128 | else | 
|  | 1129 | { | 
|  | 1130 | // Versions less than 2 didn't have an entry size, it was hard coded | 
|  | 1131 | m_kext_summary_header.entry_size = KERNEL_MODULE_ENTRY_SIZE_VERSION_1; | 
|  | 1132 | } | 
|  | 1133 | m_kext_summary_header.entry_count = data.GetU32(&offset); | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 1134 | return true; | 
|  | 1135 | } | 
|  | 1136 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1137 | } | 
|  | 1138 | } | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 1139 | m_kext_summary_header_addr.Clear(); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1140 | return false; | 
|  | 1141 | } | 
|  | 1142 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1143 | // We've either (a) just attached to a new kernel, or (b) the kexts-changed breakpoint was hit | 
|  | 1144 | // and we need to figure out what kexts have been added or removed. | 
|  | 1145 | // Read the kext summaries from the inferior kernel memory, compare them against the | 
|  | 1146 | // m_known_kexts vector and update the m_known_kexts vector as needed to keep in sync with the | 
|  | 1147 | // inferior. | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1148 |  | 
|  | 1149 | bool | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1150 | DynamicLoaderDarwinKernel::ParseKextSummaries (const Address &kext_summary_addr, uint32_t count) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1151 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1152 | KextImageInfo::collection kext_summaries; | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1153 | LogSP log(GetLogIfAnyCategoriesSet (LIBLLDB_LOG_DYNAMIC_LOADER)); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1154 | if (log) | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1155 | log->Printf ("Kexts-changed breakpoint hit, there are %d kexts currently.\n", count); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1156 |  | 
|  | 1157 | Mutex::Locker locker(m_mutex); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1158 |  | 
|  | 1159 | if (!ReadKextSummaries (kext_summary_addr, count, kext_summaries)) | 
|  | 1160 | return false; | 
|  | 1161 |  | 
| Jason Molenda | 4da2e32 | 2013-02-26 00:26:47 +0000 | [diff] [blame] | 1162 | // read the plugin.dynamic-loader.darwin-kernel.load-kexts setting -- if the user requested no | 
|  | 1163 | // kext loading, don't print any messages about kexts & don't try to read them. | 
|  | 1164 | const bool load_kexts = GetGlobalProperties()->GetLoadKexts(); | 
|  | 1165 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1166 | // By default, all kexts we've loaded in the past are marked as "remove" and all of the kexts | 
|  | 1167 | // we just found out about from ReadKextSummaries are marked as "add". | 
|  | 1168 | std::vector<bool> to_be_removed(m_known_kexts.size(), true); | 
|  | 1169 | std::vector<bool> to_be_added(count, true); | 
|  | 1170 |  | 
|  | 1171 | int number_of_new_kexts_being_added = 0; | 
|  | 1172 | int number_of_old_kexts_being_removed = m_known_kexts.size(); | 
|  | 1173 |  | 
|  | 1174 | const uint32_t new_kexts_size = kext_summaries.size(); | 
|  | 1175 | const uint32_t old_kexts_size = m_known_kexts.size(); | 
|  | 1176 |  | 
|  | 1177 | // The m_known_kexts vector may have entries that have been Cleared, | 
|  | 1178 | // or are a kernel. | 
|  | 1179 | for (uint32_t old_kext = 0; old_kext < old_kexts_size; old_kext++) | 
|  | 1180 | { | 
|  | 1181 | bool ignore = false; | 
|  | 1182 | KextImageInfo &image_info = m_known_kexts[old_kext]; | 
|  | 1183 | if (image_info.IsKernel()) | 
|  | 1184 | { | 
|  | 1185 | ignore = true; | 
|  | 1186 | } | 
|  | 1187 | else if (image_info.GetLoadAddress() == LLDB_INVALID_ADDRESS && !image_info.GetModule()) | 
|  | 1188 | { | 
|  | 1189 | ignore = true; | 
|  | 1190 | } | 
|  | 1191 |  | 
|  | 1192 | if (ignore) | 
|  | 1193 | { | 
|  | 1194 | number_of_old_kexts_being_removed--; | 
|  | 1195 | to_be_removed[old_kext] = false; | 
|  | 1196 | } | 
|  | 1197 | } | 
|  | 1198 |  | 
|  | 1199 | // Scan over the list of kexts we just read from the kernel, note those that | 
|  | 1200 | // need to be added and those already loaded. | 
|  | 1201 | for (uint32_t new_kext = 0; new_kext < new_kexts_size; new_kext++) | 
|  | 1202 | { | 
|  | 1203 | bool add_this_one = true; | 
|  | 1204 | for (uint32_t old_kext = 0; old_kext < old_kexts_size; old_kext++) | 
|  | 1205 | { | 
|  | 1206 | if (m_known_kexts[old_kext] == kext_summaries[new_kext]) | 
|  | 1207 | { | 
|  | 1208 | // We already have this kext, don't re-load it. | 
|  | 1209 | to_be_added[new_kext] = false; | 
|  | 1210 | // This kext is still present, do not remove it. | 
|  | 1211 | to_be_removed[old_kext] = false; | 
|  | 1212 |  | 
|  | 1213 | number_of_old_kexts_being_removed--; | 
|  | 1214 | add_this_one = false; | 
|  | 1215 | break; | 
|  | 1216 | } | 
|  | 1217 | } | 
|  | 1218 | if (add_this_one) | 
|  | 1219 | { | 
|  | 1220 | number_of_new_kexts_being_added++; | 
|  | 1221 | } | 
|  | 1222 | } | 
|  | 1223 |  | 
|  | 1224 | if (number_of_new_kexts_being_added == 0 && number_of_old_kexts_being_removed == 0) | 
|  | 1225 | return true; | 
|  | 1226 |  | 
| Greg Clayton | 07e66e3 | 2011-07-20 03:41:06 +0000 | [diff] [blame] | 1227 | Stream *s = &m_process->GetTarget().GetDebugger().GetOutputStream(); | 
| Jason Molenda | 4da2e32 | 2013-02-26 00:26:47 +0000 | [diff] [blame] | 1228 | if (s && load_kexts) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1229 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1230 | if (number_of_new_kexts_being_added > 0 && number_of_old_kexts_being_removed > 0) | 
|  | 1231 | { | 
|  | 1232 | s->Printf ("Loading %d kext modules and unloading %d kext modules ", number_of_new_kexts_being_added, number_of_old_kexts_being_removed); | 
|  | 1233 | } | 
|  | 1234 | else if (number_of_new_kexts_being_added > 0) | 
|  | 1235 | { | 
|  | 1236 | s->Printf ("Loading %d kext modules ", number_of_new_kexts_being_added); | 
|  | 1237 | } | 
|  | 1238 | else if (number_of_old_kexts_being_removed > 0) | 
|  | 1239 | { | 
|  | 1240 | s->Printf ("Unloading %d kext modules ", number_of_old_kexts_being_removed); | 
|  | 1241 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1242 | } | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1243 |  | 
|  | 1244 | if (log) | 
| Jason Molenda | 4da2e32 | 2013-02-26 00:26:47 +0000 | [diff] [blame] | 1245 | { | 
|  | 1246 | if (load_kexts) | 
|  | 1247 | { | 
|  | 1248 | log->Printf ("DynamicLoaderDarwinKernel::ParseKextSummaries: %d kexts added, %d kexts removed", number_of_new_kexts_being_added, number_of_old_kexts_being_removed); | 
|  | 1249 | } | 
|  | 1250 | else | 
|  | 1251 | { | 
|  | 1252 | log->Printf ("DynamicLoaderDarwinKernel::ParseKextSummaries kext loading is disabled, else would have %d kexts added, %d kexts removed", number_of_new_kexts_being_added, number_of_old_kexts_being_removed); | 
|  | 1253 | } | 
|  | 1254 | } | 
|  | 1255 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1256 |  | 
|  | 1257 | if (number_of_new_kexts_being_added > 0) | 
|  | 1258 | { | 
|  | 1259 | ModuleList loaded_module_list; | 
|  | 1260 |  | 
|  | 1261 | const uint32_t num_of_new_kexts = kext_summaries.size(); | 
|  | 1262 | for (uint32_t new_kext = 0; new_kext < num_of_new_kexts; new_kext++) | 
|  | 1263 | { | 
|  | 1264 | if (to_be_added[new_kext] == true) | 
|  | 1265 | { | 
|  | 1266 | KextImageInfo &image_info = kext_summaries[new_kext]; | 
| Jason Molenda | 4da2e32 | 2013-02-26 00:26:47 +0000 | [diff] [blame] | 1267 | if (load_kexts) | 
|  | 1268 | { | 
|  | 1269 | if (!image_info.LoadImageUsingMemoryModule (m_process)) | 
|  | 1270 | { | 
|  | 1271 | image_info.LoadImageAtFileAddress (m_process); | 
|  | 1272 | } | 
|  | 1273 | } | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1274 |  | 
|  | 1275 | m_known_kexts.push_back(image_info); | 
|  | 1276 |  | 
|  | 1277 | if (image_info.GetModule() && m_process->GetStopID() == image_info.GetProcessStopId()) | 
|  | 1278 | loaded_module_list.AppendIfNeeded (image_info.GetModule()); | 
|  | 1279 |  | 
| Jason Molenda | 4da2e32 | 2013-02-26 00:26:47 +0000 | [diff] [blame] | 1280 | if (s && load_kexts) | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1281 | s->Printf ("."); | 
|  | 1282 |  | 
|  | 1283 | if (log) | 
|  | 1284 | kext_summaries[new_kext].PutToLog (log.get()); | 
|  | 1285 | } | 
|  | 1286 | } | 
|  | 1287 | m_process->GetTarget().ModulesDidLoad (loaded_module_list); | 
|  | 1288 | } | 
|  | 1289 |  | 
|  | 1290 | if (number_of_old_kexts_being_removed > 0) | 
|  | 1291 | { | 
|  | 1292 | ModuleList loaded_module_list; | 
|  | 1293 | const uint32_t num_of_old_kexts = m_known_kexts.size(); | 
|  | 1294 | for (uint32_t old_kext = 0; old_kext < num_of_old_kexts; old_kext++) | 
|  | 1295 | { | 
|  | 1296 | ModuleList unloaded_module_list; | 
|  | 1297 | if (to_be_removed[old_kext]) | 
|  | 1298 | { | 
|  | 1299 | KextImageInfo &image_info = m_known_kexts[old_kext]; | 
|  | 1300 | // You can't unload the kernel. | 
|  | 1301 | if (!image_info.IsKernel()) | 
|  | 1302 | { | 
|  | 1303 | if (image_info.GetModule()) | 
|  | 1304 | { | 
|  | 1305 | unloaded_module_list.AppendIfNeeded (image_info.GetModule()); | 
|  | 1306 | } | 
|  | 1307 | if (s) | 
|  | 1308 | s->Printf ("."); | 
|  | 1309 | image_info.Clear(); | 
|  | 1310 | // should pull it out of the KextImageInfos vector but that would mutate the list and invalidate | 
|  | 1311 | // the to_be_removed bool vector; leaving it in place once Cleared() is relatively harmless. | 
|  | 1312 | } | 
|  | 1313 | } | 
|  | 1314 | m_process->GetTarget().ModulesDidUnload (unloaded_module_list); | 
|  | 1315 | } | 
|  | 1316 | } | 
|  | 1317 |  | 
| Jason Molenda | 4da2e32 | 2013-02-26 00:26:47 +0000 | [diff] [blame] | 1318 | if (s && load_kexts) | 
| Jason Molenda | 68b3607 | 2012-10-02 03:49:41 +0000 | [diff] [blame] | 1319 | { | 
|  | 1320 | s->Printf (" done.\n"); | 
|  | 1321 | s->Flush (); | 
|  | 1322 | } | 
|  | 1323 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1324 | return true; | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1325 | } | 
|  | 1326 |  | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1327 | uint32_t | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1328 | DynamicLoaderDarwinKernel::ReadKextSummaries (const Address &kext_summary_addr, | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1329 | uint32_t image_infos_count, | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1330 | KextImageInfo::collection &image_infos) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1331 | { | 
|  | 1332 | const ByteOrder endian = m_kernel.GetByteOrder(); | 
|  | 1333 | const uint32_t addr_size = m_kernel.GetAddressByteSize(); | 
|  | 1334 |  | 
|  | 1335 | image_infos.resize(image_infos_count); | 
|  | 1336 | const size_t count = image_infos.size() * m_kext_summary_header.entry_size; | 
|  | 1337 | DataBufferHeap data(count, 0); | 
|  | 1338 | Error error; | 
| Greg Clayton | 5b88216 | 2011-07-21 01:12:01 +0000 | [diff] [blame] | 1339 |  | 
| Greg Clayton | 0d9fc76 | 2011-07-08 03:21:57 +0000 | [diff] [blame] | 1340 | const bool prefer_file_cache = false; | 
|  | 1341 | const size_t bytes_read = m_process->GetTarget().ReadMemory (kext_summary_addr, | 
|  | 1342 | prefer_file_cache, | 
|  | 1343 | data.GetBytes(), | 
|  | 1344 | data.GetByteSize(), | 
|  | 1345 | error); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1346 | if (bytes_read == count) | 
|  | 1347 | { | 
| Greg Clayton | a63d08c | 2011-07-19 03:57:15 +0000 | [diff] [blame] | 1348 |  | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1349 | DataExtractor extractor (data.GetBytes(), data.GetByteSize(), endian, addr_size); | 
|  | 1350 | uint32_t i=0; | 
| Greg Clayton | a63d08c | 2011-07-19 03:57:15 +0000 | [diff] [blame] | 1351 | for (uint32_t kext_summary_offset = 0; | 
|  | 1352 | i < image_infos.size() && extractor.ValidOffsetForDataOfSize(kext_summary_offset, m_kext_summary_header.entry_size); | 
|  | 1353 | ++i, kext_summary_offset += m_kext_summary_header.entry_size) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1354 | { | 
| Greg Clayton | c7bece56 | 2013-01-25 18:06:21 +0000 | [diff] [blame] | 1355 | lldb::offset_t offset = kext_summary_offset; | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1356 | const void *name_data = extractor.GetData(&offset, KERNEL_MODULE_MAX_NAME); | 
|  | 1357 | if (name_data == NULL) | 
|  | 1358 | break; | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1359 | image_infos[i].SetName ((const char *) name_data); | 
|  | 1360 | UUID uuid (extractor.GetData (&offset, 16), 16); | 
|  | 1361 | image_infos[i].SetUUID (uuid); | 
|  | 1362 | image_infos[i].SetLoadAddress (extractor.GetU64(&offset)); | 
|  | 1363 | image_infos[i].SetSize (extractor.GetU64(&offset)); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1364 | } | 
|  | 1365 | if (i < image_infos.size()) | 
|  | 1366 | image_infos.resize(i); | 
|  | 1367 | } | 
|  | 1368 | else | 
|  | 1369 | { | 
|  | 1370 | image_infos.clear(); | 
|  | 1371 | } | 
|  | 1372 | return image_infos.size(); | 
|  | 1373 | } | 
|  | 1374 |  | 
|  | 1375 | bool | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1376 | DynamicLoaderDarwinKernel::ReadAllKextSummaries () | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1377 | { | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1378 | LogSP log(GetLogIfAnyCategoriesSet (LIBLLDB_LOG_DYNAMIC_LOADER)); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1379 |  | 
|  | 1380 | Mutex::Locker locker(m_mutex); | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1381 |  | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1382 | if (ReadKextSummaryHeader ()) | 
|  | 1383 | { | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1384 | if (m_kext_summary_header.entry_count > 0 && m_kext_summary_header_addr.IsValid()) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1385 | { | 
| Greg Clayton | 0d9fc76 | 2011-07-08 03:21:57 +0000 | [diff] [blame] | 1386 | Address summary_addr (m_kext_summary_header_addr); | 
| Greg Clayton | a63d08c | 2011-07-19 03:57:15 +0000 | [diff] [blame] | 1387 | summary_addr.Slide(m_kext_summary_header.GetSize()); | 
| Greg Clayton | 0d9fc76 | 2011-07-08 03:21:57 +0000 | [diff] [blame] | 1388 | if (!ParseKextSummaries (summary_addr, m_kext_summary_header.entry_count)) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1389 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1390 | m_known_kexts.clear(); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1391 | } | 
|  | 1392 | return true; | 
|  | 1393 | } | 
|  | 1394 | } | 
|  | 1395 | return false; | 
|  | 1396 | } | 
|  | 1397 |  | 
|  | 1398 | //---------------------------------------------------------------------- | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1399 | // Dump an image info structure to the file handle provided. | 
|  | 1400 | //---------------------------------------------------------------------- | 
|  | 1401 | void | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1402 | DynamicLoaderDarwinKernel::KextImageInfo::PutToLog (Log *log) const | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1403 | { | 
|  | 1404 | if (log == NULL) | 
|  | 1405 | return; | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1406 | const uint8_t *u = (uint8_t *) m_uuid.GetBytes(); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1407 |  | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1408 | if (m_load_address == LLDB_INVALID_ADDRESS) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1409 | { | 
|  | 1410 | if (u) | 
|  | 1411 | { | 
| Greg Clayton | a63d08c | 2011-07-19 03:57:15 +0000 | [diff] [blame] | 1412 | 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)", | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1413 | u[ 0], u[ 1], u[ 2], u[ 3], | 
|  | 1414 | u[ 4], u[ 5], u[ 6], u[ 7], | 
|  | 1415 | u[ 8], u[ 9], u[10], u[11], | 
|  | 1416 | u[12], u[13], u[14], u[15], | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1417 | m_name.c_str()); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1418 | } | 
|  | 1419 | else | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1420 | log->Printf("\tname=\"%s\" (UNLOADED)", m_name.c_str()); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1421 | } | 
|  | 1422 | else | 
|  | 1423 | { | 
|  | 1424 | if (u) | 
|  | 1425 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1426 | log->Printf("\taddr=0x%16.16" PRIx64 " size=0x%16.16" PRIx64 " 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\"", | 
|  | 1427 | m_load_address, m_size, | 
| Greg Clayton | a63d08c | 2011-07-19 03:57:15 +0000 | [diff] [blame] | 1428 | u[ 0], u[ 1], u[ 2], u[ 3], u[ 4], u[ 5], u[ 6], u[ 7], | 
|  | 1429 | u[ 8], u[ 9], u[10], u[11], u[12], u[13], u[14], u[15], | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1430 | m_name.c_str()); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1431 | } | 
|  | 1432 | else | 
|  | 1433 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1434 | log->Printf("\t[0x%16.16" PRIx64 " - 0x%16.16" PRIx64 ") name=\"%s\"", | 
|  | 1435 | m_load_address, m_load_address+m_size, m_name.c_str()); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1436 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1437 | } | 
|  | 1438 | } | 
|  | 1439 |  | 
|  | 1440 | //---------------------------------------------------------------------- | 
|  | 1441 | // Dump the _dyld_all_image_infos members and all current image infos | 
|  | 1442 | // that we have parsed to the file handle provided. | 
|  | 1443 | //---------------------------------------------------------------------- | 
|  | 1444 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1445 | DynamicLoaderDarwinKernel::PutToLog(Log *log) const | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1446 | { | 
|  | 1447 | if (log == NULL) | 
|  | 1448 | return; | 
|  | 1449 |  | 
|  | 1450 | Mutex::Locker locker(m_mutex); | 
| Daniel Malea | d01b295 | 2012-11-29 21:49:15 +0000 | [diff] [blame] | 1451 | log->Printf("gLoadedKextSummaries = 0x%16.16" PRIx64 " { version=%u, entry_size=%u, entry_count=%u }", | 
| Greg Clayton | 0d9fc76 | 2011-07-08 03:21:57 +0000 | [diff] [blame] | 1452 | m_kext_summary_header_addr.GetFileAddress(), | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1453 | m_kext_summary_header.version, | 
|  | 1454 | m_kext_summary_header.entry_size, | 
| Greg Clayton | a63d08c | 2011-07-19 03:57:15 +0000 | [diff] [blame] | 1455 | m_kext_summary_header.entry_count); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1456 |  | 
|  | 1457 | size_t i; | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1458 | const size_t count = m_known_kexts.size(); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1459 | if (count > 0) | 
|  | 1460 | { | 
|  | 1461 | log->PutCString("Loaded:"); | 
|  | 1462 | for (i = 0; i<count; i++) | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1463 | m_known_kexts[i].PutToLog(log); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1464 | } | 
|  | 1465 | } | 
|  | 1466 |  | 
|  | 1467 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1468 | DynamicLoaderDarwinKernel::PrivateInitialize(Process *process) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1469 | { | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1470 | DEBUG_PRINTF("DynamicLoaderDarwinKernel::%s() process state = %s\n", __FUNCTION__, StateAsCString(m_process->GetState())); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1471 | Clear(true); | 
|  | 1472 | m_process = process; | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1473 | } | 
|  | 1474 |  | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1475 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1476 | DynamicLoaderDarwinKernel::SetNotificationBreakpointIfNeeded () | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1477 | { | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1478 | if (m_break_id == LLDB_INVALID_BREAK_ID && m_kernel.GetModule()) | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1479 | { | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1480 | DEBUG_PRINTF("DynamicLoaderDarwinKernel::%s() process state = %s\n", __FUNCTION__, StateAsCString(m_process->GetState())); | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1481 |  | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 1482 |  | 
| Jim Ingham | a8558b6 | 2012-05-22 00:12:20 +0000 | [diff] [blame] | 1483 | const bool internal_bp = true; | 
| Greg Clayton | d16e1e5 | 2011-07-12 17:06:17 +0000 | [diff] [blame] | 1484 | const LazyBool skip_prologue = eLazyBoolNo; | 
| Jim Ingham | 969795f | 2011-09-21 01:17:13 +0000 | [diff] [blame] | 1485 | FileSpecList module_spec_list; | 
| Jason Molenda | 306bd0a | 2013-02-19 05:42:46 +0000 | [diff] [blame] | 1486 | module_spec_list.Append (m_kernel.GetModule()->GetFileSpec()); | 
| Jim Ingham | 969795f | 2011-09-21 01:17:13 +0000 | [diff] [blame] | 1487 | Breakpoint *bp = m_process->GetTarget().CreateBreakpoint (&module_spec_list, | 
| Jim Ingham | 87df91b | 2011-09-23 00:54:11 +0000 | [diff] [blame] | 1488 | NULL, | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1489 | "OSKextLoadedKextSummariesUpdated", | 
|  | 1490 | eFunctionNameTypeFull, | 
| Jim Ingham | a8558b6 | 2012-05-22 00:12:20 +0000 | [diff] [blame] | 1491 | skip_prologue, | 
|  | 1492 | internal_bp).get(); | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1493 |  | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1494 | bp->SetCallback (DynamicLoaderDarwinKernel::BreakpointHitCallback, this, true); | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1495 | m_break_id = bp->GetID(); | 
|  | 1496 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1497 | } | 
|  | 1498 |  | 
|  | 1499 | //---------------------------------------------------------------------- | 
|  | 1500 | // Member function that gets called when the process state changes. | 
|  | 1501 | //---------------------------------------------------------------------- | 
|  | 1502 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1503 | DynamicLoaderDarwinKernel::PrivateProcessStateChanged (Process *process, StateType state) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1504 | { | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1505 | DEBUG_PRINTF("DynamicLoaderDarwinKernel::%s(%s)\n", __FUNCTION__, StateAsCString(state)); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1506 | switch (state) | 
|  | 1507 | { | 
|  | 1508 | case eStateConnected: | 
|  | 1509 | case eStateAttaching: | 
|  | 1510 | case eStateLaunching: | 
|  | 1511 | case eStateInvalid: | 
|  | 1512 | case eStateUnloaded: | 
|  | 1513 | case eStateExited: | 
|  | 1514 | case eStateDetached: | 
|  | 1515 | Clear(false); | 
|  | 1516 | break; | 
|  | 1517 |  | 
|  | 1518 | case eStateStopped: | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1519 | UpdateIfNeeded(); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1520 | break; | 
|  | 1521 |  | 
|  | 1522 | case eStateRunning: | 
|  | 1523 | case eStateStepping: | 
|  | 1524 | case eStateCrashed: | 
|  | 1525 | case eStateSuspended: | 
|  | 1526 | break; | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1527 | } | 
|  | 1528 | } | 
|  | 1529 |  | 
|  | 1530 | ThreadPlanSP | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1531 | DynamicLoaderDarwinKernel::GetStepThroughTrampolinePlan (Thread &thread, bool stop_others) | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1532 | { | 
|  | 1533 | ThreadPlanSP thread_plan_sp; | 
| Greg Clayton | 374972e | 2011-07-09 17:15:55 +0000 | [diff] [blame] | 1534 | LogSP log(GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP)); | 
|  | 1535 | if (log) | 
|  | 1536 | log->Printf ("Could not find symbol for step through."); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1537 | return thread_plan_sp; | 
|  | 1538 | } | 
|  | 1539 |  | 
|  | 1540 | Error | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1541 | DynamicLoaderDarwinKernel::CanLoadImage () | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1542 | { | 
|  | 1543 | Error error; | 
|  | 1544 | error.SetErrorString("always unsafe to load or unload shared libraries in the darwin kernel"); | 
|  | 1545 | return error; | 
|  | 1546 | } | 
|  | 1547 |  | 
|  | 1548 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1549 | DynamicLoaderDarwinKernel::Initialize() | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1550 | { | 
|  | 1551 | PluginManager::RegisterPlugin (GetPluginNameStatic(), | 
|  | 1552 | GetPluginDescriptionStatic(), | 
| Greg Clayton | e8cd0c9 | 2012-10-19 18:02:49 +0000 | [diff] [blame] | 1553 | CreateInstance, | 
|  | 1554 | DebuggerInitialize); | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1555 | } | 
|  | 1556 |  | 
|  | 1557 | void | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1558 | DynamicLoaderDarwinKernel::Terminate() | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1559 | { | 
|  | 1560 | PluginManager::UnregisterPlugin (CreateInstance); | 
|  | 1561 | } | 
|  | 1562 |  | 
| Greg Clayton | e8cd0c9 | 2012-10-19 18:02:49 +0000 | [diff] [blame] | 1563 | void | 
|  | 1564 | DynamicLoaderDarwinKernel::DebuggerInitialize (lldb_private::Debugger &debugger) | 
|  | 1565 | { | 
|  | 1566 | if (!PluginManager::GetSettingForDynamicLoaderPlugin (debugger, DynamicLoaderDarwinKernelProperties::GetSettingName())) | 
|  | 1567 | { | 
|  | 1568 | const bool is_global_setting = true; | 
|  | 1569 | PluginManager::CreateSettingForDynamicLoaderPlugin (debugger, | 
|  | 1570 | GetGlobalProperties()->GetValueProperties(), | 
|  | 1571 | ConstString ("Properties for the DynamicLoaderDarwinKernel plug-in."), | 
|  | 1572 | is_global_setting); | 
|  | 1573 | } | 
|  | 1574 | } | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1575 |  | 
|  | 1576 | const char * | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1577 | DynamicLoaderDarwinKernel::GetPluginNameStatic() | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1578 | { | 
| Greg Clayton | 468ea4e | 2012-10-19 18:14:47 +0000 | [diff] [blame] | 1579 | return "dynamic-loader.darwin-kernel"; | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1580 | } | 
|  | 1581 |  | 
|  | 1582 | const char * | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1583 | DynamicLoaderDarwinKernel::GetPluginDescriptionStatic() | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1584 | { | 
|  | 1585 | return "Dynamic loader plug-in that watches for shared library loads/unloads in the MacOSX kernel."; | 
|  | 1586 | } | 
|  | 1587 |  | 
|  | 1588 |  | 
|  | 1589 | //------------------------------------------------------------------ | 
|  | 1590 | // PluginInterface protocol | 
|  | 1591 | //------------------------------------------------------------------ | 
|  | 1592 | const char * | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1593 | DynamicLoaderDarwinKernel::GetPluginName() | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1594 | { | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1595 | return "DynamicLoaderDarwinKernel"; | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1596 | } | 
|  | 1597 |  | 
|  | 1598 | const char * | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1599 | DynamicLoaderDarwinKernel::GetShortPluginName() | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1600 | { | 
|  | 1601 | return GetPluginNameStatic(); | 
|  | 1602 | } | 
|  | 1603 |  | 
|  | 1604 | uint32_t | 
| Greg Clayton | 944b828 | 2011-08-22 22:30:57 +0000 | [diff] [blame] | 1605 | DynamicLoaderDarwinKernel::GetPluginVersion() | 
| Greg Clayton | 7b24238 | 2011-07-08 00:48:09 +0000 | [diff] [blame] | 1606 | { | 
|  | 1607 | return 1; | 
|  | 1608 | } | 
|  | 1609 |  | 
| Greg Clayton | 1f74607 | 2012-08-29 21:13:06 +0000 | [diff] [blame] | 1610 | lldb::ByteOrder | 
|  | 1611 | DynamicLoaderDarwinKernel::GetByteOrderFromMagic (uint32_t magic) | 
|  | 1612 | { | 
|  | 1613 | switch (magic) | 
|  | 1614 | { | 
|  | 1615 | case llvm::MachO::HeaderMagic32: | 
|  | 1616 | case llvm::MachO::HeaderMagic64: | 
|  | 1617 | return lldb::endian::InlHostByteOrder(); | 
|  | 1618 |  | 
|  | 1619 | case llvm::MachO::HeaderMagic32Swapped: | 
|  | 1620 | case llvm::MachO::HeaderMagic64Swapped: | 
|  | 1621 | if (lldb::endian::InlHostByteOrder() == lldb::eByteOrderBig) | 
|  | 1622 | return lldb::eByteOrderLittle; | 
|  | 1623 | else | 
|  | 1624 | return lldb::eByteOrderBig; | 
|  | 1625 |  | 
|  | 1626 | default: | 
|  | 1627 | break; | 
|  | 1628 | } | 
|  | 1629 | return lldb::eByteOrderInvalid; | 
|  | 1630 | } | 
|  | 1631 |  |