Felsökning av C API-tillägg och CPython-interna funktioner med GDB¶
Det här dokumentet förklarar hur Python GDB-tillägget, python-gdb.py
, kan användas med GDB-debuggern för att debugga CPython-tillägg och själva CPython-tolken.
Vid felsökning av lågnivåproblem som krascher eller deadlocks är en lågnivåfelsökare, som GDB, användbar för att diagnostisera och korrigera problemet. Som standard stöder GDB (eller någon av dess frontend) inte information på hög nivå som är specifik för CPython-tolken.
Tillägget python-gdb.py
lägger till information om CPython-tolken i GDB. Tillägget hjälper till att introspektera stacken för Python-funktioner som körs för närvarande. Med tanke på ett Python-objekt som representeras av en PyObject*-pekare, visar tillägget objektets typ och värde.
Utvecklare som arbetar med tillägg till CPython eller med delar av CPython som är skrivna i C kan använda det här dokumentet för att lära sig hur man använder tillägget python-gdb.py
med GDB.
Anteckning
Detta dokument förutsätter att du är bekant med grunderna i GDB och CPython C API. Det konsoliderar vägledning från devguide och Python wiki.
Förutsättningar¶
Du måste ha:
GDB 7 eller senare. (För tidigare versioner av GDB, se
Misc/gdbinit
i källorna till Python 3.11 eller tidigare)GDB-kompatibel felsökningsinformation för Python och alla tillägg som du felsöker.
Tillägget
python-gdb.py
.
Tillägget är byggt med Python, men kan distribueras separat eller inte alls. Nedan finns tips för några vanliga system som exempel. Observera att även om instruktionerna matchar ditt system kan de vara föråldrade.
Installation med Python byggt från källkod¶
När du bygger CPython från källkod bör felsökningsinformation vara tillgänglig och byggandet bör lägga till en fil med namnet python-gdb.py
i rotkatalogen för ditt repository.
För att aktivera stödet måste du lägga till katalogen som innehåller python-gdb.py
i GDB:s ”auto-load-safe-path”. Om du inte har gjort detta kommer de senaste versionerna av GDB att skriva ut en varning med instruktioner om hur du gör detta.
Anteckning
Om du inte hittar några instruktioner för din version av GDB, skriv in detta i din konfigurationsfil (~/.gdbinit
eller ~/.config/gdb/gdbinit
):
add-auto-load-safe-path /path/to/cpython
Du kan också lägga till flera sökvägar, åtskilda med :
.
Installation för Python från en Linux-distro¶
De flesta Linux-system tillhandahåller felsökningsinformation för systemet Python i ett paket som heter python-debuginfo
, python-dbg
eller liknande. Till exempel
Fedora:
sudo dnf install gdb sudo dnf debuginfo-install python3
Ubuntu:
sudo apt install gdb python3-dbg
På flera nya Linux-system kan GDB hämta felsökningssymboler automatiskt med hjälp av debuginfod. Detta kommer dock inte att installera tillägget python-gdb.py
; du behöver i allmänhet installera debug info-paketet separat.
Använda Debug- och utvecklingsläget¶
För enklare felsökning kanske du vill göra det:
Använd en debug build av Python. (När du bygger från källkod, använd
configure --with-pydebug
. På Linux-distributioner, installera och kör ett paket sompython-debug
ellerpython-dbg
, om det finns tillgängligt)Använd runtime development mode (
-X dev
).
Båda aktiverar extra assertions och inaktiverar vissa optimeringar. Ibland döljer detta felet du försöker hitta, men i de flesta fall gör de processen enklare.
Använda tillägget python-gdb
¶
När tillägget är laddat har det två huvudfunktioner: snygga skrivare för Python-värden och ytterligare kommandon.
Vackra skrivare¶
Så här ser en GDB-bakspårning ut (avkortad) när detta tillägg är aktiverat:
#0 0x000000000041a6b1 in PyObject_Malloc (nbytes=Cannot access memory at address 0x7fffff7fefe8
) at Objects/obmalloc.c:748
#1 0x000000000041b7c0 in _PyObject_DebugMallocApi (id=111 'o', nbytes=24) at Objects/obmalloc.c:1445
#2 0x000000000041b717 in _PyObject_DebugMalloc (nbytes=24) at Objects/obmalloc.c:1412
#3 0x000000000044060a in _PyUnicode_New (length=11) at Objects/unicodeobject.c:346
#4 0x00000000004466aa in PyUnicodeUCS2_DecodeUTF8Stateful (s=0x5c2b8d "__lltrace__", size=11, errors=0x0, consumed=
0x0) at Objects/unicodeobject.c:2531
#5 0x0000000000446647 in PyUnicodeUCS2_DecodeUTF8 (s=0x5c2b8d "__lltrace__", size=11, errors=0x0)
at Objects/unicodeobject.c:2495
#6 0x0000000000440d1b in PyUnicodeUCS2_FromStringAndSize (u=0x5c2b8d "__lltrace__", size=11)
at Objects/unicodeobject.c:551
#7 0x0000000000440d94 in PyUnicodeUCS2_FromString (u=0x5c2b8d "__lltrace__") at Objects/unicodeobject.c:569
#8 0x0000000000584abd in PyDict_GetItemString (v=
{'Yuck': <type at remote 0xad4730>, '__builtins__': <module at remote 0x7ffff7fd5ee8>, '__file__': 'Lib/test/crashers/nasty_eq_vs_dict.py', '__package__': None, 'y': <Yuck(i=0) at remote 0xaacd80>, 'dict': {0: 0, 1: 1, 2: 2, 3: 3}, '__cached__': None, '__name__': '__main__', 'z': <Yuck(i=0) at remote 0xaace60>, '__doc__': None}, key=
0x5c2b8d "__lltrace__") at Objects/dictobject.c:2171
Lägg märke till hur ordboksargumentet till PyDict_GetItemString
visas som dess repr()
, snarare än en ogenomskinlig PyObject *
-pekare.
Tillägget fungerar genom att tillhandahålla en anpassad utskriftsrutin för värden av typen PyObject *
. Om du behöver komma åt detaljer på lägre nivå i ett objekt, castar du värdet till en pekare av lämplig typ. Till exempel:
(gdb) p globals
$1 = {'__builtins__': <module at remote 0x7ffff7fb1868>, '__name__':
'__main__', 'ctypes': <module at remote 0x7ffff7f14360>, '__doc__': None,
'__package__': None}
(gdb) p *(PyDictObject*)globals
$2 = {ob_refcnt = 3, ob_type = 0x3dbdf85820, ma_fill = 5, ma_used = 5,
ma_mask = 7, ma_table = 0x63d0f8, ma_lookup = 0x3dbdc7ea70
<lookdict_string>, ma_smalltable = {{me_hash = 7065186196740147912,
me_key = '__builtins__', me_value = <module at remote 0x7ffff7fb1868>},
{me_hash = -368181376027291943, me_key = '__name__',
me_value ='__main__'}, {me_hash = 0, me_key = 0x0, me_value = 0x0},
{me_hash = 0, me_key = 0x0, me_value = 0x0},
{me_hash = -9177857982131165996, me_key = 'ctypes',
me_value = <module at remote 0x7ffff7f14360>},
{me_hash = -8518757509529533123, me_key = '__doc__', me_value = None},
{me_hash = 0, me_key = 0x0, me_value = 0x0}, {
me_hash = 6614918939584953775, me_key = '__package__', me_value = None}}}
Observera att pretty-printers faktiskt inte anropar repr()
. För grundläggande typer försöker de matcha dess resultat nära.
Ett område som kan vara förvirrande är att den anpassade skrivaren för vissa typer ser ut ungefär som GDB:s inbyggda skrivare för standardtyper. Till exempel ger pretty-printern för en Python int
(PyLongObject*) en representation som inte kan skiljas från en representation av ett vanligt heltal på maskinnivå:
(gdb) p some_machine_integer
$3 = 42
(gdb) p some_python_integer
$4 = 42
Den interna strukturen kan avslöjas med en cast till PyLongObject*:
(gdb) p *(PyLongObject*)some_python_integer
$5 = {ob_base = {ob_base = {ob_refcnt = 8, ob_type = 0x3dad39f5e0}, ob_size = 1},
ob_digit = {42}}
En liknande förvirring kan uppstå med typen str
, där utdata ser ut ungefär som gdb:s inbyggda skrivare för char *
:
(gdb) p ptr_to_python_str
$6 = '__builtins__'
Pretty-printer för str
-instanser använder som standard enkla citattecken (precis som Pythons repr
för strängar), medan standardprinter för char *
-värden använder dubbla citattecken och innehåller en hexadecimal adress:
(gdb) p ptr_to_char_star
$7 = 0x6d72c0 "hello world"
Återigen kan implementeringsdetaljerna avslöjas med en cast till PyUnicodeObject*:
(gdb) p *(PyUnicodeObject*)$6
$8 = {ob_base = {ob_refcnt = 33, ob_type = 0x3dad3a95a0}, length = 12,
str = 0x7ffff2128500, hash = 7065186196740147912, state = 1, defenc = 0x0}
py-list
¶
Tillägget lägger till kommandot
py-list
, som listar Python-källkoden (om någon) för den aktuella bildrutan i den valda tråden. Den aktuella raden markeras med ett ”>”:(gdb) py-list 901 if options.profile: 902 options.profile = False 903 profile_me() 904 return 905 >906 u = UI() 907 if not u.quit: 908 try: 909 gtk.main() 910 except KeyboardInterrupt: 911 # properly quit on a keyboard interrupt...Använd
py-list START
för att lista på ett annat radnummer i Python-källan ochpy-list START,END
för att lista ett specifikt intervall av rader i Python-källan.
py-up
and py-down
¶
Kommandona
py-up
ochpy-down
är analoga med GDB:s vanliga kommandonup
ochdown
, men försöker röra sig på nivån för CPython-ramar, snarare än C-ramar.GDB kan inte alltid läsa den relevanta raminformationen, beroende på optimeringsnivån som CPython kompilerades med. Internt letar kommandona efter C-ramar som kör standardfunktionen för ramutvärdering (det vill säga den centrala bytecode-tolkningsloopen i CPython) och letar upp värdet på det relaterade
PyFrameObject *
.De sänder ut ramnumret (på C-nivå) inom tråden.
Till exempel:
(gdb) py-up #37 Frame 0x9420b04, for file /usr/lib/python2.6/site-packages/ gnome_sudoku/main.py, line 906, in start_game () u = UI() (gdb) py-up #40 Frame 0x948e82c, for file /usr/lib/python2.6/site-packages/ gnome_sudoku/gnome_sudoku.py, line 22, in start_game(main=<module at remote 0xb771b7f4>) main.start_game() (gdb) py-up Unable to find an older python frameså vi är högst upp i Python-stacken.
Ramnumren motsvarar de som visas av GDB:s standardkommando
backtrace
. Kommandot hoppar över C-rutor som inte exekverar Python-kod.Går ner igen:
(gdb) py-down #37 Frame 0x9420b04, for file /usr/lib/python2.6/site-packages/gnome_sudoku/main.py, line 906, in start_game () u = UI() (gdb) py-down #34 (unable to read python frame information) (gdb) py-down #23 (unable to read python frame information) (gdb) py-down #19 (unable to read python frame information) (gdb) py-down #14 Frame 0x99262ac, for file /usr/lib/python2.6/site-packages/gnome_sudoku/game_selector.py, line 201, in run_swallowed_dialog (self=<NewOrSavedGameSelector(new_game_model=<gtk.ListStore at remote 0x98fab44>, puzzle=None, saved_games=[{'gsd.auto_fills': 0, 'tracking': {}, 'trackers': {}, 'notes': [], 'saved_at': 1270084485, 'game': '7 8 0 0 0 0 0 5 6 0 0 9 0 8 0 1 0 0 0 4 6 0 0 0 0 7 0 6 5 0 0 0 4 7 9 2 0 0 0 9 0 1 0 0 0 3 9 7 6 0 0 0 1 8 0 6 0 0 0 0 2 8 0 0 0 5 0 4 0 6 0 0 2 1 0 0 0 0 0 4 5\n7 8 0 0 0 0 0 5 6 0 0 9 0 8 0 1 0 0 0 4 6 0 0 0 0 7 0 6 5 1 8 3 4 7 9 2 0 0 0 9 0 1 0 0 0 3 9 7 6 0 0 0 1 8 0 6 0 0 0 0 2 8 0 0 0 5 0 4 0 6 0 0 2 1 0 0 0 0 0 4 5', 'gsd.impossible_hints': 0, 'timer.__absolute_start_time__': <float at remote 0x984b474>, 'gsd.hints': 0, 'timer.active_time': <float at remote 0x984b494>, 'timer.total_time': <float at remote 0x984b464>}], dialog=<gtk.Dialog at remote 0x98faaa4>, saved_game_model=<gtk.ListStore at remote 0x98fad24>, sudoku_maker=<SudokuMaker(terminated=False, played=[], batch_siz...(truncated) swallower.run_dialog(self.dialog) (gdb) py-down #11 Frame 0x9aead74, for file /usr/lib/python2.6/site-packages/gnome_sudoku/dialog_swallower.py, line 48, in run_dialog (self=<SwappableArea(running=<gtk.Dialog at remote 0x98faaa4>, main_page=0) at remote 0x98fa6e4>, d=<gtk.Dialog at remote 0x98faaa4>) gtk.main() (gdb) py-down #8 (unable to read python frame information) (gdb) py-down Unable to find a newer python frameoch vi är längst ner i Python-stacken.
Observera att i Python 3.12 och nyare kan samma C-stackram användas för flera Python-stackramar. Detta innebär att
py-up
ochpy-down
kan flytta flera Python-ramar samtidigt. Till exempel:(gdb) py-up #6 Frame 0x7ffff7fb62b0, for file /tmp/rec.py, line 5, in recursive_function (n=0) time.sleep(5) #6 Frame 0x7ffff7fb6240, for file /tmp/rec.py, line 7, in recursive_function (n=1) recursive_function(n-1) #6 Frame 0x7ffff7fb61d0, for file /tmp/rec.py, line 7, in recursive_function (n=2) recursive_function(n-1) #6 Frame 0x7ffff7fb6160, for file /tmp/rec.py, line 7, in recursive_function (n=3) recursive_function(n-1) #6 Frame 0x7ffff7fb60f0, for file /tmp/rec.py, line 7, in recursive_function (n=4) recursive_function(n-1) #6 Frame 0x7ffff7fb6080, for file /tmp/rec.py, line 7, in recursive_function (n=5) recursive_function(n-1) #6 Frame 0x7ffff7fb6020, for file /tmp/rec.py, line 9, in <module> () recursive_function(5) (gdb) py-up Unable to find an older python frame
py-bt
¶
Kommandot
py-bt
försöker visa en bakåtspårning på Python-nivå av den aktuella tråden.Till exempel:
(gdb) py-bt #8 (unable to read python frame information) #11 Frame 0x9aead74, for file /usr/lib/python2.6/site-packages/gnome_sudoku/dialog_swallower.py, line 48, in run_dialog (self=<SwappableArea(running=<gtk.Dialog at remote 0x98faaa4>, main_page=0) at remote 0x98fa6e4>, d=<gtk.Dialog at remote 0x98faaa4>) gtk.main() #14 Frame 0x99262ac, for file /usr/lib/python2.6/site-packages/gnome_sudoku/game_selector.py, line 201, in run_swallowed_dialog (self=<NewOrSavedGameSelector(new_game_model=<gtk.ListStore at remote 0x98fab44>, puzzle=None, saved_games=[{'gsd.auto_fills': 0, 'tracking': {}, 'trackers': {}, 'notes': [], 'saved_at': 1270084485, 'game': '7 8 0 0 0 0 0 5 6 0 0 9 0 8 0 1 0 0 0 4 6 0 0 0 0 7 0 6 5 0 0 0 4 7 9 2 0 0 0 9 0 1 0 0 0 3 9 7 6 0 0 0 1 8 0 6 0 0 0 0 2 8 0 0 0 5 0 4 0 6 0 0 2 1 0 0 0 0 0 4 5\n7 8 0 0 0 0 0 5 6 0 0 9 0 8 0 1 0 0 0 4 6 0 0 0 0 7 0 6 5 1 8 3 4 7 9 2 0 0 0 9 0 1 0 0 0 3 9 7 6 0 0 0 1 8 0 6 0 0 0 0 2 8 0 0 0 5 0 4 0 6 0 0 2 1 0 0 0 0 0 4 5', 'gsd.impossible_hints': 0, 'timer.__absolute_start_time__': <float at remote 0x984b474>, 'gsd.hints': 0, 'timer.active_time': <float at remote 0x984b494>, 'timer.total_time': <float at remote 0x984b464>}], dialog=<gtk.Dialog at remote 0x98faaa4>, saved_game_model=<gtk.ListStore at remote 0x98fad24>, sudoku_maker=<SudokuMaker(terminated=False, played=[], batch_siz...(truncated) swallower.run_dialog(self.dialog) #19 (unable to read python frame information) #23 (unable to read python frame information) #34 (unable to read python frame information) #37 Frame 0x9420b04, for file /usr/lib/python2.6/site-packages/gnome_sudoku/main.py, line 906, in start_game () u = UI() #40 Frame 0x948e82c, for file /usr/lib/python2.6/site-packages/gnome_sudoku/gnome_sudoku.py, line 22, in start_game (main=<module at remote 0xb771b7f4>) main.start_game()Ramnumren motsvarar de som visas med GDB:s standardkommando
backtrace
.
py-print
¶
Kommandot
py-print
letar upp ett Python-namn och försöker skriva ut det. Det letar i lokala filer i den aktuella tråden, sedan i globala filer och slutligen i builtins:(gdb) py-print self local 'self' = <SwappableArea(running=<gtk.Dialog at remote 0x98faaa4>, main_page=0) at remote 0x98fa6e4> (gdb) py-print __name__ global '__name__' = 'gnome_sudoku.dialog_swallower' (gdb) py-print len builtin 'len' = <built-in function len> (gdb) py-print scarlet_pimpernel 'scarlet_pimpernel' not foundOm den aktuella C-rutan motsvarar flera Python-rutor, tar
py-print
bara hänsyn till den första.
py-locals
¶
Kommandot
py-locals
letar upp alla Python-locals inom den aktuella Python-rutan i den valda tråden och skriver ut deras representationer:(gdb) py-locals self = <SwappableArea(running=<gtk.Dialog at remote 0x98faaa4>, main_page=0) at remote 0x98fa6e4> d = <gtk.Dialog at remote 0x98faaa4>Om den aktuella C-rutan motsvarar flera Python-rutor visas lokaldelar från dem alla:
(gdb) py-locals Locals for recursive_function n = 0 Locals for recursive_function n = 1 Locals for recursive_function n = 2 Locals for recursive_function n = 3 Locals for recursive_function n = 4 Locals for recursive_function n = 5 Locals for <module>
Använd med GDB-kommandon¶
Tilläggskommandona kompletterar GDB:s inbyggda kommandon. Du kan till exempel använda ett ramnummer som visas av py-bt
med kommandot frame
för att gå till en specifik ram inom den valda tråden, så här:
(gdb) py-bt
(output snipped)
#68 Frame 0xaa4560, for file Lib/test/regrtest.py, line 1548, in <module> ()
main()
(gdb) frame 68
#68 0x00000000004cd1e6 in PyEval_EvalFrameEx (f=Frame 0xaa4560, for file Lib/test/regrtest.py, line 1548, in <module> (), throwflag=0) at Python/ceval.c:2665
2665 x = call_function(&sp, oparg);
(gdb) py-list
1543 # Run the tests in a context manager that temporary changes the CWD to a
1544 # temporary and writable directory. If it's not possible to create or
1545 # change the CWD, the original CWD will be used. The original CWD is
1546 # available from test_support.SAVEDCWD.
1547 with test_support.temp_cwd(TESTCWD, quiet=True):
>1548 main()
Kommandot info threads
ger dig en lista över trådarna i processen, och du kan använda kommandot thread
för att välja en annan:
(gdb) info threads
105 Thread 0x7fffefa18710 (LWP 10260) sem_wait () at ../nptl/sysdeps/unix/sysv/linux/x86_64/sem_wait.S:86
104 Thread 0x7fffdf5fe710 (LWP 10259) sem_wait () at ../nptl/sysdeps/unix/sysv/linux/x86_64/sem_wait.S:86
* 1 Thread 0x7ffff7fe2700 (LWP 10145) 0x00000038e46d73e3 in select () at ../sysdeps/unix/syscall-template.S:82
Du kan använda thread apply all COMMAND
eller (t a a COMMAND
förkortat) för att köra ett kommando på alla trådar. Med py-bt
kan du se vad varje tråd gör på Python-nivå:
(gdb) t a a py-bt
Thread 105 (Thread 0x7fffefa18710 (LWP 10260)):
#5 Frame 0x7fffd00019d0, for file /home/david/coding/python-svn/Lib/threading.py, line 155, in _acquire_restore (self=<_RLock(_Verbose__verbose=False, _RLock__owner=140737354016512, _RLock__block=<thread.lock at remote 0x858770>, _RLock__count=1) at remote 0xd7ff40>, count_owner=(1, 140737213728528), count=1, owner=140737213728528)
self.__block.acquire()
#8 Frame 0x7fffac001640, for file /home/david/coding/python-svn/Lib/threading.py, line 269, in wait (self=<_Condition(_Condition__lock=<_RLock(_Verbose__verbose=False, _RLock__owner=140737354016512, _RLock__block=<thread.lock at remote 0x858770>, _RLock__count=1) at remote 0xd7ff40>, acquire=<instancemethod at remote 0xd80260>, _is_owned=<instancemethod at remote 0xd80160>, _release_save=<instancemethod at remote 0xd803e0>, release=<instancemethod at remote 0xd802e0>, _acquire_restore=<instancemethod at remote 0xd7ee60>, _Verbose__verbose=False, _Condition__waiters=[]) at remote 0xd7fd10>, timeout=None, waiter=<thread.lock at remote 0x858a90>, saved_state=(1, 140737213728528))
self._acquire_restore(saved_state)
#12 Frame 0x7fffb8001a10, for file /home/david/coding/python-svn/Lib/test/lock_tests.py, line 348, in f ()
cond.wait()
#16 Frame 0x7fffb8001c40, for file /home/david/coding/python-svn/Lib/test/lock_tests.py, line 37, in task (tid=140737213728528)
f()
Thread 104 (Thread 0x7fffdf5fe710 (LWP 10259)):
#5 Frame 0x7fffe4001580, for file /home/david/coding/python-svn/Lib/threading.py, line 155, in _acquire_restore (self=<_RLock(_Verbose__verbose=False, _RLock__owner=140737354016512, _RLock__block=<thread.lock at remote 0x858770>, _RLock__count=1) at remote 0xd7ff40>, count_owner=(1, 140736940992272), count=1, owner=140736940992272)
self.__block.acquire()
#8 Frame 0x7fffc8002090, for file /home/david/coding/python-svn/Lib/threading.py, line 269, in wait (self=<_Condition(_Condition__lock=<_RLock(_Verbose__verbose=False, _RLock__owner=140737354016512, _RLock__block=<thread.lock at remote 0x858770>, _RLock__count=1) at remote 0xd7ff40>, acquire=<instancemethod at remote 0xd80260>, _is_owned=<instancemethod at remote 0xd80160>, _release_save=<instancemethod at remote 0xd803e0>, release=<instancemethod at remote 0xd802e0>, _acquire_restore=<instancemethod at remote 0xd7ee60>, _Verbose__verbose=False, _Condition__waiters=[]) at remote 0xd7fd10>, timeout=None, waiter=<thread.lock at remote 0x858860>, saved_state=(1, 140736940992272))
self._acquire_restore(saved_state)
#12 Frame 0x7fffac001c90, for file /home/david/coding/python-svn/Lib/test/lock_tests.py, line 348, in f ()
cond.wait()
#16 Frame 0x7fffac0011c0, for file /home/david/coding/python-svn/Lib/test/lock_tests.py, line 37, in task (tid=140736940992272)
f()
Thread 1 (Thread 0x7ffff7fe2700 (LWP 10145)):
#5 Frame 0xcb5380, for file /home/david/coding/python-svn/Lib/test/lock_tests.py, line 16, in _wait ()
time.sleep(0.01)
#8 Frame 0x7fffd00024a0, for file /home/david/coding/python-svn/Lib/test/lock_tests.py, line 378, in _check_notify (self=<ConditionTests(_testMethodName='test_notify', _resultForDoCleanups=<TestResult(_original_stdout=<cStringIO.StringO at remote 0xc191e0>, skipped=[], _mirrorOutput=False, testsRun=39, buffer=False, _original_stderr=<file at remote 0x7ffff7fc6340>, _stdout_buffer=<cStringIO.StringO at remote 0xc9c7f8>, _stderr_buffer=<cStringIO.StringO at remote 0xc9c790>, _moduleSetUpFailed=False, expectedFailures=[], errors=[], _previousTestClass=<type at remote 0x928310>, unexpectedSuccesses=[], failures=[], shouldStop=False, failfast=False) at remote 0xc185a0>, _threads=(0,), _cleanups=[], _type_equality_funcs={<type at remote 0x7eba00>: <instancemethod at remote 0xd750e0>, <type at remote 0x7e7820>: <instancemethod at remote 0xd75160>, <type at remote 0x7e30e0>: <instancemethod at remote 0xd75060>, <type at remote 0x7e7d20>: <instancemethod at remote 0xd751e0>, <type at remote 0x7f19e0...(truncated)
_wait()