Files
Mark Meserve 8760fb4976 attach: Formalize ROCAttach API (#1653)
* attach: Formalize ROCAttach API

- Make ROCAttach public with public headers
- Change detach to take a PID
  - attach and detach are now reentrant
- Cleanup of states and signal handling in ptrace session
- Fixes mixed up definition of ROCPROF_ATTACH_TOOL_LIBRARY
  - ROCPROF_ATTACH_TOOL_LIBRARY now always means the tool library loaded by the attachment target
  - ROCPROF_ATTACH_LIBRARY refers to the library used to perform attachment
- Add direct call of rocprof-attach
- Fix python library call of rocprof-attach
  - Function now named attach(), changed from main()

* attach: rocprof-compute ROCAttach updates

- Update to new library names
- Correct usage of C lib detach

* attach: add test for rocattach

- Disable ASan, TSan, and UBSan for the new parallel-attach test
- Lower log level for LSan tests, existing behavior from other tests

---------

Co-authored-by: Ammar ELWazir <aelwazir@amd.com>
2026-01-15 14:32:14 -06:00

197 rivejä
7.1 KiB
Python
Executable File

#!/usr/bin/env python3
# MIT License
#
# Copyright (c) 2025 Advanced Micro Devices, Inc. All rights reserved.
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
# THE SOFTWARE.
import argparse
import ctypes
import os
import signal
import sys
import time
ROCPROF_ATTACH_DIR = os.path.dirname(os.path.realpath(__file__))
ROCM_DIR = os.path.dirname(ROCPROF_ATTACH_DIR)
ROCPROF_ATTACH_LIBRARY = f"{ROCM_DIR}/lib/librocprofiler-sdk-rocattach.so"
def parse_arguments(args=None):
def format_help(formatter, w=120, h=40):
"""Return a wider HelpFormatter, if possible."""
try:
kwargs = {"width": w, "max_help_position": h}
formatter(None, **kwargs)
return lambda prog: formatter(prog, **kwargs)
except TypeError:
return formatter
usage_examples = """
%(prog)s, e.g.
$ rocprof-attach -p <pid> -t <tool library> [-a <attach tool library> -d <msec duration>]
$ rocprof-attach -p 12345 -t path/to/your-tool-library.so -d 5000
"""
parser = argparse.ArgumentParser(
description="rocprofiler-sdk attachment profiler for custom tool libraries ",
usage="%(prog)s [options] ",
epilog=usage_examples,
formatter_class=format_help(argparse.RawTextHelpFormatter),
)
parser.add_argument(
"-p",
"--pid",
"--attach",
help="""Attachment target's process identifier (PID).
Can also be specified in environment variable ROCPROF_ATTACH_PID. This option overrides the environment variable if both are set.""",
type=int,
required=False,
default=os.environ.get("ROCPROF_ATTACH_PID", None),
)
parser.add_argument(
"-t",
"--attach-tool-library",
help="""Comma delimited list of tool libraries to use during attachment.
Can also be specified in environment variable ROCPROF_ATTACH_TOOL_LIBRARY. This option overrides the environment variable if both are set.""",
type=str,
required=False,
default=os.environ.get("ROCPROF_ATTACH_TOOL_LIBRARY", None),
)
parser.add_argument(
"-d",
"--attach-duration-msec",
help="""Sets the amount of time in milliseconds the profiler will be attached before detaching. When unset, the profiler will wait until Enter is pressed or SIGINT (Ctrl+C) to detach.
Can also be specified in environment variable ROCPROF_ATTACH_DURATION. This option overrides the environment variable if both are set.""",
type=int,
required=False,
default=os.environ.get("ROCPROF_ATTACH_DURATION", None),
)
advanced_options = parser.add_argument_group("Advanced options")
advanced_options.add_argument(
"--attach-library",
help="""Library used to attach and detach from the target process. Default will work for nearly all configurations.
Defaults to rocprofiler-sdk-rocattach.so from this ROCm install, i.e. <ROCmdirectory>/lib/rocprofiler-sdk-rocattach.so
Can also be specified in environment variable ROCPROF_ATTACH_LIBRARY. This option overrides the environment variable if both are set.""",
type=str,
required=False,
default=os.environ.get("ROCPROF_ATTACH_LIBRARY", ROCPROF_ATTACH_LIBRARY),
)
return parser.parse_args(args)
def attach(
pid,
attach_tool_library,
attach_duration_msec,
attach_library=ROCPROF_ATTACH_LIBRARY,
):
if pid is None:
raise RuntimeError("rocprof-attach called with no PID specified")
if attach_tool_library is None:
raise RuntimeError("rocprof-attach called with no tool libraries specified")
tool_libraries_tokens = attach_tool_library.split(":")
for lib in tool_libraries_tokens:
if not os.path.exists(lib):
raise RuntimeError(f"rocprof-attach could not find tool library {lib}")
# Program option overrides environment variable. This is consumed by rocprofiler-sdk on the target program side.
os.environ["ROCPROF_ATTACH_TOOL_LIBRARY"] = attach_tool_library
print(f"Attaching to PID {pid} using library {attach_library}")
# Load the shared library into ctypes and attach
try:
c_lib = ctypes.CDLL(attach_library)
c_lib.rocattach_attach.restype = ctypes.c_int
c_lib.rocattach_attach.argtypes = [ctypes.c_int]
c_lib.rocattach_detach.restype = ctypes.c_int
c_lib.rocattach_detach.argtypes = [ctypes.c_int]
attach_status = c_lib.rocattach_attach(pid)
except Exception as e:
raise RuntimeError(f"Exception during library load and attachment: {e}")
if attach_status != 0:
raise RuntimeError(
f"Calling attach in {attach_library} returned non-zero status {attach_status}"
)
print(f"Attaching to PID {pid} using library {attach_library} :: success")
def detach():
print("Detaching. Please wait, this can take up to 1-2 minutes")
sys.stdout.flush()
try:
detach_status = c_lib.rocattach_detach(int(pid))
except Exception as e:
print(f"Exception during detachment: {e}")
if detach_status != 0:
print(
f"Calling detach in {attach_library} returned non-zero status {detach_status}"
)
else:
print(f"Detaching from PID {pid} using library {attach_library} :: success")
def signal_handler(sig, frame):
print("\nCaught signal SIGINT")
detach()
sys.exit(0)
signal.signal(signal.SIGINT, signal_handler)
if attach_duration_msec is None:
sys.stdout.write("Press Enter to detach...")
sys.stdout.flush() # Force the prompt to appear immediately
input() # Now wait for input
else:
print(f"Attaching for {attach_duration_msec} msec...\n")
sys.stdout.flush()
time.sleep(int(attach_duration_msec) / 1000)
detach()
def main(cmd_args=None):
args = parse_arguments(cmd_args)
attach(
pid=args.pid,
attach_tool_library=args.attach_tool_library,
attach_duration_msec=args.attach_duration_msec,
attach_library=args.attach_library,
)
return 0
if __name__ == "__main__":
ec = main(sys.argv[1:])
sys.exit(ec)