summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorWilliam Schultz <william.schultz@mongodb.com>2018-12-11 15:16:03 -0500
committerWilliam Schultz <william.schultz@mongodb.com>2018-12-11 15:19:23 -0500
commit35da9daad146b3e497ff7c41ab56d563dd0c36e7 (patch)
tree30420fe3688b8d3a9699ba0691cba4a3e396307a
parentcee9c4deed8bbf0c612b465be4625d5d0775d204 (diff)
downloadmongo-35da9daad146b3e497ff7c41ab56d563dd0c36e7.tar.gz
SERVER-38045 Add GDB tools for dumping the SessionCatalog in the hang analyzer
-rw-r--r--buildscripts/gdb/mongo.py236
-rw-r--r--buildscripts/gdb/mongo_printers.py29
-rwxr-xr-xbuildscripts/hang_analyzer.py3
3 files changed, 267 insertions, 1 deletions
diff --git a/buildscripts/gdb/mongo.py b/buildscripts/gdb/mongo.py
index cfc28867639..af8654e9e2d 100644
--- a/buildscripts/gdb/mongo.py
+++ b/buildscripts/gdb/mongo.py
@@ -7,6 +7,26 @@ import sys
import gdb
+# pylint: disable=invalid-name,wildcard-import
+try:
+ # Try to find and load the C++ pretty-printer library.
+ import glob
+ pp = glob.glob("/opt/mongodbtoolchain/v2/share/gcc-*/python/libstdcxx/v6/printers.py")
+ printers = pp[0]
+ path = os.path.dirname(os.path.dirname(os.path.dirname(printers)))
+ sys.path.insert(0, path)
+ from libstdcxx.v6.printers import *
+ print("Loaded libstdc++ pretty printers from '%s'" % printers)
+except ImportError as e:
+ print("Failed to load the libstdc++ pretty printers: " + str(e))
+# pylint: enable=invalid-name,wildcard-import
+
+if sys.version_info[0] >= 3:
+ # GDB only permits converting a gdb.Value instance to its numerical address when using the
+ # long() constructor in Python 2 and not when using the int() constructor. We define the
+ # 'long' class as an alias for the 'int' class in Python 3 for compatibility.
+ long = int # pylint: disable=redefined-builtin,invalid-name
+
def get_process_name():
"""Return the main binary we are attached to."""
@@ -49,6 +69,88 @@ def get_current_thread_name():
return fallback_name
+def get_global_service_context():
+ """Return the global ServiceContext object."""
+ return gdb.parse_and_eval("'mongo::(anonymous namespace)::globalServiceContext'").dereference()
+
+
+def get_session_catalog():
+ """Return the global SessionCatalog object.
+
+ Returns None if no SessionCatalog could be found.
+ """
+ # The SessionCatalog is a decoration on the ServiceContext.
+ session_catalog_dec = get_decoration(get_global_service_context(), "mongo::SessionCatalog")
+ if session_catalog_dec is None:
+ return None
+ return session_catalog_dec[1]
+
+
+def get_decorations(obj):
+ """Return an iterator to all decorations on a given object.
+
+ Each object returned by the iterator is a tuple whose first element is the type name of the
+ decoration and whose second element is the decoration object itself.
+
+ TODO: De-duplicate the logic between here and DecorablePrinter. This code was copied from there.
+ """
+ type_name = str(obj.type).replace(" ", "")
+ decorable = obj.cast(gdb.lookup_type("mongo::Decorable<{}>".format(type_name)))
+ decl_vector = decorable["_decorations"]["_registry"]["_decorationInfo"]
+ start = decl_vector["_M_impl"]["_M_start"]
+ finish = decl_vector["_M_impl"]["_M_finish"]
+
+ decorable_t = decorable.type.template_argument(0)
+ decinfo_t = gdb.lookup_type('mongo::DecorationRegistry<{}>::DecorationInfo'.format(decorable_t))
+ count = long((long(finish) - long(start)) / decinfo_t.sizeof)
+
+ for i in range(count):
+ descriptor = start[i]
+ dindex = int(descriptor["descriptor"]["_index"])
+
+ type_name = str(descriptor["constructor"])
+ type_name = type_name[0:len(type_name) - 1]
+ type_name = type_name[0:type_name.rindex(">")]
+ type_name = type_name[type_name.index("constructAt<"):].replace("constructAt<", "")
+ # get_unique_ptr should be loaded from 'mongo_printers.py'.
+ decoration_data = get_unique_ptr(decorable["_decorations"]["_decorationData"]) # pylint: disable=undefined-variable
+
+ if type_name.endswith('*'):
+ type_name = type_name[0:len(type_name) - 1]
+ type_name = type_name.rstrip()
+ type_t = gdb.lookup_type(type_name)
+ obj = decoration_data[dindex].cast(type_t)
+ yield (type_name, obj)
+
+
+def get_decoration(obj, type_name):
+ """Find a decoration on 'obj' where the string 'type_name' is in the decoration's type name.
+
+ Returns a tuple whose first element is the type name of the decoration and whose
+ second is the decoration itself. If there are multiple such decorations, returns the first one
+ that matches. Returns None if no matching decorations were found.
+ """
+ for dec_type_name, dec in get_decorations(obj):
+ if type_name in dec_type_name:
+ return (dec_type_name, dec)
+ return None
+
+
+def get_boost_optional(optional):
+ """
+ Retrieve the value stored in a boost::optional type, if it is non-empty.
+
+ Returns None if the optional is empty.
+
+ TODO: Import the boost pretty printers instead of using this custom function.
+ """
+ if not optional['m_initialized']:
+ return None
+ value_ref_type = optional.type.template_argument(0).pointer()
+ storage = optional['m_storage']['dummy_']['data']
+ return storage.cast(value_ref_type).dereference()
+
+
###################################################################################################
#
# Commands
@@ -91,6 +193,140 @@ class DumpGlobalServiceContext(gdb.Command):
DumpGlobalServiceContext()
+class GetMongoDecoration(gdb.Command):
+ """
+ Search for a decoration on an object by typename and print it e.g.
+
+ (gdb) mongo-decoration opCtx ReadConcernArgs
+
+ would print out a decoration on opCtx whose type name contains the string "ReadConcernArgs".
+ """
+
+ def __init__(self):
+ """Initialize GetMongoDecoration."""
+ RegisterMongoCommand.register(self, "mongo-decoration", gdb.COMMAND_DATA)
+
+ def invoke(self, args, _from_tty): # pylint: disable=unused-argument,no-self-use
+ """Invoke GetMongoDecoration."""
+ argarr = args.split(" ")
+ if len(argarr) < 2:
+ raise ValueError("Must provide both an object and type_name argument.")
+
+ # The object that is decorated.
+ expr = argarr[0]
+ # The substring of the decoration type that is to be printed.
+ type_name_substr = argarr[1]
+ dec = get_decoration(gdb.parse_and_eval(expr), type_name_substr)
+ if dec:
+ (type_name, obj) = dec
+ print(type_name, obj)
+ else:
+ print("No decoration found whose type name contains '" + type_name_substr + "'.")
+
+
+# Register command
+GetMongoDecoration()
+
+
+class DumpMongoDSessionCatalog(gdb.Command):
+ """Print out the mongod SessionCatalog, which maintains a table of all Sessions.
+
+ Prints out interesting information from TransactionParticipants too, which are decorations on
+ the Session. If no arguments are provided, dumps out all sessions. Can optionally provide a
+ session id argument. In that case, will only print the session for the specified id, if it is
+ found. e.g.
+
+ (gdb) dump-sessions "32cb9e84-98ad-4322-acf0-e055cad3ef73"
+
+ """
+
+ def __init__(self):
+ """Initialize DumpMongoDSessionCatalog."""
+ RegisterMongoCommand.register(self, "mongod-dump-sessions", gdb.COMMAND_DATA)
+
+ def invoke(self, args, _from_tty): # pylint: disable=unused-argument,no-self-use,too-many-locals
+ """Invoke DumpMongoDSessionCatalog."""
+ # See if a particular session id was specified.
+ argarr = args.split(" ")
+ lsid_to_find = None
+ if argarr:
+ lsid_to_find = argarr[0]
+
+ # Get the SessionCatalog and the table of sessions.
+ session_catalog = get_session_catalog()
+ if session_catalog is None:
+ print(
+ "No SessionCatalog object was found on the ServiceContext. Not dumping any sessions."
+ )
+ return
+ lsid_map = session_catalog["_sessions"]
+ session_kv_pairs = list(StdHashtableIterator(lsid_map['_M_h'])) # pylint: disable=undefined-variable
+ print("Dumping %d Session objects from the SessionCatalog" % len(session_kv_pairs))
+
+ # Optionally search for a specified session, based on its id.
+ if lsid_to_find:
+ print("Only printing information for session " + lsid_to_find + ", if found.")
+ lsids_to_print = [lsid_to_find]
+ else:
+ lsids_to_print = [str(s['first']['_id']) for s in session_kv_pairs]
+
+ for sess_kv in session_kv_pairs:
+ # The Session is stored inside the SessionRuntimeInfo object.
+ session_runtime_info = sess_kv['second']['_M_ptr'].dereference()
+ session = session_runtime_info['session']
+ # TODO: Add a custom pretty printer for LogicalSessionId.
+ lsid_str = str(session['_sessionId']['_id'])
+
+ # If we are only interested in a specific session, then we print out the entire Session
+ # object, to aid more detailed debugging.
+ if lsid_str == lsid_to_find:
+ print("SessionId", "=", lsid_str)
+ print(session)
+ # Terminate if this is the only session we care about.
+ break
+
+ # Only print info for the necessary sessions.
+ if lsid_str not in lsids_to_print:
+ continue
+
+ # If we are printing multiple sessions, we only print the most interesting fields from
+ # the Session object for the sake of efficiency. We print the session id string first so
+ # the session is easily identifiable.
+ print("Session (" + str(session.address) + "):")
+ print("SessionId", "=", lsid_str)
+ session_fields_to_print = ['_sessionId', '_checkoutOpCtx', '_killRequested']
+ for field in session_fields_to_print:
+ print(field, "=", session[field])
+
+ # Print the information from a TransactionParticipant if a session has one. Otherwise
+ # we just print the session's id and nothing else.
+ txn_part_dec = get_decoration(session, "TransactionParticipant")
+ if txn_part_dec:
+ # Only print the most interesting fields for debugging transactions issues.
+ txn_part = txn_part_dec[1]
+ fields_to_print = ['_txnState', '_activeTxnNumber']
+ print("TransactionParticipant (" + str(txn_part.address) + "):")
+ for field in fields_to_print:
+ print(field, "=", txn_part[field])
+
+ # The '_txnResourceStash' field is a boost::optional so we unpack it
+ # manually if it is non-empty. We are only interested in its Locker object for now.
+ # TODO: Load the boost pretty printers so the object will be printed clearly
+ # by default, without the need for special unpacking.
+ val = get_boost_optional(txn_part['_txnResourceStash'])
+ if val:
+ locker_addr = val["_locker"]["_M_t"]['_M_head_impl']
+ print('_txnResourceStash._locker', "@", locker_addr)
+ else:
+ print('_txnResourceStash', "=", None)
+ # Separate sessions by a newline.
+ print("")
+
+
+# Register command
+DumpMongoDSessionCatalog()
+
+
class MongoDBDumpLocks(gdb.Command):
"""Dump locks in mongod process."""
diff --git a/buildscripts/gdb/mongo_printers.py b/buildscripts/gdb/mongo_printers.py
index 95070fd2ee7..0a5245a66d3 100644
--- a/buildscripts/gdb/mongo_printers.py
+++ b/buildscripts/gdb/mongo_printers.py
@@ -4,6 +4,7 @@ from __future__ import print_function
import re
import struct
import sys
+import uuid
import gdb.printing
@@ -17,6 +18,12 @@ except ImportError as err:
print("Check with the pip command if pymongo 3.x is installed.")
bson = None
+if sys.version_info[0] >= 3:
+ # GDB only permits converting a gdb.Value instance to its numerical address when using the
+ # long() constructor in Python 2 and not when using the int() constructor. We define the
+ # 'long' class as an alias for the 'int' class in Python 3 for compatibility.
+ long = int # pylint: disable=redefined-builtin,invalid-name
+
def get_unique_ptr(obj):
"""Read the value of a libstdc++ std::unique_ptr."""
@@ -147,6 +154,25 @@ class BSONObjPrinter(object):
return "%s BSONObj %s bytes @ %s" % (ownership, size, self.ptr)
+class UUIDPrinter(object):
+ """Pretty-printer for mongo::UUID."""
+
+ def __init__(self, val):
+ """Initialize UUIDPrinter."""
+ self.val = val
+
+ @staticmethod
+ def display_hint():
+ """Display hint."""
+ return 'string'
+
+ def to_string(self):
+ """Return UUID for printing."""
+ raw_bytes = [self.val['_uuid']['_M_elems'][i] for i in range(16)]
+ uuid_hex_bytes = [hex(int(b))[2:].zfill(2) for b in raw_bytes]
+ return str(uuid.UUID("".join(uuid_hex_bytes)))
+
+
class UnorderedFastKeyTablePrinter(object):
"""Pretty-printer for mongo::UnorderedFastKeyTable<>."""
@@ -204,7 +230,7 @@ class DecorablePrinter(object):
decorable_t = val.type.template_argument(0)
decinfo_t = gdb.lookup_type(
'mongo::DecorationRegistry<{}>::DecorationInfo'.format(decorable_t))
- self.count = int((int(finish) - int(self.start)) / decinfo_t.sizeof)
+ self.count = long((long(finish) - long(self.start)) / decinfo_t.sizeof)
@staticmethod
def display_hint():
@@ -462,6 +488,7 @@ def build_pretty_printer():
pp.add('StringData', 'mongo::StringData', False, StringDataPrinter)
pp.add('UnorderedFastKeyTable', 'mongo::UnorderedFastKeyTable', True,
UnorderedFastKeyTablePrinter)
+ pp.add('UUID', 'mongo::UUID', False, UUIDPrinter)
pp.add('__wt_cursor', '__wt_cursor', False, WtCursorPrinter)
pp.add('__wt_session_impl', '__wt_session_impl', False, WtSessionImplPrinter)
pp.add('__wt_txn', '__wt_txn', False, WtTxnPrinter)
diff --git a/buildscripts/hang_analyzer.py b/buildscripts/hang_analyzer.py
index 4935fb92754..e96dbc2f841 100755
--- a/buildscripts/hang_analyzer.py
+++ b/buildscripts/hang_analyzer.py
@@ -330,6 +330,7 @@ class GDBDumper(object):
mongodb_waitsfor_graph = "mongodb-waitsfor-graph debugger_waitsfor_%s_%d.gv" % \
(process_name, pid)
mongodb_javascript_stack = "mongodb-javascript-stack"
+ mongod_dump_sessions = "mongod-dump-sessions"
# The following MongoDB python extensions do not run on Solaris.
if sys.platform.startswith("sunos"):
@@ -342,6 +343,7 @@ class GDBDumper(object):
mongodb_show_locks = ""
mongodb_waitsfor_graph = ""
mongodb_javascript_stack = ""
+ mongod_dump_sessions = ""
if not logger.mongo_process_filename:
raw_stacks_commands = []
@@ -378,6 +380,7 @@ class GDBDumper(object):
mongodb_show_locks,
mongodb_waitsfor_graph,
mongodb_javascript_stack,
+ mongod_dump_sessions,
"set confirm off",
"quit",
]