2010-06-09 00:52:24 +08:00
|
|
|
//===-- Driver.cpp ----------------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "Driver.h"
|
|
|
|
|
|
|
|
#include <getopt.h>
|
|
|
|
#include <libgen.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <termios.h>
|
|
|
|
#include <unistd.h>
|
2010-06-09 17:50:17 +08:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <limits.h>
|
2010-06-10 03:11:30 +08:00
|
|
|
#include <fcntl.h>
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
#include "IOChannel.h"
|
2012-02-08 13:23:15 +08:00
|
|
|
#include "lldb/API/SBBreakpoint.h"
|
2010-06-09 17:50:17 +08:00
|
|
|
#include "lldb/API/SBCommandInterpreter.h"
|
|
|
|
#include "lldb/API/SBCommandReturnObject.h"
|
|
|
|
#include "lldb/API/SBCommunication.h"
|
|
|
|
#include "lldb/API/SBDebugger.h"
|
|
|
|
#include "lldb/API/SBEvent.h"
|
|
|
|
#include "lldb/API/SBHostOS.h"
|
|
|
|
#include "lldb/API/SBListener.h"
|
2011-02-19 10:53:09 +08:00
|
|
|
#include "lldb/API/SBStream.h"
|
2010-06-09 17:50:17 +08:00
|
|
|
#include "lldb/API/SBTarget.h"
|
|
|
|
#include "lldb/API/SBThread.h"
|
|
|
|
#include "lldb/API/SBProcess.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
using namespace lldb;
|
|
|
|
|
|
|
|
static void reset_stdin_termios ();
|
2012-02-03 03:28:31 +08:00
|
|
|
static bool g_old_stdin_termios_is_valid = false;
|
2010-06-09 00:52:24 +08:00
|
|
|
static struct termios g_old_stdin_termios;
|
|
|
|
|
2010-09-10 01:45:09 +08:00
|
|
|
static char *g_debugger_name = (char *) "";
|
2010-11-20 04:47:54 +08:00
|
|
|
static Driver *g_driver = NULL;
|
2010-09-10 01:45:09 +08:00
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
// In the Driver::MainLoop, we change the terminal settings. This function is
|
|
|
|
// added as an atexit handler to make sure we clean them up.
|
|
|
|
static void
|
|
|
|
reset_stdin_termios ()
|
|
|
|
{
|
2012-02-03 03:28:31 +08:00
|
|
|
if (g_old_stdin_termios_is_valid)
|
|
|
|
{
|
|
|
|
g_old_stdin_termios_is_valid = false;
|
|
|
|
::tcsetattr (STDIN_FILENO, TCSANOW, &g_old_stdin_termios);
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2011-03-25 05:19:54 +08:00
|
|
|
typedef struct
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2011-03-25 05:19:54 +08:00
|
|
|
uint32_t usage_mask; // Used to mark options that can be used together. If (1 << n & usage_mask) != 0
|
|
|
|
// then this option belongs to option set n.
|
|
|
|
bool required; // This option is required (in the current usage level)
|
|
|
|
const char * long_option; // Full name for this option.
|
|
|
|
char short_option; // Single character for this option.
|
|
|
|
int option_has_arg; // no_argument, required_argument or optional_argument
|
2011-04-14 06:47:15 +08:00
|
|
|
uint32_t completion_type; // Cookie the option class can use to do define the argument completion.
|
2011-03-25 05:19:54 +08:00
|
|
|
lldb::CommandArgumentType argument_type; // Type of argument this option takes
|
|
|
|
const char * usage_text; // Full text explaining what this options does and what (if any) argument to
|
|
|
|
// pass it.
|
|
|
|
} OptionDefinition;
|
|
|
|
|
2011-09-14 07:25:31 +08:00
|
|
|
#define LLDB_3_TO_5 LLDB_OPT_SET_3|LLDB_OPT_SET_4|LLDB_OPT_SET_5
|
|
|
|
#define LLDB_4_TO_5 LLDB_OPT_SET_4|LLDB_OPT_SET_5
|
|
|
|
|
2011-03-25 05:19:54 +08:00
|
|
|
static OptionDefinition g_options[] =
|
|
|
|
{
|
2011-09-16 05:30:02 +08:00
|
|
|
{ LLDB_OPT_SET_1, true , "help" , 'h', no_argument , NULL, eArgTypeNone,
|
|
|
|
"Prints out the usage information for the LLDB debugger." },
|
|
|
|
{ LLDB_OPT_SET_2, true , "version" , 'v', no_argument , NULL, eArgTypeNone,
|
|
|
|
"Prints out the current version number of the LLDB debugger." },
|
|
|
|
{ LLDB_OPT_SET_3, true , "arch" , 'a', required_argument, NULL, eArgTypeArchitecture,
|
|
|
|
"Tells the debugger to use the specified architecture when starting and running the program. <architecture> must "
|
|
|
|
"be one of the architectures for which the program was compiled." },
|
|
|
|
{ LLDB_OPT_SET_3, true , "file" , 'f', required_argument, NULL, eArgTypeFilename,
|
|
|
|
"Tells the debugger to use the file <filename> as the program to be debugged." },
|
|
|
|
{ LLDB_OPT_SET_4, true , "attach-name" , 'n', required_argument, NULL, eArgTypeProcessName,
|
|
|
|
"Tells the debugger to attach to a process with the given name." },
|
|
|
|
{ LLDB_OPT_SET_4, true , "wait-for" , 'w', no_argument , NULL, eArgTypeNone,
|
|
|
|
"Tells the debugger to wait for a process with the given pid or name to launch before attaching." },
|
|
|
|
{ LLDB_OPT_SET_5, true , "attach-pid" , 'p', required_argument, NULL, eArgTypePid,
|
|
|
|
"Tells the debugger to attach to a process with the given pid." },
|
|
|
|
{ LLDB_3_TO_5, false, "script-language", 'l', required_argument, NULL, eArgTypeScriptLang,
|
|
|
|
"Tells the debugger to use the specified scripting language for user-defined scripts, rather than the default. "
|
|
|
|
"Valid scripting languages that can be specified include Python, Perl, Ruby and Tcl. Currently only the Python "
|
|
|
|
"extensions have been implemented." },
|
|
|
|
{ LLDB_3_TO_5, false, "debug" , 'd', no_argument , NULL, eArgTypeNone,
|
|
|
|
"Tells the debugger to print out extra information for debugging itself." },
|
|
|
|
{ LLDB_3_TO_5, false, "source" , 's', required_argument, NULL, eArgTypeFilename,
|
|
|
|
"Tells the debugger to read in and execute the file <file>, which should contain lldb commands." },
|
|
|
|
{ LLDB_3_TO_5, false, "editor" , 'e', no_argument , NULL, eArgTypeNone,
|
|
|
|
"Tells the debugger to open source files using the host's \"external editor\" mechanism." },
|
|
|
|
{ LLDB_3_TO_5, false, "no-lldbinit" , 'x', no_argument , NULL, eArgTypeNone,
|
|
|
|
"Do not automatically parse any '.lldbinit' files." },
|
2011-09-14 07:25:31 +08:00
|
|
|
{ 0, false, NULL , 0 , 0 , NULL, eArgTypeNone, NULL }
|
2010-06-09 00:52:24 +08:00
|
|
|
};
|
|
|
|
|
2011-09-14 07:25:31 +08:00
|
|
|
static const uint32_t last_option_set_with_args = 2;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
Driver::Driver () :
|
|
|
|
SBBroadcaster ("Driver"),
|
2011-08-13 08:22:20 +08:00
|
|
|
m_debugger (SBDebugger::Create(false)),
|
2010-06-09 00:52:24 +08:00
|
|
|
m_editline_pty (),
|
|
|
|
m_editline_slave_fh (NULL),
|
|
|
|
m_editline_reader (),
|
|
|
|
m_io_channel_ap (),
|
|
|
|
m_option_data (),
|
|
|
|
m_waiting_for_command (false)
|
|
|
|
{
|
2011-05-29 12:06:55 +08:00
|
|
|
// We want to be able to handle CTRL+D in the terminal to have it terminate
|
|
|
|
// certain input
|
|
|
|
m_debugger.SetCloseInputOnEOF (false);
|
2010-09-10 01:45:09 +08:00
|
|
|
g_debugger_name = (char *) m_debugger.GetInstanceName();
|
|
|
|
if (g_debugger_name == NULL)
|
|
|
|
g_debugger_name = (char *) "";
|
2010-11-20 04:47:54 +08:00
|
|
|
g_driver = this;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Driver::~Driver ()
|
|
|
|
{
|
2010-11-20 04:47:54 +08:00
|
|
|
g_driver = NULL;
|
|
|
|
g_debugger_name = NULL;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Driver::CloseIOChannelFile ()
|
|
|
|
{
|
|
|
|
// Write and End of File sequence to the file descriptor to ensure any
|
|
|
|
// read functions can exit.
|
|
|
|
char eof_str[] = "\x04";
|
|
|
|
::write (m_editline_pty.GetMasterFileDescriptor(), eof_str, strlen(eof_str));
|
|
|
|
|
|
|
|
m_editline_pty.CloseMasterFileDescriptor();
|
|
|
|
|
|
|
|
if (m_editline_slave_fh)
|
|
|
|
{
|
|
|
|
::fclose (m_editline_slave_fh);
|
|
|
|
m_editline_slave_fh = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-10 04:39:50 +08:00
|
|
|
// This function takes INDENT, which tells how many spaces to output at the front
|
|
|
|
// of each line; TEXT, which is the text that is to be output. It outputs the
|
|
|
|
// text, on multiple lines if necessary, to RESULT, with INDENT spaces at the
|
|
|
|
// front of each line. It breaks lines on spaces, tabs or newlines, shortening
|
|
|
|
// the line if necessary to not break in the middle of a word. It assumes that
|
|
|
|
// each output line should contain a maximum of OUTPUT_MAX_COLUMNS characters.
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
void
|
2010-07-10 04:39:50 +08:00
|
|
|
OutputFormattedUsageText (FILE *out, int indent, const char *text, int output_max_columns)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
int len = strlen (text);
|
|
|
|
std::string text_string (text);
|
|
|
|
|
|
|
|
// Force indentation to be reasonable.
|
|
|
|
if (indent >= output_max_columns)
|
|
|
|
indent = 0;
|
|
|
|
|
|
|
|
// Will it all fit on one line?
|
|
|
|
|
|
|
|
if (len + indent < output_max_columns)
|
|
|
|
// Output as a single line
|
2010-07-10 04:39:50 +08:00
|
|
|
fprintf (out, "%*s%s\n", indent, "", text);
|
2010-06-09 00:52:24 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// We need to break it up into multiple lines.
|
|
|
|
int text_width = output_max_columns - indent - 1;
|
|
|
|
int start = 0;
|
|
|
|
int end = start;
|
|
|
|
int final_end = len;
|
|
|
|
int sub_len;
|
|
|
|
|
|
|
|
while (end < final_end)
|
|
|
|
{
|
|
|
|
// Dont start the 'text' on a space, since we're already outputting the indentation.
|
|
|
|
while ((start < final_end) && (text[start] == ' '))
|
|
|
|
start++;
|
|
|
|
|
|
|
|
end = start + text_width;
|
|
|
|
if (end > final_end)
|
|
|
|
end = final_end;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// If we're not at the end of the text, make sure we break the line on white space.
|
|
|
|
while (end > start
|
|
|
|
&& text[end] != ' ' && text[end] != '\t' && text[end] != '\n')
|
|
|
|
end--;
|
|
|
|
}
|
|
|
|
sub_len = end - start;
|
|
|
|
std::string substring = text_string.substr (start, sub_len);
|
2010-07-10 04:39:50 +08:00
|
|
|
fprintf (out, "%*s%s\n", indent, "", substring.c_str());
|
2010-06-09 00:52:24 +08:00
|
|
|
start = end + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-03-25 05:19:54 +08:00
|
|
|
ShowUsage (FILE *out, OptionDefinition *option_table, Driver::OptionData data)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
uint32_t screen_width = 80;
|
|
|
|
uint32_t indent_level = 0;
|
|
|
|
const char *name = "lldb";
|
2010-06-16 02:47:14 +08:00
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
fprintf (out, "\nUsage:\n\n");
|
|
|
|
|
|
|
|
indent_level += 2;
|
|
|
|
|
|
|
|
|
|
|
|
// First, show each usage level set of options, e.g. <cmd> [options-for-level-0]
|
|
|
|
// <cmd> [options-for-level-1]
|
|
|
|
// etc.
|
|
|
|
|
|
|
|
uint32_t num_options;
|
2010-06-16 02:47:14 +08:00
|
|
|
uint32_t num_option_sets = 0;
|
|
|
|
|
|
|
|
for (num_options = 0; option_table[num_options].long_option != NULL; ++num_options)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-06-16 02:47:14 +08:00
|
|
|
uint32_t this_usage_mask = option_table[num_options].usage_mask;
|
|
|
|
if (this_usage_mask == LLDB_OPT_SET_ALL)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-06-16 02:47:14 +08:00
|
|
|
if (num_option_sets == 0)
|
|
|
|
num_option_sets = 1;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2010-06-16 02:47:14 +08:00
|
|
|
else
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-07-10 04:39:50 +08:00
|
|
|
for (uint32_t j = 0; j < LLDB_MAX_NUM_OPTION_SETS; j++)
|
2010-06-16 02:47:14 +08:00
|
|
|
{
|
|
|
|
if (this_usage_mask & 1 << j)
|
|
|
|
{
|
|
|
|
if (num_option_sets <= j)
|
|
|
|
num_option_sets = j + 1;
|
|
|
|
}
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2010-06-16 02:47:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
for (uint32_t opt_set = 0; opt_set < num_option_sets; opt_set++)
|
|
|
|
{
|
|
|
|
uint32_t opt_set_mask;
|
|
|
|
|
|
|
|
opt_set_mask = 1 << opt_set;
|
|
|
|
|
|
|
|
if (opt_set > 0)
|
|
|
|
fprintf (out, "\n");
|
2010-07-10 04:39:50 +08:00
|
|
|
fprintf (out, "%*s%s", indent_level, "", name);
|
2011-08-17 07:15:02 +08:00
|
|
|
bool is_help_line = false;
|
2010-06-16 02:47:14 +08:00
|
|
|
|
|
|
|
for (uint32_t i = 0; i < num_options; ++i)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-06-16 02:47:14 +08:00
|
|
|
if (option_table[i].usage_mask & opt_set_mask)
|
|
|
|
{
|
2010-10-02 03:59:14 +08:00
|
|
|
CommandArgumentType arg_type = option_table[i].argument_type;
|
2011-02-20 10:15:07 +08:00
|
|
|
const char *arg_name = SBCommandInterpreter::GetArgumentTypeAsCString (arg_type);
|
2011-08-17 07:15:02 +08:00
|
|
|
// This is a bit of a hack, but there's no way to say certain options don't have arguments yet...
|
|
|
|
// so we do it by hand here.
|
|
|
|
if (option_table[i].short_option == 'h')
|
|
|
|
is_help_line = true;
|
|
|
|
|
2010-06-16 02:47:14 +08:00
|
|
|
if (option_table[i].required)
|
|
|
|
{
|
|
|
|
if (option_table[i].option_has_arg == required_argument)
|
2011-02-20 10:15:07 +08:00
|
|
|
fprintf (out, " -%c <%s>", option_table[i].short_option, arg_name);
|
2010-06-16 02:47:14 +08:00
|
|
|
else if (option_table[i].option_has_arg == optional_argument)
|
2011-02-20 10:15:07 +08:00
|
|
|
fprintf (out, " -%c [<%s>]", option_table[i].short_option, arg_name);
|
2010-06-16 02:47:14 +08:00
|
|
|
else
|
|
|
|
fprintf (out, " -%c", option_table[i].short_option);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (option_table[i].option_has_arg == required_argument)
|
2011-02-20 10:15:07 +08:00
|
|
|
fprintf (out, " [-%c <%s>]", option_table[i].short_option, arg_name);
|
2010-06-16 02:47:14 +08:00
|
|
|
else if (option_table[i].option_has_arg == optional_argument)
|
2011-02-20 10:15:07 +08:00
|
|
|
fprintf (out, " [-%c [<%s>]]", option_table[i].short_option, arg_name);
|
2010-06-16 02:47:14 +08:00
|
|
|
else
|
|
|
|
fprintf (out, " [-%c]", option_table[i].short_option);
|
|
|
|
}
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2011-09-14 07:25:31 +08:00
|
|
|
if (!is_help_line && (opt_set <= last_option_set_with_args))
|
2011-08-17 07:15:02 +08:00
|
|
|
fprintf (out, " [[--] <PROGRAM-ARG-1> [<PROGRAM_ARG-2> ...]]");
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
fprintf (out, "\n\n");
|
|
|
|
|
|
|
|
// Now print out all the detailed information about the various options: long form, short form and help text:
|
|
|
|
// -- long_name <argument>
|
|
|
|
// - short <argument>
|
|
|
|
// help text
|
|
|
|
|
|
|
|
// This variable is used to keep track of which options' info we've printed out, because some options can be in
|
|
|
|
// more than one usage level, but we only want to print the long form of its information once.
|
|
|
|
|
|
|
|
Driver::OptionData::OptionSet options_seen;
|
|
|
|
Driver::OptionData::OptionSet::iterator pos;
|
|
|
|
|
|
|
|
indent_level += 5;
|
|
|
|
|
2010-06-16 02:47:14 +08:00
|
|
|
for (uint32_t i = 0; i < num_options; ++i)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
// Only print this option if we haven't already seen it.
|
|
|
|
pos = options_seen.find (option_table[i].short_option);
|
|
|
|
if (pos == options_seen.end())
|
|
|
|
{
|
2010-10-02 03:59:14 +08:00
|
|
|
CommandArgumentType arg_type = option_table[i].argument_type;
|
2011-02-20 10:15:07 +08:00
|
|
|
const char *arg_name = SBCommandInterpreter::GetArgumentTypeAsCString (arg_type);
|
2010-10-02 03:59:14 +08:00
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
options_seen.insert (option_table[i].short_option);
|
2010-07-10 04:39:50 +08:00
|
|
|
fprintf (out, "%*s-%c ", indent_level, "", option_table[i].short_option);
|
2010-10-02 03:59:14 +08:00
|
|
|
if (arg_type != eArgTypeNone)
|
2011-02-20 10:15:07 +08:00
|
|
|
fprintf (out, "<%s>", arg_name);
|
2010-06-09 00:52:24 +08:00
|
|
|
fprintf (out, "\n");
|
2010-07-10 04:39:50 +08:00
|
|
|
fprintf (out, "%*s--%s ", indent_level, "", option_table[i].long_option);
|
2010-10-02 03:59:14 +08:00
|
|
|
if (arg_type != eArgTypeNone)
|
2011-02-20 10:15:07 +08:00
|
|
|
fprintf (out, "<%s>", arg_name);
|
2010-06-09 00:52:24 +08:00
|
|
|
fprintf (out, "\n");
|
|
|
|
indent_level += 5;
|
2010-07-10 04:39:50 +08:00
|
|
|
OutputFormattedUsageText (out, indent_level, option_table[i].usage_text, screen_width);
|
2010-06-09 00:52:24 +08:00
|
|
|
indent_level -= 5;
|
|
|
|
fprintf (out, "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
indent_level -= 5;
|
|
|
|
|
2011-08-17 07:57:58 +08:00
|
|
|
fprintf (out, "\n%*s(If you don't provide -f then the first argument will be the file to be debugged"
|
|
|
|
"\n%*s so '%s -- <filename> [<ARG1> [<ARG2>]]' also works."
|
|
|
|
"\n%*s Remember to end the options with \"--\" if any of your arguments have a \"-\" in them.)\n\n",
|
|
|
|
indent_level, "",
|
|
|
|
indent_level, "",
|
|
|
|
name,
|
|
|
|
indent_level, "");
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-03-25 05:19:54 +08:00
|
|
|
BuildGetOptTable (OptionDefinition *expanded_option_table, std::vector<struct option> &getopt_table,
|
2010-10-13 05:57:09 +08:00
|
|
|
uint32_t num_options)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
if (num_options == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
uint32_t i;
|
|
|
|
uint32_t j;
|
|
|
|
std::bitset<256> option_seen;
|
|
|
|
|
2010-10-13 05:57:09 +08:00
|
|
|
getopt_table.resize (num_options + 1);
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
for (i = 0, j = 0; i < num_options; ++i)
|
2010-07-10 04:39:50 +08:00
|
|
|
{
|
2010-06-09 00:52:24 +08:00
|
|
|
char short_opt = expanded_option_table[i].short_option;
|
2010-07-10 04:39:50 +08:00
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
if (option_seen.test(short_opt) == false)
|
2010-07-10 04:39:50 +08:00
|
|
|
{
|
2010-10-13 05:57:09 +08:00
|
|
|
getopt_table[j].name = expanded_option_table[i].long_option;
|
|
|
|
getopt_table[j].has_arg = expanded_option_table[i].option_has_arg;
|
|
|
|
getopt_table[j].flag = NULL;
|
|
|
|
getopt_table[j].val = expanded_option_table[i].short_option;
|
2010-06-09 00:52:24 +08:00
|
|
|
option_seen.set(short_opt);
|
|
|
|
++j;
|
2010-07-10 04:39:50 +08:00
|
|
|
}
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-10-13 05:57:09 +08:00
|
|
|
getopt_table[j].name = NULL;
|
|
|
|
getopt_table[j].has_arg = 0;
|
|
|
|
getopt_table[j].flag = NULL;
|
|
|
|
getopt_table[j].val = 0;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
Driver::OptionData::OptionData () :
|
2010-12-09 06:23:24 +08:00
|
|
|
m_args(),
|
2010-06-23 09:19:29 +08:00
|
|
|
m_script_lang (lldb::eScriptLanguageDefault),
|
2010-07-10 04:39:50 +08:00
|
|
|
m_crash_log (),
|
2010-06-23 09:19:29 +08:00
|
|
|
m_source_command_files (),
|
|
|
|
m_debug_mode (false),
|
2010-07-10 04:39:50 +08:00
|
|
|
m_print_version (false),
|
2010-06-23 09:19:29 +08:00
|
|
|
m_print_help (false),
|
2011-09-14 07:25:31 +08:00
|
|
|
m_wait_for(false),
|
|
|
|
m_process_name(),
|
|
|
|
m_process_pid(LLDB_INVALID_PROCESS_ID),
|
2011-11-01 06:50:49 +08:00
|
|
|
m_use_external_editor(false),
|
2011-04-12 03:41:40 +08:00
|
|
|
m_seen_options()
|
2010-06-23 09:19:29 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Driver::OptionData::~OptionData ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Driver::OptionData::Clear ()
|
|
|
|
{
|
2010-12-09 06:23:24 +08:00
|
|
|
m_args.clear ();
|
2010-06-23 09:19:29 +08:00
|
|
|
m_script_lang = lldb::eScriptLanguageDefault;
|
|
|
|
m_source_command_files.clear ();
|
|
|
|
m_debug_mode = false;
|
|
|
|
m_print_help = false;
|
|
|
|
m_print_version = false;
|
2010-08-31 03:44:40 +08:00
|
|
|
m_use_external_editor = false;
|
2011-09-14 07:25:31 +08:00
|
|
|
m_wait_for = false;
|
|
|
|
m_process_name.erase();
|
|
|
|
m_process_pid = LLDB_INVALID_PROCESS_ID;
|
2010-06-23 09:19:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Driver::ResetOptionValues ()
|
|
|
|
{
|
|
|
|
m_option_data.Clear ();
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
Driver::GetFilename() const
|
|
|
|
{
|
2010-12-09 06:23:24 +08:00
|
|
|
if (m_option_data.m_args.empty())
|
2010-06-23 09:19:29 +08:00
|
|
|
return NULL;
|
2010-12-09 06:23:24 +08:00
|
|
|
return m_option_data.m_args.front().c_str();
|
2010-06-23 09:19:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
Driver::GetCrashLogFilename() const
|
|
|
|
{
|
|
|
|
if (m_option_data.m_crash_log.empty())
|
|
|
|
return NULL;
|
|
|
|
return m_option_data.m_crash_log.c_str();
|
|
|
|
}
|
|
|
|
|
|
|
|
lldb::ScriptLanguage
|
|
|
|
Driver::GetScriptLanguage() const
|
|
|
|
{
|
|
|
|
return m_option_data.m_script_lang;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t
|
|
|
|
Driver::GetNumSourceCommandFiles () const
|
|
|
|
{
|
|
|
|
return m_option_data.m_source_command_files.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
Driver::GetSourceCommandFileAtIndex (uint32_t idx) const
|
|
|
|
{
|
|
|
|
if (idx < m_option_data.m_source_command_files.size())
|
|
|
|
return m_option_data.m_source_command_files[idx].c_str();
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Driver::GetDebugMode() const
|
|
|
|
{
|
|
|
|
return m_option_data.m_debug_mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Check the arguments that were passed to this program to make sure they are valid and to get their
|
|
|
|
// argument values (if any). Return a boolean value indicating whether or not to start up the full
|
|
|
|
// debugger (i.e. the Command Interpreter) or not. Return FALSE if the arguments were invalid OR
|
|
|
|
// if the user only wanted help or version information.
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
SBError
|
2010-06-23 09:19:29 +08:00
|
|
|
Driver::ParseArgs (int argc, const char *argv[], FILE *out_fh, bool &exit)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
ResetOptionValues ();
|
|
|
|
|
|
|
|
SBCommandReturnObject result;
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
SBError error;
|
|
|
|
std::string option_string;
|
|
|
|
struct option *long_options = NULL;
|
2010-10-13 05:57:09 +08:00
|
|
|
std::vector<struct option> long_options_vector;
|
2010-07-10 04:39:50 +08:00
|
|
|
uint32_t num_options;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-07-10 04:39:50 +08:00
|
|
|
for (num_options = 0; g_options[num_options].long_option != NULL; ++num_options)
|
|
|
|
/* Do Nothing. */;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
if (num_options == 0)
|
|
|
|
{
|
|
|
|
if (argc > 1)
|
|
|
|
error.SetErrorStringWithFormat ("invalid number of options");
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2010-10-13 05:57:09 +08:00
|
|
|
BuildGetOptTable (g_options, long_options_vector, num_options);
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-10-13 05:57:09 +08:00
|
|
|
if (long_options_vector.empty())
|
|
|
|
long_options = NULL;
|
|
|
|
else
|
|
|
|
long_options = &long_options_vector.front();
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
if (long_options == NULL)
|
|
|
|
{
|
|
|
|
error.SetErrorStringWithFormat ("invalid long options");
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build the option_string argument for call to getopt_long.
|
|
|
|
|
|
|
|
for (int i = 0; long_options[i].name != NULL; ++i)
|
|
|
|
{
|
|
|
|
if (long_options[i].flag == NULL)
|
|
|
|
{
|
|
|
|
option_string.push_back ((char) long_options[i].val);
|
|
|
|
switch (long_options[i].has_arg)
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
case no_argument:
|
|
|
|
break;
|
|
|
|
case required_argument:
|
|
|
|
option_string.push_back (':');
|
|
|
|
break;
|
|
|
|
case optional_argument:
|
|
|
|
option_string.append ("::");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-13 08:22:20 +08:00
|
|
|
// This is kind of a pain, but since we make the debugger in the Driver's constructor, we can't
|
|
|
|
// know at that point whether we should read in init files yet. So we don't read them in in the
|
|
|
|
// Driver constructor, then set the flags back to "read them in" here, and then if we see the
|
|
|
|
// "-n" flag, we'll turn it off again. Finally we have to read them in by hand later in the
|
|
|
|
// main loop.
|
|
|
|
|
|
|
|
m_debugger.SkipLLDBInitFiles (false);
|
|
|
|
m_debugger.SkipAppInitFiles (false);
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
// Prepare for & make calls to getopt_long.
|
2010-06-14 03:18:49 +08:00
|
|
|
#if __GLIBC__
|
|
|
|
optind = 0;
|
|
|
|
#else
|
2010-06-09 00:52:24 +08:00
|
|
|
optreset = 1;
|
|
|
|
optind = 1;
|
2010-06-14 03:18:49 +08:00
|
|
|
#endif
|
2010-06-09 00:52:24 +08:00
|
|
|
int val;
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
int long_options_index = -1;
|
2010-07-10 04:39:50 +08:00
|
|
|
val = ::getopt_long (argc, const_cast<char **>(argv), option_string.c_str(), long_options, &long_options_index);
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
if (val == -1)
|
|
|
|
break;
|
|
|
|
else if (val == '?')
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
m_option_data.m_print_help = true;
|
2010-06-09 00:52:24 +08:00
|
|
|
error.SetErrorStringWithFormat ("unknown or ambiguous option");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if (val == 0)
|
|
|
|
continue;
|
|
|
|
else
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
m_option_data.m_seen_options.insert ((char) val);
|
2010-06-09 00:52:24 +08:00
|
|
|
if (long_options_index == -1)
|
|
|
|
{
|
|
|
|
for (int i = 0;
|
|
|
|
long_options[i].name || long_options[i].has_arg || long_options[i].flag || long_options[i].val;
|
|
|
|
++i)
|
|
|
|
{
|
|
|
|
if (long_options[i].val == val)
|
|
|
|
{
|
|
|
|
long_options_index = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (long_options_index >= 0)
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
const char short_option = (char) g_options[long_options_index].short_option;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
switch (short_option)
|
|
|
|
{
|
|
|
|
case 'h':
|
|
|
|
m_option_data.m_print_help = true;
|
|
|
|
break;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
case 'v':
|
|
|
|
m_option_data.m_print_version = true;
|
|
|
|
break;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
case 'c':
|
|
|
|
m_option_data.m_crash_log = optarg;
|
|
|
|
break;
|
2010-10-11 09:05:37 +08:00
|
|
|
|
2010-08-31 03:44:40 +08:00
|
|
|
case 'e':
|
|
|
|
m_option_data.m_use_external_editor = true;
|
|
|
|
break;
|
2010-10-11 09:05:37 +08:00
|
|
|
|
2011-09-14 07:25:31 +08:00
|
|
|
case 'x':
|
2010-10-11 09:05:37 +08:00
|
|
|
m_debugger.SkipLLDBInitFiles (true);
|
2011-08-13 08:22:20 +08:00
|
|
|
m_debugger.SkipAppInitFiles (true);
|
2010-10-11 09:05:37 +08:00
|
|
|
break;
|
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
case 'f':
|
|
|
|
{
|
|
|
|
SBFileSpec file(optarg);
|
|
|
|
if (file.Exists())
|
2010-12-09 06:23:24 +08:00
|
|
|
{
|
|
|
|
m_option_data.m_args.push_back (optarg);
|
|
|
|
}
|
2010-09-10 12:48:55 +08:00
|
|
|
else if (file.ResolveExecutableLocation())
|
|
|
|
{
|
|
|
|
char path[PATH_MAX];
|
2011-08-11 06:06:24 +08:00
|
|
|
file.GetPath (path, sizeof(path));
|
2010-12-09 06:23:24 +08:00
|
|
|
m_option_data.m_args.push_back (path);
|
2010-09-10 12:48:55 +08:00
|
|
|
}
|
2010-06-23 09:19:29 +08:00
|
|
|
else
|
|
|
|
error.SetErrorStringWithFormat("file specified in --file (-f) option doesn't exist: '%s'", optarg);
|
|
|
|
}
|
|
|
|
break;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
case 'a':
|
|
|
|
if (!m_debugger.SetDefaultArchitecture (optarg))
|
|
|
|
error.SetErrorStringWithFormat("invalid architecture in the -a or --arch option: '%s'", optarg);
|
|
|
|
break;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
case 'l':
|
|
|
|
m_option_data.m_script_lang = m_debugger.GetScriptingLanguage (optarg);
|
|
|
|
break;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
case 'd':
|
|
|
|
m_option_data.m_debug_mode = true;
|
|
|
|
break;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2011-09-14 07:25:31 +08:00
|
|
|
case 'n':
|
|
|
|
m_option_data.m_process_name = optarg;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'w':
|
|
|
|
m_option_data.m_wait_for = true;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'p':
|
|
|
|
{
|
|
|
|
char *remainder;
|
|
|
|
m_option_data.m_process_pid = strtol (optarg, &remainder, 0);
|
|
|
|
if (remainder == optarg || *remainder != '\0')
|
|
|
|
error.SetErrorStringWithFormat ("Could not convert process PID: \"%s\" into a pid.",
|
|
|
|
optarg);
|
|
|
|
}
|
|
|
|
break;
|
2010-06-23 09:19:29 +08:00
|
|
|
case 's':
|
|
|
|
{
|
|
|
|
SBFileSpec file(optarg);
|
|
|
|
if (file.Exists())
|
|
|
|
m_option_data.m_source_command_files.push_back (optarg);
|
2010-09-10 12:48:55 +08:00
|
|
|
else if (file.ResolveExecutableLocation())
|
|
|
|
{
|
|
|
|
char final_path[PATH_MAX];
|
2011-08-11 06:06:24 +08:00
|
|
|
file.GetPath (final_path, sizeof(final_path));
|
2010-09-10 12:48:55 +08:00
|
|
|
std::string path_str (final_path);
|
|
|
|
m_option_data.m_source_command_files.push_back (path_str);
|
|
|
|
}
|
2010-06-23 09:19:29 +08:00
|
|
|
else
|
|
|
|
error.SetErrorStringWithFormat("file specified in --source (-s) option doesn't exist: '%s'", optarg);
|
|
|
|
}
|
|
|
|
break;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
default:
|
|
|
|
m_option_data.m_print_help = true;
|
|
|
|
error.SetErrorStringWithFormat ("unrecognized option %c", short_option);
|
|
|
|
break;
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2010-06-23 09:19:29 +08:00
|
|
|
else
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
error.SetErrorStringWithFormat ("invalid option with value %i", val);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2010-06-23 09:19:29 +08:00
|
|
|
if (error.Fail())
|
2010-10-13 05:57:09 +08:00
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
return error;
|
2010-10-13 05:57:09 +08:00
|
|
|
}
|
2010-06-23 09:19:29 +08:00
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2010-06-16 02:47:14 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
if (error.Fail() || m_option_data.m_print_help)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
ShowUsage (out_fh, g_options, m_option_data);
|
2010-10-11 09:13:37 +08:00
|
|
|
exit = true;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
else if (m_option_data.m_print_version)
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
::fprintf (out_fh, "%s\n", m_debugger.GetVersionString());
|
|
|
|
exit = true;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
else if (! m_option_data.m_crash_log.empty())
|
|
|
|
{
|
|
|
|
// Handle crash log stuff here.
|
|
|
|
}
|
2011-09-14 07:25:31 +08:00
|
|
|
else if (m_option_data.m_process_name.empty() && m_option_data.m_process_pid == LLDB_INVALID_PROCESS_ID)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-12-09 06:23:24 +08:00
|
|
|
// Any arguments that are left over after option parsing are for
|
|
|
|
// the program. If a file was specified with -f then the filename
|
|
|
|
// is already in the m_option_data.m_args array, and any remaining args
|
|
|
|
// are arguments for the inferior program. If no file was specified with
|
|
|
|
// -f, then what is left is the program name followed by any arguments.
|
|
|
|
|
|
|
|
// Skip any options we consumed with getopt_long
|
|
|
|
argc -= optind;
|
|
|
|
argv += optind;
|
|
|
|
|
|
|
|
if (argc > 0)
|
|
|
|
{
|
|
|
|
for (int arg_idx=0; arg_idx<argc; ++arg_idx)
|
|
|
|
{
|
|
|
|
const char *arg = argv[arg_idx];
|
|
|
|
if (arg)
|
|
|
|
m_option_data.m_args.push_back (arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2011-09-14 07:25:31 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// Skip any options we consumed with getopt_long
|
|
|
|
argc -= optind;
|
|
|
|
argv += optind;
|
|
|
|
|
|
|
|
if (argc > 0)
|
|
|
|
::fprintf (out_fh, "Warning: program arguments are ignored when attaching.\n");
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
return error;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2010-09-30 02:35:42 +08:00
|
|
|
size_t
|
2010-06-09 00:52:24 +08:00
|
|
|
Driver::GetProcessSTDOUT ()
|
|
|
|
{
|
|
|
|
// The process has stuff waiting for stdout; get it and write it out to the appropriate place.
|
|
|
|
char stdio_buffer[1024];
|
|
|
|
size_t len;
|
2010-09-30 02:35:42 +08:00
|
|
|
size_t total_bytes = 0;
|
2010-08-27 05:32:51 +08:00
|
|
|
while ((len = m_debugger.GetSelectedTarget().GetProcess().GetSTDOUT (stdio_buffer, sizeof (stdio_buffer))) > 0)
|
2010-09-30 02:35:42 +08:00
|
|
|
{
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
m_io_channel_ap->OutWrite (stdio_buffer, len, ASYNC);
|
2010-09-30 02:35:42 +08:00
|
|
|
total_bytes += len;
|
|
|
|
}
|
|
|
|
return total_bytes;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2010-09-30 02:35:42 +08:00
|
|
|
size_t
|
2010-06-09 00:52:24 +08:00
|
|
|
Driver::GetProcessSTDERR ()
|
|
|
|
{
|
|
|
|
// The process has stuff waiting for stderr; get it and write it out to the appropriate place.
|
|
|
|
char stdio_buffer[1024];
|
|
|
|
size_t len;
|
2010-09-30 02:35:42 +08:00
|
|
|
size_t total_bytes = 0;
|
2010-08-27 05:32:51 +08:00
|
|
|
while ((len = m_debugger.GetSelectedTarget().GetProcess().GetSTDERR (stdio_buffer, sizeof (stdio_buffer))) > 0)
|
2010-09-30 02:35:42 +08:00
|
|
|
{
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
m_io_channel_ap->ErrWrite (stdio_buffer, len, ASYNC);
|
2010-09-30 02:35:42 +08:00
|
|
|
total_bytes += len;
|
|
|
|
}
|
|
|
|
return total_bytes;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2010-08-27 05:32:51 +08:00
|
|
|
Driver::UpdateSelectedThread ()
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
using namespace lldb;
|
2010-08-27 05:32:51 +08:00
|
|
|
SBProcess process(m_debugger.GetSelectedTarget().GetProcess());
|
2010-06-09 00:52:24 +08:00
|
|
|
if (process.IsValid())
|
|
|
|
{
|
2010-08-27 05:32:51 +08:00
|
|
|
SBThread curr_thread (process.GetSelectedThread());
|
2010-06-09 00:52:24 +08:00
|
|
|
SBThread thread;
|
|
|
|
StopReason curr_thread_stop_reason = eStopReasonInvalid;
|
|
|
|
curr_thread_stop_reason = curr_thread.GetStopReason();
|
|
|
|
|
|
|
|
if (!curr_thread.IsValid() ||
|
|
|
|
curr_thread_stop_reason == eStopReasonInvalid ||
|
|
|
|
curr_thread_stop_reason == eStopReasonNone)
|
|
|
|
{
|
|
|
|
// Prefer a thread that has just completed its plan over another thread as current thread.
|
|
|
|
SBThread plan_thread;
|
|
|
|
SBThread other_thread;
|
|
|
|
const size_t num_threads = process.GetNumThreads();
|
|
|
|
size_t i;
|
|
|
|
for (i = 0; i < num_threads; ++i)
|
|
|
|
{
|
|
|
|
thread = process.GetThreadAtIndex(i);
|
|
|
|
StopReason thread_stop_reason = thread.GetStopReason();
|
|
|
|
switch (thread_stop_reason)
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
case eStopReasonInvalid:
|
|
|
|
case eStopReasonNone:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eStopReasonTrace:
|
|
|
|
case eStopReasonBreakpoint:
|
|
|
|
case eStopReasonWatchpoint:
|
|
|
|
case eStopReasonSignal:
|
|
|
|
case eStopReasonException:
|
|
|
|
if (!other_thread.IsValid())
|
|
|
|
other_thread = thread;
|
|
|
|
break;
|
|
|
|
case eStopReasonPlanComplete:
|
|
|
|
if (!plan_thread.IsValid())
|
|
|
|
plan_thread = thread;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (plan_thread.IsValid())
|
2010-08-27 05:32:51 +08:00
|
|
|
process.SetSelectedThread (plan_thread);
|
2010-06-09 00:52:24 +08:00
|
|
|
else if (other_thread.IsValid())
|
2010-08-27 05:32:51 +08:00
|
|
|
process.SetSelectedThread (other_thread);
|
2010-06-09 00:52:24 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if (curr_thread.IsValid())
|
|
|
|
thread = curr_thread;
|
|
|
|
else
|
|
|
|
thread = process.GetThreadAtIndex(0);
|
|
|
|
|
|
|
|
if (thread.IsValid())
|
2010-08-27 05:32:51 +08:00
|
|
|
process.SetSelectedThread (thread);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-08 13:23:15 +08:00
|
|
|
// This function handles events that were broadcast by the process.
|
|
|
|
void
|
|
|
|
Driver::HandleBreakpointEvent (const SBEvent &event)
|
|
|
|
{
|
|
|
|
using namespace lldb;
|
|
|
|
const uint32_t event_type = SBBreakpoint::GetBreakpointEventTypeFromEvent (event);
|
|
|
|
|
|
|
|
if (event_type & eBreakpointEventTypeAdded
|
|
|
|
|| event_type & eBreakpointEventTypeRemoved
|
|
|
|
|| event_type & eBreakpointEventTypeEnabled
|
|
|
|
|| event_type & eBreakpointEventTypeDisabled
|
|
|
|
|| event_type & eBreakpointEventTypeCommandChanged
|
|
|
|
|| event_type & eBreakpointEventTypeConditionChanged
|
|
|
|
|| event_type & eBreakpointEventTypeIgnoreChanged
|
|
|
|
|| event_type & eBreakpointEventTypeLocationsResolved)
|
|
|
|
{
|
|
|
|
// Don't do anything about these events, since the breakpoint commands already echo these actions.
|
|
|
|
}
|
|
|
|
else if (event_type & eBreakpointEventTypeLocationsAdded)
|
|
|
|
{
|
|
|
|
char message[256];
|
|
|
|
uint32_t num_new_locations = SBBreakpoint::GetNumBreakpointLocationsFromEvent(event);
|
|
|
|
if (num_new_locations > 0)
|
|
|
|
{
|
|
|
|
SBBreakpoint breakpoint = SBBreakpoint::GetBreakpointFromEvent(event);
|
|
|
|
int message_len = ::snprintf (message, sizeof(message), "%d locations added to breakpoint %d\n",
|
|
|
|
num_new_locations,
|
|
|
|
breakpoint.GetID());
|
|
|
|
m_io_channel_ap->OutWrite(message, message_len, ASYNC);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (event_type & eBreakpointEventTypeLocationsRemoved)
|
|
|
|
{
|
|
|
|
// These locations just get disabled, not sure it is worth spamming folks about this on the command line.
|
|
|
|
}
|
|
|
|
else if (event_type & eBreakpointEventTypeLocationsResolved)
|
|
|
|
{
|
|
|
|
// This might be an interesting thing to note, but I'm going to leave it quiet for now, it just looked noisy.
|
|
|
|
}
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
// This function handles events that were broadcast by the process.
|
|
|
|
void
|
|
|
|
Driver::HandleProcessEvent (const SBEvent &event)
|
|
|
|
{
|
|
|
|
using namespace lldb;
|
|
|
|
const uint32_t event_type = event.GetType();
|
|
|
|
|
|
|
|
if (event_type & SBProcess::eBroadcastBitSTDOUT)
|
|
|
|
{
|
|
|
|
// The process has stdout available, get it and write it out to the
|
|
|
|
// appropriate place.
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
GetProcessSTDOUT ();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
else if (event_type & SBProcess::eBroadcastBitSTDERR)
|
|
|
|
{
|
|
|
|
// The process has stderr available, get it and write it out to the
|
|
|
|
// appropriate place.
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
GetProcessSTDERR ();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
else if (event_type & SBProcess::eBroadcastBitStateChanged)
|
|
|
|
{
|
|
|
|
// Drain all stout and stderr so we don't see any output come after
|
|
|
|
// we print our prompts
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
GetProcessSTDOUT ();
|
|
|
|
GetProcessSTDERR ();
|
2010-06-09 00:52:24 +08:00
|
|
|
// Something changed in the process; get the event and report the process's current status and location to
|
|
|
|
// the user.
|
|
|
|
StateType event_state = SBProcess::GetStateFromEvent (event);
|
|
|
|
if (event_state == eStateInvalid)
|
|
|
|
return;
|
|
|
|
|
|
|
|
SBProcess process (SBProcess::GetProcessFromEvent (event));
|
|
|
|
assert (process.IsValid());
|
|
|
|
|
|
|
|
switch (event_state)
|
|
|
|
{
|
|
|
|
case eStateInvalid:
|
|
|
|
case eStateUnloaded:
|
2011-02-04 09:58:07 +08:00
|
|
|
case eStateConnected:
|
2010-06-09 00:52:24 +08:00
|
|
|
case eStateAttaching:
|
|
|
|
case eStateLaunching:
|
|
|
|
case eStateStepping:
|
|
|
|
case eStateDetached:
|
|
|
|
{
|
|
|
|
char message[1024];
|
2011-12-02 07:28:38 +08:00
|
|
|
int message_len = ::snprintf (message, sizeof(message), "Process %llu %s\n", process.GetProcessID(),
|
2010-06-23 09:19:29 +08:00
|
|
|
m_debugger.StateAsCString (event_state));
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
m_io_channel_ap->OutWrite(message, message_len, ASYNC);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eStateRunning:
|
|
|
|
// Don't be chatty when we run...
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eStateExited:
|
2010-09-30 02:35:42 +08:00
|
|
|
{
|
|
|
|
SBCommandReturnObject result;
|
|
|
|
m_debugger.GetCommandInterpreter().HandleCommand("process status", result, false);
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
m_io_channel_ap->ErrWrite (result.GetError(), result.GetErrorSize(), ASYNC);
|
|
|
|
m_io_channel_ap->OutWrite (result.GetOutput(), result.GetOutputSize(), ASYNC);
|
2010-09-30 02:35:42 +08:00
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case eStateStopped:
|
|
|
|
case eStateCrashed:
|
|
|
|
case eStateSuspended:
|
|
|
|
// Make sure the program hasn't been auto-restarted:
|
|
|
|
if (SBProcess::GetRestartedFromEvent (event))
|
|
|
|
{
|
|
|
|
// FIXME: Do we want to report this, or would that just be annoyingly chatty?
|
|
|
|
char message[1024];
|
2011-12-02 07:28:38 +08:00
|
|
|
int message_len = ::snprintf (message, sizeof(message), "Process %llu stopped and was programmatically restarted.\n",
|
2010-06-09 00:52:24 +08:00
|
|
|
process.GetProcessID());
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
m_io_channel_ap->OutWrite(message, message_len, ASYNC);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-09-30 02:35:42 +08:00
|
|
|
SBCommandReturnObject result;
|
2010-08-27 05:32:51 +08:00
|
|
|
UpdateSelectedThread ();
|
2010-09-30 02:35:42 +08:00
|
|
|
m_debugger.GetCommandInterpreter().HandleCommand("process status", result, false);
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
m_io_channel_ap->ErrWrite (result.GetError(), result.GetErrorSize(), ASYNC);
|
|
|
|
m_io_channel_ap->OutWrite (result.GetOutput(), result.GetOutputSize(), ASYNC);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// This function handles events broadcast by the IOChannel (HasInput, UserInterrupt, or ThreadShouldExit).
|
|
|
|
|
|
|
|
bool
|
|
|
|
Driver::HandleIOEvent (const SBEvent &event)
|
|
|
|
{
|
|
|
|
bool quit = false;
|
|
|
|
|
|
|
|
const uint32_t event_type = event.GetType();
|
|
|
|
|
|
|
|
if (event_type & IOChannel::eBroadcastBitHasUserInput)
|
|
|
|
{
|
|
|
|
// We got some input (i.e. a command string) from the user; pass it off to the command interpreter for
|
|
|
|
// handling.
|
|
|
|
|
|
|
|
const char *command_string = SBEvent::GetCStringFromEvent(event);
|
|
|
|
if (command_string == NULL)
|
2010-07-10 04:39:50 +08:00
|
|
|
command_string = "";
|
2010-06-09 00:52:24 +08:00
|
|
|
SBCommandReturnObject result;
|
2011-02-19 10:53:09 +08:00
|
|
|
|
2011-05-17 03:20:50 +08:00
|
|
|
// We don't want the result to bypass the OutWrite function in IOChannel, as this can result in odd
|
|
|
|
// output orderings and problems with the prompt.
|
2011-02-19 10:53:09 +08:00
|
|
|
m_debugger.GetCommandInterpreter().HandleCommand (command_string, result, true);
|
|
|
|
|
2011-05-17 03:20:50 +08:00
|
|
|
if (result.GetOutputSize() > 0)
|
|
|
|
m_io_channel_ap->OutWrite (result.GetOutput(), result.GetOutputSize(), NO_ASYNC);
|
|
|
|
|
|
|
|
if (result.GetErrorSize() > 0)
|
|
|
|
m_io_channel_ap->OutWrite (result.GetError(), result.GetErrorSize(), NO_ASYNC);
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
// We are done getting and running our command, we can now clear the
|
|
|
|
// m_waiting_for_command so we can get another one.
|
|
|
|
m_waiting_for_command = false;
|
|
|
|
|
|
|
|
// If our editline input reader is active, it means another input reader
|
|
|
|
// got pushed onto the input reader and caused us to become deactivated.
|
|
|
|
// When the input reader above us gets popped, we will get re-activated
|
|
|
|
// and our prompt will refresh in our callback
|
|
|
|
if (m_editline_reader.IsActive())
|
|
|
|
{
|
|
|
|
ReadyForCommand ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (event_type & IOChannel::eBroadcastBitUserInterrupt)
|
|
|
|
{
|
|
|
|
// This is here to handle control-c interrupts from the user. It has not yet really been implemented.
|
|
|
|
// TO BE DONE: PROPERLY HANDLE CONTROL-C FROM USER
|
|
|
|
//m_io_channel_ap->CancelInput();
|
|
|
|
// Anything else? Send Interrupt to process?
|
|
|
|
}
|
|
|
|
else if ((event_type & IOChannel::eBroadcastBitThreadShouldExit) ||
|
|
|
|
(event_type & IOChannel::eBroadcastBitThreadDidExit))
|
|
|
|
{
|
|
|
|
// If the IOChannel thread is trying to go away, then it is definitely
|
|
|
|
// time to end the debugging session.
|
|
|
|
quit = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return quit;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Driver::MasterThreadBytesReceived (void *baton, const void *src, size_t src_len)
|
|
|
|
{
|
|
|
|
Driver *driver = (Driver*)baton;
|
|
|
|
driver->GetFromMaster ((const char *)src, src_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Driver::GetFromMaster (const char *src, size_t src_len)
|
|
|
|
{
|
|
|
|
// Echo the characters back to the Debugger's stdout, that way if you
|
|
|
|
// type characters while a command is running, you'll see what you've typed.
|
2010-06-23 09:19:29 +08:00
|
|
|
FILE *out_fh = m_debugger.GetOutputFileHandle();
|
2010-06-09 00:52:24 +08:00
|
|
|
if (out_fh)
|
|
|
|
::fwrite (src, 1, src_len, out_fh);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t
|
|
|
|
Driver::EditLineInputReaderCallback
|
|
|
|
(
|
|
|
|
void *baton,
|
|
|
|
SBInputReader *reader,
|
|
|
|
InputReaderAction notification,
|
|
|
|
const char *bytes,
|
|
|
|
size_t bytes_len
|
|
|
|
)
|
|
|
|
{
|
|
|
|
Driver *driver = (Driver *)baton;
|
|
|
|
|
|
|
|
switch (notification)
|
|
|
|
{
|
|
|
|
case eInputReaderActivate:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eInputReaderReactivate:
|
|
|
|
driver->ReadyForCommand();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eInputReaderDeactivate:
|
|
|
|
break;
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
|
|
|
|
case eInputReaderAsynchronousOutputWritten:
|
|
|
|
if (driver->m_io_channel_ap.get() != NULL)
|
|
|
|
driver->m_io_channel_ap->RefreshPrompt();
|
|
|
|
break;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-11-20 04:47:54 +08:00
|
|
|
case eInputReaderInterrupt:
|
|
|
|
if (driver->m_io_channel_ap.get() != NULL)
|
|
|
|
{
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
driver->m_io_channel_ap->OutWrite ("^C\n", 3, NO_ASYNC);
|
2010-11-20 04:47:54 +08:00
|
|
|
driver->m_io_channel_ap->RefreshPrompt();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eInputReaderEndOfFile:
|
|
|
|
if (driver->m_io_channel_ap.get() != NULL)
|
|
|
|
{
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
driver->m_io_channel_ap->OutWrite ("^D\n", 3, NO_ASYNC);
|
2010-11-20 04:47:54 +08:00
|
|
|
driver->m_io_channel_ap->RefreshPrompt ();
|
|
|
|
}
|
|
|
|
write (driver->m_editline_pty.GetMasterFileDescriptor(), "quit\n", 5);
|
|
|
|
break;
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
case eInputReaderGotToken:
|
|
|
|
write (driver->m_editline_pty.GetMasterFileDescriptor(), bytes, bytes_len);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eInputReaderDone:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return bytes_len;
|
|
|
|
}
|
|
|
|
|
2012-02-29 12:21:24 +08:00
|
|
|
// Intercept when the quit command is called and tell our driver that it is done
|
|
|
|
static bool
|
|
|
|
QuitCommandOverrideCallback (void *baton, const char **argv)
|
|
|
|
{
|
|
|
|
((Driver *)baton)->SetIsDone();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
void
|
|
|
|
Driver::MainLoop ()
|
|
|
|
{
|
|
|
|
char error_str[1024];
|
|
|
|
if (m_editline_pty.OpenFirstAvailableMaster(O_RDWR|O_NOCTTY, error_str, sizeof(error_str)) == false)
|
|
|
|
{
|
|
|
|
::fprintf (stderr, "error: failed to open driver pseudo terminal : %s", error_str);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const char *driver_slave_name = m_editline_pty.GetSlaveName (error_str, sizeof(error_str));
|
|
|
|
if (driver_slave_name == NULL)
|
|
|
|
{
|
|
|
|
::fprintf (stderr, "error: failed to get slave name for driver pseudo terminal : %s", error_str);
|
|
|
|
exit(2);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_editline_slave_fh = ::fopen (driver_slave_name, "r+");
|
|
|
|
if (m_editline_slave_fh == NULL)
|
|
|
|
{
|
|
|
|
SBError error;
|
|
|
|
error.SetErrorToErrno();
|
|
|
|
::fprintf (stderr, "error: failed to get open slave for driver pseudo terminal : %s",
|
|
|
|
error.GetCString());
|
|
|
|
exit(3);
|
|
|
|
}
|
|
|
|
|
|
|
|
::setbuf (m_editline_slave_fh, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
lldb_utility::PseudoTerminal editline_output_pty;
|
|
|
|
FILE *editline_output_slave_fh = NULL;
|
|
|
|
|
|
|
|
if (editline_output_pty.OpenFirstAvailableMaster (O_RDWR|O_NOCTTY, error_str, sizeof (error_str)) == false)
|
|
|
|
{
|
|
|
|
::fprintf (stderr, "error: failed to open output pseudo terminal : %s", error_str);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const char *output_slave_name = editline_output_pty.GetSlaveName (error_str, sizeof(error_str));
|
|
|
|
if (output_slave_name == NULL)
|
|
|
|
{
|
|
|
|
::fprintf (stderr, "error: failed to get slave name for output pseudo terminal : %s", error_str);
|
|
|
|
exit(2);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
editline_output_slave_fh = ::fopen (output_slave_name, "r+");
|
|
|
|
if (editline_output_slave_fh == NULL)
|
|
|
|
{
|
|
|
|
SBError error;
|
|
|
|
error.SetErrorToErrno();
|
|
|
|
::fprintf (stderr, "error: failed to get open slave for output pseudo terminal : %s",
|
|
|
|
error.GetCString());
|
|
|
|
exit(3);
|
|
|
|
}
|
|
|
|
::setbuf (editline_output_slave_fh, NULL);
|
|
|
|
}
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
// struct termios stdin_termios;
|
|
|
|
|
|
|
|
if (::tcgetattr(STDIN_FILENO, &g_old_stdin_termios) == 0)
|
2012-02-03 03:28:31 +08:00
|
|
|
{
|
|
|
|
g_old_stdin_termios_is_valid = true;
|
2010-06-09 00:52:24 +08:00
|
|
|
atexit (reset_stdin_termios);
|
2012-02-03 03:28:31 +08:00
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
::setbuf (stdin, NULL);
|
|
|
|
::setbuf (stdout, NULL);
|
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
m_debugger.SetErrorFileHandle (stderr, false);
|
|
|
|
m_debugger.SetOutputFileHandle (stdout, false);
|
|
|
|
m_debugger.SetInputFileHandle (stdin, true);
|
2010-08-31 03:44:40 +08:00
|
|
|
|
|
|
|
m_debugger.SetUseExternalEditor(m_option_data.m_use_external_editor);
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
// You have to drain anything that comes to the master side of the PTY. master_out_comm is
|
|
|
|
// for that purpose. The reason you need to do this is a curious reason... editline will echo
|
|
|
|
// characters to the PTY when it gets characters while el_gets is not running, and then when
|
|
|
|
// you call el_gets (or el_getc) it will try to reset the terminal back to raw mode which blocks
|
|
|
|
// if there are unconsumed characters in the out buffer.
|
|
|
|
// However, you don't need to do anything with the characters, since editline will dump these
|
|
|
|
// unconsumed characters after printing the prompt again in el_gets.
|
|
|
|
|
2010-12-04 10:39:47 +08:00
|
|
|
SBCommunication master_out_comm("driver.editline");
|
|
|
|
master_out_comm.SetCloseOnEOF (false);
|
2010-06-09 00:52:24 +08:00
|
|
|
master_out_comm.AdoptFileDesriptor(m_editline_pty.GetMasterFileDescriptor(), false);
|
|
|
|
master_out_comm.SetReadThreadBytesReceivedCallback(Driver::MasterThreadBytesReceived, this);
|
|
|
|
|
|
|
|
if (master_out_comm.ReadThreadStart () == false)
|
|
|
|
{
|
|
|
|
::fprintf (stderr, "error: failed to start master out read thread");
|
|
|
|
exit(5);
|
|
|
|
}
|
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
SBCommandInterpreter sb_interpreter = m_debugger.GetCommandInterpreter();
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2012-02-29 12:21:24 +08:00
|
|
|
// Intercept when the quit command is called and tell our driver that it is done
|
|
|
|
bool quit_success = sb_interpreter.SetCommandOverrideCallback ("quit", QuitCommandOverrideCallback, this);
|
|
|
|
assert (quit_success);
|
|
|
|
|
This patch captures and serializes all output being written by the
command line driver, including the lldb prompt being output by
editline, the asynchronous process output & error messages, and
asynchronous messages written by target stop-hooks.
As part of this it introduces a new Stream class,
StreamAsynchronousIO. A StreamAsynchronousIO object is created with a
broadcaster, who will eventually broadcast the stream's data for a
listener to handle, and an event type indicating what type of event
the broadcaster will broadcast. When the Write method is called on a
StreamAsynchronousIO object, the data is appended to an internal
string. When the Flush method is called on a StreamAsynchronousIO
object, it broadcasts it's data string and clears the string.
Anything in lldb-core that needs to generate asynchronous output for
the end-user should use the StreamAsynchronousIO objects.
I have also added a new notification type for InputReaders, to let
them know that a asynchronous output has been written. This is to
allow the input readers to, for example, refresh their prompts and
lines, if desired. I added the case statements to all the input
readers to catch this notification, but I haven't added any code for
handling them yet (except to the IOChannel input reader).
llvm-svn: 130721
2011-05-03 04:41:46 +08:00
|
|
|
m_io_channel_ap.reset (new IOChannel(m_editline_slave_fh, editline_output_slave_fh, stdout, stderr, this));
|
|
|
|
|
|
|
|
SBCommunication out_comm_2("driver.editline_output");
|
|
|
|
out_comm_2.SetCloseOnEOF (false);
|
|
|
|
out_comm_2.AdoptFileDesriptor (editline_output_pty.GetMasterFileDescriptor(), false);
|
|
|
|
out_comm_2.SetReadThreadBytesReceivedCallback (IOChannel::LibeditOutputBytesReceived, m_io_channel_ap.get());
|
|
|
|
|
|
|
|
if (out_comm_2.ReadThreadStart () == false)
|
|
|
|
{
|
|
|
|
::fprintf (stderr, "error: failed to start libedit output read thread");
|
|
|
|
exit (5);
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
struct winsize window_size;
|
|
|
|
if (isatty (STDIN_FILENO)
|
|
|
|
&& ::ioctl (STDIN_FILENO, TIOCGWINSZ, &window_size) == 0)
|
|
|
|
{
|
2010-09-04 08:03:46 +08:00
|
|
|
if (window_size.ws_col > 0)
|
2010-09-18 09:14:36 +08:00
|
|
|
m_debugger.SetTerminalWidth (window_size.ws_col);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Since input can be redirected by the debugger, we must insert our editline
|
|
|
|
// input reader in the queue so we know when our reader should be active
|
|
|
|
// and so we can receive bytes only when we are supposed to.
|
2010-06-23 09:19:29 +08:00
|
|
|
SBError err (m_editline_reader.Initialize (m_debugger,
|
|
|
|
Driver::EditLineInputReaderCallback, // callback
|
2010-06-09 00:52:24 +08:00
|
|
|
this, // baton
|
|
|
|
eInputReaderGranularityByte, // token_size
|
|
|
|
NULL, // end token - NULL means never done
|
|
|
|
NULL, // prompt - taken care of elsewhere
|
|
|
|
false)); // echo input - don't need Debugger
|
|
|
|
// to do this, we handle it elsewhere
|
|
|
|
|
|
|
|
if (err.Fail())
|
|
|
|
{
|
|
|
|
::fprintf (stderr, "error: %s", err.GetCString());
|
|
|
|
exit (6);
|
|
|
|
}
|
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
m_debugger.PushInputReader (m_editline_reader);
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
SBListener listener(m_debugger.GetListener());
|
2012-02-16 14:50:00 +08:00
|
|
|
listener.StartListeningForEventClass(m_debugger,
|
|
|
|
SBTarget::GetBroadcasterClassName(),
|
|
|
|
SBTarget::eBroadcastBitBreakpointChanged);
|
2010-06-09 00:52:24 +08:00
|
|
|
if (listener.IsValid())
|
|
|
|
{
|
|
|
|
|
|
|
|
listener.StartListeningForEvents (*m_io_channel_ap,
|
|
|
|
IOChannel::eBroadcastBitHasUserInput |
|
|
|
|
IOChannel::eBroadcastBitUserInterrupt |
|
|
|
|
IOChannel::eBroadcastBitThreadShouldExit |
|
|
|
|
IOChannel::eBroadcastBitThreadDidStart |
|
|
|
|
IOChannel::eBroadcastBitThreadDidExit);
|
|
|
|
|
|
|
|
if (m_io_channel_ap->Start ())
|
|
|
|
{
|
|
|
|
bool iochannel_thread_exited = false;
|
|
|
|
|
|
|
|
listener.StartListeningForEvents (sb_interpreter.GetBroadcaster(),
|
2011-05-04 04:53:11 +08:00
|
|
|
SBCommandInterpreter::eBroadcastBitQuitCommandReceived |
|
|
|
|
SBCommandInterpreter::eBroadcastBitAsynchronousOutputData |
|
|
|
|
SBCommandInterpreter::eBroadcastBitAsynchronousErrorData);
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
// Before we handle any options from the command line, we parse the
|
|
|
|
// .lldbinit file in the user's home directory.
|
|
|
|
SBCommandReturnObject result;
|
|
|
|
sb_interpreter.SourceInitFileInHomeDirectory(result);
|
|
|
|
if (GetDebugMode())
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
result.PutError (m_debugger.GetErrorFileHandle());
|
|
|
|
result.PutOutput (m_debugger.GetOutputFileHandle());
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Now we handle options we got from the command line
|
|
|
|
char command_string[PATH_MAX * 2];
|
|
|
|
const size_t num_source_command_files = GetNumSourceCommandFiles();
|
|
|
|
if (num_source_command_files > 0)
|
|
|
|
{
|
|
|
|
for (size_t i=0; i < num_source_command_files; ++i)
|
|
|
|
{
|
|
|
|
const char *command_file = GetSourceCommandFileAtIndex(i);
|
2010-07-29 05:16:11 +08:00
|
|
|
::snprintf (command_string, sizeof(command_string), "command source '%s'", command_file);
|
2010-06-23 09:19:29 +08:00
|
|
|
m_debugger.GetCommandInterpreter().HandleCommand (command_string, result, false);
|
2010-06-09 00:52:24 +08:00
|
|
|
if (GetDebugMode())
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
result.PutError (m_debugger.GetErrorFileHandle());
|
|
|
|
result.PutOutput (m_debugger.GetOutputFileHandle());
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-12-09 06:23:24 +08:00
|
|
|
const size_t num_args = m_option_data.m_args.size();
|
|
|
|
if (num_args > 0)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
char arch_name[64];
|
2010-06-23 09:19:29 +08:00
|
|
|
if (m_debugger.GetDefaultArchitecture (arch_name, sizeof (arch_name)))
|
2010-12-09 06:23:24 +08:00
|
|
|
::snprintf (command_string,
|
|
|
|
sizeof (command_string),
|
Centralized a lot of the status information for processes,
threads, and stack frame down in the lldb_private::Process,
lldb_private::Thread, lldb_private::StackFrameList and the
lldb_private::StackFrame classes. We had some command line
commands that had duplicate versions of the process status
output ("thread list" and "process status" for example).
Removed the "file" command and placed it where it should
have been: "target create". Made an alias for "file" to
"target create" so we stay compatible with GDB commands.
We can now have multple usable targets in lldb at the
same time. This is nice for comparing two runs of a program
or debugging more than one binary at the same time. The
new command is "target select <target-idx>" and also to see
a list of the current targets you can use the new "target list"
command. The flow in a debug session can be:
(lldb) target create /path/to/exe/a.out
(lldb) breakpoint set --name main
(lldb) run
... hit breakpoint
(lldb) target create /bin/ls
(lldb) run /tmp
Process 36001 exited with status = 0 (0x00000000)
(lldb) target list
Current targets:
target #0: /tmp/args/a.out ( arch=x86_64-apple-darwin, platform=localhost, pid=35999, state=stopped )
* target #1: /bin/ls ( arch=x86_64-apple-darwin, platform=localhost, pid=36001, state=exited )
(lldb) target select 0
Current targets:
* target #0: /tmp/args/a.out ( arch=x86_64-apple-darwin, platform=localhost, pid=35999, state=stopped )
target #1: /bin/ls ( arch=x86_64-apple-darwin, platform=localhost, pid=36001, state=exited )
(lldb) bt
* thread #1: tid = 0x2d03, 0x0000000100000b9a a.out`main + 42 at main.c:16, stop reason = breakpoint 1.1
frame #0: 0x0000000100000b9a a.out`main + 42 at main.c:16
frame #1: 0x0000000100000b64 a.out`start + 52
Above we created a target for "a.out" and ran and hit a
breakpoint at "main". Then we created a new target for /bin/ls
and ran it. Then we listed the targest and selected our original
"a.out" program, so we showed two concurent debug sessions
going on at the same time.
llvm-svn: 129695
2011-04-18 16:33:37 +08:00
|
|
|
"target create --arch=%s '%s'",
|
2010-12-09 06:23:24 +08:00
|
|
|
arch_name,
|
|
|
|
m_option_data.m_args[0].c_str());
|
2010-06-09 00:52:24 +08:00
|
|
|
else
|
2010-12-09 06:23:24 +08:00
|
|
|
::snprintf (command_string,
|
|
|
|
sizeof(command_string),
|
Centralized a lot of the status information for processes,
threads, and stack frame down in the lldb_private::Process,
lldb_private::Thread, lldb_private::StackFrameList and the
lldb_private::StackFrame classes. We had some command line
commands that had duplicate versions of the process status
output ("thread list" and "process status" for example).
Removed the "file" command and placed it where it should
have been: "target create". Made an alias for "file" to
"target create" so we stay compatible with GDB commands.
We can now have multple usable targets in lldb at the
same time. This is nice for comparing two runs of a program
or debugging more than one binary at the same time. The
new command is "target select <target-idx>" and also to see
a list of the current targets you can use the new "target list"
command. The flow in a debug session can be:
(lldb) target create /path/to/exe/a.out
(lldb) breakpoint set --name main
(lldb) run
... hit breakpoint
(lldb) target create /bin/ls
(lldb) run /tmp
Process 36001 exited with status = 0 (0x00000000)
(lldb) target list
Current targets:
target #0: /tmp/args/a.out ( arch=x86_64-apple-darwin, platform=localhost, pid=35999, state=stopped )
* target #1: /bin/ls ( arch=x86_64-apple-darwin, platform=localhost, pid=36001, state=exited )
(lldb) target select 0
Current targets:
* target #0: /tmp/args/a.out ( arch=x86_64-apple-darwin, platform=localhost, pid=35999, state=stopped )
target #1: /bin/ls ( arch=x86_64-apple-darwin, platform=localhost, pid=36001, state=exited )
(lldb) bt
* thread #1: tid = 0x2d03, 0x0000000100000b9a a.out`main + 42 at main.c:16, stop reason = breakpoint 1.1
frame #0: 0x0000000100000b9a a.out`main + 42 at main.c:16
frame #1: 0x0000000100000b64 a.out`start + 52
Above we created a target for "a.out" and ran and hit a
breakpoint at "main". Then we created a new target for /bin/ls
and ran it. Then we listed the targest and selected our original
"a.out" program, so we showed two concurent debug sessions
going on at the same time.
llvm-svn: 129695
2011-04-18 16:33:37 +08:00
|
|
|
"target create '%s'",
|
2010-12-09 06:23:24 +08:00
|
|
|
m_option_data.m_args[0].c_str());
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
m_debugger.HandleCommand (command_string);
|
2010-12-09 06:23:24 +08:00
|
|
|
|
|
|
|
if (num_args > 1)
|
|
|
|
{
|
2011-11-08 10:43:13 +08:00
|
|
|
m_debugger.HandleCommand ("settings clear target.run-args");
|
2010-12-09 06:23:24 +08:00
|
|
|
char arg_cstr[1024];
|
|
|
|
for (size_t arg_idx = 1; arg_idx < num_args; ++arg_idx)
|
|
|
|
{
|
2011-09-14 07:25:31 +08:00
|
|
|
::snprintf (arg_cstr,
|
|
|
|
sizeof(arg_cstr),
|
2011-11-08 10:43:13 +08:00
|
|
|
"settings append target.run-args \"%s\"",
|
2011-09-14 07:25:31 +08:00
|
|
|
m_option_data.m_args[arg_idx].c_str());
|
2010-12-09 06:23:24 +08:00
|
|
|
m_debugger.HandleCommand (arg_cstr);
|
|
|
|
}
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Now that all option parsing is done, we try and parse the .lldbinit
|
|
|
|
// file in the current working directory
|
|
|
|
sb_interpreter.SourceInitFileInCurrentWorkingDirectory (result);
|
|
|
|
if (GetDebugMode())
|
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
result.PutError(m_debugger.GetErrorFileHandle());
|
|
|
|
result.PutOutput(m_debugger.GetOutputFileHandle());
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
SBEvent event;
|
|
|
|
|
|
|
|
// Make sure the IO channel is started up before we try to tell it we
|
|
|
|
// are ready for input
|
|
|
|
listener.WaitForEventForBroadcasterWithType (UINT32_MAX,
|
|
|
|
*m_io_channel_ap,
|
|
|
|
IOChannel::eBroadcastBitThreadDidStart,
|
|
|
|
event);
|
2011-09-14 07:25:31 +08:00
|
|
|
// If we were asked to attach, then do that here:
|
|
|
|
// I'm going to use the command string rather than directly
|
|
|
|
// calling the API's because then I don't have to recode the
|
|
|
|
// event handling here.
|
|
|
|
if (!m_option_data.m_process_name.empty()
|
|
|
|
|| m_option_data.m_process_pid != LLDB_INVALID_PROCESS_ID)
|
|
|
|
{
|
|
|
|
std::string command_str("process attach ");
|
|
|
|
if (m_option_data.m_process_pid != LLDB_INVALID_PROCESS_ID)
|
|
|
|
{
|
|
|
|
command_str.append("-p ");
|
|
|
|
char pid_buffer[32];
|
2011-12-02 07:28:38 +08:00
|
|
|
::snprintf (pid_buffer, sizeof(pid_buffer), "%llu", m_option_data.m_process_pid);
|
2011-09-14 07:25:31 +08:00
|
|
|
command_str.append(pid_buffer);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
command_str.append("-n \"");
|
|
|
|
command_str.append(m_option_data.m_process_name);
|
|
|
|
command_str.push_back('\"');
|
|
|
|
if (m_option_data.m_wait_for)
|
|
|
|
command_str.append(" -w");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_debugger.GetOutputFileHandle())
|
|
|
|
::fprintf (m_debugger.GetOutputFileHandle(),
|
|
|
|
"Attaching to process with:\n %s\n",
|
|
|
|
command_str.c_str());
|
|
|
|
|
|
|
|
// Force the attach to be synchronous:
|
|
|
|
bool orig_async = m_debugger.GetAsync();
|
|
|
|
m_debugger.SetAsync(true);
|
|
|
|
m_debugger.HandleCommand(command_str.c_str());
|
|
|
|
m_debugger.SetAsync(orig_async);
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
ReadyForCommand ();
|
|
|
|
|
2012-02-29 12:21:24 +08:00
|
|
|
while (!GetIsDone())
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
listener.WaitForEvent (UINT32_MAX, event);
|
|
|
|
if (event.IsValid())
|
|
|
|
{
|
|
|
|
if (event.GetBroadcaster().IsValid())
|
|
|
|
{
|
|
|
|
uint32_t event_type = event.GetType();
|
|
|
|
if (event.BroadcasterMatchesRef (*m_io_channel_ap))
|
|
|
|
{
|
|
|
|
if ((event_type & IOChannel::eBroadcastBitThreadShouldExit) ||
|
|
|
|
(event_type & IOChannel::eBroadcastBitThreadDidExit))
|
|
|
|
{
|
2012-02-29 12:21:24 +08:00
|
|
|
SetIsDone();
|
2010-06-09 00:52:24 +08:00
|
|
|
if (event_type & IOChannel::eBroadcastBitThreadDidExit)
|
|
|
|
iochannel_thread_exited = true;
|
|
|
|
}
|
|
|
|
else
|
2012-02-29 12:21:24 +08:00
|
|
|
{
|
|
|
|
if (HandleIOEvent (event))
|
|
|
|
SetIsDone();
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2012-02-08 13:23:15 +08:00
|
|
|
else if (SBProcess::EventIsProcessEvent (event))
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
HandleProcessEvent (event);
|
|
|
|
}
|
2012-02-08 13:23:15 +08:00
|
|
|
else if (SBBreakpoint::EventIsBreakpointEvent (event))
|
|
|
|
{
|
|
|
|
HandleBreakpointEvent (event);
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
else if (event.BroadcasterMatchesRef (sb_interpreter.GetBroadcaster()))
|
|
|
|
{
|
2012-02-29 12:21:24 +08:00
|
|
|
// TODO: deprecate the eBroadcastBitQuitCommandReceived event
|
|
|
|
// now that we have SBCommandInterpreter::SetCommandOverrideCallback()
|
|
|
|
// that can take over a command
|
2010-06-09 00:52:24 +08:00
|
|
|
if (event_type & SBCommandInterpreter::eBroadcastBitQuitCommandReceived)
|
2012-01-31 12:56:17 +08:00
|
|
|
{
|
2012-02-29 12:21:24 +08:00
|
|
|
SetIsDone();
|
2012-01-31 12:56:17 +08:00
|
|
|
}
|
2011-05-04 04:53:11 +08:00
|
|
|
else if (event_type & SBCommandInterpreter::eBroadcastBitAsynchronousErrorData)
|
|
|
|
{
|
|
|
|
const char *data = SBEvent::GetCStringFromEvent (event);
|
|
|
|
m_io_channel_ap->ErrWrite (data, strlen(data), ASYNC);
|
|
|
|
}
|
|
|
|
else if (event_type & SBCommandInterpreter::eBroadcastBitAsynchronousOutputData)
|
|
|
|
{
|
|
|
|
const char *data = SBEvent::GetCStringFromEvent (event);
|
|
|
|
m_io_channel_ap->OutWrite (data, strlen(data), ASYNC);
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-03 03:28:31 +08:00
|
|
|
editline_output_pty.CloseMasterFileDescriptor();
|
|
|
|
master_out_comm.Disconnect();
|
|
|
|
out_comm_2.Disconnect();
|
|
|
|
reset_stdin_termios();
|
|
|
|
fclose (stdin);
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
CloseIOChannelFile ();
|
|
|
|
|
|
|
|
if (!iochannel_thread_exited)
|
|
|
|
{
|
2010-07-14 08:18:15 +08:00
|
|
|
event.Clear();
|
2010-06-09 00:52:24 +08:00
|
|
|
listener.GetNextEventForBroadcasterWithType (*m_io_channel_ap,
|
|
|
|
IOChannel::eBroadcastBitThreadDidExit,
|
|
|
|
event);
|
|
|
|
if (!event.IsValid())
|
|
|
|
{
|
|
|
|
// Send end EOF to the driver file descriptor
|
|
|
|
m_io_channel_ap->Stop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-16 05:30:02 +08:00
|
|
|
SBDebugger::Destroy (m_debugger);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Driver::ReadyForCommand ()
|
|
|
|
{
|
|
|
|
if (m_waiting_for_command == false)
|
|
|
|
{
|
|
|
|
m_waiting_for_command = true;
|
|
|
|
BroadcastEventByType (Driver::eBroadcastBitReadyForInput, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-10 01:45:09 +08:00
|
|
|
void
|
|
|
|
sigwinch_handler (int signo)
|
|
|
|
{
|
|
|
|
struct winsize window_size;
|
|
|
|
if (isatty (STDIN_FILENO)
|
|
|
|
&& ::ioctl (STDIN_FILENO, TIOCGWINSZ, &window_size) == 0)
|
|
|
|
{
|
|
|
|
if ((window_size.ws_col > 0) && (strlen (g_debugger_name) > 0))
|
|
|
|
{
|
|
|
|
char width_str_buffer[25];
|
|
|
|
::sprintf (width_str_buffer, "%d", window_size.ws_col);
|
|
|
|
SBDebugger::SetInternalVariable ("term-width", width_str_buffer, g_debugger_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-20 04:47:54 +08:00
|
|
|
void
|
|
|
|
sigint_handler (int signo)
|
|
|
|
{
|
|
|
|
static bool g_interrupt_sent = false;
|
|
|
|
if (g_driver)
|
|
|
|
{
|
|
|
|
if (!g_interrupt_sent)
|
|
|
|
{
|
|
|
|
g_interrupt_sent = true;
|
|
|
|
g_driver->GetDebugger().DispatchInputInterrupt();
|
|
|
|
g_interrupt_sent = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
exit (signo);
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
int
|
2011-01-28 04:15:39 +08:00
|
|
|
main (int argc, char const *argv[], const char *envp[])
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
|
|
|
SBDebugger::Initialize();
|
|
|
|
|
2010-11-08 05:02:03 +08:00
|
|
|
SBHostOS::ThreadCreated ("<lldb.driver.main-thread>");
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2010-10-19 11:25:40 +08:00
|
|
|
signal (SIGPIPE, SIG_IGN);
|
2010-09-10 01:45:09 +08:00
|
|
|
signal (SIGWINCH, sigwinch_handler);
|
2010-11-20 04:47:54 +08:00
|
|
|
signal (SIGINT, sigint_handler);
|
2010-09-10 01:45:09 +08:00
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
// Create a scope for driver so that the driver object will destroy itself
|
|
|
|
// before SBDebugger::Terminate() is called.
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-06-23 09:19:29 +08:00
|
|
|
Driver driver;
|
|
|
|
|
|
|
|
bool exit = false;
|
|
|
|
SBError error (driver.ParseArgs (argc, argv, stdout, exit));
|
|
|
|
if (error.Fail())
|
|
|
|
{
|
|
|
|
const char *error_cstr = error.GetCString ();
|
|
|
|
if (error_cstr)
|
|
|
|
::fprintf (stderr, "error: %s\n", error_cstr);
|
|
|
|
}
|
|
|
|
else if (!exit)
|
|
|
|
{
|
|
|
|
driver.MainLoop ();
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
SBDebugger::Terminate();
|
|
|
|
return 0;
|
|
|
|
}
|