The implementation of categories is now synchronization safe
Code cleanup:
- The Format Manager implementation is now split between two files: FormatClasses.{h|cpp} where the
actual formatter classes (ValueFormat, SummaryFormat, ...) are implemented and
FormatManager.{h|cpp} where the infrastructure classes (FormatNavigator, FormatManager, ...)
are contained. The wrapper code always remains in Debugger.{h|cpp}
- Several leftover fields, methods and comments from previous design choices have been removed
type category subcommands (enable, disable, delete) now can take a list of category names as input
- for type category enable, saying "enable A B C" is the same as saying
enable C
enable B
enable A
(the ordering is relevant in enabling categories, and it is expected that a user typing
enable A B C wants to look into category A, then into B, then into C and not the other
way round)
- for the other two commands, the order is not really relevant (however, the same inverted ordering
is used for consistency)
llvm-svn: 135494
2011-07-20 02:03:25 +08:00
|
|
|
//===-- FormatClasses.cpp ----------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
// C Includes
|
|
|
|
|
|
|
|
// C++ Includes
|
|
|
|
#include <ostream>
|
|
|
|
|
|
|
|
// Other libraries and framework includes
|
|
|
|
|
|
|
|
// Project includes
|
|
|
|
#include "lldb/lldb-public.h"
|
|
|
|
#include "lldb/lldb-enumerations.h"
|
|
|
|
|
|
|
|
#include "lldb/Core/Debugger.h"
|
|
|
|
#include "lldb/Core/FormatClasses.h"
|
|
|
|
#include "lldb/Core/StreamString.h"
|
2011-07-24 08:14:56 +08:00
|
|
|
#include "lldb/Interpreter/CommandInterpreter.h"
|
The implementation of categories is now synchronization safe
Code cleanup:
- The Format Manager implementation is now split between two files: FormatClasses.{h|cpp} where the
actual formatter classes (ValueFormat, SummaryFormat, ...) are implemented and
FormatManager.{h|cpp} where the infrastructure classes (FormatNavigator, FormatManager, ...)
are contained. The wrapper code always remains in Debugger.{h|cpp}
- Several leftover fields, methods and comments from previous design choices have been removed
type category subcommands (enable, disable, delete) now can take a list of category names as input
- for type category enable, saying "enable A B C" is the same as saying
enable C
enable B
enable A
(the ordering is relevant in enabling categories, and it is expected that a user typing
enable A B C wants to look into category A, then into B, then into C and not the other
way round)
- for the other two commands, the order is not really relevant (however, the same inverted ordering
is used for consistency)
llvm-svn: 135494
2011-07-20 02:03:25 +08:00
|
|
|
#include "lldb/Symbol/ClangASTType.h"
|
|
|
|
#include "lldb/Target/StackFrame.h"
|
2011-07-24 08:14:56 +08:00
|
|
|
#include "lldb/Target/Target.h"
|
The implementation of categories is now synchronization safe
Code cleanup:
- The Format Manager implementation is now split between two files: FormatClasses.{h|cpp} where the
actual formatter classes (ValueFormat, SummaryFormat, ...) are implemented and
FormatManager.{h|cpp} where the infrastructure classes (FormatNavigator, FormatManager, ...)
are contained. The wrapper code always remains in Debugger.{h|cpp}
- Several leftover fields, methods and comments from previous design choices have been removed
type category subcommands (enable, disable, delete) now can take a list of category names as input
- for type category enable, saying "enable A B C" is the same as saying
enable C
enable B
enable A
(the ordering is relevant in enabling categories, and it is expected that a user typing
enable A B C wants to look into category A, then into B, then into C and not the other
way round)
- for the other two commands, the order is not really relevant (however, the same inverted ordering
is used for consistency)
llvm-svn: 135494
2011-07-20 02:03:25 +08:00
|
|
|
|
|
|
|
using namespace lldb;
|
|
|
|
using namespace lldb_private;
|
|
|
|
|
|
|
|
std::string
|
|
|
|
ValueFormat::FormatObject(lldb::ValueObjectSP object)
|
|
|
|
{
|
|
|
|
if (!object.get())
|
|
|
|
return "NULL";
|
|
|
|
|
|
|
|
StreamString sstr;
|
|
|
|
|
|
|
|
if (ClangASTType::DumpTypeValue (object->GetClangAST(), // The clang AST
|
|
|
|
object->GetClangType(), // The clang type to display
|
|
|
|
&sstr,
|
|
|
|
m_format, // Format to display this type with
|
|
|
|
object->GetDataExtractor(), // Data to extract from
|
|
|
|
0, // Byte offset into "data"
|
|
|
|
object->GetByteSize(), // Byte size of item in "data"
|
|
|
|
object->GetBitfieldBitSize(), // Bitfield bit size
|
|
|
|
object->GetBitfieldBitOffset())) // Bitfield bit offset
|
|
|
|
return (sstr.GetString());
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return ("unsufficient data for value");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
StringSummaryFormat::FormatObject(lldb::ValueObjectSP object)
|
|
|
|
{
|
|
|
|
if (!object.get())
|
|
|
|
return "NULL";
|
|
|
|
|
|
|
|
StreamString s;
|
|
|
|
ExecutionContext exe_ctx;
|
|
|
|
object->GetExecutionContextScope()->CalculateExecutionContext(exe_ctx);
|
|
|
|
SymbolContext sc;
|
|
|
|
if (exe_ctx.frame)
|
|
|
|
sc = exe_ctx.frame->GetSymbolContext(lldb::eSymbolContextEverything);
|
|
|
|
|
|
|
|
if (m_show_members_oneliner)
|
|
|
|
{
|
2011-07-22 08:16:08 +08:00
|
|
|
ValueObjectSP synth_vobj = object->GetSyntheticValue(lldb::eUseSyntheticFilter);
|
|
|
|
const uint32_t num_children = synth_vobj->GetNumChildren();
|
The implementation of categories is now synchronization safe
Code cleanup:
- The Format Manager implementation is now split between two files: FormatClasses.{h|cpp} where the
actual formatter classes (ValueFormat, SummaryFormat, ...) are implemented and
FormatManager.{h|cpp} where the infrastructure classes (FormatNavigator, FormatManager, ...)
are contained. The wrapper code always remains in Debugger.{h|cpp}
- Several leftover fields, methods and comments from previous design choices have been removed
type category subcommands (enable, disable, delete) now can take a list of category names as input
- for type category enable, saying "enable A B C" is the same as saying
enable C
enable B
enable A
(the ordering is relevant in enabling categories, and it is expected that a user typing
enable A B C wants to look into category A, then into B, then into C and not the other
way round)
- for the other two commands, the order is not really relevant (however, the same inverted ordering
is used for consistency)
llvm-svn: 135494
2011-07-20 02:03:25 +08:00
|
|
|
if (num_children)
|
|
|
|
{
|
|
|
|
s.PutChar('(');
|
|
|
|
|
|
|
|
for (uint32_t idx=0; idx<num_children; ++idx)
|
|
|
|
{
|
2011-07-22 08:16:08 +08:00
|
|
|
lldb::ValueObjectSP child_sp(synth_vobj->GetChildAtIndex(idx, true));
|
The implementation of categories is now synchronization safe
Code cleanup:
- The Format Manager implementation is now split between two files: FormatClasses.{h|cpp} where the
actual formatter classes (ValueFormat, SummaryFormat, ...) are implemented and
FormatManager.{h|cpp} where the infrastructure classes (FormatNavigator, FormatManager, ...)
are contained. The wrapper code always remains in Debugger.{h|cpp}
- Several leftover fields, methods and comments from previous design choices have been removed
type category subcommands (enable, disable, delete) now can take a list of category names as input
- for type category enable, saying "enable A B C" is the same as saying
enable C
enable B
enable A
(the ordering is relevant in enabling categories, and it is expected that a user typing
enable A B C wants to look into category A, then into B, then into C and not the other
way round)
- for the other two commands, the order is not really relevant (however, the same inverted ordering
is used for consistency)
llvm-svn: 135494
2011-07-20 02:03:25 +08:00
|
|
|
if (child_sp.get())
|
|
|
|
{
|
|
|
|
if (idx)
|
|
|
|
s.PutCString(", ");
|
|
|
|
s.PutCString(child_sp.get()->GetName().AsCString());
|
|
|
|
s.PutChar('=');
|
|
|
|
s.PutCString(child_sp.get()->GetPrintableRepresentation());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s.PutChar(')');
|
|
|
|
|
|
|
|
return s.GetString();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return "";
|
|
|
|
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (Debugger::FormatPrompt(m_format.c_str(), &sc, &exe_ctx, &sc.line_entry.range.GetBaseAddress(), s, NULL, object.get()))
|
|
|
|
return s.GetString();
|
|
|
|
else
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
StringSummaryFormat::GetDescription()
|
|
|
|
{
|
|
|
|
StreamString sstr;
|
|
|
|
sstr.Printf ("`%s`%s%s%s%s%s%s", m_format.c_str(),
|
|
|
|
m_cascades ? "" : " (not cascading)",
|
|
|
|
m_dont_show_children ? "" : " (show children)",
|
|
|
|
m_dont_show_value ? " (hide value)" : "",
|
|
|
|
m_show_members_oneliner ? " (one-line printout)" : "",
|
|
|
|
m_skip_pointers ? " (skip pointers)" : "",
|
|
|
|
m_skip_references ? " (skip references)" : "");
|
|
|
|
return sstr.GetString();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
ScriptSummaryFormat::FormatObject(lldb::ValueObjectSP object)
|
|
|
|
{
|
|
|
|
return std::string(ScriptInterpreterPython::CallPythonScriptFunction(m_function_name.c_str(),
|
|
|
|
object).c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
ScriptSummaryFormat::GetDescription()
|
|
|
|
{
|
|
|
|
StreamString sstr;
|
|
|
|
sstr.Printf ("%s%s%s%s%s%s\n%s", m_cascades ? "" : " (not cascading)",
|
|
|
|
m_dont_show_children ? "" : " (show children)",
|
|
|
|
m_dont_show_value ? " (hide value)" : "",
|
|
|
|
m_show_members_oneliner ? " (one-line printout)" : "",
|
|
|
|
m_skip_pointers ? " (skip pointers)" : "",
|
|
|
|
m_skip_references ? " (skip references)" : "",
|
|
|
|
m_python_script.c_str());
|
|
|
|
return sstr.GetString();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-07-22 08:16:08 +08:00
|
|
|
std::string
|
|
|
|
SyntheticFilter::GetDescription()
|
|
|
|
{
|
|
|
|
StreamString sstr;
|
|
|
|
sstr.Printf("%s%s%s {\n",
|
|
|
|
m_cascades ? "" : " (not cascading)",
|
|
|
|
m_skip_pointers ? " (skip pointers)" : "",
|
|
|
|
m_skip_references ? " (skip references)" : "");
|
|
|
|
|
|
|
|
for (int i = 0; i < GetCount(); i++)
|
|
|
|
{
|
|
|
|
sstr.Printf(" %s\n",
|
|
|
|
GetExpressionPathAtIndex(i).c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
sstr.Printf("}");
|
|
|
|
return sstr.GetString();
|
|
|
|
}
|
2011-07-24 08:14:56 +08:00
|
|
|
|
|
|
|
SyntheticScriptProvider::FrontEnd::FrontEnd(std::string pclass,
|
|
|
|
lldb::ValueObjectSP be) :
|
|
|
|
SyntheticChildrenFrontEnd(be),
|
|
|
|
m_python_class(pclass)
|
|
|
|
{
|
2011-07-26 00:59:05 +08:00
|
|
|
if (be.get() == NULL)
|
|
|
|
{
|
|
|
|
m_interpreter = NULL;
|
|
|
|
m_wrapper = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-07-24 08:14:56 +08:00
|
|
|
m_interpreter = be->GetUpdatePoint().GetTarget()->GetDebugger().GetCommandInterpreter().GetScriptInterpreter();
|
2011-07-26 00:59:05 +08:00
|
|
|
|
|
|
|
if (m_interpreter == NULL)
|
|
|
|
m_wrapper = NULL;
|
|
|
|
else
|
|
|
|
m_wrapper = (PyObject*)m_interpreter->CreateSyntheticScriptedProvider(m_python_class, m_backend);
|
2011-07-24 08:14:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
SyntheticScriptProvider::GetDescription()
|
|
|
|
{
|
|
|
|
StreamString sstr;
|
2011-07-26 00:59:05 +08:00
|
|
|
sstr.Printf("%s%s%s Python class %s",
|
2011-07-24 08:14:56 +08:00
|
|
|
m_cascades ? "" : " (not cascading)",
|
|
|
|
m_skip_pointers ? " (skip pointers)" : "",
|
|
|
|
m_skip_references ? " (skip references)" : "",
|
|
|
|
m_python_class.c_str());
|
|
|
|
|
|
|
|
return sstr.GetString();
|
|
|
|
}
|