QGIS API Documentation 3.41.0-Master (57ec4277f5e)
Loading...
Searching...
No Matches
qgsopenclutils.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsopenclutils.cpp - QgsOpenClUtils
3
4 ---------------------
5 begin : 11.4.2018
6 copyright : (C) 2018 by elpaso
7 email : elpaso at itopen dot it
8 ***************************************************************************
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 ***************************************************************************/
16#include "qgsopenclutils.h"
17#include "moc_qgsopenclutils.cpp"
18#include "qgssettings.h"
19#include "qgsmessagelog.h"
20#include "qgslogger.h"
21
22#include <QLibrary>
23
24#include <QTextStream>
25#include <QFile>
26#include <QDebug>
27
28#ifdef Q_OS_WIN
29#if defined(UNICODE) && !defined(_UNICODE)
30#define _UNICODE
31#endif
32#include <windows.h>
33#include <tchar.h>
34#endif
35
36#if defined(_MSC_VER)
37#include <windows.h>
38#include <excpt.h>
39#endif
40
41QLatin1String QgsOpenClUtils::SETTINGS_GLOBAL_ENABLED_KEY = QLatin1String( "OpenClEnabled" );
42QLatin1String QgsOpenClUtils::SETTINGS_DEFAULT_DEVICE_KEY = QLatin1String( "OpenClDefaultDevice" );
43QLatin1String QgsOpenClUtils::LOGMESSAGE_TAG = QLatin1String( "OpenCL" );
44bool QgsOpenClUtils::sAvailable = false;
45
46Q_GLOBAL_STATIC( QString, sSourcePath )
47
48
49const std::vector<cl::Device> QgsOpenClUtils::devices()
50{
51 std::vector<cl::Platform> platforms;
52 cl::Platform::get( &platforms );
53 std::vector<cl::Device> existingDevices;
54 for ( const auto &p : platforms )
55 {
56 const std::string platver = p.getInfo<CL_PLATFORM_VERSION>();
57 QgsMessageLog::logMessage( QObject::tr( "Found OpenCL platform %1: %2" )
58 .arg( QString::fromStdString( platver ),
59 QString::fromStdString( p.getInfo<CL_PLATFORM_NAME>() ) ),
60 LOGMESSAGE_TAG, Qgis::MessageLevel::Info );
61 if ( platver.find( "OpenCL " ) != std::string::npos )
62 {
63 std::vector<cl::Device> _devices;
64 // Check for a device
65 try
66 {
67 p.getDevices( CL_DEVICE_TYPE_ALL, &_devices );
68 }
69 catch ( cl::Error &e )
70 {
71 QgsMessageLog::logMessage( QObject::tr( "Error %1 on platform %3 searching for OpenCL device: %2" )
72 .arg( errorText( e.err() ),
73 QString::fromStdString( e.what() ),
74 QString::fromStdString( p.getInfo<CL_PLATFORM_NAME>() ) ),
75 LOGMESSAGE_TAG, Qgis::MessageLevel::Warning );
76 }
77 if ( _devices.size() > 0 )
78 {
79 for ( unsigned long i = 0; i < _devices.size(); i++ )
80 {
81 QgsMessageLog::logMessage( QObject::tr( "Found OpenCL device: %1" )
82 .arg( deviceId( _devices[i] ) ),
83 LOGMESSAGE_TAG, Qgis::MessageLevel::Info );
84 existingDevices.push_back( _devices[i] );
85 }
86 }
87 }
88 }
89 return existingDevices;
90}
91
92void QgsOpenClUtils::init()
93{
94 static std::once_flag initialized;
95 std::call_once( initialized, [ = ]( )
96 {
97#ifdef Q_OS_MAC
98 QLibrary openCLLib { QStringLiteral( "/System/Library/Frameworks/OpenCL.framework/Versions/Current/OpenCL" ) };
99#else
100 QLibrary openCLLib { QStringLiteral( "OpenCL" ) };
101#endif
102 openCLLib.setLoadHints( QLibrary::LoadHint::ResolveAllSymbolsHint );
103 if ( ! openCLLib.load() )
104 {
105 QgsMessageLog::logMessage( QObject::tr( "Error loading OpenCL library: %1" )
106 .arg( openCLLib.errorString() ),
108 return;
109 }
110
111#ifdef Q_OS_WIN
112#ifdef _UNICODE
113#define _T(x) L##x
114#else
115#define _T(x) x
116#endif
117 HMODULE hModule = GetModuleHandle( _T( "OpenCL.dll" ) );
118 if ( hModule )
119 {
120 TCHAR pszFileName[1024];
121 if ( GetModuleFileName( hModule, pszFileName, 1024 ) < 1024 )
122 {
123 QgsMessageLog::logMessage( QObject::tr( "Found OpenCL library filename %1" )
124 .arg( pszFileName ),
126
127 DWORD dwUseless;
128 DWORD dwLen = GetFileVersionInfoSize( pszFileName, &dwUseless );
129 if ( dwLen )
130 {
131 LPTSTR lpVI = ( LPTSTR ) malloc( dwLen * sizeof( TCHAR ) );
132 if ( lpVI )
133 {
134 if ( GetFileVersionInfo( pszFileName, 0, dwLen, lpVI ) )
135 {
136 VS_FIXEDFILEINFO *lpFFI;
137 if ( VerQueryValue( lpVI, _T( "\\" ), ( LPVOID * ) &lpFFI, ( UINT * ) &dwUseless ) )
138 {
139 QgsMessageLog::logMessage( QObject::tr( "OpenCL Product version: %1.%2.%3.%4" )
140 .arg( lpFFI->dwProductVersionMS >> 16 )
141 .arg( lpFFI->dwProductVersionMS & 0xffff )
142 .arg( lpFFI->dwProductVersionLS >> 16 )
143 .arg( lpFFI->dwProductVersionLS & 0xffff ),
145 }
146
147 struct LANGANDCODEPAGE
148 {
149 WORD wLanguage;
150 WORD wCodePage;
151 } *lpTranslate;
152
153 DWORD cbTranslate;
154
155 if ( VerQueryValue( lpVI, _T( "\\VarFileInfo\\Translation" ), ( LPVOID * ) &lpTranslate, ( UINT * ) &cbTranslate ) && cbTranslate >= sizeof( struct LANGANDCODEPAGE ) )
156 {
157 QStringList items = QStringList()
158 << QStringLiteral( "Comments" )
159 << QStringLiteral( "InternalName" )
160 << QStringLiteral( "ProductName" )
161 << QStringLiteral( "CompanyName" )
162 << QStringLiteral( "LegalCopyright" )
163 << QStringLiteral( "ProductVersion" )
164 << QStringLiteral( "FileDescription" )
165 << QStringLiteral( "LegalTrademarks" )
166 << QStringLiteral( "PrivateBuild" )
167 << QStringLiteral( "FileVersion" )
168 << QStringLiteral( "OriginalFilename" )
169 << QStringLiteral( "SpecialBuild" );
170 for ( auto d : items )
171 {
172 LPTSTR lpBuffer;
173 QString subBlock = QString( QStringLiteral( "\\StringFileInfo\\%1%2\\%3" ) )
174 .arg( lpTranslate[0].wLanguage, 4, 16, QLatin1Char( '0' ) )
175 .arg( lpTranslate[0].wCodePage, 4, 16, QLatin1Char( '0' ) )
176 .arg( d );
177
178 QgsDebugMsgLevel( QString( "d:%1 subBlock:%2" ).arg( d ).arg( subBlock ), 2 );
179
180 BOOL r = VerQueryValue( lpVI,
181#ifdef UNICODE
182 subBlock.toStdWString().c_str(),
183#else
184 subBlock.toUtf8(),
185#endif
186 ( LPVOID * )&lpBuffer, ( UINT * )&dwUseless );
187
188 if ( r && lpBuffer && lpBuffer != INVALID_HANDLE_VALUE && dwUseless < 1023 )
189 {
190 QgsMessageLog::logMessage( QObject::tr( "Found OpenCL version info %1: %2" )
191 .arg( d )
192#ifdef UNICODE
193 .arg( QString::fromUtf16( ( const ushort * ) lpBuffer ) ),
194#else
195 .arg( QString::fromLocal8Bit( lpBuffer ) ),
196#endif
198 }
199 }
200 }
201 }
202
203 free( lpVI );
204 }
205 }
206 }
207 else
208 {
209 QgsMessageLog::logMessage( QObject::tr( "No module handle to OpenCL library" ), LOGMESSAGE_TAG, Qgis::MessageLevel::Warning );
210 }
211 }
212 else
213 {
214 QgsMessageLog::logMessage( QObject::tr( "No module handle to OpenCL library" ), LOGMESSAGE_TAG, Qgis::MessageLevel::Warning );
215 }
216#endif
217
218 try
219 {
220 activate( preferredDevice() );
221 }
222 catch ( cl::Error &e )
223 {
224 QgsMessageLog::logMessage( QObject::tr( "Error %1 initializing OpenCL device: %2" )
225 .arg( errorText( e.err() ), QString::fromStdString( e.what() ) ),
227 }
228
229 } );
230}
231
233{
234 return *sSourcePath();
235}
236
237void QgsOpenClUtils::setSourcePath( const QString &value )
238{
239 *sSourcePath() = value;
240}
241
243{
244 return deviceInfo( infoType, activeDevice( ) );
245}
246
247QString QgsOpenClUtils::deviceInfo( const Info infoType, cl::Device device )
248{
249 try
250 {
251 switch ( infoType )
252 {
253 case Info::Vendor:
254 return QString::fromStdString( device.getInfo<CL_DEVICE_VENDOR>() );
255 case Info::Profile:
256 return QString::fromStdString( device.getInfo<CL_DEVICE_PROFILE>() );
257 case Info::Version:
258 return QString::fromStdString( device.getInfo<CL_DEVICE_VERSION>() );
260 return device.getInfo<CL_DEVICE_IMAGE_SUPPORT>() ? QStringLiteral( "True" ) : QStringLiteral( "False" );
262 return QString::number( device.getInfo<CL_DEVICE_IMAGE2D_MAX_HEIGHT>() );
264 return QString::number( device.getInfo<CL_DEVICE_MAX_MEM_ALLOC_SIZE>() );
266 return QString::number( device.getInfo<CL_DEVICE_IMAGE2D_MAX_WIDTH>() );
267 case Info::Type:
268 {
269 const unsigned long type( device.getInfo<CL_DEVICE_TYPE>() );
270 int mappedType;
271 switch ( type )
272 {
273 case CL_DEVICE_TYPE_CPU:
275 break;
276 case CL_DEVICE_TYPE_GPU:
278 break;
279 default:
281 }
282 const QMetaEnum metaEnum = QMetaEnum::fromType<QgsOpenClUtils::HardwareType>();
283 return metaEnum.valueToKey( mappedType );
284 }
285 case Info::Name:
286 return QString::fromStdString( device.getInfo<CL_DEVICE_NAME>() );
287 }
288 }
289 catch ( cl::Error &e )
290 {
291 // This can be a legitimate error when initializing, let's log it quietly
292 QgsDebugMsgLevel( QStringLiteral( "Error %1 getting info for OpenCL device: %2" )
293 .arg( errorText( e.err() ), QString::fromStdString( e.what() ) ),
294 4 );
295 return QString();
296 }
297 return QString();
298}
299
300
302{
303 return QgsSettings().value( SETTINGS_GLOBAL_ENABLED_KEY, false, QgsSettings::Section::Core ).toBool();
304}
305
307{
308 return cl::Device::getDefault();
309}
310
312{
313 QString version;
314 if ( cl::Platform::getDefault()() )
315 {
316 const std::string platver = cl::Platform::getDefault().getInfo<CL_PLATFORM_VERSION>();
317 if ( platver.find( "OpenCL " ) != std::string::npos )
318 {
319 version = QString::fromStdString( platver.substr( 7 ) ).split( ' ' ).first();
320 }
321 }
322 return version;
323}
324
325void QgsOpenClUtils::storePreferredDevice( const QString deviceId )
326{
327 QgsSettings().setValue( SETTINGS_DEFAULT_DEVICE_KEY, deviceId, QgsSettings::Section::Core );
328}
329
331{
332 return QgsSettings().value( SETTINGS_DEFAULT_DEVICE_KEY, QString( ), QgsSettings::Section::Core ).toString();
333}
334
335QString QgsOpenClUtils::deviceId( const cl::Device device )
336{
337 return QStringLiteral( "%1|%2|%3|%4" )
338 .arg( deviceInfo( QgsOpenClUtils::Info::Name, device ) )
341 .arg( deviceInfo( QgsOpenClUtils::Info::Type, device ) );
342}
343
344#if defined(_MSC_VER)
345static void emitLogMessageForSEHException( int exceptionCode )
346{
347 QgsMessageLog::logMessage( QObject::tr( "Unexpected exception of code %1 occurred while searching for OpenCL device. Note that the application may become unreliable and may need to be restarted." ).arg( exceptionCode ),
349}
350#endif
351
352bool QgsOpenClUtils::activate( const QString &preferredDeviceId )
353{
354#if defined(_MSC_VER)
355 // Try to capture hard crashes such as https://github.com/qgis/QGIS/issues/59617
356 __try
357 {
358 // We cannot combine together __try and try in the same function.
359 return activateInternal( preferredDeviceId );
360 }
361 __except ( EXCEPTION_EXECUTE_HANDLER )
362 {
363 emitLogMessageForSEHException( GetExceptionCode() );
364 return false;
365 }
366#else
367 return activateInternal( preferredDeviceId );
368#endif
369}
370
371bool QgsOpenClUtils::activateInternal( const QString &preferredDeviceId )
372{
373 if ( deviceId( activeDevice() ) == preferredDeviceId )
374 {
375 sAvailable = true;
376 return false;
377 }
378
379 try
380 {
381 std::vector<cl::Platform> platforms;
382 cl::Platform::get( &platforms );
383 cl::Platform plat;
384 cl::Device dev;
385 bool deviceFound = false;
386 for ( const auto &p : platforms )
387 {
388 if ( deviceFound )
389 break;
390 const std::string platver = p.getInfo<CL_PLATFORM_VERSION>();
391 QgsDebugMsgLevel( QStringLiteral( "Found OpenCL platform %1: %2" ).arg( QString::fromStdString( platver ), QString::fromStdString( p.getInfo<CL_PLATFORM_NAME>() ) ), 2 );
392 if ( platver.find( "OpenCL " ) != std::string::npos )
393 {
394 std::vector<cl::Device> devices;
395 // Search for a device
396 try
397 {
398 p.getDevices( CL_DEVICE_TYPE_ALL, &devices );
399 // First search for the preferred device
400 if ( ! preferredDeviceId.isEmpty() )
401 {
402 for ( const auto &_dev : devices )
403 {
404 if ( preferredDeviceId == deviceId( _dev ) )
405 {
406 // Got one!
407 plat = p;
408 dev = _dev;
409 deviceFound = true;
410 break;
411 }
412 }
413 }
414 // Not found or preferred device id not set: get the first GPU
415 if ( ! deviceFound )
416 {
417 for ( const auto &_dev : devices )
418 {
419 if ( _dev.getInfo<CL_DEVICE_TYPE>() == CL_DEVICE_TYPE_GPU )
420 {
421 // Got one!
422 plat = p;
423 dev = _dev;
424 deviceFound = true;
425 break;
426 }
427 }
428 }
429 // Still nothing? Get the first device
430 if ( ! deviceFound )
431 {
432 for ( const auto &_dev : devices )
433 {
434 if ( _dev.getInfo<CL_DEVICE_TYPE>() == CL_DEVICE_TYPE_CPU )
435 {
436 // Got one!
437 plat = p;
438 dev = _dev;
439 deviceFound = true;
440 break;
441 }
442 }
443 }
444 if ( ! deviceFound )
445 {
446 QgsMessageLog::logMessage( QObject::tr( "No OpenCL device could be found." ), LOGMESSAGE_TAG, Qgis::MessageLevel::Warning );
447 }
448 }
449 catch ( cl::Error &e )
450 {
451 QgsDebugError( QStringLiteral( "Error %1 on platform %3 searching for OpenCL device: %2" )
452 .arg( errorText( e.err() ),
453 QString::fromStdString( e.what() ),
454 QString::fromStdString( p.getInfo<CL_PLATFORM_NAME>() ) ) );
455 }
456
457 }
458 }
459 if ( ! plat() )
460 {
461 QgsMessageLog::logMessage( QObject::tr( "No OpenCL platform found." ), LOGMESSAGE_TAG, Qgis::MessageLevel::Warning );
462 sAvailable = false;
463 }
464 else
465 {
466 const cl::Platform newP = cl::Platform::setDefault( plat );
467 if ( newP != plat )
468 {
469 QgsMessageLog::logMessage( QObject::tr( "Error setting default platform." ),
471 sAvailable = false;
472 }
473 else
474 {
475 cl::Device::setDefault( dev );
476 QgsMessageLog::logMessage( QObject::tr( "Active OpenCL device: %1" )
477 .arg( QString::fromStdString( dev.getInfo<CL_DEVICE_NAME>() ) ),
479 sAvailable = true;
480 }
481 }
482 }
483
484 catch ( cl::Error &e )
485 {
486 QgsMessageLog::logMessage( QObject::tr( "Error %1 searching for OpenCL device: %2" )
487 .arg( errorText( e.err() ), QString::fromStdString( e.what() ) ),
489 sAvailable = false;
490 }
491
492 return sAvailable;
493}
494
495QString QgsOpenClUtils::deviceDescription( const cl::Device device )
496{
497 return QStringLiteral(
498 "Type: <b>%9</b><br>"
499 "Name: <b>%1</b><br>"
500 "Vendor: <b>%2</b><br>"
501 "Profile: <b>%3</b><br>"
502 "Version: <b>%4</b><br>"
503 "Image support: <b>%5</b><br>"
504 "Max image2d width: <b>%6</b><br>"
505 "Max image2d height: <b>%7</b><br>"
506 "Max mem alloc size: <b>%8</b><br>"
516}
517
518QString QgsOpenClUtils::deviceDescription( const QString deviceId )
519{
520 for ( const auto &dev : devices( ) )
521 {
522 if ( QgsOpenClUtils::deviceId( dev ) == deviceId )
523 return deviceDescription( dev );
524 }
525 return QString();
526}
527
529{
530 init();
531 return sAvailable;
532}
533
534void QgsOpenClUtils::setEnabled( bool enabled )
535{
536 QgsSettings().setValue( SETTINGS_GLOBAL_ENABLED_KEY, enabled, QgsSettings::Section::Core );
537}
538
539
540
541QString QgsOpenClUtils::sourceFromPath( const QString &path )
542{
543 // TODO: check for compatibility with current platform ( cl_khr_fp64 )
544 // Try to load the program sources
545 QString source_str;
546 QFile file( path );
547 if ( file.open( QFile::ReadOnly | QFile::Text ) )
548 {
549 QTextStream in( &file );
550 source_str = in.readAll();
551 file.close();
552 }
553 else
554 {
555 QgsMessageLog::logMessage( QObject::tr( "Could not load OpenCL program from path %1." ).arg( path ), LOGMESSAGE_TAG, Qgis::MessageLevel::Warning );
556 }
557 return source_str;
558}
559
560QString QgsOpenClUtils::sourceFromBaseName( const QString &baseName )
561{
562 const QString path = QStringLiteral( "%1/%2.cl" ).arg( sourcePath(), baseName );
563 return sourceFromPath( path );
564}
565
566QString QgsOpenClUtils::buildLog( cl::BuildError &error )
567{
568 cl::BuildLogType build_logs = error.getBuildLog();
569 QString build_log;
570 if ( build_logs.size() > 0 )
571 build_log = QString::fromStdString( build_logs[0].second );
572 return build_log;
573}
574
575QString QgsOpenClUtils::errorText( const int errorCode )
576{
577 switch ( errorCode )
578 {
579 case 0: return QStringLiteral( "CL_SUCCESS" );
580 case -1: return QStringLiteral( "CL_DEVICE_NOT_FOUND" );
581 case -2: return QStringLiteral( "CL_DEVICE_NOT_AVAILABLE" );
582 case -3: return QStringLiteral( "CL_COMPILER_NOT_AVAILABLE" );
583 case -4: return QStringLiteral( "CL_MEM_OBJECT_ALLOCATION_FAILURE" );
584 case -5: return QStringLiteral( "CL_OUT_OF_RESOURCES" );
585 case -6: return QStringLiteral( "CL_OUT_OF_HOST_MEMORY" );
586 case -7: return QStringLiteral( "CL_PROFILING_INFO_NOT_AVAILABLE" );
587 case -8: return QStringLiteral( "CL_MEM_COPY_OVERLAP" );
588 case -9: return QStringLiteral( "CL_IMAGE_FORMAT_MISMATCH" );
589 case -10: return QStringLiteral( "CL_IMAGE_FORMAT_NOT_SUPPORTED" );
590 case -12: return QStringLiteral( "CL_MAP_FAILURE" );
591 case -13: return QStringLiteral( "CL_MISALIGNED_SUB_BUFFER_OFFSET" );
592 case -14: return QStringLiteral( "CL_EXEC_STATUS_ERROR_FOR_EVENTS_IN_WAIT_LIST" );
593 case -15: return QStringLiteral( "CL_COMPILE_PROGRAM_FAILURE" );
594 case -16: return QStringLiteral( "CL_LINKER_NOT_AVAILABLE" );
595 case -17: return QStringLiteral( "CL_LINK_PROGRAM_FAILURE" );
596 case -18: return QStringLiteral( "CL_DEVICE_PARTITION_FAILED" );
597 case -19: return QStringLiteral( "CL_KERNEL_ARG_INFO_NOT_AVAILABLE" );
598 case -30: return QStringLiteral( "CL_INVALID_VALUE" );
599 case -31: return QStringLiteral( "CL_INVALID_DEVICE_TYPE" );
600 case -32: return QStringLiteral( "CL_INVALID_PLATFORM" );
601 case -33: return QStringLiteral( "CL_INVALID_DEVICE" );
602 case -34: return QStringLiteral( "CL_INVALID_CONTEXT" );
603 case -35: return QStringLiteral( "CL_INVALID_QUEUE_PROPERTIES" );
604 case -36: return QStringLiteral( "CL_INVALID_COMMAND_QUEUE" );
605 case -37: return QStringLiteral( "CL_INVALID_HOST_PTR" );
606 case -38: return QStringLiteral( "CL_INVALID_MEM_OBJECT" );
607 case -39: return QStringLiteral( "CL_INVALID_IMAGE_FORMAT_DESCRIPTOR" );
608 case -40: return QStringLiteral( "CL_INVALID_IMAGE_SIZE" );
609 case -41: return QStringLiteral( "CL_INVALID_SAMPLER" );
610 case -42: return QStringLiteral( "CL_INVALID_BINARY" );
611 case -43: return QStringLiteral( "CL_INVALID_BUILD_OPTIONS" );
612 case -44: return QStringLiteral( "CL_INVALID_PROGRAM" );
613 case -45: return QStringLiteral( "CL_INVALID_PROGRAM_EXECUTABLE" );
614 case -46: return QStringLiteral( "CL_INVALID_KERNEL_NAME" );
615 case -47: return QStringLiteral( "CL_INVALID_KERNEL_DEFINITION" );
616 case -48: return QStringLiteral( "CL_INVALID_KERNEL" );
617 case -49: return QStringLiteral( "CL_INVALID_ARG_INDEX" );
618 case -50: return QStringLiteral( "CL_INVALID_ARG_VALUE" );
619 case -51: return QStringLiteral( "CL_INVALID_ARG_SIZE" );
620 case -52: return QStringLiteral( "CL_INVALID_KERNEL_ARGS" );
621 case -53: return QStringLiteral( "CL_INVALID_WORK_DIMENSION" );
622 case -54: return QStringLiteral( "CL_INVALID_WORK_GROUP_SIZE" );
623 case -55: return QStringLiteral( "CL_INVALID_WORK_ITEM_SIZE" );
624 case -56: return QStringLiteral( "CL_INVALID_GLOBAL_OFFSET" );
625 case -57: return QStringLiteral( "CL_INVALID_EVENT_WAIT_LIST" );
626 case -58: return QStringLiteral( "CL_INVALID_EVENT" );
627 case -59: return QStringLiteral( "CL_INVALID_OPERATION" );
628 case -60: return QStringLiteral( "CL_INVALID_GL_OBJECT" );
629 case -61: return QStringLiteral( "CL_INVALID_BUFFER_SIZE" );
630 case -62: return QStringLiteral( "CL_INVALID_MIP_LEVEL" );
631 case -63: return QStringLiteral( "CL_INVALID_GLOBAL_WORK_SIZE" );
632 case -64: return QStringLiteral( "CL_INVALID_PROPERTY" );
633 case -65: return QStringLiteral( "CL_INVALID_IMAGE_DESCRIPTOR" );
634 case -66: return QStringLiteral( "CL_INVALID_COMPILER_OPTIONS" );
635 case -67: return QStringLiteral( "CL_INVALID_LINKER_OPTIONS" );
636 case -68: return QStringLiteral( "CL_INVALID_DEVICE_PARTITION_COUNT" );
637 case -69: return QStringLiteral( "CL_INVALID_PIPE_SIZE" );
638 case -70: return QStringLiteral( "CL_INVALID_DEVICE_QUEUE" );
639 case -71: return QStringLiteral( "CL_INVALID_SPEC_ID" );
640 case -72: return QStringLiteral( "CL_MAX_SIZE_RESTRICTION_EXCEEDED" );
641 case -1002: return QStringLiteral( "CL_INVALID_D3D10_DEVICE_KHR" );
642 case -1003: return QStringLiteral( "CL_INVALID_D3D10_RESOURCE_KHR" );
643 case -1004: return QStringLiteral( "CL_D3D10_RESOURCE_ALREADY_ACQUIRED_KHR" );
644 case -1005: return QStringLiteral( "CL_D3D10_RESOURCE_NOT_ACQUIRED_KHR" );
645 case -1006: return QStringLiteral( "CL_INVALID_D3D11_DEVICE_KHR" );
646 case -1007: return QStringLiteral( "CL_INVALID_D3D11_RESOURCE_KHR" );
647 case -1008: return QStringLiteral( "CL_D3D11_RESOURCE_ALREADY_ACQUIRED_KHR" );
648 case -1009: return QStringLiteral( "CL_D3D11_RESOURCE_NOT_ACQUIRED_KHR" );
649 case -1010: return QStringLiteral( "CL_INVALID_DX9_MEDIA_ADAPTER_KHR" );
650 case -1011: return QStringLiteral( "CL_INVALID_DX9_MEDIA_SURFACE_KHR" );
651 case -1012: return QStringLiteral( "CL_DX9_MEDIA_SURFACE_ALREADY_ACQUIRED_KHR" );
652 case -1013: return QStringLiteral( "CL_DX9_MEDIA_SURFACE_NOT_ACQUIRED_KHR" );
653 case -1093: return QStringLiteral( "CL_INVALID_EGL_OBJECT_KHR" );
654 case -1092: return QStringLiteral( "CL_EGL_RESOURCE_NOT_ACQUIRED_KHR" );
655 case -1001: return QStringLiteral( "CL_PLATFORM_NOT_FOUND_KHR" );
656 case -1057: return QStringLiteral( "CL_DEVICE_PARTITION_FAILED_EXT" );
657 case -1058: return QStringLiteral( "CL_INVALID_PARTITION_COUNT_EXT" );
658 case -1059: return QStringLiteral( "CL_INVALID_PARTITION_NAME_EXT" );
659 case -1094: return QStringLiteral( "CL_INVALID_ACCELERATOR_INTEL" );
660 case -1095: return QStringLiteral( "CL_INVALID_ACCELERATOR_TYPE_INTEL" );
661 case -1096: return QStringLiteral( "CL_INVALID_ACCELERATOR_DESCRIPTOR_INTEL" );
662 case -1097: return QStringLiteral( "CL_ACCELERATOR_TYPE_NOT_SUPPORTED_INTEL" );
663 case -1000: return QStringLiteral( "CL_INVALID_GL_SHAREGROUP_REFERENCE_KHR" );
664 case -1098: return QStringLiteral( "CL_INVALID_VA_API_MEDIA_ADAPTER_INTEL" );
665 case -1099: return QStringLiteral( "CL_INVALID_VA_API_MEDIA_SURFACE_INTEL" );
666 case -1100: return QStringLiteral( "CL_VA_API_MEDIA_SURFACE_ALREADY_ACQUIRED_INTEL" );
667 case -1101: return QStringLiteral( "CL_VA_API_MEDIA_SURFACE_NOT_ACQUIRED_INTEL" );
668 default: return QStringLiteral( "CL_UNKNOWN_ERROR" );
669 }
670}
671
673{
674 // Depending on the platform version, to avoid a crash
675 // we need to use the legacy calls to C API instead of the 2.0
676 // compatible C++ API.
677 cl::Context context( QgsOpenClUtils::context() );
678 if ( QgsOpenClUtils::activePlatformVersion().toFloat() >= 200 )
679 {
680 return cl::CommandQueue( context );
681 }
682 else // legacy
683 {
684 cl::Device device( QgsOpenClUtils::activeDevice() );
685 const cl_command_queue_properties properties = 0;
687 cl_command_queue queue = clCreateCommandQueue( context(), device(), properties, nullptr );
689 return cl::CommandQueue( queue, true );
690 }
691}
692
694{
695 static cl::Context context;
696 static std::once_flag contextCreated;
697 std::call_once( contextCreated, [ = ]()
698 {
699 if ( available() && cl::Platform::getDefault()() && cl::Device::getDefault()() )
700 {
701 context = cl::Context( cl::Device::getDefault() );
702 }
703 } );
704 return context;
705}
706
707cl::Program QgsOpenClUtils::buildProgram( const cl::Context &, const QString &source, ExceptionBehavior exceptionBehavior )
708{
709 // Deprecated: ignore context and use default
710 return buildProgram( source, exceptionBehavior );
711}
712
713cl::Program QgsOpenClUtils::buildProgram( const QString &source, QgsOpenClUtils::ExceptionBehavior exceptionBehavior )
714{
715 cl::Program program;
716 try
717 {
718 program = cl::Program( QgsOpenClUtils::context(), source.toStdString( ) );
719 // OpenCL version for compatibility with older hardware, but it's up to
720 // llvm to support latest CL versions
721 bool ok;
722 const float version( QgsOpenClUtils::activePlatformVersion().toFloat( &ok ) );
723 if ( ok && version < 2.0f )
724 {
725 program.build( QStringLiteral( "-cl-std=CL%1 -I\"%2\"" )
727 .arg( sourcePath() ).toStdString().c_str() );
728 }
729 else
730 {
731 program.build( QStringLiteral( "-I\"%1\"" )
732 .arg( sourcePath() ).toStdString().c_str() );
733 }
734 }
735 catch ( cl::BuildError &e )
736 {
737 QString build_log( buildLog( e ) );
738 if ( build_log.isEmpty() )
739 build_log = QObject::tr( "Build logs not available!" );
740 const QString err = QObject::tr( "Error building OpenCL program: %1" )
741 .arg( build_log );
743 if ( exceptionBehavior == Throw )
744 throw e;
745 }
746 catch ( cl::Error &e )
747 {
748 const QString err = QObject::tr( "Error %1 building OpenCL program in %2" )
749 .arg( errorText( e.err() ), QString::fromStdString( e.what() ) );
751 throw e;
752 }
753 return program;
754}
@ Warning
Warning message.
Definition qgis.h:156
@ Critical
Critical/error message.
Definition qgis.h:157
@ Info
Information message.
Definition qgis.h:155
@ Success
Used for reporting a successful operation.
Definition qgis.h:158
static void logMessage(const QString &message, const QString &tag=QString(), Qgis::MessageLevel level=Qgis::MessageLevel::Warning, bool notifyUser=true)
Adds a message to the log instance (and creates it if necessary).
The QgsOpenClUtils class is responsible for common OpenCL operations such as.
static Q_DECL_DEPRECATED cl::Program buildProgram(const cl::Context &context, const QString &source, ExceptionBehavior exceptionBehavior=Catch)
Build the program from source in the given context and depending on exceptionBehavior can throw or ca...
static void setSourcePath(const QString &value)
Set the base path to OpenCL program directory.
static QString buildLog(cl::BuildError &error)
Extract and return the build log error from error.
static void storePreferredDevice(const QString deviceId)
Store in the settings the preferred deviceId device identifier.
static QString sourcePath()
Returns the base path to OpenCL program directory.
static cl::Context context()
Context factory.
static cl::Device activeDevice()
Returns the active device.
static bool enabled()
Returns true if OpenCL is enabled in the user settings.
static bool available()
Checks whether a suitable OpenCL platform and device is available on this system and initialize the Q...
static QString deviceDescription(const cl::Device device)
Returns a formatted description for the device.
static QString activeDeviceInfo(const Info infoType=Info::Name)
Returns infoType information about the active (default) device.
static const std::vector< cl::Device > devices()
Returns a list of OpenCL devices found on this sysytem.
static void setEnabled(bool enabled)
Set the OpenCL user setting to enabled.
static QString preferredDevice()
Read from the settings the preferred device identifier.
static QString deviceInfo(const Info infoType, cl::Device device)
Returns infoType information about the device.
static QString errorText(const int errorCode)
Returns a string representation from an OpenCL errorCode.
static cl::CommandQueue commandQueue()
Create an OpenCL command queue from the default context.
static QString sourceFromPath(const QString &path)
Read an OpenCL source file from path.
static QString sourceFromBaseName(const QString &baseName)
Returns the full path to a an OpenCL source file from the baseName ('.cl' extension is automatically ...
ExceptionBehavior
The ExceptionBehavior enum define how exceptions generated by OpenCL should be treated.
@ Throw
Write errors in the message log and re-throw exceptions.
static QString deviceId(const cl::Device device)
Create a string identifier from a device.
Info
The Info enum maps to OpenCL info constants.
static QLatin1String LOGMESSAGE_TAG
OpenCL string for message logs.
static QString activePlatformVersion()
Returns the active platform OpenCL version string (e.g.
This class is a composition of two QSettings instances:
Definition qgssettings.h:64
QVariant value(const QString &key, const QVariant &defaultValue=QVariant(), Section section=NoSection) const
Returns the value for setting key.
void setValue(const QString &key, const QVariant &value, QgsSettings::Section section=QgsSettings::NoSection)
Sets the value of setting key to value.
#define Q_NOWARN_DEPRECATED_POP
Definition qgis.h:6643
#define Q_NOWARN_DEPRECATED_PUSH
Definition qgis.h:6642
Q_GLOBAL_STATIC(QReadWriteLock, sDefinitionCacheLock)
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:39
#define QgsDebugError(str)
Definition qgslogger.h:38