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': 329012d4fcaSEnrico Granata m_stop_on_error.SetCurrentValue(Args::StringToBoolean(option_arg, true, &success)); 330e16c50a1SJim Ingham if (!success) 33186edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid value for stop-on-error: %s", option_arg); 332e16c50a1SJim Ingham break; 333e16c50a1SJim Ingham case 'c': 334e16c50a1SJim Ingham m_stop_on_continue = Args::StringToBoolean(option_arg, true, &success); 335e16c50a1SJim Ingham if (!success) 33686edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid value for stop-on-continue: %s", option_arg); 337e16c50a1SJim Ingham break; 338e16c50a1SJim Ingham default: 33986edbf41SGreg Clayton error.SetErrorStringWithFormat ("unrecognized option '%c'", short_option); 340e16c50a1SJim Ingham break; 341e16c50a1SJim Ingham } 342e16c50a1SJim Ingham 343e16c50a1SJim Ingham return error; 344e16c50a1SJim Ingham } 345e16c50a1SJim Ingham 346e16c50a1SJim Ingham void 347f6b8b581SGreg Clayton OptionParsingStarting () 348e16c50a1SJim Ingham { 349012d4fcaSEnrico Granata m_stop_on_error.Clear(); 350e16c50a1SJim Ingham m_stop_on_continue = true; 351e16c50a1SJim Ingham } 352e16c50a1SJim Ingham 353e0d378b3SGreg Clayton const OptionDefinition* 354e16c50a1SJim Ingham GetDefinitions () 355e16c50a1SJim Ingham { 356e16c50a1SJim Ingham return g_option_table; 357e16c50a1SJim Ingham } 358e16c50a1SJim Ingham 359e16c50a1SJim Ingham // Options table: Required for subclasses of Options. 360e16c50a1SJim Ingham 361e0d378b3SGreg Clayton static OptionDefinition g_option_table[]; 362e16c50a1SJim Ingham 363e16c50a1SJim Ingham // Instance variables to hold the values for command options. 364e16c50a1SJim Ingham 365012d4fcaSEnrico Granata OptionValueBoolean m_stop_on_error; 366e16c50a1SJim Ingham bool m_stop_on_continue; 367e16c50a1SJim Ingham }; 368e16c50a1SJim Ingham 369ebc09c36SJim Ingham bool 3705a988416SJim Ingham DoExecute(Args& command, CommandReturnObject &result) 371ebc09c36SJim Ingham { 372c7bece56SGreg Clayton const size_t argc = command.GetArgumentCount(); 373ebc09c36SJim Ingham if (argc == 1) 374ebc09c36SJim Ingham { 3755a988416SJim Ingham const char *filename = command.GetArgumentAtIndex(0); 376ebc09c36SJim Ingham 377ebc09c36SJim Ingham result.AppendMessageWithFormat ("Executing commands in '%s'.\n", filename); 378ebc09c36SJim Ingham 3791ee3853fSJohnny Chen FileSpec cmd_file (filename, true); 380e16c50a1SJim Ingham ExecutionContext *exe_ctx = NULL; // Just use the default context. 381e16c50a1SJim Ingham bool echo_commands = true; 382e16c50a1SJim Ingham bool print_results = true; 383012d4fcaSEnrico Granata bool stop_on_error = m_options.m_stop_on_error.OptionWasSet() ? (bool)m_options.m_stop_on_error : m_interpreter.GetStopCmdSourceOnError(); 384ebc09c36SJim Ingham 385e16c50a1SJim Ingham m_interpreter.HandleCommandsFromFile (cmd_file, 386e16c50a1SJim Ingham exe_ctx, 387e16c50a1SJim Ingham m_options.m_stop_on_continue, 388012d4fcaSEnrico Granata stop_on_error, 389e16c50a1SJim Ingham echo_commands, 390e16c50a1SJim Ingham print_results, 3915f5ab602SEnrico Granata eLazyBoolCalculate, 392e16c50a1SJim Ingham result); 393ebc09c36SJim Ingham } 394ebc09c36SJim Ingham else 395ebc09c36SJim Ingham { 396ebc09c36SJim Ingham result.AppendErrorWithFormat("'%s' takes exactly one executable filename argument.\n", GetCommandName()); 397ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 398ebc09c36SJim Ingham } 399ebc09c36SJim Ingham return result.Succeeded(); 400ebc09c36SJim Ingham 401ebc09c36SJim Ingham } 4025a988416SJim Ingham CommandOptions m_options; 403ebc09c36SJim Ingham }; 404ebc09c36SJim Ingham 405e0d378b3SGreg Clayton OptionDefinition 406e16c50a1SJim Ingham CommandObjectCommandsSource::CommandOptions::g_option_table[] = 407e16c50a1SJim Ingham { 408e16c50a1SJim Ingham { LLDB_OPT_SET_ALL, false, "stop-on-error", 'e', required_argument, NULL, 0, eArgTypeBoolean, "If true, stop executing commands on error."}, 409e16c50a1SJim Ingham { LLDB_OPT_SET_ALL, false, "stop-on-continue", 'c', required_argument, NULL, 0, eArgTypeBoolean, "If true, stop executing commands on continue."}, 410e16c50a1SJim Ingham { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 411e16c50a1SJim Ingham }; 412e16c50a1SJim Ingham 413ebc09c36SJim Ingham #pragma mark CommandObjectCommandsAlias 414ebc09c36SJim Ingham //------------------------------------------------------------------------- 415ebc09c36SJim Ingham // CommandObjectCommandsAlias 416ebc09c36SJim Ingham //------------------------------------------------------------------------- 417ebc09c36SJim Ingham 418be93a35aSEnrico Granata static const char *g_python_command_instructions = "Enter your Python command(s). Type 'DONE' to end.\n" 419be93a35aSEnrico Granata "You must define a Python function with this signature:\n" 42040d55710SEnrico Granata "def my_command_impl(debugger, args, result, internal_dict):"; 421be93a35aSEnrico Granata 422be93a35aSEnrico Granata 4235a988416SJim Ingham class CommandObjectCommandsAlias : public CommandObjectRaw 424ebc09c36SJim Ingham { 425be93a35aSEnrico Granata 426be93a35aSEnrico Granata 427ebc09c36SJim Ingham public: 428a7015092SGreg Clayton CommandObjectCommandsAlias (CommandInterpreter &interpreter) : 4295a988416SJim Ingham CommandObjectRaw (interpreter, 4300e5e5a79SGreg Clayton "command alias", 431e3d26315SCaroline Tice "Allow users to define their own debugger command abbreviations.", 432405fe67fSCaroline Tice NULL) 433ebc09c36SJim Ingham { 434ebc09c36SJim Ingham SetHelpLong( 435ebc09c36SJim Ingham "'alias' allows the user to create a short-cut or abbreviation for long \n\ 436ebc09c36SJim Ingham commands, multi-word commands, and commands that take particular options. \n\ 437ebc09c36SJim Ingham Below are some simple examples of how one might use the 'alias' command: \n\ 43869c12ccbSJason Molenda \n 'command alias sc script' // Creates the abbreviation 'sc' for the 'script' \n\ 439ebc09c36SJim Ingham // command. \n\ 44069c12ccbSJason Molenda 'command alias bp breakpoint' // Creates the abbreviation 'bp' for the 'breakpoint' \n\ 441ebc09c36SJim Ingham // command. Since breakpoint commands are two-word \n\ 442ebc09c36SJim Ingham // commands, the user will still need to enter the \n\ 443ebc09c36SJim Ingham // second word after 'bp', e.g. 'bp enable' or \n\ 444ebc09c36SJim Ingham // 'bp delete'. \n\ 44569c12ccbSJason Molenda 'command alias bpl breakpoint list' // Creates the abbreviation 'bpl' for the \n\ 446ebc09c36SJim Ingham // two-word command 'breakpoint list'. \n\ 447ebc09c36SJim Ingham \nAn alias can include some options for the command, with the values either \n\ 448ebc09c36SJim Ingham filled in at the time the alias is created, or specified as positional \n\ 449ebc09c36SJim Ingham arguments, to be filled in when the alias is invoked. The following example \n\ 450ebc09c36SJim Ingham shows how to create aliases with options: \n\ 451ebc09c36SJim Ingham \n\ 45269c12ccbSJason Molenda 'command alias bfl breakpoint set -f %1 -l %2' \n\ 453ebc09c36SJim Ingham \nThis creates the abbreviation 'bfl' (for break-file-line), with the -f and -l \n\ 454ebc09c36SJim Ingham options already part of the alias. So if the user wants to set a breakpoint \n\ 455ebc09c36SJim Ingham by file and line without explicitly having to use the -f and -l options, the \n\ 456ebc09c36SJim Ingham user can now use 'bfl' instead. The '%1' and '%2' are positional placeholders \n\ 457ebc09c36SJim Ingham for the actual arguments that will be passed when the alias command is used. \n\ 458ebc09c36SJim Ingham The number in the placeholder refers to the position/order the actual value \n\ 45981ded935SJim Ingham occupies when the alias is used. All the occurrences of '%1' in the alias \n\ 460ebc09c36SJim Ingham will be replaced with the first argument, all the occurrences of '%2' in the \n\ 461ebc09c36SJim Ingham alias will be replaced with the second argument, and so on. This also allows \n\ 462ebc09c36SJim Ingham actual arguments to be used multiple times within an alias (see 'process \n\ 46381ded935SJim Ingham launch' example below). \n\ 46481ded935SJim Ingham Note: the positional arguments must substitute as whole words in the resultant\n\ 46581ded935SJim Ingham command, so you can't at present do something like:\n\ 46681ded935SJim Ingham \n\ 46769c12ccbSJason Molenda command alias bcppfl breakpoint set -f %1.cpp -l %2\n\ 46881ded935SJim Ingham \n\ 46981ded935SJim Ingham to get the file extension \".cpp\" automatically appended. For more complex\n\ 47081ded935SJim Ingham aliasing, use the \"command regex\" command instead.\n\ 47181ded935SJim Ingham \nSo in the 'bfl' case, the actual file value will be \n\ 472ebc09c36SJim Ingham filled in with the first argument following 'bfl' and the actual line number \n\ 473ebc09c36SJim Ingham value will be filled in with the second argument. The user would use this \n\ 474ebc09c36SJim Ingham alias as follows: \n\ 47569c12ccbSJason Molenda \n (lldb) command alias bfl breakpoint set -f %1 -l %2 \n\ 476ebc09c36SJim Ingham <... some time later ...> \n\ 47709799af6SCaroline Tice (lldb) bfl my-file.c 137 \n\ 478ebc09c36SJim Ingham \nThis would be the same as if the user had entered \n\ 479ebc09c36SJim Ingham 'breakpoint set -f my-file.c -l 137'. \n\ 480ebc09c36SJim Ingham \nAnother example: \n\ 48169c12ccbSJason Molenda \n (lldb) command alias pltty process launch -s -o %1 -e %1 \n\ 48209799af6SCaroline Tice (lldb) pltty /dev/tty0 \n\ 483ebc09c36SJim Ingham // becomes 'process launch -s -o /dev/tty0 -e /dev/tty0' \n\ 484ebc09c36SJim Ingham \nIf the user always wanted to pass the same value to a particular option, the \n\ 485ebc09c36SJim Ingham alias could be defined with that value directly in the alias as a constant, \n\ 486ebc09c36SJim Ingham rather than using a positional placeholder: \n\ 48769c12ccbSJason Molenda \n command alias bl3 breakpoint set -f %1 -l 3 // Always sets a breakpoint on line \n\ 488ebc09c36SJim Ingham // 3 of whatever file is indicated. \n"); 489ebc09c36SJim Ingham 490405fe67fSCaroline Tice CommandArgumentEntry arg1; 491405fe67fSCaroline Tice CommandArgumentEntry arg2; 492405fe67fSCaroline Tice CommandArgumentEntry arg3; 493405fe67fSCaroline Tice CommandArgumentData alias_arg; 494405fe67fSCaroline Tice CommandArgumentData cmd_arg; 495405fe67fSCaroline Tice CommandArgumentData options_arg; 496405fe67fSCaroline Tice 497405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 498405fe67fSCaroline Tice alias_arg.arg_type = eArgTypeAliasName; 499405fe67fSCaroline Tice alias_arg.arg_repetition = eArgRepeatPlain; 500405fe67fSCaroline Tice 501405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 502405fe67fSCaroline Tice arg1.push_back (alias_arg); 503405fe67fSCaroline Tice 504405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 505405fe67fSCaroline Tice cmd_arg.arg_type = eArgTypeCommandName; 506405fe67fSCaroline Tice cmd_arg.arg_repetition = eArgRepeatPlain; 507405fe67fSCaroline Tice 508405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 509405fe67fSCaroline Tice arg2.push_back (cmd_arg); 510405fe67fSCaroline Tice 511405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 512405fe67fSCaroline Tice options_arg.arg_type = eArgTypeAliasOptions; 513405fe67fSCaroline Tice options_arg.arg_repetition = eArgRepeatOptional; 514405fe67fSCaroline Tice 515405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 516405fe67fSCaroline Tice arg3.push_back (options_arg); 517405fe67fSCaroline Tice 518405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 519405fe67fSCaroline Tice m_arguments.push_back (arg1); 520405fe67fSCaroline Tice m_arguments.push_back (arg2); 521405fe67fSCaroline Tice m_arguments.push_back (arg3); 522ebc09c36SJim Ingham } 523ebc09c36SJim Ingham 524ebc09c36SJim Ingham ~CommandObjectCommandsAlias () 525ebc09c36SJim Ingham { 526ebc09c36SJim Ingham } 527ebc09c36SJim Ingham 5285a988416SJim Ingham protected: 5295a988416SJim Ingham virtual bool 5305a988416SJim Ingham DoExecute (const char *raw_command_line, CommandReturnObject &result) 531844d2303SCaroline Tice { 532844d2303SCaroline Tice Args args (raw_command_line); 533844d2303SCaroline Tice std::string raw_command_string (raw_command_line); 534844d2303SCaroline Tice 535844d2303SCaroline Tice size_t argc = args.GetArgumentCount(); 536844d2303SCaroline Tice 537844d2303SCaroline Tice if (argc < 2) 538844d2303SCaroline Tice { 539844d2303SCaroline Tice result.AppendError ("'alias' requires at least two arguments"); 540844d2303SCaroline Tice result.SetStatus (eReturnStatusFailed); 541844d2303SCaroline Tice return false; 542844d2303SCaroline Tice } 543844d2303SCaroline Tice 544844d2303SCaroline Tice // Get the alias command. 545844d2303SCaroline Tice 546844d2303SCaroline Tice const std::string alias_command = args.GetArgumentAtIndex (0); 547844d2303SCaroline Tice 548844d2303SCaroline Tice // Strip the new alias name off 'raw_command_string' (leave it on args, which gets passed to 'Execute', which 549844d2303SCaroline Tice // does the stripping itself. 550844d2303SCaroline Tice size_t pos = raw_command_string.find (alias_command); 551844d2303SCaroline Tice if (pos == 0) 552844d2303SCaroline Tice { 553844d2303SCaroline Tice raw_command_string = raw_command_string.substr (alias_command.size()); 554844d2303SCaroline Tice pos = raw_command_string.find_first_not_of (' '); 555844d2303SCaroline Tice if ((pos != std::string::npos) && (pos > 0)) 556844d2303SCaroline Tice raw_command_string = raw_command_string.substr (pos); 557844d2303SCaroline Tice } 558844d2303SCaroline Tice else 559844d2303SCaroline Tice { 560844d2303SCaroline Tice result.AppendError ("Error parsing command string. No alias created."); 561844d2303SCaroline Tice result.SetStatus (eReturnStatusFailed); 562844d2303SCaroline Tice return false; 563844d2303SCaroline Tice } 564844d2303SCaroline Tice 565844d2303SCaroline Tice 566844d2303SCaroline Tice // Verify that the command is alias-able. 567844d2303SCaroline Tice if (m_interpreter.CommandExists (alias_command.c_str())) 568844d2303SCaroline Tice { 569844d2303SCaroline Tice result.AppendErrorWithFormat ("'%s' is a permanent debugger command and cannot be redefined.\n", 570844d2303SCaroline Tice alias_command.c_str()); 571844d2303SCaroline Tice result.SetStatus (eReturnStatusFailed); 572844d2303SCaroline Tice return false; 573844d2303SCaroline Tice } 574844d2303SCaroline Tice 575844d2303SCaroline Tice // Get CommandObject that is being aliased. The command name is read from the front of raw_command_string. 576844d2303SCaroline Tice // raw_command_string is returned with the name of the command object stripped off the front. 577844d2303SCaroline Tice CommandObject *cmd_obj = m_interpreter.GetCommandObjectForCommand (raw_command_string); 578844d2303SCaroline Tice 579844d2303SCaroline Tice if (!cmd_obj) 580844d2303SCaroline Tice { 58186edbf41SGreg Clayton result.AppendErrorWithFormat ("invalid command given to 'alias'. '%s' does not begin with a valid command." 582844d2303SCaroline Tice " No alias created.", raw_command_string.c_str()); 583844d2303SCaroline Tice result.SetStatus (eReturnStatusFailed); 584844d2303SCaroline Tice return false; 585844d2303SCaroline Tice } 586844d2303SCaroline Tice else if (!cmd_obj->WantsRawCommandString ()) 587844d2303SCaroline Tice { 588844d2303SCaroline Tice // Note that args was initialized with the original command, and has not been updated to this point. 589844d2303SCaroline Tice // Therefore can we pass it to the version of Execute that does not need/expect raw input in the alias. 5905a988416SJim Ingham return HandleAliasingNormalCommand (args, result); 591844d2303SCaroline Tice } 592844d2303SCaroline Tice else 593844d2303SCaroline Tice { 5945a988416SJim Ingham return HandleAliasingRawCommand (alias_command, raw_command_string, *cmd_obj, result); 5955a988416SJim Ingham } 5965a988416SJim Ingham return result.Succeeded(); 5975a988416SJim Ingham } 5985a988416SJim Ingham 5995a988416SJim Ingham bool 6005a988416SJim Ingham HandleAliasingRawCommand (const std::string &alias_command, std::string &raw_command_string, CommandObject &cmd_obj, CommandReturnObject &result) 6015a988416SJim Ingham { 602844d2303SCaroline Tice // Verify & handle any options/arguments passed to the alias command 603844d2303SCaroline Tice 604844d2303SCaroline Tice OptionArgVectorSP option_arg_vector_sp = OptionArgVectorSP (new OptionArgVector); 605844d2303SCaroline Tice OptionArgVector *option_arg_vector = option_arg_vector_sp.get(); 606844d2303SCaroline Tice 6075a988416SJim Ingham CommandObjectSP cmd_obj_sp = m_interpreter.GetCommandSPExact (cmd_obj.GetCommandName(), false); 608844d2303SCaroline Tice 609ca90c47eSCaroline Tice if (!m_interpreter.ProcessAliasOptionsArgs (cmd_obj_sp, raw_command_string.c_str(), option_arg_vector_sp)) 610844d2303SCaroline Tice { 611844d2303SCaroline Tice result.AppendError ("Unable to create requested alias.\n"); 612ca90c47eSCaroline Tice result.SetStatus (eReturnStatusFailed); 613844d2303SCaroline Tice return false; 614844d2303SCaroline Tice } 615844d2303SCaroline Tice 616844d2303SCaroline Tice // Create the alias 617844d2303SCaroline Tice if (m_interpreter.AliasExists (alias_command.c_str()) 618844d2303SCaroline Tice || m_interpreter.UserCommandExists (alias_command.c_str())) 619844d2303SCaroline Tice { 620844d2303SCaroline Tice OptionArgVectorSP temp_option_arg_sp (m_interpreter.GetAliasOptions (alias_command.c_str())); 621844d2303SCaroline Tice if (temp_option_arg_sp.get()) 622844d2303SCaroline Tice { 623844d2303SCaroline Tice if (option_arg_vector->size() == 0) 624844d2303SCaroline Tice m_interpreter.RemoveAliasOptions (alias_command.c_str()); 625844d2303SCaroline Tice } 626844d2303SCaroline Tice result.AppendWarningWithFormat ("Overwriting existing definition for '%s'.\n", 627844d2303SCaroline Tice alias_command.c_str()); 628844d2303SCaroline Tice } 629844d2303SCaroline Tice 630472362e6SCaroline Tice if (cmd_obj_sp) 631472362e6SCaroline Tice { 632844d2303SCaroline Tice m_interpreter.AddAlias (alias_command.c_str(), cmd_obj_sp); 633844d2303SCaroline Tice if (option_arg_vector->size() > 0) 634844d2303SCaroline Tice m_interpreter.AddOrReplaceAliasOptions (alias_command.c_str(), option_arg_vector_sp); 635844d2303SCaroline Tice result.SetStatus (eReturnStatusSuccessFinishNoResult); 636844d2303SCaroline Tice } 637472362e6SCaroline Tice else 638472362e6SCaroline Tice { 639472362e6SCaroline Tice result.AppendError ("Unable to create requested alias.\n"); 640472362e6SCaroline Tice result.SetStatus (eReturnStatusFailed); 641472362e6SCaroline Tice } 642844d2303SCaroline Tice return result.Succeeded (); 643844d2303SCaroline Tice } 644ebc09c36SJim Ingham 645ebc09c36SJim Ingham bool 6465a988416SJim Ingham HandleAliasingNormalCommand (Args& args, CommandReturnObject &result) 647ebc09c36SJim Ingham { 648867b185dSCaroline Tice size_t argc = args.GetArgumentCount(); 649ebc09c36SJim Ingham 650ebc09c36SJim Ingham if (argc < 2) 651ebc09c36SJim Ingham { 652ebc09c36SJim Ingham result.AppendError ("'alias' requires at least two arguments"); 653ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 654ebc09c36SJim Ingham return false; 655ebc09c36SJim Ingham } 656ebc09c36SJim Ingham 657ebc09c36SJim Ingham const std::string alias_command = args.GetArgumentAtIndex(0); 658ebc09c36SJim Ingham const std::string actual_command = args.GetArgumentAtIndex(1); 659ebc09c36SJim Ingham 660ebc09c36SJim Ingham args.Shift(); // Shift the alias command word off the argument vector. 661ebc09c36SJim Ingham args.Shift(); // Shift the old command word off the argument vector. 662ebc09c36SJim Ingham 663ebc09c36SJim Ingham // Verify that the command is alias'able, and get the appropriate command object. 664ebc09c36SJim Ingham 665a7015092SGreg Clayton if (m_interpreter.CommandExists (alias_command.c_str())) 666ebc09c36SJim Ingham { 667ebc09c36SJim Ingham result.AppendErrorWithFormat ("'%s' is a permanent debugger command and cannot be redefined.\n", 668ebc09c36SJim Ingham alias_command.c_str()); 669ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 670ebc09c36SJim Ingham } 671ebc09c36SJim Ingham else 672ebc09c36SJim Ingham { 673a7015092SGreg Clayton CommandObjectSP command_obj_sp(m_interpreter.GetCommandSPExact (actual_command.c_str(), true)); 674ebc09c36SJim Ingham CommandObjectSP subcommand_obj_sp; 675ebc09c36SJim Ingham bool use_subcommand = false; 676ebc09c36SJim Ingham if (command_obj_sp.get()) 677ebc09c36SJim Ingham { 678ebc09c36SJim Ingham CommandObject *cmd_obj = command_obj_sp.get(); 679c982c768SGreg Clayton CommandObject *sub_cmd_obj = NULL; 680ebc09c36SJim Ingham OptionArgVectorSP option_arg_vector_sp = OptionArgVectorSP (new OptionArgVector); 681ebc09c36SJim Ingham OptionArgVector *option_arg_vector = option_arg_vector_sp.get(); 682ebc09c36SJim Ingham 683844d2303SCaroline Tice while (cmd_obj->IsMultiwordObject() && args.GetArgumentCount() > 0) 684ebc09c36SJim Ingham { 685ebc09c36SJim Ingham if (argc >= 3) 686ebc09c36SJim Ingham { 687ebc09c36SJim Ingham const std::string sub_command = args.GetArgumentAtIndex(0); 688ebc09c36SJim Ingham assert (sub_command.length() != 0); 689998255bfSGreg Clayton subcommand_obj_sp = cmd_obj->GetSubcommandSP (sub_command.c_str()); 690ebc09c36SJim Ingham if (subcommand_obj_sp.get()) 691ebc09c36SJim Ingham { 692ebc09c36SJim Ingham sub_cmd_obj = subcommand_obj_sp.get(); 693ebc09c36SJim Ingham use_subcommand = true; 694ebc09c36SJim Ingham args.Shift(); // Shift the sub_command word off the argument vector. 695844d2303SCaroline Tice cmd_obj = sub_cmd_obj; 696ebc09c36SJim Ingham } 697ebc09c36SJim Ingham else 698ebc09c36SJim Ingham { 699f415eeb4SCaroline Tice result.AppendErrorWithFormat("'%s' is not a valid sub-command of '%s'. " 700f415eeb4SCaroline Tice "Unable to create alias.\n", 701f415eeb4SCaroline Tice sub_command.c_str(), actual_command.c_str()); 702ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 703ebc09c36SJim Ingham return false; 704ebc09c36SJim Ingham } 705ebc09c36SJim Ingham } 706ebc09c36SJim Ingham } 707ebc09c36SJim Ingham 708ebc09c36SJim Ingham // Verify & handle any options/arguments passed to the alias command 709ebc09c36SJim Ingham 710ebc09c36SJim Ingham if (args.GetArgumentCount () > 0) 711ebc09c36SJim Ingham { 712ca90c47eSCaroline Tice CommandObjectSP tmp_sp = m_interpreter.GetCommandSPExact (cmd_obj->GetCommandName(), false); 713ebc09c36SJim Ingham if (use_subcommand) 714ca90c47eSCaroline Tice tmp_sp = m_interpreter.GetCommandSPExact (sub_cmd_obj->GetCommandName(), false); 715ca90c47eSCaroline Tice 716ca90c47eSCaroline Tice std::string args_string; 717ca90c47eSCaroline Tice args.GetCommandString (args_string); 718ca90c47eSCaroline Tice 719ca90c47eSCaroline Tice if (!m_interpreter.ProcessAliasOptionsArgs (tmp_sp, args_string.c_str(), option_arg_vector_sp)) 720ebc09c36SJim Ingham { 721ca90c47eSCaroline Tice result.AppendError ("Unable to create requested alias.\n"); 722ca90c47eSCaroline Tice result.SetStatus (eReturnStatusFailed); 723e7941795SCaroline Tice return false; 724867b185dSCaroline Tice } 725867b185dSCaroline Tice } 726867b185dSCaroline Tice 727ebc09c36SJim Ingham // Create the alias. 728ebc09c36SJim Ingham 729a7015092SGreg Clayton if (m_interpreter.AliasExists (alias_command.c_str()) 730a7015092SGreg Clayton || m_interpreter.UserCommandExists (alias_command.c_str())) 731ebc09c36SJim Ingham { 732a7015092SGreg Clayton OptionArgVectorSP tmp_option_arg_sp (m_interpreter.GetAliasOptions (alias_command.c_str())); 733ebc09c36SJim Ingham if (tmp_option_arg_sp.get()) 734ebc09c36SJim Ingham { 735ebc09c36SJim Ingham if (option_arg_vector->size() == 0) 736a7015092SGreg Clayton m_interpreter.RemoveAliasOptions (alias_command.c_str()); 737ebc09c36SJim Ingham } 738ebc09c36SJim Ingham result.AppendWarningWithFormat ("Overwriting existing definition for '%s'.\n", 739ebc09c36SJim Ingham alias_command.c_str()); 740ebc09c36SJim Ingham } 741ebc09c36SJim Ingham 742ebc09c36SJim Ingham if (use_subcommand) 743a7015092SGreg Clayton m_interpreter.AddAlias (alias_command.c_str(), subcommand_obj_sp); 744ebc09c36SJim Ingham else 745a7015092SGreg Clayton m_interpreter.AddAlias (alias_command.c_str(), command_obj_sp); 746ebc09c36SJim Ingham if (option_arg_vector->size() > 0) 747a7015092SGreg Clayton m_interpreter.AddOrReplaceAliasOptions (alias_command.c_str(), option_arg_vector_sp); 748ebc09c36SJim Ingham result.SetStatus (eReturnStatusSuccessFinishNoResult); 749ebc09c36SJim Ingham } 750ebc09c36SJim Ingham else 751ebc09c36SJim Ingham { 752ebc09c36SJim Ingham result.AppendErrorWithFormat ("'%s' is not an existing command.\n", actual_command.c_str()); 753ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 754e7941795SCaroline Tice return false; 755ebc09c36SJim Ingham } 756ebc09c36SJim Ingham } 757ebc09c36SJim Ingham 758ebc09c36SJim Ingham return result.Succeeded(); 759ebc09c36SJim Ingham } 7605a988416SJim Ingham 761ebc09c36SJim Ingham }; 762ebc09c36SJim Ingham 763ebc09c36SJim Ingham #pragma mark CommandObjectCommandsUnalias 764ebc09c36SJim Ingham //------------------------------------------------------------------------- 765ebc09c36SJim Ingham // CommandObjectCommandsUnalias 766ebc09c36SJim Ingham //------------------------------------------------------------------------- 767ebc09c36SJim Ingham 7685a988416SJim Ingham class CommandObjectCommandsUnalias : public CommandObjectParsed 769ebc09c36SJim Ingham { 770ebc09c36SJim Ingham public: 771a7015092SGreg Clayton CommandObjectCommandsUnalias (CommandInterpreter &interpreter) : 7725a988416SJim Ingham CommandObjectParsed (interpreter, 7730e5e5a79SGreg Clayton "command unalias", 77486ddae50SCaroline Tice "Allow the user to remove/delete a user-defined command abbreviation.", 775405fe67fSCaroline Tice NULL) 776ebc09c36SJim Ingham { 777405fe67fSCaroline Tice CommandArgumentEntry arg; 778405fe67fSCaroline Tice CommandArgumentData alias_arg; 779405fe67fSCaroline Tice 780405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 781405fe67fSCaroline Tice alias_arg.arg_type = eArgTypeAliasName; 782405fe67fSCaroline Tice alias_arg.arg_repetition = eArgRepeatPlain; 783405fe67fSCaroline Tice 784405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 785405fe67fSCaroline Tice arg.push_back (alias_arg); 786405fe67fSCaroline Tice 787405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 788405fe67fSCaroline Tice m_arguments.push_back (arg); 789ebc09c36SJim Ingham } 790ebc09c36SJim Ingham 791ebc09c36SJim Ingham ~CommandObjectCommandsUnalias() 792ebc09c36SJim Ingham { 793ebc09c36SJim Ingham } 794ebc09c36SJim Ingham 7955a988416SJim Ingham protected: 796ebc09c36SJim Ingham bool 7975a988416SJim Ingham DoExecute (Args& args, CommandReturnObject &result) 798ebc09c36SJim Ingham { 799ebc09c36SJim Ingham CommandObject::CommandMap::iterator pos; 800ebc09c36SJim Ingham CommandObject *cmd_obj; 801ebc09c36SJim Ingham 802ebc09c36SJim Ingham if (args.GetArgumentCount() != 0) 803ebc09c36SJim Ingham { 804ebc09c36SJim Ingham const char *command_name = args.GetArgumentAtIndex(0); 805a7015092SGreg Clayton cmd_obj = m_interpreter.GetCommandObject(command_name); 806ebc09c36SJim Ingham if (cmd_obj) 807ebc09c36SJim Ingham { 808a7015092SGreg Clayton if (m_interpreter.CommandExists (command_name)) 809ebc09c36SJim Ingham { 810ebc09c36SJim Ingham result.AppendErrorWithFormat ("'%s' is a permanent debugger command and cannot be removed.\n", 811ebc09c36SJim Ingham command_name); 812ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 813ebc09c36SJim Ingham } 814ebc09c36SJim Ingham else 815ebc09c36SJim Ingham { 816ebc09c36SJim Ingham 817a7015092SGreg Clayton if (m_interpreter.RemoveAlias (command_name) == false) 818ebc09c36SJim Ingham { 819a7015092SGreg Clayton if (m_interpreter.AliasExists (command_name)) 820ebc09c36SJim Ingham result.AppendErrorWithFormat ("Error occurred while attempting to unalias '%s'.\n", 821ebc09c36SJim Ingham command_name); 822ebc09c36SJim Ingham else 823ebc09c36SJim Ingham result.AppendErrorWithFormat ("'%s' is not an existing alias.\n", command_name); 824ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 825ebc09c36SJim Ingham } 826ebc09c36SJim Ingham else 827ebc09c36SJim Ingham result.SetStatus (eReturnStatusSuccessFinishNoResult); 828ebc09c36SJim Ingham } 829ebc09c36SJim Ingham } 830ebc09c36SJim Ingham else 831ebc09c36SJim Ingham { 832ebc09c36SJim Ingham result.AppendErrorWithFormat ("'%s' is not a known command.\nTry 'help' to see a " 833ebc09c36SJim Ingham "current list of commands.\n", 834ebc09c36SJim Ingham command_name); 835ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 836ebc09c36SJim Ingham } 837ebc09c36SJim Ingham } 838ebc09c36SJim Ingham else 839ebc09c36SJim Ingham { 840ebc09c36SJim Ingham result.AppendError ("must call 'unalias' with a valid alias"); 841ebc09c36SJim Ingham result.SetStatus (eReturnStatusFailed); 842ebc09c36SJim Ingham } 843ebc09c36SJim Ingham 844ebc09c36SJim Ingham return result.Succeeded(); 845ebc09c36SJim Ingham } 846ebc09c36SJim Ingham }; 847ebc09c36SJim Ingham 848de164aaaSGreg Clayton //------------------------------------------------------------------------- 849de164aaaSGreg Clayton // CommandObjectCommandsAddRegex 850de164aaaSGreg Clayton //------------------------------------------------------------------------- 8515a988416SJim Ingham #pragma mark CommandObjectCommandsAddRegex 852de164aaaSGreg Clayton 8535a988416SJim Ingham class CommandObjectCommandsAddRegex : public CommandObjectParsed 854de164aaaSGreg Clayton { 855de164aaaSGreg Clayton public: 856de164aaaSGreg Clayton CommandObjectCommandsAddRegex (CommandInterpreter &interpreter) : 8575a988416SJim Ingham CommandObjectParsed (interpreter, 8580e5e5a79SGreg Clayton "command regex", 859de164aaaSGreg Clayton "Allow the user to create a regular expression command.", 8600e5e5a79SGreg Clayton "command regex <cmd-name> [s/<regex>/<subst>/ ...]"), 861de164aaaSGreg Clayton m_options (interpreter) 862de164aaaSGreg Clayton { 8630e5e5a79SGreg Clayton SetHelpLong( 8640e5e5a79SGreg Clayton "This command allows the user to create powerful regular expression commands\n" 8650e5e5a79SGreg Clayton "with substitutions. The regular expressions and substitutions are specified\n" 8660e5e5a79SGreg Clayton "using the regular exression substitution format of:\n" 8670e5e5a79SGreg Clayton "\n" 8680e5e5a79SGreg Clayton " s/<regex>/<subst>/\n" 8690e5e5a79SGreg Clayton "\n" 8700e5e5a79SGreg Clayton "<regex> is a regular expression that can use parenthesis to capture regular\n" 8710e5e5a79SGreg Clayton "expression input and substitute the captured matches in the output using %1\n" 8720e5e5a79SGreg Clayton "for the first match, %2 for the second, and so on.\n" 8730e5e5a79SGreg Clayton "\n" 8740e5e5a79SGreg Clayton "The regular expressions can all be specified on the command line if more than\n" 8750e5e5a79SGreg Clayton "one argument is provided. If just the command name is provided on the command\n" 8760e5e5a79SGreg Clayton "line, then the regular expressions and substitutions can be entered on separate\n" 8770e5e5a79SGreg Clayton " lines, followed by an empty line to terminate the command definition.\n" 8780e5e5a79SGreg Clayton "\n" 8790e5e5a79SGreg Clayton "EXAMPLES\n" 8800e5e5a79SGreg Clayton "\n" 881adc43c99SSean Callanan "The following example will define a regular expression command named 'f' that\n" 8820e5e5a79SGreg Clayton "will call 'finish' if there are no arguments, or 'frame select <frame-idx>' if\n" 8830e5e5a79SGreg Clayton "a number follows 'f':\n" 884adc43c99SSean Callanan "\n" 8850e5e5a79SGreg Clayton " (lldb) command regex f s/^$/finish/ 's/([0-9]+)/frame select %1/'\n" 886adc43c99SSean Callanan "\n" 8870e5e5a79SGreg Clayton ); 888de164aaaSGreg Clayton } 889de164aaaSGreg Clayton 890de164aaaSGreg Clayton ~CommandObjectCommandsAddRegex() 891de164aaaSGreg Clayton { 892de164aaaSGreg Clayton } 893de164aaaSGreg Clayton 894de164aaaSGreg Clayton 8955a988416SJim Ingham protected: 896de164aaaSGreg Clayton bool 8975a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 898de164aaaSGreg Clayton { 8995a988416SJim Ingham const size_t argc = command.GetArgumentCount(); 9000e5e5a79SGreg Clayton if (argc == 0) 901de164aaaSGreg Clayton { 90269c12ccbSJason Molenda result.AppendError ("usage: 'command regex <command-name> [s/<regex1>/<subst1>/ s/<regex2>/<subst2>/ ...]'\n"); 9030e5e5a79SGreg Clayton result.SetStatus (eReturnStatusFailed); 9040e5e5a79SGreg Clayton } 9050e5e5a79SGreg Clayton else 9060e5e5a79SGreg Clayton { 9070e5e5a79SGreg Clayton Error error; 9085a988416SJim Ingham const char *name = command.GetArgumentAtIndex(0); 909de164aaaSGreg Clayton m_regex_cmd_ap.reset (new CommandObjectRegexCommand (m_interpreter, 910de164aaaSGreg Clayton name, 911de164aaaSGreg Clayton m_options.GetHelp (), 912de164aaaSGreg Clayton m_options.GetSyntax (), 913de164aaaSGreg Clayton 10)); 9140e5e5a79SGreg Clayton 9150e5e5a79SGreg Clayton if (argc == 1) 9160e5e5a79SGreg Clayton { 9170e5e5a79SGreg Clayton InputReaderSP reader_sp (new InputReader(m_interpreter.GetDebugger())); 918de164aaaSGreg Clayton if (reader_sp) 919de164aaaSGreg Clayton { 9200e5e5a79SGreg Clayton error =reader_sp->Initialize (CommandObjectCommandsAddRegex::InputReaderCallback, 921de164aaaSGreg Clayton this, // baton 922de164aaaSGreg Clayton eInputReaderGranularityLine, // token size, to pass to callback function 9230e5e5a79SGreg Clayton NULL, // end token 924de164aaaSGreg Clayton "> ", // prompt 9250e5e5a79SGreg Clayton true); // echo input 9260e5e5a79SGreg Clayton if (error.Success()) 927de164aaaSGreg Clayton { 928de164aaaSGreg Clayton m_interpreter.GetDebugger().PushInputReader (reader_sp); 929de164aaaSGreg Clayton result.SetStatus (eReturnStatusSuccessFinishNoResult); 9300e5e5a79SGreg Clayton return true; 931de164aaaSGreg Clayton } 932de164aaaSGreg Clayton } 933de164aaaSGreg Clayton } 934de164aaaSGreg Clayton else 935de164aaaSGreg Clayton { 9360e5e5a79SGreg Clayton for (size_t arg_idx = 1; arg_idx < argc; ++arg_idx) 9370e5e5a79SGreg Clayton { 9385a988416SJim Ingham llvm::StringRef arg_strref (command.GetArgumentAtIndex(arg_idx)); 9390e5e5a79SGreg Clayton error = AppendRegexSubstitution (arg_strref); 9400e5e5a79SGreg Clayton if (error.Fail()) 9410e5e5a79SGreg Clayton break; 9420e5e5a79SGreg Clayton } 9430e5e5a79SGreg Clayton 9440e5e5a79SGreg Clayton if (error.Success()) 9450e5e5a79SGreg Clayton { 9460e5e5a79SGreg Clayton AddRegexCommandToInterpreter(); 9470e5e5a79SGreg Clayton } 9480e5e5a79SGreg Clayton } 9490e5e5a79SGreg Clayton if (error.Fail()) 9500e5e5a79SGreg Clayton { 9510e5e5a79SGreg Clayton result.AppendError (error.AsCString()); 952de164aaaSGreg Clayton result.SetStatus (eReturnStatusFailed); 953de164aaaSGreg Clayton } 9540e5e5a79SGreg Clayton } 9550e5e5a79SGreg Clayton 956de164aaaSGreg Clayton return result.Succeeded(); 957de164aaaSGreg Clayton } 958de164aaaSGreg Clayton 9590e5e5a79SGreg Clayton Error 9600e5e5a79SGreg Clayton AppendRegexSubstitution (const llvm::StringRef ®ex_sed) 961de164aaaSGreg Clayton { 9620e5e5a79SGreg Clayton Error error; 9630e5e5a79SGreg Clayton 9640e5e5a79SGreg Clayton if (m_regex_cmd_ap.get() == NULL) 965de164aaaSGreg Clayton { 9660e5e5a79SGreg Clayton error.SetErrorStringWithFormat("invalid regular expression command object for: '%.*s'", 9670e5e5a79SGreg Clayton (int)regex_sed.size(), 9680e5e5a79SGreg Clayton regex_sed.data()); 9690e5e5a79SGreg Clayton return error; 970de164aaaSGreg Clayton } 9710e5e5a79SGreg Clayton 9720e5e5a79SGreg Clayton size_t regex_sed_size = regex_sed.size(); 9730e5e5a79SGreg Clayton 9740e5e5a79SGreg Clayton if (regex_sed_size <= 1) 9750e5e5a79SGreg Clayton { 9760e5e5a79SGreg Clayton error.SetErrorStringWithFormat("regular expression substitution string is too short: '%.*s'", 9770e5e5a79SGreg Clayton (int)regex_sed.size(), 9780e5e5a79SGreg Clayton regex_sed.data()); 9790e5e5a79SGreg Clayton return error; 9800e5e5a79SGreg Clayton } 9810e5e5a79SGreg Clayton 9820e5e5a79SGreg Clayton if (regex_sed[0] != 's') 9830e5e5a79SGreg Clayton { 9840e5e5a79SGreg Clayton error.SetErrorStringWithFormat("regular expression substitution string doesn't start with 's': '%.*s'", 9850e5e5a79SGreg Clayton (int)regex_sed.size(), 9860e5e5a79SGreg Clayton regex_sed.data()); 9870e5e5a79SGreg Clayton return error; 9880e5e5a79SGreg Clayton } 9890e5e5a79SGreg Clayton const size_t first_separator_char_pos = 1; 9900e5e5a79SGreg Clayton // use the char that follows 's' as the regex separator character 9910e5e5a79SGreg Clayton // so we can have "s/<regex>/<subst>/" or "s|<regex>|<subst>|" 9920e5e5a79SGreg Clayton const char separator_char = regex_sed[first_separator_char_pos]; 9930e5e5a79SGreg Clayton const size_t second_separator_char_pos = regex_sed.find (separator_char, first_separator_char_pos + 1); 9940e5e5a79SGreg Clayton 9950e5e5a79SGreg Clayton if (second_separator_char_pos == std::string::npos) 9960e5e5a79SGreg Clayton { 9970e5e5a79SGreg Clayton error.SetErrorStringWithFormat("missing second '%c' separator char after '%.*s'", 9980e5e5a79SGreg Clayton separator_char, 9990e5e5a79SGreg Clayton (int)(regex_sed.size() - first_separator_char_pos - 1), 10000e5e5a79SGreg Clayton regex_sed.data() + (first_separator_char_pos + 1)); 10010e5e5a79SGreg Clayton return error; 10020e5e5a79SGreg Clayton } 10030e5e5a79SGreg Clayton 10040e5e5a79SGreg Clayton const size_t third_separator_char_pos = regex_sed.find (separator_char, second_separator_char_pos + 1); 10050e5e5a79SGreg Clayton 10060e5e5a79SGreg Clayton if (third_separator_char_pos == std::string::npos) 10070e5e5a79SGreg Clayton { 10080e5e5a79SGreg Clayton error.SetErrorStringWithFormat("missing third '%c' separator char after '%.*s'", 10090e5e5a79SGreg Clayton separator_char, 10100e5e5a79SGreg Clayton (int)(regex_sed.size() - second_separator_char_pos - 1), 10110e5e5a79SGreg Clayton regex_sed.data() + (second_separator_char_pos + 1)); 10120e5e5a79SGreg Clayton return error; 10130e5e5a79SGreg Clayton } 10140e5e5a79SGreg Clayton 10150e5e5a79SGreg Clayton if (third_separator_char_pos != regex_sed_size - 1) 10160e5e5a79SGreg Clayton { 10170e5e5a79SGreg Clayton // Make sure that everything that follows the last regex 10180e5e5a79SGreg Clayton // separator char 10190e5e5a79SGreg Clayton if (regex_sed.find_first_not_of("\t\n\v\f\r ", third_separator_char_pos + 1) != std::string::npos) 10200e5e5a79SGreg Clayton { 10210e5e5a79SGreg Clayton error.SetErrorStringWithFormat("extra data found after the '%.*s' regular expression substitution string: '%.*s'", 10220e5e5a79SGreg Clayton (int)third_separator_char_pos + 1, 10230e5e5a79SGreg Clayton regex_sed.data(), 10240e5e5a79SGreg Clayton (int)(regex_sed.size() - third_separator_char_pos - 1), 10250e5e5a79SGreg Clayton regex_sed.data() + (third_separator_char_pos + 1)); 10260e5e5a79SGreg Clayton return error; 10270e5e5a79SGreg Clayton } 10280e5e5a79SGreg Clayton 10290e5e5a79SGreg Clayton } 10300e5e5a79SGreg Clayton else if (first_separator_char_pos + 1 == second_separator_char_pos) 10310e5e5a79SGreg Clayton { 10320e5e5a79SGreg Clayton error.SetErrorStringWithFormat("<regex> can't be empty in 's%c<regex>%c<subst>%c' string: '%.*s'", 10330e5e5a79SGreg Clayton separator_char, 10340e5e5a79SGreg Clayton separator_char, 10350e5e5a79SGreg Clayton separator_char, 10360e5e5a79SGreg Clayton (int)regex_sed.size(), 10370e5e5a79SGreg Clayton regex_sed.data()); 10380e5e5a79SGreg Clayton return error; 10390e5e5a79SGreg Clayton } 10400e5e5a79SGreg Clayton else if (second_separator_char_pos + 1 == third_separator_char_pos) 10410e5e5a79SGreg Clayton { 10420e5e5a79SGreg Clayton error.SetErrorStringWithFormat("<subst> can't be empty in 's%c<regex>%c<subst>%c' string: '%.*s'", 10430e5e5a79SGreg Clayton separator_char, 10440e5e5a79SGreg Clayton separator_char, 10450e5e5a79SGreg Clayton separator_char, 10460e5e5a79SGreg Clayton (int)regex_sed.size(), 10470e5e5a79SGreg Clayton regex_sed.data()); 10480e5e5a79SGreg Clayton return error; 10490e5e5a79SGreg Clayton } 10500e5e5a79SGreg Clayton std::string regex(regex_sed.substr(first_separator_char_pos + 1, second_separator_char_pos - first_separator_char_pos - 1)); 10510e5e5a79SGreg Clayton std::string subst(regex_sed.substr(second_separator_char_pos + 1, third_separator_char_pos - second_separator_char_pos - 1)); 10520e5e5a79SGreg Clayton m_regex_cmd_ap->AddRegexCommand (regex.c_str(), 10530e5e5a79SGreg Clayton subst.c_str()); 10540e5e5a79SGreg Clayton return error; 1055de164aaaSGreg Clayton } 1056de164aaaSGreg Clayton 1057de164aaaSGreg Clayton void 10580e5e5a79SGreg Clayton AddRegexCommandToInterpreter() 1059de164aaaSGreg Clayton { 1060de164aaaSGreg Clayton if (m_regex_cmd_ap.get()) 1061de164aaaSGreg Clayton { 1062de164aaaSGreg Clayton if (m_regex_cmd_ap->HasRegexEntries()) 1063de164aaaSGreg Clayton { 1064de164aaaSGreg Clayton CommandObjectSP cmd_sp (m_regex_cmd_ap.release()); 1065de164aaaSGreg Clayton m_interpreter.AddCommand(cmd_sp->GetCommandName(), cmd_sp, true); 1066de164aaaSGreg Clayton } 1067de164aaaSGreg Clayton } 1068de164aaaSGreg Clayton } 1069de164aaaSGreg Clayton 10700e5e5a79SGreg Clayton void 10710e5e5a79SGreg Clayton InputReaderDidCancel() 10720e5e5a79SGreg Clayton { 10730e5e5a79SGreg Clayton m_regex_cmd_ap.reset(); 10740e5e5a79SGreg Clayton } 10750e5e5a79SGreg Clayton 1076de164aaaSGreg Clayton static size_t 1077de164aaaSGreg Clayton InputReaderCallback (void *baton, 1078de164aaaSGreg Clayton InputReader &reader, 1079de164aaaSGreg Clayton lldb::InputReaderAction notification, 1080de164aaaSGreg Clayton const char *bytes, 10815a988416SJim Ingham size_t bytes_len) 10825a988416SJim Ingham { 10835a988416SJim Ingham CommandObjectCommandsAddRegex *add_regex_cmd = (CommandObjectCommandsAddRegex *) baton; 10845a988416SJim Ingham bool batch_mode = reader.GetDebugger().GetCommandInterpreter().GetBatchCommandMode(); 10855a988416SJim Ingham 10865a988416SJim Ingham switch (notification) 10875a988416SJim Ingham { 10885a988416SJim Ingham case eInputReaderActivate: 10895a988416SJim Ingham if (!batch_mode) 10905a988416SJim Ingham { 10915a988416SJim Ingham StreamSP out_stream = reader.GetDebugger().GetAsyncOutputStream (); 10925a988416SJim Ingham out_stream->Printf("%s\n", "Enter regular expressions in the form 's/<regex>/<subst>/' and terminate with an empty line:"); 10935a988416SJim Ingham out_stream->Flush(); 10945a988416SJim Ingham } 10955a988416SJim Ingham break; 10965a988416SJim Ingham case eInputReaderReactivate: 10975a988416SJim Ingham break; 10985a988416SJim Ingham 10995a988416SJim Ingham case eInputReaderDeactivate: 11005a988416SJim Ingham break; 11015a988416SJim Ingham 11025a988416SJim Ingham case eInputReaderAsynchronousOutputWritten: 11035a988416SJim Ingham break; 11045a988416SJim Ingham 11055a988416SJim Ingham case eInputReaderGotToken: 11065a988416SJim Ingham while (bytes_len > 0 && (bytes[bytes_len-1] == '\r' || bytes[bytes_len-1] == '\n')) 11075a988416SJim Ingham --bytes_len; 11085a988416SJim Ingham if (bytes_len == 0) 11095a988416SJim Ingham reader.SetIsDone(true); 11105a988416SJim Ingham else if (bytes) 11115a988416SJim Ingham { 11125a988416SJim Ingham llvm::StringRef bytes_strref (bytes, bytes_len); 11135a988416SJim Ingham Error error (add_regex_cmd->AppendRegexSubstitution (bytes_strref)); 11145a988416SJim Ingham if (error.Fail()) 11155a988416SJim Ingham { 11165a988416SJim Ingham if (!batch_mode) 11175a988416SJim Ingham { 11185a988416SJim Ingham StreamSP out_stream = reader.GetDebugger().GetAsyncOutputStream(); 11195a988416SJim Ingham out_stream->Printf("error: %s\n", error.AsCString()); 11205a988416SJim Ingham out_stream->Flush(); 11215a988416SJim Ingham } 11225a988416SJim Ingham add_regex_cmd->InputReaderDidCancel (); 11235a988416SJim Ingham reader.SetIsDone (true); 11245a988416SJim Ingham } 11255a988416SJim Ingham } 11265a988416SJim Ingham break; 11275a988416SJim Ingham 11285a988416SJim Ingham case eInputReaderInterrupt: 11295a988416SJim Ingham { 11305a988416SJim Ingham reader.SetIsDone (true); 11315a988416SJim Ingham if (!batch_mode) 11325a988416SJim Ingham { 11335a988416SJim Ingham StreamSP out_stream = reader.GetDebugger().GetAsyncOutputStream(); 11345a988416SJim Ingham out_stream->PutCString("Regular expression command creations was cancelled.\n"); 11355a988416SJim Ingham out_stream->Flush(); 11365a988416SJim Ingham } 11375a988416SJim Ingham add_regex_cmd->InputReaderDidCancel (); 11385a988416SJim Ingham } 11395a988416SJim Ingham break; 11405a988416SJim Ingham 11415a988416SJim Ingham case eInputReaderEndOfFile: 11425a988416SJim Ingham reader.SetIsDone (true); 11435a988416SJim Ingham break; 11445a988416SJim Ingham 11455a988416SJim Ingham case eInputReaderDone: 11465a988416SJim Ingham add_regex_cmd->AddRegexCommandToInterpreter(); 11475a988416SJim Ingham break; 11485a988416SJim Ingham } 11495a988416SJim Ingham 11505a988416SJim Ingham return bytes_len; 11515a988416SJim Ingham } 11525a988416SJim Ingham 1153de164aaaSGreg Clayton private: 11547b0992d9SGreg Clayton std::unique_ptr<CommandObjectRegexCommand> m_regex_cmd_ap; 1155de164aaaSGreg Clayton 1156de164aaaSGreg Clayton class CommandOptions : public Options 1157de164aaaSGreg Clayton { 1158de164aaaSGreg Clayton public: 1159de164aaaSGreg Clayton 1160de164aaaSGreg Clayton CommandOptions (CommandInterpreter &interpreter) : 1161de164aaaSGreg Clayton Options (interpreter) 1162de164aaaSGreg Clayton { 1163de164aaaSGreg Clayton } 1164de164aaaSGreg Clayton 1165de164aaaSGreg Clayton virtual 1166de164aaaSGreg Clayton ~CommandOptions (){} 1167de164aaaSGreg Clayton 1168de164aaaSGreg Clayton virtual Error 1169de164aaaSGreg Clayton SetOptionValue (uint32_t option_idx, const char *option_arg) 1170de164aaaSGreg Clayton { 1171de164aaaSGreg Clayton Error error; 11723bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 1173de164aaaSGreg Clayton 1174de164aaaSGreg Clayton switch (short_option) 1175de164aaaSGreg Clayton { 1176de164aaaSGreg Clayton case 'h': 1177de164aaaSGreg Clayton m_help.assign (option_arg); 1178de164aaaSGreg Clayton break; 1179de164aaaSGreg Clayton case 's': 1180de164aaaSGreg Clayton m_syntax.assign (option_arg); 1181de164aaaSGreg Clayton break; 1182de164aaaSGreg Clayton 1183de164aaaSGreg Clayton default: 118486edbf41SGreg Clayton error.SetErrorStringWithFormat ("unrecognized option '%c'", short_option); 1185de164aaaSGreg Clayton break; 1186de164aaaSGreg Clayton } 1187de164aaaSGreg Clayton 1188de164aaaSGreg Clayton return error; 1189de164aaaSGreg Clayton } 1190de164aaaSGreg Clayton 1191de164aaaSGreg Clayton void 1192de164aaaSGreg Clayton OptionParsingStarting () 1193de164aaaSGreg Clayton { 1194de164aaaSGreg Clayton m_help.clear(); 1195de164aaaSGreg Clayton m_syntax.clear(); 1196de164aaaSGreg Clayton } 1197de164aaaSGreg Clayton 1198de164aaaSGreg Clayton const OptionDefinition* 1199de164aaaSGreg Clayton GetDefinitions () 1200de164aaaSGreg Clayton { 1201de164aaaSGreg Clayton return g_option_table; 1202de164aaaSGreg Clayton } 1203de164aaaSGreg Clayton 1204de164aaaSGreg Clayton // Options table: Required for subclasses of Options. 1205de164aaaSGreg Clayton 1206de164aaaSGreg Clayton static OptionDefinition g_option_table[]; 1207de164aaaSGreg Clayton 1208de164aaaSGreg Clayton const char * 1209de164aaaSGreg Clayton GetHelp () 1210de164aaaSGreg Clayton { 1211de164aaaSGreg Clayton if (m_help.empty()) 1212de164aaaSGreg Clayton return NULL; 1213de164aaaSGreg Clayton return m_help.c_str(); 1214de164aaaSGreg Clayton } 1215de164aaaSGreg Clayton const char * 1216de164aaaSGreg Clayton GetSyntax () 1217de164aaaSGreg Clayton { 1218de164aaaSGreg Clayton if (m_syntax.empty()) 1219de164aaaSGreg Clayton return NULL; 1220de164aaaSGreg Clayton return m_syntax.c_str(); 1221de164aaaSGreg Clayton } 1222de164aaaSGreg Clayton // Instance variables to hold the values for command options. 1223de164aaaSGreg Clayton protected: 1224de164aaaSGreg Clayton std::string m_help; 1225de164aaaSGreg Clayton std::string m_syntax; 1226de164aaaSGreg Clayton }; 1227de164aaaSGreg Clayton 1228de164aaaSGreg Clayton virtual Options * 1229de164aaaSGreg Clayton GetOptions () 1230de164aaaSGreg Clayton { 1231de164aaaSGreg Clayton return &m_options; 1232de164aaaSGreg Clayton } 1233de164aaaSGreg Clayton 12345a988416SJim Ingham CommandOptions m_options; 1235de164aaaSGreg Clayton }; 1236de164aaaSGreg Clayton 1237de164aaaSGreg Clayton OptionDefinition 1238de164aaaSGreg Clayton CommandObjectCommandsAddRegex::CommandOptions::g_option_table[] = 1239de164aaaSGreg Clayton { 1240de164aaaSGreg Clayton { LLDB_OPT_SET_1, false, "help" , 'h', required_argument, NULL, 0, eArgTypeNone, "The help text to display for this command."}, 1241de164aaaSGreg Clayton { LLDB_OPT_SET_1, false, "syntax", 's', required_argument, NULL, 0, eArgTypeNone, "A syntax string showing the typical usage syntax."}, 1242de164aaaSGreg Clayton { 0 , false, NULL , 0 , 0 , NULL, 0, eArgTypeNone, NULL } 1243de164aaaSGreg Clayton }; 1244de164aaaSGreg Clayton 1245de164aaaSGreg Clayton 12465a988416SJim Ingham class CommandObjectPythonFunction : public CommandObjectRaw 1247223383edSEnrico Granata { 1248223383edSEnrico Granata private: 1249223383edSEnrico Granata std::string m_function_name; 12500a305db7SEnrico Granata ScriptedCommandSynchronicity m_synchro; 1251fac939e9SEnrico Granata bool m_fetched_help_long; 1252223383edSEnrico Granata 1253223383edSEnrico Granata public: 1254223383edSEnrico Granata 1255223383edSEnrico Granata CommandObjectPythonFunction (CommandInterpreter &interpreter, 1256223383edSEnrico Granata std::string name, 12570a305db7SEnrico Granata std::string funct, 12580a305db7SEnrico Granata ScriptedCommandSynchronicity synch) : 12595a988416SJim Ingham CommandObjectRaw (interpreter, 1260223383edSEnrico Granata name.c_str(), 1261223383edSEnrico Granata (std::string("Run Python function ") + funct).c_str(), 1262223383edSEnrico Granata NULL), 12630a305db7SEnrico Granata m_function_name(funct), 1264fac939e9SEnrico Granata m_synchro(synch), 1265fac939e9SEnrico Granata m_fetched_help_long(false) 1266223383edSEnrico Granata { 1267223383edSEnrico Granata } 1268223383edSEnrico Granata 1269223383edSEnrico Granata virtual 1270223383edSEnrico Granata ~CommandObjectPythonFunction () 1271223383edSEnrico Granata { 1272223383edSEnrico Granata } 1273223383edSEnrico Granata 1274223383edSEnrico Granata virtual bool 12753a18e319SGreg Clayton IsRemovable () const 12765a988416SJim Ingham { 12775a988416SJim Ingham return true; 12785a988416SJim Ingham } 12795a988416SJim Ingham 12805a988416SJim Ingham const std::string& 12815a988416SJim Ingham GetFunctionName () 12825a988416SJim Ingham { 12835a988416SJim Ingham return m_function_name; 12845a988416SJim Ingham } 12855a988416SJim Ingham 12865a988416SJim Ingham ScriptedCommandSynchronicity 12875a988416SJim Ingham GetSynchronicity () 12885a988416SJim Ingham { 12895a988416SJim Ingham return m_synchro; 12905a988416SJim Ingham } 12915a988416SJim Ingham 1292fac939e9SEnrico Granata virtual const char * 1293fac939e9SEnrico Granata GetHelpLong () 1294fac939e9SEnrico Granata { 1295fac939e9SEnrico Granata if (!m_fetched_help_long) 1296fac939e9SEnrico Granata { 1297fac939e9SEnrico Granata ScriptInterpreter* scripter = m_interpreter.GetScriptInterpreter(); 1298fac939e9SEnrico Granata if (scripter) 1299fac939e9SEnrico Granata { 1300fac939e9SEnrico Granata std::string docstring; 1301fac939e9SEnrico Granata m_fetched_help_long = scripter->GetDocumentationForItem(m_function_name.c_str(),docstring); 1302fac939e9SEnrico Granata if (!docstring.empty()) 1303fac939e9SEnrico Granata SetHelpLong(docstring); 1304fac939e9SEnrico Granata } 1305fac939e9SEnrico Granata } 1306fac939e9SEnrico Granata return CommandObjectRaw::GetHelpLong(); 1307fac939e9SEnrico Granata } 1308fac939e9SEnrico Granata 13095a988416SJim Ingham protected: 13105a988416SJim Ingham virtual bool 13115a988416SJim Ingham DoExecute (const char *raw_command_line, CommandReturnObject &result) 1312223383edSEnrico Granata { 1313223383edSEnrico Granata ScriptInterpreter* scripter = m_interpreter.GetScriptInterpreter(); 1314223383edSEnrico Granata 1315223383edSEnrico Granata Error error; 1316223383edSEnrico Granata 131770f11f88SJim Ingham result.SetStatus(eReturnStatusInvalid); 131870f11f88SJim Ingham 1319223383edSEnrico Granata if (!scripter || scripter->RunScriptBasedCommand(m_function_name.c_str(), 1320223383edSEnrico Granata raw_command_line, 13210a305db7SEnrico Granata m_synchro, 1322223383edSEnrico Granata result, 1323223383edSEnrico Granata error) == false) 1324223383edSEnrico Granata { 1325223383edSEnrico Granata result.AppendError(error.AsCString()); 1326223383edSEnrico Granata result.SetStatus(eReturnStatusFailed); 1327223383edSEnrico Granata } 1328223383edSEnrico Granata else 132970f11f88SJim Ingham { 133070f11f88SJim Ingham // Don't change the status if the command already set it... 133170f11f88SJim Ingham if (result.GetStatus() == eReturnStatusInvalid) 133270f11f88SJim Ingham { 1333*b4675a4eSEnrico Granata if (result.GetErrorData() && result.GetErrorData()[0]) 1334*b4675a4eSEnrico Granata result.SetStatus(eReturnStatusFailed); 1335*b4675a4eSEnrico Granata else if (result.GetOutputData() == NULL || result.GetOutputData()[0] == '\0') 1336223383edSEnrico Granata result.SetStatus(eReturnStatusSuccessFinishNoResult); 133770f11f88SJim Ingham else 133870f11f88SJim Ingham result.SetStatus(eReturnStatusSuccessFinishResult); 133970f11f88SJim Ingham } 134070f11f88SJim Ingham } 1341223383edSEnrico Granata 1342223383edSEnrico Granata return result.Succeeded(); 1343223383edSEnrico Granata } 1344223383edSEnrico Granata 1345223383edSEnrico Granata }; 1346223383edSEnrico Granata 1347a9dbf432SEnrico Granata //------------------------------------------------------------------------- 1348a9dbf432SEnrico Granata // CommandObjectCommandsScriptImport 1349a9dbf432SEnrico Granata //------------------------------------------------------------------------- 1350a9dbf432SEnrico Granata 13515a988416SJim Ingham class CommandObjectCommandsScriptImport : public CommandObjectParsed 1352a9dbf432SEnrico Granata { 13535a988416SJim Ingham public: 13545a988416SJim Ingham CommandObjectCommandsScriptImport (CommandInterpreter &interpreter) : 13555a988416SJim Ingham CommandObjectParsed (interpreter, 13565a988416SJim Ingham "command script import", 13575a988416SJim Ingham "Import a scripting module in LLDB.", 13585a988416SJim Ingham NULL), 13595a988416SJim Ingham m_options(interpreter) 13605a988416SJim Ingham { 13615a988416SJim Ingham CommandArgumentEntry arg1; 13625a988416SJim Ingham CommandArgumentData cmd_arg; 13635a988416SJim Ingham 13645a988416SJim Ingham // Define the first (and only) variant of this arg. 13655a988416SJim Ingham cmd_arg.arg_type = eArgTypeFilename; 13665a988416SJim Ingham cmd_arg.arg_repetition = eArgRepeatPlain; 13675a988416SJim Ingham 13685a988416SJim Ingham // There is only one variant this argument could be; put it into the argument entry. 13695a988416SJim Ingham arg1.push_back (cmd_arg); 13705a988416SJim Ingham 13715a988416SJim Ingham // Push the data for the first argument into the m_arguments vector. 13725a988416SJim Ingham m_arguments.push_back (arg1); 13735a988416SJim Ingham } 13745a988416SJim Ingham 13755a988416SJim Ingham ~CommandObjectCommandsScriptImport () 13765a988416SJim Ingham { 13775a988416SJim Ingham } 13785a988416SJim Ingham 1379c7bece56SGreg Clayton virtual int 13805a988416SJim Ingham HandleArgumentCompletion (Args &input, 13815a988416SJim Ingham int &cursor_index, 13825a988416SJim Ingham int &cursor_char_position, 13835a988416SJim Ingham OptionElementVector &opt_element_vector, 13845a988416SJim Ingham int match_start_point, 13855a988416SJim Ingham int max_return_elements, 13865a988416SJim Ingham bool &word_complete, 13875a988416SJim Ingham StringList &matches) 13885a988416SJim Ingham { 13895a988416SJim Ingham std::string completion_str (input.GetArgumentAtIndex(cursor_index)); 13905a988416SJim Ingham completion_str.erase (cursor_char_position); 13915a988416SJim Ingham 13925a988416SJim Ingham CommandCompletions::InvokeCommonCompletionCallbacks (m_interpreter, 13935a988416SJim Ingham CommandCompletions::eDiskFileCompletion, 13945a988416SJim Ingham completion_str.c_str(), 13955a988416SJim Ingham match_start_point, 13965a988416SJim Ingham max_return_elements, 13975a988416SJim Ingham NULL, 13985a988416SJim Ingham word_complete, 13995a988416SJim Ingham matches); 14005a988416SJim Ingham return matches.GetSize(); 14015a988416SJim Ingham } 14025a988416SJim Ingham 14035a988416SJim Ingham virtual Options * 14045a988416SJim Ingham GetOptions () 14055a988416SJim Ingham { 14065a988416SJim Ingham return &m_options; 14075a988416SJim Ingham } 14085a988416SJim Ingham 14095a988416SJim Ingham protected: 14100a305db7SEnrico Granata 14110a305db7SEnrico Granata class CommandOptions : public Options 14120a305db7SEnrico Granata { 14130a305db7SEnrico Granata public: 14140a305db7SEnrico Granata 14150a305db7SEnrico Granata CommandOptions (CommandInterpreter &interpreter) : 14160a305db7SEnrico Granata Options (interpreter) 14170a305db7SEnrico Granata { 14180a305db7SEnrico Granata } 14190a305db7SEnrico Granata 14200a305db7SEnrico Granata virtual 14210a305db7SEnrico Granata ~CommandOptions (){} 14220a305db7SEnrico Granata 14230a305db7SEnrico Granata virtual Error 14240a305db7SEnrico Granata SetOptionValue (uint32_t option_idx, const char *option_arg) 14250a305db7SEnrico Granata { 14260a305db7SEnrico Granata Error error; 14273bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 14280a305db7SEnrico Granata 14290a305db7SEnrico Granata switch (short_option) 14300a305db7SEnrico Granata { 14310a305db7SEnrico Granata case 'r': 14320a305db7SEnrico Granata m_allow_reload = true; 14330a305db7SEnrico Granata break; 14340a305db7SEnrico Granata default: 14350a305db7SEnrico Granata error.SetErrorStringWithFormat ("unrecognized option '%c'", short_option); 14360a305db7SEnrico Granata break; 14370a305db7SEnrico Granata } 14380a305db7SEnrico Granata 14390a305db7SEnrico Granata return error; 14400a305db7SEnrico Granata } 14410a305db7SEnrico Granata 14420a305db7SEnrico Granata void 14430a305db7SEnrico Granata OptionParsingStarting () 14440a305db7SEnrico Granata { 1445e0c70f1bSEnrico Granata m_allow_reload = true; 14460a305db7SEnrico Granata } 14470a305db7SEnrico Granata 14480a305db7SEnrico Granata const OptionDefinition* 14490a305db7SEnrico Granata GetDefinitions () 14500a305db7SEnrico Granata { 14510a305db7SEnrico Granata return g_option_table; 14520a305db7SEnrico Granata } 14530a305db7SEnrico Granata 14540a305db7SEnrico Granata // Options table: Required for subclasses of Options. 14550a305db7SEnrico Granata 14560a305db7SEnrico Granata static OptionDefinition g_option_table[]; 14570a305db7SEnrico Granata 14580a305db7SEnrico Granata // Instance variables to hold the values for command options. 14590a305db7SEnrico Granata 14600a305db7SEnrico Granata bool m_allow_reload; 14610a305db7SEnrico Granata }; 14620a305db7SEnrico Granata 1463a9dbf432SEnrico Granata bool 14645a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 1465a9dbf432SEnrico Granata { 1466a9dbf432SEnrico Granata 1467a9dbf432SEnrico Granata if (m_interpreter.GetDebugger().GetScriptLanguage() != lldb::eScriptLanguagePython) 1468a9dbf432SEnrico Granata { 1469a9dbf432SEnrico Granata result.AppendError ("only scripting language supported for module importing is currently Python"); 1470a9dbf432SEnrico Granata result.SetStatus (eReturnStatusFailed); 1471a9dbf432SEnrico Granata return false; 1472a9dbf432SEnrico Granata } 1473a9dbf432SEnrico Granata 14745a988416SJim Ingham size_t argc = command.GetArgumentCount(); 1475a9dbf432SEnrico Granata 1476a9dbf432SEnrico Granata if (argc != 1) 1477a9dbf432SEnrico Granata { 1478a9dbf432SEnrico Granata result.AppendError ("'command script import' requires one argument"); 1479a9dbf432SEnrico Granata result.SetStatus (eReturnStatusFailed); 1480a9dbf432SEnrico Granata return false; 1481a9dbf432SEnrico Granata } 1482a9dbf432SEnrico Granata 14835a988416SJim Ingham std::string path = command.GetArgumentAtIndex(0); 1484a9dbf432SEnrico Granata Error error; 1485a9dbf432SEnrico Granata 1486c9d645d3SGreg Clayton const bool init_session = true; 1487078551c7SEnrico Granata // FIXME: this is necessary because CommandObject::CheckRequirements() assumes that 1488078551c7SEnrico Granata // commands won't ever be recursively invoked, but it's actually possible to craft 1489078551c7SEnrico Granata // a Python script that does other "command script imports" in __lldb_init_module 1490078551c7SEnrico Granata // the real fix is to have recursive commands possible with a CommandInvocation object 1491078551c7SEnrico Granata // separate from the CommandObject itself, so that recursive command invocations 1492078551c7SEnrico Granata // won't stomp on each other (wrt to execution contents, options, and more) 1493078551c7SEnrico Granata m_exe_ctx.Clear(); 1494a9dbf432SEnrico Granata if (m_interpreter.GetScriptInterpreter()->LoadScriptingModule(path.c_str(), 14950a305db7SEnrico Granata m_options.m_allow_reload, 1496c9d645d3SGreg Clayton init_session, 1497a9dbf432SEnrico Granata error)) 1498a9dbf432SEnrico Granata { 1499a9dbf432SEnrico Granata result.SetStatus (eReturnStatusSuccessFinishNoResult); 1500a9dbf432SEnrico Granata } 1501a9dbf432SEnrico Granata else 1502a9dbf432SEnrico Granata { 1503a9dbf432SEnrico Granata result.AppendErrorWithFormat("module importing failed: %s", error.AsCString()); 1504a9dbf432SEnrico Granata result.SetStatus (eReturnStatusFailed); 1505a9dbf432SEnrico Granata } 1506a9dbf432SEnrico Granata 1507a9dbf432SEnrico Granata return result.Succeeded(); 1508a9dbf432SEnrico Granata } 15090a305db7SEnrico Granata 15105a988416SJim Ingham CommandOptions m_options; 1511a9dbf432SEnrico Granata }; 1512223383edSEnrico Granata 15130a305db7SEnrico Granata OptionDefinition 15140a305db7SEnrico Granata CommandObjectCommandsScriptImport::CommandOptions::g_option_table[] = 15150a305db7SEnrico Granata { 1516e0c70f1bSEnrico 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."}, 15170a305db7SEnrico Granata { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 15180a305db7SEnrico Granata }; 15190a305db7SEnrico Granata 15200a305db7SEnrico Granata 1521223383edSEnrico Granata //------------------------------------------------------------------------- 1522223383edSEnrico Granata // CommandObjectCommandsScriptAdd 1523223383edSEnrico Granata //------------------------------------------------------------------------- 1524223383edSEnrico Granata 15255a988416SJim Ingham class CommandObjectCommandsScriptAdd : public CommandObjectParsed 1526223383edSEnrico Granata { 15275a988416SJim Ingham public: 15285a988416SJim Ingham CommandObjectCommandsScriptAdd(CommandInterpreter &interpreter) : 15295a988416SJim Ingham CommandObjectParsed (interpreter, 15305a988416SJim Ingham "command script add", 15315a988416SJim Ingham "Add a scripted function as an LLDB command.", 15325a988416SJim Ingham NULL), 15335a988416SJim Ingham m_options (interpreter) 15345a988416SJim Ingham { 15355a988416SJim Ingham CommandArgumentEntry arg1; 15365a988416SJim Ingham CommandArgumentData cmd_arg; 15375a988416SJim Ingham 15385a988416SJim Ingham // Define the first (and only) variant of this arg. 15395a988416SJim Ingham cmd_arg.arg_type = eArgTypeCommandName; 15405a988416SJim Ingham cmd_arg.arg_repetition = eArgRepeatPlain; 15415a988416SJim Ingham 15425a988416SJim Ingham // There is only one variant this argument could be; put it into the argument entry. 15435a988416SJim Ingham arg1.push_back (cmd_arg); 15445a988416SJim Ingham 15455a988416SJim Ingham // Push the data for the first argument into the m_arguments vector. 15465a988416SJim Ingham m_arguments.push_back (arg1); 15475a988416SJim Ingham } 15485a988416SJim Ingham 15495a988416SJim Ingham ~CommandObjectCommandsScriptAdd () 15505a988416SJim Ingham { 15515a988416SJim Ingham } 15525a988416SJim Ingham 15535a988416SJim Ingham virtual Options * 15545a988416SJim Ingham GetOptions () 15555a988416SJim Ingham { 15565a988416SJim Ingham return &m_options; 15575a988416SJim Ingham } 15585a988416SJim Ingham 15595a988416SJim Ingham protected: 1560223383edSEnrico Granata 1561223383edSEnrico Granata class CommandOptions : public Options 1562223383edSEnrico Granata { 1563223383edSEnrico Granata public: 1564223383edSEnrico Granata 1565223383edSEnrico Granata CommandOptions (CommandInterpreter &interpreter) : 1566223383edSEnrico Granata Options (interpreter) 1567223383edSEnrico Granata { 1568223383edSEnrico Granata } 1569223383edSEnrico Granata 1570223383edSEnrico Granata virtual 1571223383edSEnrico Granata ~CommandOptions (){} 1572223383edSEnrico Granata 1573223383edSEnrico Granata virtual Error 1574223383edSEnrico Granata SetOptionValue (uint32_t option_idx, const char *option_arg) 1575223383edSEnrico Granata { 1576223383edSEnrico Granata Error error; 15773bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 1578223383edSEnrico Granata 1579223383edSEnrico Granata switch (short_option) 1580223383edSEnrico Granata { 1581223383edSEnrico Granata case 'f': 1582223383edSEnrico Granata m_funct_name = std::string(option_arg); 1583223383edSEnrico Granata break; 15840a305db7SEnrico Granata case 's': 15850a305db7SEnrico Granata m_synchronous = (ScriptedCommandSynchronicity) Args::StringToOptionEnum(option_arg, g_option_table[option_idx].enum_values, 0, error); 15860a305db7SEnrico Granata if (!error.Success()) 15870a305db7SEnrico Granata error.SetErrorStringWithFormat ("unrecognized value for synchronicity '%s'", option_arg); 15880a305db7SEnrico Granata break; 1589223383edSEnrico Granata default: 159086edbf41SGreg Clayton error.SetErrorStringWithFormat ("unrecognized option '%c'", short_option); 1591223383edSEnrico Granata break; 1592223383edSEnrico Granata } 1593223383edSEnrico Granata 1594223383edSEnrico Granata return error; 1595223383edSEnrico Granata } 1596223383edSEnrico Granata 1597223383edSEnrico Granata void 1598223383edSEnrico Granata OptionParsingStarting () 1599223383edSEnrico Granata { 1600223383edSEnrico Granata m_funct_name = ""; 16010a305db7SEnrico Granata m_synchronous = eScriptedCommandSynchronicitySynchronous; 1602223383edSEnrico Granata } 1603223383edSEnrico Granata 1604223383edSEnrico Granata const OptionDefinition* 1605223383edSEnrico Granata GetDefinitions () 1606223383edSEnrico Granata { 1607223383edSEnrico Granata return g_option_table; 1608223383edSEnrico Granata } 1609223383edSEnrico Granata 1610223383edSEnrico Granata // Options table: Required for subclasses of Options. 1611223383edSEnrico Granata 1612223383edSEnrico Granata static OptionDefinition g_option_table[]; 1613223383edSEnrico Granata 1614223383edSEnrico Granata // Instance variables to hold the values for command options. 1615223383edSEnrico Granata 1616223383edSEnrico Granata std::string m_funct_name; 16170a305db7SEnrico Granata ScriptedCommandSynchronicity m_synchronous; 1618223383edSEnrico Granata }; 1619223383edSEnrico Granata 16205a988416SJim Ingham private: 1621223383edSEnrico Granata class PythonAliasReader : public InputReaderEZ 1622223383edSEnrico Granata { 1623223383edSEnrico Granata private: 1624223383edSEnrico Granata CommandInterpreter& m_interpreter; 1625223383edSEnrico Granata std::string m_cmd_name; 16260a305db7SEnrico Granata ScriptedCommandSynchronicity m_synchronous; 1627223383edSEnrico Granata StringList m_user_input; 1628223383edSEnrico Granata DISALLOW_COPY_AND_ASSIGN (PythonAliasReader); 1629223383edSEnrico Granata public: 1630223383edSEnrico Granata PythonAliasReader(Debugger& debugger, 1631223383edSEnrico Granata CommandInterpreter& interpreter, 16320a305db7SEnrico Granata std::string cmd_name, 16330a305db7SEnrico Granata ScriptedCommandSynchronicity synch) : 1634223383edSEnrico Granata InputReaderEZ(debugger), 1635223383edSEnrico Granata m_interpreter(interpreter), 1636223383edSEnrico Granata m_cmd_name(cmd_name), 16370a305db7SEnrico Granata m_synchronous(synch), 1638223383edSEnrico Granata m_user_input() 1639223383edSEnrico Granata {} 1640223383edSEnrico Granata 1641223383edSEnrico Granata virtual 1642223383edSEnrico Granata ~PythonAliasReader() 1643223383edSEnrico Granata { 1644223383edSEnrico Granata } 1645223383edSEnrico Granata 1646223383edSEnrico Granata virtual void ActivateHandler(HandlerData& data) 1647223383edSEnrico Granata { 1648223383edSEnrico Granata StreamSP out_stream = data.GetOutStream(); 1649223383edSEnrico Granata bool batch_mode = data.GetBatchMode(); 1650223383edSEnrico Granata if (!batch_mode) 1651223383edSEnrico Granata { 1652223383edSEnrico Granata out_stream->Printf ("%s\n", g_python_command_instructions); 1653223383edSEnrico Granata if (data.reader.GetPrompt()) 1654223383edSEnrico Granata out_stream->Printf ("%s", data.reader.GetPrompt()); 1655223383edSEnrico Granata out_stream->Flush(); 1656223383edSEnrico Granata } 1657223383edSEnrico Granata } 1658223383edSEnrico Granata 1659223383edSEnrico Granata virtual void ReactivateHandler(HandlerData& data) 1660223383edSEnrico Granata { 1661223383edSEnrico Granata StreamSP out_stream = data.GetOutStream(); 1662223383edSEnrico Granata bool batch_mode = data.GetBatchMode(); 1663223383edSEnrico Granata if (data.reader.GetPrompt() && !batch_mode) 1664223383edSEnrico Granata { 1665223383edSEnrico Granata out_stream->Printf ("%s", data.reader.GetPrompt()); 1666223383edSEnrico Granata out_stream->Flush(); 1667223383edSEnrico Granata } 1668223383edSEnrico Granata } 1669223383edSEnrico Granata virtual void GotTokenHandler(HandlerData& data) 1670223383edSEnrico Granata { 1671223383edSEnrico Granata StreamSP out_stream = data.GetOutStream(); 1672223383edSEnrico Granata bool batch_mode = data.GetBatchMode(); 1673223383edSEnrico Granata if (data.bytes && data.bytes_len) 1674223383edSEnrico Granata { 1675223383edSEnrico Granata m_user_input.AppendString(data.bytes, data.bytes_len); 1676223383edSEnrico Granata } 1677223383edSEnrico Granata if (!data.reader.IsDone() && data.reader.GetPrompt() && !batch_mode) 1678223383edSEnrico Granata { 1679223383edSEnrico Granata out_stream->Printf ("%s", data.reader.GetPrompt()); 1680223383edSEnrico Granata out_stream->Flush(); 1681223383edSEnrico Granata } 1682223383edSEnrico Granata } 1683223383edSEnrico Granata virtual void InterruptHandler(HandlerData& data) 1684223383edSEnrico Granata { 1685223383edSEnrico Granata StreamSP out_stream = data.GetOutStream(); 1686223383edSEnrico Granata bool batch_mode = data.GetBatchMode(); 1687223383edSEnrico Granata data.reader.SetIsDone (true); 1688223383edSEnrico Granata if (!batch_mode) 1689223383edSEnrico Granata { 16900a305db7SEnrico Granata out_stream->Printf ("Warning: No script attached.\n"); 1691223383edSEnrico Granata out_stream->Flush(); 1692223383edSEnrico Granata } 1693223383edSEnrico Granata } 1694223383edSEnrico Granata virtual void EOFHandler(HandlerData& data) 1695223383edSEnrico Granata { 1696223383edSEnrico Granata data.reader.SetIsDone (true); 1697223383edSEnrico Granata } 1698223383edSEnrico Granata virtual void DoneHandler(HandlerData& data) 1699223383edSEnrico Granata { 1700223383edSEnrico Granata StreamSP out_stream = data.GetOutStream(); 1701223383edSEnrico Granata 1702223383edSEnrico Granata ScriptInterpreter *interpreter = data.reader.GetDebugger().GetCommandInterpreter().GetScriptInterpreter(); 1703223383edSEnrico Granata if (!interpreter) 1704223383edSEnrico Granata { 17050a305db7SEnrico Granata out_stream->Printf ("Script interpreter missing: no script attached.\n"); 1706223383edSEnrico Granata out_stream->Flush(); 1707223383edSEnrico Granata return; 1708223383edSEnrico Granata } 1709a73b7df7SEnrico Granata std::string funct_name_str; 1710223383edSEnrico Granata if (!interpreter->GenerateScriptAliasFunction (m_user_input, 1711a73b7df7SEnrico Granata funct_name_str)) 1712223383edSEnrico Granata { 17130a305db7SEnrico Granata out_stream->Printf ("Unable to create function: no script attached.\n"); 1714223383edSEnrico Granata out_stream->Flush(); 1715223383edSEnrico Granata return; 1716223383edSEnrico Granata } 1717a73b7df7SEnrico Granata if (funct_name_str.empty()) 1718223383edSEnrico Granata { 17190a305db7SEnrico Granata out_stream->Printf ("Unable to obtain a function name: no script attached.\n"); 1720223383edSEnrico Granata out_stream->Flush(); 1721223383edSEnrico Granata return; 1722223383edSEnrico Granata } 1723223383edSEnrico Granata // everything should be fine now, let's add this alias 1724223383edSEnrico Granata 1725223383edSEnrico Granata CommandObjectSP command_obj_sp(new CommandObjectPythonFunction(m_interpreter, 1726223383edSEnrico Granata m_cmd_name, 1727a73b7df7SEnrico Granata funct_name_str.c_str(), 17280a305db7SEnrico Granata m_synchronous)); 1729223383edSEnrico Granata 17300a305db7SEnrico Granata if (!m_interpreter.AddUserCommand(m_cmd_name, command_obj_sp, true)) 1731223383edSEnrico Granata { 17320a305db7SEnrico Granata out_stream->Printf ("Unable to add selected command: no script attached.\n"); 1733223383edSEnrico Granata out_stream->Flush(); 1734223383edSEnrico Granata return; 1735223383edSEnrico Granata } 1736223383edSEnrico Granata } 1737223383edSEnrico Granata }; 1738223383edSEnrico Granata 17395a988416SJim Ingham protected: 1740223383edSEnrico Granata bool 17415a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 1742223383edSEnrico Granata { 174399f0b8f9SEnrico Granata 174499f0b8f9SEnrico Granata if (m_interpreter.GetDebugger().GetScriptLanguage() != lldb::eScriptLanguagePython) 174599f0b8f9SEnrico Granata { 174699f0b8f9SEnrico Granata result.AppendError ("only scripting language supported for scripted commands is currently Python"); 174799f0b8f9SEnrico Granata result.SetStatus (eReturnStatusFailed); 174899f0b8f9SEnrico Granata return false; 174999f0b8f9SEnrico Granata } 175099f0b8f9SEnrico Granata 17515a988416SJim Ingham size_t argc = command.GetArgumentCount(); 1752223383edSEnrico Granata 1753223383edSEnrico Granata if (argc != 1) 1754223383edSEnrico Granata { 1755223383edSEnrico Granata result.AppendError ("'command script add' requires one argument"); 1756223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1757223383edSEnrico Granata return false; 1758223383edSEnrico Granata } 1759223383edSEnrico Granata 17605a988416SJim Ingham std::string cmd_name = command.GetArgumentAtIndex(0); 1761223383edSEnrico Granata 1762223383edSEnrico Granata if (m_options.m_funct_name.empty()) 1763223383edSEnrico Granata { 1764223383edSEnrico Granata InputReaderSP reader_sp (new PythonAliasReader (m_interpreter.GetDebugger(), 1765223383edSEnrico Granata m_interpreter, 17660a305db7SEnrico Granata cmd_name, 17670a305db7SEnrico Granata m_options.m_synchronous)); 1768223383edSEnrico Granata 1769223383edSEnrico Granata if (reader_sp) 1770223383edSEnrico Granata { 1771223383edSEnrico Granata 1772223383edSEnrico Granata InputReaderEZ::InitializationParameters ipr; 1773223383edSEnrico Granata 1774223383edSEnrico Granata Error err (reader_sp->Initialize (ipr.SetBaton(NULL).SetPrompt(" "))); 1775223383edSEnrico Granata if (err.Success()) 1776223383edSEnrico Granata { 1777223383edSEnrico Granata m_interpreter.GetDebugger().PushInputReader (reader_sp); 1778223383edSEnrico Granata result.SetStatus (eReturnStatusSuccessFinishNoResult); 1779223383edSEnrico Granata } 1780223383edSEnrico Granata else 1781223383edSEnrico Granata { 1782223383edSEnrico Granata result.AppendError (err.AsCString()); 1783223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1784223383edSEnrico Granata } 1785223383edSEnrico Granata } 1786223383edSEnrico Granata else 1787223383edSEnrico Granata { 1788223383edSEnrico Granata result.AppendError("out of memory"); 1789223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1790223383edSEnrico Granata } 1791223383edSEnrico Granata } 1792223383edSEnrico Granata else 1793223383edSEnrico Granata { 17940a305db7SEnrico Granata CommandObjectSP new_cmd(new CommandObjectPythonFunction(m_interpreter, 17950a305db7SEnrico Granata cmd_name, 17960a305db7SEnrico Granata m_options.m_funct_name, 17970a305db7SEnrico Granata m_options.m_synchronous)); 17980a305db7SEnrico Granata if (m_interpreter.AddUserCommand(cmd_name, new_cmd, true)) 1799223383edSEnrico Granata { 1800223383edSEnrico Granata result.SetStatus (eReturnStatusSuccessFinishNoResult); 1801223383edSEnrico Granata } 1802223383edSEnrico Granata else 1803223383edSEnrico Granata { 1804223383edSEnrico Granata result.AppendError("cannot add command"); 1805223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1806223383edSEnrico Granata } 1807223383edSEnrico Granata } 1808223383edSEnrico Granata 1809223383edSEnrico Granata return result.Succeeded(); 1810223383edSEnrico Granata 1811223383edSEnrico Granata } 18125a988416SJim Ingham 18135a988416SJim Ingham CommandOptions m_options; 1814223383edSEnrico Granata }; 1815223383edSEnrico Granata 18160a305db7SEnrico Granata static OptionEnumValueElement g_script_synchro_type[] = 18170a305db7SEnrico Granata { 18180a305db7SEnrico Granata { eScriptedCommandSynchronicitySynchronous, "synchronous", "Run synchronous"}, 18190a305db7SEnrico Granata { eScriptedCommandSynchronicityAsynchronous, "asynchronous", "Run asynchronous"}, 18200a305db7SEnrico Granata { eScriptedCommandSynchronicityCurrentValue, "current", "Do not alter current setting"}, 18210a305db7SEnrico Granata { 0, NULL, NULL } 18220a305db7SEnrico Granata }; 18230a305db7SEnrico Granata 1824223383edSEnrico Granata OptionDefinition 1825223383edSEnrico Granata CommandObjectCommandsScriptAdd::CommandOptions::g_option_table[] = 1826223383edSEnrico Granata { 18279128ee2fSEnrico Granata { LLDB_OPT_SET_1, false, "function", 'f', required_argument, NULL, 0, eArgTypePythonFunction, "Name of the Python function to bind to this command name."}, 18280a305db7SEnrico 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."}, 1829223383edSEnrico Granata { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 1830223383edSEnrico Granata }; 1831223383edSEnrico Granata 1832223383edSEnrico Granata //------------------------------------------------------------------------- 1833223383edSEnrico Granata // CommandObjectCommandsScriptList 1834223383edSEnrico Granata //------------------------------------------------------------------------- 1835223383edSEnrico Granata 18365a988416SJim Ingham class CommandObjectCommandsScriptList : public CommandObjectParsed 1837223383edSEnrico Granata { 1838223383edSEnrico Granata private: 1839223383edSEnrico Granata 1840223383edSEnrico Granata public: 1841223383edSEnrico Granata CommandObjectCommandsScriptList(CommandInterpreter &interpreter) : 18425a988416SJim Ingham CommandObjectParsed (interpreter, 1843223383edSEnrico Granata "command script list", 1844223383edSEnrico Granata "List defined scripted commands.", 1845223383edSEnrico Granata NULL) 1846223383edSEnrico Granata { 1847223383edSEnrico Granata } 1848223383edSEnrico Granata 1849223383edSEnrico Granata ~CommandObjectCommandsScriptList () 1850223383edSEnrico Granata { 1851223383edSEnrico Granata } 1852223383edSEnrico Granata 1853223383edSEnrico Granata bool 18545a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 1855223383edSEnrico Granata { 1856223383edSEnrico Granata 1857223383edSEnrico Granata m_interpreter.GetHelp(result, 1858223383edSEnrico Granata CommandInterpreter::eCommandTypesUserDef); 1859223383edSEnrico Granata 1860223383edSEnrico Granata result.SetStatus (eReturnStatusSuccessFinishResult); 1861223383edSEnrico Granata 1862223383edSEnrico Granata return true; 1863223383edSEnrico Granata 1864223383edSEnrico Granata 1865223383edSEnrico Granata } 1866223383edSEnrico Granata }; 1867223383edSEnrico Granata 1868223383edSEnrico Granata //------------------------------------------------------------------------- 1869223383edSEnrico Granata // CommandObjectCommandsScriptClear 1870223383edSEnrico Granata //------------------------------------------------------------------------- 1871223383edSEnrico Granata 18725a988416SJim Ingham class CommandObjectCommandsScriptClear : public CommandObjectParsed 1873223383edSEnrico Granata { 1874223383edSEnrico Granata private: 1875223383edSEnrico Granata 1876223383edSEnrico Granata public: 1877223383edSEnrico Granata CommandObjectCommandsScriptClear(CommandInterpreter &interpreter) : 18785a988416SJim Ingham CommandObjectParsed (interpreter, 1879223383edSEnrico Granata "command script clear", 1880223383edSEnrico Granata "Delete all scripted commands.", 1881223383edSEnrico Granata NULL) 1882223383edSEnrico Granata { 1883223383edSEnrico Granata } 1884223383edSEnrico Granata 1885223383edSEnrico Granata ~CommandObjectCommandsScriptClear () 1886223383edSEnrico Granata { 1887223383edSEnrico Granata } 1888223383edSEnrico Granata 18895a988416SJim Ingham protected: 1890223383edSEnrico Granata bool 18915a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 1892223383edSEnrico Granata { 1893223383edSEnrico Granata 1894223383edSEnrico Granata m_interpreter.RemoveAllUser(); 1895223383edSEnrico Granata 1896223383edSEnrico Granata result.SetStatus (eReturnStatusSuccessFinishResult); 1897223383edSEnrico Granata 1898223383edSEnrico Granata return true; 1899223383edSEnrico Granata } 1900223383edSEnrico Granata }; 1901223383edSEnrico Granata 1902223383edSEnrico Granata //------------------------------------------------------------------------- 1903223383edSEnrico Granata // CommandObjectCommandsScriptDelete 1904223383edSEnrico Granata //------------------------------------------------------------------------- 1905223383edSEnrico Granata 19065a988416SJim Ingham class CommandObjectCommandsScriptDelete : public CommandObjectParsed 1907223383edSEnrico Granata { 1908223383edSEnrico Granata public: 1909223383edSEnrico Granata CommandObjectCommandsScriptDelete(CommandInterpreter &interpreter) : 19105a988416SJim Ingham CommandObjectParsed (interpreter, 1911223383edSEnrico Granata "command script delete", 1912223383edSEnrico Granata "Delete a scripted command.", 1913223383edSEnrico Granata NULL) 1914223383edSEnrico Granata { 1915223383edSEnrico Granata CommandArgumentEntry arg1; 1916223383edSEnrico Granata CommandArgumentData cmd_arg; 1917223383edSEnrico Granata 1918223383edSEnrico Granata // Define the first (and only) variant of this arg. 1919223383edSEnrico Granata cmd_arg.arg_type = eArgTypeCommandName; 1920223383edSEnrico Granata cmd_arg.arg_repetition = eArgRepeatPlain; 1921223383edSEnrico Granata 1922223383edSEnrico Granata // There is only one variant this argument could be; put it into the argument entry. 1923223383edSEnrico Granata arg1.push_back (cmd_arg); 1924223383edSEnrico Granata 1925223383edSEnrico Granata // Push the data for the first argument into the m_arguments vector. 1926223383edSEnrico Granata m_arguments.push_back (arg1); 1927223383edSEnrico Granata } 1928223383edSEnrico Granata 1929223383edSEnrico Granata ~CommandObjectCommandsScriptDelete () 1930223383edSEnrico Granata { 1931223383edSEnrico Granata } 1932223383edSEnrico Granata 19335a988416SJim Ingham protected: 1934223383edSEnrico Granata bool 19355a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 1936223383edSEnrico Granata { 1937223383edSEnrico Granata 19385a988416SJim Ingham size_t argc = command.GetArgumentCount(); 1939223383edSEnrico Granata 1940223383edSEnrico Granata if (argc != 1) 1941223383edSEnrico Granata { 1942223383edSEnrico Granata result.AppendError ("'command script delete' requires one argument"); 1943223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1944223383edSEnrico Granata return false; 1945223383edSEnrico Granata } 1946223383edSEnrico Granata 19475a988416SJim Ingham const char* cmd_name = command.GetArgumentAtIndex(0); 1948223383edSEnrico Granata 1949223383edSEnrico Granata if (cmd_name && *cmd_name && m_interpreter.HasUserCommands() && m_interpreter.UserCommandExists(cmd_name)) 1950223383edSEnrico Granata { 1951223383edSEnrico Granata m_interpreter.RemoveUser(cmd_name); 1952223383edSEnrico Granata result.SetStatus (eReturnStatusSuccessFinishResult); 1953223383edSEnrico Granata } 1954223383edSEnrico Granata else 1955223383edSEnrico Granata { 1956223383edSEnrico Granata result.AppendErrorWithFormat ("command %s not found", cmd_name); 1957223383edSEnrico Granata result.SetStatus (eReturnStatusFailed); 1958223383edSEnrico Granata } 1959223383edSEnrico Granata 1960223383edSEnrico Granata return result.Succeeded(); 1961223383edSEnrico Granata 1962223383edSEnrico Granata } 1963223383edSEnrico Granata }; 1964223383edSEnrico Granata 1965223383edSEnrico Granata #pragma mark CommandObjectMultiwordCommandsScript 1966223383edSEnrico Granata 1967223383edSEnrico Granata //------------------------------------------------------------------------- 1968223383edSEnrico Granata // CommandObjectMultiwordCommandsScript 1969223383edSEnrico Granata //------------------------------------------------------------------------- 1970223383edSEnrico Granata 1971223383edSEnrico Granata class CommandObjectMultiwordCommandsScript : public CommandObjectMultiword 1972223383edSEnrico Granata { 1973223383edSEnrico Granata public: 1974223383edSEnrico Granata CommandObjectMultiwordCommandsScript (CommandInterpreter &interpreter) : 1975223383edSEnrico Granata CommandObjectMultiword (interpreter, 1976223383edSEnrico Granata "command script", 1977223383edSEnrico Granata "A set of commands for managing or customizing script commands.", 1978223383edSEnrico Granata "command script <subcommand> [<subcommand-options>]") 1979223383edSEnrico Granata { 1980223383edSEnrico Granata LoadSubCommand ("add", CommandObjectSP (new CommandObjectCommandsScriptAdd (interpreter))); 1981223383edSEnrico Granata LoadSubCommand ("delete", CommandObjectSP (new CommandObjectCommandsScriptDelete (interpreter))); 1982223383edSEnrico Granata LoadSubCommand ("clear", CommandObjectSP (new CommandObjectCommandsScriptClear (interpreter))); 1983223383edSEnrico Granata LoadSubCommand ("list", CommandObjectSP (new CommandObjectCommandsScriptList (interpreter))); 1984a9dbf432SEnrico Granata LoadSubCommand ("import", CommandObjectSP (new CommandObjectCommandsScriptImport (interpreter))); 1985223383edSEnrico Granata } 1986223383edSEnrico Granata 1987223383edSEnrico Granata ~CommandObjectMultiwordCommandsScript () 1988223383edSEnrico Granata { 1989223383edSEnrico Granata } 1990223383edSEnrico Granata 1991223383edSEnrico Granata }; 1992223383edSEnrico Granata 1993223383edSEnrico Granata 1994ebc09c36SJim Ingham #pragma mark CommandObjectMultiwordCommands 1995ebc09c36SJim Ingham 1996ebc09c36SJim Ingham //------------------------------------------------------------------------- 1997ebc09c36SJim Ingham // CommandObjectMultiwordCommands 1998ebc09c36SJim Ingham //------------------------------------------------------------------------- 1999ebc09c36SJim Ingham 2000ebc09c36SJim Ingham CommandObjectMultiwordCommands::CommandObjectMultiwordCommands (CommandInterpreter &interpreter) : 2001a7015092SGreg Clayton CommandObjectMultiword (interpreter, 20020e5e5a79SGreg Clayton "command", 20033f4c09c1SCaroline Tice "A set of commands for managing or customizing the debugger commands.", 20040e5e5a79SGreg Clayton "command <subcommand> [<subcommand-options>]") 2005ebc09c36SJim Ingham { 2006a7015092SGreg Clayton LoadSubCommand ("source", CommandObjectSP (new CommandObjectCommandsSource (interpreter))); 2007a7015092SGreg Clayton LoadSubCommand ("alias", CommandObjectSP (new CommandObjectCommandsAlias (interpreter))); 2008a7015092SGreg Clayton LoadSubCommand ("unalias", CommandObjectSP (new CommandObjectCommandsUnalias (interpreter))); 2009de164aaaSGreg Clayton LoadSubCommand ("regex", CommandObjectSP (new CommandObjectCommandsAddRegex (interpreter))); 2010a5a97ebeSJim Ingham LoadSubCommand ("history", CommandObjectSP (new CommandObjectCommandsHistory (interpreter))); 2011223383edSEnrico Granata LoadSubCommand ("script", CommandObjectSP (new CommandObjectMultiwordCommandsScript (interpreter))); 2012ebc09c36SJim Ingham } 2013ebc09c36SJim Ingham 2014ebc09c36SJim Ingham CommandObjectMultiwordCommands::~CommandObjectMultiwordCommands () 2015ebc09c36SJim Ingham { 2016ebc09c36SJim Ingham } 2017ebc09c36SJim Ingham 2018