Skip to content

Commit c7f8730

Browse files
author
Enrico Granata
committed
Added formatters for libc++ (http://libcxx.llvm.org):
std::string has a summary provider std::vector std::list and std::map have both a summary and a synthetic children provider Given the usage of a custom namespace (std::__1::classname) for the implementation of libc++, we keep both libstdcpp and libc++ formatters enabled at the same time since that raises no conflicts and enabled for seamless transition between the two The formatters for libc++ reside in a libcxx category, and are loaded from libcxx.py (to be found in examples/synthetic) The formatters-stl test cases have been divided to be separate for libcxx and libstdcpp. This separation is necessary because (a) we need different compiler flags for libc++ than for libstdcpp (b) libc++ inlines a lot more than libstdcpp and some code changes were required to accommodate this difference llvm-svn: 152570
1 parent 701a6b4 commit c7f8730

23 files changed

+1417
-7
lines changed

lldb/examples/synthetic/libcxx.py

+481
Large diffs are not rendered by default.

lldb/include/lldb/Core/FormatManager.h

+4
Original file line numberDiff line numberDiff line change
@@ -670,6 +670,7 @@ class FormatManager : public IFormatChangeListener
670670
ConstString m_default_category_name;
671671
ConstString m_system_category_name;
672672
ConstString m_gnu_cpp_category_name;
673+
ConstString m_libcxx_category_name;
673674
ConstString m_objc_category_name;
674675
ConstString m_corefoundation_category_name;
675676
ConstString m_coregraphics_category_name;
@@ -690,6 +691,9 @@ class FormatManager : public IFormatChangeListener
690691
void
691692
LoadSTLFormatters();
692693

694+
void
695+
LoadLibcxxFormatters();
696+
693697
void
694698
LoadSystemFormatters();
695699

lldb/scripts/Python/finish-swig-Python-LLDB.sh

+14
Original file line numberDiff line numberDiff line change
@@ -184,6 +184,20 @@ else
184184
fi
185185
fi
186186

187+
if [ -f "${SRC_ROOT}/examples/synthetic/libcxx.py" ]
188+
then
189+
if [ $Debug == 1 ]
190+
then
191+
echo "Copying libcxx.py to ${framework_python_dir}"
192+
fi
193+
cp "${SRC_ROOT}/examples/synthetic/libcxx.py" "${framework_python_dir}"
194+
else
195+
if [ $Debug == 1 ]
196+
then
197+
echo "Unable to find ${SRC_ROOT}/examples/synthetic/libcxx.py"
198+
fi
199+
fi
200+
187201
# Copy the ObjC formatters over to the framework Python directory
188202
if [ -f "${SRC_ROOT}/examples/summaries/objc.py" ]
189203
then

lldb/source/Core/FormatManager.cpp

+54
Original file line numberDiff line numberDiff line change
@@ -580,6 +580,7 @@ FormatManager::FormatManager() :
580580
m_default_category_name(ConstString("default")),
581581
m_system_category_name(ConstString("system")),
582582
m_gnu_cpp_category_name(ConstString("gnu-libstdc++")),
583+
m_libcxx_category_name(ConstString("libcxx")),
583584
m_objc_category_name(ConstString("objc")),
584585
m_corefoundation_category_name(ConstString("CoreFoundation")),
585586
m_coregraphics_category_name(ConstString("CoreGraphics")),
@@ -590,6 +591,7 @@ FormatManager::FormatManager() :
590591

591592
LoadSystemFormatters();
592593
LoadSTLFormatters();
594+
LoadLibcxxFormatters();
593595
#ifndef LLDB_DISABLE_PYTHON
594596
LoadObjCFormatters();
595597
#endif
@@ -600,6 +602,7 @@ FormatManager::FormatManager() :
600602
//EnableCategory(m_coreservices_category_name,CategoryMap::Last);
601603
//EnableCategory(m_coregraphics_category_name,CategoryMap::Last);
602604
EnableCategory(m_gnu_cpp_category_name,CategoryMap::Last);
605+
EnableCategory(m_libcxx_category_name,CategoryMap::Last);
603606
//EnableCategory(m_vectortypes_category_name,CategoryMap::Last);
604607
EnableCategory(m_system_category_name,CategoryMap::Last);
605608
}
@@ -653,6 +656,57 @@ FormatManager::LoadSTLFormatters()
653656
#endif
654657
}
655658

