| /** |
| # Copyright (c) 2022, NVIDIA CORPORATION. All rights reserved. |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| **/ |
| |
| package cuda |
| |
| import ( |
| "fmt" |
| |
| "github.com/NVIDIA/go-nvml/pkg/dl" |
| ) |
| |
| /* |
| #cgo linux LDFLAGS: -Wl,--export-dynamic -Wl,--unresolved-symbols=ignore-in-object-files |
| #cgo darwin LDFLAGS: -Wl,-undefined,dynamic_lookup |
| |
| #ifdef _WIN32 |
| #define CUDAAPI __stdcall |
| #else |
| #define CUDAAPI |
| #endif |
| |
| typedef int CUdevice; |
| |
| typedef enum CUdevice_attribute_enum { |
| CU_DEVICE_ATTRIBUTE_COMPUTE_CAPABILITY_MAJOR = 75, |
| CU_DEVICE_ATTRIBUTE_COMPUTE_CAPABILITY_MINOR = 76 |
| } CUdevice_attribute; |
| |
| typedef enum cudaError_enum { |
| CUDA_SUCCESS = 0 |
| } CUresult; |
| |
| CUresult CUDAAPI cuInit(unsigned int Flags); |
| CUresult CUDAAPI cuDriverGetVersion(int *driverVersion); |
| CUresult CUDAAPI cuDeviceGet(CUdevice *device, int ordinal); |
| CUresult CUDAAPI cuDeviceGetAttribute(int *pi, CUdevice_attribute attrib, CUdevice dev); |
| */ |
| import "C" |
| |
| const ( |
| libraryName = "libcuda.so.1" |
| libraryLoadFlags = dl.RTLD_LAZY | dl.RTLD_GLOBAL |
| ) |
| |
| // cuda stores a reference the cuda dynamic library |
| var lib *dl.DynamicLibrary |
| |
| // Version returns the CUDA version of the driver as a string or an error if this |
| // cannot be determined. |
| func Version() (string, error) { |
| lib, err := load() |
| if err != nil { |
| return "", err |
| } |
| defer lib.Close() |
| |
| if err := lib.Lookup("cuDriverGetVersion"); err != nil { |
| return "", fmt.Errorf("failed to lookup symbol: %v", err) |
| } |
| |
| var version C.int |
| if result := C.cuDriverGetVersion(&version); result != C.CUDA_SUCCESS { |
| return "", fmt.Errorf("failed to get CUDA version: result=%v", result) |
| } |
| |
| major := version / 1000 |
| minor := version % 100 / 10 |
| |
| return fmt.Sprintf("%d.%d", major, minor), nil |
| } |
| |
| // ComputeCapability returns the CUDA compute capability of a device with the specified index as a string |
| // or an error if this cannot be determined. |
| func ComputeCapability(index int) (string, error) { |
| lib, err := load() |
| if err != nil { |
| return "", err |
| } |
| defer lib.Close() |
| |
| if err := lib.Lookup("cuInit"); err != nil { |
| return "", fmt.Errorf("failed to lookup symbol: %v", err) |
| } |
| if err := lib.Lookup("cuDeviceGet"); err != nil { |
| return "", fmt.Errorf("failed to lookup symbol: %v", err) |
| } |
| if err := lib.Lookup("cuDeviceGetAttribute"); err != nil { |
| return "", fmt.Errorf("failed to lookup symbol: %v", err) |
| } |
| |
| if result := C.cuInit(C.uint(0)); result != C.CUDA_SUCCESS { |
| return "", fmt.Errorf("failed to initialize CUDA: result=%v", result) |
| } |
| |
| var device C.CUdevice |
| // NOTE: We only query the first device |
| if result := C.cuDeviceGet(&device, C.int(index)); result != C.CUDA_SUCCESS { |
| return "", fmt.Errorf("failed to get CUDA device %v: result=%v", 0, result) |
| } |
| |
| var major C.int |
| if result := C.cuDeviceGetAttribute(&major, C.CU_DEVICE_ATTRIBUTE_COMPUTE_CAPABILITY_MAJOR, device); result != C.CUDA_SUCCESS { |
| return "", fmt.Errorf("failed to get CUDA compute capability major for device %v : result=%v", 0, result) |
| } |
| |
| var minor C.int |
| if result := C.cuDeviceGetAttribute(&minor, C.CU_DEVICE_ATTRIBUTE_COMPUTE_CAPABILITY_MINOR, device); result != C.CUDA_SUCCESS { |
| return "", fmt.Errorf("failed to get CUDA compute capability minor for device %v: result=%v", 0, result) |
| } |
| |
| return fmt.Sprintf("%d.%d", major, minor), nil |
| } |
| |
| func load() (*dl.DynamicLibrary, error) { |
| lib := dl.New(libraryName, libraryLoadFlags) |
| if lib == nil { |
| return nil, fmt.Errorf("error instantiating DynamicLibrary for CUDA") |
| } |
| err := lib.Open() |
| if err != nil { |
| return nil, fmt.Errorf("error opening DynamicLibrary for CUDA: %v", err) |
| } |
| |
| return lib, nil |
| } |