2014-07-08 03:11:14 -04:00
# include "EmberCLPch.h"
# include "OpenCLWrapper.h"
namespace EmberCLns
{
/// <summary>
/// Constructor that sets everything to an uninitialized state.
/// No OpenCL setup is done here, the caller must explicitly do it.
/// </summary>
OpenCLWrapper : : OpenCLWrapper ( )
{
m_Init = false ;
m_Shared = false ;
m_PlatformIndex = 0 ;
m_DeviceIndex = 0 ;
m_LocalMemSize = 0 ;
cl : : Platform : : get ( & m_Platforms ) ;
m_Devices . resize ( m_Platforms . size ( ) ) ;
for ( size_t i = 0 ; i < m_Platforms . size ( ) ; i + + )
m_Platforms [ i ] . getDevices ( CL_DEVICE_TYPE_ALL , & m_Devices [ i ] ) ;
}
/// <summary>
/// Determine if OpenCL is available on the system.
/// </summary>
/// <returns>True if any OpenCL platform and at least 1 device within that platform exists on the system, else false.</returns>
bool OpenCLWrapper : : CheckOpenCL ( )
{
for ( size_t i = 0 ; i < m_Platforms . size ( ) ; i + + )
for ( size_t j = 0 ; j < m_Devices [ i ] . size ( ) ; j + + )
return true ;
return false ;
}
/// <summary>
/// Initialize the specified platform and device.
/// This can be shared with OpenGL.
/// </summary>
/// <param name="platform">The index platform of the platform to use</param>
/// <param name="device">The index device of the device to use</param>
/// <param name="shared">True if shared with OpenGL, else false.</param>
/// <returns>True if success, else false.</returns>
2014-12-06 00:05:09 -05:00
bool OpenCLWrapper : : Init ( uint platform , uint device , bool shared )
2014-07-08 03:11:14 -04:00
{
cl_int err ;
m_Init = false ;
m_ErrorReport . clear ( ) ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
if ( m_Platforms . size ( ) > 0 )
{
if ( platform < m_Platforms . size ( ) & & platform < m_Devices . size ( ) )
{
m_PlatformIndex = platform ; //Platform is ok, now do context.
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
if ( CreateContext ( shared ) )
{
//Context is ok, now do device.
if ( device < m_Devices [ m_PlatformIndex ] . size ( ) )
{
//At least one GPU device is present, so create a command queue.
m_Queue = cl : : CommandQueue ( m_Context , m_Devices [ m_PlatformIndex ] [ device ] , 0 , & err ) ;
if ( CheckCL ( err , " cl::CommandQueue() " ) )
{
m_DeviceIndex = device ;
m_Platform = m_Platforms [ m_PlatformIndex ] ;
m_Device = m_Devices [ m_PlatformIndex ] [ device ] ;
m_DeviceVec . clear ( ) ;
m_DeviceVec . push_back ( m_Device ) ;
2014-12-07 02:51:44 -05:00
m_LocalMemSize = uint ( GetInfo < cl_ulong > ( m_PlatformIndex , m_DeviceIndex , CL_DEVICE_LOCAL_MEM_SIZE ) ) ;
2014-07-08 03:11:14 -04:00
m_Shared = shared ;
m_Init = true ; //Command queue is ok, it's now ok to begin building and running programs.
}
}
}
}
}
return m_Init ;
}
/// <summary>
/// Compile and add the program, using the specified entry point.
/// If a program with the same name already exists then it will be replaced.
/// </summary>
/// <param name="name">The name of the program</param>
/// <param name="program">The program source</param>
/// <param name="entryPoint">The name of the entry point kernel function in the program</param>
/// <returns>True if success, else false.</returns>
2014-12-05 21:30:46 -05:00
bool OpenCLWrapper : : AddProgram ( const string & name , const string & program , const string & entryPoint , bool doublePrecision )
2014-07-08 03:11:14 -04:00
{
Spk spk ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
if ( CreateSPK ( name , program , entryPoint , spk , doublePrecision ) )
{
for ( size_t i = 0 ; i < m_Programs . size ( ) ; i + + )
{
if ( name = = m_Programs [ i ] . m_Name )
{
m_Programs [ i ] = spk ;
return true ;
}
}
//Nothing was found, so add.
m_Programs . push_back ( spk ) ;
return true ;
}
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
return false ;
}
/// <summary>
/// Clear the programs.
/// </summary>
void OpenCLWrapper : : ClearPrograms ( )
{
m_Programs . clear ( ) ;
}
/// <summary>
/// Add a buffer with the specified size and name.
/// Three possible actions to take:
/// Buffer didn't exist, so create and add.
/// Buffer existed, but was a different size. Replace.
/// Buffer existed with the same size, do nothing.
/// </summary>
/// <param name="name">The name of the buffer</param>
/// <param name="size">The size in bytes of the buffer</param>
/// <param name="flags">The buffer flags. Default: CL_MEM_READ_WRITE.</param>
/// <returns>True if success, else false.</returns>
2014-10-14 11:53:15 -04:00
bool OpenCLWrapper : : AddBuffer ( const string & name , size_t size , cl_mem_flags flags )
2014-07-08 03:11:14 -04:00
{
cl_int err ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
if ( m_Init )
{
int bufferIndex = FindBufferIndex ( name ) ;
if ( bufferIndex = = - 1 ) //If the buffer didn't exist, create and add.
{
2014-12-05 21:30:46 -05:00
cl : : Buffer buff ( m_Context , flags , size , nullptr , & err ) ;
2014-07-08 03:11:14 -04:00
if ( ! CheckCL ( err , " cl::Buffer() " ) )
return false ;
NamedBuffer nb ( buff , name ) ;
m_Buffers . push_back ( nb ) ;
}
else if ( GetBufferSize ( bufferIndex ) ! = size ) //If it did exist, only create and add if the sizes were different.
{
2014-12-05 21:30:46 -05:00
m_Buffers [ bufferIndex ] = NamedBuffer ( cl : : Buffer ( m_Context , flags , 0 , nullptr , & err ) , " emptybuffer " ) ; //First clear out the original so the two don't exist in memory at once.
2014-07-08 03:11:14 -04:00
2014-12-05 21:30:46 -05:00
cl : : Buffer buff ( m_Context , flags , size , nullptr , & err ) ; //Create the new buffer.
2014-07-08 03:11:14 -04:00
if ( ! CheckCL ( err , " cl::Buffer() " ) )
return false ;
2014-10-14 11:53:15 -04:00
NamedBuffer nb ( buff , name ) ; //Make a named buffer out of the new buffer.
2014-07-08 03:11:14 -04:00
2014-10-14 11:53:15 -04:00
m_Buffers [ bufferIndex ] = nb ; //Finally, assign.
2014-07-08 03:11:14 -04:00
}
//If the buffer existed and the sizes were the same, take no action.
return true ;
}
return false ;
}
/// <summary>
/// Add and/or write a buffer of data with the specified name to the list of buffers.
/// Three possible actions to take:
/// Buffer didn't exist, so create and add.
/// Buffer existed, but was a different size. Replace.
/// Buffer existed with the same size, copy data.
/// </summary>
/// <param name="name">The name of the buffer</param>
/// <param name="data">A pointer to the buffer</param>
/// <param name="size">The size in bytes of the buffer</param>
2014-10-14 11:53:15 -04:00
/// <param name="flags">The buffer flags. Default: CL_MEM_READ_WRITE.</param>
2014-07-08 03:11:14 -04:00
/// <returns>True if success, else false.</returns>
2014-10-14 11:53:15 -04:00
bool OpenCLWrapper : : AddAndWriteBuffer ( const string & name , void * data , size_t size , cl_mem_flags flags )
2014-07-08 03:11:14 -04:00
{
bool b = false ;
2014-10-14 11:53:15 -04:00
if ( AddBuffer ( name , size , flags ) )
b = WriteBuffer ( name , data , size ) ;
2014-07-08 03:11:14 -04:00
return b ;
}
/// <summary>
/// Write data to an existing buffer with the specified name.
/// </summary>
/// <param name="name">The name of the buffer</param>
/// <param name="data">A pointer to the buffer</param>
/// <param name="size">The size in bytes of the buffer</param>
/// <returns>True if success, else false.</returns>
2014-10-14 11:53:15 -04:00
bool OpenCLWrapper : : WriteBuffer ( const string & name , void * data , size_t size )
2014-07-08 03:11:14 -04:00
{
int bufferIndex = FindBufferIndex ( name ) ;
2014-12-05 21:30:46 -05:00
return bufferIndex ! = - 1 ? WriteBuffer ( bufferIndex , data , size ) : false ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Write data to an existing buffer at the specified index.
/// </summary>
/// <param name="bufferIndex">The index of the buffer</param>
/// <param name="data">A pointer to the buffer</param>
/// <param name="size">The size in bytes of the buffer</param>
/// <returns>True if success, else false.</returns>
2014-12-06 00:05:09 -05:00
bool OpenCLWrapper : : WriteBuffer ( uint bufferIndex , void * data , size_t size )
2014-07-08 03:11:14 -04:00
{
if ( m_Init & & ( bufferIndex < m_Buffers . size ( ) ) & & ( GetBufferSize ( bufferIndex ) = = size ) )
{
cl : : Event e ;
2014-12-05 21:30:46 -05:00
cl_int err = m_Queue . enqueueWriteBuffer ( m_Buffers [ bufferIndex ] . m_Buffer , CL_TRUE , 0 , size , data , nullptr , & e ) ;
2014-07-08 03:11:14 -04:00
e . wait ( ) ;
m_Queue . finish ( ) ;
if ( CheckCL ( err , " cl::CommandQueue::enqueueWriteBuffer() " ) )
return true ;
}
return false ;
}
/// <summary>
/// Read data from an existing buffer with the specified name.
/// </summary>
/// <param name="name">The name of the buffer</param>
/// <param name="data">A pointer to a buffer to copy the data to</param>
/// <param name="size">The size in bytes of the buffer</param>
/// <returns>True if success, else false.</returns>
2014-10-14 11:53:15 -04:00
bool OpenCLWrapper : : ReadBuffer ( const string & name , void * data , size_t size )
2014-07-08 03:11:14 -04:00
{
int bufferIndex = FindBufferIndex ( name ) ;
2014-12-05 21:30:46 -05:00
return bufferIndex ! = - 1 ? ReadBuffer ( bufferIndex , data , size ) : false ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Read data from an existing buffer at the specified index.
/// </summary>
/// <param name="bufferIndex">The index of the buffer</param>
/// <param name="data">A pointer to a buffer to copy the data to</param>
/// <param name="size">The size in bytes of the buffer</param>
/// <returns>True if success, else false.</returns>
2014-12-06 00:05:09 -05:00
bool OpenCLWrapper : : ReadBuffer ( uint bufferIndex , void * data , size_t size )
2014-07-08 03:11:14 -04:00
{
if ( m_Init & & ( bufferIndex < m_Buffers . size ( ) ) & & ( GetBufferSize ( bufferIndex ) = = size ) )
{
cl : : Event e ;
2014-12-05 21:30:46 -05:00
cl_int err = m_Queue . enqueueReadBuffer ( m_Buffers [ bufferIndex ] . m_Buffer , CL_TRUE , 0 , size , data , nullptr , & e ) ;
2014-07-08 03:11:14 -04:00
e . wait ( ) ;
m_Queue . finish ( ) ;
if ( CheckCL ( err , " cl::CommandQueue::enqueueReadBuffer() " ) )
return true ;
}
return false ;
}
/// <summary>
/// Find the index of the buffer with the specified name.
/// </summary>
/// <param name="name">The name of the buffer to search for</param>
/// <returns>The index if found, else -1.</returns>
2014-10-14 11:53:15 -04:00
int OpenCLWrapper : : FindBufferIndex ( const string & name )
2014-07-08 03:11:14 -04:00
{
2014-12-07 02:51:44 -05:00
for ( size_t i = 0 ; i < m_Buffers . size ( ) ; i + + )
2014-07-08 03:11:14 -04:00
if ( m_Buffers [ i ] . m_Name = = name )
2014-12-07 02:51:44 -05:00
return int ( i ) ;
2014-07-08 03:11:14 -04:00
return - 1 ;
}
/// <summary>
/// Get the size of the buffer with the specified name.
/// </summary>
/// <param name="name">The name of the buffer to search for</param>
/// <returns>The size of the buffer if found, else 0.</returns>
2014-12-06 00:05:09 -05:00
uint OpenCLWrapper : : GetBufferSize ( const string & name )
2014-07-08 03:11:14 -04:00
{
2014-12-05 21:30:46 -05:00
int bufferIndex = FindBufferIndex ( name ) ;
2014-07-08 03:11:14 -04:00
2014-12-05 21:30:46 -05:00
return bufferIndex ! = - 1 ? GetBufferSize ( bufferIndex ) : 0 ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Get the size of the buffer at the specified index.
/// </summary>
/// <param name="name">The index of the buffer to get the size of</param>
/// <returns>The size of the buffer if found, else 0.</returns>
2014-12-06 00:05:09 -05:00
uint OpenCLWrapper : : GetBufferSize ( uint bufferIndex )
2014-07-08 03:11:14 -04:00
{
2014-12-07 02:51:44 -05:00
if ( m_Init & & ( bufferIndex < m_Buffers . size ( ) ) )
return uint ( m_Buffers [ bufferIndex ] . m_Buffer . getInfo < CL_MEM_SIZE > ( nullptr ) ) ;
2014-07-08 03:11:14 -04:00
return 0 ;
}
/// <summary>
/// Clear all buffers.
/// </summary>
void OpenCLWrapper : : ClearBuffers ( )
{
m_Buffers . clear ( ) ;
}
/// <summary>
/// Add and/or write a new 2D image.
/// Three possible actions to take:
/// Image didn't exist, so create and add.
/// Image existed, but was a different size. Replace.
/// Image existed with the same size, copy data.
/// </summary>
/// <param name="name">The name of the image to add/replace</param>
/// <param name="flags">The memory flags</param>
/// <param name="format">The image format</param>
/// <param name="width">The width in pixels of the image</param>
/// <param name="height">The height in pixels of the image</param>
/// <param name="row_pitch">The row pitch (usually zero)</param>
/// <param name="data">The image data. Default: NULL.</param>
/// <param name="shared">True if shared with an OpenGL texture, else false. Default: false.</param>
/// <param name="texName">The texture ID of the shared OpenGL texture if shared. Default: 0.</param>
/// <returns>True if success, else false.</returns>
2014-10-14 11:53:15 -04:00
bool OpenCLWrapper : : AddAndWriteImage ( const string & name , cl_mem_flags flags , const cl : : ImageFormat & format , : : size_t width , : : size_t height , : : size_t row_pitch , void * data , bool shared , GLuint texName )
2014-07-08 03:11:14 -04:00
{
cl_int err ;
if ( m_Init )
{
int imageIndex = FindImageIndex ( name , shared ) ;
if ( imageIndex = = - 1 ) //If the image didn't exist, create and add.
{
if ( shared )
{
//::wglMakeCurrent(wglGetCurrentDC(), wglGetCurrentContext());
IMAGEGL2D imageGL ( m_Context , flags , GL_TEXTURE_2D , 0 , texName , & err ) ;
NamedImage2DGL namedImageGL ( imageGL , name ) ;
if ( CheckCL ( err , " cl::ImageGL() " ) )
{
m_GLImages . push_back ( namedImageGL ) ;
if ( data )
2014-12-07 02:51:44 -05:00
return WriteImage2D ( uint ( m_GLImages . size ( ) - 1 ) , true , width , height , row_pitch , data ) ; //OpenGL images/textures require a separate write.
2014-07-08 03:11:14 -04:00
else
return true ;
}
}
else
{
NamedImage2D namedImage ( cl : : Image2D ( m_Context , flags , format , width , height , row_pitch , data , & err ) , name ) ;
if ( CheckCL ( err , " cl::Image2D() " ) )
{
m_Images . push_back ( namedImage ) ;
return true ;
}
}
}
else //It did exist, so create new if sizes are different. Write if data is not NULL.
{
if ( shared )
{
IMAGEGL2D imageGL = m_GLImages [ imageIndex ] . m_Image ;
if ( ! CompareImageParams ( imageGL , flags , format , width , height , row_pitch ) )
{
NamedImage2DGL namedImageGL ( IMAGEGL2D ( m_Context , flags , GL_TEXTURE_2D , 0 , texName , & err ) , name ) ; //Sizes are different, so create new.
if ( CheckCL ( err , " cl::ImageGL() " ) )
{
m_GLImages [ imageIndex ] = namedImageGL ;
}
else
return false ;
}
//Write data to new image since OpenGL images/textures require a separate write, must match new size.
if ( data )
return WriteImage2D ( imageIndex , true , width , height , row_pitch , data ) ;
else
return true ;
}
else
{
2014-10-14 11:53:15 -04:00
if ( ! CompareImageParams ( m_Images [ imageIndex ] . m_Image , flags , format , width , height , row_pitch ) )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
m_Images [ imageIndex ] = NamedImage2D ( ) ; //First clear out the original so the two don't exist in memory at once.
2014-07-08 03:11:14 -04:00
NamedImage2D namedImage ( cl : : Image2D ( m_Context , flags , format , width , height , row_pitch , data , & err ) , name ) ;
if ( CheckCL ( err , " cl::Image2D() " ) )
{
m_Images [ imageIndex ] = namedImage ;
return true ;
}
}
else if ( data )
return WriteImage2D ( imageIndex , false , width , height , row_pitch , data ) ;
else //Strange case: images were same dimensions but no data was passed in, so do nothing.
return true ;
}
}
}
return false ;
}
/// <summary>
/// Write data to an existing 2D image at the specified index.
/// </summary>
/// <param name="index">The index of the image</param>
/// <param name="shared">True if shared with an OpenGL texture, else false.</param>
/// <param name="width">The width in pixels of the image</param>
/// <param name="height">The height in pixels of the image</param>
/// <param name="row_pitch">The row pitch (usually zero)</param>
/// <param name="data">The image data</param>
/// <returns>True if success, else false.</returns>
2014-12-06 00:05:09 -05:00
bool OpenCLWrapper : : WriteImage2D ( uint index , bool shared , : : size_t width , : : size_t height , : : size_t row_pitch , void * data )
2014-07-08 03:11:14 -04:00
{
if ( m_Init )
{
cl_int err ;
cl : : Event e ;
cl : : size_t < 3 > origin , region ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
origin [ 0 ] = 0 ;
origin [ 1 ] = 0 ;
origin [ 2 ] = 0 ;
region [ 0 ] = width ;
region [ 1 ] = height ;
region [ 2 ] = 1 ;
if ( shared & & index < m_GLImages . size ( ) )
{
IMAGEGL2D imageGL = m_GLImages [ index ] . m_Image ;
if ( EnqueueAcquireGLObjects ( imageGL ) )
{
2014-12-05 21:30:46 -05:00
err = m_Queue . enqueueWriteImage ( imageGL , CL_TRUE , origin , region , row_pitch , 0 , data , nullptr , & e ) ;
2014-07-08 03:11:14 -04:00
e . wait ( ) ;
m_Queue . finish ( ) ;
bool b = EnqueueReleaseGLObjects ( imageGL ) ;
return CheckCL ( err , " cl::enqueueWriteImage() " ) && b ;
}
}
else if ( ! shared & & index < m_Images . size ( ) )
{
2014-12-05 21:30:46 -05:00
err = m_Queue . enqueueWriteImage ( m_Images [ index ] . m_Image , CL_TRUE , origin , region , row_pitch , 0 , data , nullptr , & e ) ;
2014-07-08 03:11:14 -04:00
e . wait ( ) ;
m_Queue . finish ( ) ;
return CheckCL ( err , " cl::enqueueWriteImage() " ) ;
}
}
return false ;
}
/// <summary>
/// Read data from an existing 2D image with the specified name.
/// </summary>
/// <param name="name">The name of the image</param>
/// <param name="width">The width in pixels of the image</param>
/// <param name="height">The height in pixels of the image</param>
/// <param name="row_pitch">The row pitch (usually zero)</param>
/// <param name="shared">True if shared with an OpenGL texture, else false.</param>
/// <param name="data">A pointer to a buffer to copy the data to</param>
/// <returns>True if success, else false.</returns>
2014-10-14 11:53:15 -04:00
bool OpenCLWrapper : : ReadImage ( const string & name , : : size_t width , : : size_t height , : : size_t row_pitch , bool shared , void * data )
2014-07-08 03:11:14 -04:00
{
if ( m_Init )
{
int imageIndex = FindImageIndex ( name , shared ) ;
if ( imageIndex ! = - 1 )
return ReadImage ( imageIndex , width , height , row_pitch , shared , data ) ;
}
return false ;
}
/// <summary>
/// Read data from an existing 2D image at the specified index.
/// </summary>
/// <param name="name">The name of the image</param>
/// <param name="width">The width in pixels of the image</param>
/// <param name="height">The height in pixels of the image</param>
/// <param name="row_pitch">The row pitch (usually zero)</param>
/// <param name="shared">True if shared with an OpenGL texture, else false.</param>
/// <param name="data">A pointer to a buffer to copy the data to</param>
/// <returns>True if success, else false.</returns>
2014-12-06 00:05:09 -05:00
bool OpenCLWrapper : : ReadImage ( uint imageIndex , : : size_t width , : : size_t height , : : size_t row_pitch , bool shared , void * data )
2014-07-08 03:11:14 -04:00
{
if ( m_Init )
{
cl_int err ;
cl : : Event e ;
cl : : size_t < 3 > origin , region ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
origin [ 0 ] = 0 ;
origin [ 1 ] = 0 ;
origin [ 2 ] = 0 ;
region [ 0 ] = width ;
region [ 1 ] = height ;
region [ 2 ] = 1 ;
if ( shared & & imageIndex < m_GLImages . size ( ) )
{
IMAGEGL2D imageGL = m_GLImages [ imageIndex ] . m_Image ;
if ( EnqueueAcquireGLObjects ( imageGL ) )
{
err = m_Queue . enqueueReadImage ( m_GLImages [ imageIndex ] . m_Image , true , origin , region , row_pitch , 0 , data ) ;
bool b = EnqueueReleaseGLObjects ( m_GLImages [ imageIndex ] . m_Image ) ;
return CheckCL ( err , " cl::enqueueReadImage() " ) && b ;
}
}
else if ( ! shared & & imageIndex < m_Images . size ( ) )
{
err = m_Queue . enqueueReadImage ( m_Images [ imageIndex ] . m_Image , true , origin , region , row_pitch , 0 , data ) ;
return CheckCL ( err , " cl::enqueueReadImage() " ) ;
}
}
return false ;
}
/// <summary>
/// Find the index of the 2D image with the specified name.
/// </summary>
/// <param name="name">The name of the image to search for</param>
/// <param name="shared">True if shared with an OpenGL texture, else false.</param>
/// <returns>The index if found, else -1.</returns>
2014-10-14 11:53:15 -04:00
int OpenCLWrapper : : FindImageIndex ( const string & name , bool shared )
2014-07-08 03:11:14 -04:00
{
if ( shared )
{
2014-12-07 02:51:44 -05:00
for ( size_t i = 0 ; i < m_GLImages . size ( ) ; i + + )
2014-07-08 03:11:14 -04:00
if ( m_GLImages [ i ] . m_Name = = name )
2014-12-09 03:24:28 -05:00
return int ( i ) ;
2014-07-08 03:11:14 -04:00
}
else
{
2014-12-07 02:51:44 -05:00
for ( size_t i = 0 ; i < m_Images . size ( ) ; i + + )
2014-07-08 03:11:14 -04:00
if ( m_Images [ i ] . m_Name = = name )
2014-12-09 03:24:28 -05:00
return int ( i ) ;
2014-07-08 03:11:14 -04:00
}
return - 1 ;
}
/// <summary>
/// Get the size of the 2D image with the specified name.
/// </summary>
/// <param name="name">The name of the image to search for</param>
/// <param name="shared">True if shared with an OpenGL texture, else false.</param>
/// <returns>The size of the 2D image if found, else 0.</returns>
2014-12-06 00:05:09 -05:00
uint OpenCLWrapper : : GetImageSize ( const string & name , bool shared )
2014-07-08 03:11:14 -04:00
{
int imageIndex = FindImageIndex ( name , shared ) ;
return GetImageSize ( imageIndex , shared ) ;
}
/// <summary>
/// Get the size of the 2D image at the specified index.
/// </summary>
/// <param name="imageIndex">Index of the image to search for</param>
/// <param name="shared">True if shared with an OpenGL texture, else false.</param>
/// <returns>The size of the 2D image if found, else 0.</returns>
2014-12-06 00:05:09 -05:00
uint OpenCLWrapper : : GetImageSize ( uint imageIndex , bool shared )
2014-07-08 03:11:14 -04:00
{
size_t size = 0 ;
if ( m_Init )
{
if ( shared & & imageIndex < m_GLImages . size ( ) )
{
vector < cl : : Memory > images ;
images . push_back ( m_GLImages [ imageIndex ] . m_Image ) ;
IMAGEGL2D image = m_GLImages [ imageIndex ] . m_Image ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
if ( EnqueueAcquireGLObjects ( & images ) )
2014-12-05 21:30:46 -05:00
size = image . getImageInfo < CL_IMAGE_WIDTH > ( nullptr ) * image . getImageInfo < CL_IMAGE_HEIGHT > ( nullptr ) * image . getImageInfo < CL_IMAGE_ELEMENT_SIZE > ( nullptr ) ; //Should pitch be checked here?
2014-07-08 03:11:14 -04:00
EnqueueReleaseGLObjects ( & images ) ;
}
else if ( ! shared & & imageIndex < m_Images . size ( ) )
{
cl : : Image2D image = m_Images [ imageIndex ] . m_Image ;
2014-12-05 21:30:46 -05:00
size = image . getImageInfo < CL_IMAGE_WIDTH > ( nullptr ) * image . getImageInfo < CL_IMAGE_HEIGHT > ( nullptr ) * image . getImageInfo < CL_IMAGE_ELEMENT_SIZE > ( nullptr ) ; //Should pitch be checked here?
2014-07-08 03:11:14 -04:00
}
}
2014-12-07 02:51:44 -05:00
return uint ( size ) ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Compare the passed in image with the specified parameters.
/// </summary>
/// <param name="image">The image to compare</param>
/// <param name="flags">The memory flags to compare (ommitted)</param>
/// <param name="format">The format to compare</param>
/// <param name="width">The width to compare</param>
/// <param name="height">The height to compare</param>
/// <param name="row_pitch">The row_pitch to compare (omitted)</param>
/// <returns>True if all parameters matched, else false.</returns>
bool OpenCLWrapper : : CompareImageParams ( cl : : Image & image , cl_mem_flags flags , const cl : : ImageFormat & format , : : size_t width , : : size_t height , : : size_t row_pitch )
{
2014-12-05 21:30:46 -05:00
cl_image_format tempFormat = image . getImageInfo < CL_IMAGE_FORMAT > ( nullptr ) ;
2014-07-08 03:11:14 -04:00
return ( /*image.getImageInfo<CL_MEM_FLAGS>() == flags &&*/
2014-12-05 21:30:46 -05:00
tempFormat . image_channel_data_type = = format . image_channel_data_type & &
tempFormat . image_channel_order = = format . image_channel_order & &
image . getImageInfo < CL_IMAGE_WIDTH > ( nullptr ) = = width & &
image . getImageInfo < CL_IMAGE_HEIGHT > ( nullptr ) = = height /* &&
2014-07-08 03:11:14 -04:00
image . getImageInfo < CL_IMAGE_ROW_PITCH > ( ) = = row_pitch */ ) ; //Pitch will be (width * bytes per pixel) + padding.
}
/// <summary>
/// Clear all images.
/// </summary>
/// <param name="shared">True to clear shared images, else clear regular images.</param>
void OpenCLWrapper : : ClearImages ( bool shared )
{
if ( shared )
m_GLImages . clear ( ) ;
else
m_Images . clear ( ) ;
}
/// <summary>
/// Create a 2D image and store in the image passed in.
/// </summary>
/// <param name="image2D">The 2D image to store the newly created image in</param>
/// <param name="flags">The memory flags to use</param>
/// <param name="format">The format to use</param>
/// <param name="width">The width in pixels of the image</param>
/// <param name="height">The height in pixels of the image</param>
/// <param name="row_pitch">The row pitch (usually zero)</param>
/// <param name="data">The image data. Default: NULL.</param>
/// <returns>True if success, else false.</returns>
bool OpenCLWrapper : : CreateImage2D ( cl : : Image2D & image2D , cl_mem_flags flags , cl : : ImageFormat format , : : size_t width , : : size_t height , : : size_t row_pitch , void * data )
{
if ( m_Init )
{
cl_int err ;
image2D = cl : : Image2D ( m_Context ,
flags ,
format ,
width ,
height ,
row_pitch ,
data ,
& err ) ;
return CheckCL ( err , " cl::Image2D() " ) ;
}
return false ;
}
/// <summary>
/// Create a 2D image shared with an OpenGL texture and store in the image passed in.
/// </summary>
/// <param name="image2DGL">The 2D image to store the newly created image in</param>
/// <param name="flags">The memory flags to use</param>
/// <param name="target">The target</param>
/// <param name="miplevel">The mip map level</param>
/// <param name="texobj">The texture ID of the shared OpenGL texture</param>
/// <returns>True if success, else false.</returns>
bool OpenCLWrapper : : CreateImage2DGL ( IMAGEGL2D & image2DGL , cl_mem_flags flags , GLenum target , GLint miplevel , GLuint texobj )
{
if ( m_Init )
{
cl_int err ;
image2DGL = IMAGEGL2D ( m_Context ,
flags ,
target ,
miplevel ,
texobj ,
& err ) ;
return CheckCL ( err , " cl::ImageGL() " ) ;
}
return false ;
}
/// <summary>
/// Acquire the shared 2D image with the specified name.
/// </summary>
/// <param name="name">The name of the image to acquire</param>
/// <returns>True if success, else false.</returns>
2014-10-14 11:53:15 -04:00
bool OpenCLWrapper : : EnqueueAcquireGLObjects ( const string & name )
2014-07-08 03:11:14 -04:00
{
int index = FindImageIndex ( name , true ) ;
if ( index ! = - 1 )
return EnqueueAcquireGLObjects ( m_GLImages [ index ] . m_Image ) ;
return false ;
}
/// <summary>
/// Acquire the shared 2D image.
/// </summary>
/// <param name="image">The image to acquire</param>
/// <returns>True if success, else false.</returns>
bool OpenCLWrapper : : EnqueueAcquireGLObjects ( IMAGEGL2D & image )
{
if ( m_Init & & m_Shared )
{
vector < cl : : Memory > images ;
images . push_back ( image ) ;
cl_int err = m_Queue . enqueueAcquireGLObjects ( & images ) ;
m_Queue . finish ( ) ;
return CheckCL ( err , " cl::CommandQueue::enqueueAcquireGLObjects() " ) ;
}
return false ;
}
/// <summary>
/// Reelease the shared 2D image with the specified name.
/// </summary>
/// <param name="name">The name of the image to release</param>
/// <returns>True if success, else false.</returns>
2014-10-14 11:53:15 -04:00
bool OpenCLWrapper : : EnqueueReleaseGLObjects ( const string & name )
2014-07-08 03:11:14 -04:00
{
int index = FindImageIndex ( name , true ) ;
if ( index ! = - 1 )
return EnqueueReleaseGLObjects ( m_GLImages [ index ] . m_Image ) ;
return false ;
}
/// <summary>
/// Release the shared 2D image.
/// </summary>
/// <param name="image">The image to release</param>
/// <returns>True if success, else false.</returns>
bool OpenCLWrapper : : EnqueueReleaseGLObjects ( IMAGEGL2D & image )
{
if ( m_Init & & m_Shared )
{
vector < cl : : Memory > images ;
images . push_back ( image ) ;
cl_int err = m_Queue . enqueueReleaseGLObjects ( & images ) ;
m_Queue . finish ( ) ;
return CheckCL ( err , " cl::CommandQueue::enqueueReleaseGLObjects() " ) ;
}
return false ;
}
/// <summary>
/// Acquire a vector of shared OpenGL memory objects.
/// </summary>
/// <param name="memObjects">The memory objects to acquire</param>
/// <returns>True if success, else false.</returns>
bool OpenCLWrapper : : EnqueueAcquireGLObjects ( const VECTOR_CLASS < cl : : Memory > * memObjects )
{
if ( m_Init & & m_Shared )
{
cl_int err = m_Queue . enqueueAcquireGLObjects ( memObjects ) ;
m_Queue . finish ( ) ;
return CheckCL ( err , " cl::CommandQueue::enqueueAcquireGLObjects() " ) ;
}
return false ;
}
/// <summary>
/// Release a vector of shared OpenGL memory objects.
/// </summary>
/// <param name="memObjects">The memory objects to release</param>
/// <returns>True if success, else false.</returns>
bool OpenCLWrapper : : EnqueueReleaseGLObjects ( const VECTOR_CLASS < cl : : Memory > * memObjects )
{
if ( m_Init & & m_Shared )
{
cl_int err = m_Queue . enqueueReleaseGLObjects ( memObjects ) ;
m_Queue . finish ( ) ;
return CheckCL ( err , " cl::CommandQueue::enqueueReleaseGLObjects() " ) ;
}
return false ;
}
/// <summary>
/// Create a texture sampler.
/// </summary>
/// <param name="sampler">The sampler to store the newly created sampler in</param>
/// <param name="normalizedCoords">True to use normalized coordinates, else don't.</param>
/// <param name="addressingMode">The addressing mode to use</param>
/// <param name="filterMode">The filter mode to use</param>
/// <returns>True if success, else false.</returns>
bool OpenCLWrapper : : CreateSampler ( cl : : Sampler & sampler , cl_bool normalizedCoords , cl_addressing_mode addressingMode , cl_filter_mode filterMode )
{
cl_int err ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
sampler = cl : : Sampler ( m_Context ,
normalizedCoords ,
addressingMode ,
filterMode ,
& err ) ;
return CheckCL ( err , " cl::Sampler() " ) ;
}
/// <summary>
/// Set the argument at the specified index for the kernel at the specified index to be
/// the buffer with the specified name.
/// </summary>
/// <param name="kernelIndex">Index of the kernel</param>
/// <param name="argIndex">Index of the argument</param>
/// <param name="name">The name of the buffer</param>
/// <returns>True if success, else false.</returns>
2014-12-06 00:05:09 -05:00
bool OpenCLWrapper : : SetBufferArg ( uint kernelIndex , uint argIndex , const string & name )
2014-07-08 03:11:14 -04:00
{
int bufferIndex = OpenCLWrapper : : FindBufferIndex ( name ) ;
return bufferIndex ! = - 1 ? SetBufferArg ( kernelIndex , argIndex , bufferIndex ) : false ;
}
/// <summary>
/// Set the argument at the specified index for the kernel at the specified index to be
/// the buffer at the specified index.
/// </summary>
/// <param name="kernelIndex">Index of the kernel</param>
/// <param name="argIndex">Index of the argument</param>
/// <param name="bufferIndex">Index of the buffer</param>
/// <returns>True if success, else false.</returns>
2014-12-06 00:05:09 -05:00
bool OpenCLWrapper : : SetBufferArg ( uint kernelIndex , uint argIndex , uint bufferIndex )
2014-07-08 03:11:14 -04:00
{
if ( m_Init & & bufferIndex < m_Buffers . size ( ) )
return SetArg < cl : : Buffer > ( kernelIndex , argIndex , m_Buffers [ bufferIndex ] . m_Buffer ) ;
return false ;
}
/// <summary>
/// Set the argument at the specified index for the kernel at the specified index to be
/// the 2D image with the specified name.
/// </summary>
/// <param name="kernelIndex">Index of the kernel</param>
/// <param name="argIndex">Index of the argument</param>
/// <param name="shared">True if shared with an OpenGL texture, else false</param>
/// <param name="name">The name of the 2D image</param>
/// <returns>True if success, else false.</returns>
2014-12-06 00:05:09 -05:00
bool OpenCLWrapper : : SetImageArg ( uint kernelIndex , uint argIndex , bool shared , const string & name )
2014-07-08 03:11:14 -04:00
{
if ( m_Init )
{
int imageIndex = FindImageIndex ( name , shared ) ;
return SetImageArg ( kernelIndex , argIndex , shared , imageIndex ) ;
}
return false ;
}
/// <summary>
/// Set the argument at the specified index for the kernel at the specified index to be
/// the 2D image at the specified index.
/// </summary>
/// <param name="kernelIndex">Index of the kernel</param>
/// <param name="argIndex">Index of the argument</param>
/// <param name="shared">True if shared with an OpenGL texture, else false</param>
/// <param name="imageIndex">Index of the 2D image</param>
/// <returns>True if success, else false.</returns>
2014-12-06 00:05:09 -05:00
bool OpenCLWrapper : : SetImageArg ( uint kernelIndex , uint argIndex , bool shared , uint imageIndex )
2014-07-08 03:11:14 -04:00
{
cl_int err ;
if ( m_Init )
{
if ( shared & & imageIndex < m_GLImages . size ( ) )
{
err = m_Programs [ kernelIndex ] . m_Kernel . setArg ( argIndex , m_GLImages [ imageIndex ] . m_Image ) ;
return CheckCL ( err , " cl::Kernel::setArg() " ) ;
}
else if ( ! shared & & imageIndex < m_Images . size ( ) )
{
err = m_Programs [ kernelIndex ] . m_Kernel . setArg ( argIndex , m_Images [ imageIndex ] . m_Image ) ;
return CheckCL ( err , " cl::Kernel::setArg() " ) ;
}
}
return false ;
}
/// <summary>
/// Find the index of the kernel with the specified name.
/// </summary>
/// <param name="name">The name of the kernel to search for</param>
/// <returns>The index if found, else -1.</returns>
2014-10-14 11:53:15 -04:00
int OpenCLWrapper : : FindKernelIndex ( const string & name )
2014-07-08 03:11:14 -04:00
{
2014-12-07 02:51:44 -05:00
for ( size_t i = 0 ; i < m_Programs . size ( ) ; i + + )
2014-07-08 03:11:14 -04:00
if ( m_Programs [ i ] . m_Name = = name )
2014-12-07 02:51:44 -05:00
return int ( i ) ;
2014-07-08 03:11:14 -04:00
return - 1 ;
}
/// <summary>
/// Run the kernel at the specified index, using the specified grid and block dimensions.
/// </summary>
/// <param name="kernelIndex">Index of the kernel to run</param>
/// <param name="totalGridWidth">Total width of the grid</param>
/// <param name="totalGridHeight">Total height of the grid</param>
/// <param name="totalGridDepth">The total depth grid</param>
/// <param name="blockWidth">Width of each block</param>
/// <param name="blockHeight">Height of each block</param>
/// <param name="blockDepth">Depth of each block</param>
/// <returns>True if success, else false.</returns>
2014-12-06 00:05:09 -05:00
bool OpenCLWrapper : : RunKernel ( uint kernelIndex , uint totalGridWidth , uint totalGridHeight , uint totalGridDepth ,
uint blockWidth , uint blockHeight , uint blockDepth )
2014-07-08 03:11:14 -04:00
{
if ( m_Init & & kernelIndex < m_Programs . size ( ) )
{
cl : : Event e ;
cl_int err = m_Queue . enqueueNDRangeKernel ( m_Programs [ kernelIndex ] . m_Kernel ,
cl : : NullRange ,
cl : : NDRange ( totalGridWidth , totalGridHeight , totalGridDepth ) ,
cl : : NDRange ( blockWidth , blockHeight , blockDepth ) ,
2014-12-05 21:30:46 -05:00
nullptr ,
2014-07-08 03:11:14 -04:00
& e ) ;
e . wait ( ) ;
m_Queue . finish ( ) ;
return CheckCL ( err , " cl::CommandQueue::enqueueNDRangeKernel() " ) ;
}
return false ;
}
/// <summary>
/// Get device information for the specified field.
/// Template argument expected to be cl_ulong, cl_uint or cl_int;
/// </summary>
/// <param name="name">The device field/feature to query</param>
/// <returns>The value of the field</returns>
template < typename T >
2014-12-09 03:24:28 -05:00
T OpenCLWrapper : : GetInfo ( size_t platform , size_t device , cl_device_info name ) const
2014-07-08 03:11:14 -04:00
{
T val ;
if ( platform < m_Devices . size ( ) & & device < m_Devices [ platform ] . size ( ) )
m_Devices [ platform ] [ device ] . getInfo ( name , & val ) ;
return val ;
}
/// <summary>
/// Get the platform name at the specified index.
/// </summary>
/// <param name="i">The platform index to get the name of</param>
/// <returns>The platform name if found, else empty string</returns>
string OpenCLWrapper : : PlatformName ( size_t platform )
{
if ( platform < m_Platforms . size ( ) )
2014-12-05 21:30:46 -05:00
return m_Platforms [ platform ] . getInfo < CL_PLATFORM_VENDOR > ( nullptr ) + " " + m_Platforms [ platform ] . getInfo < CL_PLATFORM_NAME > ( nullptr ) + " " + m_Platforms [ platform ] . getInfo < CL_PLATFORM_VERSION > ( nullptr ) ;
2014-07-08 03:11:14 -04:00
else
return " " ;
}
/// <summary>
/// Get all available platform names on the system as a vector of strings.
/// </summary>
/// <returns>All available platform names on the system as a vector of strings</returns>
vector < string > OpenCLWrapper : : PlatformNames ( )
{
vector < string > platforms ;
platforms . reserve ( m_Platforms . size ( ) ) ;
2014-12-07 02:51:44 -05:00
for ( size_t i = 0 ; i < m_Platforms . size ( ) ; i + + )
2014-07-08 03:11:14 -04:00
platforms . push_back ( PlatformName ( i ) ) ;
return platforms ;
}
/// <summary>
/// Get the device name at the specified index on the platform
/// at the specified index.
/// </summary>
/// <param name="platform">The platform index of the device</param>
/// <param name="device">The device index</param>
/// <returns>The name of the device if found, else empty string</returns>
string OpenCLWrapper : : DeviceName ( size_t platform , size_t device )
{
string s ;
if ( platform < m_Platforms . size ( ) & & platform < m_Devices . size ( ) )
if ( device < m_Devices [ platform ] . size ( ) )
2014-12-05 21:30:46 -05:00
s = m_Devices [ platform ] [ device ] . getInfo < CL_DEVICE_VENDOR > ( nullptr ) + " " + m_Devices [ platform ] [ device ] . getInfo < CL_DEVICE_NAME > ( nullptr ) ; // + " " + m_Devices[platform][device].getInfo<CL_DEVICE_VERSION>();
2014-07-08 03:11:14 -04:00
return s ;
}
/// <summary>
/// Get all available device names on the platform at the specified index as a vector of strings.
/// </summary>
/// <param name="platform">The platform index of the devices to query</param>
/// <returns>All available device names on the platform at the specified index as a vector of strings</returns>
vector < string > OpenCLWrapper : : DeviceNames ( size_t platform )
{
2014-12-06 00:05:09 -05:00
uint i = 0 ;
2014-07-08 03:11:14 -04:00
string s ;
vector < string > devices ;
do
{
s = DeviceName ( platform , i ) ;
if ( s ! = " " )
devices . push_back ( s ) ;
i + + ;
} while ( s ! = " " ) ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
return devices ;
}
/// <summary>
/// Get all availabe device and platform names as one contiguous string.
/// </summary>
/// <returns>A string with all available device and platform names</returns>
string OpenCLWrapper : : DeviceAndPlatformNames ( )
{
ostringstream os ;
vector < string > deviceNames ;
for ( size_t platform = 0 ; platform < m_Platforms . size ( ) ; platform + + )
{
os < < PlatformName ( platform ) < < endl ;
deviceNames = DeviceNames ( platform ) ;
for ( size_t device = 0 ; device < m_Devices [ platform ] . size ( ) ; device + + )
os < < " \t " < < deviceNames [ device ] < < endl ;
}
return os . str ( ) ;
}
/// <summary>
/// Get all information about the currently used device.
/// </summary>
/// <returns>A string with all information about the currently used device</returns>
string OpenCLWrapper : : DumpInfo ( )
{
ostringstream os ;
vector < size_t > sizes ;
os . imbue ( std : : locale ( " " ) ) ;
for ( size_t platform = 0 ; platform < m_Platforms . size ( ) ; platform + + )
{
os < < " Platform " < < platform < < " : " < < PlatformName ( platform ) < < endl ;
for ( size_t device = 0 ; device < m_Devices [ platform ] . size ( ) ; device + + )
{
os < < " Device " < < device < < " : " < < DeviceName ( platform , device ) < < endl ;
os < < " CL_DEVICE_OPENCL_C_VERSION: " < < GetInfo < string > ( platform , device , CL_DEVICE_OPENCL_C_VERSION ) < < endl ;
os < < " CL_DEVICE_LOCAL_MEM_SIZE: " < < GetInfo < cl_ulong > ( platform , device , CL_DEVICE_LOCAL_MEM_SIZE ) < < endl ;
os < < " CL_DEVICE_LOCAL_MEM_TYPE: " < < GetInfo < cl_uint > ( platform , device , CL_DEVICE_LOCAL_MEM_TYPE ) < < endl ;
os < < " CL_DEVICE_MAX_COMPUTE_UNITS: " < < GetInfo < cl_uint > ( platform , device , CL_DEVICE_MAX_COMPUTE_UNITS ) < < endl ;
os < < " CL_DEVICE_MAX_READ_IMAGE_ARGS: " < < GetInfo < cl_uint > ( platform , device , CL_DEVICE_MAX_READ_IMAGE_ARGS ) < < endl ;
os < < " CL_DEVICE_MAX_WRITE_IMAGE_ARGS: " < < GetInfo < cl_uint > ( platform , device , CL_DEVICE_MAX_WRITE_IMAGE_ARGS ) < < endl ;
os < < " CL_DEVICE_MAX_MEM_ALLOC_SIZE: " < < GetInfo < cl_ulong > ( platform , device , CL_DEVICE_MAX_MEM_ALLOC_SIZE ) < < endl ;
2014-12-09 03:24:28 -05:00
os < < " CL_DEVICE_ADDRESS_BITS: " < < GetInfo < cl_uint > ( platform , device , CL_DEVICE_ADDRESS_BITS ) < < endl ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
os < < " CL_DEVICE_GLOBAL_MEM_CACHE_TYPE: " < < GetInfo < cl_uint > ( platform , device , CL_DEVICE_GLOBAL_MEM_CACHE_TYPE ) < < endl ;
os < < " CL_DEVICE_GLOBAL_MEM_CACHELINE_SIZE: " < < GetInfo < cl_uint > ( platform , device , CL_DEVICE_GLOBAL_MEM_CACHELINE_SIZE ) < < endl ;
os < < " CL_DEVICE_GLOBAL_MEM_CACHE_SIZE: " < < GetInfo < cl_ulong > ( platform , device , CL_DEVICE_GLOBAL_MEM_CACHE_SIZE ) < < endl ;
os < < " CL_DEVICE_GLOBAL_MEM_SIZE: " < < GetInfo < cl_ulong > ( platform , device , CL_DEVICE_GLOBAL_MEM_SIZE ) < < endl ;
os < < " CL_DEVICE_MAX_CONSTANT_BUFFER_SIZE: " < < GetInfo < cl_ulong > ( platform , device , CL_DEVICE_MAX_CONSTANT_BUFFER_SIZE ) < < endl ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
os < < " CL_DEVICE_MAX_CONSTANT_ARGS: " < < GetInfo < cl_uint > ( platform , device , CL_DEVICE_MAX_CONSTANT_ARGS ) < < endl ;
os < < " CL_DEVICE_MAX_WORK_ITEM_DIMENSIONS: " < < GetInfo < cl_uint > ( platform , device , CL_DEVICE_MAX_WORK_ITEM_DIMENSIONS ) < < endl ;
os < < " CL_DEVICE_MAX_WORK_GROUP_SIZE: " < < GetInfo < : : size_t > ( platform , device , CL_DEVICE_MAX_WORK_GROUP_SIZE ) < < endl ;
sizes = GetInfo < vector < : : size_t > > ( platform , device , CL_DEVICE_MAX_WORK_ITEM_SIZES ) ;
os < < " CL_DEVICE_MAX_WORK_ITEM_SIZES: " < < sizes [ 0 ] < < " , " < < sizes [ 1 ] < < " , " < < sizes [ 2 ] < < endl < < endl ;
if ( device ! = m_Devices [ platform ] . size ( ) - 1 & & platform ! = m_Platforms . size ( ) - 1 )
os < < endl ;
}
os < < endl ;
}
return os . str ( ) ;
}
/// <summary>
/// OpenCL properties, getters only.
/// </summary>
2014-11-28 04:37:51 -05:00
bool OpenCLWrapper : : Ok ( ) const { return m_Init ; }
bool OpenCLWrapper : : Shared ( ) const { return m_Shared ; }
cl : : Context OpenCLWrapper : : Context ( ) const { return m_Context ; }
2014-12-06 00:05:09 -05:00
uint OpenCLWrapper : : PlatformIndex ( ) const { return m_PlatformIndex ; }
uint OpenCLWrapper : : DeviceIndex ( ) const { return m_DeviceIndex ; }
2014-12-09 03:24:28 -05:00
size_t OpenCLWrapper : : GlobalMemSize ( ) const { return GetInfo < cl_ulong > ( PlatformIndex ( ) , DeviceIndex ( ) , CL_DEVICE_GLOBAL_MEM_SIZE ) ; }
2014-12-06 00:05:09 -05:00
uint OpenCLWrapper : : LocalMemSize ( ) const { return m_LocalMemSize ; }
2014-12-09 03:24:28 -05:00
size_t OpenCLWrapper : : MaxAllocSize ( ) const { return GetInfo < cl_ulong > ( PlatformIndex ( ) , DeviceIndex ( ) , CL_DEVICE_MAX_MEM_ALLOC_SIZE ) ; }
2014-07-08 03:11:14 -04:00
/// <summary>
/// Makes the even grid dims.
/// </summary>
/// <param name="blockW">The block w.</param>
/// <param name="blockH">The block h.</param>
/// <param name="gridW">The grid w.</param>
/// <param name="gridH">The grid h.</param>
2014-12-06 00:05:09 -05:00
void OpenCLWrapper : : MakeEvenGridDims ( uint blockW , uint blockH , uint & gridW , uint & gridH )
2014-07-08 03:11:14 -04:00
{
if ( gridW % blockW ! = 0 )
gridW + = ( blockW - ( gridW % blockW ) ) ;
if ( gridH % blockH ! = 0 )
gridH + = ( blockH - ( gridH % blockH ) ) ;
}
/// <summary>
/// Create a context that is optionall shared with OpenGL.
/// </summary>
/// <param name="shared">True if shared with OpenGL, else not shared.</param>
/// <returns>True if success, else false.</returns>
bool OpenCLWrapper : : CreateContext ( bool shared )
{
cl_int err ;
if ( shared )
{
//Define OS-specific context properties and create the OpenCL context.
# if defined (__APPLE__) || defined(MACOSX)
CGLContextObj kCGLContext = CGLGetCurrentContext ( ) ;
CGLShareGroupObj kCGLShareGroup = CGLGetShareGroup ( kCGLContext ) ;
cl_context_properties props [ ] =
{
CL_CONTEXT_PROPERTY_USE_CGL_SHAREGROUP_APPLE , ( cl_context_properties ) kCGLShareGroup ,
0
} ;
2014-12-05 21:30:46 -05:00
m_Context = cl : : Context ( CL_DEVICE_TYPE_GPU , props , nullptr , nullptr , & err ) ; //May need to tinker with this on Mac.
2014-07-08 03:11:14 -04:00
# else
# if defined WIN32
cl_context_properties props [ ] =
{
CL_GL_CONTEXT_KHR , ( cl_context_properties ) wglGetCurrentContext ( ) ,
CL_WGL_HDC_KHR , ( cl_context_properties ) wglGetCurrentDC ( ) ,
2014-12-07 02:51:44 -05:00
CL_CONTEXT_PLATFORM , reinterpret_cast < cl_context_properties > ( ( m_Platforms [ m_PlatformIndex ] ) ( ) ) ,
2014-07-08 03:11:14 -04:00
0
} ;
2014-12-05 21:30:46 -05:00
m_Context = cl : : Context ( CL_DEVICE_TYPE_GPU , props , nullptr , nullptr , & err ) ;
2014-07-08 03:11:14 -04:00
# else
cl_context_properties props [ ] =
{
2014-12-07 02:51:44 -05:00
CL_GL_CONTEXT_KHR , cl_context_properties ( glXGetCurrentContext ( ) ) ,
CL_GLX_DISPLAY_KHR , cl_context_properties ( glXGetCurrentDisplay ( ) ) ,
CL_CONTEXT_PLATFORM , reinterpret_cast < cl_context_properties > ( ( m_Platforms [ m_PlatformIndex ] ) ( ) ) ,
2014-07-08 03:11:14 -04:00
0
} ;
2014-12-05 21:30:46 -05:00
m_Context = cl : : Context ( CL_DEVICE_TYPE_GPU , props , nullptr , nullptr , & err ) ;
2014-07-08 03:11:14 -04:00
# endif
# endif
}
else
{
cl_context_properties props [ 3 ] =
{
CL_CONTEXT_PLATFORM ,
2014-12-07 02:51:44 -05:00
reinterpret_cast < cl_context_properties > ( ( m_Platforms [ m_PlatformIndex ] ) ( ) ) ,
2014-07-08 03:11:14 -04:00
0
} ;
2014-12-05 21:30:46 -05:00
m_Context = cl : : Context ( CL_DEVICE_TYPE_ALL , props , nullptr , nullptr , & err ) ;
2014-07-08 03:11:14 -04:00
}
return CheckCL ( err , " cl::Context() " ) ;
}
/// <summary>
/// Create an Spk object created by compiling the program arguments passed in.
/// </summary>
/// <param name="name">The name of the program</param>
/// <param name="program">The source of the program</param>
/// <param name="entryPoint">The name of the entry point kernel function in the program</param>
/// <param name="spk">The Spk object to store the resulting compiled program in</param>
/// <returns>True if success, else false.</returns>
2014-12-05 21:30:46 -05:00
bool OpenCLWrapper : : CreateSPK ( const string & name , const string & program , const string & entryPoint , Spk & spk , bool doublePrecision )
2014-07-08 03:11:14 -04:00
{
if ( m_Init )
{
cl_int err ;
spk . m_Name = name ;
2015-02-26 16:18:55 -05:00
spk . m_Source = cl : : Program : : Sources ( 1 , std : : make_pair ( program . c_str ( ) , program . length ( ) + 1 ) ) ;
2014-07-08 03:11:14 -04:00
spk . m_Program = cl : : Program ( m_Context , spk . m_Source ) ;
if ( doublePrecision )
err = spk . m_Program . build ( m_DeviceVec , " -cl-mad-enable " ) ; //Tinker with other options later.
else
err = spk . m_Program . build ( m_DeviceVec , " -cl-mad-enable -cl-no-signed-zeros -cl-single-precision-constant " ) ;
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
//err = spk.m_Program.build(m_DeviceVec, "-cl-single-precision-constant");
//err = spk.m_Program.build(m_DeviceVec, "-cl-mad-enable -cl-single-precision-constant");
2014-07-08 03:11:14 -04:00
//err = spk.m_Program.build(m_DeviceVec, "-cl-mad-enable -cl-no-signed-zeros -cl-fast-relaxed-math -cl-single-precision-constant");//This can cause some rounding.
//err = spk.m_Program.build(m_DeviceVec, "-cl-mad-enable -cl-single-precision-constant");
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
if ( CheckCL ( err , " cl::Program::build() " ) )
{
//Building of program is ok, now create kernel with the specified entry point.
spk . m_Kernel = cl : : Kernel ( spk . m_Program , entryPoint . c_str ( ) , & err ) ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
if ( CheckCL ( err , " cl::Kernel() " ) )
return true ; //Everything is ok.
2015-03-22 15:46:10 -04:00
}
else
{
for ( auto & i : m_DeviceVec )
m_ErrorReport . push_back ( spk . m_Program . getBuildInfo < CL_PROGRAM_BUILD_LOG > ( i ) ) ;
2015-02-25 08:20:03 -05:00
}
2014-07-08 03:11:14 -04:00
}
return false ;
}
/// <summary>
/// Check an OpenCL return value for errors.
/// </summary>
/// <param name="err">The error code to inspect</param>
/// <param name="name">A description of where the value was gotten from</param>
/// <returns>True if success, else false.</returns>
bool OpenCLWrapper : : CheckCL ( cl_int err , const char * name )
{
if ( err ! = CL_SUCCESS )
{
ostringstream ss ;
ss < < " ERROR: " < < ErrorToStringCL ( err ) < < " in " < < name < < " . " < < std : : endl ;
m_ErrorReport . push_back ( ss . str ( ) ) ;
}
return err = = CL_SUCCESS ;
}
/// <summary>
/// Translate an OpenCL error code into a human readable string.
/// </summary>
/// <param name="err">The error code to translate</param>
/// <returns>A human readable description of the error passed in</returns>
std : : string OpenCLWrapper : : ErrorToStringCL ( cl_int err )
{
switch ( err )
{
case CL_SUCCESS : return " Success " ;
case CL_DEVICE_NOT_FOUND : return " Device not found " ;
case CL_DEVICE_NOT_AVAILABLE : return " Device not available " ;
case CL_COMPILER_NOT_AVAILABLE : return " Compiler not available " ;
case CL_MEM_OBJECT_ALLOCATION_FAILURE : return " Memory object allocation failure " ;
case CL_OUT_OF_RESOURCES : return " Out of resources " ;
case CL_OUT_OF_HOST_MEMORY : return " Out of host memory " ;
case CL_PROFILING_INFO_NOT_AVAILABLE : return " Profiling information not available " ;
case CL_MEM_COPY_OVERLAP : return " Memory copy overlap " ;
case CL_IMAGE_FORMAT_MISMATCH : return " Image format mismatch " ;
case CL_IMAGE_FORMAT_NOT_SUPPORTED : return " Image format not supported " ;
case CL_BUILD_PROGRAM_FAILURE : return " Program build failure " ;
case CL_MAP_FAILURE : return " Map failure " ;
case CL_MISALIGNED_SUB_BUFFER_OFFSET : return " Misaligned sub buffer offset " ;
case CL_EXEC_STATUS_ERROR_FOR_EVENTS_IN_WAIT_LIST : return " Exec status error for events in wait list " ;
case CL_INVALID_VALUE : return " Invalid value " ;
case CL_INVALID_DEVICE_TYPE : return " Invalid device type " ;
case CL_INVALID_PLATFORM : return " Invalid platform " ;
case CL_INVALID_DEVICE : return " Invalid device " ;
case CL_INVALID_CONTEXT : return " Invalid context " ;
case CL_INVALID_QUEUE_PROPERTIES : return " Invalid queue properties " ;
case CL_INVALID_COMMAND_QUEUE : return " Invalid command queue " ;
case CL_INVALID_HOST_PTR : return " Invalid host pointer " ;
case CL_INVALID_MEM_OBJECT : return " Invalid memory object " ;
case CL_INVALID_IMAGE_FORMAT_DESCRIPTOR : return " Invalid image format descriptor " ;
case CL_INVALID_IMAGE_SIZE : return " Invalid image size " ;
case CL_INVALID_SAMPLER : return " Invalid sampler " ;
case CL_INVALID_BINARY : return " Invalid binary " ;
case CL_INVALID_BUILD_OPTIONS : return " Invalid build options " ;
case CL_INVALID_PROGRAM : return " Invalid program " ;
case CL_INVALID_PROGRAM_EXECUTABLE : return " Invalid program executable " ;
case CL_INVALID_KERNEL_NAME : return " Invalid kernel name " ;
case CL_INVALID_KERNEL_DEFINITION : return " Invalid kernel definition " ;
case CL_INVALID_KERNEL : return " Invalid kernel " ;
case CL_INVALID_ARG_INDEX : return " Invalid argument index " ;
case CL_INVALID_ARG_VALUE : return " Invalid argument value " ;
case CL_INVALID_ARG_SIZE : return " Invalid argument size " ;
case CL_INVALID_KERNEL_ARGS : return " Invalid kernel arguments " ;
case CL_INVALID_WORK_DIMENSION : return " Invalid work dimension " ;
case CL_INVALID_WORK_GROUP_SIZE : return " Invalid work group size " ;
case CL_INVALID_WORK_ITEM_SIZE : return " Invalid work item size " ;
case CL_INVALID_GLOBAL_OFFSET : return " Invalid global offset " ;
case CL_INVALID_EVENT_WAIT_LIST : return " Invalid event wait list " ;
case CL_INVALID_EVENT : return " Invalid event " ;
case CL_INVALID_OPERATION : return " Invalid operation " ;
case CL_INVALID_GL_OBJECT : return " Invalid OpenGL object " ;
case CL_INVALID_BUFFER_SIZE : return " Invalid buffer size " ;
case CL_INVALID_MIP_LEVEL : return " Invalid mip-map level " ;
case CL_INVALID_GLOBAL_WORK_SIZE : return " Invalid global work size " ;
case CL_INVALID_PROPERTY : return " Invalid property " ;
default :
{
ostringstream ss ;
ss < < " <Unknown error code> " < < err ;
return ss . str ( ) ;
}
}
}
2014-12-05 21:30:46 -05:00
}