Coverage for cuda / core / _utils / enum_explanations_helpers.py: 96.55%
58 statements
« prev ^ index » next coverage.py v7.14.0, created at 2026-05-22 01:37 +0000
« prev ^ index » next coverage.py v7.14.0, created at 2026-05-22 01:37 +0000
1# SPDX-FileCopyrightText: Copyright (c) 2025-2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2# SPDX-License-Identifier: Apache-2.0
4"""Internal support for error-enum explanations.
6``cuda_core`` keeps frozen 13.1.1 fallback tables for older ``cuda-bindings``
7releases. Driver/runtime error enums carry usable ``__doc__`` text starting in
8the 12.x backport line at ``cuda-bindings`` 12.9.6, and in the mainline 13.x
9series at ``cuda-bindings`` 13.2.0. This module decides which source to use
10and normalizes generated docstrings so user-facing ``CUDAError`` messages stay
11presentable.
13The cleanup rules here were derived while validating generated enum docstrings
14in PR #1805. Keep them narrow and remove them when codegen quirks or fallback
15support are no longer needed.
16"""
18from __future__ import annotations
20import importlib.metadata
21import re
22from collections.abc import Callable
23from typing import Any
25_MIN_12X_BINDING_VERSION_FOR_ENUM_DOCSTRINGS = (12, 9, 6)
26_MIN_13X_BINDING_VERSION_FOR_ENUM_DOCSTRINGS = (13, 2, 0)
27_RST_INLINE_ROLE_RE = re.compile(r":(?:[a-z]+:)?[a-z]+:`([^`]+)`")
28_WORDWRAP_HYPHEN_AFTER_RE = re.compile(r"(?<=[0-9A-Za-z_])- (?=[0-9A-Za-z_])")
29_WORDWRAP_HYPHEN_BEFORE_RE = re.compile(r"(?<=[0-9A-Za-z_]) -(?=[0-9A-Za-z_])")
30_ExplanationTable = dict[int, str | tuple[str, ...]]
31_ExplanationTableLoader = Callable[[], _ExplanationTable]
34# ``version.pyx`` cannot be reused here (circular import via ``cuda_utils``).
35def _binding_version() -> tuple[int, int, int]:
36 """Return the installed ``cuda-bindings`` version, or a conservative old value."""
37 try:
38 version = importlib.metadata.version("cuda-bindings")
39 except importlib.metadata.PackageNotFoundError:
40 return (0, 0, 0) # For very old versions of cuda-python
42 parts = version.partition("+")[0].split(".")[:3]
43 return tuple(int(v) for v in parts)
46def _binding_version_has_usable_enum_docstrings(version: tuple[int, int, int]) -> bool:
47 """Whether released bindings are known to carry usable error-enum ``__doc__`` text."""
48 return ( 1aTdefgUVWXYZNOPQbcR
49 _MIN_12X_BINDING_VERSION_FOR_ENUM_DOCSTRINGS <= version < (13, 0, 0)
50 or version >= _MIN_13X_BINDING_VERSION_FOR_ENUM_DOCSTRINGS
51 )
54def _fix_hyphenation_wordwrap_spacing(s: str) -> str:
55 """Remove spaces around hyphens introduced by line wrapping in generated ``__doc__`` text.
57 This targets asymmetric wrap artifacts such as ``non- linear`` or
58 ``GPU- Direct`` while leaving intentional ``a - b`` separators alone.
59 """
60 prev = None 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
61 while prev != s: 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
62 prev = s 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
63 s = _WORDWRAP_HYPHEN_AFTER_RE.sub("-", s) 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
64 s = _WORDWRAP_HYPHEN_BEFORE_RE.sub("-", s) 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
65 return s 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
68def clean_enum_member_docstring(doc: str | None) -> str | None:
69 """Turn an enum member ``__doc__`` into plain text.
71 The generated enum docstrings are already close to user-facing prose, but
72 they may contain Sphinx inline roles, line wrapping, or a small known
73 codegen defect. Normalize only those differences so the text is suitable
74 for error messages.
75 """
76 if doc is None: 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKSbc
77 return None 1S
78 s = doc 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
79 # Known codegen bug on cudaErrorIncompatibleDriverContext. Remove once fixed
80 # in cuda-bindings code generation.
81 s = s.replace("\n:py:obj:`~.Interactions`", ' "Interactions ') 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
82 # Drop a leading "~." or "." after removing the surrounding RST inline role.
83 s = _RST_INLINE_ROLE_RE.sub(lambda m: re.sub(r"^~?\.", "", m.group(1)), s) 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
84 # Strip simple bold emphasis markers.
85 s = re.sub(r"\*\*([^*]+)\*\*", r"\1", s) 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
86 # Strip simple italic emphasis markers.
87 s = re.sub(r"\*([^*]+)\*", r"\1", s) 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
88 # Collapse wrapped lines and repeated spaces.
89 s = re.sub(r"\s+", " ", s).strip() 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
90 s = _fix_hyphenation_wordwrap_spacing(s) 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
91 return s 1ijklmnopqrhdefgstuvwxyzABCDEFGHIJKbc
94class DocstringBackedExplanations:
95 """Compatibility shim exposing enum-member ``__doc__`` text via ``dict.get``.
97 Keeps the existing ``.get(int(error))`` lookup shape used by ``cuda_utils.pyx``.
98 """
100 __slots__ = ("_enum_type",)
102 def __init__(self, enum_type: Any) -> None:
103 self._enum_type = enum_type 1aLMNbcR
105 def get(self, code: int, default: str | None = None) -> str | None:
106 try: 1ijklmnopqrhdefgstuLMbc
107 member = self._enum_type(code) 1ijklmnopqrhdefgstuLMbc
108 except ValueError: 1M
109 return default 1M
111 raw_doc = member.__doc__ 1ijklmnopqrhdefgstuLbc
112 if raw_doc is None: 1ijklmnopqrhdefgstuLbc
113 return default 1hL
115 return clean_enum_member_docstring(raw_doc) 1ijklmnopqrhdefgstubc
118def get_best_available_explanations(
119 enum_type: Any,
120 fallback: _ExplanationTable | _ExplanationTableLoader,
121) -> DocstringBackedExplanations | _ExplanationTable:
122 """Pick one explanation source per bindings version.
124 Use enum-member ``__doc__`` only for bindings versions known to expose
125 usable per-member text (12.9.6+ in the 12.x backport line, 13.2.0+ in the
126 13.x mainline). Otherwise keep using the frozen 13.1.1 fallback tables.
127 """
128 if not _binding_version_has_usable_enum_docstrings(_binding_version()): 1aNOPQbcR
129 if callable(fallback): 1OPQ
130 return fallback() 1O
131 return fallback 1PQ
132 return DocstringBackedExplanations(enum_type) 1aNbcR