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>
|
2012-11-30 05:49:15 +08:00
|
|
|
#include <inttypes.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.
|
2012-12-04 08:32:51 +08:00
|
|
|
int short_option; // Single character for this option.
|
2011-03-25 05:19:54 +08:00
|
|
|
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[] =
|
|
|
|
{
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_OPT_SET_1, true , "help" , 'h', no_argument , 0, eArgTypeNone,
|
2011-09-16 05:30:02 +08:00
|
|
|
"Prints out the usage information for the LLDB debugger." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_OPT_SET_2, true , "version" , 'v', no_argument , 0, eArgTypeNone,
|
2011-09-16 05:30:02 +08:00
|
|
|
"Prints out the current version number of the LLDB debugger." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_OPT_SET_3, true , "arch" , 'a', required_argument, 0, eArgTypeArchitecture,
|
2011-09-16 05:30:02 +08:00
|
|
|
"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." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_OPT_SET_3, true , "file" , 'f', required_argument, 0, eArgTypeFilename,
|
2011-09-16 05:30:02 +08:00
|
|
|
"Tells the debugger to use the file <filename> as the program to be debugged." },
|
2012-10-24 11:29:40 +08:00
|
|
|
{ LLDB_OPT_SET_3, false, "core" , 'c', required_argument, 0, eArgTypeFilename,
|
2012-08-16 06:10:42 +08:00
|
|
|
"Tells the debugger to use the fullpath to <path> as the core file." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_OPT_SET_4, true , "attach-name" , 'n', required_argument, 0, eArgTypeProcessName,
|
2011-09-16 05:30:02 +08:00
|
|
|
"Tells the debugger to attach to a process with the given name." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_OPT_SET_4, true , "wait-for" , 'w', no_argument , 0, eArgTypeNone,
|
2011-09-16 05:30:02 +08:00
|
|
|
"Tells the debugger to wait for a process with the given pid or name to launch before attaching." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_OPT_SET_5, true , "attach-pid" , 'p', required_argument, 0, eArgTypePid,
|
2011-09-16 05:30:02 +08:00
|
|
|
"Tells the debugger to attach to a process with the given pid." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_3_TO_5, false, "script-language", 'l', required_argument, 0, eArgTypeScriptLang,
|
2011-09-16 05:30:02 +08:00
|
|
|
"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." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_3_TO_5, false, "debug" , 'd', no_argument , 0, eArgTypeNone,
|
2011-09-16 05:30:02 +08:00
|
|
|
"Tells the debugger to print out extra information for debugging itself." },
|
2013-09-14 08:20:24 +08:00
|
|
|
{ LLDB_3_TO_5, false, "source-quietly" , 'b', no_argument , 0, eArgTypeNone,
|
|
|
|
"Tells the debugger to print out extra information for debugging itself." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_3_TO_5, false, "source" , 's', required_argument, 0, eArgTypeFilename,
|
2011-09-16 05:30:02 +08:00
|
|
|
"Tells the debugger to read in and execute the file <file>, which should contain lldb commands." },
|
2013-09-14 08:20:24 +08:00
|
|
|
{ LLDB_3_TO_5, false, "one-line" , 'o', required_argument, 0, eArgTypeNone,
|
|
|
|
"Tells the debugger to execute this one-line lldb command." },
|
|
|
|
{ LLDB_3_TO_5, false, "source-before-file" , 'S', required_argument, 0, eArgTypeFilename,
|
|
|
|
"Tells the debugger to read in and execute the file <file>, which should contain lldb commands." },
|
|
|
|
{ LLDB_3_TO_5, false, "one-line-before-file" , 'O', required_argument, 0, eArgTypeNone,
|
|
|
|
"Tells the debugger to execute this one-line lldb command." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_3_TO_5, false, "editor" , 'e', no_argument , 0, eArgTypeNone,
|
2011-09-16 05:30:02 +08:00
|
|
|
"Tells the debugger to open source files using the host's \"external editor\" mechanism." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ LLDB_3_TO_5, false, "no-lldbinit" , 'x', no_argument , 0, eArgTypeNone,
|
2011-09-16 05:30:02 +08:00
|
|
|
"Do not automatically parse any '.lldbinit' files." },
|
2013-09-14 08:20:24 +08:00
|
|
|
{ LLDB_3_TO_5, false, "no-use-colors" , 'X', no_argument , 0, eArgTypeNone,
|
2013-05-24 04:47:45 +08:00
|
|
|
"Do not use colors." },
|
|
|
|
{ LLDB_OPT_SET_6, true , "python-path" , 'P', no_argument , 0, eArgTypeNone,
|
2012-12-22 06:22:26 +08:00
|
|
|
"Prints out the path to the lldb.py file for this version of lldb." },
|
2012-09-12 02:11:16 +08:00
|
|
|
{ 0, false, NULL , 0 , 0 , 0, 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 (),
|
<rdar://problem/13764135>
The "lldb" driver was interfering with STDOUT and STDERR if the output was over 1024 charcters long. The output was grabbing 1024 characters at a time, before it output the characters, it was writing characters to the screen to clear the current line. This has been fixed.
I also fixed the command interpreter from mixing the "(lldb) " prompt in with program output by always manually checking for program output. This was done by having the command interpreter know when it is in the middle of executing a command by setting a bool. This was needed since sometimes when a command would run the target, like with a command like 'expression (int)printf("hello\n")', the process would push a new input reader, and then pop it when it was done. This popping of the input reader would cause the command interpreter to get sent a reactivated message (from the private process state thread) and cause it to ask for another command, even though we were still in the middle of the command ('expression (int)printf("hello\n")'). Now we set a bool to true, run the command and set the bool to false. If we get reactivated while we are in the middle of a command, we don't say we are ready for a new command. This coupled with emitting the STDOUT/STDERR first after each command, followed by the command results, followed by then saying we are ready for a new command, should help cleanup the command line output on all platforms.
llvm-svn: 181807
2013-05-15 01:36:51 +08:00
|
|
|
m_executing_user_command (false),
|
2012-12-18 01:40:07 +08:00
|
|
|
m_waiting_for_command (false),
|
|
|
|
m_done(false)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
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 ()
|
|
|
|
{
|
2012-05-17 06:01:10 +08:00
|
|
|
// Write an End of File sequence to the file descriptor to ensure any
|
2010-06-09 00:52:24 +08:00
|
|
|
// 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),
|
2012-08-16 06:10:42 +08:00
|
|
|
m_core_file (),
|
2010-07-10 04:39:50 +08:00
|
|
|
m_crash_log (),
|
2013-09-14 08:20:24 +08:00
|
|
|
m_initial_commands (),
|
|
|
|
m_after_file_commands (),
|
2010-06-23 09:19:29 +08:00
|
|
|
m_debug_mode (false),
|
2013-09-14 08:20:24 +08:00
|
|
|
m_source_quietly(false),
|
2010-07-10 04:39:50 +08:00
|
|
|
m_print_version (false),
|
2012-12-22 06:22:26 +08:00
|
|
|
m_print_python_path (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;
|
2013-09-14 08:20:24 +08:00
|
|
|
m_initial_commands.clear ();
|
|
|
|
m_after_file_commands.clear ();
|
2010-06-23 09:19:29 +08:00
|
|
|
m_debug_mode = false;
|
2013-09-14 08:20:24 +08:00
|
|
|
m_source_quietly = false;
|
2010-06-23 09:19:29 +08:00
|
|
|
m_print_help = false;
|
|
|
|
m_print_version = false;
|
2012-12-22 06:22:26 +08:00
|
|
|
m_print_python_path = 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
|
|
|
}
|
|
|
|
|
2013-09-14 08:20:24 +08:00
|
|
|
void
|
|
|
|
Driver::OptionData::AddInitialCommand (const char *command, bool before_file, bool is_file, SBError &error)
|
|
|
|
{
|
|
|
|
std::vector<std::pair<bool, std::string> > *command_set;
|
|
|
|
if (before_file)
|
|
|
|
command_set = &(m_initial_commands);
|
|
|
|
else
|
|
|
|
command_set = &(m_after_file_commands);
|
|
|
|
|
|
|
|
if (is_file)
|
|
|
|
{
|
|
|
|
SBFileSpec file(command);
|
|
|
|
if (file.Exists())
|
|
|
|
command_set->push_back (std::pair<bool, std::string> (true, optarg));
|
|
|
|
else if (file.ResolveExecutableLocation())
|
|
|
|
{
|
|
|
|
char final_path[PATH_MAX];
|
|
|
|
file.GetPath (final_path, sizeof(final_path));
|
|
|
|
std::string path_str (final_path);
|
|
|
|
command_set->push_back (std::pair<bool, std::string> (true, path_str));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
error.SetErrorStringWithFormat("file specified in --source (-s) option doesn't exist: '%s'", optarg);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
command_set->push_back (std::pair<bool, std::string> (false, optarg));
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2013-09-14 08:20:24 +08:00
|
|
|
void
|
|
|
|
Driver::ExecuteInitialCommands (bool before_file)
|
2010-06-23 09:19:29 +08:00
|
|
|
{
|
2013-09-14 08:20:24 +08:00
|
|
|
size_t num_commands;
|
|
|
|
std::vector<std::pair<bool, std::string> > *command_set;
|
|
|
|
if (before_file)
|
|
|
|
command_set = &(m_option_data.m_initial_commands);
|
|
|
|
else
|
|
|
|
command_set = &(m_option_data.m_after_file_commands);
|
|
|
|
|
|
|
|
num_commands = command_set->size();
|
|
|
|
SBCommandReturnObject result;
|
|
|
|
bool old_async = GetDebugger().GetAsync();
|
|
|
|
GetDebugger().SetAsync(false);
|
|
|
|
for (size_t idx = 0; idx < num_commands; idx++)
|
|
|
|
{
|
|
|
|
bool is_file = (*command_set)[idx].first;
|
|
|
|
const char *command = (*command_set)[idx].second.c_str();
|
|
|
|
char command_string[PATH_MAX * 2];
|
|
|
|
const bool dump_stream_only_if_no_immediate = true;
|
|
|
|
const char *executed_command = command;
|
|
|
|
if (is_file)
|
|
|
|
{
|
|
|
|
::snprintf (command_string, sizeof(command_string), "command source '%s'", command);
|
|
|
|
executed_command = command_string;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_debugger.GetCommandInterpreter().HandleCommand (executed_command, result, false);
|
|
|
|
if (!m_option_data.m_source_quietly || result.Succeeded() == false)
|
|
|
|
{
|
|
|
|
const size_t output_size = result.GetOutputSize();
|
|
|
|
if (output_size > 0)
|
|
|
|
m_io_channel_ap->OutWrite (result.GetOutput(dump_stream_only_if_no_immediate), output_size, NO_ASYNC);
|
|
|
|
const size_t error_size = result.GetErrorSize();
|
|
|
|
if (error_size > 0)
|
|
|
|
m_io_channel_ap->OutWrite (result.GetError(dump_stream_only_if_no_immediate), error_size, NO_ASYNC);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result.Succeeded() == false)
|
|
|
|
{
|
|
|
|
char error_buffer[1024];
|
|
|
|
size_t error_size;
|
|
|
|
const char *type = before_file ? "before file" : "after_file";
|
|
|
|
if (is_file)
|
|
|
|
error_size = ::snprintf(error_buffer, sizeof(error_buffer), "Aborting %s command execution, command file: '%s' failed.\n", type, command);
|
|
|
|
else
|
|
|
|
error_size = ::snprintf(error_buffer, sizeof(error_buffer), "Aborting %s command execution, command: '%s' failed.\n", type, command);
|
|
|
|
|
|
|
|
m_io_channel_ap->OutWrite(error_buffer, error_size, NO_ASYNC);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
result.Clear();
|
|
|
|
}
|
|
|
|
GetDebugger().SetAsync(old_async);
|
2010-06-23 09:19:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2013-04-05 04:35:24 +08:00
|
|
|
// Build the option_string argument for call to getopt_long_only.
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
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);
|
|
|
|
|
2013-04-05 04:35:24 +08:00
|
|
|
// Prepare for & make calls to getopt_long_only.
|
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;
|
2013-04-05 04:35:24 +08:00
|
|
|
val = ::getopt_long_only (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)
|
|
|
|
{
|
2012-12-04 08:32:51 +08:00
|
|
|
const int short_option = 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
|
|
|
|
2012-12-22 06:22:26 +08:00
|
|
|
case 'P':
|
|
|
|
m_option_data.m_print_python_path = true;
|
|
|
|
break;
|
|
|
|
|
2010-06-23 09:19:29 +08:00
|
|
|
case 'c':
|
2012-08-16 06:10:42 +08:00
|
|
|
{
|
|
|
|
SBFileSpec file(optarg);
|
|
|
|
if (file.Exists())
|
|
|
|
{
|
|
|
|
m_option_data.m_core_file = optarg;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
error.SetErrorStringWithFormat("file specified in --core (-c) option doesn't exist: '%s'", optarg);
|
|
|
|
}
|
2010-06-23 09:19:29 +08:00
|
|
|
break;
|
2012-08-16 06:10:42 +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;
|
|
|
|
|
2013-09-14 08:20:24 +08:00
|
|
|
case 'X':
|
2013-05-24 04:47:45 +08:00
|
|
|
m_debugger.SetUseColor (false);
|
|
|
|
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
|
|
|
|
2013-09-14 08:20:24 +08:00
|
|
|
case 'q':
|
|
|
|
m_option_data.m_source_quietly = true;
|
|
|
|
break;
|
|
|
|
|
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':
|
2013-09-14 08:20:24 +08:00
|
|
|
m_option_data.AddInitialCommand(optarg, false, true, error);
|
|
|
|
break;
|
|
|
|
case 'o':
|
|
|
|
m_option_data.AddInitialCommand(optarg, false, false, error);
|
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
m_option_data.AddInitialCommand(optarg, true, true, error);
|
|
|
|
break;
|
|
|
|
case 'O':
|
|
|
|
m_option_data.AddInitialCommand(optarg, true, false, error);
|
2010-06-23 09:19:29 +08:00
|
|
|
break;
|
|
|
|
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
|
|
|
}
|
2012-12-22 06:22:26 +08:00
|
|
|
else if (m_option_data.m_print_python_path)
|
|
|
|
{
|
|
|
|
SBFileSpec python_file_spec = SBHostOS::GetLLDBPythonPath();
|
|
|
|
if (python_file_spec.IsValid())
|
|
|
|
{
|
|
|
|
char python_path[PATH_MAX];
|
|
|
|
size_t num_chars = python_file_spec.GetPath(python_path, PATH_MAX);
|
|
|
|
if (num_chars < PATH_MAX)
|
|
|
|
{
|
|
|
|
::fprintf (out_fh, "%s\n", python_path);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
::fprintf (out_fh, "<PATH TOO LONG>\n");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
::fprintf (out_fh, "<COULD NOT FIND PATH>\n");
|
|
|
|
exit = true;
|
|
|
|
}
|
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.
|
|
|
|
|
2013-04-05 04:35:24 +08:00
|
|
|
// Skip any options we consumed with getopt_long_only
|
2010-12-09 06:23:24 +08:00
|
|
|
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
|
|
|
|
{
|
2013-04-05 04:35:24 +08:00
|
|
|
// Skip any options we consumed with getopt_long_only
|
2011-09-14 07:25:31 +08:00
|
|
|
argc -= optind;
|
2012-07-17 11:23:13 +08:00
|
|
|
//argv += optind; // Commented out to keep static analyzer happy
|
2011-09-14 07:25:31 +08:00
|
|
|
|
|
|
|
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
|
|
|
{
|
<rdar://problem/13764135>
The "lldb" driver was interfering with STDOUT and STDERR if the output was over 1024 charcters long. The output was grabbing 1024 characters at a time, before it output the characters, it was writing characters to the screen to clear the current line. This has been fixed.
I also fixed the command interpreter from mixing the "(lldb) " prompt in with program output by always manually checking for program output. This was done by having the command interpreter know when it is in the middle of executing a command by setting a bool. This was needed since sometimes when a command would run the target, like with a command like 'expression (int)printf("hello\n")', the process would push a new input reader, and then pop it when it was done. This popping of the input reader would cause the command interpreter to get sent a reactivated message (from the private process state thread) and cause it to ask for another command, even though we were still in the middle of the command ('expression (int)printf("hello\n")'). Now we set a bool to true, run the command and set the bool to false. If we get reactivated while we are in the middle of a command, we don't say we are ready for a new command. This coupled with emitting the STDOUT/STDERR first after each command, followed by the command results, followed by then saying we are ready for a new command, should help cleanup the command line output on all platforms.
llvm-svn: 181807
2013-05-15 01:36:51 +08:00
|
|
|
m_io_channel_ap->OutWrite (stdio_buffer, len, NO_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
|
|
|
{
|
<rdar://problem/13764135>
The "lldb" driver was interfering with STDOUT and STDERR if the output was over 1024 charcters long. The output was grabbing 1024 characters at a time, before it output the characters, it was writing characters to the screen to clear the current line. This has been fixed.
I also fixed the command interpreter from mixing the "(lldb) " prompt in with program output by always manually checking for program output. This was done by having the command interpreter know when it is in the middle of executing a command by setting a bool. This was needed since sometimes when a command would run the target, like with a command like 'expression (int)printf("hello\n")', the process would push a new input reader, and then pop it when it was done. This popping of the input reader would cause the command interpreter to get sent a reactivated message (from the private process state thread) and cause it to ask for another command, even though we were still in the middle of the command ('expression (int)printf("hello\n")'). Now we set a bool to true, run the command and set the bool to false. If we get reactivated while we are in the middle of a command, we don't say we are ready for a new command. This coupled with emitting the STDOUT/STDERR first after each command, followed by the command results, followed by then saying we are ready for a new command, should help cleanup the command line output on all platforms.
llvm-svn: 181807
2013-05-15 01:36:51 +08:00
|
|
|
m_io_channel_ap->ErrWrite (stdio_buffer, len, NO_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)
|
|
|
|
{
|
|
|
|
case eStopReasonInvalid:
|
|
|
|
case eStopReasonNone:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eStopReasonTrace:
|
|
|
|
case eStopReasonBreakpoint:
|
|
|
|
case eStopReasonWatchpoint:
|
|
|
|
case eStopReasonSignal:
|
|
|
|
case eStopReasonException:
|
2012-12-05 08:16:59 +08:00
|
|
|
case eStopReasonExec:
|
2012-12-21 07:08:03 +08:00
|
|
|
case eStopReasonThreadExiting:
|
2010-06-09 00:52:24 +08:00
|
|
|
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);
|
2012-03-06 08:37:27 +08:00
|
|
|
int message_len = ::snprintf (message, sizeof(message), "%d location%s added to breakpoint %d\n",
|
2012-02-08 13:23:15 +08:00
|
|
|
num_new_locations,
|
2012-09-28 09:50:47 +08:00
|
|
|
num_new_locations == 1 ? "" : "s",
|
2012-02-08 13:23:15 +08:00
|
|
|
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];
|
2012-11-30 05:49:15 +08:00
|
|
|
int message_len = ::snprintf (message, sizeof(message), "Process %" PRIu64 " %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))
|
|
|
|
{
|
2013-02-09 09:29:05 +08:00
|
|
|
size_t num_reasons = SBProcess::GetNumRestartedReasonsFromEvent(event);
|
|
|
|
if (num_reasons > 0)
|
|
|
|
{
|
2010-06-09 00:52:24 +08:00
|
|
|
// FIXME: Do we want to report this, or would that just be annoyingly chatty?
|
2013-02-09 09:29:05 +08:00
|
|
|
if (num_reasons == 1)
|
|
|
|
{
|
|
|
|
char message[1024];
|
|
|
|
const char *reason = SBProcess::GetRestartedReasonAtIndexFromEvent (event, 0);
|
|
|
|
int message_len = ::snprintf (message, sizeof(message), "Process %" PRIu64 " stopped and restarted: %s\n",
|
|
|
|
process.GetProcessID(), reason ? reason : "<UNKNOWN REASON>");
|
|
|
|
m_io_channel_ap->OutWrite(message, message_len, ASYNC);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
char message[1024];
|
|
|
|
int message_len = ::snprintf (message, sizeof(message), "Process %" PRIu64 " stopped and restarted, reasons:\n",
|
2010-06-09 00:52:24 +08:00
|
|
|
process.GetProcessID());
|
2013-02-09 09:29:05 +08:00
|
|
|
m_io_channel_ap->OutWrite(message, message_len, ASYNC);
|
|
|
|
for (size_t i = 0; i < num_reasons; i++)
|
|
|
|
{
|
|
|
|
const char *reason = SBProcess::GetRestartedReasonAtIndexFromEvent (event, i);
|
|
|
|
int message_len = ::snprintf(message, sizeof(message), "\t%s\n", reason ? reason : "<UNKNOWN REASON>");
|
|
|
|
m_io_channel_ap->OutWrite(message, message_len, ASYNC);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-05-09 07:06:07 +08:00
|
|
|
if (GetDebugger().GetSelectedTarget() == process.GetTarget())
|
|
|
|
{
|
|
|
|
SBCommandReturnObject result;
|
|
|
|
UpdateSelectedThread ();
|
|
|
|
m_debugger.GetCommandInterpreter().HandleCommand("process status", result, false);
|
|
|
|
m_io_channel_ap->ErrWrite (result.GetError(), result.GetErrorSize(), ASYNC);
|
|
|
|
m_io_channel_ap->OutWrite (result.GetOutput(), result.GetOutputSize(), ASYNC);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
SBStream out_stream;
|
|
|
|
uint32_t target_idx = GetDebugger().GetIndexOfTarget(process.GetTarget());
|
|
|
|
if (target_idx != UINT32_MAX)
|
|
|
|
out_stream.Printf ("Target %d: (", target_idx);
|
|
|
|
else
|
|
|
|
out_stream.Printf ("Target <unknown index>: (");
|
|
|
|
process.GetTarget().GetDescription (out_stream, eDescriptionLevelBrief);
|
|
|
|
out_stream.Printf (") stopped.\n");
|
|
|
|
m_io_channel_ap->OutWrite (out_stream.GetData(), out_stream.GetSize(), ASYNC);
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-11 02:32:14 +08:00
|
|
|
void
|
|
|
|
Driver::HandleThreadEvent (const SBEvent &event)
|
|
|
|
{
|
|
|
|
// At present the only thread event we handle is the Frame Changed event, and all we do for that is just
|
|
|
|
// reprint the thread status for that thread.
|
|
|
|
using namespace lldb;
|
|
|
|
const uint32_t event_type = event.GetType();
|
2012-12-11 10:31:48 +08:00
|
|
|
if (event_type == SBThread::eBroadcastBitStackChanged
|
|
|
|
|| event_type == SBThread::eBroadcastBitThreadSelected)
|
2012-10-11 02:32:14 +08:00
|
|
|
{
|
|
|
|
SBThread thread = SBThread::GetThreadFromEvent (event);
|
|
|
|
if (thread.IsValid())
|
|
|
|
{
|
|
|
|
SBStream out_stream;
|
|
|
|
thread.GetStatus(out_stream);
|
|
|
|
m_io_channel_ap->OutWrite (out_stream.GetData (), out_stream.GetSize (), ASYNC);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
// 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.
|
<rdar://problem/13764135>
The "lldb" driver was interfering with STDOUT and STDERR if the output was over 1024 charcters long. The output was grabbing 1024 characters at a time, before it output the characters, it was writing characters to the screen to clear the current line. This has been fixed.
I also fixed the command interpreter from mixing the "(lldb) " prompt in with program output by always manually checking for program output. This was done by having the command interpreter know when it is in the middle of executing a command by setting a bool. This was needed since sometimes when a command would run the target, like with a command like 'expression (int)printf("hello\n")', the process would push a new input reader, and then pop it when it was done. This popping of the input reader would cause the command interpreter to get sent a reactivated message (from the private process state thread) and cause it to ask for another command, even though we were still in the middle of the command ('expression (int)printf("hello\n")'). Now we set a bool to true, run the command and set the bool to false. If we get reactivated while we are in the middle of a command, we don't say we are ready for a new command. This coupled with emitting the STDOUT/STDERR first after each command, followed by the command results, followed by then saying we are ready for a new command, should help cleanup the command line output on all platforms.
llvm-svn: 181807
2013-05-15 01:36:51 +08:00
|
|
|
|
|
|
|
// Note that we are in the process of executing a command
|
|
|
|
m_executing_user_command = true;
|
|
|
|
|
2011-02-19 10:53:09 +08:00
|
|
|
m_debugger.GetCommandInterpreter().HandleCommand (command_string, result, true);
|
|
|
|
|
<rdar://problem/13764135>
The "lldb" driver was interfering with STDOUT and STDERR if the output was over 1024 charcters long. The output was grabbing 1024 characters at a time, before it output the characters, it was writing characters to the screen to clear the current line. This has been fixed.
I also fixed the command interpreter from mixing the "(lldb) " prompt in with program output by always manually checking for program output. This was done by having the command interpreter know when it is in the middle of executing a command by setting a bool. This was needed since sometimes when a command would run the target, like with a command like 'expression (int)printf("hello\n")', the process would push a new input reader, and then pop it when it was done. This popping of the input reader would cause the command interpreter to get sent a reactivated message (from the private process state thread) and cause it to ask for another command, even though we were still in the middle of the command ('expression (int)printf("hello\n")'). Now we set a bool to true, run the command and set the bool to false. If we get reactivated while we are in the middle of a command, we don't say we are ready for a new command. This coupled with emitting the STDOUT/STDERR first after each command, followed by the command results, followed by then saying we are ready for a new command, should help cleanup the command line output on all platforms.
llvm-svn: 181807
2013-05-15 01:36:51 +08:00
|
|
|
// Note that we are back from executing a user command
|
|
|
|
m_executing_user_command = false;
|
|
|
|
|
|
|
|
// Display any STDOUT/STDERR _prior_ to emitting the command result text
|
|
|
|
GetProcessSTDOUT ();
|
|
|
|
GetProcessSTDERR ();
|
|
|
|
|
2012-10-17 04:57:12 +08:00
|
|
|
const bool only_if_no_immediate = true;
|
|
|
|
|
<rdar://problem/13764135>
The "lldb" driver was interfering with STDOUT and STDERR if the output was over 1024 charcters long. The output was grabbing 1024 characters at a time, before it output the characters, it was writing characters to the screen to clear the current line. This has been fixed.
I also fixed the command interpreter from mixing the "(lldb) " prompt in with program output by always manually checking for program output. This was done by having the command interpreter know when it is in the middle of executing a command by setting a bool. This was needed since sometimes when a command would run the target, like with a command like 'expression (int)printf("hello\n")', the process would push a new input reader, and then pop it when it was done. This popping of the input reader would cause the command interpreter to get sent a reactivated message (from the private process state thread) and cause it to ask for another command, even though we were still in the middle of the command ('expression (int)printf("hello\n")'). Now we set a bool to true, run the command and set the bool to false. If we get reactivated while we are in the middle of a command, we don't say we are ready for a new command. This coupled with emitting the STDOUT/STDERR first after each command, followed by the command results, followed by then saying we are ready for a new command, should help cleanup the command line output on all platforms.
llvm-svn: 181807
2013-05-15 01:36:51 +08:00
|
|
|
// Now emit the command output text from the command we just executed
|
2012-10-17 05:11:14 +08:00
|
|
|
const size_t output_size = result.GetOutputSize();
|
2012-10-17 04:57:12 +08:00
|
|
|
if (output_size > 0)
|
|
|
|
m_io_channel_ap->OutWrite (result.GetOutput(only_if_no_immediate), output_size, NO_ASYNC);
|
|
|
|
|
<rdar://problem/13764135>
The "lldb" driver was interfering with STDOUT and STDERR if the output was over 1024 charcters long. The output was grabbing 1024 characters at a time, before it output the characters, it was writing characters to the screen to clear the current line. This has been fixed.
I also fixed the command interpreter from mixing the "(lldb) " prompt in with program output by always manually checking for program output. This was done by having the command interpreter know when it is in the middle of executing a command by setting a bool. This was needed since sometimes when a command would run the target, like with a command like 'expression (int)printf("hello\n")', the process would push a new input reader, and then pop it when it was done. This popping of the input reader would cause the command interpreter to get sent a reactivated message (from the private process state thread) and cause it to ask for another command, even though we were still in the middle of the command ('expression (int)printf("hello\n")'). Now we set a bool to true, run the command and set the bool to false. If we get reactivated while we are in the middle of a command, we don't say we are ready for a new command. This coupled with emitting the STDOUT/STDERR first after each command, followed by the command results, followed by then saying we are ready for a new command, should help cleanup the command line output on all platforms.
llvm-svn: 181807
2013-05-15 01:36:51 +08:00
|
|
|
// Now emit the command error text from the command we just executed
|
2012-10-17 05:11:14 +08:00
|
|
|
const size_t error_size = result.GetErrorSize();
|
2012-10-17 04:57:12 +08:00
|
|
|
if (error_size > 0)
|
|
|
|
m_io_channel_ap->OutWrite (result.GetError(only_if_no_immediate), error_size, NO_ASYNC);
|
2011-05-17 03:20:50 +08:00
|
|
|
|
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:
|
<rdar://problem/13764135>
The "lldb" driver was interfering with STDOUT and STDERR if the output was over 1024 charcters long. The output was grabbing 1024 characters at a time, before it output the characters, it was writing characters to the screen to clear the current line. This has been fixed.
I also fixed the command interpreter from mixing the "(lldb) " prompt in with program output by always manually checking for program output. This was done by having the command interpreter know when it is in the middle of executing a command by setting a bool. This was needed since sometimes when a command would run the target, like with a command like 'expression (int)printf("hello\n")', the process would push a new input reader, and then pop it when it was done. This popping of the input reader would cause the command interpreter to get sent a reactivated message (from the private process state thread) and cause it to ask for another command, even though we were still in the middle of the command ('expression (int)printf("hello\n")'). Now we set a bool to true, run the command and set the bool to false. If we get reactivated while we are in the middle of a command, we don't say we are ready for a new command. This coupled with emitting the STDOUT/STDERR first after each command, followed by the command results, followed by then saying we are ready for a new command, should help cleanup the command line output on all platforms.
llvm-svn: 181807
2013-05-15 01:36:51 +08:00
|
|
|
if (driver->m_executing_user_command == false)
|
|
|
|
driver->ReadyForCommand();
|
2010-06-09 00:52:24 +08:00
|
|
|
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)
|
|
|
|
{
|
2012-07-28 07:57:19 +08:00
|
|
|
SBProcess process(driver->GetDebugger().GetSelectedTarget().GetProcess());
|
2012-06-01 09:07:02 +08:00
|
|
|
if (!driver->m_io_channel_ap->EditLineHasCharacters()
|
2012-07-28 07:57:19 +08:00
|
|
|
&& process.IsValid()
|
|
|
|
&& (process.GetState() == lldb::eStateRunning || process.GetState() == lldb::eStateAttaching))
|
2012-06-01 09:07:02 +08:00
|
|
|
{
|
2012-07-28 07:57:19 +08:00
|
|
|
process.SendAsyncInterrupt ();
|
2012-06-01 09:07:02 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
driver->m_io_channel_ap->OutWrite ("^C\n", 3, NO_ASYNC);
|
|
|
|
// I wish I could erase the entire input line, but there's no public API for that.
|
|
|
|
driver->m_io_channel_ap->EraseCharsBeforeCursor();
|
|
|
|
driver->m_io_channel_ap->RefreshPrompt();
|
|
|
|
}
|
2010-11-20 04:47:54 +08:00
|
|
|
}
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
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());
|
2010-06-09 00:52:24 +08:00
|
|
|
if (listener.IsValid())
|
|
|
|
{
|
|
|
|
|
2012-10-11 02:32:14 +08:00
|
|
|
listener.StartListeningForEventClass(m_debugger,
|
|
|
|
SBTarget::GetBroadcasterClassName(),
|
|
|
|
SBTarget::eBroadcastBitBreakpointChanged);
|
|
|
|
listener.StartListeningForEventClass(m_debugger,
|
|
|
|
SBThread::GetBroadcasterClassName(),
|
2012-12-11 10:31:48 +08:00
|
|
|
SBThread::eBroadcastBitStackChanged |
|
|
|
|
SBThread::eBroadcastBitThreadSelected);
|
2010-06-09 00:52:24 +08:00
|
|
|
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
|
2013-09-14 08:20:24 +08:00
|
|
|
// First source in the commands specified to be run before the file arguments are processed.
|
|
|
|
ExecuteInitialCommands(true);
|
|
|
|
|
2012-08-16 06:10:42 +08:00
|
|
|
// Was there a core file specified?
|
|
|
|
std::string core_file_spec("");
|
|
|
|
if (!m_option_data.m_core_file.empty())
|
|
|
|
core_file_spec.append("--core ").append(m_option_data.m_core_file);
|
|
|
|
|
2013-09-14 08:20:24 +08:00
|
|
|
char command_string[PATH_MAX * 2];
|
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),
|
2012-08-16 06:10:42 +08:00
|
|
|
"target create --arch=%s %s \"%s\"",
|
2010-12-09 06:23:24 +08:00
|
|
|
arch_name,
|
2012-08-16 06:10:42 +08:00
|
|
|
core_file_spec.c_str(),
|
2010-12-09 06:23:24 +08:00
|
|
|
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),
|
2012-08-16 06:10:42 +08:00
|
|
|
"target create %s \"%s\"",
|
|
|
|
core_file_spec.c_str(),
|
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
|
|
|
}
|
2012-08-16 06:10:42 +08:00
|
|
|
else if (!core_file_spec.empty())
|
|
|
|
{
|
|
|
|
::snprintf (command_string,
|
|
|
|
sizeof(command_string),
|
|
|
|
"target create %s",
|
|
|
|
core_file_spec.c_str());
|
|
|
|
m_debugger.HandleCommand (command_string);;
|
|
|
|
}
|
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
|
|
|
}
|
2013-09-14 08:20:24 +08:00
|
|
|
|
|
|
|
// Now execute the commands specified for after the file arguments are processed.
|
|
|
|
ExecuteInitialCommands(false);
|
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];
|
2012-11-30 05:49:15 +08:00
|
|
|
::snprintf (pid_buffer, sizeof(pid_buffer), "%" PRIu64, 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);
|
|
|
|
}
|
2012-10-11 02:32:14 +08:00
|
|
|
else if (SBThread::EventIsThreadEvent (event))
|
|
|
|
{
|
|
|
|
HandleThreadEvent (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
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-23 07:31:28 +08:00
|
|
|
master_out_comm.SetReadThreadBytesReceivedCallback(NULL, NULL);
|
2012-02-03 03:28:31 +08:00
|
|
|
master_out_comm.Disconnect();
|
2013-05-23 07:31:28 +08:00
|
|
|
master_out_comm.ReadThreadStop();
|
|
|
|
|
|
|
|
out_comm_2.SetReadThreadBytesReceivedCallback(NULL, NULL);
|
2012-02-03 03:28:31 +08:00
|
|
|
out_comm_2.Disconnect();
|
2013-05-23 07:31:28 +08:00
|
|
|
out_comm_2.ReadThreadStop();
|
|
|
|
|
|
|
|
editline_output_pty.CloseMasterFileDescriptor();
|
2012-02-03 03:28:31 +08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-23 06:56:55 +08:00
|
|
|
void
|
|
|
|
Driver::ResizeWindow (unsigned short col)
|
|
|
|
{
|
|
|
|
GetDebugger().SetTerminalWidth (col);
|
|
|
|
if (m_io_channel_ap.get() != NULL)
|
|
|
|
{
|
|
|
|
m_io_channel_ap->ElResize();
|
|
|
|
}
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
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)
|
|
|
|
{
|
2012-04-27 05:39:32 +08:00
|
|
|
if ((window_size.ws_col > 0) && g_driver != NULL)
|
2010-09-10 01:45:09 +08:00
|
|
|
{
|
2013-02-23 06:56:55 +08:00
|
|
|
g_driver->ResizeWindow (window_size.ws_col);
|
2010-09-10 01:45:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2012-12-01 04:23:19 +08:00
|
|
|
void
|
|
|
|
sigtstp_handler (int signo)
|
|
|
|
{
|
|
|
|
g_driver->GetDebugger().SaveInputTerminalState();
|
|
|
|
signal (signo, SIG_DFL);
|
|
|
|
kill (getpid(), signo);
|
|
|
|
signal (signo, sigtstp_handler);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
sigcont_handler (int signo)
|
|
|
|
{
|
|
|
|
g_driver->GetDebugger().RestoreInputTerminalState();
|
|
|
|
signal (signo, SIG_DFL);
|
|
|
|
kill (getpid(), signo);
|
|
|
|
signal (signo, sigcont_handler);
|
|
|
|
}
|
|
|
|
|
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);
|
2012-12-01 04:23:19 +08:00
|
|
|
signal (SIGTSTP, sigtstp_handler);
|
|
|
|
signal (SIGCONT, sigcont_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;
|
|
|
|
}
|