it logs the function calls, their arguments and the return values. This is not complete or polished, but I am committing it now, at the request of someone who really wants to use it, even though it's not really done. It currently does not attempt to log all the functions, just the most important ones. I will be making further adjustments to the API logging code over the next few days/weeks. (Suggestions for improvements are welcome). Update the Python build scripts to re-build the swig C++ file whenever the python-extensions.swig file is modified. Correct the help for 'log enable' command (give it the correct number & type of arguments). llvm-svn: 117349
359 lines
10 KiB
C++
359 lines
10 KiB
C++
//===-- InputReader.cpp -----------------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include <string>
|
|
|
|
#include "lldb/Core/InputReader.h"
|
|
#include "lldb/Core/Debugger.h"
|
|
|
|
using namespace lldb;
|
|
using namespace lldb_private;
|
|
|
|
InputReader::InputReader (Debugger &debugger) :
|
|
m_debugger (debugger),
|
|
m_callback (NULL),
|
|
m_callback_baton (NULL),
|
|
m_end_token (),
|
|
m_granularity (eInputReaderGranularityInvalid),
|
|
m_done (true),
|
|
m_echo (true),
|
|
m_active (false),
|
|
m_reader_done (false)
|
|
{
|
|
}
|
|
|
|
InputReader::~InputReader ()
|
|
{
|
|
}
|
|
|
|
Error
|
|
InputReader::Initialize
|
|
(
|
|
Callback callback,
|
|
void *baton,
|
|
lldb::InputReaderGranularity granularity,
|
|
const char *end_token,
|
|
const char *prompt,
|
|
bool echo
|
|
)
|
|
{
|
|
Error err;
|
|
m_callback = callback;
|
|
m_callback_baton = baton,
|
|
m_granularity = granularity;
|
|
if (end_token != NULL)
|
|
m_end_token = end_token;
|
|
if (prompt != NULL)
|
|
m_prompt = prompt;
|
|
m_done = true;
|
|
m_echo = echo;
|
|
|
|
if (m_granularity == eInputReaderGranularityInvalid)
|
|
{
|
|
err.SetErrorString ("Invalid read token size: Reader must be initialized with a token size other than 'eInputReaderGranularityInvalid'.");
|
|
}
|
|
else
|
|
if (end_token != NULL && granularity != eInputReaderGranularityInvalid)
|
|
{
|
|
if (granularity == eInputReaderGranularityByte)
|
|
{
|
|
// Check to see if end_token is longer than one byte.
|
|
|
|
if (strlen (end_token) > 1)
|
|
{
|
|
err.SetErrorString ("Invalid end token: End token cannot be larger than specified token size (byte).");
|
|
}
|
|
}
|
|
else if (granularity == eInputReaderGranularityWord)
|
|
{
|
|
// Check to see if m_end_token contains any white space (i.e. is multiple words).
|
|
|
|
const char *white_space = " \t\n";
|
|
size_t pos = m_end_token.find_first_of (white_space);
|
|
if (pos != std::string::npos)
|
|
{
|
|
err.SetErrorString ("Invalid end token: End token cannot be larger than specified token size (word).");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Check to see if m_end_token contains any newlines; cannot handle multi-line end tokens.
|
|
|
|
size_t pos = m_end_token.find_first_of ('\n');
|
|
if (pos != std::string::npos)
|
|
{
|
|
err.SetErrorString ("Invalid end token: End token cannot contain a newline.");
|
|
}
|
|
}
|
|
}
|
|
|
|
m_done = err.Fail();
|
|
|
|
return err;
|
|
}
|
|
|
|
size_t
|
|
InputReader::HandleRawBytes (const char *bytes, size_t bytes_len)
|
|
{
|
|
const char *end_token = NULL;
|
|
|
|
if (m_end_token.empty() == false)
|
|
{
|
|
end_token = ::strstr (bytes, m_end_token.c_str());
|
|
if (end_token >= bytes + bytes_len)
|
|
end_token = NULL;
|
|
}
|
|
|
|
const char *p = bytes;
|
|
const char *end = bytes + bytes_len;
|
|
|
|
switch (m_granularity)
|
|
{
|
|
case eInputReaderGranularityInvalid:
|
|
break;
|
|
|
|
case eInputReaderGranularityByte:
|
|
while (p < end)
|
|
{
|
|
if (end_token == p)
|
|
{
|
|
p += m_end_token.size();
|
|
SetIsDone(true);
|
|
break;
|
|
}
|
|
|
|
if (m_callback (m_callback_baton, *this, eInputReaderGotToken, p, 1) == 0)
|
|
break;
|
|
++p;
|
|
if (IsDone())
|
|
break;
|
|
}
|
|
// Return how many bytes were handled.
|
|
return p - bytes;
|
|
break;
|
|
|
|
|
|
case eInputReaderGranularityWord:
|
|
{
|
|
char quote = '\0';
|
|
const char *word_start = NULL;
|
|
bool send_word = false;
|
|
for (; p < end; ++p, send_word = false)
|
|
{
|
|
if (end_token && end_token == p)
|
|
{
|
|
p += m_end_token.size();
|
|
SetIsDone(true);
|
|
break;
|
|
}
|
|
|
|
const char ch = *p;
|
|
if (isspace(ch) && (!quote || (quote == ch && p[-1] != '\\')))
|
|
{
|
|
// We have a space character or the terminating quote
|
|
send_word = word_start != NULL;
|
|
quote = '\0';
|
|
}
|
|
else if (quote)
|
|
{
|
|
// We are in the middle of a quoted character
|
|
continue;
|
|
}
|
|
else if (ch == '"' || ch == '\'' || ch == '`')
|
|
quote = ch;
|
|
else if (word_start == NULL)
|
|
{
|
|
// We have the first character in a word
|
|
word_start = p;
|
|
}
|
|
|
|
if (send_word)
|
|
{
|
|
const size_t word_len = p - word_start;
|
|
size_t bytes_handled = m_callback (m_callback_baton,
|
|
*this,
|
|
eInputReaderGotToken,
|
|
word_start,
|
|
word_len);
|
|
|
|
if (bytes_handled != word_len)
|
|
return word_start - bytes + bytes_handled;
|
|
|
|
if (IsDone())
|
|
return p - bytes;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
case eInputReaderGranularityLine:
|
|
{
|
|
const char *line_start = bytes;
|
|
const char *end_line = NULL;
|
|
while (p < end)
|
|
{
|
|
const char ch = *p;
|
|
if (ch == '\n' || ch == '\r')
|
|
{
|
|
size_t line_length = p - line_start;
|
|
// Now skip the newline character
|
|
++p;
|
|
// Skip a complete DOS newline if we run into one
|
|
if (ch == 0xd && p < end && *p == 0xa)
|
|
++p;
|
|
|
|
if (line_start <= end_token && end_token < line_start + line_length)
|
|
{
|
|
SetIsDone(true);
|
|
m_callback (m_callback_baton,
|
|
*this,
|
|
eInputReaderGotToken,
|
|
line_start,
|
|
end_token - line_start);
|
|
|
|
return p - bytes;
|
|
}
|
|
|
|
size_t bytes_handled = m_callback (m_callback_baton,
|
|
*this,
|
|
eInputReaderGotToken,
|
|
line_start,
|
|
line_length);
|
|
|
|
end_line = p;
|
|
|
|
if (bytes_handled != line_length)
|
|
{
|
|
// The input reader wasn't able to handle all the data
|
|
return line_start - bytes + bytes_handled;
|
|
}
|
|
|
|
|
|
if (IsDone())
|
|
return p - bytes;
|
|
|
|
line_start = p;
|
|
}
|
|
else
|
|
{
|
|
++p;
|
|
}
|
|
}
|
|
|
|
if (end_line)
|
|
return end_line - bytes;
|
|
}
|
|
break;
|
|
|
|
|
|
case eInputReaderGranularityAll:
|
|
{
|
|
// Nothing should be handle unless we see our end token
|
|
if (end_token)
|
|
{
|
|
size_t length = end_token - bytes;
|
|
size_t bytes_handled = m_callback (m_callback_baton,
|
|
*this,
|
|
eInputReaderGotToken,
|
|
bytes,
|
|
length);
|
|
m_done = true;
|
|
|
|
p += bytes_handled + m_end_token.size();
|
|
|
|
// Consume any white space, such as newlines, beyond the end token
|
|
|
|
while (p < end && isspace(*p))
|
|
++p;
|
|
|
|
if (bytes_handled != length)
|
|
return bytes_handled;
|
|
else
|
|
{
|
|
return p - bytes;
|
|
//return bytes_handled + m_end_token.size();
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
const char *
|
|
InputReader::GetPrompt () const
|
|
{
|
|
if (!m_prompt.empty())
|
|
return m_prompt.c_str();
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
InputReader::RefreshPrompt ()
|
|
{
|
|
if (!m_prompt.empty())
|
|
{
|
|
FILE *out_fh = m_debugger.GetOutputFileHandle();
|
|
if (out_fh)
|
|
::fprintf (out_fh, "%s", m_prompt.c_str());
|
|
}
|
|
}
|
|
|
|
void
|
|
InputReader::Notify (InputReaderAction notification)
|
|
{
|
|
switch (notification)
|
|
{
|
|
case eInputReaderActivate:
|
|
case eInputReaderReactivate:
|
|
m_active = true;
|
|
m_reader_done.SetValue(false, eBroadcastAlways);
|
|
break;
|
|
|
|
case eInputReaderDeactivate:
|
|
case eInputReaderDone:
|
|
m_active = false;
|
|
break;
|
|
|
|
case eInputReaderGotToken:
|
|
return; // We don't notify the tokens here, it is done in HandleRawBytes
|
|
}
|
|
if (m_callback)
|
|
m_callback (m_callback_baton, *this, notification, NULL, 0);
|
|
if (notification == eInputReaderDone)
|
|
m_reader_done.SetValue(true, eBroadcastAlways);
|
|
}
|
|
|
|
void
|
|
InputReader::WaitOnReaderIsDone ()
|
|
{
|
|
m_reader_done.WaitForValueEqualTo (true);
|
|
}
|
|
|
|
const char *
|
|
InputReader::GranularityAsCString (lldb::InputReaderGranularity granularity)
|
|
{
|
|
switch (granularity)
|
|
{
|
|
case eInputReaderGranularityInvalid: return "invalid";
|
|
case eInputReaderGranularityByte: return "byte";
|
|
case eInputReaderGranularityWord: return "word";
|
|
case eInputReaderGranularityLine: return "line";
|
|
case eInputReaderGranularityAll: return "all";
|
|
}
|
|
|
|
static char unknown_state_string[64];
|
|
snprintf(unknown_state_string, sizeof (unknown_state_string), "InputReaderGranularity = %i", granularity);
|
|
return unknown_state_string;
|
|
}
|
|
|