From 3504e76a9957c053b6ab862c997e1a4c89479cc8 Mon Sep 17 00:00:00 2001 From: Oguzhan Karakaya Date: Fri, 6 May 2022 10:55:39 +0200 Subject: [PATCH] gdb-oneapi support in STAT Corefile merger and generic gdb_oneapi driver is implemented. Frame parser is pretified and function names are abbreviated programmatically. Script "oneapi_gdb_test.py" is provided for testing regex parser methotds implemented for the driver. System test is done using stat serial attach mechanism. After setting STAT_GDB environment variable as gdb_oneapi, stat-cl -G -w -t 5 -I can be run to generate bt graph. Additionally, STATmain merge --oneapi -c /path/to/core can be run to generate bt graph from core file. Co-authored-by: Oguzhan Karakaya --- scripts/Makefile.am | 2 +- scripts/core_file_merger.py | 35 +++++- scripts/oneapi_gdb.py | 226 ++++++++++++++++++++++++++++++++++++ scripts/oneapi_gdb_test.py | 219 ++++++++++++++++++++++++++++++++++ scripts/stat_cuda_gdb.py | 5 +- 5 files changed, 484 insertions(+), 3 deletions(-) create mode 100644 scripts/oneapi_gdb.py create mode 100644 scripts/oneapi_gdb_test.py diff --git a/scripts/Makefile.am b/scripts/Makefile.am index 19b5e8d8..8615099b 100644 --- a/scripts/Makefile.am +++ b/scripts/Makefile.am @@ -2,7 +2,7 @@ dist_bin_SCRIPTS = run_one_sec bin_SCRIPTS = STAT stat-cl stat-script stat-core-merger attach-helper python_PYTHON = STATmain.py stat_merge_base.py STATmerge.py core_file_merger.py bg_core_backtrace.py attach_helper.py STAThelper.py if ENABLE_GDB -python_PYTHON += stat_cuda_gdb.py cuda_gdb.py gdb.py roc_gdb.py +python_PYTHON += stat_cuda_gdb.py cuda_gdb.py gdb.py roc_gdb.py oneapi_gdb.py endif if ENABLE_GUI python_PYTHON += STATGUI.py STATview.py diff --git a/scripts/core_file_merger.py b/scripts/core_file_merger.py index aee86354..68fa3955 100755 --- a/scripts/core_file_merger.py +++ b/scripts/core_file_merger.py @@ -169,6 +169,36 @@ def open(self, corefile, executable = None): self.subprocess = subprocess.Popen(args, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.STDOUT) return self.readlines() +class GdbOneapi(Gdb): + + def __init__ (self, options): + Gdb.__init__(self, options) + self.executable = "GPU" + + def open(self, corefile, executable = None): + args = [] + args.append('gdb-oneapi') + args.append('-ex') + args.append("set pagination 0") + args.append('-ex') + args.append("cd %s" %(self.directory)) + args.append('-ex') + args.append("path %s" %(self.objectpath)) + args.append('-ex') + args.append("directory %s" %(self.sourcepath)) + args.append('-ex') + args.append("set filename-display absolute") + args.append('-ex') + target_string = "target core " + self.corefile = corefile + if os.path.isabs(self.corefile): + target_string += " %s" % (self.corefile) + else: + target_string += " %s/%s" % (self.coredir, self.corefile) + args.append(target_string) + self.subprocess = subprocess.Popen(args, universal_newlines=True, stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.STDOUT) + return self.readlines() + ############################################################################### class CoreFile: @@ -353,6 +383,8 @@ def process_core(self): logging.info("Connecting gdb to the core file (%s)"%self.coreData['coreFile']) if CoreFile.__options['cuda'] == 1: gdb = CudaGdb(CoreFile.__options) + elif CoreFile.__options['oneapi'] == 1: + gdb = GdbOneapi(CoreFile.__options) else: gdb = Gdb(CoreFile.__options) executable = gdb.executable @@ -468,7 +500,7 @@ def process_core(self): in_thread,functions = True,[] #In some cases, gdb will quit the stack trace early - elif 'Backtrace stopped: ' in line: + elif 'Backtrace stopped: ' in line and CoreFile.__options['oneapi'] != 1: logging.critical("GDB: Backtrace stopped") if not force: sys.exit(2) @@ -513,6 +545,7 @@ def __init__(self): self.arg_map["force"] = StatMergerArgs.StatMergerArgElement("r", False, int, 0, "whether to force parsing on warnings and errors") self.arg_map["threads"] = StatMergerArgs.StatMergerArgElement("T", False, int, 1, "max number of threads") self.arg_map["cuda"] = StatMergerArgs.StatMergerArgElement("C", False, int, 0, "set if running on cuda cores") + self.arg_map["oneapi"] = StatMergerArgs.StatMergerArgElement("n", False, int, 0, "set if using gdb-oneapi") self.arg_map["jobid"] = self.StatMergerArgElement("j", False, None, None, "[LW] delineate traces based on Job ID in the core file") self.arg_map["exe"] = StatMergerArgs.StatMergerArgElement("x", True, str, "NULL", "[LW] the executable path") diff --git a/scripts/oneapi_gdb.py b/scripts/oneapi_gdb.py new file mode 100644 index 00000000..e4db4f5c --- /dev/null +++ b/scripts/oneapi_gdb.py @@ -0,0 +1,226 @@ +#!/bin/env python + +"""@package STATview +Visualizes dot graphs outputted by STAT.""" + +__copyright__ = """Modifications Copyright (C) 2022 Intel Corporation +SPDX-License-Identifier: BSD-3-Clause""" +__license__ = """Produced by Intel Corporation for Lawrence Livermore National Security, LLC. +Written by M. Oguzhan Karakaya oguzhan.karakaya@intel.com +LLNL-CODE-750488. +All rights reserved. + +This file is part of STAT. For details, see http://www.github.com/LLNL/STAT. Please also read STAT/LICENSE. + +Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: + + Redistributions of source code must retain the above copyright notice, this list of conditions and the disclaimer below. + Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the disclaimer (as noted below) in the documentation and/or other materials provided with the distribution. + Neither the name of the LLNS/LLNL nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL LAWRENCE LIVERMORE NATIONAL SECURITY, LLC, THE U.S. DEPARTMENT OF ENERGY OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +""" +__author__ = ["M. Oguzhan Karakaya "] +__version_major__ = 4 +__version_minor__ = 1 +__version_revision__ = 0 +__version__ = "%d.%d.%d" %(__version_major__, __version_minor__, __version_revision__) + +import sys +import os +import logging +import re +from gdb import GdbDriver, check_lines + +def expand_simd_spec(simd): + """ + Converts a merged simd specifier into a list of individual simd lanes. + `simd` string is expected to be either an individual lane number + or a comma-separated range list inside square brackets. + """ + if simd[0] == '[': + simd = simd[1:] + lanes = [] + for item1 in simd.split(" "): + for item2 in item1.split(","): + if item2.isnumeric(): + lanes += item2 + else: + pair = item2.split("-") + for num in range(int(pair[0]), int(pair[1])+1): + lanes += str(num) + return lanes + +def parse_thread_info_line(line, parse_simd_lanes = False): + """ + Extracts thread IDs from a `thread info` output produced by + Intel(R) Distribution for GDB*. See oneapi_gdb_test.py for + sample inputs and outputs. + """ + tid_single_simd = re.compile( + r"^[\s|*]*(\d+(\.?\d+)?)(?:(?::(\d+))?)\s*(?:Thread|LWP)") + tid_multiple_simd = re.compile( + r"^[\s|*]*(\d+(\.?\d+)?)(?:(?::\[(.*?)\])?)\s*(?:Thread|LWP)") + match = re.search(tid_multiple_simd, line) + if not match: + return [] + if not parse_simd_lanes: + return [ match[1] ] + if not match[3]: + match = re.search(tid_single_simd, line) + if not match[3]: + return [ match[1] ] + return [ match[1] + ":" + match[3]] + return [ match[1] + ":" + x for x in expand_simd_spec(match[3]) ] + +def clean_cpp_template_brackets_and_call_signature(string): + """ + Prettify function name output by Intel(R) Distribution for GDB*. + Long template parameters and function signatures are collapsed. + """ + sub_strings = [] + (found, begin, end, counter) = (False, 0, 0, 0) + for idx, char in enumerate(string): + if char == "<" and not found: + found = True + begin = idx + counter = 1 + elif char == "<" and found: + counter += 1 + elif char == ">" and found and counter > 1: + counter -= 1 + elif char == ">" and found and counter == 1: + end = idx + counter -= 1 + found = False + sub_strings.append(string[begin:end+1:]) + # Removing template variables from the function name + for sub_string in sub_strings: + string = string.replace(sub_string, "<...>") + # Removing call signature for the function name + ptrn = re.compile(r"\([^()]+\)(?=[^()]*$)") + match = ptrn.findall(string) + if match: + string = string.replace(match[-1], "(...)") + + return string + +def parse_frameinfo_from_backtrace(string): + """ + Takes the gdb backtrace output and call + clean_cpp_template_brackets_and_call_signature function over + extracted function names. See oneapi_gdb_test.py for + sample inputs and outputs. + """ + function = 'Unknown' + source = '??' + linenum = 0 + error = True + + frame_fromat = re.compile( + r"^#\d+\s+(?:0x\S+)*(?: in )*(.*)\s\(.*\)\s(?:at|from)\s([^:]+)(?::)?(\d+)?$") + match = frame_fromat.match(string) + if match: + error = False + function, source, linenum = match.groups() + if linenum is None: + linenum = 0 + else: + linenum = int(linenum) + function = clean_cpp_template_brackets_and_call_signature(function) + return {'function':function, + 'source':source, + 'linenum':linenum, + 'error':error} + +class OneAPIGdbDriver(GdbDriver): + """ A class to drive Intel(R) Distribution for GDB* """ + + gdb_command = 'gdb-oneapi' + parse_simd_lanes = False + if 'STAT_COLLECT_SIMD_BT' in os.environ and \ + os.environ['STAT_COLLECT_SIMD_BT'] == "1": + parse_simd_lanes = True + + def get_thread_list(self): + """ + Gets the list of threads in the target process. For + Intel(R) Distribution for GDB* this function extracts SIMD + lane information as part of a thread ID. + It is to improve the resulting representation by adding the + information on number of active SIMD lanes along with stack + trace information. + """ + logging.info('gdb-oneapi: info threads') + tids = [] + lines = self.communicate("info threads") + logging.debug('%s', repr(lines)) + for line in lines: + if "inactive" in line: + continue + tids += parse_thread_info_line(line, self.parse_simd_lanes) + return tids + + def bt(self, thread_id): + """ + Gets a backtrace from the requested thread id. + returns list of frames, where each frame is a map of attributes. + """ + logging.info('GDB thread bt ID %d', thread_id) + ret = [] + + lines = self.communicate(f"thread apply {thread_id} bt") + + lines = lines[2:] + for line_num, line in enumerate(lines): + if line[0] != '#': + continue + logging.debug('Parsing line #%d: %d', line_num, line) + ret.append(parse_frameinfo_from_backtrace(line)) + return ret + + def attach(self): + """ + Attaches to the target process and checkes if there is + an error related to gdbserver-gt at the gdb console output. + """ + logging.info('GDB attach to PID %d', self.pid) + lines = self.communicate(f"attach {self.pid}") + # We need to check if there is an error with starting + # gdbserver-gt and inform the user for possible missing + # device backtrace features. + ptrn = re.compile( + r"^intelgt\:\s*(\S*)\s*failed to start\.") + for line in lines: + match = ptrn.match(line) + if match: + gdbserver_name = match.groups()[0] + logging.error(""" +OneAPI %s initialization failed for process %d. Device backtrace will not be available.""", + gdbserver_name, self.pid) + logging.debug('%s', repr(lines)) + return check_lines(lines) + + def target_core(self, path): + """ + Intel(R) Distribution for GDB* targets a core dump + file given by path. + """ + logging.info('GDB open core %s', path) + self.communicate(f"target core {path}") + +if __name__ == "__main__": + gdb = OneAPIGdbDriver(0, 'debug', 'log.txt') + if gdb.launch() is False: + sys.stderr.write('gdb launch failed\n') + sys.exit(1) + gdb.target_core(sys.argv[1]) + thread_ids = gdb.get_thread_list() + thread_ids.sort() + for tid in thread_ids: + bt = gdb.bt(tid) + print("Thread", tid) + for i, frame in enumerate(bt): + print(f"{i}) {frame['source']}:{frame['linenum']}") + print() + gdb.close() diff --git a/scripts/oneapi_gdb_test.py b/scripts/oneapi_gdb_test.py new file mode 100644 index 00000000..627ea8de --- /dev/null +++ b/scripts/oneapi_gdb_test.py @@ -0,0 +1,219 @@ +#!/bin/env python + +"""@package STATview +Visualizes dot graphs outputted by STAT.""" + +__copyright__ = """Modifications Copyright (C) 2022 Intel Corporation +SPDX-License-Identifier: BSD-3-Clause""" +__license__ = """Produced by Intel Corporation for Lawrence Livermore National Security, LLC. +Written by M. Oguzhan Karakaya oguzhan.karakaya@intel.com +LLNL-CODE-750488. +All rights reserved. + +This file is part of STAT. For details, see http://www.github.com/LLNL/STAT. Please also read STAT/LICENSE. + +Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: + + Redistributions of source code must retain the above copyright notice, this list of conditions and the disclaimer below. + Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the disclaimer (as noted below) in the documentation and/or other materials provided with the distribution. + Neither the name of the LLNS/LLNL nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL LAWRENCE LIVERMORE NATIONAL SECURITY, LLC, THE U.S. DEPARTMENT OF ENERGY OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +""" +__author__ = ["M. Oguzhan Karakaya "] +__version_major__ = 4 +__version_minor__ = 1 +__version_revision__ = 0 +__version__ = "%d.%d.%d" %(__version_major__, __version_minor__, __version_revision__) + +import unittest +from oneapi_gdb import OneAPIGdbDriver, parse_thread_info_line, parse_frameinfo_from_backtrace + +class TestGDBParsing(unittest.TestCase): + """ + Testing OneAPIGdbDriver and utility functions + in oneapi_gdb module + """ + + def test_oneapigdbdriver_class_statics(self): + """ + Tests class static variables + """ + self.assertEqual(OneAPIGdbDriver.gdb_command, "gdb-oneapi") + self.assertFalse(OneAPIGdbDriver.parse_simd_lanes) + + def test_info_threads(self): + """ + Tests parse_thread_info_line method + """ + + tids = parse_thread_info_line( + "2.1 Thread 1.1073741824", + parse_simd_lanes=False) + self.assertEqual(tids, ["2.1"]) + + tids = parse_thread_info_line( + " * 2.1 Thread 1.1073741824", + parse_simd_lanes=False) + self.assertEqual(tids, ["2.1"]) + + tids = parse_thread_info_line( + "2.1 Thread 1.1073741824", + parse_simd_lanes=True) + self.assertEqual(tids, ["2.1"]) + + tids = parse_thread_info_line( + " * 2.1 Thread 1.1073741824", + parse_simd_lanes=True) + self.assertEqual(tids, ["2.1"]) + + tids = parse_thread_info_line( + "2.1:[1-7] Thread 1.1073741824", + parse_simd_lanes=False) + self.assertEqual(tids, ["2.1"]) + + tids = parse_thread_info_line( + "2.1:[1-7] Thread 1.1073741824", + parse_simd_lanes=True) + self.assertEqual(tids, ["2.1:1", + "2.1:2", "2.1:3", "2.1:4", + "2.1:5", "2.1:6", "2.1:7" ]) + + tids = parse_thread_info_line( + "2.1:[1,3-4,7] Thread 1.1073741824", + parse_simd_lanes=True) + self.assertEqual(tids, ["2.1:1", + "2.1:3", "2.1:4", "2.1:7" ]) + + tids = parse_thread_info_line( + "2.1:[1 3 5 7] Thread 1.1073741824", + parse_simd_lanes=True) + self.assertEqual(tids, ["2.1:1", + "2.1:3", "2.1:5", "2.1:7" ]) + + tids = parse_thread_info_line( + "96:[0-2] LWP 1073742144", + parse_simd_lanes=True) + self.assertEqual(tids, ["96:0", + "96:1", "96:2"]) + + tids = parse_thread_info_line( + "96:[0-2] LWP 1073742144", + parse_simd_lanes=False) + self.assertEqual(tids, ["96"]) + + tids = parse_thread_info_line( + " wiID=\"2.1\") at test.cpp:72", + parse_simd_lanes=True) + self.assertEqual(tids, []) + + def test_bt(self): + """ + Tests parse_frameinfo_from_backtrace method. + """ + + info = parse_frameinfo_from_backtrace("#0 SubKernel0::FunctionX1 (a=0x75569dba0000, " + "b=0x75569db70000, o=0x75569db60000, id=8) at /" + "path/to/demo/app/./sub_kernel_0.h:37") + self.assertEqual(info, { + 'function': 'SubKernel0::FunctionX1', + 'source': '/path/to/demo/app/./sub_kernel_0.h', + 'linenum': 37, + 'error': False}) + + info = parse_frameinfo_from_backtrace("#1 0x00006000f319ee80 in SubKernel0::Function" + "A (a=0x75569dba0000, b=0x75569db70000, o=0x755" + "69db60000, local_id=8, id=8) at /path/to/demo/" + "app/./sub_kernel_0.h:87") + self.assertEqual(info, { + 'function': 'SubKernel0::FunctionA', + 'source': '/path/to/demo/app/./sub_kernel_0.h', + 'linenum': 87, + 'error': False}) + + info = parse_frameinfo_from_backtrace("#2 0x00006000f31a2250 in SubKernel0::Function" + "K (a=0x75569dba0000, b=0x75569db70000, o=0x755" + "69db60000, local_id=8, group_id=0, global_id=8" + ") at /path/to/demo/app/./sub_kernel_0.h:134") + self.assertEqual(info, { + 'function': 'SubKernel0::FunctionK', + 'source': '/path/to/demo/app/./sub_kernel_0.h', + 'linenum': 134, + 'error': False}) + + info = parse_frameinfo_from_backtrace("#3 0x00006000f31c27a0 in SubKernel<0>::SubKer" + "nelImp >, c" + "l::sycl::accessor > const>(cl:" + ":sycl::accessor > const&, cl::" + "sycl::accessor > const&, cl::s" + "ycl::accessor > const&, cl::sy" + "cl::nd_item<1>&) (in_acc_a=..., in_acc_b=..., " + "out_acc=..., item=...) at /path/to/demo/app/./" + "sub_kernel_0.h:154") + self.assertEqual(info, { + 'function': 'SubKernel<...>::SubKernelImp<...>(...)', + 'source': '/path/to/demo/app/./sub_kernel_0.h', + 'linenum': 154, + 'error': False}) + + info = parse_frameinfo_from_backtrace("#4 0x00006000f31c3040 in SimpleKernel<0, cl::" + "sycl::accessor >, cl::sycl::ac" + "cessor > >::operator()(cl::syc" + "l::nd_item<1>) const (this=0x75569de02990, ite" + "m=<>) at main.cpp:24") + self.assertEqual(info, { + 'function': 'SimpleKernel<...>::operator()(...) const', + 'source': 'main.cpp', + 'linenum': 24, + 'error': False}) + + info = parse_frameinfo_from_backtrace("#5 0x00006000f3197120 in typeinfo name for Si" + "mpleKernel<0, cl::sycl::accessor >, cl::sycl::accessor > > (" + "_arg_in_acc_a=cl::sycl::id<1> = {...}, _arg_in" + "_acc_a=cl::sycl::id<1> = {...}, _arg_in_acc_a=" + "cl::sycl::id<1> = {...}, _arg_in_acc_a=cl::syc" + "l::id<1> = {...}, _arg_in_acc_b=cl::sycl::id<1" + "> = {...}, _arg_in_acc_b=cl::sycl::id<1> = {.." + ".}, _arg_in_acc_b=cl::sycl::id<1> = {...}, _ar" + "g_in_acc_b=cl::sycl::id<1> = {...}, _arg_out_a" + "cc=cl::sycl::id<1> = {...}, _arg_out_acc=cl::s" + "ycl::id<1> = {...}, _arg_out_acc=cl::sycl::id<" + "1> = {...}, _arg_out_acc=cl::sycl::id<1> = {.." + ".}) at //path/to/oneapi/compiler/../include/sy" + "cl/CL/sycl/handler.hpp:939") + self.assertEqual(info, { + 'function': 'typeinfo name for SimpleKernel<...>', + 'source': '//path/to/oneapi/compiler/../include/sycl/CL/sycl/handler.hpp', + 'linenum': 939, + 'error': False}) + +if __name__ == '__main__': + unittest.main() diff --git a/scripts/stat_cuda_gdb.py b/scripts/stat_cuda_gdb.py index 83e9f068..303f9e49 100644 --- a/scripts/stat_cuda_gdb.py +++ b/scripts/stat_cuda_gdb.py @@ -3,7 +3,7 @@ """@package STATview Visualizes dot graphs outputted by STAT.""" -__copyright__ = """Copyright (c) 2007-2018, Lawrence Livermore National Security, LLC.""" +__copyright__ = """Copyright (c) 2007-2022, Lawrence Livermore National Security, LLC.""" __license__ = """Produced at the Lawrence Livermore National Laboratory Written by Gregory Lee , Dorian Arnold, Matthew LeGendre, Dong Ahn, Bronis de Supinski, Barton Miller, Martin Schulz, Niklas Nielson, Nicklas Bo Jensen, Jesper Nielson, and Sven Karlsson. LLNL-CODE-750488. @@ -34,6 +34,7 @@ from gdb import GdbDriver from cuda_gdb import CudaGdbDriver from roc_gdb import RocGdbDriver + from oneapi_gdb import OneAPIGdbDriver except Exception as e: print(e) gdb_instances = {} @@ -55,6 +56,8 @@ def new_gdb_instance(pid, gdb_type='gdb'): gdb = CudaGdbDriver(pid, 'error', 'stderr') elif 'rocgdb' in os.environ['STAT_GDB']: gdb = RocGdbDriver(pid, 'error', 'stderr') + elif 'gdb-oneapi' in os.environ['STAT_GDB']: + gdb = OneAPIGdbDriver(pid, 'error', 'stderr') else: gdb = GdbDriver(pid, 'error', 'stderr') except: