| |
| |
|
|
| import functools |
| import os |
| import shutil |
|
|
| from cuda.pathfinder._binaries import supported_nvidia_binaries |
| from cuda.pathfinder._utils.env_vars import get_cuda_home_or_path |
| from cuda.pathfinder._utils.find_sub_dirs import find_sub_dirs_all_sitepackages |
| from cuda.pathfinder._utils.platform_aware import IS_WINDOWS |
|
|
|
|
| class UnsupportedBinaryError(Exception): |
| def __init__(self, utility: str) -> None: |
| super().__init__(utility) |
| self.utility = utility |
|
|
| def __str__(self) -> str: |
| supported_utilities = ", ".join(supported_nvidia_binaries.SUPPORTED_BINARIES) |
| return f"Binary '{self.utility}' is not supported. Supported utilities are: {supported_utilities}" |
|
|
|
|
| def _normalize_utility_name(utility_name: str) -> str: |
| """Normalize utility name by adding .exe on Windows if needed.""" |
| if IS_WINDOWS and not utility_name.lower().endswith((".exe", ".bat", ".cmd")): |
| return f"{utility_name}.exe" |
| return utility_name |
|
|
|
|
| @functools.cache |
| def find_nvidia_binary_utility(utility_name: str) -> str | None: |
| """Locate a CUDA binary utility executable. |
| |
| Args: |
| utility_name (str): The name of the binary utility to find |
| (e.g., ``"nvdisasm"``, ``"cuobjdump"``). On Windows, the ``.exe`` |
| extension will be automatically appended if not present. The function |
| also recognizes ``.bat`` and ``.cmd`` files on Windows. |
| |
| Returns: |
| str or None: Absolute path to the discovered executable, or ``None`` |
| if the utility cannot be found. The returned path is normalized |
| (absolute and with resolved separators). |
| |
| Raises: |
| UnsupportedBinaryError: If ``utility_name`` is not in the supported set |
| (see ``SUPPORTED_BINARY_UTILITIES``). |
| |
| Search order: |
| 1. **NVIDIA Python wheels** |
| |
| - Scan installed distributions (``site-packages``) for binary layouts |
| shipped in NVIDIA wheels (e.g., ``cuda-nvcc``). |
| |
| 2. **Conda environments** |
| |
| - Check Conda-style installation prefixes via ``CONDA_PREFIX`` |
| environment variable, which use platform-specific bin directory |
| layouts (``Library/bin`` on Windows, ``bin`` on Linux). |
| |
| 3. **CUDA Toolkit environment variables** |
| |
| - Use ``CUDA_HOME`` or ``CUDA_PATH`` (in that order), searching |
| ``bin/x64``, ``bin/x86_64``, and ``bin`` subdirectories on Windows, |
| or just ``bin`` on Linux. |
| |
| Note: |
| Results are cached using ``@functools.cache`` for performance. The cache |
| persists for the lifetime of the process. |
| |
| On Windows, executables are identified by their file extensions |
| (``.exe``, ``.bat``, ``.cmd``). On Unix-like systems, executables |
| are identified by the ``X_OK`` (execute) permission bit. |
| |
| Example: |
| >>> from cuda.pathfinder import find_nvidia_binary_utility |
| >>> nvdisasm = find_nvidia_binary_utility("nvdisasm") |
| >>> if nvdisasm: |
| ... print(f"Found nvdisasm at: {nvdisasm}") |
| """ |
| if utility_name not in supported_nvidia_binaries.SUPPORTED_BINARIES: |
| raise UnsupportedBinaryError(utility_name) |
|
|
| |
| candidate_dirs = supported_nvidia_binaries.SITE_PACKAGES_BINDIRS.get(utility_name, ()) |
| dirs = [] |
|
|
| for sub_dir in candidate_dirs: |
| dirs.extend(find_sub_dirs_all_sitepackages(sub_dir.split(os.sep))) |
|
|
| |
| if (conda_prefix := os.environ.get("CONDA_PREFIX")) is not None: |
| if IS_WINDOWS: |
| dirs.append(os.path.join(conda_prefix, "Library", "bin")) |
| else: |
| dirs.append(os.path.join(conda_prefix, "bin")) |
|
|
| |
| if (cuda_home := get_cuda_home_or_path()) is not None: |
| if IS_WINDOWS: |
| dirs.append(os.path.join(cuda_home, "bin", "x64")) |
| dirs.append(os.path.join(cuda_home, "bin", "x86_64")) |
| dirs.append(os.path.join(cuda_home, "bin")) |
|
|
| normalized_name = _normalize_utility_name(utility_name) |
| return shutil.which(normalized_name, path=os.pathsep.join(dirs)) |
|
|