MLIR 23.0.0git
CudaRuntimeWrappers.cpp
Go to the documentation of this file.
1//===- CudaRuntimeWrappers.cpp - MLIR CUDA API wrapper library ------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// Implements C wrappers around the CUDA library for easy linking in ORC jit.
10// Also adds some debugging helpers that are helpful when writing MLIR code to
11// run on GPUs.
12//
13//===----------------------------------------------------------------------===//
14
16
17#include <cstdio>
18
19#include "cuda.h"
20#include "cuda_bf16.h"
21#include "cuda_fp16.h"
22
23#ifdef MLIR_ENABLE_CUDA_CUSPARSE
24#include "cusparse.h"
25#ifdef MLIR_ENABLE_CUDA_CUSPARSELT
26#include "cusparseLt.h"
27#endif // MLIR_ENABLE_CUDA_CUSPARSELT
28#endif // MLIR_ENABLE_CUDA_CUSPARSE
29
30#ifdef _WIN32
31#include <malloc.h>
32#define MLIR_CUDA_WRAPPERS_EXPORT __declspec(dllexport)
33#else
34#define MLIR_CUDA_WRAPPERS_EXPORT __attribute__((visibility("default")))
35#endif // _WIN32
36
37#define CUDA_REPORT_IF_ERROR(expr) \
38 [](CUresult result) { \
39 if (!result) \
40 return; \
41 const char *name = nullptr; \
42 cuGetErrorName(result, &name); \
43 if (!name) \
44 name = "<unknown>"; \
45 fprintf(stderr, "'%s' failed with '%s'\n", #expr, name); \
46 }(expr)
47
48#define CUSPARSE_REPORT_IF_ERROR(expr) \
49 { \
50 cusparseStatus_t status = (expr); \
51 if (status != CUSPARSE_STATUS_SUCCESS) { \
52 fprintf(stderr, "cuSPARSE '%s' failed with '%s'\n", #expr, \
53 cusparseGetErrorString(status)); \
54 } \
55 }
56
57thread_local static int32_t defaultDevice = 0;
58
59/// Helper method that checks environment value for debugging.
60static bool isDebugEnabled() {
61 const char *kDebugEnvironmentVariable = "MLIR_CUDA_DEBUG";
62 static bool isEnabled = getenv(kDebugEnvironmentVariable) != nullptr;
63 return isEnabled;
64}
65
66#define debug_print(fmt, ...) \
67 do { \
68 if (isDebugEnabled()) \
69 fprintf(stderr, "%s:%d:%s(): " fmt, "CudaRuntimeWrappers.cpp", __LINE__, \
70 __func__, __VA_ARGS__); \
71 } while (0)
72
73// Returns default CUdevice
74static CUdevice getDefaultCuDevice() {
75 CUdevice device;
76 CUDA_REPORT_IF_ERROR(cuDeviceGet(&device, /*ordinal=*/defaultDevice));
77 return device;
78}
79
80// Make the primary context of the current default device current for the
81// duration
82// of the instance and restore the previous context on destruction.
84public:
86 // Static reference to CUDA primary context for device ordinal
87 // defaultDevice.
88 static CUcontext context = [] {
89 CUDA_REPORT_IF_ERROR(cuInit(/*flags=*/0));
90 CUcontext ctx;
91 // Note: this does not affect the current context.
93 cuDevicePrimaryCtxRetain(&ctx, getDefaultCuDevice()));
94 return ctx;
95 }();
96
97 CUDA_REPORT_IF_ERROR(cuCtxPushCurrent(context));
98 }
99
100 ~ScopedContext() { CUDA_REPORT_IF_ERROR(cuCtxPopCurrent(nullptr)); }
101};
102
103#ifdef MLIR_ENABLE_CUDA_CUSPARSE
104// Note that (1) Nvidia confirms the safety to share handle across multiple
105// instances, and streams. (2) Clients are responsible to call the @mgpu
106// environment initialization/destruction in a thread-safe manner, e.g.,
107// at the beginning of the program before multi-threads are created.
108static cusparseHandle_t cusparse_env = nullptr;
109
110#ifdef MLIR_ENABLE_CUDA_CUSPARSELT
111// cusparseLtHandle_t is not a pointer type, so we need an additional flag to
112// indicate whether it is initialized.
113static cusparseLtHandle_t cusparseLt_env;
114static bool cusparseLt_initiated = false;
115
116#endif // MLIR_ENABLE_CUDA_CUSPARSELT
117#endif // MLIR_ENABLE_CUDA_CUSPARSE
118
119extern "C" MLIR_CUDA_WRAPPERS_EXPORT CUmodule
120mgpuModuleLoad(void *data, size_t /*gpuBlobSize*/) {
121 ScopedContext scopedContext;
122 CUmodule module = nullptr;
123 CUDA_REPORT_IF_ERROR(cuModuleLoadData(&module, data));
124 return module;
125}
126
127extern "C" MLIR_CUDA_WRAPPERS_EXPORT CUmodule
128mgpuModuleLoadJIT(void *data, int optLevel, size_t /*assmeblySize*/) {
129 ScopedContext scopedContext;
130 CUmodule module = nullptr;
131 char jitErrorBuffer[4096] = {0};
132 CUjit_option jitOptions[] = {CU_JIT_ERROR_LOG_BUFFER,
133 CU_JIT_ERROR_LOG_BUFFER_SIZE_BYTES,
134 CU_JIT_OPTIMIZATION_LEVEL};
135 void *jitOptionsVals[] = {jitErrorBuffer,
136 reinterpret_cast<void *>(sizeof(jitErrorBuffer)),
137 reinterpret_cast<void *>(optLevel)};
138
139 CUresult result =
140 cuModuleLoadDataEx(&module, data, 3, jitOptions, jitOptionsVals);
141 if (result) {
142 fprintf(stderr, "JIT compilation failed with: '%s'\n", jitErrorBuffer);
144 }
145 return module;
146}
147
148extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuModuleUnload(CUmodule module) {
149 // Tolerate CUDA_ERROR_DEINITIALIZED: this can run from a global destructor
150 // after the CUDA primary context has already been torn down.
151 CUresult result = cuModuleUnload(module);
152 if (result != CUDA_SUCCESS && result != CUDA_ERROR_DEINITIALIZED)
154}
155
156extern "C" MLIR_CUDA_WRAPPERS_EXPORT CUfunction
157mgpuModuleGetFunction(CUmodule module, const char *name) {
158 CUfunction function = nullptr;
159 CUDA_REPORT_IF_ERROR(cuModuleGetFunction(&function, module, name));
160 return function;
161}
162
163// The wrapper uses intptr_t instead of CUDA's unsigned int to match
164// the type of MLIR's index type. This avoids the need for casts in the
165// generated MLIR code.
166extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
167mgpuLaunchKernel(CUfunction function, intptr_t gridX, intptr_t gridY,
168 intptr_t gridZ, intptr_t blockX, intptr_t blockY,
169 intptr_t blockZ, int32_t smem, CUstream stream, void **params,
170 void **extra, size_t /*paramsCount*/) {
171 ScopedContext scopedContext;
172 if (smem > 0) {
173 // Avoid checking driver as it's more expensive than if statement
174 int32_t maxShmem = 0;
175 CUdevice device = getDefaultCuDevice();
176 CUDA_REPORT_IF_ERROR(cuDeviceGet(&device, /*ordinal=*/defaultDevice));
177 CUDA_REPORT_IF_ERROR(cuDeviceGetAttribute(
178 &maxShmem, CU_DEVICE_ATTRIBUTE_MAX_SHARED_MEMORY_PER_BLOCK_OPTIN,
179 device));
180 if (maxShmem < smem) {
181 fprintf(stderr,
182 "Requested shared memory (%dkb) is larger than maximum allowed "
183 "shared memory (%dkb) for this device\n",
184 smem, maxShmem);
185 }
186 CUDA_REPORT_IF_ERROR(cuFuncSetAttribute(
187 function, CU_FUNC_ATTRIBUTE_MAX_DYNAMIC_SHARED_SIZE_BYTES, smem));
188 }
189 debug_print("Launching kernel, grid=%ld,%ld,%ld, "
190 "threads: %ld, %ld, %ld, "
191 "smem: %dkb\n",
192 gridX, gridY, gridZ, blockX, blockY, blockZ, smem);
193 CUDA_REPORT_IF_ERROR(cuLaunchKernel(function, gridX, gridY, gridZ, blockX,
194 blockY, blockZ, smem, stream, params,
195 extra));
196}
197
199 ScopedContext scopedContext;
200 CUstream stream = nullptr;
201 CUDA_REPORT_IF_ERROR(cuStreamCreate(&stream, CU_STREAM_NON_BLOCKING));
202 return stream;
203}
204
205extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuStreamDestroy(CUstream stream) {
206 CUDA_REPORT_IF_ERROR(cuStreamDestroy(stream));
207}
208
209extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
210mgpuStreamSynchronize(CUstream stream) {
211 CUDA_REPORT_IF_ERROR(cuStreamSynchronize(stream));
212}
213
214extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuStreamWaitEvent(CUstream stream,
215 CUevent event) {
216 CUDA_REPORT_IF_ERROR(cuStreamWaitEvent(stream, event, /*flags=*/0));
217}
218
220 ScopedContext scopedContext;
221 CUevent event = nullptr;
222 CUDA_REPORT_IF_ERROR(cuEventCreate(&event, CU_EVENT_DISABLE_TIMING));
223 return event;
224}
225
226extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuEventDestroy(CUevent event) {
227 CUDA_REPORT_IF_ERROR(cuEventDestroy(event));
228}
229
230extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuEventSynchronize(CUevent event) {
231 CUDA_REPORT_IF_ERROR(cuEventSynchronize(event));
232}
233
234extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuEventRecord(CUevent event,
235 CUstream stream) {
236 CUDA_REPORT_IF_ERROR(cuEventRecord(event, stream));
237}
238
239extern "C" MLIR_CUDA_WRAPPERS_EXPORT void *
240mgpuMemAlloc(uint64_t sizeBytes, CUstream stream, bool isHostShared) {
241 ScopedContext scopedContext;
242 CUdeviceptr ptr = 0;
243 if (sizeBytes == 0)
244 return reinterpret_cast<void *>(ptr);
245
246 if (isHostShared) {
248 cuMemAllocManaged(&ptr, sizeBytes, CU_MEM_ATTACH_GLOBAL));
249 return reinterpret_cast<void *>(ptr);
250 }
251 CUDA_REPORT_IF_ERROR(cuMemAlloc(&ptr, sizeBytes));
252 return reinterpret_cast<void *>(ptr);
253}
254
255extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuMemFree(void *ptr,
256 CUstream /*stream*/) {
257 CUDA_REPORT_IF_ERROR(cuMemFree(reinterpret_cast<CUdeviceptr>(ptr)));
258}
259
260extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
261mgpuMemcpy(void *dst, void *src, size_t sizeBytes, CUstream stream) {
262 CUDA_REPORT_IF_ERROR(cuMemcpyAsync(reinterpret_cast<CUdeviceptr>(dst),
263 reinterpret_cast<CUdeviceptr>(src),
264 sizeBytes, stream));
265}
266
267extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
268mgpuMemset32(void *dst, unsigned int value, size_t count, CUstream stream) {
269 CUDA_REPORT_IF_ERROR(cuMemsetD32Async(reinterpret_cast<CUdeviceptr>(dst),
270 value, count, stream));
271}
272
273extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
274mgpuMemset16(void *dst, unsigned short value, size_t count, CUstream stream) {
275 CUDA_REPORT_IF_ERROR(cuMemsetD16Async(reinterpret_cast<CUdeviceptr>(dst),
276 value, count, stream));
277}
278
279///
280/// Helper functions for writing mlir example code
281///
282
283// Allows to register byte array with the CUDA runtime. Helpful until we have
284// transfer functions implemented.
285extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
286mgpuMemHostRegister(void *ptr, uint64_t sizeBytes) {
287 ScopedContext scopedContext;
288 CUDA_REPORT_IF_ERROR(cuMemHostRegister(ptr, sizeBytes, /*flags=*/0));
289}
290
291/// Registers a memref with the CUDA runtime. `descriptor` is a pointer to a
292/// ranked memref descriptor struct of rank `rank`. Helpful until we have
293/// transfer functions implemented.
294extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
296 int64_t elementSizeBytes) {
297 // Only densely packed tensors are currently supported.
298#ifdef _WIN32
299 int64_t *denseStrides = (int64_t *)_alloca(rank * sizeof(int64_t));
300#else
301 int64_t *denseStrides = (int64_t *)alloca(rank * sizeof(int64_t));
302#endif // _WIN32
303 int64_t *sizes = descriptor->sizes;
304 for (int64_t i = rank - 1, runningStride = 1; i >= 0; i--) {
305 denseStrides[i] = runningStride;
306 runningStride *= sizes[i];
307 }
308 uint64_t sizeBytes = sizes[0] * denseStrides[0] * elementSizeBytes;
309 int64_t *strides = &sizes[rank];
310 (void)strides;
311 for (unsigned i = 0; i < rank; ++i)
312 assert(strides[i] == denseStrides[i] &&
313 "Mismatch in computed dense strides");
314
315 auto *ptr = descriptor->data + descriptor->offset * elementSizeBytes;
316 mgpuMemHostRegister(ptr, sizeBytes);
317}
318
319// Allows to unregister byte array with the CUDA runtime.
321 ScopedContext scopedContext;
322 CUDA_REPORT_IF_ERROR(cuMemHostUnregister(ptr));
323}
324
325/// Unregisters a memref with the CUDA runtime. `descriptor` is a pointer to a
326/// ranked memref descriptor struct of rank `rank`
327extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
329 StridedMemRefType<char, 1> *descriptor,
330 int64_t elementSizeBytes) {
331 auto *ptr = descriptor->data + descriptor->offset * elementSizeBytes;
333}
334
335extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuSetDefaultDevice(int32_t device) {
336 defaultDevice = device;
337}
338
339///
340/// Runtime methods using CUDA 12.0+ driver
341///
342
343#if (CUDA_VERSION >= 12000)
344
346 CUfunction function, intptr_t gridX, intptr_t gridY, intptr_t gridZ,
347 intptr_t clusterX, intptr_t clusterY, intptr_t clusterZ, intptr_t blockX,
348 intptr_t blockY, intptr_t blockZ, int32_t smem, CUstream stream,
349 void **params, void **extra) {
350 ScopedContext scopedContext;
351 if (smem > 0) {
352 int32_t maxShmem = 0;
353 CUdevice device = getDefaultCuDevice();
354 CUDA_REPORT_IF_ERROR(cuDeviceGet(&device, /*ordinal=*/defaultDevice));
355 CUDA_REPORT_IF_ERROR(cuDeviceGetAttribute(
356 &maxShmem, CU_DEVICE_ATTRIBUTE_MAX_SHARED_MEMORY_PER_BLOCK_OPTIN,
357 device));
358 if (maxShmem < smem) {
359 fprintf(stderr,
360 "Requested shared memory (%dkb) is larger than maximum allowed "
361 "shared memory (%dkb) for this device\n",
362 smem, maxShmem);
363 }
364 CUDA_REPORT_IF_ERROR(cuFuncSetAttribute(
365 function, CU_FUNC_ATTRIBUTE_MAX_DYNAMIC_SHARED_SIZE_BYTES, smem));
366 }
367
368 CUlaunchConfig config;
369 config.gridDimX = gridX;
370 config.gridDimY = gridY;
371 config.gridDimZ = gridZ;
372 config.blockDimX = blockX;
373 config.blockDimY = blockY;
374 config.blockDimZ = blockZ;
375 config.sharedMemBytes = smem;
376 config.hStream = stream;
377
378 CUlaunchAttribute launchAttrs[3];
379 int numAttrs = 0;
380
381 bool hasCluster = clusterX > 0 && clusterY > 0 && clusterZ > 0;
382 if (hasCluster) {
383 launchAttrs[numAttrs].id = CU_LAUNCH_ATTRIBUTE_CLUSTER_DIMENSION;
384 launchAttrs[numAttrs].value.clusterDim.x = clusterX;
385 launchAttrs[numAttrs].value.clusterDim.y = clusterY;
386 launchAttrs[numAttrs].value.clusterDim.z = clusterZ;
387 numAttrs++;
388
389 launchAttrs[numAttrs].id =
390 CU_LAUNCH_ATTRIBUTE_CLUSTER_SCHEDULING_POLICY_PREFERENCE;
391 launchAttrs[numAttrs].value.clusterSchedulingPolicyPreference =
392 CU_CLUSTER_SCHEDULING_POLICY_SPREAD;
393 numAttrs++;
394 }
395
396 launchAttrs[numAttrs].id = CU_LAUNCH_ATTRIBUTE_COOPERATIVE;
397 launchAttrs[numAttrs].value.cooperative = 1;
398 numAttrs++;
399
400 config.numAttrs = numAttrs;
401 config.attrs = launchAttrs;
402
403 debug_print("Launching cooperative kernel (cluster=%d), "
404 "grid=%ld,%ld,%ld, "
405 "threads: %ld, %ld, %ld, "
406 "smem: %dkb\n",
407 hasCluster, gridX, gridY, gridZ, blockX, blockY, blockZ, smem);
408
409 CUDA_REPORT_IF_ERROR(cuLaunchKernelEx(&config, function, params, extra));
410}
411
412extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuLaunchClusterKernel(
413 CUfunction function, intptr_t clusterX, intptr_t clusterY,
414 intptr_t clusterZ, intptr_t gridX, intptr_t gridY, intptr_t gridZ,
415 intptr_t blockX, intptr_t blockY, intptr_t blockZ, int32_t smem,
416 CUstream stream, void **params, void **extra, size_t /*paramsCount*/) {
417 ScopedContext scopedContext;
418 if (smem > 0) {
419 // Avoid checking driver as it's more expensive than if statement
420 int32_t maxShmem = 0;
421 CUdevice device = getDefaultCuDevice();
422 CUDA_REPORT_IF_ERROR(cuDeviceGet(&device, /*ordinal=*/defaultDevice));
423 CUDA_REPORT_IF_ERROR(cuDeviceGetAttribute(
424 &maxShmem, CU_DEVICE_ATTRIBUTE_MAX_SHARED_MEMORY_PER_BLOCK_OPTIN,
425 device));
426 if (maxShmem < smem) {
427 fprintf(stderr,
428 "Requested shared memory (%dkb) is larger than maximum allowed "
429 "shared memory (%dkb) for this device\n",
430 smem, maxShmem);
431 }
432 CUDA_REPORT_IF_ERROR(cuFuncSetAttribute(
433 function, CU_FUNC_ATTRIBUTE_MAX_DYNAMIC_SHARED_SIZE_BYTES, smem));
434 }
435 CUlaunchConfig config;
436 config.gridDimX = gridX;
437 config.gridDimY = gridY;
438 config.gridDimZ = gridZ;
439 config.blockDimX = blockX;
440 config.blockDimY = blockY;
441 config.blockDimZ = blockZ;
442 config.sharedMemBytes = smem;
443 config.hStream = stream;
444 CUlaunchAttribute launchAttr[2];
445 launchAttr[0].id = CU_LAUNCH_ATTRIBUTE_CLUSTER_DIMENSION;
446 launchAttr[0].value.clusterDim.x = clusterX;
447 launchAttr[0].value.clusterDim.y = clusterY;
448 launchAttr[0].value.clusterDim.z = clusterZ;
449 launchAttr[1].id = CU_LAUNCH_ATTRIBUTE_CLUSTER_SCHEDULING_POLICY_PREFERENCE;
450 launchAttr[1].value.clusterSchedulingPolicyPreference =
451 CU_CLUSTER_SCHEDULING_POLICY_SPREAD;
452 config.numAttrs = 2;
453 config.attrs = launchAttr;
454
455 debug_print("Launching kernel,"
456 "cluster: %ld, %ld, %ld, "
457 "grid=%ld,%ld,%ld, "
458 "threads: %ld, %ld, %ld, "
459 "smem: %dkb\n",
460 clusterX, clusterY, clusterZ, gridX, gridY, gridZ, blockX, blockY,
461 blockZ, smem);
462
463 CUDA_REPORT_IF_ERROR(cuLaunchKernelEx(&config, function, params, extra));
464}
465
466extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuTensorMapEncodeTiled(
467 CUtensorMap *tensorMap, // Tensor map object
468 CUtensorMapDataType tensorDataType, // Tensor data type
469 cuuint32_t tensorRank, // Dimensionality of tensor
470 void *globalAddress, // Starting address
471 const cuuint64_t *globalDim, // Tensor size (number of elements)
472 const cuuint64_t *globalStrides, // Stride size (in bytes)
473 const cuuint32_t *boxDim, // Traversal box (number of elments)
474 const cuuint32_t *elementStrides, // Traversal stride
475 CUtensorMapInterleave interleave, // Type of interleaved layout
476 CUtensorMapSwizzle swizzle, // Bank swizzling pattern
477 CUtensorMapL2promotion l2Promotion, // L2 promotion size
478 CUtensorMapFloatOOBfill oobFill // Padding zfill or NaN fill
479) {
480 ScopedContext scopedContext;
481 CUDA_REPORT_IF_ERROR(cuTensorMapEncodeTiled(
482 tensorMap, tensorDataType, tensorRank, globalAddress, globalDim,
483 globalStrides, boxDim, elementStrides, interleave, swizzle, l2Promotion,
484 oobFill));
485 debug_print("Created TMA descriptor\n Addr: %p\n"
486 "data type : %d\n"
487 "rank : %d\n"
488 "globalDim[5]: %zu, %zu, %zu, %zu, %zu\n"
489 "globalStrides[5]: %zu, %zu, %zu, %zu, %zu\n"
490 "boxDim[5]: %u, %u, %u, %u, %u\n"
491 "elementStrides[5]: %u, %u, %u, %u, %u\n"
492 "interleave: %u \n"
493 "swizzle: %u \n"
494 "l2Promotion: %u \n"
495 "oobFill: %u \n",
496 (void *)&tensorMap, tensorDataType, tensorRank, globalDim[0],
497 globalDim[1], globalDim[2], globalDim[3], globalDim[4],
498 globalStrides[0], globalStrides[1], globalStrides[2],
499 globalStrides[3], globalStrides[4], boxDim[0], boxDim[1],
500 boxDim[2], boxDim[3], boxDim[4], elementStrides[0],
501 elementStrides[1], elementStrides[2], elementStrides[3],
502 elementStrides[4], interleave, swizzle, l2Promotion, oobFill);
503}
504
505template <int Rank>
506void mgpuGetMemRefDataAndShape(void *rawDescriptor, char **addr,
507 uint64_t *globalDim, uint64_t *globalStrides,
508 const CUtensorMapDataType tensorDataType) {
509 auto descriptor =
510 reinterpret_cast<StridedMemRefType<char, Rank> *>(rawDescriptor);
511 *addr = descriptor->data;
512 for (int i = 0; i < Rank; ++i) {
513 globalDim[i] = static_cast<uint64_t>(descriptor->sizes[Rank - i - 1]);
514 }
515 static constexpr int elementSizeInBytes[] = {1, 2, 4, 4, 8, 8, 2,
516 4, 8, 2, 4, 4, 4};
517 for (int i = 0; i < Rank - 1; ++i) {
518 globalStrides[i] = static_cast<uint64_t>(
519 descriptor->strides[Rank - i - 2] * elementSizeInBytes[tensorDataType]);
520 }
521}
522
523extern "C" MLIR_CUDA_WRAPPERS_EXPORT void *mgpuTensorMapEncodeTiledMemref(
524 int64_t tensorRank, // Dimensionality of tensor
525 void *rankedDescriptor, // Ranked MemRef descriptor
526 const CUtensorMapDataType tensorDataType, // Stride size (in bytes)
527 CUtensorMapInterleave interleave, // Type of interleaved layout
528 CUtensorMapSwizzle swizzle, // Bank swizzling pattern
529 CUtensorMapL2promotion l2Promotion, // L2 promotion size
530 CUtensorMapFloatOOBfill oobFill, // Padding zfill or NaN fill
531 int64_t *inputBoxDims // Tensor size (number of elements)
532) {
533 CUtensorMap tensorMap;
534
535 uint32_t boxDim[5] = {1, 1, 1, 1, 1}, elementStrides[5] = {1, 1, 1, 1, 1};
536 uint64_t globalDim[5] = {1, 1, 1, 1, 1}, globalStrides[5] = {0};
537 uint32_t tensorRank32 = uint32_t(tensorRank);
538
539 char *globalAddress = nullptr;
540 switch (tensorRank) {
541 case 1:
542 mgpuGetMemRefDataAndShape<1>(rankedDescriptor, &globalAddress, globalDim,
543 globalStrides, tensorDataType);
544 break;
545 case 2:
546 mgpuGetMemRefDataAndShape<2>(rankedDescriptor, &globalAddress, globalDim,
547 globalStrides, tensorDataType);
548 break;
549 case 3:
550 mgpuGetMemRefDataAndShape<3>(rankedDescriptor, &globalAddress, globalDim,
551 globalStrides, tensorDataType);
552 break;
553 case 4:
554 mgpuGetMemRefDataAndShape<4>(rankedDescriptor, &globalAddress, globalDim,
555 globalStrides, tensorDataType);
556 break;
557 case 5:
558 mgpuGetMemRefDataAndShape<5>(rankedDescriptor, &globalAddress, globalDim,
559 globalStrides, tensorDataType);
560 break;
561 default:
562 fprintf(
563 stderr,
564 "'mgpuTensorMapEncodeTiledMemref' failed with 'rank is too high'\n");
565 return nullptr;
566 }
567
568 for (int64_t r = 0; r < tensorRank; ++r) {
569 boxDim[r] = static_cast<uint32_t>(inputBoxDims[tensorRank - r - 1]);
570 }
571
572 ScopedContext scopedContext;
573 mgpuTensorMapEncodeTiled(&tensorMap, tensorDataType, tensorRank32,
574 globalAddress, globalDim, globalStrides, boxDim,
575 elementStrides, interleave, swizzle, l2Promotion,
576 oobFill);
577 // Copy created tensor map to device
578 CUdeviceptr dTensorMap;
579 CUDA_REPORT_IF_ERROR(cuMemAlloc(&dTensorMap, sizeof(CUtensorMap)));
580 CUDA_REPORT_IF_ERROR(cuMemcpy(dTensorMap,
581 reinterpret_cast<CUdeviceptr>(&tensorMap),
582 sizeof(CUtensorMap)));
583 return reinterpret_cast<void *>(dTensorMap);
584}
585#endif
586
587#ifdef MLIR_ENABLE_CUDA_CUSPARSE
588
589///
590/// Wrapper methods for the cuSparse library.
591///
592
593// Some macro magic to get float/double alpha and beta on host.
594// TODO: add support to passing alpha and beta as arguments
595#define ALPHABETA(dtp, alpha, beta) \
596 __nv_bfloat16(alpha##16bf) = 1.0f; \
597 __nv_bfloat16(beta##16bf) = 1.0f; \
598 __half(alpha##16f) = 1.0f; \
599 __half(beta##16f) = 1.0f; \
600 float(alpha##f) = 1.0f; \
601 float(beta##f) = 1.0f; \
602 double(alpha##d) = 1.0; \
603 double(beta##d) = 1.0; \
604 const void *(alpha##p) = nullptr; \
605 const void *(beta##p) = nullptr; \
606 if (dtp == CUDA_R_16BF || dtp == CUDA_C_16BF) { \
607 (alpha##p) = reinterpret_cast<void *>(&(alpha##16bf)); \
608 (beta##p) = reinterpret_cast<void *>(&(beta##16bf)); \
609 } else if (dtp == CUDA_R_16F || dtp == CUDA_C_16F) { \
610 (alpha##p) = reinterpret_cast<void *>(&(alpha##16f)); \
611 (beta##p) = reinterpret_cast<void *>(&(beta##16f)); \
612 } else if (dtp == CUDA_R_32F || dtp == CUDA_C_32F) { \
613 (alpha##p) = reinterpret_cast<void *>(&(alpha##f)); \
614 (beta##p) = reinterpret_cast<void *>(&(beta##f)); \
615 } else { \
616 (alpha##p) = reinterpret_cast<void *>(&(alpha##d)); \
617 (beta##p) = reinterpret_cast<void *>(&(beta##d)); \
618 }
619
620extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuCreateSparseEnv() {
621 // ScopedContext is for cuda initialization.
622 ScopedContext scopedContext;
623 assert(!cusparse_env && "client called mgpuCreateSparseEnv() twice");
624 CUSPARSE_REPORT_IF_ERROR(cusparseCreate(&cusparse_env));
625}
626
627extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuDestroySparseEnv() {
628 assert(cusparse_env && "client did not call mgpuCreateSparseEnv()");
629 CUSPARSE_REPORT_IF_ERROR(cusparseDestroy(cusparse_env));
630 cusparse_env = nullptr;
631}
632
633extern "C" MLIR_CUDA_WRAPPERS_EXPORT void *
634mgpuCreateDnVec(intptr_t size, void *values, int32_t dtp, CUstream /*stream*/) {
635 cusparseDnVecDescr_t vec = nullptr;
636 auto dTp = static_cast<cudaDataType_t>(dtp);
637 CUSPARSE_REPORT_IF_ERROR(cusparseCreateDnVec(&vec, size, values, dTp))
638 return reinterpret_cast<void *>(vec);
639}
640
641extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
642mgpuDestroyDnVec(void *v, CUstream /*stream*/) {
643 cusparseDnVecDescr_t vec = reinterpret_cast<cusparseDnVecDescr_t>(v);
644 CUSPARSE_REPORT_IF_ERROR(cusparseDestroyDnVec(vec))
645}
646
647extern "C" MLIR_CUDA_WRAPPERS_EXPORT void *
648mgpuCreateDnMat(intptr_t rows, intptr_t cols, void *values, int32_t dtp,
649 CUstream /*stream*/) {
650 cusparseDnMatDescr_t mat = nullptr;
651 auto dTp = static_cast<cudaDataType_t>(dtp);
652 CUSPARSE_REPORT_IF_ERROR(cusparseCreateDnMat(&mat, rows, cols, /*ld=*/cols,
653 values, dTp, CUSPARSE_ORDER_ROW))
654 return reinterpret_cast<void *>(mat);
655}
656
657extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
658mgpuDestroyDnMat(void *m, CUstream /*stream*/) {
659 cusparseDnMatDescr_t mat = reinterpret_cast<cusparseDnMatDescr_t>(m);
660 CUSPARSE_REPORT_IF_ERROR(cusparseDestroyDnMat(mat))
661}
662
663extern "C" MLIR_CUDA_WRAPPERS_EXPORT void *
664mgpuCreateCoo(intptr_t rows, intptr_t cols, intptr_t nnz, void *rowIdxs,
665 void *colIdxs, void *values, int32_t itp, int32_t dtp,
666 CUstream /*stream*/) {
667 cusparseSpMatDescr_t mat = nullptr;
668 auto iTp = static_cast<cusparseIndexType_t>(itp);
669 auto dTp = static_cast<cudaDataType_t>(dtp);
670 CUSPARSE_REPORT_IF_ERROR(cusparseCreateCoo(&mat, rows, cols, nnz, rowIdxs,
671 colIdxs, values, iTp,
672 CUSPARSE_INDEX_BASE_ZERO, dTp))
673 return reinterpret_cast<void *>(mat);
674}
675
676#ifdef CUSPARSE_COO_AOS // deprecated in cuSPARSE 11.2
677extern "C" MLIR_CUDA_WRAPPERS_EXPORT void *
678mgpuCreateCooAoS(intptr_t rows, intptr_t cols, intptr_t nnz, void *idxs,
679 void *values, int32_t itp, int32_t dtp, CUstream /*stream*/) {
680 cusparseSpMatDescr_t mat = nullptr;
681 auto iTp = static_cast<cusparseIndexType_t>(itp);
682 auto dTp = static_cast<cudaDataType_t>(dtp);
683 CUSPARSE_REPORT_IF_ERROR(cusparseCreateCooAoS(
684 &mat, rows, cols, nnz, idxs, values, iTp, CUSPARSE_INDEX_BASE_ZERO, dTp))
685 return reinterpret_cast<void *>(mat);
686}
687#endif // CUSPARSE_COO_AOS
688
689extern "C" MLIR_CUDA_WRAPPERS_EXPORT void *
690mgpuCreateCsr(intptr_t rows, intptr_t cols, intptr_t nnz, void *rowPos,
691 void *colIdxs, void *values, int32_t ptp, int32_t itp,
692 int32_t dtp, CUstream /*stream*/) {
693 cusparseSpMatDescr_t mat = nullptr;
694 auto pTp = static_cast<cusparseIndexType_t>(ptp);
695 auto iTp = static_cast<cusparseIndexType_t>(itp);
696 auto dTp = static_cast<cudaDataType_t>(dtp);
697 CUSPARSE_REPORT_IF_ERROR(cusparseCreateCsr(&mat, rows, cols, nnz, rowPos,
698 colIdxs, values, pTp, iTp,
699 CUSPARSE_INDEX_BASE_ZERO, dTp))
700 return reinterpret_cast<void *>(mat);
701}
702
703extern "C" MLIR_CUDA_WRAPPERS_EXPORT void *
704mgpuCreateCsc(intptr_t rows, intptr_t cols, intptr_t nnz, void *colPos,
705 void *rowIdxs, void *values, int32_t ptp, int32_t itp,
706 int32_t dtp, CUstream /*stream*/) {
707 cusparseSpMatDescr_t mat = nullptr;
708 auto pTp = static_cast<cusparseIndexType_t>(ptp);
709 auto iTp = static_cast<cusparseIndexType_t>(itp);
710 auto dTp = static_cast<cudaDataType_t>(dtp);
711 CUSPARSE_REPORT_IF_ERROR(cusparseCreateCsc(&mat, rows, cols, nnz, colPos,
712 rowIdxs, values, pTp, iTp,
713 CUSPARSE_INDEX_BASE_ZERO, dTp))
714 return reinterpret_cast<void *>(mat);
715}
716
717extern "C" MLIR_CUDA_WRAPPERS_EXPORT void *
718mgpuCreateBsr(intptr_t brows, intptr_t bcols, intptr_t bnnz, intptr_t rBsz,
719 intptr_t cBsz, void *rowPos, void *colIdxs, void *values,
720 int32_t ptp, int32_t itp, int32_t dtp, CUstream /*stream*/) {
721 cusparseSpMatDescr_t mat = nullptr;
722#if CUSPARSE_VERSION >= 12100
723 auto pTp = static_cast<cusparseIndexType_t>(ptp);
724 auto iTp = static_cast<cusparseIndexType_t>(itp);
725 auto dTp = static_cast<cudaDataType_t>(dtp);
726 CUSPARSE_REPORT_IF_ERROR(cusparseCreateBsr(
727 &mat, brows, bcols, bnnz, rBsz, cBsz, rowPos, colIdxs, values, pTp, iTp,
728 CUSPARSE_INDEX_BASE_ZERO, dTp, CUSPARSE_ORDER_ROW))
729#endif
730 return reinterpret_cast<void *>(mat);
731}
732
733extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
734mgpuDestroySpMat(void *m, CUstream /*stream*/) {
735 cusparseSpMatDescr_t mat = reinterpret_cast<cusparseSpMatDescr_t>(m);
736 CUSPARSE_REPORT_IF_ERROR(cusparseDestroySpMat(mat))
737}
738
739extern "C" MLIR_CUDA_WRAPPERS_EXPORT intptr_t mgpuSpMVBufferSize(
740 int32_t ma, void *a, void *x, void *y, int32_t ctp, CUstream /*stream*/) {
741 assert(cusparse_env && "client did not call mgpuCreateSparseEnv()");
742 cusparseOperation_t modeA = static_cast<cusparseOperation_t>(ma);
743 cusparseSpMatDescr_t matA = reinterpret_cast<cusparseSpMatDescr_t>(a);
744 cusparseDnVecDescr_t vecX = reinterpret_cast<cusparseDnVecDescr_t>(x);
745 cusparseDnVecDescr_t vecY = reinterpret_cast<cusparseDnVecDescr_t>(y);
746 cudaDataType_t cTp = static_cast<cudaDataType_t>(ctp);
747 ALPHABETA(cTp, alpha, beta)
748 size_t bufferSize = 0;
749 CUSPARSE_REPORT_IF_ERROR(cusparseSpMV_bufferSize(
750 cusparse_env, modeA, alphap, matA, vecX, betap, vecY, cTp,
751 CUSPARSE_SPMV_ALG_DEFAULT, &bufferSize))
752 return bufferSize;
753}
754
755extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuSpMV(int32_t ma, void *a, void *x,
756 void *y, int32_t ctp,
757 void *buf,
758 CUstream /*stream*/) {
759 assert(cusparse_env && "client did not call mgpuCreateSparseEnv()");
760 cusparseOperation_t modeA = static_cast<cusparseOperation_t>(ma);
761 cusparseSpMatDescr_t matA = reinterpret_cast<cusparseSpMatDescr_t>(a);
762 cusparseDnVecDescr_t vecX = reinterpret_cast<cusparseDnVecDescr_t>(x);
763 cusparseDnVecDescr_t vecY = reinterpret_cast<cusparseDnVecDescr_t>(y);
764 cudaDataType_t cTp = static_cast<cudaDataType_t>(ctp);
765 ALPHABETA(cTp, alpha, beta)
766 CUSPARSE_REPORT_IF_ERROR(cusparseSpMV(cusparse_env, modeA, alphap, matA, vecX,
767 betap, vecY, cTp,
768 CUSPARSE_SPMV_ALG_DEFAULT, buf))
769}
770
772mgpuSpMMBufferSize(int32_t ma, int32_t mb, void *a, void *b, void *c,
773 int32_t ctp, CUstream /*stream*/) {
774 assert(cusparse_env && "client did not call mgpuCreateSparseEnv()");
775 cusparseOperation_t modeA = static_cast<cusparseOperation_t>(ma);
776 cusparseOperation_t modeB = static_cast<cusparseOperation_t>(mb);
777 cusparseSpMatDescr_t matA = reinterpret_cast<cusparseSpMatDescr_t>(a);
778 cusparseDnMatDescr_t matB = reinterpret_cast<cusparseDnMatDescr_t>(b);
779 cusparseDnMatDescr_t matC = reinterpret_cast<cusparseDnMatDescr_t>(c);
780 cudaDataType_t cTp = static_cast<cudaDataType_t>(ctp);
781 ALPHABETA(cTp, alpha, beta)
782 size_t bufferSize = 0;
783 CUSPARSE_REPORT_IF_ERROR(cusparseSpMM_bufferSize(
784 cusparse_env, modeA, modeB, alphap, matA, matB, betap, matC, cTp,
785 CUSPARSE_SPMM_ALG_DEFAULT, &bufferSize))
786 return bufferSize;
787}
788
789extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuSpMM(int32_t ma, int32_t mb,
790 void *a, void *b, void *c,
791 int32_t ctp, void *buf,
792 CUstream /*stream*/) {
793 assert(cusparse_env && "client did not call mgpuCreateSparseEnv()");
794 cusparseOperation_t modeA = static_cast<cusparseOperation_t>(ma);
795 cusparseOperation_t modeB = static_cast<cusparseOperation_t>(mb);
796 cusparseSpMatDescr_t matA = reinterpret_cast<cusparseSpMatDescr_t>(a);
797 cusparseDnMatDescr_t matB = reinterpret_cast<cusparseDnMatDescr_t>(b);
798 cusparseDnMatDescr_t matC = reinterpret_cast<cusparseDnMatDescr_t>(c);
799 cudaDataType_t cTp = static_cast<cudaDataType_t>(ctp);
800 ALPHABETA(cTp, alpha, beta)
801 CUSPARSE_REPORT_IF_ERROR(cusparseSpMM(cusparse_env, modeA, modeB, alphap,
802 matA, matB, betap, matC, cTp,
803 CUSPARSE_SPMM_ALG_DEFAULT, buf))
804}
805
807mgpuSDDMMBufferSize(int32_t ma, int32_t mb, void *a, void *b, void *c,
808 int32_t ctp, CUstream /*stream*/) {
809 assert(cusparse_env && "client did not call mgpuCreateSparseEnv()");
810 cusparseOperation_t modeA = static_cast<cusparseOperation_t>(ma);
811 cusparseOperation_t modeB = static_cast<cusparseOperation_t>(mb);
812 cusparseDnMatDescr_t matA = reinterpret_cast<cusparseDnMatDescr_t>(a);
813 cusparseDnMatDescr_t matB = reinterpret_cast<cusparseDnMatDescr_t>(b);
814 cusparseSpMatDescr_t matC = reinterpret_cast<cusparseSpMatDescr_t>(c);
815 auto cTp = static_cast<cudaDataType_t>(ctp);
816 ALPHABETA(cTp, alpha, beta)
817 size_t bufferSize = 0;
818 CUSPARSE_REPORT_IF_ERROR(cusparseSDDMM_bufferSize(
819 cusparse_env, modeA, modeB, alphap, matA, matB, betap, matC, cTp,
820 CUSPARSE_SDDMM_ALG_DEFAULT, &bufferSize))
821 return bufferSize;
822}
823
824extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuSDDMM(int32_t ma, int32_t mb,
825 void *a, void *b, void *c,
826 int32_t ctp, void *buf,
827 CUstream /*stream*/) {
828 assert(cusparse_env && "client did not call mgpuCreateSparseEnv()");
829 cusparseOperation_t modeA = static_cast<cusparseOperation_t>(ma);
830 cusparseOperation_t modeB = static_cast<cusparseOperation_t>(mb);
831 cusparseDnMatDescr_t matA = reinterpret_cast<cusparseDnMatDescr_t>(a);
832 cusparseDnMatDescr_t matB = reinterpret_cast<cusparseDnMatDescr_t>(b);
833 cusparseSpMatDescr_t matC = reinterpret_cast<cusparseSpMatDescr_t>(c);
834 auto cTp = static_cast<cudaDataType_t>(ctp);
835 ALPHABETA(cTp, alpha, beta)
836 CUSPARSE_REPORT_IF_ERROR(cusparseSDDMM(cusparse_env, modeA, modeB, alphap,
837 matA, matB, betap, matC, cTp,
838 CUSPARSE_SDDMM_ALG_DEFAULT, buf))
839}
840
841extern "C" MLIR_CUDA_WRAPPERS_EXPORT void *
842mgpuSpGEMMCreateDescr(CUstream /*stream*/) {
843 cusparseSpGEMMDescr_t spgemmDesc = nullptr;
844 CUSPARSE_REPORT_IF_ERROR(cusparseSpGEMM_createDescr(&spgemmDesc))
845 return reinterpret_cast<void *>(spgemmDesc);
846}
847
848extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
849mgpuSpGEMMDestroyDescr(void *s, CUstream /*stream*/) {
850 cusparseSpGEMMDescr_t spgemmDesc = reinterpret_cast<cusparseSpGEMMDescr_t>(s);
851 CUSPARSE_REPORT_IF_ERROR(cusparseSpGEMM_destroyDescr(spgemmDesc))
852}
853
854extern "C" MLIR_CUDA_WRAPPERS_EXPORT intptr_t mgpuSpGEMMWorkEstimation(
855 void *s, int32_t ma, int32_t mb, void *a, void *b, void *c, int32_t ctp,
856 intptr_t bs, void *buf, CUstream /*stream*/) {
857 cusparseSpGEMMDescr_t spgemmDesc = reinterpret_cast<cusparseSpGEMMDescr_t>(s);
858 cusparseOperation_t modeA = static_cast<cusparseOperation_t>(ma);
859 cusparseOperation_t modeB = static_cast<cusparseOperation_t>(mb);
860 cusparseSpMatDescr_t matA = reinterpret_cast<cusparseSpMatDescr_t>(a);
861 cusparseSpMatDescr_t matB = reinterpret_cast<cusparseSpMatDescr_t>(b);
862 cusparseSpMatDescr_t matC = reinterpret_cast<cusparseSpMatDescr_t>(c);
863 auto cTp = static_cast<cudaDataType_t>(ctp);
864 ALPHABETA(cTp, alpha, beta)
865 size_t newBufferSize = bs;
866 CUSPARSE_REPORT_IF_ERROR(cusparseSpGEMM_workEstimation(
867 cusparse_env, modeA, modeB, alphap, matA, matB, betap, matC, cTp,
868 CUSPARSE_SPGEMM_DEFAULT, spgemmDesc, &newBufferSize, buf))
869 return newBufferSize;
870}
871
873mgpuSpGEMMCompute(void *s, int32_t ma, int32_t mb, void *a, void *b, void *c,
874 int32_t ctp, intptr_t bsz2, void *buf2, CUstream /*stream*/) {
875 cusparseSpGEMMDescr_t spgemmDesc = reinterpret_cast<cusparseSpGEMMDescr_t>(s);
876 cusparseOperation_t modeA = static_cast<cusparseOperation_t>(ma);
877 cusparseOperation_t modeB = static_cast<cusparseOperation_t>(mb);
878 cusparseSpMatDescr_t matA = reinterpret_cast<cusparseSpMatDescr_t>(a);
879 cusparseSpMatDescr_t matB = reinterpret_cast<cusparseSpMatDescr_t>(b);
880 cusparseSpMatDescr_t matC = reinterpret_cast<cusparseSpMatDescr_t>(c);
881 auto cTp = static_cast<cudaDataType_t>(ctp);
882 ALPHABETA(cTp, alpha, beta)
883 size_t newBufferSize2 = bsz2;
884 CUSPARSE_REPORT_IF_ERROR(cusparseSpGEMM_compute(
885 cusparse_env, modeA, modeB, alphap, matA, matB, betap, matC, cTp,
886 CUSPARSE_SPGEMM_DEFAULT, spgemmDesc, &newBufferSize2, buf2))
887 return newBufferSize2;
888}
889
890extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
891mgpuSpGEMMCopy(void *s, int32_t ma, int32_t mb, void *a, void *b, void *c,
892 int32_t ctp, CUstream /*stream*/) {
893 cusparseSpGEMMDescr_t spgemmDesc = reinterpret_cast<cusparseSpGEMMDescr_t>(s);
894 cusparseOperation_t modeA = static_cast<cusparseOperation_t>(ma);
895 cusparseOperation_t modeB = static_cast<cusparseOperation_t>(mb);
896 cusparseSpMatDescr_t matA = reinterpret_cast<cusparseSpMatDescr_t>(a);
897 cusparseSpMatDescr_t matB = reinterpret_cast<cusparseSpMatDescr_t>(b);
898 cusparseSpMatDescr_t matC = reinterpret_cast<cusparseSpMatDescr_t>(c);
899 auto cTp = static_cast<cudaDataType_t>(ctp);
900 ALPHABETA(cTp, alpha, beta)
902 cusparseSpGEMM_copy(cusparse_env, modeA, modeB, alphap, matA, matB, betap,
903 matC, cTp, CUSPARSE_SPGEMM_DEFAULT, spgemmDesc))
904}
905
906extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
907mgpuSpMatGetSize(void *m, void *r, void *c, void *n, CUstream /*stream*/) {
908 cusparseConstSpMatDescr_t matDescr =
909 reinterpret_cast<cusparseConstSpMatDescr_t>(m);
910 int64_t *rows = reinterpret_cast<int64_t *>(r);
911 int64_t *cols = reinterpret_cast<int64_t *>(c);
912 int64_t *nnz = reinterpret_cast<int64_t *>(n);
913 CUSPARSE_REPORT_IF_ERROR(cusparseSpMatGetSize(matDescr, rows, cols, nnz));
914}
915
916extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
917mgpuSetCsrPointers(void *m, void *p, void *c, void *v, CUstream /*stream*/) {
918 cusparseSpMatDescr_t matDescr = reinterpret_cast<cusparseSpMatDescr_t>(m);
919 CUSPARSE_REPORT_IF_ERROR(cusparseCsrSetPointers(matDescr, p, c, v));
920}
921
922#ifdef MLIR_ENABLE_CUDA_CUSPARSELT
923
924///
925/// Wrapper methods for the cuSparseLt library.
926///
927
928struct cusparseLtSpMatHandleAndData {
929 cusparseLtMatDescriptor_t mat;
930 // TODO: the following three are associated with the SpMM operator rather than
931 // the sparse matrix. Create workspace buffers and pass them to the SpMM
932 // execution.
933 cusparseLtMatmulAlgSelection_t alg_sel;
934 cusparseLtMatmulPlan_t plan;
935 cusparseLtMatmulDescriptor_t matmul;
936 void *values{nullptr};
937};
938
939struct cusparseLtDnMatHandleAndData {
940 cusparseLtMatDescriptor_t mat;
941 void *values{nullptr};
942};
943
944static_assert(sizeof(cusparseLtHandle_t) == 11024,
945 "Unexpected cusparseLt handle size");
946static_assert(sizeof(cusparseLtSpMatHandleAndData) == 44104,
947 "Unexpected cusparseLt sparse matrix handle size");
948static_assert(sizeof(cusparseLtDnMatHandleAndData) == 11032,
949 "Unexpected cusparseLt dense matrix handle size");
950
951extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuCreateSparseLtEnv() {
952 // ScopedContext is for cuda initialization.
953 ScopedContext scopedContext;
954 assert(!cusparseLt_initiated &&
955 "client called mgpuCreateSparseLtEnv() twice");
956 // Note that cuSparseLt still uses cusparseStatus_t.
957 CUSPARSE_REPORT_IF_ERROR(cusparseLtInit(&cusparseLt_env));
958 cusparseLt_initiated = true;
959}
960
961extern "C" MLIR_CUDA_WRAPPERS_EXPORT void mgpuDestroySparseLtEnv() {
962 assert(cusparseLt_initiated && "client did not call mgpuCreateSparseLtEnv()");
963 CUSPARSE_REPORT_IF_ERROR(cusparseLtDestroy(&cusparseLt_env));
964 cusparseLt_initiated = false;
965}
966
967extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
968mgpuCreateCuSparseLtDnMat(void *dh, intptr_t rows, intptr_t cols, void *values,
969 int32_t dtp, CUstream /*stream*/) {
970 assert(cusparseLt_initiated && "client did not call mgpuCreateSparseLtEnv()");
971 auto dnmat_handle = reinterpret_cast<cusparseLtDnMatHandleAndData *>(dh);
972 dnmat_handle->values = values;
973 auto dTp = static_cast<cudaDataType_t>(dtp);
974 // Assume row-major when deciding lda.
975 const uint32_t alignment = 16;
976 CUSPARSE_REPORT_IF_ERROR(cusparseLtDenseDescriptorInit(
977 &cusparseLt_env, &(dnmat_handle->mat), rows, cols, /*lda=*/cols,
978 alignment, dTp, CUSPARSE_ORDER_ROW))
979}
980
981extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
982mgpuDestroyCuSparseLtDnMat(void *dh, CUstream /*stream*/) {
983 auto dnmat_handle = reinterpret_cast<cusparseLtDnMatHandleAndData *>(dh);
984 CUSPARSE_REPORT_IF_ERROR(cusparseLtMatDescriptorDestroy(&(dnmat_handle->mat)))
985}
986
987extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
988mgpuCusparseLtCreate2To4SpMat(void *sh, intptr_t rows, intptr_t cols,
989 void *values, int32_t dtp, CUstream /*stream*/) {
990 assert(cusparseLt_initiated && "client did not call mgpuCreateSparseLtEnv()");
991 auto spmat_handle = reinterpret_cast<cusparseLtSpMatHandleAndData *>(sh);
992 spmat_handle->values = values;
993 auto dTp = static_cast<cudaDataType_t>(dtp);
994 // Assume row-major when deciding lda.
995 const uint32_t alignment = 16;
996 CUSPARSE_REPORT_IF_ERROR(cusparseLtStructuredDescriptorInit(
997 &cusparseLt_env, &(spmat_handle->mat), rows, cols, /*ld=*/cols, alignment,
998 dTp, CUSPARSE_ORDER_ROW, CUSPARSELT_SPARSITY_50_PERCENT))
999}
1000
1001extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
1002mgpuDestroyCuSparseLtSpMat(void *sh, CUstream /*stream*/) {
1003 auto spmat_handle = reinterpret_cast<cusparseLtSpMatHandleAndData *>(sh);
1004 CUSPARSE_REPORT_IF_ERROR(cusparseLtMatDescriptorDestroy(&(spmat_handle->mat)))
1005}
1006
1007// Several things are being done in this stage, algorithm selection, planning,
1008// and returning workspace and compressed matrices data buffer sizes.
1009// The parameter prune_flag is used to indicate whether pruning and pruning
1010// check will happen 0 means not prune or prune check, 1 means prune, 2 means
1011// prune & prune check
1012extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
1013mgpuCuSparseLtSpMMBufferSize(void *bs, int32_t ma, int32_t mb, void *a, void *b,
1014 void *c, int32_t ctp, int32_t prune_flag,
1015 CUstream stream) {
1016 assert(cusparseLt_initiated && "client did not call mgpuCreateSparseLtEnv()");
1017 // TODO: support more advanced settings, e.g., the input right operand is a
1018 // sparse matrix assuming matA is the sparse matrix
1019 auto matA = reinterpret_cast<cusparseLtSpMatHandleAndData *>(a);
1020 auto matB = reinterpret_cast<cusparseLtDnMatHandleAndData *>(b);
1021 auto matC = reinterpret_cast<cusparseLtDnMatHandleAndData *>(c);
1022 auto workspace_size = reinterpret_cast<size_t *>(bs);
1023 auto compressed_size = &(reinterpret_cast<size_t *>(bs)[1]);
1024 auto compressed_buffer_size = &(reinterpret_cast<size_t *>(bs)[2]);
1025 auto cTp = static_cast<cusparseComputeType>(ctp);
1026
1027 cusparseOperation_t modeA = static_cast<cusparseOperation_t>(ma);
1028 cusparseOperation_t modeB = static_cast<cusparseOperation_t>(mb);
1029 CUSPARSE_REPORT_IF_ERROR(cusparseLtMatmulDescriptorInit(
1030 &cusparseLt_env, &(matA->matmul), modeA, modeB, &(matA->mat),
1031 &(matB->mat), &(matC->mat), &(matC->mat), cTp))
1032 CUSPARSE_REPORT_IF_ERROR(cusparseLtMatmulAlgSelectionInit(
1033 &cusparseLt_env, &(matA->alg_sel), &(matA->matmul),
1034 CUSPARSELT_MATMUL_ALG_DEFAULT))
1035 int alg = 0;
1036 CUSPARSE_REPORT_IF_ERROR(cusparseLtMatmulAlgSetAttribute(
1037 &cusparseLt_env, &(matA->alg_sel), CUSPARSELT_MATMUL_ALG_CONFIG_ID, &alg,
1038 sizeof(alg)))
1039
1040 CUSPARSE_REPORT_IF_ERROR(cusparseLtMatmulPlanInit(
1041 &cusparseLt_env, &(matA->plan), &(matA->matmul), &(matA->alg_sel)))
1042
1043 // Pruning step (in-place).
1044 if (prune_flag > 0)
1045 CUSPARSE_REPORT_IF_ERROR(cusparseLtSpMMAPrune(
1046 &cusparseLt_env, &(matA->matmul), matA->values, matA->values,
1047 CUSPARSELT_PRUNE_SPMMA_STRIP, stream))
1048
1049 // Check structure of A.
1050 // Note that this adds a synchronization on the stream.
1051 // TODO: Do we want that?
1052 if (prune_flag == 2) {
1053 int *dvalid = (int *)mgpuMemAlloc(sizeof(int), stream, false);
1054 CUSPARSE_REPORT_IF_ERROR(cusparseLtSpMMAPruneCheck(
1055 &cusparseLt_env, &(matA->matmul), matA->values, dvalid, stream))
1056 int valid = 0;
1057 mgpuMemcpy(&valid, dvalid, sizeof(int), stream);
1058 mgpuStreamSynchronize(stream);
1059 mgpuMemFree(dvalid, stream);
1060 if (valid != 0)
1061 fprintf(stderr, "CUPARSE-LT: sparse matrix is not 2:4; computed results "
1062 "will be invalid\n");
1063 }
1064
1065 CUSPARSE_REPORT_IF_ERROR(cusparseLtMatmulGetWorkspace(
1066 &cusparseLt_env, &(matA->plan), workspace_size))
1067 CUSPARSE_REPORT_IF_ERROR(cusparseLtSpMMACompressedSize(
1068 &cusparseLt_env, &(matA->plan), compressed_size, compressed_buffer_size))
1069}
1070
1071extern "C" MLIR_CUDA_WRAPPERS_EXPORT void
1072mgpuCuSparseLtSpMM(void *a, void *b, void *c, void *d_workspace,
1073 void *dA_compressed, void *dA_compressedBuffer,
1074 CUstream stream) {
1075 assert(cusparseLt_initiated && "client did not call mgpuCreateSparseLtEnv()");
1076 auto matA = reinterpret_cast<cusparseLtSpMatHandleAndData *>(a);
1077 auto matB = reinterpret_cast<cusparseLtDnMatHandleAndData *>(b);
1078 auto matC = reinterpret_cast<cusparseLtDnMatHandleAndData *>(c);
1079
1080 ALPHABETA(CUDA_R_32F, alpha, beta)
1082 cusparseLtSpMMACompress(&cusparseLt_env, &(matA->plan), (matA->values),
1083 dA_compressed, dA_compressedBuffer, stream))
1084
1085 // TODO: add support to multi-stream execution
1086 // Perform the matrix multiplication. D = A*B+C using C==D for now
1088 cusparseLtMatmul(&cusparseLt_env, &(matA->plan), alphap, dA_compressed,
1089 matB->values, betap, matC->values,
1090 /*dD*/ matC->values, d_workspace, nullptr, 0))
1091
1092 CUSPARSE_REPORT_IF_ERROR(cusparseLtMatDescriptorDestroy(&(matA->mat)))
1093 // destroy the plan associated with the sparse matrix
1094 CUSPARSE_REPORT_IF_ERROR(cusparseLtMatmulPlanDestroy(&(matA->plan)))
1095}
1096
1097#endif // MLIR_ENABLE_CUDA_CUSPARSELT
1098#endif // MLIR_ENABLE_CUDA_CUSPARSE
#define CUSPARSE_REPORT_IF_ERROR(expr)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuStreamWaitEvent(CUstream stream, CUevent event)
#define MLIR_CUDA_WRAPPERS_EXPORT
MLIR_CUDA_WRAPPERS_EXPORT void mgpuModuleUnload(CUmodule module)
MLIR_CUDA_WRAPPERS_EXPORT void * mgpuMemAlloc(uint64_t sizeBytes, CUstream stream, bool isHostShared)
#define CUDA_REPORT_IF_ERROR(expr)
static thread_local int32_t defaultDevice
MLIR_CUDA_WRAPPERS_EXPORT void mgpuMemHostRegisterMemRef(int64_t rank, StridedMemRefType< char, 1 > *descriptor, int64_t elementSizeBytes)
Registers a memref with the CUDA runtime.
MLIR_CUDA_WRAPPERS_EXPORT CUmodule mgpuModuleLoadJIT(void *data, int optLevel, size_t)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuMemHostUnregister(void *ptr)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuEventRecord(CUevent event, CUstream stream)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuMemcpy(void *dst, void *src, size_t sizeBytes, CUstream stream)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuMemset16(void *dst, unsigned short value, size_t count, CUstream stream)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuMemset32(void *dst, unsigned int value, size_t count, CUstream stream)
MLIR_CUDA_WRAPPERS_EXPORT CUfunction mgpuModuleGetFunction(CUmodule module, const char *name)
#define debug_print(fmt,...)
MLIR_CUDA_WRAPPERS_EXPORT CUevent mgpuEventCreate()
MLIR_CUDA_WRAPPERS_EXPORT void mgpuMemHostRegister(void *ptr, uint64_t sizeBytes)
Helper functions for writing mlir example code.
static bool isDebugEnabled()
Helper method that checks environment value for debugging.
MLIR_CUDA_WRAPPERS_EXPORT void mgpuStreamSynchronize(CUstream stream)
MLIR_CUDA_WRAPPERS_EXPORT CUstream mgpuStreamCreate()
MLIR_CUDA_WRAPPERS_EXPORT void mgpuMemFree(void *ptr, CUstream)
MLIR_CUDA_WRAPPERS_EXPORT CUmodule mgpuModuleLoad(void *data, size_t)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuSetDefaultDevice(int32_t device)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuStreamDestroy(CUstream stream)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuLaunchKernel(CUfunction function, intptr_t gridX, intptr_t gridY, intptr_t gridZ, intptr_t blockX, intptr_t blockY, intptr_t blockZ, int32_t smem, CUstream stream, void **params, void **extra, size_t)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuMemHostUnregisterMemRef(int64_t rank, StridedMemRefType< char, 1 > *descriptor, int64_t elementSizeBytes)
Unregisters a memref with the CUDA runtime.
MLIR_CUDA_WRAPPERS_EXPORT void mgpuEventSynchronize(CUevent event)
MLIR_CUDA_WRAPPERS_EXPORT void mgpuEventDestroy(CUevent event)
static CUdevice getDefaultCuDevice()
b
Return true if permutation is a valid permutation of the outer_dims_perm (case OuterOrInnerPerm::Oute...
if(!isCopyOut)
void mgpuLaunchKernelCooperative(hipFunction_t function, intptr_t gridX, intptr_t gridY, intptr_t gridZ, intptr_t clusterX, intptr_t clusterY, intptr_t clusterZ, intptr_t blockX, intptr_t blockY, intptr_t blockZ, int32_t smem, hipStream_t stream, void **params, void **)
StridedMemRef descriptor type with static rank.