Project
Loading...
Searching...
No Matches
GPUReconstructionLibrary.cxx
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11
14
15#ifdef _WIN32
16#include <windows.h>
17#include <winbase.h>
18#include <conio.h>
19#else
20#include <dlfcn.h>
21#include <pthread.h>
22#include <unistd.h>
23#endif
24
25#include "GPUReconstruction.h"
26#include "GPUReconstructionAvailableBackends.h"
27
28#include "utils/qlibload.h"
29
30#include "GPULogging.h"
31
32using namespace o2::gpu;
33
35{
37 cfg.deviceType = type;
38 cfg.forceDeviceType = forceType;
39 cfg.master = master;
40 return CreateInstance(cfg);
41}
42
44{
45 GPUReconstruction* retVal = nullptr;
47#ifdef DEBUG_STREAMER
48 if (type != DeviceType::CPU) {
49 GPUError("Cannot create GPUReconstruction for a non-CPU device if DEBUG_STREAMER are enabled");
50 return nullptr;
51 }
52#endif
53 if (type == DeviceType::CPU) {
55 } else {
56 auto* loader = GetLibraryInstance(type, true);
57 if (loader && (retVal = (*loader)->GetPtr(cfg))) {
58 retVal->mMyLib = *loader;
59 }
60 }
61
62 if (retVal == nullptr) {
63 if (cfg.forceDeviceType) {
64 GPUError("Error: Could not load GPUReconstruction for specified device: %s (%u)", GPUDataTypes::DEVICE_TYPE_NAMES[type], cfg.deviceType);
65 } else if (type != DeviceType::CPU) {
66 GPUError("Could not load GPUReconstruction for device type %s (%u), falling back to CPU version", GPUDataTypes::DEVICE_TYPE_NAMES[type], cfg.deviceType);
67 GPUSettingsDeviceBackend cfg2 = cfg;
68 cfg2.deviceType = DeviceType::CPU;
69 retVal = CreateInstance(cfg2);
70 }
71 } else {
72 GPUInfo("Created GPUReconstruction instance for device type %s (%u)%s", GPUDataTypes::DEVICE_TYPE_NAMES[type], cfg.deviceType, cfg.master ? " (slave)" : "");
73 }
74
75 return retVal;
76}
77
79{
80 if (type == DeviceType::CPU) {
81 return true;
82 } else {
83 auto* loader = GetLibraryInstance(type, verbose);
84 return loader != nullptr && (*loader)->LoadLibrary() == 0;
85 }
86}
87
88std::shared_ptr<GPUReconstruction::LibraryLoader>* GPUReconstruction::GetLibraryInstance(DeviceType type, bool verbose)
89{
90 if (type == DeviceType::CPU) {
91 return nullptr;
92 } else if (type == DeviceType::CUDA) {
93#ifdef CUDA_ENABLED
94 return &sLibCUDA;
95#endif
96 } else if (type == DeviceType::HIP) {
97#ifdef HIP_ENABLED
98 return &sLibHIP;
99#endif
100 } else if (type == DeviceType::OCL) {
101#ifdef OPENCL_ENABLED
102 return &sLibOCL;
103#endif
104 } else {
105 GPUError("Error: Invalid device type %u", (uint32_t)type);
106 return nullptr;
107 }
108 if (verbose) {
109 GPUInfo("%s Support not compiled in for device type %u (%s)", GPUDataTypes::DEVICE_TYPE_NAMES[type], (uint32_t)type, GPUDataTypes::DEVICE_TYPE_NAMES[type]);
110 }
111 return nullptr;
112}
113
115{
117 if (t == DeviceType::INVALID_DEVICE) {
118 GPUError("Invalid device type: %s", type);
119 return nullptr;
120 }
121 return CreateInstance(t, forceType, master);
122}
123
124std::shared_ptr<GPUReconstruction::LibraryLoader> GPUReconstruction::sLibCUDA(new GPUReconstruction::LibraryLoader("lib" LIBRARY_PREFIX "GPUTrackingCUDA" LIBRARY_EXTENSION, "GPUReconstruction_Create_CUDA"));
125std::shared_ptr<GPUReconstruction::LibraryLoader> GPUReconstruction::sLibHIP(new GPUReconstruction::LibraryLoader("lib" LIBRARY_PREFIX "GPUTrackingHIP" LIBRARY_EXTENSION, "GPUReconstruction_Create_HIP"));
126std::shared_ptr<GPUReconstruction::LibraryLoader> GPUReconstruction::sLibOCL(new GPUReconstruction::LibraryLoader("lib" LIBRARY_PREFIX "GPUTrackingOCL" LIBRARY_EXTENSION, "GPUReconstruction_Create_OCL"));
127
128GPUReconstruction::LibraryLoader::LibraryLoader(const char* lib, const char* func) : mLibName(lib), mFuncName(func), mGPULib(nullptr), mGPUEntry(nullptr) {}
129
131
132int32_t GPUReconstruction::LibraryLoader::LoadLibrary()
133{
134 static std::mutex mut;
135 std::lock_guard<std::mutex> lock(mut);
136
137 if (mGPUEntry) {
138 return 0;
139 }
140
141 LIBRARY_TYPE hGPULib;
142 hGPULib = LIBRARY_LOAD(mLibName);
143 if (hGPULib == nullptr) {
144#ifndef _WIN32
145 GPUImportant("The following error occured during dlopen: %s", dlerror());
146#endif
147 GPUError("Error Opening cagpu library for GPU Tracker (%s)", mLibName);
148 return 1;
149 } else {
150 void* createFunc = LIBRARY_FUNCTION(hGPULib, mFuncName);
151 if (createFunc == nullptr) {
152 GPUError("Error fetching entry function in GPU library\n");
153 LIBRARY_CLOSE(hGPULib);
154 return 1;
155 } else {
156 mGPULib = (void*)(size_t)hGPULib;
157 mGPUEntry = createFunc;
158 GPUInfo("GPU Tracker library loaded and GPU tracker object created sucessfully");
159 }
160 }
161 return 0;
162}
163
164GPUReconstruction* GPUReconstruction::LibraryLoader::GetPtr(const GPUSettingsDeviceBackend& cfg)
165{
166 if (LoadLibrary()) {
167 return nullptr;
168 }
169 if (mGPUEntry == nullptr) {
170 return nullptr;
171 }
172 GPUReconstruction* (*tmp)(const GPUSettingsDeviceBackend& cfg) = (GPUReconstruction * (*)(const GPUSettingsDeviceBackend& cfg)) mGPUEntry;
173 return tmp(cfg);
174}
175
176int32_t GPUReconstruction::LibraryLoader::CloseLibrary()
177{
178 if (mGPUEntry == nullptr) {
179 return 1;
180 }
181 LIBRARY_CLOSE((LIBRARY_TYPE)(size_t)mGPULib);
182 mGPULib = nullptr;
183 mGPUEntry = nullptr;
184 return 0;
185}
int32_t retVal
static DeviceType GetDeviceType(const char *type)
static constexpr const char *const DEVICE_TYPE_NAMES[]
LibraryLoader(const LibraryLoader &)=delete
static std::shared_ptr< LibraryLoader > sLibHIP
GPUDataTypes::DeviceType DeviceType
static std::shared_ptr< LibraryLoader > * GetLibraryInstance(DeviceType type, bool verbose)
static bool CheckInstanceAvailable(DeviceType type, bool verbose)
static std::shared_ptr< LibraryLoader > sLibOCL
static GPUReconstruction * CreateInstance(const GPUSettingsDeviceBackend &cfg)
static GPUReconstruction * GPUReconstruction_Create_CPU(const GPUSettingsDeviceBackend &cfg)
static std::shared_ptr< LibraryLoader > sLibCUDA
GLenum func
Definition glcorearb.h:778
GLint GLint GLsizei GLint GLenum GLenum type
Definition glcorearb.h:275
#define LIBRARY_EXTENSION
Definition qlibload.h:25
#define LIBRARY_CLOSE
Definition qlibload.h:28
#define LIBRARY_TYPE
Definition qlibload.h:26
#define LIBRARY_PREFIX
Definition qlibload.h:35
#define LIBRARY_LOAD(name)
Definition qlibload.h:27
#define LIBRARY_FUNCTION
Definition qlibload.h:29