forked from OSchip/llvm-project
618 lines
19 KiB
C++
618 lines
19 KiB
C++
//===-- IOChannel.cpp -------------------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "IOChannel.h"
|
|
|
|
#include <map>
|
|
|
|
#include "lldb/API/SBCommandInterpreter.h"
|
|
#include "lldb/API/SBDebugger.h"
|
|
#include "lldb/API/SBError.h"
|
|
#include "lldb/API/SBEvent.h"
|
|
#include "lldb/API/SBFileSpec.h"
|
|
#include "lldb/API/SBHostOS.h"
|
|
#include "lldb/API/SBListener.h"
|
|
#include "lldb/API/SBStringList.h"
|
|
|
|
#include <string.h>
|
|
#include <limits.h>
|
|
|
|
using namespace lldb;
|
|
|
|
typedef std::map<EditLine *, std::string> PromptMap;
|
|
const char *g_default_prompt = "(lldb) ";
|
|
PromptMap g_prompt_map;
|
|
|
|
// Printing the following string causes libedit to back up to the beginning of the line & blank it out.
|
|
const char undo_prompt_string[4] = { (char) 13, (char) 27, (char) 91, (char) 75};
|
|
|
|
static const char*
|
|
el_prompt(EditLine *el)
|
|
{
|
|
PromptMap::const_iterator pos = g_prompt_map.find (el);
|
|
if (pos == g_prompt_map.end())
|
|
return g_default_prompt;
|
|
return pos->second.c_str();
|
|
}
|
|
|
|
const char *
|
|
IOChannel::GetPrompt ()
|
|
{
|
|
PromptMap::const_iterator pos = g_prompt_map.find (m_edit_line);
|
|
if (pos == g_prompt_map.end())
|
|
return g_default_prompt;
|
|
return pos->second.c_str();
|
|
}
|
|
|
|
bool
|
|
IOChannel::EditLineHasCharacters ()
|
|
{
|
|
const LineInfo *line_info = el_line(m_edit_line);
|
|
if (line_info)
|
|
return line_info->cursor != line_info->buffer;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
|
|
void
|
|
IOChannel::EraseCharsBeforeCursor ()
|
|
{
|
|
const LineInfo *line_info = el_line(m_edit_line);
|
|
el_deletestr(m_edit_line, line_info->cursor - line_info->buffer);
|
|
}
|
|
|
|
unsigned char
|
|
IOChannel::ElCompletionFn (EditLine *e, int ch)
|
|
{
|
|
IOChannel *io_channel;
|
|
if (el_get(e, EL_CLIENTDATA, &io_channel) == 0)
|
|
{
|
|
return io_channel->HandleCompletion (e, ch);
|
|
}
|
|
else
|
|
{
|
|
return CC_ERROR;
|
|
}
|
|
}
|
|
|
|
unsigned char
|
|
IOChannel::HandleCompletion (EditLine *e, int ch)
|
|
{
|
|
assert (e == m_edit_line);
|
|
|
|
const LineInfo *line_info = el_line(m_edit_line);
|
|
SBStringList completions;
|
|
int page_size = 40;
|
|
|
|
int num_completions = m_driver->GetDebugger().GetCommandInterpreter().HandleCompletion (line_info->buffer,
|
|
line_info->cursor,
|
|
line_info->lastchar,
|
|
0,
|
|
-1,
|
|
completions);
|
|
|
|
if (num_completions == -1)
|
|
{
|
|
el_insertstr (m_edit_line, m_completion_key);
|
|
return CC_REDISPLAY;
|
|
}
|
|
else if (num_completions == -2)
|
|
{
|
|
el_deletestr (m_edit_line, line_info->cursor - line_info->buffer);
|
|
el_insertstr (m_edit_line, completions.GetStringAtIndex(0));
|
|
return CC_REDISPLAY;
|
|
}
|
|
|
|
// If we get a longer match display that first.
|
|
const char *completion_str = completions.GetStringAtIndex(0);
|
|
if (completion_str != NULL && *completion_str != '\0')
|
|
{
|
|
el_insertstr (m_edit_line, completion_str);
|
|
return CC_REDISPLAY;
|
|
}
|
|
|
|
if (num_completions > 1)
|
|
{
|
|
const char *comment = "\nAvailable completions:";
|
|
|
|
int num_elements = num_completions + 1;
|
|
OutWrite(comment, strlen (comment), NO_ASYNC);
|
|
if (num_completions < page_size)
|
|
{
|
|
for (int i = 1; i < num_elements; i++)
|
|
{
|
|
completion_str = completions.GetStringAtIndex(i);
|
|
OutWrite("\n\t", 2, NO_ASYNC);
|
|
OutWrite(completion_str, strlen (completion_str), NO_ASYNC);
|
|
}
|
|
OutWrite ("\n", 1, NO_ASYNC);
|
|
}
|
|
else
|
|
{
|
|
int cur_pos = 1;
|
|
char reply;
|
|
int got_char;
|
|
while (cur_pos < num_elements)
|
|
{
|
|
int endpoint = cur_pos + page_size;
|
|
if (endpoint > num_elements)
|
|
endpoint = num_elements;
|
|
for (; cur_pos < endpoint; cur_pos++)
|
|
{
|
|
completion_str = completions.GetStringAtIndex(cur_pos);
|
|
OutWrite("\n\t", 2, NO_ASYNC);
|
|
OutWrite(completion_str, strlen (completion_str), NO_ASYNC);
|
|
}
|
|
|
|
if (cur_pos >= num_elements)
|
|
{
|
|
OutWrite("\n", 1, NO_ASYNC);
|
|
break;
|
|
}
|
|
|
|
OutWrite("\nMore (Y/n/a): ", strlen ("\nMore (Y/n/a): "), NO_ASYNC);
|
|
reply = 'n';
|
|
got_char = el_getc(m_edit_line, &reply);
|
|
if (got_char == -1 || reply == 'n')
|
|
break;
|
|
if (reply == 'a')
|
|
page_size = num_elements - cur_pos;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
if (num_completions == 0)
|
|
return CC_REFRESH_BEEP;
|
|
else
|
|
return CC_REDISPLAY;
|
|
}
|
|
|
|
IOChannel::IOChannel
|
|
(
|
|
FILE *editline_in,
|
|
FILE *editline_out,
|
|
FILE *out,
|
|
FILE *err,
|
|
Driver *driver
|
|
) :
|
|
SBBroadcaster ("IOChannel"),
|
|
m_output_mutex (),
|
|
m_enter_elgets_time (),
|
|
m_driver (driver),
|
|
m_read_thread (LLDB_INVALID_HOST_THREAD),
|
|
m_read_thread_should_exit (false),
|
|
m_out_file (out),
|
|
m_err_file (err),
|
|
m_command_queue (),
|
|
m_completion_key ("\t"),
|
|
m_edit_line (::el_init (SBHostOS::GetProgramFileSpec().GetFilename(), editline_in, editline_out, editline_out)),
|
|
m_history (history_init()),
|
|
m_history_event(),
|
|
m_getting_command (false),
|
|
m_expecting_prompt (false),
|
|
m_prompt_str (),
|
|
m_refresh_request_pending (false)
|
|
{
|
|
assert (m_edit_line);
|
|
::el_set (m_edit_line, EL_PROMPT, el_prompt);
|
|
::el_set (m_edit_line, EL_EDITOR, "emacs");
|
|
::el_set (m_edit_line, EL_HIST, history, m_history);
|
|
|
|
el_set (m_edit_line, EL_ADDFN, "lldb_complete",
|
|
"LLDB completion function",
|
|
IOChannel::ElCompletionFn);
|
|
el_set (m_edit_line, EL_BIND, m_completion_key, "lldb_complete", NULL);
|
|
el_set (m_edit_line, EL_BIND, "^r", "em-inc-search-prev", NULL); // Cycle through backwards search, entering string
|
|
el_set (m_edit_line, EL_BIND, "^w", "ed-delete-prev-word", NULL); // Delete previous word, behave like bash does.
|
|
el_set (m_edit_line, EL_CLIENTDATA, this);
|
|
|
|
// Source $PWD/.editrc then $HOME/.editrc
|
|
::el_source (m_edit_line, NULL);
|
|
|
|
assert (m_history);
|
|
::history (m_history, &m_history_event, H_SETSIZE, 800);
|
|
::history (m_history, &m_history_event, H_SETUNIQUE, 1);
|
|
// Load history
|
|
HistorySaveLoad (false);
|
|
|
|
// Set up mutex to make sure OutErr, OutWrite and RefreshPrompt do not interfere
|
|
// with each other when writing.
|
|
|
|
int error;
|
|
::pthread_mutexattr_t attr;
|
|
error = ::pthread_mutexattr_init (&attr);
|
|
assert (error == 0);
|
|
error = ::pthread_mutexattr_settype (&attr, PTHREAD_MUTEX_RECURSIVE);
|
|
assert (error == 0);
|
|
error = ::pthread_mutex_init (&m_output_mutex, &attr);
|
|
assert (error == 0);
|
|
error = ::pthread_mutexattr_destroy (&attr);
|
|
assert (error == 0);
|
|
|
|
// Initialize time that ::el_gets was last called.
|
|
|
|
m_enter_elgets_time.tv_sec = 0;
|
|
m_enter_elgets_time.tv_usec = 0;
|
|
}
|
|
|
|
IOChannel::~IOChannel ()
|
|
{
|
|
// Save history
|
|
HistorySaveLoad (true);
|
|
|
|
if (m_history != NULL)
|
|
{
|
|
::history_end (m_history);
|
|
m_history = NULL;
|
|
}
|
|
|
|
if (m_edit_line != NULL)
|
|
{
|
|
::el_end (m_edit_line);
|
|
m_edit_line = NULL;
|
|
}
|
|
|
|
::pthread_mutex_destroy (&m_output_mutex);
|
|
}
|
|
|
|
void
|
|
IOChannel::HistorySaveLoad (bool save)
|
|
{
|
|
if (m_history != NULL)
|
|
{
|
|
char history_path[PATH_MAX];
|
|
::snprintf (history_path, sizeof(history_path), "~/.%s-history", SBHostOS::GetProgramFileSpec().GetFilename());
|
|
if ((size_t)SBFileSpec::ResolvePath (history_path, history_path, sizeof(history_path)) < sizeof(history_path) - 1)
|
|
{
|
|
const char *path_ptr = history_path;
|
|
if (save)
|
|
::history (m_history, &m_history_event, H_SAVE, path_ptr);
|
|
else
|
|
::history (m_history, &m_history_event, H_LOAD, path_ptr);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
IOChannel::LibeditOutputBytesReceived (void *baton, const void *src, size_t src_len)
|
|
{
|
|
// Make this a member variable.
|
|
// static std::string prompt_str;
|
|
IOChannel *io_channel = (IOChannel *) baton;
|
|
IOLocker locker (io_channel->m_output_mutex);
|
|
const char *bytes = (const char *) src;
|
|
|
|
if (io_channel->IsGettingCommand() && io_channel->m_expecting_prompt)
|
|
{
|
|
io_channel->m_prompt_str.append (bytes, src_len);
|
|
// Log this to make sure the prompt is really what you think it is.
|
|
if (io_channel->m_prompt_str.find (el_prompt(io_channel->m_edit_line)) == 0)
|
|
{
|
|
io_channel->m_expecting_prompt = false;
|
|
io_channel->m_refresh_request_pending = false;
|
|
io_channel->OutWrite (io_channel->m_prompt_str.c_str(),
|
|
io_channel->m_prompt_str.size(), NO_ASYNC);
|
|
io_channel->m_prompt_str.clear();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (io_channel->m_prompt_str.size() > 0)
|
|
io_channel->m_prompt_str.clear();
|
|
std::string tmp_str (bytes, src_len);
|
|
if (tmp_str.find (el_prompt (io_channel->m_edit_line)) == 0)
|
|
io_channel->m_refresh_request_pending = false;
|
|
io_channel->OutWrite (bytes, src_len, NO_ASYNC);
|
|
}
|
|
}
|
|
|
|
bool
|
|
IOChannel::LibeditGetInput (std::string &new_line)
|
|
{
|
|
if (m_edit_line != NULL)
|
|
{
|
|
int line_len = 0;
|
|
|
|
// Set boolean indicating whether or not el_gets is trying to get input (i.e. whether or not to attempt
|
|
// to refresh the prompt after writing data).
|
|
SetGettingCommand (true);
|
|
m_expecting_prompt = true;
|
|
|
|
// Call el_gets to prompt the user and read the user's input.
|
|
const char *line = ::el_gets (m_edit_line, &line_len);
|
|
|
|
// Re-set the boolean indicating whether or not el_gets is trying to get input.
|
|
SetGettingCommand (false);
|
|
|
|
if (line)
|
|
{
|
|
// strip any newlines off the end of the string...
|
|
while (line_len > 0 && (line[line_len - 1] == '\n' || line[line_len - 1] == '\r'))
|
|
--line_len;
|
|
if (line_len > 0)
|
|
{
|
|
::history (m_history, &m_history_event, H_ENTER, line);
|
|
new_line.assign (line, line_len); // Omit the newline
|
|
}
|
|
else
|
|
{
|
|
// Someone just hit ENTER, return the empty string
|
|
new_line.clear();
|
|
}
|
|
// Return true to indicate success even if a string is empty
|
|
return true;
|
|
}
|
|
}
|
|
// Return false to indicate failure. This can happen when the file handle
|
|
// is closed (EOF).
|
|
new_line.clear();
|
|
return false;
|
|
}
|
|
|
|
void *
|
|
IOChannel::IOReadThread (void *ptr)
|
|
{
|
|
IOChannel *myself = static_cast<IOChannel *> (ptr);
|
|
myself->Run();
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
IOChannel::Run ()
|
|
{
|
|
SBListener listener("IOChannel::Run");
|
|
std::string new_line;
|
|
|
|
SBBroadcaster interpreter_broadcaster (m_driver->GetDebugger().GetCommandInterpreter().GetBroadcaster());
|
|
listener.StartListeningForEvents (interpreter_broadcaster,
|
|
SBCommandInterpreter::eBroadcastBitResetPrompt |
|
|
SBCommandInterpreter::eBroadcastBitThreadShouldExit |
|
|
SBCommandInterpreter::eBroadcastBitQuitCommandReceived);
|
|
|
|
listener.StartListeningForEvents (*this,
|
|
IOChannel::eBroadcastBitThreadShouldExit);
|
|
|
|
listener.StartListeningForEvents (*m_driver,
|
|
Driver::eBroadcastBitReadyForInput |
|
|
Driver::eBroadcastBitThreadShouldExit);
|
|
|
|
// Let anyone know that the IO channel is up and listening and ready for events
|
|
BroadcastEventByType (eBroadcastBitThreadDidStart);
|
|
bool done = false;
|
|
while (!done)
|
|
{
|
|
SBEvent event;
|
|
|
|
listener.WaitForEvent (UINT32_MAX, event);
|
|
if (!event.IsValid())
|
|
continue;
|
|
|
|
const uint32_t event_type = event.GetType();
|
|
|
|
if (event.GetBroadcaster().IsValid())
|
|
{
|
|
if (event.BroadcasterMatchesPtr (m_driver))
|
|
{
|
|
if (event_type & Driver::eBroadcastBitReadyForInput)
|
|
{
|
|
std::string line;
|
|
|
|
if (CommandQueueIsEmpty())
|
|
{
|
|
if (LibeditGetInput(line) == false)
|
|
{
|
|
// EOF or some other file error occurred
|
|
done = true;
|
|
continue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
GetCommandFromQueue (line);
|
|
}
|
|
|
|
// TO BE DONE: FIGURE OUT WHICH COMMANDS SHOULD NOT BE REPEATED IF USER PRESSES PLAIN 'RETURN'
|
|
// AND TAKE CARE OF THAT HERE.
|
|
|
|
SBEvent line_event(IOChannel::eBroadcastBitHasUserInput,
|
|
line.c_str(),
|
|
line.size());
|
|
BroadcastEvent (line_event);
|
|
}
|
|
else if (event_type & Driver::eBroadcastBitThreadShouldExit)
|
|
{
|
|
done = true;
|
|
continue;
|
|
}
|
|
}
|
|
else if (event.BroadcasterMatchesRef (interpreter_broadcaster))
|
|
{
|
|
switch (event_type)
|
|
{
|
|
case SBCommandInterpreter::eBroadcastBitResetPrompt:
|
|
{
|
|
const char *new_prompt = SBEvent::GetCStringFromEvent (event);
|
|
if (new_prompt)
|
|
g_prompt_map[m_edit_line] = new_prompt;
|
|
}
|
|
break;
|
|
|
|
case SBCommandInterpreter::eBroadcastBitThreadShouldExit:
|
|
case SBCommandInterpreter::eBroadcastBitQuitCommandReceived:
|
|
done = true;
|
|
break;
|
|
}
|
|
}
|
|
else if (event.BroadcasterMatchesPtr (this))
|
|
{
|
|
if (event_type & IOChannel::eBroadcastBitThreadShouldExit)
|
|
{
|
|
done = true;
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
BroadcastEventByType (IOChannel::eBroadcastBitThreadDidExit);
|
|
m_driver = NULL;
|
|
m_read_thread = NULL;
|
|
}
|
|
|
|
bool
|
|
IOChannel::Start ()
|
|
{
|
|
if (IS_VALID_LLDB_HOST_THREAD(m_read_thread))
|
|
return true;
|
|
|
|
m_read_thread = SBHostOS::ThreadCreate ("<lldb.driver.commandline_io>", IOChannel::IOReadThread, this,
|
|
NULL);
|
|
|
|
return (IS_VALID_LLDB_HOST_THREAD(m_read_thread));
|
|
}
|
|
|
|
bool
|
|
IOChannel::Stop ()
|
|
{
|
|
if (!IS_VALID_LLDB_HOST_THREAD(m_read_thread))
|
|
return true;
|
|
|
|
BroadcastEventByType (eBroadcastBitThreadShouldExit);
|
|
|
|
// Don't call Host::ThreadCancel since el_gets won't respond to this
|
|
// function call -- the thread will just die and all local variables in
|
|
// IOChannel::Run() won't get destructed down which is bad since there is
|
|
// a local listener holding onto broadcasters... To ensure proper shutdown,
|
|
// a ^D (control-D) sequence (0x04) should be written to other end of the
|
|
// the "in" file handle that was passed into the contructor as closing the
|
|
// file handle doesn't seem to make el_gets() exit....
|
|
return SBHostOS::ThreadJoin (m_read_thread, NULL, NULL);
|
|
}
|
|
|
|
void
|
|
IOChannel::RefreshPrompt ()
|
|
{
|
|
// If we are not in the middle of getting input from the user, there is no need to
|
|
// refresh the prompt.
|
|
IOLocker locker (m_output_mutex);
|
|
if (! IsGettingCommand())
|
|
return;
|
|
|
|
// If we haven't finished writing the prompt, there's no need to refresh it.
|
|
if (m_expecting_prompt)
|
|
return;
|
|
|
|
if (m_refresh_request_pending)
|
|
return;
|
|
|
|
::el_set (m_edit_line, EL_REFRESH);
|
|
m_refresh_request_pending = true;
|
|
}
|
|
|
|
void
|
|
IOChannel::OutWrite (const char *buffer, size_t len, bool asynchronous)
|
|
{
|
|
if (len == 0)
|
|
return;
|
|
|
|
// We're in the process of exiting -- IOChannel::Run() has already completed
|
|
// and set m_driver to NULL - it is time for us to leave now. We might not
|
|
// print the final ^D to stdout in this case. We need to do some re-work on
|
|
// how the I/O streams are managed at some point.
|
|
if (m_driver == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Use the mutex to make sure OutWrite and ErrWrite do not interfere with each other's output.
|
|
IOLocker locker (m_output_mutex);
|
|
if (m_driver->EditlineReaderIsTop() && asynchronous)
|
|
::fwrite (undo_prompt_string, 1, 4, m_out_file);
|
|
::fwrite (buffer, 1, len, m_out_file);
|
|
if (asynchronous)
|
|
m_driver->GetDebugger().NotifyTopInputReader (eInputReaderAsynchronousOutputWritten);
|
|
}
|
|
|
|
void
|
|
IOChannel::ErrWrite (const char *buffer, size_t len, bool asynchronous)
|
|
{
|
|
if (len == 0)
|
|
return;
|
|
|
|
// Use the mutex to make sure OutWrite and ErrWrite do not interfere with each other's output.
|
|
IOLocker locker (m_output_mutex);
|
|
if (asynchronous)
|
|
::fwrite (undo_prompt_string, 1, 4, m_err_file);
|
|
::fwrite (buffer, 1, len, m_err_file);
|
|
if (asynchronous)
|
|
m_driver->GetDebugger().NotifyTopInputReader (eInputReaderAsynchronousOutputWritten);
|
|
}
|
|
|
|
void
|
|
IOChannel::AddCommandToQueue (const char *command)
|
|
{
|
|
m_command_queue.push (std::string(command));
|
|
}
|
|
|
|
bool
|
|
IOChannel::GetCommandFromQueue (std::string &cmd)
|
|
{
|
|
if (m_command_queue.empty())
|
|
return false;
|
|
cmd.swap(m_command_queue.front());
|
|
m_command_queue.pop ();
|
|
return true;
|
|
}
|
|
|
|
int
|
|
IOChannel::CommandQueueSize () const
|
|
{
|
|
return m_command_queue.size();
|
|
}
|
|
|
|
void
|
|
IOChannel::ClearCommandQueue ()
|
|
{
|
|
while (!m_command_queue.empty())
|
|
m_command_queue.pop();
|
|
}
|
|
|
|
bool
|
|
IOChannel::CommandQueueIsEmpty () const
|
|
{
|
|
return m_command_queue.empty();
|
|
}
|
|
|
|
bool
|
|
IOChannel::IsGettingCommand () const
|
|
{
|
|
return m_getting_command;
|
|
}
|
|
|
|
void
|
|
IOChannel::SetGettingCommand (bool new_value)
|
|
{
|
|
m_getting_command = new_value;
|
|
}
|
|
|
|
IOLocker::IOLocker (pthread_mutex_t &mutex) :
|
|
m_mutex_ptr (&mutex)
|
|
{
|
|
if (m_mutex_ptr)
|
|
::pthread_mutex_lock (m_mutex_ptr);
|
|
|
|
}
|
|
|
|
IOLocker::~IOLocker ()
|
|
{
|
|
if (m_mutex_ptr)
|
|
::pthread_mutex_unlock (m_mutex_ptr);
|
|
}
|