Skip to content
Snippets Groups Projects
Commit d857e338 authored by Mohammed Bilal Akhtar's avatar Mohammed Bilal Akhtar
Browse files

Add support for backtraces/info about active fibres

parent 028c6fe9
No related branches found
No related tags found
1 merge request!2Add support for backtraces/info about active fibres
...@@ -7,6 +7,7 @@ from contextlib import contextmanager ...@@ -7,6 +7,7 @@ from contextlib import contextmanager
class FibreSupport(): class FibreSupport():
def __init__(self): def __init__(self):
FibreSupport.list = [] FibreSupport.list = []
FibreSupport.active = {}
FibreSupport.saved = False FibreSupport.saved = False
if (gdb.lookup_symbol("_globalStackList")[0] == None): if (gdb.lookup_symbol("_globalStackList")[0] == None):
print() print()
...@@ -28,9 +29,22 @@ class FibreSupport(): ...@@ -28,9 +29,22 @@ class FibreSupport():
while (next != first): while (next != first):
FibreSupport.list.append(next) FibreSupport.list.append(next)
next = next['link'][1]['next'] next = next['link'][1]['next']
orig_thread = gdb.selected_thread()
for thread in gdb.selected_inferior().threads():
thread.switch()
currStack = str(gdb.parse_and_eval("Context::currStack"))
# Cache the registers for this thread, in case it represents
# a fibre
FibreSupport.active[currStack] = {
'rsp': str(gdb.parse_and_eval("$rsp")).split(None, 1)[0],
'rbp': str(gdb.parse_and_eval("$rbp")).split(None, 1)[0],
'rip': str(gdb.parse_and_eval("$rip")).split(None, 1)[0]
}
orig_thread.switch()
def cont_handler(event): def cont_handler(event):
FibreSupport.list = [] FibreSupport.list = []
FibreSupport.active = {}
# ideally restore() would be hooked up with cont_handler, # ideally restore() would be hooked up with cont_handler,
# but gdb currently fails with segmentation fault when trying # but gdb currently fails with segmentation fault when trying
...@@ -56,19 +70,28 @@ class FibreSupport(): ...@@ -56,19 +70,28 @@ class FibreSupport():
# if current pthread: use current register context # if current pthread: use current register context
if (arg == gdb.parse_and_eval("Context::currStack")): if (arg == gdb.parse_and_eval("Context::currStack")):
return True return True
# retrieve fibre's register context # Check active fibre cache in case this fibre is in it
ftype = gdb.lookup_type('Fibre').pointer() # (FibreSupport.active is more up-to-date than
ptr = gdb.Value(arg).cast(ftype) # StackContext for retrieving stack pointers)
rsp = ptr['stackPointer'] argstr = str(arg)
if (rsp == 0): if (argstr in FibreSupport.active):
if not silent: rsp = FibreSupport.active[argstr]['rsp']
print("cannot access stack pointer - active in different thread?") rbp = FibreSupport.active[argstr]['rbp']
return False rip = FibreSupport.active[argstr]['rip']
ptype = gdb.lookup_type('uintptr_t').pointer() else:
rsp = rsp + 40 # cf. STACK_PUSH in src/generic/regsave.h # retrieve fibre's register context
rbp = gdb.Value(rsp).cast(ptype).dereference() ftype = gdb.lookup_type('Fibre').pointer()
rsp = rsp + 8 ptr = gdb.Value(arg).cast(ftype)
rip = gdb.Value(rsp).cast(ptype).dereference() rsp = ptr['stackPointer']
if (rsp == 0):
if not silent:
print("cannot access stack pointer - active in different thread?")
return False
ptype = gdb.lookup_type('uintptr_t').pointer()
rsp = rsp + 40 # cf. STACK_PUSH in src/generic/regsave.h
rbp = gdb.Value(rsp).cast(ptype).dereference()
rsp = rsp + 8
rip = gdb.Value(rsp).cast(ptype).dereference()
# enable fibre's register context # enable fibre's register context
gdb.execute("set $rsp = " + str(rsp)) gdb.execute("set $rsp = " + str(rsp))
gdb.execute("set $rbp = " + str(rbp)) gdb.execute("set $rbp = " + str(rbp))
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment