1 //===-- CommandObjectDisassemble.cpp ----------------------------*- C++ -*-===// 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 10 #include "CommandObjectDisassemble.h" 11 12 // C Includes 13 // C++ Includes 14 // Other libraries and framework includes 15 // Project includes 16 #include "lldb/Core/AddressRange.h" 17 #include "lldb/Interpreter/Args.h" 18 #include "lldb/Interpreter/CommandCompletions.h" 19 #include "lldb/Interpreter/CommandInterpreter.h" 20 #include "lldb/Interpreter/CommandReturnObject.h" 21 #include "lldb/Core/Disassembler.h" 22 #include "lldb/Interpreter/Options.h" 23 #include "lldb/Core/SourceManager.h" 24 #include "lldb/Target/StackFrame.h" 25 #include "lldb/Symbol/Symbol.h" 26 #include "lldb/Target/Process.h" 27 #include "lldb/Target/Target.h" 28 29 #define DEFAULT_DISASM_BYTE_SIZE 32 30 #define DEFAULT_DISASM_NUM_INS 4 31 32 using namespace lldb; 33 using namespace lldb_private; 34 35 CommandObjectDisassemble::CommandOptions::CommandOptions (CommandInterpreter &interpreter) : 36 Options(interpreter), 37 num_lines_context(0), 38 num_instructions (0), 39 func_name(), 40 cur_function (false), 41 start_addr(), 42 end_addr (), 43 at_pc (false), 44 frame_line (false), 45 plugin_name (), 46 arch(), 47 some_location_specified (false) 48 { 49 OptionParsingStarting(); 50 } 51 52 CommandObjectDisassemble::CommandOptions::~CommandOptions () 53 { 54 } 55 56 Error 57 CommandObjectDisassemble::CommandOptions::SetOptionValue (uint32_t option_idx, const char *option_arg) 58 { 59 Error error; 60 61 char short_option = (char) m_getopt_table[option_idx].val; 62 63 bool success; 64 65 switch (short_option) 66 { 67 case 'm': 68 show_mixed = true; 69 break; 70 71 case 'C': 72 num_lines_context = Args::StringToUInt32(option_arg, 0, 0, &success); 73 if (!success) 74 error.SetErrorStringWithFormat ("Invalid num context lines string: \"%s\".\n", option_arg); 75 break; 76 77 case 'c': 78 num_instructions = Args::StringToUInt32(option_arg, 0, 0, &success); 79 if (!success) 80 error.SetErrorStringWithFormat ("Invalid num of instructions string: \"%s\".\n", option_arg); 81 break; 82 83 case 'b': 84 show_bytes = true; 85 break; 86 87 case 's': 88 start_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 0); 89 if (start_addr == LLDB_INVALID_ADDRESS) 90 start_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 16); 91 92 if (start_addr == LLDB_INVALID_ADDRESS) 93 error.SetErrorStringWithFormat ("Invalid start address string '%s'.\n", option_arg); 94 some_location_specified = true; 95 break; 96 case 'e': 97 end_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 0); 98 if (end_addr == LLDB_INVALID_ADDRESS) 99 end_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 16); 100 101 if (end_addr == LLDB_INVALID_ADDRESS) 102 error.SetErrorStringWithFormat ("Invalid end address string '%s'.\n", option_arg); 103 break; 104 some_location_specified = true; 105 case 'n': 106 func_name.assign (option_arg); 107 some_location_specified = true; 108 break; 109 110 case 'p': 111 at_pc = true; 112 some_location_specified = true; 113 break; 114 115 case 'l': 116 frame_line = true; 117 // Disassemble the current source line kind of implies showing mixed 118 // source code context. 119 show_mixed = true; 120 some_location_specified = true; 121 break; 122 123 case 'P': 124 plugin_name.assign (option_arg); 125 break; 126 127 case 'r': 128 raw = true; 129 break; 130 131 case 'f': 132 cur_function = true; 133 some_location_specified = true; 134 break; 135 136 case 'a': 137 arch.SetTriple (option_arg, m_interpreter.GetPlatform (true).get()); 138 break; 139 140 default: 141 error.SetErrorStringWithFormat("Unrecognized short option '%c'.\n", short_option); 142 break; 143 } 144 145 return error; 146 } 147 148 void 149 CommandObjectDisassemble::CommandOptions::OptionParsingStarting () 150 { 151 show_mixed = false; 152 show_bytes = false; 153 num_lines_context = 0; 154 num_instructions = 0; 155 func_name.clear(); 156 cur_function = false; 157 at_pc = false; 158 frame_line = false; 159 start_addr = LLDB_INVALID_ADDRESS; 160 end_addr = LLDB_INVALID_ADDRESS; 161 raw = false; 162 plugin_name.clear(); 163 arch.Clear(); 164 some_location_specified = false; 165 } 166 167 Error 168 CommandObjectDisassemble::CommandOptions::OptionParsingFinished () 169 { 170 if (!some_location_specified) 171 at_pc = true; 172 return Error(); 173 174 } 175 176 const OptionDefinition* 177 CommandObjectDisassemble::CommandOptions::GetDefinitions () 178 { 179 return g_option_table; 180 } 181 182 OptionDefinition 183 CommandObjectDisassemble::CommandOptions::g_option_table[] = 184 { 185 { LLDB_OPT_SET_ALL , false , "bytes", 'b', no_argument , NULL, 0, eArgTypeNone, "Show opcode bytes when disassembling."}, 186 { LLDB_OPT_SET_ALL , false , "context", 'C', required_argument , NULL, 0, eArgTypeNumLines, "Number of context lines of source to show."}, 187 { LLDB_OPT_SET_ALL , false , "mixed", 'm', no_argument , NULL, 0, eArgTypeNone, "Enable mixed source and assembly display."}, 188 { LLDB_OPT_SET_ALL , false , "raw", 'r', no_argument , NULL, 0, eArgTypeNone, "Print raw disassembly with no symbol information."}, 189 { LLDB_OPT_SET_ALL , false , "plugin", 'P', required_argument , NULL, 0, eArgTypePlugin, "Name of the disassembler plugin you want to use."}, 190 { LLDB_OPT_SET_ALL , false , "arch", 'a', required_argument , NULL, 0, eArgTypeArchitecture,"Specify the architecture to use from cross disassembly."}, 191 { LLDB_OPT_SET_1 | 192 LLDB_OPT_SET_2 , true , "start-address" , 's', required_argument , NULL, 0, eArgTypeStartAddress,"Address at which to start disassembling."}, 193 { LLDB_OPT_SET_1 , false , "end-address" , 'e', required_argument , NULL, 0, eArgTypeEndAddress, "Address at which to end disassembling."}, 194 { LLDB_OPT_SET_2 | 195 LLDB_OPT_SET_3 | 196 LLDB_OPT_SET_4 | 197 LLDB_OPT_SET_5 , false , "count", 'c', required_argument , NULL, 0, eArgTypeNumLines, "Number of instructions to display."}, 198 { LLDB_OPT_SET_3 , false , "name", 'n', required_argument , NULL, CommandCompletions::eSymbolCompletion, eArgTypeFunctionName, "Disassemble entire contents of the given function name."}, 199 { LLDB_OPT_SET_4 , false , "frame", 'f', no_argument , NULL, 0, eArgTypeNone, "Disassemble from the start of the current frame's function."}, 200 { LLDB_OPT_SET_5 , false , "pc", 'p', no_argument , NULL, 0, eArgTypeNone, "Disassemble around the current pc."}, 201 { LLDB_OPT_SET_6 , false , "line", 'l', no_argument , NULL, 0, eArgTypeNone, "Disassemble the current frame's current source line instructions if there debug line table information, else disasemble around the pc."}, 202 { 0 , false , NULL, 0, 0 , NULL, 0, eArgTypeNone, NULL } 203 }; 204 205 206 207 //------------------------------------------------------------------------- 208 // CommandObjectDisassemble 209 //------------------------------------------------------------------------- 210 211 CommandObjectDisassemble::CommandObjectDisassemble (CommandInterpreter &interpreter) : 212 CommandObject (interpreter, 213 "disassemble", 214 "Disassemble bytes in the current function, or elsewhere in the executable program as specified by the user.", 215 "disassemble [<cmd-options>]"), 216 m_options (interpreter) 217 { 218 } 219 220 CommandObjectDisassemble::~CommandObjectDisassemble() 221 { 222 } 223 224 bool 225 CommandObjectDisassemble::Execute 226 ( 227 Args& command, 228 CommandReturnObject &result 229 ) 230 { 231 Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get(); 232 if (target == NULL) 233 { 234 result.AppendError ("invalid target, create a debug target using the 'target create' command"); 235 result.SetStatus (eReturnStatusFailed); 236 return false; 237 } 238 if (!m_options.arch.IsValid()) 239 m_options.arch = target->GetArchitecture(); 240 241 if (!m_options.arch.IsValid()) 242 { 243 result.AppendError ("use the --arch option or set the target architecure to disassemble"); 244 result.SetStatus (eReturnStatusFailed); 245 return false; 246 } 247 248 const char *plugin_name = m_options.GetPluginName (); 249 Disassembler *disassembler = Disassembler::FindPlugin(m_options.arch, plugin_name); 250 251 if (disassembler == NULL) 252 { 253 if (plugin_name) 254 result.AppendErrorWithFormat ("Unable to find Disassembler plug-in named '%s' that supports the '%s' architecture.\n", 255 plugin_name, 256 m_options.arch.GetArchitectureName()); 257 else 258 result.AppendErrorWithFormat ("Unable to find Disassembler plug-in for the '%s' architecture.\n", 259 m_options.arch.GetArchitectureName()); 260 result.SetStatus (eReturnStatusFailed); 261 return false; 262 } 263 264 result.SetStatus (eReturnStatusSuccessFinishResult); 265 266 if (command.GetArgumentCount() != 0) 267 { 268 result.AppendErrorWithFormat ("\"disassemble\" arguments are specified as options.\n"); 269 GetOptions()->GenerateOptionUsage (result.GetErrorStream(), this); 270 result.SetStatus (eReturnStatusFailed); 271 return false; 272 } 273 274 if (m_options.show_mixed && m_options.num_lines_context == 0) 275 m_options.num_lines_context = 1; 276 277 ExecutionContext exe_ctx(m_interpreter.GetExecutionContext()); 278 // Always show the PC in the disassembly 279 uint32_t options = Disassembler::eOptionMarkPCAddress; 280 281 // Mark the source line for the current PC only if we are doing mixed source and assembly 282 if (m_options.show_mixed) 283 options |= Disassembler::eOptionMarkPCSourceLine; 284 285 if (m_options.show_bytes) 286 options |= Disassembler::eOptionShowBytes; 287 288 if (m_options.raw) 289 options |= Disassembler::eOptionRawOuput; 290 291 if (!m_options.func_name.empty()) 292 { 293 ConstString name(m_options.func_name.c_str()); 294 295 if (Disassembler::Disassemble (m_interpreter.GetDebugger(), 296 m_options.arch, 297 plugin_name, 298 exe_ctx, 299 name, 300 NULL, // Module * 301 m_options.num_instructions, 302 m_options.show_mixed ? m_options.num_lines_context : 0, 303 options, 304 result.GetOutputStream())) 305 { 306 result.SetStatus (eReturnStatusSuccessFinishResult); 307 } 308 else 309 { 310 result.AppendErrorWithFormat ("Unable to find symbol with name '%s'.\n", name.GetCString()); 311 result.SetStatus (eReturnStatusFailed); 312 } 313 } 314 else 315 { 316 AddressRange range; 317 if (m_options.frame_line) 318 { 319 if (exe_ctx.frame == NULL) 320 { 321 result.AppendError ("Cannot disassemble around the current line without a selected frame.\n"); 322 result.SetStatus (eReturnStatusFailed); 323 return false; 324 } 325 LineEntry pc_line_entry (exe_ctx.frame->GetSymbolContext(eSymbolContextLineEntry).line_entry); 326 if (pc_line_entry.IsValid()) 327 { 328 range = pc_line_entry.range; 329 } 330 else 331 { 332 m_options.at_pc = true; // No line entry, so just disassemble around the current pc 333 m_options.show_mixed = false; 334 } 335 } 336 else if (m_options.cur_function) 337 { 338 if (exe_ctx.frame == NULL) 339 { 340 result.AppendError ("Cannot disassemble around the current function without a selected frame.\n"); 341 result.SetStatus (eReturnStatusFailed); 342 return false; 343 } 344 Symbol *symbol = exe_ctx.frame->GetSymbolContext(eSymbolContextSymbol).symbol; 345 if (symbol) 346 range = symbol->GetAddressRangeRef(); 347 } 348 349 // Did the "m_options.frame_line" find a valid range already? If so 350 // skip the rest... 351 if (range.GetByteSize() == 0) 352 { 353 if (m_options.at_pc) 354 { 355 if (exe_ctx.frame == NULL) 356 { 357 result.AppendError ("Cannot disassemble around the current PC without a selected frame.\n"); 358 result.SetStatus (eReturnStatusFailed); 359 return false; 360 } 361 range.GetBaseAddress() = exe_ctx.frame->GetFrameCodeAddress(); 362 if (m_options.num_instructions == 0) 363 { 364 // Disassembling at the PC always disassembles some number of instructions (not the whole function). 365 m_options.num_instructions = DEFAULT_DISASM_NUM_INS; 366 } 367 } 368 else 369 { 370 range.GetBaseAddress().SetOffset (m_options.start_addr); 371 if (range.GetBaseAddress().IsValid()) 372 { 373 if (m_options.end_addr != LLDB_INVALID_ADDRESS) 374 { 375 if (m_options.end_addr <= m_options.start_addr) 376 { 377 result.AppendErrorWithFormat ("End address before start address.\n"); 378 result.SetStatus (eReturnStatusFailed); 379 return false; 380 } 381 range.SetByteSize (m_options.end_addr - m_options.start_addr); 382 } 383 } 384 } 385 } 386 387 if (m_options.num_instructions != 0) 388 { 389 if (!range.GetBaseAddress().IsValid()) 390 { 391 // The default action is to disassemble the current frame function. 392 if (exe_ctx.frame) 393 { 394 SymbolContext sc(exe_ctx.frame->GetSymbolContext(eSymbolContextFunction | eSymbolContextSymbol)); 395 if (sc.function) 396 range.GetBaseAddress() = sc.function->GetAddressRange().GetBaseAddress(); 397 else if (sc.symbol && sc.symbol->GetAddressRangePtr()) 398 range.GetBaseAddress() = sc.symbol->GetAddressRangePtr()->GetBaseAddress(); 399 else 400 range.GetBaseAddress() = exe_ctx.frame->GetFrameCodeAddress(); 401 } 402 403 if (!range.GetBaseAddress().IsValid()) 404 { 405 result.AppendError ("invalid frame"); 406 result.SetStatus (eReturnStatusFailed); 407 return false; 408 } 409 } 410 411 if (Disassembler::Disassemble (m_interpreter.GetDebugger(), 412 m_options.arch, 413 plugin_name, 414 exe_ctx, 415 range.GetBaseAddress(), 416 m_options.num_instructions, 417 m_options.show_mixed ? m_options.num_lines_context : 0, 418 options, 419 result.GetOutputStream())) 420 { 421 result.SetStatus (eReturnStatusSuccessFinishResult); 422 } 423 else 424 { 425 result.AppendErrorWithFormat ("Failed to disassemble memory at 0x%8.8llx.\n", m_options.start_addr); 426 result.SetStatus (eReturnStatusFailed); 427 } 428 } 429 else 430 { 431 if (!range.GetBaseAddress().IsValid()) 432 { 433 // The default action is to disassemble the current frame function. 434 if (exe_ctx.frame) 435 { 436 SymbolContext sc(exe_ctx.frame->GetSymbolContext(eSymbolContextFunction | eSymbolContextSymbol)); 437 if (sc.function) 438 range = sc.function->GetAddressRange(); 439 else if (sc.symbol && sc.symbol->GetAddressRangePtr()) 440 range = *sc.symbol->GetAddressRangePtr(); 441 else 442 range.GetBaseAddress() = exe_ctx.frame->GetFrameCodeAddress(); 443 } 444 else 445 { 446 result.AppendError ("invalid frame"); 447 result.SetStatus (eReturnStatusFailed); 448 return false; 449 } 450 } 451 if (range.GetByteSize() == 0) 452 range.SetByteSize(DEFAULT_DISASM_BYTE_SIZE); 453 454 if (Disassembler::Disassemble (m_interpreter.GetDebugger(), 455 m_options.arch, 456 plugin_name, 457 exe_ctx, 458 range, 459 m_options.num_instructions, 460 m_options.show_mixed ? m_options.num_lines_context : 0, 461 options, 462 result.GetOutputStream())) 463 { 464 result.SetStatus (eReturnStatusSuccessFinishResult); 465 } 466 else 467 { 468 result.AppendErrorWithFormat ("Failed to disassemble memory at 0x%8.8llx.\n", m_options.start_addr); 469 result.SetStatus (eReturnStatusFailed); 470 } 471 } 472 } 473 474 return result.Succeeded(); 475 } 476