659+
void
660+
FormatManager::LoadLibcxxFormatters()
661+
{
662+
TypeSummaryImpl::Flags stl_summary_flags;
663+
stl_summary_flags.SetCascades(true)
664+
.SetSkipPointers(false)
665+
.SetSkipReferences(false)
666+
.SetDontShowChildren(true)
667+
.SetDontShowValue(true)
668+
.SetShowMembersOneLiner(false)
669+
.SetHideItemNames(false);
670+
671+
std::string code(" libcxx.stdstring_SummaryProvider(valobj,dict)");
672+
lldb::TypeSummaryImplSP std_string_summary_sp(new ScriptSummaryFormat(stl_summary_flags, "libcxx.stdstring_SummaryProvider",code.c_str()));
673+
674+
TypeCategoryImpl::SharedPointer libcxx_category_sp = GetCategory(m_libcxx_category_name);
675+
676+
libcxx_category_sp->GetSummaryNavigator()->Add(ConstString("std::__1::string"),
677+
std_string_summary_sp);
678+
libcxx_category_sp->GetSummaryNavigator()->Add(ConstString("std::__1::basic_string<char, class std::__1::char_traits<char>, class std::__1::allocator<char> >"),
679+
std_string_summary_sp);
680+
681+
682+
#ifndef LLDB_DISABLE_PYTHON
683+
684+
SyntheticChildren::Flags stl_synth_flags;
685+
stl_synth_flags.SetCascades(true).SetSkipPointers(false).SetSkipReferences(false);
686+
687+
libcxx_category_sp->GetRegexSyntheticNavigator()->Add(RegularExpressionSP(new RegularExpression("^(std::__1::)vector<.+>$")),
688+
SyntheticChildrenSP(new TypeSyntheticImpl(stl_synth_flags,
689+
"libcxx.stdvector_SynthProvider")));
690+
libcxx_category_sp->GetRegexSyntheticNavigator()->Add(RegularExpressionSP(new RegularExpression("^(std::__1::)list<.+>$")),
691+
SyntheticChildrenSP(new TypeSyntheticImpl(stl_synth_flags,
692+
"libcxx.stdlist_SynthProvider")));
693+
libcxx_category_sp->GetRegexSyntheticNavigator()->Add(RegularExpressionSP(new RegularExpression("^(std::__1::)map<.+> >$")),
694+
SyntheticChildrenSP(new TypeSyntheticImpl(stl_synth_flags,
695+
"libcxx.stdmap_SynthProvider")));
696+
697+
stl_summary_flags.SetDontShowChildren(false);
698+
code.assign(" libcxx.stdvector_SummaryProvider(valobj,dict)");
699+
libcxx_category_sp->GetRegexSummaryNavigator()->Add(RegularExpressionSP(new RegularExpression("^(std::__1::)vector<.+>$")),
700+
TypeSummaryImplSP(new ScriptSummaryFormat(stl_summary_flags, "libcxx.stdvector_SummaryProvider",code.c_str())));
701+
code.assign(" libcxx.stdlist_SummaryProvider(valobj,dict)");
702+
libcxx_category_sp->GetRegexSummaryNavigator()->Add(RegularExpressionSP(new RegularExpression("^(std::__1::)list<.+>$")),
703+
TypeSummaryImplSP(new ScriptSummaryFormat(stl_summary_flags, "libcxx.stdlist_SummaryProvider",code.c_str())));
704+
code.assign(" libcxx.stdmap_SummaryProvider(valobj,dict)");
705+
libcxx_category_sp->GetRegexSummaryNavigator()->Add(RegularExpressionSP(new RegularExpression("^(std::__1::)map<.+> >$")),
706+
TypeSummaryImplSP(new ScriptSummaryFormat(stl_summary_flags, "libcxx.stdmap_SummaryProvider",code.c_str())));
707+
#endif
708+
}
709+
656710
void
657711
FormatManager::LoadSystemFormatters()
658712
{

lldb/source/Interpreter/ScriptInterpreterPython.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -279,7 +279,7 @@ ScriptInterpreterPython::ScriptInterpreterPython (CommandInterpreter &interprete
279279
int old_count = Debugger::TestDebuggerRefCount();
280280

281281

282-
run_string.Printf ("run_one_line (%s, 'import copy, os, re, sys, uuid, lldb, gnu_libstdcpp, objc')", m_dictionary_name.c_str());
282+
run_string.Printf ("run_one_line (%s, 'import copy, os, re, sys, uuid, lldb, gnu_libstdcpp, libcxx, objc')", m_dictionary_name.c_str());
283283
PyRun_SimpleString (run_string.GetData());
284284

285285
// WARNING: temporary code that loads Cocoa formatters - this should be done on a per-platform basis rather than loading the whole set
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
LEVEL = ../../../../../make
2+
3+
CXX_SOURCES := main.cpp
4+
5+
include $(LEVEL)/Makefile.rules
6+
7+
CXXFLAGS += -stdlib=libc++ -O0
8+
LDFLAGS += -stdlib=libc++
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,175 @@
1+
"""
2+
Test lldb data formatter subsystem.
3+
"""
4+
5+
import os, time
6+
import unittest2
7+
import lldb
8+
from lldbtest import *
9+
10+
class LibcxxListDataFormatterTestCase(TestBase):
11+
12+
mydir = os.path.join("functionalities", "data-formatter", "data-formatter-stl", "libcxx", "list")
13+
14+
@unittest2.skipUnless(sys.platform.startswith("darwin"), "requires Darwin")
15+
def test_with_dsym_and_run_command(self):
16+
"""Test data formatter commands."""
17+
self.buildDsym()
18+
self.data_formatter_commands()
19+
20+
def test_with_dwarf_and_run_command(self):
21+
"""Test data formatter commands."""
22+
self.buildDwarf()
23+
self.data_formatter_commands()
24+
25+
def setUp(self):
26+
# Call super's setUp().
27+
TestBase.setUp(self)
28+
# Find the line number to break at.
29+
self.line = line_number('main.cpp', '// Set break point at this line.')
30+
self.line2 = line_number('main.cpp', '// Set second break point at this line.')
31+
32+
def data_formatter_commands(self):
33+
"""Test that that file and class static variables display correctly."""
34+
self.runCmd("file a.out", CURRENT_EXECUTABLE_SET)
35+
36+
self.expect("breakpoint set -f main.cpp -l %d" % self.line,
37+
BREAKPOINT_CREATED,
38+
startstr = "Breakpoint created: 1: file ='main.cpp', line = %d" %
39+
self.line)
40+
self.expect("breakpoint set -f main.cpp -l %d" % self.line2,
41+
BREAKPOINT_CREATED,
42+
startstr = "Breakpoint created: 2: file ='main.cpp', line = %d" %
43+
self.line2)
44+
45+
self.runCmd("run", RUN_SUCCEEDED)
46+
47+
# The stop reason of the thread should be breakpoint.
48+
self.expect("thread list", STOPPED_DUE_TO_BREAKPOINT,
49+
substrs = ['stopped',
50+
'stop reason = breakpoint'])
51+
52+
# This is the function to remove the custom formats in order to have a
53+
# clean slate for the next test case.
54+
def cleanup():
55+
self.runCmd('type format clear', check=False)
56+
self.runCmd('type summary clear', check=False)
57+
self.runCmd('type filter clear', check=False)
58+
self.runCmd('type synth clear', check=False)
59+
self.runCmd("settings set target.max-children-count 256", check=False)
60+
61+
# Execute the cleanup function during test case tear down.
62+
self.addTearDownHook(cleanup)
63+
64+
self.runCmd("frame variable numbers_list -T")
65+
self.runCmd("type summary add std::int_list std::string_list int_list string_list --summary-string \"list has ${svar%#} items\" -e")
66+
self.runCmd("type format add -f hex int")
67+
68+
self.expect("frame variable numbers_list --raw", matching=False,
69+
substrs = ['list has 0 items',
70+
'{}'])
71+
72+
self.expect("frame variable numbers_list",
73+
substrs = ['list has 0 items',
74+
'{}'])
75+
76+
self.expect("p numbers_list",
77+
substrs = ['list has 0 items',
78+
'{}'])
79+
80+
self.runCmd("n")
81+
82+
self.expect("frame variable numbers_list",
83+
substrs = ['list has 1 items',
84+
'[0] = ',
85+
'0x12345678'])
86+
87+
self.runCmd("n");self.runCmd("n");self.runCmd("n");
88+
89+
self.expect("frame variable numbers_list",
90+
substrs = ['list has 4 items',
91+
'[0] = ',
92+
'0x12345678',
93+
'[1] =',
94+
'0x11223344',
95+
'[2] =',
96+
'0xbeeffeed',
97+
'[3] =',
98+
'0x00abba00'])
99+
100+
self.runCmd("n");self.runCmd("n");
101+
102+
self.expect("frame variable numbers_list",
103+
substrs = ['list has 6 items',
104+
'[0] = ',
105+
'0x12345678',
106+
'0x11223344',
107+
'0xbeeffeed',
108+
'0x00abba00',
109+
'[4] =',
110+
'0x0abcdef0',
111+
'[5] =',
112+
'0x0cab0cab'])
113+
114+
self.expect("p numbers_list",
115+
substrs = ['list has 6 items',
116+
'[0] = ',
117+
'0x12345678',
118+
'0x11223344',
119+
'0xbeeffeed',
120+
'0x00abba00',
121+
'[4] =',
122+
'0x0abcdef0',
123+
'[5] =',
124+
'0x0cab0cab'])
125+
126+
# check access-by-index
127+
self.expect("frame variable numbers_list[0]",
128+
substrs = ['0x12345678']);
129+
self.expect("frame variable numbers_list[1]",
130+
substrs = ['0x11223344']);
131+
132+
self.runCmd("n")
133+
134+
self.expect("frame variable numbers_list",
135+
substrs = ['list has 0 items',
136+
'{}'])
137+
138+
self.runCmd("n");self.runCmd("n");self.runCmd("n");self.runCmd("n");
139+
140+
self.expect("frame variable numbers_list",
141+
substrs = ['list has 4 items',
142+
'[0] = ', '1',
143+
'[1] = ', '2',
144+
'[2] = ', '3',
145+
'[3] = ', '4'])
146+
147+
self.runCmd("type format delete int")
148+
149+
self.runCmd("c")
150+
151+
self.expect("frame variable text_list",
152+
substrs = ['list has 3 items',
153+
'[0]', 'goofy',
154+
'[1]', 'is',
155+
'[2]', 'smart'])
156+
157+
self.expect("p text_list",
158+
substrs = ['list has 3 items',
159+
'\"goofy\"',
160+
'\"is\"',
161+
'\"smart\"'])
162+
163+
self.runCmd("n")
164+
165+
# check access-by-index
166+
self.expect("frame variable text_list[0]",
167+
substrs = ['goofy']);
168+
self.expect("frame variable text_list[3]",
169+
substrs = ['!!!']);
170+
171+
if __name__ == '__main__':
172+
import atexit
173+
lldb.SBDebugger.Initialize()
174+
atexit.register(lambda: lldb.SBDebugger.Terminate())
175+
unittest2.main()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
#include <string>
2+
#define _LIBCPP_INLINE_VISIBILITY
3+
#include <list>
4+
5+
6+
typedef std::list<int> int_list;
7+
typedef std::list<std::string> string_list;
8+
9+
int main()
10+
{
11+
int_list numbers_list;
12+
13+
(numbers_list.push_back(0x12345678)); // Set break point at this line.
14+
(numbers_list.push_back(0x11223344));
15+
(numbers_list.push_back(0xBEEFFEED));
16+
(numbers_list.push_back(0x00ABBA00));
17+
(numbers_list.push_back(0x0ABCDEF0));
18+
(numbers_list.push_back(0x0CAB0CAB));
19+
20+
numbers_list.clear();
21+
22+
(numbers_list.push_back(1));
23+
(numbers_list.push_back(2));
24+
(numbers_list.push_back(3));
25+
(numbers_list.push_back(4));
26+
27+
string_list text_list;
28+
(text_list.push_back(std::string("goofy")));
29+
(text_list.push_back(std::string("is")));
30+
(text_list.push_back(std::string("smart")));
31+
32+
(text_list.push_back(std::string("!!!"))); // Set second break point at this line.
33+
34+
return 0;
35+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
LEVEL = ../../../../../make
2+
3+
CXX_SOURCES := main.cpp
4+
5+
include $(LEVEL)/Makefile.rules
6+
7+
CXXFLAGS += -stdlib=libc++ -O0
8+
LDFLAGS += -stdlib=libc++

0 commit comments

Comments
 (0)