Coverage for cuda/core/_memory/_legacy.py: 98.39%
62 statements
« prev ^ index » next coverage.py v7.14.1, created at 2026-06-13 01:38 +0000
« prev ^ index » next coverage.py v7.14.1, created at 2026-06-13 01:38 +0000
1# SPDX-FileCopyrightText: Copyright (c) 2024-2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
2#
3# SPDX-License-Identifier: Apache-2.0
5from __future__ import annotations
7from typing import TYPE_CHECKING
9if TYPE_CHECKING:
10 from cuda.core._stream import Stream
11 from cuda.core.graph import GraphBuilder
12 from cuda.core.typing import DevicePointerType
14from cuda.core._memory._buffer import Buffer, MemoryResource
15from cuda.core._utils.cuda_utils import (
16 _check_driver_error as raise_if_driver_error,
17)
18from cuda.core._utils.cuda_utils import (
19 driver,
20)
22__all__ = ["LegacyPinnedMemoryResource"]
25class LegacyPinnedMemoryResource(MemoryResource):
26 """Create a pinned memory resource that uses legacy cuMemAllocHost/cudaMallocHost
27 APIs.
28 """
30 # TODO: support creating this MR with flags that are later passed to cuMemHostAlloc?
32 def allocate(self, size: int, *, stream: Stream | GraphBuilder | None = None) -> Buffer:
33 """Allocate a buffer of the requested size.
35 ``cuMemAllocHost`` is synchronous, so this resource ignores any
36 supplied stream. The argument is accepted (and validated when
37 non-``None``) for interface conformance with stream-ordered
38 memory resources.
40 Parameters
41 ----------
42 size : int
43 The size of the buffer to allocate, in bytes.
44 stream : Stream, optional
45 Keyword-only. Validated when provided but otherwise unused.
47 Returns
48 -------
49 Buffer
50 The allocated buffer object, which is accessible on both host and device.
51 """
52 from cuda.core._stream import Stream_accept 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL=?./MNO:PQRSTUVWXYZ0123456789!#$%'()*+;,
54 if stream is not None: 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL=?./MNO:PQRSTUVWXYZ0123456789!#$%'()*+;,
55 Stream_accept(stream)
56 if size: 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL=?./MNO:PQRSTUVWXYZ0123456789!#$%'()*+;,
57 err, ptr = driver.cuMemAllocHost(size) 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL=?./MNO:PQRSTUVWXYZ0123456789!#$%'()*+,
58 raise_if_driver_error(err) 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL=?./MNO:PQRSTUVWXYZ0123456789!#$%'()*+,
59 else:
60 ptr = 0 1;
61 return Buffer._init(ptr, size, self) 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL=?./MNO:PQRSTUVWXYZ0123456789!#$%'()*+;,
63 def deallocate(self, ptr: DevicePointerType, size: int, *, stream: Stream | GraphBuilder | None = None) -> None:
64 """Deallocate a buffer previously allocated by this resource.
66 Parameters
67 ----------
68 ptr : :obj:`~_memory.DevicePointerType`
69 The pointer or handle to the buffer to deallocate.
70 size : int
71 The size of the buffer to deallocate, in bytes.
72 stream : Stream, optional
73 Keyword-only. If provided, ``stream.sync()`` is called before the
74 host allocation is freed. ``None`` skips the sync.
75 """
76 from cuda.core._stream import Stream_accept 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL./MNO:PQRSTUVWXYZ0123456789!#$%'()*+;@,
78 if stream is not None: 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL./MNO:PQRSTUVWXYZ0123456789!#$%'()*+;@,
79 Stream_accept(stream).sync() 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL./MNO:PQRSTUVWXYZ0123456789!#$%'()*+;@,
81 if size: 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL./MNO:PQRSTUVWXYZ0123456789!#$%'()*+;@,
82 (err,) = driver.cuMemFreeHost(ptr) 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL./MNO:PQRSTUVWXYZ0123456789!#$%'()*+@,
83 raise_if_driver_error(err) 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKL./MNO:PQRSTUVWXYZ0123456789!#$%'()*+@,
85 @property
86 def is_device_accessible(self) -> bool:
87 """bool: this memory resource provides device-accessible buffers."""
88 return True 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!#$%'()*+,
90 @property
91 def is_host_accessible(self) -> bool:
92 """bool: this memory resource provides host-accessible buffers."""
93 return True 1cdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!#$%'()*+,
95 @property
96 def device_id(self) -> int:
97 """This memory resource is not bound to any GPU."""
98 raise RuntimeError("a pinned memory resource is not bound to any GPU") 1[
101class _SynchronousMemoryResource(MemoryResource):
102 __slots__ = ("_device_id",)
104 def __init__(self, device_id: int) -> None:
105 from .._device import Device 1b-
107 self._device_id = Device(device_id).device_id 1b-
109 def allocate(self, size: int, *, stream: Stream | GraphBuilder | None = None) -> Buffer:
110 # cuMemAlloc is synchronous; stream is accepted (and validated)
111 # for interface conformance but not used.
112 from cuda.core._stream import Stream_accept 1b-
114 if stream is not None: 1b-
115 Stream_accept(stream) 1b
116 if size: 1b-
117 err, ptr = driver.cuMemAlloc(size) 1b-
118 raise_if_driver_error(err) 1b-
119 else:
120 ptr = 0 1b
121 return Buffer._init(ptr, size, self) 1b-
123 def deallocate(self, ptr: DevicePointerType, size: int, *, stream: Stream | GraphBuilder | None = None) -> None:
124 from cuda.core._stream import Stream_accept 1b-
126 if stream is not None: 1b-
127 Stream_accept(stream).sync() 1b-
128 if size: 1b-
129 (err,) = driver.cuMemFree(ptr) 1b-
130 raise_if_driver_error(err) 1b-
132 @property
133 def is_device_accessible(self) -> bool:
134 return True 1b
136 @property
137 def is_host_accessible(self) -> bool:
138 return False 1b
140 @property
141 def device_id(self) -> int:
142 return self._device_id 1b