1ebc09c36SJim Ingham //===-- CommandObjectSource.cpp ---------------------------------*- C++ -*-===// 2ebc09c36SJim Ingham // 3ebc09c36SJim Ingham // The LLVM Compiler Infrastructure 4ebc09c36SJim Ingham // 5ebc09c36SJim Ingham // This file is distributed under the University of Illinois Open Source 6ebc09c36SJim Ingham // License. See LICENSE.TXT for details. 7ebc09c36SJim Ingham // 8ebc09c36SJim Ingham //===----------------------------------------------------------------------===// 9ebc09c36SJim Ingham 1093a64300SDaniel Malea #include "lldb/lldb-python.h" 1193a64300SDaniel Malea 12ebc09c36SJim Ingham #include "CommandObjectCommands.h" 13ebc09c36SJim Ingham 14ebc09c36SJim Ingham // C Includes 15ebc09c36SJim Ingham // C++ Includes 16ebc09c36SJim Ingham // Other libraries and framework includes 170e5e5a79SGreg Clayton #include "llvm/ADT/StringRef.h" 180e5e5a79SGreg Clayton 19ebc09c36SJim Ingham // Project includes 20ebc09c36SJim Ingham #include "lldb/Core/Debugger.h" 21de164aaaSGreg Clayton #include "lldb/Core/InputReader.h" 22be93a35aSEnrico Granata #include "lldb/Core/InputReaderEZ.h" 23be93a35aSEnrico Granata #include "lldb/Core/StringList.h" 24de164aaaSGreg Clayton #include "lldb/Interpreter/Args.h" 257594f14fSEnrico Granata #include "lldb/Interpreter/CommandHistory.h" 26ebc09c36SJim Ingham #include "lldb/Interpreter/CommandInterpreter.h" 27de164aaaSGreg Clayton #include "lldb/Interpreter/CommandObjectRegexCommand.h" 28ebc09c36SJim Ingham #include "lldb/Interpreter/CommandReturnObject.h" 29012d4fcaSEnrico Granata #include "lldb/Interpreter/OptionValueBoolean.h" 307594f14fSEnrico Granata #include "lldb/Interpreter/OptionValueUInt64.h" 31ebc09c36SJim Ingham #include "lldb/Interpreter/Options.h" 3299f0b8f9SEnrico Granata #include "lldb/Interpreter/ScriptInterpreter.h" 3399f0b8f9SEnrico Granata #include "lldb/Interpreter/ScriptInterpreterPython.h" 34ebc09c36SJim Ingham 35ebc09c36SJim Ingham using namespace lldb; 36ebc09c36SJim Ingham using namespace lldb_private; 37ebc09c36SJim Ingham 38ebc09c36SJim Ingham //------------------------------------------------------------------------- 39ebc09c36SJim Ingham // CommandObjectCommandsSource 40ebc09c36SJim Ingham //------------------------------------------------------------------------- 41ebc09c36SJim Ingham 425a988416SJim Ingham class CommandObjectCommandsHistory : public CommandObjectParsed 43a5a97ebeSJim Ingham { 445a988416SJim Ingham public: 455a988416SJim Ingham CommandObjectCommandsHistory(CommandInterpreter &interpreter) : 465a988416SJim Ingham CommandObjectParsed (interpreter, 475a988416SJim Ingham "command history", 485a988416SJim Ingham "Dump the history of commands in this session.", 495a988416SJim Ingham NULL), 505a988416SJim Ingham m_options (interpreter) 515a988416SJim Ingham { 525a988416SJim Ingham } 535a988416SJim Ingham 545a988416SJim Ingham ~CommandObjectCommandsHistory () {} 555a988416SJim Ingham 565a988416SJim Ingham virtual Options * 575a988416SJim Ingham GetOptions () 585a988416SJim Ingham { 595a988416SJim Ingham return &m_options; 605a988416SJim Ingham } 615a988416SJim Ingham 625a988416SJim Ingham protected: 63a5a97ebeSJim Ingham 64a5a97ebeSJim Ingham class CommandOptions : public Options 65a5a97ebeSJim Ingham { 66a5a97ebeSJim Ingham public: 67a5a97ebeSJim Ingham 68a5a97ebeSJim Ingham CommandOptions (CommandInterpreter &interpreter) : 697594f14fSEnrico Granata Options (interpreter), 707594f14fSEnrico Granata m_start_idx(0), 717594f14fSEnrico Granata m_stop_idx(0), 727594f14fSEnrico Granata m_count(0), 7363123b64SEnrico Granata m_clear(false) 74a5a97ebeSJim Ingham { 75a5a97ebeSJim Ingham } 76a5a97ebeSJim Ingham 77a5a97ebeSJim Ingham virtual 78a5a97ebeSJim Ingham ~CommandOptions (){} 79a5a97ebeSJim Ingham 80a5a97ebeSJim Ingham virtual Error 81a5a97ebeSJim Ingham SetOptionValue (uint32_t option_idx, const char *option_arg) 82a5a97ebeSJim Ingham { 83a5a97ebeSJim Ingham Error error; 843bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 85a5a97ebeSJim Ingham 86a5a97ebeSJim Ingham switch (short_option) 87a5a97ebeSJim Ingham { 88a5a97ebeSJim Ingham case 'c': 897594f14fSEnrico Granata error = m_count.SetValueFromCString(option_arg,eVarSetOperationAssign); 90a5a97ebeSJim Ingham break; 91a5a97ebeSJim Ingham case 's': 927594f14fSEnrico Granata if (option_arg && strcmp("end", option_arg) == 0) 937594f14fSEnrico Granata { 947594f14fSEnrico Granata m_start_idx.SetCurrentValue(UINT64_MAX); 957594f14fSEnrico Granata m_start_idx.SetOptionWasSet(); 967594f14fSEnrico Granata } 977594f14fSEnrico Granata else 987594f14fSEnrico Granata error = m_start_idx.SetValueFromCString(option_arg,eVarSetOperationAssign); 997594f14fSEnrico Granata break; 1007594f14fSEnrico Granata case 'e': 1017594f14fSEnrico Granata error = m_stop_idx.SetValueFromCString(option_arg,eVarSetOperationAssign); 1027594f14fSEnrico Granata break; 10363123b64SEnrico Granata case 'C': 10463123b64SEnrico Granata m_clear.SetCurrentValue(true); 10563123b64SEnrico Granata m_clear.SetOptionWasSet(); 106a5a97ebeSJim Ingham break; 107a5a97ebeSJim Ingham default: 10886edbf41SGreg Clayton error.SetErrorStringWithFormat ("unrecognized option '%c'", short_option); 109a5a97ebeSJim Ingham break; 110a5a97ebeSJim Ingham } 111a5a97ebeSJim Ingham 112a5a97ebeSJim Ingham return error; 113a5a97ebeSJim Ingham } 114a5a97ebeSJim Ingham 115a5a97ebeSJim Ingham void 116a5a97ebeSJim Ingham OptionParsingStarting () 117a5a97ebeSJim Ingham { 1187594f14fSEnrico Granata m_start_idx.Clear(); 1197594f14fSEnrico Granata m_stop_idx.Clear(); 1207594f14fSEnrico Granata m_count.Clear(); 12163123b64SEnrico Granata m_clear.Clear(); 122a5a97ebeSJim Ingham } 123a5a97ebeSJim Ingham 124a5a97ebeSJim Ingham const OptionDefinition* 125a5a97ebeSJim Ingham GetDefinitions () 126a5a97ebeSJim Ingham { 127a5a97ebeSJim Ingham return g_option_table; 128a5a97ebeSJim Ingham } 129a5a97ebeSJim Ingham 130a5a97ebeSJim Ingham // Options table: Required for subclasses of Options. 131a5a97ebeSJim Ingham 132a5a97ebeSJim Ingham static OptionDefinition g_option_table[]; 133a5a97ebeSJim Ingham 134a5a97ebeSJim Ingham // Instance variables to hold the values for command options. 135a5a97ebeSJim Ingham 1367594f14fSEnrico Granata OptionValueUInt64 m_start_idx; 1377594f14fSEnrico Granata OptionValueUInt64 m_stop_idx; 1387594f14fSEnrico Granata OptionValueUInt64 m_count; 13963123b64SEnrico Granata OptionValueBoolean m_clear; 140a5a97ebeSJim Ingham }; 141a5a97ebeSJim Ingham 142a5a97ebeSJim Ingham bool 1435a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 144a5a97ebeSJim Ingham { 14563123b64SEnrico Granata if (m_options.m_clear.GetCurrentValue() && m_options.m_clear.OptionWasSet()) 1467594f14fSEnrico Granata { 1477594f14fSEnrico Granata m_interpreter.GetCommandHistory().Clear(); 1487594f14fSEnrico Granata result.SetStatus(lldb::eReturnStatusSuccessFinishNoResult); 1497594f14fSEnrico Granata } 1507594f14fSEnrico Granata else 1517594f14fSEnrico Granata { 1527594f14fSEnrico Granata if (m_options.m_start_idx.OptionWasSet() && m_options.m_stop_idx.OptionWasSet() && m_options.m_count.OptionWasSet()) 1537594f14fSEnrico Granata { 1547594f14fSEnrico Granata result.AppendError("--count, --start-index and --end-index cannot be all specified in the same invocation"); 1557594f14fSEnrico Granata result.SetStatus(lldb::eReturnStatusFailed); 1567594f14fSEnrico Granata } 1577594f14fSEnrico Granata else 1587594f14fSEnrico Granata { 1597594f14fSEnrico Granata std::pair<bool,uint64_t> start_idx = {m_options.m_start_idx.OptionWasSet(),m_options.m_start_idx.GetCurrentValue()}; 1607594f14fSEnrico Granata std::pair<bool,uint64_t> stop_idx = {m_options.m_stop_idx.OptionWasSet(),m_options.m_stop_idx.GetCurrentValue()}; 1617594f14fSEnrico Granata std::pair<bool,uint64_t> count = {m_options.m_count.OptionWasSet(),m_options.m_count.GetCurrentValue()}; 162a5a97ebeSJim Ingham 1637594f14fSEnrico Granata const CommandHistory& history(m_interpreter.GetCommandHistory()); 1647594f14fSEnrico Granata 1657594f14fSEnrico Granata if (start_idx.first && start_idx.second == UINT64_MAX) 1667594f14fSEnrico Granata { 1677594f14fSEnrico Granata if (count.first) 1687594f14fSEnrico Granata { 1697594f14fSEnrico Granata start_idx.second = history.GetSize() - count.second; 1707594f14fSEnrico Granata stop_idx.second = history.GetSize() - 1; 1717594f14fSEnrico Granata } 1727594f14fSEnrico Granata else if (stop_idx.first) 1737594f14fSEnrico Granata { 1747594f14fSEnrico Granata start_idx.second = stop_idx.second; 1757594f14fSEnrico Granata stop_idx.second = history.GetSize() - 1; 1767594f14fSEnrico Granata } 1777594f14fSEnrico Granata else 1787594f14fSEnrico Granata { 1797594f14fSEnrico Granata start_idx.second = 0; 1807594f14fSEnrico Granata stop_idx.second = history.GetSize() - 1; 1817594f14fSEnrico Granata } 1827594f14fSEnrico Granata } 1837594f14fSEnrico Granata else 1847594f14fSEnrico Granata { 1857594f14fSEnrico Granata if (!start_idx.first && !stop_idx.first && !count.first) 1867594f14fSEnrico Granata { 1877594f14fSEnrico Granata start_idx.second = 0; 1887594f14fSEnrico Granata stop_idx.second = history.GetSize() - 1; 1897594f14fSEnrico Granata } 1907594f14fSEnrico Granata else if (start_idx.first) 1917594f14fSEnrico Granata { 1927594f14fSEnrico Granata if (count.first) 1937594f14fSEnrico Granata { 1947594f14fSEnrico Granata stop_idx.second = start_idx.second + count.second - 1; 1957594f14fSEnrico Granata } 1967594f14fSEnrico Granata else if (!stop_idx.first) 1977594f14fSEnrico Granata { 1987594f14fSEnrico Granata stop_idx.second = history.GetSize() - 1; 1997594f14fSEnrico Granata } 2007594f14fSEnrico Granata } 2017594f14fSEnrico Granata else if (stop_idx.first) 2027594f14fSEnrico Granata { 2037594f14fSEnrico Granata if (count.first) 2047594f14fSEnrico Granata { 2057594f14fSEnrico Granata if (stop_idx.second >= count.second) 2067594f14fSEnrico Granata start_idx.second = stop_idx.second - count.second + 1; 2077594f14fSEnrico Granata else 2087594f14fSEnrico Granata start_idx.second = 0; 2097594f14fSEnrico Granata } 2107594f14fSEnrico Granata } 2117594f14fSEnrico Granata else /* if (count.first) */ 2127594f14fSEnrico Granata { 2137594f14fSEnrico Granata start_idx.second = 0; 2147594f14fSEnrico Granata stop_idx.second = count.second - 1; 2157594f14fSEnrico Granata } 2167594f14fSEnrico Granata } 2177594f14fSEnrico Granata history.Dump(result.GetOutputStream(), start_idx.second, stop_idx.second); 2187594f14fSEnrico Granata } 2197594f14fSEnrico Granata } 220a5a97ebeSJim Ingham return result.Succeeded(); 221a5a97ebeSJim Ingham 222a5a97ebeSJim Ingham } 2235a988416SJim Ingham 2245a988416SJim Ingham CommandOptions m_options; 225a5a97ebeSJim Ingham }; 226a5a97ebeSJim Ingham 227a5a97ebeSJim Ingham OptionDefinition 228a5a97ebeSJim Ingham CommandObjectCommandsHistory::CommandOptions::g_option_table[] = 229a5a97ebeSJim Ingham { 230a5a97ebeSJim Ingham { LLDB_OPT_SET_1, false, "count", 'c', required_argument, NULL, 0, eArgTypeUnsignedInteger, "How many history commands to print."}, 2317594f14fSEnrico Granata { LLDB_OPT_SET_1, false, "start-index", 's', required_argument, NULL, 0, eArgTypeUnsignedInteger, "Index at which to start printing history commands (or end to mean tail mode)."}, 232a5a97ebeSJim Ingham { LLDB_OPT_SET_1, false, "end-index", 'e', required_argument, NULL, 0, eArgTypeUnsignedInteger, "Index at which to stop printing history commands."}, 23363123b64SEnrico Granata { LLDB_OPT_SET_2, false, "clear", 'C', no_argument, NULL, 0, eArgTypeBoolean, "Clears the current command history."}, 234a5a97ebeSJim Ingham { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 235a5a97ebeSJim Ingham }; 236a5a97ebeSJim Ingham 237a5a97ebeSJim Ingham 238a5a97ebeSJim Ingham //------------------------------------------------------------------------- 239a5a97ebeSJim Ingham // CommandObjectCommandsSource 240a5a97ebeSJim Ingham //------------------------------------------------------------------------- 241a5a97ebeSJim Ingham 2425a988416SJim Ingham class CommandObjectCommandsSource : public CommandObjectParsed 243ebc09c36SJim Ingham { 2445a988416SJim Ingham public: 2455a988416SJim Ingham CommandObjectCommandsSource(CommandInterpreter &interpreter) : 2465a988416SJim Ingham CommandObjectParsed (interpreter, 2475a988416SJim Ingham "command source", 2485a988416SJim Ingham "Read in debugger commands from the file <filename> and execute them.", 2495a988416SJim Ingham NULL), 2505a988416SJim Ingham m_options (interpreter) 2515a988416SJim Ingham { 2525a988416SJim Ingham CommandArgumentEntry arg; 2535a988416SJim Ingham CommandArgumentData file_arg; 2545a988416SJim Ingham 2555a988416SJim Ingham // Define the first (and only) variant of this arg. 2565a988416SJim Ingham file_arg.arg_type = eArgTypeFilename; 2575a988416SJim Ingham file_arg.arg_repetition = eArgRepeatPlain; 2585a988416SJim Ingham 2595a988416SJim Ingham // There is only one variant this argument could be; put it into the argument entry. 2605a988416SJim Ingham arg.push_back (file_arg); 2615a988416SJim Ingham 2625a988416SJim Ingham // Push the data for the first argument into the m_arguments vector. 2635a988416SJim Ingham m_arguments.push_back (arg); 2645a988416SJim Ingham } 2655a988416SJim Ingham 2665a988416SJim Ingham ~CommandObjectCommandsSource () {} 2675a988416SJim Ingham 2685a988416SJim Ingham virtual const char* 2695a988416SJim Ingham GetRepeatCommand (Args ¤t_command_args, uint32_t index) 2705a988416SJim Ingham { 2715a988416SJim Ingham return ""; 2725a988416SJim Ingham } 2735a988416SJim Ingham 274c7bece56SGreg Clayton virtual int 2755a988416SJim Ingham HandleArgumentCompletion (Args &input, 2765a988416SJim Ingham int &cursor_index, 2775a988416SJim Ingham int &cursor_char_position, 2785a988416SJim Ingham OptionElementVector &opt_element_vector, 2795a988416SJim Ingham int match_start_point, 2805a988416SJim Ingham int max_return_elements, 2815a988416SJim Ingham bool &word_complete, 2825a988416SJim Ingham StringList &matches) 2835a988416SJim Ingham { 2845a988416SJim Ingham std::string completion_str (input.GetArgumentAtIndex(cursor_index)); 2855a988416SJim Ingham completion_str.erase (cursor_char_position); 2865a988416SJim Ingham 2875a988416SJim Ingham CommandCompletions::InvokeCommonCompletionCallbacks (m_interpreter, 2885a988416SJim Ingham CommandCompletions::eDiskFileCompletion, 2895a988416SJim Ingham completion_str.c_str(), 2905a988416SJim Ingham match_start_point, 2915a988416SJim Ingham max_return_elements, 2925a988416SJim Ingham NULL, 2935a988416SJim Ingham word_complete, 2945a988416SJim Ingham matches); 2955a988416SJim Ingham return matches.GetSize(); 2965a988416SJim Ingham } 2975a988416SJim Ingham 2985a988416SJim Ingham virtual Options * 2995a988416SJim Ingham GetOptions () 3005a988416SJim Ingham { 3015a988416SJim Ingham return &m_options; 3025a988416SJim Ingham } 3035a988416SJim Ingham 3045a988416SJim Ingham protected: 305e16c50a1SJim Ingham 306e16c50a1SJim Ingham class CommandOptions : public Options 307e16c50a1SJim Ingham { 308e16c50a1SJim Ingham public: 309e16c50a1SJim Ingham 310eb0103f2SGreg Clayton CommandOptions (CommandInterpreter &interpreter) : 311012d4fcaSEnrico Granata Options (interpreter), 312012d4fcaSEnrico Granata m_stop_on_error (true) 313eb0103f2SGreg Clayton { 314eb0103f2SGreg Clayton } 315e16c50a1SJim Ingham 316e16c50a1SJim Ingham virtual 317e16c50a1SJim Ingham ~CommandOptions (){} 318e16c50a1SJim Ingham 319e16c50a1SJim Ingham virtual Error 320f6b8b581SGreg Clayton SetOptionValue (uint32_t option_idx, const char *option_arg) 321e16c50a1SJim Ingham { 322e16c50a1SJim Ingham Error error; 3233bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 324e16c50a1SJim Ingham bool success; 325e16c50a1SJim Ingham 326e16c50a1SJim Ingham switch (short_option) 327e16c50a1SJim Ingham { 328e16c50a1SJim Ingham case 'e': 32915571f15SEnrico Granata error = m_stop_on_error.SetValueFromCString(option_arg); 330e16c50a1SJim Ingham break; 331e16c50a1SJim Ingham case 'c': 332e16c50a1SJim Ingham m_stop_on_continue = Args::StringToBoolean(option_arg, true, &success); 333e16c50a1SJim Ingham if (!success) 33486edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid value for stop-on-continue: %s", option_arg); 335e16c50a1SJim Ingham break; 336e16c50a1SJim Ingham default: 33786edbf41SGreg Clayton error.SetErrorStringWithFormat ("unrecognized option '%c'", short_option); 338e16c50a1SJim Ingham break; 339e16c50a1SJim Ingham } 340e16c50a1SJim Ingham 341e16c50a1SJim Ingham return error; 342e16c50a1SJim Ingham } 343e16c50a1SJim Ingham 344e16c50a1SJim Ingham void 345f6b8b581SGreg Clayton OptionParsingStarting () 346e16c50a1SJim Ingham { 347012d4fcaSEnrico Granata m_stop_on_error.Clear(); 348e16c50a1SJim Ingham m_stop_on_continue = true; 349e16c50a1SJim Ingham } 350e16c50a1SJim Ingham 351e0d378b3SGreg Clayton const OptionDefinition* 352e16c50a1SJim Ingham GetDefinitions () 353e16c50a1SJim Ingham { 354e16c50a1SJim Ingham return g_option_table; 355e16c50a1SJim Ingham } 356e16c50a1SJim Ingham 357e16c50a1SJim Ingham // Options table: Required for subclasses of Options. 358e16c50a1SJim Ingham 359e0d378b3SGreg Clayton static OptionDefinition g_option_table[]; 360e16c50a1SJim Ingham 361e16c50a1SJim Ingham // Instance variables to hold the values for command options. 362e16c50a1SJim Ingham 363012d4fcaSEnrico Granata OptionValueBoolean m_stop_on_error; 364e16c50a1SJim Ingham bool m_stop_on_continue; 365e16c50a1SJim Ingham }; 366e16c50a1SJim Ingham 367ebc09c36SJim Ingham bool 3685a988416SJim Ingham DoExecute(Args& command, CommandReturnObject &result) 369ebc09c36SJim Ingham { 370c7bece56SGreg Clayton const size_t argc = command.GetArgumentCount(); 371ebc09c36SJim Ingham if (argc == 1) 372ebc09c36SJim Ingham { 3735a988416SJim Ingham const char *filename = command.GetArgumentAtIndex(0); 374ebc09c36SJim Ingham 375ebc09c36SJim Ingham result.AppendMessageWithFormat ("Executing commands in '%s'.\n", filename); 376ebc09c36SJim Ingham 3771ee3853fSJohnny Chen FileSpec cmd_file (filename, true); 378e16c50a1SJim Ingham ExecutionContext *exe_ctx = NULL; // Just use the default context. 379e16c50a1SJim Ingham bool echo_commands = true; 380e16c50a1SJim Ingham bool print_results = true; 381012d4fcaSEnrico Granata bool stop_on_error = m_options.m_stop_on_error.OptionWasSet() ? (bool)m_options.m_stop_on_error : m_interpreter.GetStopCmdSourceOnError(); 382ebc09c36SJim Ingham 383e16c50a1SJim Ingham m_interpreter.HandleCommandsFromFile (cmd_file, 384e16c50a1SJim Ingham exe_ctx, 385e16c50a1SJim Ingham m_options.m_stop_on_continue, 386012d4fcaSEnrico Granata stop_on_error, 387e16c50a1SJim Ingham echo_commands, 388e16c50a1SJim Ingham print_results, 3895f5ab602SEnrico Granata eLazyBoolCalculate, 390e16c50a1SJim Ingham result); 391ebc09c36SJim Ingham } 392ebc09c36SJim Ingham else 393ebc09c36SJim Ingham { 394ebc09c36SJim Ingham result.AppendErrorWithFormat("'%s' takes exactly one executable filename argument.\n", GetCommandName()); 395ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 396ebc09c36SJim Ingham } 397ebc09c36SJim Ingham return result.Succeeded(); 398ebc09c36SJim Ingham 399ebc09c36SJim Ingham } 4005a988416SJim Ingham CommandOptions m_options; 401ebc09c36SJim Ingham }; 402ebc09c36SJim Ingham 403e0d378b3SGreg Clayton OptionDefinition 404e16c50a1SJim Ingham CommandObjectCommandsSource::CommandOptions::g_option_table[] = 405e16c50a1SJim Ingham { 406e16c50a1SJim Ingham { LLDB_OPT_SET_ALL, false, "stop-on-error", 'e', required_argument, NULL, 0, eArgTypeBoolean, "If true, stop executing commands on error."}, 407e16c50a1SJim Ingham { LLDB_OPT_SET_ALL, false, "stop-on-continue", 'c', required_argument, NULL, 0, eArgTypeBoolean, "If true, stop executing commands on continue."}, 408e16c50a1SJim Ingham { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 409e16c50a1SJim Ingham }; 410e16c50a1SJim Ingham 411ebc09c36SJim Ingham #pragma mark CommandObjectCommandsAlias 412ebc09c36SJim Ingham //------------------------------------------------------------------------- 413ebc09c36SJim Ingham // CommandObjectCommandsAlias 414ebc09c36SJim Ingham //------------------------------------------------------------------------- 415ebc09c36SJim Ingham 416be93a35aSEnrico Granata static const char *g_python_command_instructions = "Enter your Python command(s). Type 'DONE' to end.\n" 417be93a35aSEnrico Granata "You must define a Python function with this signature:\n" 41840d55710SEnrico Granata "def my_command_impl(debugger, args, result, internal_dict):"; 419be93a35aSEnrico Granata 420be93a35aSEnrico Granata 4215a988416SJim Ingham class CommandObjectCommandsAlias : public CommandObjectRaw 422ebc09c36SJim Ingham { 423be93a35aSEnrico Granata 424be93a35aSEnrico Granata 425ebc09c36SJim Ingham public: 426a7015092SGreg Clayton CommandObjectCommandsAlias (CommandInterpreter &interpreter) : 4275a988416SJim Ingham CommandObjectRaw (interpreter, 4280e5e5a79SGreg Clayton "command alias", 429e3d26315SCaroline Tice "Allow users to define their own debugger command abbreviations.", 430405fe67fSCaroline Tice NULL) 431ebc09c36SJim Ingham { 432ebc09c36SJim Ingham SetHelpLong( 433ebc09c36SJim Ingham "'alias' allows the user to create a short-cut or abbreviation for long \n\ 434ebc09c36SJim Ingham commands, multi-word commands, and commands that take particular options. \n\ 435ebc09c36SJim Ingham Below are some simple examples of how one might use the 'alias' command: \n\ 43669c12ccbSJason Molenda \n 'command alias sc script' // Creates the abbreviation 'sc' for the 'script' \n\ 437ebc09c36SJim Ingham // command. \n\ 43869c12ccbSJason Molenda 'command alias bp breakpoint' // Creates the abbreviation 'bp' for the 'breakpoint' \n\ 439ebc09c36SJim Ingham // command. Since breakpoint commands are two-word \n\ 440ebc09c36SJim Ingham // commands, the user will still need to enter the \n\ 441ebc09c36SJim Ingham // second word after 'bp', e.g. 'bp enable' or \n\ 442ebc09c36SJim Ingham // 'bp delete'. \n\ 44369c12ccbSJason Molenda 'command alias bpl breakpoint list' // Creates the abbreviation 'bpl' for the \n\ 444ebc09c36SJim Ingham // two-word command 'breakpoint list'. \n\ 445ebc09c36SJim Ingham \nAn alias can include some options for the command, with the values either \n\ 446ebc09c36SJim Ingham filled in at the time the alias is created, or specified as positional \n\ 447ebc09c36SJim Ingham arguments, to be filled in when the alias is invoked. The following example \n\ 448ebc09c36SJim Ingham shows how to create aliases with options: \n\ 449ebc09c36SJim Ingham \n\ 45069c12ccbSJason Molenda 'command alias bfl breakpoint set -f %1 -l %2' \n\ 451ebc09c36SJim Ingham \nThis creates the abbreviation 'bfl' (for break-file-line), with the -f and -l \n\ 452ebc09c36SJim Ingham options already part of the alias. So if the user wants to set a breakpoint \n\ 453ebc09c36SJim Ingham by file and line without explicitly having to use the -f and -l options, the \n\ 454ebc09c36SJim Ingham user can now use 'bfl' instead. The '%1' and '%2' are positional placeholders \n\ 455ebc09c36SJim Ingham for the actual arguments that will be passed when the alias command is used. \n\ 456ebc09c36SJim Ingham The number in the placeholder refers to the position/order the actual value \n\ 45781ded935SJim Ingham occupies when the alias is used. All the occurrences of '%1' in the alias \n\ 458ebc09c36SJim Ingham will be replaced with the first argument, all the occurrences of '%2' in the \n\ 459ebc09c36SJim Ingham alias will be replaced with the second argument, and so on. This also allows \n\ 460ebc09c36SJim Ingham actual arguments to be used multiple times within an alias (see 'process \n\ 46181ded935SJim Ingham launch' example below). \n\ 46281ded935SJim Ingham Note: the positional arguments must substitute as whole words in the resultant\n\ 46381ded935SJim Ingham command, so you can't at present do something like:\n\ 46481ded935SJim Ingham \n\ 46569c12ccbSJason Molenda command alias bcppfl breakpoint set -f %1.cpp -l %2\n\ 46681ded935SJim Ingham \n\ 46781ded935SJim Ingham to get the file extension \".cpp\" automatically appended. For more complex\n\ 46881ded935SJim Ingham aliasing, use the \"command regex\" command instead.\n\ 46981ded935SJim Ingham \nSo in the 'bfl' case, the actual file value will be \n\ 470ebc09c36SJim Ingham filled in with the first argument following 'bfl' and the actual line number \n\ 471ebc09c36SJim Ingham value will be filled in with the second argument. The user would use this \n\ 472ebc09c36SJim Ingham alias as follows: \n\ 47369c12ccbSJason Molenda \n (lldb) command alias bfl breakpoint set -f %1 -l %2 \n\ 474ebc09c36SJim Ingham <... some time later ...> \n\ 47509799af6SCaroline Tice (lldb) bfl my-file.c 137 \n\ 476ebc09c36SJim Ingham \nThis would be the same as if the user had entered \n\ 477ebc09c36SJim Ingham 'breakpoint set -f my-file.c -l 137'. \n\ 478ebc09c36SJim Ingham \nAnother example: \n\ 47969c12ccbSJason Molenda \n (lldb) command alias pltty process launch -s -o %1 -e %1 \n\ 48009799af6SCaroline Tice (lldb) pltty /dev/tty0 \n\ 481ebc09c36SJim Ingham // becomes 'process launch -s -o /dev/tty0 -e /dev/tty0' \n\ 482ebc09c36SJim Ingham \nIf the user always wanted to pass the same value to a particular option, the \n\ 483ebc09c36SJim Ingham alias could be defined with that value directly in the alias as a constant, \n\ 484ebc09c36SJim Ingham rather than using a positional placeholder: \n\ 48569c12ccbSJason Molenda \n command alias bl3 breakpoint set -f %1 -l 3 // Always sets a breakpoint on line \n\ 486ebc09c36SJim Ingham // 3 of whatever file is indicated. \n"); 487ebc09c36SJim Ingham 488405fe67fSCaroline Tice CommandArgumentEntry arg1; 489405fe67fSCaroline Tice CommandArgumentEntry arg2; 490405fe67fSCaroline Tice CommandArgumentEntry arg3; 491405fe67fSCaroline Tice CommandArgumentData alias_arg; 492405fe67fSCaroline Tice CommandArgumentData cmd_arg; 493405fe67fSCaroline Tice CommandArgumentData options_arg; 494405fe67fSCaroline Tice 495405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 496405fe67fSCaroline Tice alias_arg.arg_type = eArgTypeAliasName; 497405fe67fSCaroline Tice alias_arg.arg_repetition = eArgRepeatPlain; 498405fe67fSCaroline Tice 499405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 500405fe67fSCaroline Tice arg1.push_back (alias_arg); 501405fe67fSCaroline Tice 502405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 503405fe67fSCaroline Tice cmd_arg.arg_type = eArgTypeCommandName; 504405fe67fSCaroline Tice cmd_arg.arg_repetition = eArgRepeatPlain; 505405fe67fSCaroline Tice 506405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 507405fe67fSCaroline Tice arg2.push_back (cmd_arg); 508405fe67fSCaroline Tice 509405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 510405fe67fSCaroline Tice options_arg.arg_type = eArgTypeAliasOptions; 511405fe67fSCaroline Tice options_arg.arg_repetition = eArgRepeatOptional; 512405fe67fSCaroline Tice 513405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 514405fe67fSCaroline Tice arg3.push_back (options_arg); 515405fe67fSCaroline Tice 516405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 517405fe67fSCaroline Tice m_arguments.push_back (arg1); 518405fe67fSCaroline Tice m_arguments.push_back (arg2); 519405fe67fSCaroline Tice m_arguments.push_back (arg3); 520ebc09c36SJim Ingham } 521ebc09c36SJim Ingham 522ebc09c36SJim Ingham ~CommandObjectCommandsAlias () 523ebc09c36SJim Ingham { 524ebc09c36SJim Ingham } 525ebc09c36SJim Ingham 5265a988416SJim Ingham protected: 5275a988416SJim Ingham virtual bool 5285a988416SJim Ingham DoExecute (const char *raw_command_line, CommandReturnObject &result) 529844d2303SCaroline Tice { 530844d2303SCaroline Tice Args args (raw_command_line); 531844d2303SCaroline Tice std::string raw_command_string (raw_command_line); 532844d2303SCaroline Tice 533844d2303SCaroline Tice size_t argc = args.GetArgumentCount(); 534844d2303SCaroline Tice 535844d2303SCaroline Tice if (argc < 2) 536844d2303SCaroline Tice { 537844d2303SCaroline Tice result.AppendError ("'alias' requires at least two arguments"); 538844d2303SCaroline Tice result.SetStatus (eReturnStatusFailed); 539844d2303SCaroline Tice return false; 540844d2303SCaroline Tice } 541844d2303SCaroline Tice 542844d2303SCaroline Tice // Get the alias command. 543844d2303SCaroline Tice 544844d2303SCaroline Tice const std::string alias_command = args.GetArgumentAtIndex (0); 545844d2303SCaroline Tice 546844d2303SCaroline Tice // Strip the new alias name off 'raw_command_string' (leave it on args, which gets passed to 'Execute', which 547844d2303SCaroline Tice // does the stripping itself. 548844d2303SCaroline Tice size_t pos = raw_command_string.find (alias_command); 549844d2303SCaroline Tice if (pos == 0) 550844d2303SCaroline Tice { 551844d2303SCaroline Tice raw_command_string = raw_command_string.substr (alias_command.size()); 552844d2303SCaroline Tice pos = raw_command_string.find_first_not_of (' '); 553844d2303SCaroline Tice if ((pos != std::string::npos) && (pos > 0)) 554844d2303SCaroline Tice raw_command_string = raw_command_string.substr (pos); 555844d2303SCaroline Tice } 556844d2303SCaroline Tice else 557844d2303SCaroline Tice { 558844d2303SCaroline Tice result.AppendError ("Error parsing command string. No alias created."); 559844d2303SCaroline Tice result.SetStatus (eReturnStatusFailed); 560844d2303SCaroline Tice return false; 561844d2303SCaroline Tice } 562844d2303SCaroline Tice 563844d2303SCaroline Tice 564844d2303SCaroline Tice // Verify that the command is alias-able. 565844d2303SCaroline Tice if (m_interpreter.CommandExists (alias_command.c_str())) 566844d2303SCaroline Tice { 567844d2303SCaroline Tice result.AppendErrorWithFormat ("'%s' is a permanent debugger command and cannot be redefined.\n", 568844d2303SCaroline Tice alias_command.c_str()); 569844d2303SCaroline Tice result.SetStatus (eReturnStatusFailed); 570844d2303SCaroline Tice return false; 571844d2303SCaroline Tice } 572844d2303SCaroline Tice 573844d2303SCaroline Tice // Get CommandObject that is being aliased. The command name is read from the front of raw_command_string. 574844d2303SCaroline Tice // raw_command_string is returned with the name of the command object stripped off the front. 575844d2303SCaroline Tice CommandObject *cmd_obj = m_interpreter.GetCommandObjectForCommand (raw_command_string); 576844d2303SCaroline Tice 577844d2303SCaroline Tice if (!cmd_obj) 578844d2303SCaroline Tice { 57986edbf41SGreg Clayton result.AppendErrorWithFormat ("invalid command given to 'alias'. '%s' does not begin with a valid command." 580844d2303SCaroline Tice " No alias created.", raw_command_string.c_str()); 581844d2303SCaroline Tice result.SetStatus (eReturnStatusFailed); 582844d2303SCaroline Tice return false; 583844d2303SCaroline Tice } 584844d2303SCaroline Tice else if (!cmd_obj->WantsRawCommandString ()) 585844d2303SCaroline Tice { 586844d2303SCaroline Tice // Note that args was initialized with the original command, and has not been updated to this point. 587844d2303SCaroline Tice // Therefore can we pass it to the version of Execute that does not need/expect raw input in the alias. 5885a988416SJim Ingham return HandleAliasingNormalCommand (args, result); 589844d2303SCaroline Tice } 590844d2303SCaroline Tice else 591844d2303SCaroline Tice { 5925a988416SJim Ingham return HandleAliasingRawCommand (alias_command, raw_command_string, *cmd_obj, result); 5935a988416SJim Ingham } 5945a988416SJim Ingham return result.Succeeded(); 5955a988416SJim Ingham } 5965a988416SJim Ingham 5975a988416SJim Ingham bool 5985a988416SJim Ingham HandleAliasingRawCommand (const std::string &alias_command, std::string &raw_command_string, CommandObject &cmd_obj, CommandReturnObject &result) 5995a988416SJim Ingham { 600844d2303SCaroline Tice // Verify & handle any options/arguments passed to the alias command 601844d2303SCaroline Tice 602844d2303SCaroline Tice OptionArgVectorSP option_arg_vector_sp = OptionArgVectorSP (new OptionArgVector); 603844d2303SCaroline Tice OptionArgVector *option_arg_vector = option_arg_vector_sp.get(); 604844d2303SCaroline Tice 6055a988416SJim Ingham CommandObjectSP cmd_obj_sp = m_interpreter.GetCommandSPExact (cmd_obj.GetCommandName(), false); 606844d2303SCaroline Tice 607ca90c47eSCaroline Tice if (!m_interpreter.ProcessAliasOptionsArgs (cmd_obj_sp, raw_command_string.c_str(), option_arg_vector_sp)) 608844d2303SCaroline Tice { 609844d2303SCaroline Tice result.AppendError ("Unable to create requested alias.\n"); 610ca90c47eSCaroline Tice result.SetStatus (eReturnStatusFailed); 611844d2303SCaroline Tice return false; 612844d2303SCaroline Tice } 613844d2303SCaroline Tice 614844d2303SCaroline Tice // Create the alias 615844d2303SCaroline Tice if (m_interpreter.AliasExists (alias_command.c_str()) 616844d2303SCaroline Tice || m_interpreter.UserCommandExists (alias_command.c_str())) 617844d2303SCaroline Tice { 618844d2303SCaroline Tice OptionArgVectorSP temp_option_arg_sp (m_interpreter.GetAliasOptions (alias_command.c_str())); 619844d2303SCaroline Tice if (temp_option_arg_sp.get()) 620844d2303SCaroline Tice { 621844d2303SCaroline Tice if (option_arg_vector->size() == 0) 622844d2303SCaroline Tice m_interpreter.RemoveAliasOptions (alias_command.c_str()); 623844d2303SCaroline Tice } 624844d2303SCaroline Tice result.AppendWarningWithFormat ("Overwriting existing definition for '%s'.\n", 625844d2303SCaroline Tice alias_command.c_str()); 626844d2303SCaroline Tice } 627844d2303SCaroline Tice 628472362e6SCaroline Tice if (cmd_obj_sp) 629472362e6SCaroline Tice { 630844d2303SCaroline Tice m_interpreter.AddAlias (alias_command.c_str(), cmd_obj_sp); 631844d2303SCaroline Tice if (option_arg_vector->size() > 0) 632844d2303SCaroline Tice m_interpreter.AddOrReplaceAliasOptions (alias_command.c_str(), option_arg_vector_sp); 633844d2303SCaroline Tice result.SetStatus (eReturnStatusSuccessFinishNoResult); 634844d2303SCaroline Tice } 635472362e6SCaroline Tice else 636472362e6SCaroline Tice { 637472362e6SCaroline Tice result.AppendError ("Unable to create requested alias.\n"); 638472362e6SCaroline Tice result.SetStatus (eReturnStatusFailed); 639472362e6SCaroline Tice } 640844d2303SCaroline Tice return result.Succeeded (); 641844d2303SCaroline Tice } 642ebc09c36SJim Ingham 643ebc09c36SJim Ingham bool 6445a988416SJim Ingham HandleAliasingNormalCommand (Args& args, CommandReturnObject &result) 645ebc09c36SJim Ingham { 646867b185dSCaroline Tice size_t argc = args.GetArgumentCount(); 647ebc09c36SJim Ingham 648ebc09c36SJim Ingham if (argc < 2) 649ebc09c36SJim Ingham { 650ebc09c36SJim Ingham result.AppendError ("'alias' requires at least two arguments"); 651ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 652ebc09c36SJim Ingham return false; 653ebc09c36SJim Ingham } 654ebc09c36SJim Ingham 655ebc09c36SJim Ingham const std::string alias_command = args.GetArgumentAtIndex(0); 656ebc09c36SJim Ingham const std::string actual_command = args.GetArgumentAtIndex(1); 657ebc09c36SJim Ingham 658ebc09c36SJim Ingham args.Shift(); // Shift the alias command word off the argument vector. 659ebc09c36SJim Ingham args.Shift(); // Shift the old command word off the argument vector. 660ebc09c36SJim Ingham 661ebc09c36SJim Ingham // Verify that the command is alias'able, and get the appropriate command object. 662ebc09c36SJim Ingham 663a7015092SGreg Clayton if (m_interpreter.CommandExists (alias_command.c_str())) 664ebc09c36SJim Ingham { 665ebc09c36SJim Ingham result.AppendErrorWithFormat ("'%s' is a permanent debugger command and cannot be redefined.\n", 666ebc09c36SJim Ingham alias_command.c_str()); 667ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 668ebc09c36SJim Ingham } 669ebc09c36SJim Ingham else 670ebc09c36SJim Ingham { 671a7015092SGreg Clayton CommandObjectSP command_obj_sp(m_interpreter.GetCommandSPExact (actual_command.c_str(), true)); 672ebc09c36SJim Ingham CommandObjectSP subcommand_obj_sp; 673ebc09c36SJim Ingham bool use_subcommand = false; 674ebc09c36SJim Ingham if (command_obj_sp.get()) 675ebc09c36SJim Ingham { 676ebc09c36SJim Ingham CommandObject *cmd_obj = command_obj_sp.get(); 677c982c768SGreg Clayton CommandObject *sub_cmd_obj = NULL; 678ebc09c36SJim Ingham OptionArgVectorSP option_arg_vector_sp = OptionArgVectorSP (new OptionArgVector); 679ebc09c36SJim Ingham OptionArgVector *option_arg_vector = option_arg_vector_sp.get(); 680ebc09c36SJim Ingham 681844d2303SCaroline Tice while (cmd_obj->IsMultiwordObject() && args.GetArgumentCount() > 0) 682ebc09c36SJim Ingham { 683ebc09c36SJim Ingham if (argc >= 3) 684ebc09c36SJim Ingham { 685ebc09c36SJim Ingham const std::string sub_command = args.GetArgumentAtIndex(0); 686ebc09c36SJim Ingham assert (sub_command.length() != 0); 687998255bfSGreg Clayton subcommand_obj_sp = cmd_obj->GetSubcommandSP (sub_command.c_str()); 688ebc09c36SJim Ingham if (subcommand_obj_sp.get()) 689ebc09c36SJim Ingham { 690ebc09c36SJim Ingham sub_cmd_obj = subcommand_obj_sp.get(); 691ebc09c36SJim Ingham use_subcommand = true; 692ebc09c36SJim Ingham args.Shift(); // Shift the sub_command word off the argument vector. 693844d2303SCaroline Tice cmd_obj = sub_cmd_obj; 694ebc09c36SJim Ingham } 695ebc09c36SJim Ingham else 696ebc09c36SJim Ingham { 697f415eeb4SCaroline Tice result.AppendErrorWithFormat("'%s' is not a valid sub-command of '%s'. " 698f415eeb4SCaroline Tice "Unable to create alias.\n", 699f415eeb4SCaroline Tice sub_command.c_str(), actual_command.c_str()); 700ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 701ebc09c36SJim Ingham return false; 702ebc09c36SJim Ingham } 703ebc09c36SJim Ingham } 704ebc09c36SJim Ingham } 705ebc09c36SJim Ingham 706ebc09c36SJim Ingham // Verify & handle any options/arguments passed to the alias command 707ebc09c36SJim Ingham 708ebc09c36SJim Ingham if (args.GetArgumentCount () > 0) 709ebc09c36SJim Ingham { 710ca90c47eSCaroline Tice CommandObjectSP tmp_sp = m_interpreter.GetCommandSPExact (cmd_obj->GetCommandName(), false); 711ebc09c36SJim Ingham if (use_subcommand) 712ca90c47eSCaroline Tice tmp_sp = m_interpreter.GetCommandSPExact (sub_cmd_obj->GetCommandName(), false); 713ca90c47eSCaroline Tice 714ca90c47eSCaroline Tice std::string args_string; 715ca90c47eSCaroline Tice args.GetCommandString (args_string); 716ca90c47eSCaroline Tice 717ca90c47eSCaroline Tice if (!m_interpreter.ProcessAliasOptionsArgs (tmp_sp, args_string.c_str(), option_arg_vector_sp)) 718ebc09c36SJim Ingham { 719ca90c47eSCaroline Tice result.AppendError ("Unable to create requested alias.\n"); 720ca90c47eSCaroline Tice result.SetStatus (eReturnStatusFailed); 721e7941795SCaroline Tice return false; 722867b185dSCaroline Tice } 723867b185dSCaroline Tice } 724867b185dSCaroline Tice 725ebc09c36SJim Ingham // Create the alias. 726ebc09c36SJim Ingham 727a7015092SGreg Clayton if (m_interpreter.AliasExists (alias_command.c_str()) 728a7015092SGreg Clayton || m_interpreter.UserCommandExists (alias_command.c_str())) 729ebc09c36SJim Ingham { 730a7015092SGreg Clayton OptionArgVectorSP tmp_option_arg_sp (m_interpreter.GetAliasOptions (alias_command.c_str())); 731ebc09c36SJim Ingham if (tmp_option_arg_sp.get()) 732ebc09c36SJim Ingham { 733ebc09c36SJim Ingham if (option_arg_vector->size() == 0) 734a7015092SGreg Clayton m_interpreter.RemoveAliasOptions (alias_command.c_str()); 735ebc09c36SJim Ingham } 736ebc09c36SJim Ingham result.AppendWarningWithFormat ("Overwriting existing definition for '%s'.\n", 737ebc09c36SJim Ingham alias_command.c_str()); 738ebc09c36SJim Ingham } 739ebc09c36SJim Ingham 740ebc09c36SJim Ingham if (use_subcommand) 741a7015092SGreg Clayton m_interpreter.AddAlias (alias_command.c_str(), subcommand_obj_sp); 742ebc09c36SJim Ingham else 743a7015092SGreg Clayton m_interpreter.AddAlias (alias_command.c_str(), command_obj_sp); 744ebc09c36SJim Ingham if (option_arg_vector->size() > 0) 745a7015092SGreg Clayton m_interpreter.AddOrReplaceAliasOptions (alias_command.c_str(), option_arg_vector_sp); 746ebc09c36SJim Ingham result.SetStatus (eReturnStatusSuccessFinishNoResult); 747ebc09c36SJim Ingham } 748ebc09c36SJim Ingham else 749ebc09c36SJim Ingham { 750ebc09c36SJim Ingham result.AppendErrorWithFormat ("'%s' is not an existing command.\n", actual_command.c_str()); 751ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 752e7941795SCaroline Tice return false; 753ebc09c36SJim Ingham } 754ebc09c36SJim Ingham } 755ebc09c36SJim Ingham 756ebc09c36SJim Ingham return result.Succeeded(); 757ebc09c36SJim Ingham } 7585a988416SJim Ingham 759ebc09c36SJim Ingham }; 760ebc09c36SJim Ingham 761ebc09c36SJim Ingham #pragma mark CommandObjectCommandsUnalias 762ebc09c36SJim Ingham //------------------------------------------------------------------------- 763ebc09c36SJim Ingham // CommandObjectCommandsUnalias 764ebc09c36SJim Ingham //------------------------------------------------------------------------- 765ebc09c36SJim Ingham 7665a988416SJim Ingham class CommandObjectCommandsUnalias : public CommandObjectParsed 767ebc09c36SJim Ingham { 768ebc09c36SJim Ingham public: 769a7015092SGreg Clayton CommandObjectCommandsUnalias (CommandInterpreter &interpreter) : 7705a988416SJim Ingham CommandObjectParsed (interpreter, 7710e5e5a79SGreg Clayton "command unalias", 77286ddae50SCaroline Tice "Allow the user to remove/delete a user-defined command abbreviation.", 773405fe67fSCaroline Tice NULL) 774ebc09c36SJim Ingham { 775405fe67fSCaroline Tice CommandArgumentEntry arg; 776405fe67fSCaroline Tice CommandArgumentData alias_arg; 777405fe67fSCaroline Tice 778405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 779405fe67fSCaroline Tice alias_arg.arg_type = eArgTypeAliasName; 780405fe67fSCaroline Tice alias_arg.arg_repetition = eArgRepeatPlain; 781405fe67fSCaroline Tice 782405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 783405fe67fSCaroline Tice arg.push_back (alias_arg); 784405fe67fSCaroline Tice 785405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 786405fe67fSCaroline Tice m_arguments.push_back (arg); 787ebc09c36SJim Ingham } 788ebc09c36SJim Ingham 789ebc09c36SJim Ingham ~CommandObjectCommandsUnalias() 790ebc09c36SJim Ingham { 791ebc09c36SJim Ingham } 792ebc09c36SJim Ingham 7935a988416SJim Ingham protected: 794ebc09c36SJim Ingham bool 7955a988416SJim Ingham DoExecute (Args& args, CommandReturnObject &result) 796ebc09c36SJim Ingham { 797ebc09c36SJim Ingham CommandObject::CommandMap::iterator pos; 798ebc09c36SJim Ingham CommandObject *cmd_obj; 799ebc09c36SJim Ingham 800ebc09c36SJim Ingham if (args.GetArgumentCount() != 0) 801ebc09c36SJim Ingham { 802ebc09c36SJim Ingham const char *command_name = args.GetArgumentAtIndex(0); 803a7015092SGreg Clayton cmd_obj = m_interpreter.GetCommandObject(command_name); 804ebc09c36SJim Ingham if (cmd_obj) 805ebc09c36SJim Ingham { 806a7015092SGreg Clayton if (m_interpreter.CommandExists (command_name)) 807ebc09c36SJim Ingham { 808ebc09c36SJim Ingham result.AppendErrorWithFormat ("'%s' is a permanent debugger command and cannot be removed.\n", 809ebc09c36SJim Ingham command_name); 810ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 811ebc09c36SJim Ingham } 812ebc09c36SJim Ingham else 813ebc09c36SJim Ingham { 814ebc09c36SJim Ingham 815a7015092SGreg Clayton if (m_interpreter.RemoveAlias (command_name) == false) 816ebc09c36SJim Ingham { 817a7015092SGreg Clayton if (m_interpreter.AliasExists (command_name)) 818ebc09c36SJim Ingham result.AppendErrorWithFormat ("Error occurred while attempting to unalias '%s'.\n", 819ebc09c36SJim Ingham command_name); 820ebc09c36SJim Ingham else 821ebc09c36SJim Ingham result.AppendErrorWithFormat ("'%s' is not an existing alias.\n", command_name); 822ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 823ebc09c36SJim Ingham } 824ebc09c36SJim Ingham else 825ebc09c36SJim Ingham result.SetStatus (eReturnStatusSuccessFinishNoResult); 826ebc09c36SJim Ingham } 827ebc09c36SJim Ingham } 828ebc09c36SJim Ingham else 829ebc09c36SJim Ingham { 830ebc09c36SJim Ingham result.AppendErrorWithFormat ("'%s' is not a known command.\nTry 'help' to see a " 831ebc09c36SJim Ingham "current list of commands.\n", 832ebc09c36SJim Ingham command_name); 833ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 834ebc09c36SJim Ingham } 835ebc09c36SJim Ingham } 836ebc09c36SJim Ingham else 837ebc09c36SJim Ingham { 838ebc09c36SJim Ingham result.AppendError ("must call 'unalias' with a valid alias"); 839ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 840ebc09c36SJim Ingham } 841ebc09c36SJim Ingham 842ebc09c36SJim Ingham return result.Succeeded(); 843ebc09c36SJim Ingham } 844ebc09c36SJim Ingham }; 845ebc09c36SJim Ingham 846de164aaaSGreg Clayton //------------------------------------------------------------------------- 847de164aaaSGreg Clayton // CommandObjectCommandsAddRegex 848de164aaaSGreg Clayton //------------------------------------------------------------------------- 8495a988416SJim Ingham #pragma mark CommandObjectCommandsAddRegex 850de164aaaSGreg Clayton 8515a988416SJim Ingham class CommandObjectCommandsAddRegex : public CommandObjectParsed 852de164aaaSGreg Clayton { 853de164aaaSGreg Clayton public: 854de164aaaSGreg Clayton CommandObjectCommandsAddRegex (CommandInterpreter &interpreter) : 8555a988416SJim Ingham CommandObjectParsed (interpreter, 8560e5e5a79SGreg Clayton "command regex", 857de164aaaSGreg Clayton "Allow the user to create a regular expression command.", 8580e5e5a79SGreg Clayton "command regex <cmd-name> [s/<regex>/<subst>/ ...]"), 859de164aaaSGreg Clayton m_options (interpreter) 860de164aaaSGreg Clayton { 8610e5e5a79SGreg Clayton SetHelpLong( 8620e5e5a79SGreg Clayton "This command allows the user to create powerful regular expression commands\n" 8630e5e5a79SGreg Clayton "with substitutions. The regular expressions and substitutions are specified\n" 8640e5e5a79SGreg Clayton "using the regular exression substitution format of:\n" 8650e5e5a79SGreg Clayton "\n" 8660e5e5a79SGreg Clayton " s/<regex>/<subst>/\n" 8670e5e5a79SGreg Clayton "\n" 8680e5e5a79SGreg Clayton "<regex> is a regular expression that can use parenthesis to capture regular\n" 8690e5e5a79SGreg Clayton "expression input and substitute the captured matches in the output using %1\n" 8700e5e5a79SGreg Clayton "for the first match, %2 for the second, and so on.\n" 8710e5e5a79SGreg Clayton "\n" 8720e5e5a79SGreg Clayton "The regular expressions can all be specified on the command line if more than\n" 8730e5e5a79SGreg Clayton "one argument is provided. If just the command name is provided on the command\n" 8740e5e5a79SGreg Clayton "line, then the regular expressions and substitutions can be entered on separate\n" 8750e5e5a79SGreg Clayton " lines, followed by an empty line to terminate the command definition.\n" 8760e5e5a79SGreg Clayton "\n" 8770e5e5a79SGreg Clayton "EXAMPLES\n" 8780e5e5a79SGreg Clayton "\n" 879adc43c99SSean Callanan "The following example will define a regular expression command named 'f' that\n" 8800e5e5a79SGreg Clayton "will call 'finish' if there are no arguments, or 'frame select <frame-idx>' if\n" 8810e5e5a79SGreg Clayton "a number follows 'f':\n" 882adc43c99SSean Callanan "\n" 8830e5e5a79SGreg Clayton " (lldb) command regex f s/^$/finish/ 's/([0-9]+)/frame select %1/'\n" 884adc43c99SSean Callanan "\n" 8850e5e5a79SGreg Clayton ); 886de164aaaSGreg Clayton } 887de164aaaSGreg Clayton 888de164aaaSGreg Clayton ~CommandObjectCommandsAddRegex() 889de164aaaSGreg Clayton { 890de164aaaSGreg Clayton } 891de164aaaSGreg Clayton 892de164aaaSGreg Clayton 8935a988416SJim Ingham protected: 894de164aaaSGreg Clayton bool 8955a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 896de164aaaSGreg Clayton { 8975a988416SJim Ingham const size_t argc = command.GetArgumentCount(); 8980e5e5a79SGreg Clayton if (argc == 0) 899de164aaaSGreg Clayton { 90069c12ccbSJason Molenda result.AppendError ("usage: 'command regex <command-name> [s/<regex1>/<subst1>/ s/<regex2>/<subst2>/ ...]'\n"); 9010e5e5a79SGreg Clayton result.SetStatus (eReturnStatusFailed); 9020e5e5a79SGreg Clayton } 9030e5e5a79SGreg Clayton else 9040e5e5a79SGreg Clayton { 9050e5e5a79SGreg Clayton Error error; 9065a988416SJim Ingham const char *name = command.GetArgumentAtIndex(0); 907de164aaaSGreg Clayton m_regex_cmd_ap.reset (new CommandObjectRegexCommand (m_interpreter, 908de164aaaSGreg Clayton name, 909de164aaaSGreg Clayton m_options.GetHelp (), 910de164aaaSGreg Clayton m_options.GetSyntax (), 911de164aaaSGreg Clayton 10)); 9120e5e5a79SGreg Clayton 9130e5e5a79SGreg Clayton if (argc == 1) 9140e5e5a79SGreg Clayton { 9150e5e5a79SGreg Clayton InputReaderSP reader_sp (new InputReader(m_interpreter.GetDebugger())); 916de164aaaSGreg Clayton if (reader_sp) 917de164aaaSGreg Clayton { 9180e5e5a79SGreg Clayton error =reader_sp->Initialize (CommandObjectCommandsAddRegex::InputReaderCallback, 919de164aaaSGreg Clayton this, // baton 920de164aaaSGreg Clayton eInputReaderGranularityLine, // token size, to pass to callback function 9210e5e5a79SGreg Clayton NULL, // end token 922de164aaaSGreg Clayton "> ", // prompt 9230e5e5a79SGreg Clayton true); // echo input 9240e5e5a79SGreg Clayton if (error.Success()) 925de164aaaSGreg Clayton { 926de164aaaSGreg Clayton m_interpreter.GetDebugger().PushInputReader (reader_sp); 927de164aaaSGreg Clayton result.SetStatus (eReturnStatusSuccessFinishNoResult); 9280e5e5a79SGreg Clayton return true; 929de164aaaSGreg Clayton } 930de164aaaSGreg Clayton } 931de164aaaSGreg Clayton } 932de164aaaSGreg Clayton else 933de164aaaSGreg Clayton { 9340e5e5a79SGreg Clayton for (size_t arg_idx = 1; arg_idx < argc; ++arg_idx) 9350e5e5a79SGreg Clayton { 9365a988416SJim Ingham llvm::StringRef arg_strref (command.GetArgumentAtIndex(arg_idx)); 9370e5e5a79SGreg Clayton error = AppendRegexSubstitution (arg_strref); 9380e5e5a79SGreg Clayton if (error.Fail()) 9390e5e5a79SGreg Clayton break; 9400e5e5a79SGreg Clayton } 9410e5e5a79SGreg Clayton 9420e5e5a79SGreg Clayton if (error.Success()) 9430e5e5a79SGreg Clayton { 9440e5e5a79SGreg Clayton AddRegexCommandToInterpreter(); 9450e5e5a79SGreg Clayton } 9460e5e5a79SGreg Clayton } 9470e5e5a79SGreg Clayton if (error.Fail()) 9480e5e5a79SGreg Clayton { 9490e5e5a79SGreg Clayton result.AppendError (error.AsCString()); 950de164aaaSGreg Clayton result.SetStatus (eReturnStatusFailed); 951de164aaaSGreg Clayton } 9520e5e5a79SGreg Clayton } 9530e5e5a79SGreg Clayton 954de164aaaSGreg Clayton return result.Succeeded(); 955de164aaaSGreg Clayton } 956de164aaaSGreg Clayton 9570e5e5a79SGreg Clayton Error 9580e5e5a79SGreg Clayton AppendRegexSubstitution (const llvm::StringRef ®ex_sed) 959de164aaaSGreg Clayton { 9600e5e5a79SGreg Clayton Error error; 9610e5e5a79SGreg Clayton 9620e5e5a79SGreg Clayton if (m_regex_cmd_ap.get() == NULL) 963de164aaaSGreg Clayton { 9640e5e5a79SGreg Clayton error.SetErrorStringWithFormat("invalid regular expression command object for: '%.*s'", 9650e5e5a79SGreg Clayton (int)regex_sed.size(), 9660e5e5a79SGreg Clayton regex_sed.data()); 9670e5e5a79SGreg Clayton return error; 968de164aaaSGreg Clayton } 9690e5e5a79SGreg Clayton 9700e5e5a79SGreg Clayton size_t regex_sed_size = regex_sed.size(); 9710e5e5a79SGreg Clayton 9720e5e5a79SGreg Clayton if (regex_sed_size <= 1) 9730e5e5a79SGreg Clayton { 9740e5e5a79SGreg Clayton error.SetErrorStringWithFormat("regular expression substitution string is too short: '%.*s'", 9750e5e5a79SGreg Clayton (int)regex_sed.size(), 9760e5e5a79SGreg Clayton regex_sed.data()); 9770e5e5a79SGreg Clayton return error; 9780e5e5a79SGreg Clayton } 9790e5e5a79SGreg Clayton 9800e5e5a79SGreg Clayton if (regex_sed[0] != 's') 9810e5e5a79SGreg Clayton { 9820e5e5a79SGreg Clayton error.SetErrorStringWithFormat("regular expression substitution string doesn't start with 's': '%.*s'", 9830e5e5a79SGreg Clayton (int)regex_sed.size(), 9840e5e5a79SGreg Clayton regex_sed.data()); 9850e5e5a79SGreg Clayton return error; 9860e5e5a79SGreg Clayton } 9870e5e5a79SGreg Clayton const size_t first_separator_char_pos = 1; 9880e5e5a79SGreg Clayton // use the char that follows 's' as the regex separator character 9890e5e5a79SGreg Clayton // so we can have "s/<regex>/<subst>/" or "s|<regex>|<subst>|" 9900e5e5a79SGreg Clayton const char separator_char = regex_sed[first_separator_char_pos]; 9910e5e5a79SGreg Clayton const size_t second_separator_char_pos = regex_sed.find (separator_char, first_separator_char_pos + 1); 9920e5e5a79SGreg Clayton 9930e5e5a79SGreg Clayton if (second_separator_char_pos == std::string::npos) 9940e5e5a79SGreg Clayton { 9950e5e5a79SGreg Clayton error.SetErrorStringWithFormat("missing second '%c' separator char after '%.*s'", 9960e5e5a79SGreg Clayton separator_char, 9970e5e5a79SGreg Clayton (int)(regex_sed.size() - first_separator_char_pos - 1), 9980e5e5a79SGreg Clayton regex_sed.data() + (first_separator_char_pos + 1)); 9990e5e5a79SGreg Clayton return error; 10000e5e5a79SGreg Clayton } 10010e5e5a79SGreg Clayton 10020e5e5a79SGreg Clayton const size_t third_separator_char_pos = regex_sed.find (separator_char, second_separator_char_pos + 1); 10030e5e5a79SGreg Clayton 10040e5e5a79SGreg Clayton if (third_separator_char_pos == std::string::npos) 10050e5e5a79SGreg Clayton { 10060e5e5a79SGreg Clayton error.SetErrorStringWithFormat("missing third '%c' separator char after '%.*s'", 10070e5e5a79SGreg Clayton separator_char, 10080e5e5a79SGreg Clayton (int)(regex_sed.size() - second_separator_char_pos - 1), 10090e5e5a79SGreg Clayton regex_sed.data() + (second_separator_char_pos + 1)); 10100e5e5a79SGreg Clayton return error; 10110e5e5a79SGreg Clayton } 10120e5e5a79SGreg Clayton 10130e5e5a79SGreg Clayton if (third_separator_char_pos != regex_sed_size - 1) 10140e5e5a79SGreg Clayton { 10150e5e5a79SGreg Clayton // Make sure that everything that follows the last regex 10160e5e5a79SGreg Clayton // separator char 10170e5e5a79SGreg Clayton if (regex_sed.find_first_not_of("\t\n\v\f\r ", third_separator_char_pos + 1) != std::string::npos) 10180e5e5a79SGreg Clayton { 10190e5e5a79SGreg Clayton error.SetErrorStringWithFormat("extra data found after the '%.*s' regular expression substitution string: '%.*s'", 10200e5e5a79SGreg Clayton (int)third_separator_char_pos + 1, 10210e5e5a79SGreg Clayton regex_sed.data(), 10220e5e5a79SGreg Clayton (int)(regex_sed.size() - third_separator_char_pos - 1), 10230e5e5a79SGreg Clayton regex_sed.data() + (third_separator_char_pos + 1)); 10240e5e5a79SGreg Clayton return error; 10250e5e5a79SGreg Clayton } 10260e5e5a79SGreg Clayton 10270e5e5a79SGreg Clayton } 10280e5e5a79SGreg Clayton else if (first_separator_char_pos + 1 == second_separator_char_pos) 10290e5e5a79SGreg Clayton { 10300e5e5a79SGreg Clayton error.SetErrorStringWithFormat("<regex> can't be empty in 's%c<regex>%c<subst>%c' string: '%.*s'", 10310e5e5a79SGreg Clayton separator_char, 10320e5e5a79SGreg Clayton separator_char, 10330e5e5a79SGreg Clayton separator_char, 10340e5e5a79SGreg Clayton (int)regex_sed.size(), 10350e5e5a79SGreg Clayton regex_sed.data()); 10360e5e5a79SGreg Clayton return error; 10370e5e5a79SGreg Clayton } 10380e5e5a79SGreg Clayton else if (second_separator_char_pos + 1 == third_separator_char_pos) 10390e5e5a79SGreg Clayton { 10400e5e5a79SGreg Clayton error.SetErrorStringWithFormat("<subst> can't be empty in 's%c<regex>%c<subst>%c' string: '%.*s'", 10410e5e5a79SGreg Clayton separator_char, 10420e5e5a79SGreg Clayton separator_char, 10430e5e5a79SGreg Clayton separator_char, 10440e5e5a79SGreg Clayton (int)regex_sed.size(), 10450e5e5a79SGreg Clayton regex_sed.data()); 10460e5e5a79SGreg Clayton return error; 10470e5e5a79SGreg Clayton } 10480e5e5a79SGreg Clayton std::string regex(regex_sed.substr(first_separator_char_pos + 1, second_separator_char_pos - first_separator_char_pos - 1)); 10490e5e5a79SGreg Clayton std::string subst(regex_sed.substr(second_separator_char_pos + 1, third_separator_char_pos - second_separator_char_pos - 1)); 10500e5e5a79SGreg Clayton m_regex_cmd_ap->AddRegexCommand (regex.c_str(), 10510e5e5a79SGreg Clayton subst.c_str()); 10520e5e5a79SGreg Clayton return error; 1053de164aaaSGreg Clayton } 1054de164aaaSGreg Clayton 1055de164aaaSGreg Clayton void 10560e5e5a79SGreg Clayton AddRegexCommandToInterpreter() 1057de164aaaSGreg Clayton { 1058de164aaaSGreg Clayton if (m_regex_cmd_ap.get()) 1059de164aaaSGreg Clayton { 1060de164aaaSGreg Clayton if (m_regex_cmd_ap->HasRegexEntries()) 1061de164aaaSGreg Clayton { 1062de164aaaSGreg Clayton CommandObjectSP cmd_sp (m_regex_cmd_ap.release()); 1063de164aaaSGreg Clayton m_interpreter.AddCommand(cmd_sp->GetCommandName(), cmd_sp, true); 1064de164aaaSGreg Clayton } 1065de164aaaSGreg Clayton } 1066de164aaaSGreg Clayton } 1067de164aaaSGreg Clayton 10680e5e5a79SGreg Clayton void 10690e5e5a79SGreg Clayton InputReaderDidCancel() 10700e5e5a79SGreg Clayton { 10710e5e5a79SGreg Clayton m_regex_cmd_ap.reset(); 10720e5e5a79SGreg Clayton } 10730e5e5a79SGreg Clayton 1074de164aaaSGreg Clayton static size_t 1075de164aaaSGreg Clayton InputReaderCallback (void *baton, 1076de164aaaSGreg Clayton InputReader &reader, 1077de164aaaSGreg Clayton lldb::InputReaderAction notification, 1078de164aaaSGreg Clayton const char *bytes, 10795a988416SJim Ingham size_t bytes_len) 10805a988416SJim Ingham { 10815a988416SJim Ingham CommandObjectCommandsAddRegex *add_regex_cmd = (CommandObjectCommandsAddRegex *) baton; 10825a988416SJim Ingham bool batch_mode = reader.GetDebugger().GetCommandInterpreter().GetBatchCommandMode(); 10835a988416SJim Ingham 10845a988416SJim Ingham switch (notification) 10855a988416SJim Ingham { 10865a988416SJim Ingham case eInputReaderActivate: 10875a988416SJim Ingham if (!batch_mode) 10885a988416SJim Ingham { 10895a988416SJim Ingham StreamSP out_stream = reader.GetDebugger().GetAsyncOutputStream (); 10905a988416SJim Ingham out_stream->Printf("%s\n", "Enter regular expressions in the form 's/<regex>/<subst>/' and terminate with an empty line:"); 10915a988416SJim Ingham out_stream->Flush(); 10925a988416SJim Ingham } 10935a988416SJim Ingham break; 10945a988416SJim Ingham case eInputReaderReactivate: 10955a988416SJim Ingham break; 10965a988416SJim Ingham 10975a988416SJim Ingham case eInputReaderDeactivate: 10985a988416SJim Ingham break; 10995a988416SJim Ingham 11005a988416SJim Ingham case eInputReaderAsynchronousOutputWritten: 11015a988416SJim Ingham break; 11025a988416SJim Ingham 11035a988416SJim Ingham case eInputReaderGotToken: 11045a988416SJim Ingham while (bytes_len > 0 && (bytes[bytes_len-1] == '\r' || bytes[bytes_len-1] == '\n')) 11055a988416SJim Ingham --bytes_len; 11065a988416SJim Ingham if (bytes_len == 0) 11075a988416SJim Ingham reader.SetIsDone(true); 11085a988416SJim Ingham else if (bytes) 11095a988416SJim Ingham { 11105a988416SJim Ingham llvm::StringRef bytes_strref (bytes, bytes_len); 11115a988416SJim Ingham Error error (add_regex_cmd->AppendRegexSubstitution (bytes_strref)); 11125a988416SJim Ingham if (error.Fail()) 11135a988416SJim Ingham { 11145a988416SJim Ingham if (!batch_mode) 11155a988416SJim Ingham { 11165a988416SJim Ingham StreamSP out_stream = reader.GetDebugger().GetAsyncOutputStream(); 11175a988416SJim Ingham out_stream->Printf("error: %s\n", error.AsCString()); 11185a988416SJim Ingham out_stream->Flush(); 11195a988416SJim Ingham } 11205a988416SJim Ingham add_regex_cmd->InputReaderDidCancel (); 11215a988416SJim Ingham reader.SetIsDone (true); 11225a988416SJim Ingham } 11235a988416SJim Ingham } 11245a988416SJim Ingham break; 11255a988416SJim Ingham 11265a988416SJim Ingham case eInputReaderInterrupt: 11275a988416SJim Ingham { 11285a988416SJim Ingham reader.SetIsDone (true); 11295a988416SJim Ingham if (!batch_mode) 11305a988416SJim Ingham { 11315a988416SJim Ingham StreamSP out_stream = reader.GetDebugger().GetAsyncOutputStream(); 11325a988416SJim Ingham out_stream->PutCString("Regular expression command creations was cancelled.\n"); 11335a988416SJim Ingham out_stream->Flush(); 11345a988416SJim Ingham } 11355a988416SJim Ingham add_regex_cmd->InputReaderDidCancel (); 11365a988416SJim Ingham } 11375a988416SJim Ingham break; 11385a988416SJim Ingham 11395a988416SJim Ingham case eInputReaderEndOfFile: 11405a988416SJim Ingham reader.SetIsDone (true); 11415a988416SJim Ingham break; 11425a988416SJim Ingham 11435a988416SJim Ingham case eInputReaderDone: 11445a988416SJim Ingham add_regex_cmd->AddRegexCommandToInterpreter(); 11455a988416SJim Ingham break; 11465a988416SJim Ingham } 11475a988416SJim Ingham 11485a988416SJim Ingham return bytes_len; 11495a988416SJim Ingham } 11505a988416SJim Ingham 1151de164aaaSGreg Clayton private: 11527b0992d9SGreg Clayton std::unique_ptr<CommandObjectRegexCommand> m_regex_cmd_ap; 1153de164aaaSGreg Clayton 1154de164aaaSGreg Clayton class CommandOptions : public Options 1155de164aaaSGreg Clayton { 1156de164aaaSGreg Clayton public: 1157de164aaaSGreg Clayton 1158de164aaaSGreg Clayton CommandOptions (CommandInterpreter &interpreter) : 1159de164aaaSGreg Clayton Options (interpreter) 1160de164aaaSGreg Clayton { 1161de164aaaSGreg Clayton } 1162de164aaaSGreg Clayton 1163de164aaaSGreg Clayton virtual 1164de164aaaSGreg Clayton ~CommandOptions (){} 1165de164aaaSGreg Clayton 1166de164aaaSGreg Clayton virtual Error 1167de164aaaSGreg Clayton SetOptionValue (uint32_t option_idx, const char *option_arg) 1168de164aaaSGreg Clayton { 1169de164aaaSGreg Clayton Error error; 11703bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 1171de164aaaSGreg Clayton 1172de164aaaSGreg Clayton switch (short_option) 1173de164aaaSGreg Clayton { 1174de164aaaSGreg Clayton case 'h': 1175de164aaaSGreg Clayton m_help.assign (option_arg); 1176de164aaaSGreg Clayton break; 1177de164aaaSGreg Clayton case 's': 1178de164aaaSGreg Clayton m_syntax.assign (option_arg); 1179de164aaaSGreg Clayton break; 1180de164aaaSGreg Clayton 1181de164aaaSGreg Clayton default: 118286edbf41SGreg Clayton error.SetErrorStringWithFormat ("unrecognized option '%c'", short_option); 1183de164aaaSGreg Clayton break; 1184de164aaaSGreg Clayton } 1185de164aaaSGreg Clayton 1186de164aaaSGreg Clayton return error; 1187de164aaaSGreg Clayton } 1188de164aaaSGreg Clayton 1189de164aaaSGreg Clayton void 1190de164aaaSGreg Clayton OptionParsingStarting () 1191de164aaaSGreg Clayton { 1192de164aaaSGreg Clayton m_help.clear(); 1193de164aaaSGreg Clayton m_syntax.clear(); 1194de164aaaSGreg Clayton } 1195de164aaaSGreg Clayton 1196de164aaaSGreg Clayton const OptionDefinition* 1197de164aaaSGreg Clayton GetDefinitions () 1198de164aaaSGreg Clayton { 1199de164aaaSGreg Clayton return g_option_table; 1200de164aaaSGreg Clayton } 1201de164aaaSGreg Clayton 1202de164aaaSGreg Clayton // Options table: Required for subclasses of Options. 1203de164aaaSGreg Clayton 1204de164aaaSGreg Clayton static OptionDefinition g_option_table[]; 1205de164aaaSGreg Clayton 1206de164aaaSGreg Clayton const char * 1207de164aaaSGreg Clayton GetHelp () 1208de164aaaSGreg Clayton { 1209de164aaaSGreg Clayton if (m_help.empty()) 1210de164aaaSGreg Clayton return NULL; 1211de164aaaSGreg Clayton return m_help.c_str(); 1212de164aaaSGreg Clayton } 1213de164aaaSGreg Clayton const char * 1214de164aaaSGreg Clayton GetSyntax () 1215de164aaaSGreg Clayton { 1216de164aaaSGreg Clayton if (m_syntax.empty()) 1217de164aaaSGreg Clayton return NULL; 1218de164aaaSGreg Clayton return m_syntax.c_str(); 1219de164aaaSGreg Clayton } 1220de164aaaSGreg Clayton // Instance variables to hold the values for command options. 1221de164aaaSGreg Clayton protected: 1222de164aaaSGreg Clayton std::string m_help; 1223de164aaaSGreg Clayton std::string m_syntax; 1224de164aaaSGreg Clayton }; 1225de164aaaSGreg Clayton 1226de164aaaSGreg Clayton virtual Options * 1227de164aaaSGreg Clayton GetOptions () 1228de164aaaSGreg Clayton { 1229de164aaaSGreg Clayton return &m_options; 1230de164aaaSGreg Clayton } 1231de164aaaSGreg Clayton 12325a988416SJim Ingham CommandOptions m_options; 1233de164aaaSGreg Clayton }; 1234de164aaaSGreg Clayton 1235de164aaaSGreg Clayton OptionDefinition 1236de164aaaSGreg Clayton CommandObjectCommandsAddRegex::CommandOptions::g_option_table[] = 1237de164aaaSGreg Clayton { 1238de164aaaSGreg Clayton { LLDB_OPT_SET_1, false, "help" , 'h', required_argument, NULL, 0, eArgTypeNone, "The help text to display for this command."}, 1239de164aaaSGreg Clayton { LLDB_OPT_SET_1, false, "syntax", 's', required_argument, NULL, 0, eArgTypeNone, "A syntax string showing the typical usage syntax."}, 1240de164aaaSGreg Clayton { 0 , false, NULL , 0 , 0 , NULL, 0, eArgTypeNone, NULL } 1241de164aaaSGreg Clayton }; 1242de164aaaSGreg Clayton 1243de164aaaSGreg Clayton 12445a988416SJim Ingham class CommandObjectPythonFunction : public CommandObjectRaw 1245223383edSEnrico Granata { 1246223383edSEnrico Granata private: 1247223383edSEnrico Granata std::string m_function_name; 12480a305db7SEnrico Granata ScriptedCommandSynchronicity m_synchro; 1249fac939e9SEnrico Granata bool m_fetched_help_long; 1250223383edSEnrico Granata 1251223383edSEnrico Granata public: 1252223383edSEnrico Granata 1253223383edSEnrico Granata CommandObjectPythonFunction (CommandInterpreter &interpreter, 1254223383edSEnrico Granata std::string name, 12550a305db7SEnrico Granata std::string funct, 12560a305db7SEnrico Granata ScriptedCommandSynchronicity synch) : 12575a988416SJim Ingham CommandObjectRaw (interpreter, 1258223383edSEnrico Granata name.c_str(), 1259223383edSEnrico Granata (std::string("Run Python function ") + funct).c_str(), 1260223383edSEnrico Granata NULL), 12610a305db7SEnrico Granata m_function_name(funct), 1262fac939e9SEnrico Granata m_synchro(synch), 1263fac939e9SEnrico Granata m_fetched_help_long(false) 1264223383edSEnrico Granata { 1265223383edSEnrico Granata } 1266223383edSEnrico Granata 1267223383edSEnrico Granata virtual 1268223383edSEnrico Granata ~CommandObjectPythonFunction () 1269223383edSEnrico Granata { 1270223383edSEnrico Granata } 1271223383edSEnrico Granata 1272223383edSEnrico Granata virtual bool 12733a18e319SGreg Clayton IsRemovable () const 12745a988416SJim Ingham { 12755a988416SJim Ingham return true; 12765a988416SJim Ingham } 12775a988416SJim Ingham 12785a988416SJim Ingham const std::string& 12795a988416SJim Ingham GetFunctionName () 12805a988416SJim Ingham { 12815a988416SJim Ingham return m_function_name; 12825a988416SJim Ingham } 12835a988416SJim Ingham 12845a988416SJim Ingham ScriptedCommandSynchronicity 12855a988416SJim Ingham GetSynchronicity () 12865a988416SJim Ingham { 12875a988416SJim Ingham return m_synchro; 12885a988416SJim Ingham } 12895a988416SJim Ingham 1290fac939e9SEnrico Granata virtual const char * 1291fac939e9SEnrico Granata GetHelpLong () 1292fac939e9SEnrico Granata { 1293fac939e9SEnrico Granata if (!m_fetched_help_long) 1294fac939e9SEnrico Granata { 1295fac939e9SEnrico Granata ScriptInterpreter* scripter = m_interpreter.GetScriptInterpreter(); 1296fac939e9SEnrico Granata if (scripter) 1297fac939e9SEnrico Granata { 1298fac939e9SEnrico Granata std::string docstring; 1299fac939e9SEnrico Granata m_fetched_help_long = scripter->GetDocumentationForItem(m_function_name.c_str(),docstring); 1300fac939e9SEnrico Granata if (!docstring.empty()) 1301fac939e9SEnrico Granata SetHelpLong(docstring); 1302fac939e9SEnrico Granata } 1303fac939e9SEnrico Granata } 1304fac939e9SEnrico Granata return CommandObjectRaw::GetHelpLong(); 1305fac939e9SEnrico Granata } 1306fac939e9SEnrico Granata 13075a988416SJim Ingham protected: 13085a988416SJim Ingham virtual bool 13095a988416SJim Ingham DoExecute (const char *raw_command_line, CommandReturnObject &result) 1310223383edSEnrico Granata { 1311223383edSEnrico Granata ScriptInterpreter* scripter = m_interpreter.GetScriptInterpreter(); 1312223383edSEnrico Granata 1313223383edSEnrico Granata Error error; 1314223383edSEnrico Granata 131570f11f88SJim Ingham result.SetStatus(eReturnStatusInvalid); 131670f11f88SJim Ingham 1317223383edSEnrico Granata if (!scripter || scripter->RunScriptBasedCommand(m_function_name.c_str(), 1318223383edSEnrico Granata raw_command_line, 13190a305db7SEnrico Granata m_synchro, 1320223383edSEnrico Granata result, 1321223383edSEnrico Granata error) == false) 1322223383edSEnrico Granata { 1323223383edSEnrico Granata result.AppendError(error.AsCString()); 1324223383edSEnrico Granata result.SetStatus(eReturnStatusFailed); 1325223383edSEnrico Granata } 1326223383edSEnrico Granata else 132770f11f88SJim Ingham { 132870f11f88SJim Ingham // Don't change the status if the command already set it... 132970f11f88SJim Ingham if (result.GetStatus() == eReturnStatusInvalid) 133070f11f88SJim Ingham { 1331*9a71a7d8SDaniel Malea if (result.GetOutputData() == NULL || result.GetOutputData()[0] == '\0') 1332223383edSEnrico Granata result.SetStatus(eReturnStatusSuccessFinishNoResult); 133370f11f88SJim Ingham else 133470f11f88SJim Ingham result.SetStatus(eReturnStatusSuccessFinishResult); 133570f11f88SJim Ingham } 133670f11f88SJim Ingham } 1337223383edSEnrico Granata 1338223383edSEnrico Granata return result.Succeeded(); 1339223383edSEnrico Granata } 1340223383edSEnrico Granata 1341223383edSEnrico Granata }; 1342223383edSEnrico Granata 1343a9dbf432SEnrico Granata //------------------------------------------------------------------------- 1344a9dbf432SEnrico Granata // CommandObjectCommandsScriptImport 1345a9dbf432SEnrico Granata //------------------------------------------------------------------------- 1346a9dbf432SEnrico Granata 13475a988416SJim Ingham class CommandObjectCommandsScriptImport : public CommandObjectParsed 1348a9dbf432SEnrico Granata { 13495a988416SJim Ingham public: 13505a988416SJim Ingham CommandObjectCommandsScriptImport (CommandInterpreter &interpreter) : 13515a988416SJim Ingham CommandObjectParsed (interpreter, 13525a988416SJim Ingham "command script import", 13535a988416SJim Ingham "Import a scripting module in LLDB.", 13545a988416SJim Ingham NULL), 13555a988416SJim Ingham m_options(interpreter) 13565a988416SJim Ingham { 13575a988416SJim Ingham CommandArgumentEntry arg1; 13585a988416SJim Ingham CommandArgumentData cmd_arg; 13595a988416SJim Ingham 13605a988416SJim Ingham // Define the first (and only) variant of this arg. 13615a988416SJim Ingham cmd_arg.arg_type = eArgTypeFilename; 13625a988416SJim Ingham cmd_arg.arg_repetition = eArgRepeatPlain; 13635a988416SJim Ingham 13645a988416SJim Ingham // There is only one variant this argument could be; put it into the argument entry. 13655a988416SJim Ingham arg1.push_back (cmd_arg); 13665a988416SJim Ingham 13675a988416SJim Ingham // Push the data for the first argument into the m_arguments vector. 13685a988416SJim Ingham m_arguments.push_back (arg1); 13695a988416SJim Ingham } 13705a988416SJim Ingham 13715a988416SJim Ingham ~CommandObjectCommandsScriptImport () 13725a988416SJim Ingham { 13735a988416SJim Ingham } 13745a988416SJim Ingham 1375c7bece56SGreg Clayton virtual int 13765a988416SJim Ingham HandleArgumentCompletion (Args &input, 13775a988416SJim Ingham int &cursor_index, 13785a988416SJim Ingham int &cursor_char_position, 13795a988416SJim Ingham OptionElementVector &opt_element_vector, 13805a988416SJim Ingham int match_start_point, 13815a988416SJim Ingham int max_return_elements, 13825a988416SJim Ingham bool &word_complete, 13835a988416SJim Ingham StringList &matches) 13845a988416SJim Ingham { 13855a988416SJim Ingham std::string completion_str (input.GetArgumentAtIndex(cursor_index)); 13865a988416SJim Ingham completion_str.erase (cursor_char_position); 13875a988416SJim Ingham 13885a988416SJim Ingham CommandCompletions::InvokeCommonCompletionCallbacks (m_interpreter, 13895a988416SJim Ingham CommandCompletions::eDiskFileCompletion, 13905a988416SJim Ingham completion_str.c_str(), 13915a988416SJim Ingham match_start_point, 13925a988416SJim Ingham max_return_elements, 13935a988416SJim Ingham NULL, 13945a988416SJim Ingham word_complete, 13955a988416SJim Ingham matches); 13965a988416SJim Ingham return matches.GetSize(); 13975a988416SJim Ingham } 13985a988416SJim Ingham 13995a988416SJim Ingham virtual Options * 14005a988416SJim Ingham GetOptions () 14015a988416SJim Ingham { 14025a988416SJim Ingham return &m_options; 14035a988416SJim Ingham } 14045a988416SJim Ingham 14055a988416SJim Ingham protected: 14060a305db7SEnrico Granata 14070a305db7SEnrico Granata class CommandOptions : public Options 14080a305db7SEnrico Granata { 14090a305db7SEnrico Granata public: 14100a305db7SEnrico Granata 14110a305db7SEnrico Granata CommandOptions (CommandInterpreter &interpreter) : 14120a305db7SEnrico Granata Options (interpreter) 14130a305db7SEnrico Granata { 14140a305db7SEnrico Granata } 14150a305db7SEnrico Granata 14160a305db7SEnrico Granata virtual 14170a305db7SEnrico Granata ~CommandOptions (){} 14180a305db7SEnrico Granata 14190a305db7SEnrico Granata virtual Error 14200a305db7SEnrico Granata SetOptionValue (uint32_t option_idx, const char *option_arg) 14210a305db7SEnrico Granata { 14220a305db7SEnrico Granata Error error; 14233bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 14240a305db7SEnrico Granata 14250a305db7SEnrico Granata switch (short_option) 14260a305db7SEnrico Granata { 14270a305db7SEnrico Granata case 'r': 14280a305db7SEnrico Granata m_allow_reload = true; 14290a305db7SEnrico Granata break; 14300a305db7SEnrico Granata default: 14310a305db7SEnrico Granata error.SetErrorStringWithFormat ("unrecognized option '%c'", short_option); 14320a305db7SEnrico Granata break; 14330a305db7SEnrico Granata } 14340a305db7SEnrico Granata 14350a305db7SEnrico Granata return error; 14360a305db7SEnrico Granata } 14370a305db7SEnrico Granata 14380a305db7SEnrico Granata void 14390a305db7SEnrico Granata OptionParsingStarting () 14400a305db7SEnrico Granata { 1441e0c70f1bSEnrico Granata m_allow_reload = true; 14420a305db7SEnrico Granata } 14430a305db7SEnrico Granata 14440a305db7SEnrico Granata const OptionDefinition* 14450a305db7SEnrico Granata GetDefinitions () 14460a305db7SEnrico Granata { 14470a305db7SEnrico Granata return g_option_table; 14480a305db7SEnrico Granata } 14490a305db7SEnrico Granata 14500a305db7SEnrico Granata // Options table: Required for subclasses of Options. 14510a305db7SEnrico Granata 14520a305db7SEnrico Granata static OptionDefinition g_option_table[]; 14530a305db7SEnrico Granata 14540a305db7SEnrico Granata // Instance variables to hold the values for command options. 14550a305db7SEnrico Granata 14560a305db7SEnrico Granata bool m_allow_reload; 14570a305db7SEnrico Granata }; 14580a305db7SEnrico Granata 1459a9dbf432SEnrico Granata bool 14605a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 1461a9dbf432SEnrico Granata { 1462a9dbf432SEnrico Granata 1463a9dbf432SEnrico Granata if (m_interpreter.GetDebugger().GetScriptLanguage() != lldb::eScriptLanguagePython) 1464a9dbf432SEnrico Granata { 1465a9dbf432SEnrico Granata result.AppendError ("only scripting language supported for module importing is currently Python"); 1466a9dbf432SEnrico Granata result.SetStatus (eReturnStatusFailed); 1467a9dbf432SEnrico Granata return false; 1468a9dbf432SEnrico Granata } 1469a9dbf432SEnrico Granata 14705a988416SJim Ingham size_t argc = command.GetArgumentCount(); 1471a9dbf432SEnrico Granata 1472a9dbf432SEnrico Granata if (argc != 1) 1473a9dbf432SEnrico Granata { 1474a9dbf432SEnrico Granata result.AppendError ("'command script import' requires one argument"); 1475a9dbf432SEnrico Granata result.SetStatus (eReturnStatusFailed); 1476a9dbf432SEnrico Granata return false; 1477a9dbf432SEnrico Granata } 1478a9dbf432SEnrico Granata 14795a988416SJim Ingham std::string path = command.GetArgumentAtIndex(0); 1480a9dbf432SEnrico Granata Error error; 1481a9dbf432SEnrico Granata 1482c9d645d3SGreg Clayton const bool init_session = true; 1483078551c7SEnrico Granata // FIXME: this is necessary because CommandObject::CheckRequirements() assumes that 1484078551c7SEnrico Granata // commands won't ever be recursively invoked, but it's actually possible to craft 1485078551c7SEnrico Granata // a Python script that does other "command script imports" in __lldb_init_module 1486078551c7SEnrico Granata // the real fix is to have recursive commands possible with a CommandInvocation object 1487078551c7SEnrico Granata // separate from the CommandObject itself, so that recursive command invocations 1488078551c7SEnrico Granata // won't stomp on each other (wrt to execution contents, options, and more) 1489078551c7SEnrico Granata m_exe_ctx.Clear(); 1490a9dbf432SEnrico Granata if (m_interpreter.GetScriptInterpreter()->LoadScriptingModule(path.c_str(), 14910a305db7SEnrico Granata m_options.m_allow_reload, 1492c9d645d3SGreg Clayton init_session, 1493a9dbf432SEnrico Granata error)) 1494a9dbf432SEnrico Granata { 1495a9dbf432SEnrico Granata result.SetStatus (eReturnStatusSuccessFinishNoResult); 1496a9dbf432SEnrico Granata } 1497a9dbf432SEnrico Granata else 1498a9dbf432SEnrico Granata { 1499a9dbf432SEnrico Granata result.AppendErrorWithFormat("module importing failed: %s", error.AsCString()); 1500a9dbf432SEnrico Granata result.SetStatus (eReturnStatusFailed); 1501a9dbf432SEnrico Granata } 1502a9dbf432SEnrico Granata 1503a9dbf432SEnrico Granata return result.Succeeded(); 1504a9dbf432SEnrico Granata } 15050a305db7SEnrico Granata 15065a988416SJim Ingham CommandOptions m_options; 1507a9dbf432SEnrico Granata }; 1508223383edSEnrico Granata 15090a305db7SEnrico Granata OptionDefinition 15100a305db7SEnrico Granata CommandObjectCommandsScriptImport::CommandOptions::g_option_table[] = 15110a305db7SEnrico Granata { 1512e0c70f1bSEnrico Granata { LLDB_OPT_SET_1, false, "allow-reload", 'r', no_argument, NULL, 0, eArgTypeNone, "Allow the script to be loaded even if it was already loaded before. This argument exists for backwards compatibility, but reloading is always allowed, whether you specify it or not."}, 15130a305db7SEnrico Granata { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 15140a305db7SEnrico Granata }; 15150a305db7SEnrico Granata 15160a305db7SEnrico Granata 1517223383edSEnrico Granata //------------------------------------------------------------------------- 1518223383edSEnrico Granata // CommandObjectCommandsScriptAdd 1519223383edSEnrico Granata //------------------------------------------------------------------------- 1520223383edSEnrico Granata 15215a988416SJim Ingham class CommandObjectCommandsScriptAdd : public CommandObjectParsed 1522223383edSEnrico Granata { 15235a988416SJim Ingham public: 15245a988416SJim Ingham CommandObjectCommandsScriptAdd(CommandInterpreter &interpreter) : 15255a988416SJim Ingham CommandObjectParsed (interpreter, 15265a988416SJim Ingham "command script add", 15275a988416SJim Ingham "Add a scripted function as an LLDB command.", 15285a988416SJim Ingham NULL), 15295a988416SJim Ingham m_options (interpreter) 15305a988416SJim Ingham { 15315a988416SJim Ingham CommandArgumentEntry arg1; 15325a988416SJim Ingham CommandArgumentData cmd_arg; 15335a988416SJim Ingham 15345a988416SJim Ingham // Define the first (and only) variant of this arg. 15355a988416SJim Ingham cmd_arg.arg_type = eArgTypeCommandName; 15365a988416SJim Ingham cmd_arg.arg_repetition = eArgRepeatPlain; 15375a988416SJim Ingham 15385a988416SJim Ingham // There is only one variant this argument could be; put it into the argument entry. 15395a988416SJim Ingham arg1.push_back (cmd_arg); 15405a988416SJim Ingham 15415a988416SJim Ingham // Push the data for the first argument into the m_arguments vector. 15425a988416SJim Ingham m_arguments.push_back (arg1); 15435a988416SJim Ingham } 15445a988416SJim Ingham 15455a988416SJim Ingham ~CommandObjectCommandsScriptAdd () 15465a988416SJim Ingham { 15475a988416SJim Ingham } 15485a988416SJim Ingham 15495a988416SJim Ingham virtual Options * 15505a988416SJim Ingham GetOptions () 15515a988416SJim Ingham { 15525a988416SJim Ingham return &m_options; 15535a988416SJim Ingham } 15545a988416SJim Ingham 15555a988416SJim Ingham protected: 1556223383edSEnrico Granata 1557223383edSEnrico Granata class CommandOptions : public Options 1558223383edSEnrico Granata { 1559223383edSEnrico Granata public: 1560223383edSEnrico Granata 1561223383edSEnrico Granata CommandOptions (CommandInterpreter &interpreter) : 1562223383edSEnrico Granata Options (interpreter) 1563223383edSEnrico Granata { 1564223383edSEnrico Granata } 1565223383edSEnrico Granata 1566223383edSEnrico Granata virtual 1567223383edSEnrico Granata ~CommandOptions (){} 1568223383edSEnrico Granata 1569223383edSEnrico Granata virtual Error 1570223383edSEnrico Granata SetOptionValue (uint32_t option_idx, const char *option_arg) 1571223383edSEnrico Granata { 1572223383edSEnrico Granata Error error; 15733bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 1574223383edSEnrico Granata 1575223383edSEnrico Granata switch (short_option) 1576223383edSEnrico Granata { 1577223383edSEnrico Granata case 'f': 1578223383edSEnrico Granata m_funct_name = std::string(option_arg); 1579223383edSEnrico Granata break; 15800a305db7SEnrico Granata case 's': 15810a305db7SEnrico Granata m_synchronous = (ScriptedCommandSynchronicity) Args::StringToOptionEnum(option_arg, g_option_table[option_idx].enum_values, 0, error); 15820a305db7SEnrico Granata if (!error.Success()) 15830a305db7SEnrico Granata error.SetErrorStringWithFormat ("unrecognized value for synchronicity '%s'", option_arg); 15840a305db7SEnrico Granata break; 1585223383edSEnrico Granata default: 158686edbf41SGreg Clayton error.SetErrorStringWithFormat ("unrecognized option '%c'", short_option); 1587223383edSEnrico Granata break; 1588223383edSEnrico Granata } 1589223383edSEnrico Granata 1590223383edSEnrico Granata return error; 1591223383edSEnrico Granata } 1592223383edSEnrico Granata 1593223383edSEnrico Granata void 1594223383edSEnrico Granata OptionParsingStarting () 1595223383edSEnrico Granata { 1596223383edSEnrico Granata m_funct_name = ""; 15970a305db7SEnrico Granata m_synchronous = eScriptedCommandSynchronicitySynchronous; 1598223383edSEnrico Granata } 1599223383edSEnrico Granata 1600223383edSEnrico Granata const OptionDefinition* 1601223383edSEnrico Granata GetDefinitions () 1602223383edSEnrico Granata { 1603223383edSEnrico Granata return g_option_table; 1604223383edSEnrico Granata } 1605223383edSEnrico Granata 1606223383edSEnrico Granata // Options table: Required for subclasses of Options. 1607223383edSEnrico Granata 1608223383edSEnrico Granata static OptionDefinition g_option_table[]; 1609223383edSEnrico Granata 1610223383edSEnrico Granata // Instance variables to hold the values for command options. 1611223383edSEnrico Granata 1612223383edSEnrico Granata std::string m_funct_name; 16130a305db7SEnrico Granata ScriptedCommandSynchronicity m_synchronous; 1614223383edSEnrico Granata }; 1615223383edSEnrico Granata 16165a988416SJim Ingham private: 1617223383edSEnrico Granata class PythonAliasReader : public InputReaderEZ 1618223383edSEnrico Granata { 1619223383edSEnrico Granata private: 1620223383edSEnrico Granata CommandInterpreter& m_interpreter; 1621223383edSEnrico Granata std::string m_cmd_name; 16220a305db7SEnrico Granata ScriptedCommandSynchronicity m_synchronous; 1623223383edSEnrico Granata StringList m_user_input; 1624223383edSEnrico Granata DISALLOW_COPY_AND_ASSIGN (PythonAliasReader); 1625223383edSEnrico Granata public: 1626223383edSEnrico Granata PythonAliasReader(Debugger& debugger, 1627223383edSEnrico Granata CommandInterpreter& interpreter, 16280a305db7SEnrico Granata std::string cmd_name, 16290a305db7SEnrico Granata ScriptedCommandSynchronicity synch) : 1630223383edSEnrico Granata InputReaderEZ(debugger), 1631223383edSEnrico Granata m_interpreter(interpreter), 1632223383edSEnrico Granata m_cmd_name(cmd_name), 16330a305db7SEnrico Granata m_synchronous(synch), 1634223383edSEnrico Granata m_user_input() 1635223383edSEnrico Granata {} 1636223383edSEnrico Granata 1637223383edSEnrico Granata virtual 1638223383edSEnrico Granata ~PythonAliasReader() 1639223383edSEnrico Granata { 1640223383edSEnrico Granata } 1641223383edSEnrico Granata 1642223383edSEnrico Granata virtual void ActivateHandler(HandlerData& data) 1643223383edSEnrico Granata { 1644223383edSEnrico Granata StreamSP out_stream = data.GetOutStream(); 1645223383edSEnrico Granata bool batch_mode = data.GetBatchMode(); 1646223383edSEnrico Granata if (!batch_mode) 1647223383edSEnrico Granata { 1648223383edSEnrico Granata out_stream->Printf ("%s\n", g_python_command_instructions); 1649223383edSEnrico Granata if (data.reader.GetPrompt()) 1650223383edSEnrico Granata out_stream->Printf ("%s", data.reader.GetPrompt()); 1651223383edSEnrico Granata out_stream->Flush(); 1652223383edSEnrico Granata } 1653223383edSEnrico Granata } 1654223383edSEnrico Granata 1655223383edSEnrico Granata virtual void ReactivateHandler(HandlerData& data) 1656223383edSEnrico Granata { 1657223383edSEnrico Granata StreamSP out_stream = data.GetOutStream(); 1658223383edSEnrico Granata bool batch_mode = data.GetBatchMode(); 1659223383edSEnrico Granata if (data.reader.GetPrompt() && !batch_mode) 1660223383edSEnrico Granata { 1661223383edSEnrico Granata out_stream->Printf ("%s", data.reader.GetPrompt()); 1662223383edSEnrico Granata out_stream->Flush(); 1663223383edSEnrico Granata } 1664223383edSEnrico Granata } 1665223383edSEnrico Granata virtual void GotTokenHandler(HandlerData& data) 1666223383edSEnrico Granata { 1667223383edSEnrico Granata StreamSP out_stream = data.GetOutStream(); 1668223383edSEnrico Granata bool batch_mode = data.GetBatchMode(); 1669223383edSEnrico Granata if (data.bytes && data.bytes_len) 1670223383edSEnrico Granata { 1671223383edSEnrico Granata m_user_input.AppendString(data.bytes, data.bytes_len); 1672223383edSEnrico Granata } 1673223383edSEnrico Granata if (!data.reader.IsDone() && data.reader.GetPrompt() && !batch_mode) 1674223383edSEnrico Granata { 1675223383edSEnrico Granata out_stream->Printf ("%s", data.reader.GetPrompt()); 1676223383edSEnrico Granata out_stream->Flush(); 1677223383edSEnrico Granata } 1678223383edSEnrico Granata } 1679223383edSEnrico Granata virtual void InterruptHandler(HandlerData& data) 1680223383edSEnrico Granata { 1681223383edSEnrico Granata StreamSP out_stream = data.GetOutStream(); 1682223383edSEnrico Granata bool batch_mode = data.GetBatchMode(); 1683223383edSEnrico Granata data.reader.SetIsDone (true); 1684223383edSEnrico Granata if (!batch_mode) 1685223383edSEnrico Granata { 16860a305db7SEnrico Granata out_stream->Printf ("Warning: No script attached.\n"); 1687223383edSEnrico Granata out_stream->Flush(); 1688223383edSEnrico Granata } 1689223383edSEnrico Granata } 1690223383edSEnrico Granata virtual void EOFHandler(HandlerData& data) 1691223383edSEnrico Granata { 1692223383edSEnrico Granata data.reader.SetIsDone (true); 1693223383edSEnrico Granata } 1694223383edSEnrico Granata virtual void DoneHandler(HandlerData& data) 1695223383edSEnrico Granata { 1696223383edSEnrico Granata StreamSP out_stream = data.GetOutStream(); 1697223383edSEnrico Granata 1698223383edSEnrico Granata ScriptInterpreter *interpreter = data.reader.GetDebugger().GetCommandInterpreter().GetScriptInterpreter(); 1699223383edSEnrico Granata if (!interpreter) 1700223383edSEnrico Granata { 17010a305db7SEnrico Granata out_stream->Printf ("Script interpreter missing: no script attached.\n"); 1702223383edSEnrico Granata out_stream->Flush(); 1703223383edSEnrico Granata return; 1704223383edSEnrico Granata } 1705a73b7df7SEnrico Granata std::string funct_name_str; 1706223383edSEnrico Granata if (!interpreter->GenerateScriptAliasFunction (m_user_input, 1707a73b7df7SEnrico Granata funct_name_str)) 1708223383edSEnrico Granata { 17090a305db7SEnrico Granata out_stream->Printf ("Unable to create function: no script attached.\n"); 1710223383edSEnrico Granata out_stream->Flush(); 1711223383edSEnrico Granata return; 1712223383edSEnrico Granata } 1713a73b7df7SEnrico Granata if (funct_name_str.empty()) 1714223383edSEnrico Granata { 17150a305db7SEnrico Granata out_stream->Printf ("Unable to obtain a function name: no script attached.\n"); 1716223383edSEnrico Granata out_stream->Flush(); 1717223383edSEnrico Granata return; 1718223383edSEnrico Granata } 1719223383edSEnrico Granata // everything should be fine now, let's add this alias 1720223383edSEnrico Granata 1721223383edSEnrico Granata CommandObjectSP command_obj_sp(new CommandObjectPythonFunction(m_interpreter, 1722223383edSEnrico Granata m_cmd_name, 1723a73b7df7SEnrico Granata funct_name_str.c_str(), 17240a305db7SEnrico Granata m_synchronous)); 1725223383edSEnrico Granata 17260a305db7SEnrico Granata if (!m_interpreter.AddUserCommand(m_cmd_name, command_obj_sp, true)) 1727223383edSEnrico Granata { 17280a305db7SEnrico Granata out_stream->Printf ("Unable to add selected command: no script attached.\n"); 1729223383edSEnrico Granata out_stream->Flush(); 1730223383edSEnrico Granata return; 1731223383edSEnrico Granata } 1732223383edSEnrico Granata } 1733223383edSEnrico Granata }; 1734223383edSEnrico Granata 17355a988416SJim Ingham protected: 1736223383edSEnrico Granata bool 17375a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 1738223383edSEnrico Granata { 173999f0b8f9SEnrico Granata 174099f0b8f9SEnrico Granata if (m_interpreter.GetDebugger().GetScriptLanguage() != lldb::eScriptLanguagePython) 174199f0b8f9SEnrico Granata { 174299f0b8f9SEnrico Granata result.AppendError ("only scripting language supported for scripted commands is currently Python"); 174399f0b8f9SEnrico Granata result.SetStatus (eReturnStatusFailed); 174499f0b8f9SEnrico Granata return false; 174599f0b8f9SEnrico Granata } 174699f0b8f9SEnrico Granata 17475a988416SJim Ingham size_t argc = command.GetArgumentCount(); 1748223383edSEnrico Granata 1749223383edSEnrico Granata if (argc != 1) 1750223383edSEnrico Granata { 1751223383edSEnrico Granata result.AppendError ("'command script add' requires one argument"); 1752223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1753223383edSEnrico Granata return false; 1754223383edSEnrico Granata } 1755223383edSEnrico Granata 17565a988416SJim Ingham std::string cmd_name = command.GetArgumentAtIndex(0); 1757223383edSEnrico Granata 1758223383edSEnrico Granata if (m_options.m_funct_name.empty()) 1759223383edSEnrico Granata { 1760223383edSEnrico Granata InputReaderSP reader_sp (new PythonAliasReader (m_interpreter.GetDebugger(), 1761223383edSEnrico Granata m_interpreter, 17620a305db7SEnrico Granata cmd_name, 17630a305db7SEnrico Granata m_options.m_synchronous)); 1764223383edSEnrico Granata 1765223383edSEnrico Granata if (reader_sp) 1766223383edSEnrico Granata { 1767223383edSEnrico Granata 1768223383edSEnrico Granata InputReaderEZ::InitializationParameters ipr; 1769223383edSEnrico Granata 1770223383edSEnrico Granata Error err (reader_sp->Initialize (ipr.SetBaton(NULL).SetPrompt(" "))); 1771223383edSEnrico Granata if (err.Success()) 1772223383edSEnrico Granata { 1773223383edSEnrico Granata m_interpreter.GetDebugger().PushInputReader (reader_sp); 1774223383edSEnrico Granata result.SetStatus (eReturnStatusSuccessFinishNoResult); 1775223383edSEnrico Granata } 1776223383edSEnrico Granata else 1777223383edSEnrico Granata { 1778223383edSEnrico Granata result.AppendError (err.AsCString()); 1779223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1780223383edSEnrico Granata } 1781223383edSEnrico Granata } 1782223383edSEnrico Granata else 1783223383edSEnrico Granata { 1784223383edSEnrico Granata result.AppendError("out of memory"); 1785223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1786223383edSEnrico Granata } 1787223383edSEnrico Granata } 1788223383edSEnrico Granata else 1789223383edSEnrico Granata { 17900a305db7SEnrico Granata CommandObjectSP new_cmd(new CommandObjectPythonFunction(m_interpreter, 17910a305db7SEnrico Granata cmd_name, 17920a305db7SEnrico Granata m_options.m_funct_name, 17930a305db7SEnrico Granata m_options.m_synchronous)); 17940a305db7SEnrico Granata if (m_interpreter.AddUserCommand(cmd_name, new_cmd, true)) 1795223383edSEnrico Granata { 1796223383edSEnrico Granata result.SetStatus (eReturnStatusSuccessFinishNoResult); 1797223383edSEnrico Granata } 1798223383edSEnrico Granata else 1799223383edSEnrico Granata { 1800223383edSEnrico Granata result.AppendError("cannot add command"); 1801223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1802223383edSEnrico Granata } 1803223383edSEnrico Granata } 1804223383edSEnrico Granata 1805223383edSEnrico Granata return result.Succeeded(); 1806223383edSEnrico Granata 1807223383edSEnrico Granata } 18085a988416SJim Ingham 18095a988416SJim Ingham CommandOptions m_options; 1810223383edSEnrico Granata }; 1811223383edSEnrico Granata 18120a305db7SEnrico Granata static OptionEnumValueElement g_script_synchro_type[] = 18130a305db7SEnrico Granata { 18140a305db7SEnrico Granata { eScriptedCommandSynchronicitySynchronous, "synchronous", "Run synchronous"}, 18150a305db7SEnrico Granata { eScriptedCommandSynchronicityAsynchronous, "asynchronous", "Run asynchronous"}, 18160a305db7SEnrico Granata { eScriptedCommandSynchronicityCurrentValue, "current", "Do not alter current setting"}, 18170a305db7SEnrico Granata { 0, NULL, NULL } 18180a305db7SEnrico Granata }; 18190a305db7SEnrico Granata 1820223383edSEnrico Granata OptionDefinition 1821223383edSEnrico Granata CommandObjectCommandsScriptAdd::CommandOptions::g_option_table[] = 1822223383edSEnrico Granata { 18239128ee2fSEnrico Granata { LLDB_OPT_SET_1, false, "function", 'f', required_argument, NULL, 0, eArgTypePythonFunction, "Name of the Python function to bind to this command name."}, 18240a305db7SEnrico Granata { LLDB_OPT_SET_1, false, "synchronicity", 's', required_argument, g_script_synchro_type, 0, eArgTypeScriptedCommandSynchronicity, "Set the synchronicity of this command's executions with regard to LLDB event system."}, 1825223383edSEnrico Granata { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 1826223383edSEnrico Granata }; 1827223383edSEnrico Granata 1828223383edSEnrico Granata //------------------------------------------------------------------------- 1829223383edSEnrico Granata // CommandObjectCommandsScriptList 1830223383edSEnrico Granata //------------------------------------------------------------------------- 1831223383edSEnrico Granata 18325a988416SJim Ingham class CommandObjectCommandsScriptList : public CommandObjectParsed 1833223383edSEnrico Granata { 1834223383edSEnrico Granata private: 1835223383edSEnrico Granata 1836223383edSEnrico Granata public: 1837223383edSEnrico Granata CommandObjectCommandsScriptList(CommandInterpreter &interpreter) : 18385a988416SJim Ingham CommandObjectParsed (interpreter, 1839223383edSEnrico Granata "command script list", 1840223383edSEnrico Granata "List defined scripted commands.", 1841223383edSEnrico Granata NULL) 1842223383edSEnrico Granata { 1843223383edSEnrico Granata } 1844223383edSEnrico Granata 1845223383edSEnrico Granata ~CommandObjectCommandsScriptList () 1846223383edSEnrico Granata { 1847223383edSEnrico Granata } 1848223383edSEnrico Granata 1849223383edSEnrico Granata bool 18505a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 1851223383edSEnrico Granata { 1852223383edSEnrico Granata 1853223383edSEnrico Granata m_interpreter.GetHelp(result, 1854223383edSEnrico Granata CommandInterpreter::eCommandTypesUserDef); 1855223383edSEnrico Granata 1856223383edSEnrico Granata result.SetStatus (eReturnStatusSuccessFinishResult); 1857223383edSEnrico Granata 1858223383edSEnrico Granata return true; 1859223383edSEnrico Granata 1860223383edSEnrico Granata 1861223383edSEnrico Granata } 1862223383edSEnrico Granata }; 1863223383edSEnrico Granata 1864223383edSEnrico Granata //------------------------------------------------------------------------- 1865223383edSEnrico Granata // CommandObjectCommandsScriptClear 1866223383edSEnrico Granata //------------------------------------------------------------------------- 1867223383edSEnrico Granata 18685a988416SJim Ingham class CommandObjectCommandsScriptClear : public CommandObjectParsed 1869223383edSEnrico Granata { 1870223383edSEnrico Granata private: 1871223383edSEnrico Granata 1872223383edSEnrico Granata public: 1873223383edSEnrico Granata CommandObjectCommandsScriptClear(CommandInterpreter &interpreter) : 18745a988416SJim Ingham CommandObjectParsed (interpreter, 1875223383edSEnrico Granata "command script clear", 1876223383edSEnrico Granata "Delete all scripted commands.", 1877223383edSEnrico Granata NULL) 1878223383edSEnrico Granata { 1879223383edSEnrico Granata } 1880223383edSEnrico Granata 1881223383edSEnrico Granata ~CommandObjectCommandsScriptClear () 1882223383edSEnrico Granata { 1883223383edSEnrico Granata } 1884223383edSEnrico Granata 18855a988416SJim Ingham protected: 1886223383edSEnrico Granata bool 18875a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 1888223383edSEnrico Granata { 1889223383edSEnrico Granata 1890223383edSEnrico Granata m_interpreter.RemoveAllUser(); 1891223383edSEnrico Granata 1892223383edSEnrico Granata result.SetStatus (eReturnStatusSuccessFinishResult); 1893223383edSEnrico Granata 1894223383edSEnrico Granata return true; 1895223383edSEnrico Granata } 1896223383edSEnrico Granata }; 1897223383edSEnrico Granata 1898223383edSEnrico Granata //------------------------------------------------------------------------- 1899223383edSEnrico Granata // CommandObjectCommandsScriptDelete 1900223383edSEnrico Granata //------------------------------------------------------------------------- 1901223383edSEnrico Granata 19025a988416SJim Ingham class CommandObjectCommandsScriptDelete : public CommandObjectParsed 1903223383edSEnrico Granata { 1904223383edSEnrico Granata public: 1905223383edSEnrico Granata CommandObjectCommandsScriptDelete(CommandInterpreter &interpreter) : 19065a988416SJim Ingham CommandObjectParsed (interpreter, 1907223383edSEnrico Granata "command script delete", 1908223383edSEnrico Granata "Delete a scripted command.", 1909223383edSEnrico Granata NULL) 1910223383edSEnrico Granata { 1911223383edSEnrico Granata CommandArgumentEntry arg1; 1912223383edSEnrico Granata CommandArgumentData cmd_arg; 1913223383edSEnrico Granata 1914223383edSEnrico Granata // Define the first (and only) variant of this arg. 1915223383edSEnrico Granata cmd_arg.arg_type = eArgTypeCommandName; 1916223383edSEnrico Granata cmd_arg.arg_repetition = eArgRepeatPlain; 1917223383edSEnrico Granata 1918223383edSEnrico Granata // There is only one variant this argument could be; put it into the argument entry. 1919223383edSEnrico Granata arg1.push_back (cmd_arg); 1920223383edSEnrico Granata 1921223383edSEnrico Granata // Push the data for the first argument into the m_arguments vector. 1922223383edSEnrico Granata m_arguments.push_back (arg1); 1923223383edSEnrico Granata } 1924223383edSEnrico Granata 1925223383edSEnrico Granata ~CommandObjectCommandsScriptDelete () 1926223383edSEnrico Granata { 1927223383edSEnrico Granata } 1928223383edSEnrico Granata 19295a988416SJim Ingham protected: 1930223383edSEnrico Granata bool 19315a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 1932223383edSEnrico Granata { 1933223383edSEnrico Granata 19345a988416SJim Ingham size_t argc = command.GetArgumentCount(); 1935223383edSEnrico Granata 1936223383edSEnrico Granata if (argc != 1) 1937223383edSEnrico Granata { 1938223383edSEnrico Granata result.AppendError ("'command script delete' requires one argument"); 1939223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1940223383edSEnrico Granata return false; 1941223383edSEnrico Granata } 1942223383edSEnrico Granata 19435a988416SJim Ingham const char* cmd_name = command.GetArgumentAtIndex(0); 1944223383edSEnrico Granata 1945223383edSEnrico Granata if (cmd_name && *cmd_name && m_interpreter.HasUserCommands() && m_interpreter.UserCommandExists(cmd_name)) 1946223383edSEnrico Granata { 1947223383edSEnrico Granata m_interpreter.RemoveUser(cmd_name); 1948223383edSEnrico Granata result.SetStatus (eReturnStatusSuccessFinishResult); 1949223383edSEnrico Granata } 1950223383edSEnrico Granata else 1951223383edSEnrico Granata { 1952223383edSEnrico Granata result.AppendErrorWithFormat ("command %s not found", cmd_name); 1953223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1954223383edSEnrico Granata } 1955223383edSEnrico Granata 1956223383edSEnrico Granata return result.Succeeded(); 1957223383edSEnrico Granata 1958223383edSEnrico Granata } 1959223383edSEnrico Granata }; 1960223383edSEnrico Granata 1961223383edSEnrico Granata #pragma mark CommandObjectMultiwordCommandsScript 1962223383edSEnrico Granata 1963223383edSEnrico Granata //------------------------------------------------------------------------- 1964223383edSEnrico Granata // CommandObjectMultiwordCommandsScript 1965223383edSEnrico Granata //------------------------------------------------------------------------- 1966223383edSEnrico Granata 1967223383edSEnrico Granata class CommandObjectMultiwordCommandsScript : public CommandObjectMultiword 1968223383edSEnrico Granata { 1969223383edSEnrico Granata public: 1970223383edSEnrico Granata CommandObjectMultiwordCommandsScript (CommandInterpreter &interpreter) : 1971223383edSEnrico Granata CommandObjectMultiword (interpreter, 1972223383edSEnrico Granata "command script", 1973223383edSEnrico Granata "A set of commands for managing or customizing script commands.", 1974223383edSEnrico Granata "command script <subcommand> [<subcommand-options>]") 1975223383edSEnrico Granata { 1976223383edSEnrico Granata LoadSubCommand ("add", CommandObjectSP (new CommandObjectCommandsScriptAdd (interpreter))); 1977223383edSEnrico Granata LoadSubCommand ("delete", CommandObjectSP (new CommandObjectCommandsScriptDelete (interpreter))); 1978223383edSEnrico Granata LoadSubCommand ("clear", CommandObjectSP (new CommandObjectCommandsScriptClear (interpreter))); 1979223383edSEnrico Granata LoadSubCommand ("list", CommandObjectSP (new CommandObjectCommandsScriptList (interpreter))); 1980a9dbf432SEnrico Granata LoadSubCommand ("import", CommandObjectSP (new CommandObjectCommandsScriptImport (interpreter))); 1981223383edSEnrico Granata } 1982223383edSEnrico Granata 1983223383edSEnrico Granata ~CommandObjectMultiwordCommandsScript () 1984223383edSEnrico Granata { 1985223383edSEnrico Granata } 1986223383edSEnrico Granata 1987223383edSEnrico Granata }; 1988223383edSEnrico Granata 1989223383edSEnrico Granata 1990ebc09c36SJim Ingham #pragma mark CommandObjectMultiwordCommands 1991ebc09c36SJim Ingham 1992ebc09c36SJim Ingham //------------------------------------------------------------------------- 1993ebc09c36SJim Ingham // CommandObjectMultiwordCommands 1994ebc09c36SJim Ingham //------------------------------------------------------------------------- 1995ebc09c36SJim Ingham 1996ebc09c36SJim Ingham CommandObjectMultiwordCommands::CommandObjectMultiwordCommands (CommandInterpreter &interpreter) : 1997a7015092SGreg Clayton CommandObjectMultiword (interpreter, 19980e5e5a79SGreg Clayton "command", 19993f4c09c1SCaroline Tice "A set of commands for managing or customizing the debugger commands.", 20000e5e5a79SGreg Clayton "command <subcommand> [<subcommand-options>]") 2001ebc09c36SJim Ingham { 2002a7015092SGreg Clayton LoadSubCommand ("source", CommandObjectSP (new CommandObjectCommandsSource (interpreter))); 2003a7015092SGreg Clayton LoadSubCommand ("alias", CommandObjectSP (new CommandObjectCommandsAlias (interpreter))); 2004a7015092SGreg Clayton LoadSubCommand ("unalias", CommandObjectSP (new CommandObjectCommandsUnalias (interpreter))); 2005de164aaaSGreg Clayton LoadSubCommand ("regex", CommandObjectSP (new CommandObjectCommandsAddRegex (interpreter))); 2006a5a97ebeSJim Ingham LoadSubCommand ("history", CommandObjectSP (new CommandObjectCommandsHistory (interpreter))); 2007223383edSEnrico Granata LoadSubCommand ("script", CommandObjectSP (new CommandObjectMultiwordCommandsScript (interpreter))); 2008ebc09c36SJim Ingham } 2009ebc09c36SJim Ingham 2010ebc09c36SJim Ingham CommandObjectMultiwordCommands::~CommandObjectMultiwordCommands () 2011ebc09c36SJim Ingham { 2012ebc09c36SJim Ingham } 2013ebc09c36SJim Ingham 2014