opencl_ruby_ffi 0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/LICENSE +22 -0
- data/lib/opencl_ruby_ffi.rb +18 -0
- data/lib/opencl_ruby_ffi/Arithmetic_gen.rb +3566 -0
- data/lib/opencl_ruby_ffi/Buffer.rb +84 -0
- data/lib/opencl_ruby_ffi/CommandQueue.rb +1544 -0
- data/lib/opencl_ruby_ffi/Context.rb +326 -0
- data/lib/opencl_ruby_ffi/Device.rb +188 -0
- data/lib/opencl_ruby_ffi/Event.rb +152 -0
- data/lib/opencl_ruby_ffi/Image.rb +270 -0
- data/lib/opencl_ruby_ffi/Kernel.rb +183 -0
- data/lib/opencl_ruby_ffi/Mem.rb +131 -0
- data/lib/opencl_ruby_ffi/Platform.rb +119 -0
- data/lib/opencl_ruby_ffi/Program.rb +245 -0
- data/lib/opencl_ruby_ffi/Sampler.rb +51 -0
- data/lib/opencl_ruby_ffi/opencl_ruby_ffi_base.rb +320 -0
- data/lib/opencl_ruby_ffi/opencl_ruby_ffi_base_gen.rb +1826 -0
- data/opencl_ruby_ffi.gemspec +16 -0
- metadata +110 -0
@@ -0,0 +1,131 @@
|
|
1
|
+
module OpenCL
|
2
|
+
|
3
|
+
# Attaches a callback to memobj the will be called on memobj destruction
|
4
|
+
#
|
5
|
+
# ==== Attributes
|
6
|
+
#
|
7
|
+
# * +memobj+ - the Mem to attach the callback to
|
8
|
+
# * +options+ - a hash containing named options
|
9
|
+
# * +block+ - if provided, a callback invoked when memobj is released. Signature of the callback is { |Mem, FFI::Pointer to user_data| ... }
|
10
|
+
#
|
11
|
+
# ==== Options
|
12
|
+
#
|
13
|
+
# * +:user_data+ - a Pointer (or convertible to Pointer using to_ptr) to the memory area to pass to the callback
|
14
|
+
def self.set_mem_object_destructor_callback( memobj, options = {}, &proc )
|
15
|
+
@@callbacks.push( block ) if block
|
16
|
+
error = OpenCL.clSetMemObjectDestructorCallback( memobj, block, options[:user_data] )
|
17
|
+
OpenCL.error_check(error)
|
18
|
+
return self
|
19
|
+
end
|
20
|
+
|
21
|
+
# Maps the cl_mem object of OpenCL
|
22
|
+
class Mem
|
23
|
+
|
24
|
+
# Returns the Context associated to the Mem
|
25
|
+
def context
|
26
|
+
ptr = FFI::MemoryPointer.new( Context )
|
27
|
+
error = OpenCL.clGetMemObjectInfo(self, Mem::CONTEXT, Context.size, ptr, nil)
|
28
|
+
OpenCL.error_check(error)
|
29
|
+
return OpenCL::Context::new( ptr.read_pointer )
|
30
|
+
end
|
31
|
+
|
32
|
+
# Returns the Platform associated to the Mem
|
33
|
+
def platform
|
34
|
+
return self.context.platform
|
35
|
+
end
|
36
|
+
|
37
|
+
# Returns the Buffer this Buffer was created from using create_sub_buffer
|
38
|
+
def associated_memobject
|
39
|
+
ptr = FFI::MemoryPointer.new( Mem )
|
40
|
+
error = OpenCL.clGetMemObjectInfo(self, Mem::ASSOCIATED_MEMOBJECT, Mem.size, ptr, nil)
|
41
|
+
OpenCL.error_check(error)
|
42
|
+
return nil if ptr.read_pointer.null?
|
43
|
+
return OpenCL::Mem::new( ptr.read_pointer )
|
44
|
+
end
|
45
|
+
|
46
|
+
##
|
47
|
+
# :method: offset()
|
48
|
+
# Returns the offset used to create this Buffer using create_sub_buffer
|
49
|
+
|
50
|
+
##
|
51
|
+
# :method: size()
|
52
|
+
# Returns the size of the Buffer
|
53
|
+
%w( OFFSET SIZE ).each { |prop|
|
54
|
+
eval OpenCL.get_info("Mem", :size_t, prop)
|
55
|
+
}
|
56
|
+
|
57
|
+
##
|
58
|
+
# :method: map_count()
|
59
|
+
# Returns the number of times this Mem is mapped
|
60
|
+
|
61
|
+
##
|
62
|
+
# :method: reference_count()
|
63
|
+
# Returns the Mem reference counter
|
64
|
+
%w( MAP_COUNT REFERENCE_COUNT ).each { |prop|
|
65
|
+
eval OpenCL.get_info("Mem", :cl_uint, prop)
|
66
|
+
}
|
67
|
+
|
68
|
+
##
|
69
|
+
# :method: type()
|
70
|
+
# Returns an OpenCL::Mem::Type corresponding to the Mem
|
71
|
+
eval OpenCL.get_info("Mem", :cl_mem_object_type, "TYPE")
|
72
|
+
|
73
|
+
##
|
74
|
+
# :method: flags()
|
75
|
+
# Returns an OpenCL::Mem::Flags corresponding to the flags used at Mem creation
|
76
|
+
eval OpenCL.get_info("Mem", :cl_mem_flags, "FLAGS")
|
77
|
+
|
78
|
+
##
|
79
|
+
# :method: host_ptr()
|
80
|
+
# Returns the host Pointer specified at Mem creation or the pointer + the ofsset if it is a sub-buffer. A null Pointer is returned otherwise.
|
81
|
+
eval OpenCL.get_info("Mem", :pointer, "HOST_PTR")
|
82
|
+
|
83
|
+
# Attaches a callback to memobj that will be called on the memobj destruction
|
84
|
+
#
|
85
|
+
# ==== Attributes
|
86
|
+
#
|
87
|
+
# * +options+ - a hash containing named options
|
88
|
+
# * +block+ - if provided, a callback invoked when memobj is released. Signature of the callback is { |Mem, FFI::Pointer to user_data| ... }
|
89
|
+
#
|
90
|
+
# ==== Options
|
91
|
+
#
|
92
|
+
# * +:user_data+ - a Pointer (or convertible to Pointer using to_ptr) to the memory area to pass to the callback
|
93
|
+
def set_destructor_callback( options = {}, &proc )
|
94
|
+
return OpenCL.set_mem_object_destructor_callback( self, options, &proc )
|
95
|
+
end
|
96
|
+
|
97
|
+
# Returns the texture_target argument specified in create_from_GL_texture for Mem
|
98
|
+
def GL_texture_target
|
99
|
+
param_value = MemoryPointer.new( :cl_GLenum )
|
100
|
+
error = OpenCL.clGetGLTextureInfo( self, OpenCL::GL_TEXTURE_TARGET, param_value.size, param_value, nil )
|
101
|
+
OpenCL.error_check(error)
|
102
|
+
return param_value.read_cl_GLenum
|
103
|
+
end
|
104
|
+
|
105
|
+
# Returns the miplevel argument specified in create_from_GL_texture for Mem
|
106
|
+
def GL_mimap_level
|
107
|
+
param_value = MemoryPointer.new( :cl_GLint )
|
108
|
+
error = OpenCL.clGetGLTextureInfo( self, OpenCL::GL_MIPMAP_LEVEL, param_value.size, param_value, nil )
|
109
|
+
OpenCL.error_check(error)
|
110
|
+
return param_value.read_cl_GLint
|
111
|
+
end
|
112
|
+
|
113
|
+
# Returns the type of the GL object associated with Mem
|
114
|
+
def GL_object_type
|
115
|
+
param_value = MemoryPointer.new( :cl_gl_object_type )
|
116
|
+
error = OpenCL.clGetGLObjectInfo( self, param_value, nil )
|
117
|
+
OpenCL.error_check(error)
|
118
|
+
return OpenCL::GLObjectType(param_value.read_cl_gl_object_type)
|
119
|
+
end
|
120
|
+
|
121
|
+
# Returns the name of the GL object associated with Mem
|
122
|
+
def GL_object_name
|
123
|
+
param_value = MemoryPointer.new( :cl_GLuint )
|
124
|
+
error = OpenCL.clGetGLObjectInfo( self, nil, param_value )
|
125
|
+
OpenCL.error_check(error)
|
126
|
+
return param_value.read_cl_GLuint
|
127
|
+
end
|
128
|
+
|
129
|
+
end
|
130
|
+
|
131
|
+
end
|
@@ -0,0 +1,119 @@
|
|
1
|
+
module OpenCL
|
2
|
+
|
3
|
+
# Returns an FFI::Function corresponding to an extension function
|
4
|
+
#
|
5
|
+
# ==== Attributes
|
6
|
+
#
|
7
|
+
# * +name+ - a String representing the name of the function
|
8
|
+
# * +return_type+ - the type of data returned by the function
|
9
|
+
# * +param_types+ - an Array of types, corresponding to the parameters type
|
10
|
+
# * +options+ - if given, a hash of named options that will be given to FFI::Function::new. See FFI doc for details.
|
11
|
+
def self.get_extension_function( name, return_type, param_types, options = {} )
|
12
|
+
name_p = FFI::MemoryPointer.from_string(name)
|
13
|
+
ptr = OpenCL.clGetExtensionFunctionAddress( name_p )
|
14
|
+
return nil if ptr.null?
|
15
|
+
return FFI::Function::new(return_type, param_types, ptr, options)
|
16
|
+
end
|
17
|
+
|
18
|
+
# Returns an Array of Platforms containing the available OpenCL platforms
|
19
|
+
def self.get_platforms
|
20
|
+
ptr1 = FFI::MemoryPointer.new(:cl_uint , 1)
|
21
|
+
|
22
|
+
error = OpenCL::clGetPlatformIDs(0, nil, ptr1)
|
23
|
+
OpenCL.error_check(error)
|
24
|
+
ptr2 = FFI::MemoryPointer.new(:pointer, ptr1.read_uint)
|
25
|
+
error = OpenCL::clGetPlatformIDs(ptr1.read_uint(), ptr2, nil)
|
26
|
+
OpenCL.error_check(error)
|
27
|
+
return ptr2.get_array_of_pointer(0,ptr1.read_uint()).collect { |platform_ptr|
|
28
|
+
OpenCL::Platform.new(platform_ptr)
|
29
|
+
}
|
30
|
+
end
|
31
|
+
|
32
|
+
# Returns an FFI::Function corresponding to an extension function for the Platform
|
33
|
+
#
|
34
|
+
# ==== Attributes
|
35
|
+
#
|
36
|
+
# * +platform+ - the Platform to be queried
|
37
|
+
# * +name+ - a String representing the name of the function
|
38
|
+
# * +return_type+ - the type of data returned by the function
|
39
|
+
# * +param_types+ - an Array of types, corresponding to the parameters type
|
40
|
+
# * +options+ - if given, a hash of named options that will be given to FFI::Function::new. See FFI doc for details.
|
41
|
+
def get_extension_function_for_platform( platform, name, return_type, param_types, options = {} )
|
42
|
+
OpenCL.error_check(OpenCL::INVALID_OPERATION) if self.version_number < 1.2
|
43
|
+
name_p = FFI::MemoryPointer.from_string(name)
|
44
|
+
ptr = OpenCL.clGetExtensionFunctionAddressForPlatform( platform, name_p )
|
45
|
+
return nil if ptr.null?
|
46
|
+
return FFI::Function::new(return_type, param_types, ptr, options)
|
47
|
+
end
|
48
|
+
|
49
|
+
|
50
|
+
# Maps the cl_platform_id object of OpenCL
|
51
|
+
class Platform
|
52
|
+
%w(PROFILE VERSION NAME VENDOR EXTENSIONS).each { |prop|
|
53
|
+
eval OpenCL.get_info("Platform", :string, prop)
|
54
|
+
}
|
55
|
+
|
56
|
+
# Returns an Array of Device corresponding to the available devices on the Platform
|
57
|
+
# The type of the desired devices can be specified
|
58
|
+
def devices(type = OpenCL::Device::Type::ALL)
|
59
|
+
ptr1 = FFI::MemoryPointer.new(:cl_uint , 1)
|
60
|
+
error = OpenCL::clGetDeviceIDs(self, type, 0, nil, ptr1)
|
61
|
+
OpenCL.error_check(error)
|
62
|
+
ptr2 = FFI::MemoryPointer.new(:pointer, ptr1.read_uint)
|
63
|
+
error = OpenCL::clGetDeviceIDs(self, type, ptr1.read_uint(), ptr2, nil)
|
64
|
+
OpenCL.error_check(error)
|
65
|
+
return ptr2.get_array_of_pointer(0, ptr1.read_uint()).collect { |device_ptr|
|
66
|
+
OpenCL::Device.new(device_ptr)
|
67
|
+
}
|
68
|
+
end
|
69
|
+
|
70
|
+
# returs a floating point number corresponding to the OpenCL version of the Platform
|
71
|
+
def version_number
|
72
|
+
ver = self.version
|
73
|
+
n = ver.scan(/OpenCL (\d+\.\d+)/)
|
74
|
+
return n.first.first.to_f
|
75
|
+
end
|
76
|
+
|
77
|
+
# Returns an FFI::Function corresponding to an extension function for a Platform
|
78
|
+
#
|
79
|
+
# ==== Attributes
|
80
|
+
#
|
81
|
+
# * +name+ - a String representing the name of the function
|
82
|
+
# * +return_type+ - the type of data returned by the function
|
83
|
+
# * +param_types+ - an Array of types, corresponding to the parameters type
|
84
|
+
# * +options+ - if given, a hash of named options that will be given to FFI::Function::new. See FFI doc for details.
|
85
|
+
def get_extension_function( name, return_type, param_types, options = {} )
|
86
|
+
OpenCL.error_check(OpenCL::INVALID_OPERATION) if self.version_number < 1.2
|
87
|
+
name_p = FFI::MemoryPointer.from_string(name)
|
88
|
+
ptr = OpenCL.clGetExtensionFunctionAddressForPlatform( self, name_p )
|
89
|
+
return nil if ptr.null?
|
90
|
+
return FFI::Function::new(return_type, param_types, ptr, options)
|
91
|
+
end
|
92
|
+
|
93
|
+
# Creates a Context gathering devices of a certain type and belonging to this Platform
|
94
|
+
#
|
95
|
+
# ==== Attributes
|
96
|
+
#
|
97
|
+
# * +type+ - type of device to be used
|
98
|
+
# * +options+ - if given, a hash of named options
|
99
|
+
# * +block+ - if provided, a callback invoked when error arise in the context. Signature of the callback is { |FFI::Pointer to null terminated c string, FFI::Pointer to binary data, :size_t number of bytes of binary data, FFI::Pointer to user_data| ... }
|
100
|
+
#
|
101
|
+
# ==== Options
|
102
|
+
#
|
103
|
+
# * +:properties+ - a list of :cl_context_properties, the Platform will be prepended
|
104
|
+
# * +:user_data+ - an FFI::Pointer or an object that can be converted into one using to_ptr. The pointer is passed to the callback.
|
105
|
+
def create_context_from_type(type, options = {}, &block)
|
106
|
+
props = [ OpenCL::Context::PLATFORM, self ]
|
107
|
+
if options[:properties] then
|
108
|
+
props = props + options[:properties]
|
109
|
+
else
|
110
|
+
props.push( 0 )
|
111
|
+
end
|
112
|
+
opts = options.clone
|
113
|
+
opts[:properties] = props
|
114
|
+
OpenCL.create_context_from_type(type, opts, &block)
|
115
|
+
end
|
116
|
+
|
117
|
+
end
|
118
|
+
|
119
|
+
end
|
@@ -0,0 +1,245 @@
|
|
1
|
+
module OpenCL
|
2
|
+
|
3
|
+
# Builds (compile and link) a Program created from sources or binary
|
4
|
+
#
|
5
|
+
# ==== Attributes
|
6
|
+
#
|
7
|
+
# * +program+ - the program to build
|
8
|
+
# * +options+ - a hash containing named options
|
9
|
+
# * +block+ - if provided, a callback invoked when the Program is built. Signature of the callback is { |Program, FFI::Pointer to user_data| ... }
|
10
|
+
#
|
11
|
+
# ==== Options
|
12
|
+
#
|
13
|
+
# * +:device_list+ - an Array of Device to build the program for
|
14
|
+
# * +:user_data+ - a Pointer (or convertible to Pointer using to_ptr) to the memory area to pass to the callback
|
15
|
+
# * +:options+ - a String containing the options to use for the build
|
16
|
+
def self.build_program(program, options = {}, &block)
|
17
|
+
@@callbacks.push( block ) if block
|
18
|
+
opt = ""
|
19
|
+
opt = options[:options] if options[:options]
|
20
|
+
options_p = FFI::MemoryPointer.from_string(opt)
|
21
|
+
devices = options[:device_list]
|
22
|
+
devices = [devices].flatten if devices
|
23
|
+
devices_p = nil
|
24
|
+
num_devices = 0
|
25
|
+
if devices and devices.size > 0 then
|
26
|
+
num_devices = devices.size
|
27
|
+
devices_p = FFI::MemoryPointer.new( Device, num_devices)
|
28
|
+
num_devices.times { |indx|
|
29
|
+
devices_p.put_pointer(indx, devices[indx])
|
30
|
+
}
|
31
|
+
end
|
32
|
+
err = OpenCL.clBuildProgram(program, num_devices, devices_p, options_p, block, options[:user_data] )
|
33
|
+
OpenCL.error_check(err)
|
34
|
+
return program
|
35
|
+
end
|
36
|
+
|
37
|
+
# Creates a Program from sources
|
38
|
+
#
|
39
|
+
# ==== Attributes
|
40
|
+
#
|
41
|
+
# * +context+ - Context the created Program will be associated to
|
42
|
+
# * +strings+ - a single or an Array of String repesenting the program source code
|
43
|
+
def self.create_program_with_source(context, strings)
|
44
|
+
strs = nil
|
45
|
+
if strings == nil then
|
46
|
+
raise OpenCL::Error::new(OpenCL::Error.getErrorString(OpenCL::Error::INVALID_VALUE))
|
47
|
+
else
|
48
|
+
strs = [strings].flatten
|
49
|
+
end
|
50
|
+
n_strs = strs.size
|
51
|
+
strs_lengths = FFI::MemoryPointer.new( :size_t, n_strs )
|
52
|
+
c_strs = FFI::MemoryPointer.new( :pointer, n_strs )
|
53
|
+
|
54
|
+
c_strs_p = []
|
55
|
+
strs.each { |str|
|
56
|
+
if str then
|
57
|
+
c_strs_p.push (FFI::MemoryPointer.from_string(str))
|
58
|
+
end
|
59
|
+
}
|
60
|
+
raise OpenCL::Error::new(OpenCL::Error.getErrorString(OpenCL::Error::INVALID_VALUE)) if c_strs_p.size == 0
|
61
|
+
|
62
|
+
c_strs = FFI::MemoryPointer.new( :pointer, c_strs_p.size )
|
63
|
+
c_strs_length = FFI::MemoryPointer.new( :size_t, c_strs_p.size )
|
64
|
+
c_strs_p.each_with_index { |p, i|
|
65
|
+
c_strs[i].write_pointer(p)
|
66
|
+
c_strs_length[i].write_size_t(p.size)
|
67
|
+
}
|
68
|
+
pointer_err = FFI::MemoryPointer.new( :cl_int )
|
69
|
+
program_ptr = OpenCL.clCreateProgramWithSource(context, c_strs_p.size, c_strs, c_strs_length, pointer_err)
|
70
|
+
OpenCL.error_check(pointer_err.read_cl_int)
|
71
|
+
return OpenCL::Program::new( program_ptr, false )
|
72
|
+
end
|
73
|
+
|
74
|
+
# Maps the cl_program object of OpenCL
|
75
|
+
class Program
|
76
|
+
alias_method :orig_method_missing, :method_missing
|
77
|
+
|
78
|
+
# Intercepts a call to a missing method and tries to see if it is defined as a Kernel inside
|
79
|
+
# the Program. It then calls the Kernel enqueue_with_args method. Thanks pyopencl (Andreas Klöeckner) for the idea
|
80
|
+
def method_missing(m, *a, &b)
|
81
|
+
m_string = m.to_s
|
82
|
+
k = nil
|
83
|
+
begin
|
84
|
+
k = self.create_kernel(m_string)
|
85
|
+
rescue OpenCL::Error
|
86
|
+
k = nil
|
87
|
+
end
|
88
|
+
if k then
|
89
|
+
k.enqueue_with_args(*a, &b)
|
90
|
+
else
|
91
|
+
orig_method_missing(m, *args, &b)
|
92
|
+
end
|
93
|
+
end
|
94
|
+
# Returns an Array containing the sizes of the binary inside the Program for each device
|
95
|
+
eval OpenCL.get_info_array("Program", :size_t, "BINARY_SIZES")
|
96
|
+
|
97
|
+
# Returns the number of Kernels defined in the Program
|
98
|
+
eval OpenCL.get_info("Program", :size_t, "NUM_KERNELS")
|
99
|
+
|
100
|
+
# Returns an Array of String representing the Kernel names inside the Program
|
101
|
+
def kernel_names
|
102
|
+
kernel_names_size = FFI::MemoryPointer.new( :size_t )
|
103
|
+
error = OpenCL.clGetProgramInfo( self, OpenCL::Program::KERNEL_NAMES, 0, nil, kernel_names_size)
|
104
|
+
OpenCL.error_check(error)
|
105
|
+
k_names = FFI::MemoryPointer.new( kernel_names_size.read_size_t )
|
106
|
+
error = OpenCL.clGetProgramInfo( self, OpenCL::Program::KERNEL_NAMES, kernel_names_size.read_size_t, k_names, nil)
|
107
|
+
OpenCL.error_check(error)
|
108
|
+
k_names_string = k_names.read_string
|
109
|
+
returns k_names_string.split(";")
|
110
|
+
end
|
111
|
+
|
112
|
+
# Returns the concatenated Program sources
|
113
|
+
eval OpenCL.get_info("Program", :string, "SOURCE")
|
114
|
+
|
115
|
+
# Returns the BuildStatus of the Program
|
116
|
+
def build_status(devs = nil)
|
117
|
+
devs = self.devices if not devs
|
118
|
+
devs = [devs].flatten
|
119
|
+
ptr = FFI::MemoryPointer.new( :cl_build_status )
|
120
|
+
return devs.collect { |dev|
|
121
|
+
error = OpenCL.clGetProgramBuildInfo(self, dev, OpenCL::Program::BUILD_STATUS, ptr.size, ptr, nil)
|
122
|
+
OpenCL.error_check(error)
|
123
|
+
OpenCL::BuildStatus::new(ptr.read_cl_build_status)
|
124
|
+
}
|
125
|
+
end
|
126
|
+
|
127
|
+
# Returns the BinaryType for each Device associated to the Program or the Device specified
|
128
|
+
def binary_type(devs = nil)
|
129
|
+
devs = self.devices if not devs
|
130
|
+
devs = [devs].flatten
|
131
|
+
ptr = FFI::MemoryPointer.new( :cl_program_binary_type )
|
132
|
+
return devs.collect { |dev|
|
133
|
+
error = OpenCL.clGetProgramBuildInfo(self, dev, OpenCL::Program::BINARY_TYPE, ptr.size, ptr, nil)
|
134
|
+
OpenCL.error_check(error)
|
135
|
+
OpenCL::Program::BinaryType::new(ptr.read_cl_program_binary_type)
|
136
|
+
}
|
137
|
+
end
|
138
|
+
|
139
|
+
# Returns the build options for each Device associated to the Program or the Device specified
|
140
|
+
def build_options(devs = nil)
|
141
|
+
devs = self.devices if not devs
|
142
|
+
devs = [devs].flatten
|
143
|
+
return devs.collect { |dev|
|
144
|
+
ptr1 = FFI::MemoryPointer.new( :size_t, 1)
|
145
|
+
error = OpenCL.clGetProgramBuildInfo(self, dev, OpenCL::Program::BUILD_OPTIONS, 0, nil, ptr1)
|
146
|
+
OpenCL.error_check(error)
|
147
|
+
ptr2 = FFI::MemoryPointer.new( ptr1.read_size_t )
|
148
|
+
error = OpenCL.clGetProgramBuildInfo(self, dev, OpenCL::Program::BUILD_OPTIONS, ptr1.read_size_t, ptr2, nil)
|
149
|
+
OpenCL.error_check(error)
|
150
|
+
ptr2.read_string
|
151
|
+
}
|
152
|
+
end
|
153
|
+
|
154
|
+
# Returns the build log for each Device associated to the Program or the Device specified
|
155
|
+
def build_log(devs = nil)
|
156
|
+
devs = self.devices if not devs
|
157
|
+
devs = [devs].flatten
|
158
|
+
return devs.collect { |dev|
|
159
|
+
ptr1 = FFI::MemoryPointer.new( :size_t, 1)
|
160
|
+
error = OpenCL.clGetProgramBuildInfo(self, dev, OpenCL::Program::BUILD_LOG, 0, nil, ptr1)
|
161
|
+
OpenCL.error_check(error)
|
162
|
+
ptr2 = FFI::MemoryPointer.new( ptr1.read_size_t )
|
163
|
+
error = OpenCL.clGetProgramBuildInfo(self, dev, OpenCL::Program::BUILD_LOG, ptr1.read_size_t, ptr2, nil)
|
164
|
+
OpenCL.error_check(error)
|
165
|
+
ptr2.read_string
|
166
|
+
}
|
167
|
+
end
|
168
|
+
|
169
|
+
# Returns the binaries associated to the Program for each Device
|
170
|
+
def binaries
|
171
|
+
sizes = self.binary_sizes
|
172
|
+
bin_array = FFI::MemoryPointer.new( :pointer, sizes.length )
|
173
|
+
sizes.length
|
174
|
+
total_size = 0
|
175
|
+
sizes.each_with_index { |s, i|
|
176
|
+
total_size += s
|
177
|
+
bin_array[i].write_pointer(FFI::MemoryPointer.new(s))
|
178
|
+
}
|
179
|
+
error = OpenCL.clGetProgramInfo(self, Program::BINARIES, total_size, bin_array, nil)
|
180
|
+
OpenCL.error_check(error)
|
181
|
+
bins = []
|
182
|
+
sizes.each_with_index { |s, i|
|
183
|
+
bins.push bin_array[i].read_pointer.read_bytes(s)
|
184
|
+
}
|
185
|
+
return bins
|
186
|
+
end
|
187
|
+
|
188
|
+
# Builds (compile and link) the Program created from sources or binary
|
189
|
+
#
|
190
|
+
# ==== Attributes
|
191
|
+
#
|
192
|
+
# * +options+ - a hash containing named options
|
193
|
+
# * +block+ - if provided, a callback invoked when error arise in the context. Signature of the callback is { |Program, FFI::Pointer to user_data| ... }
|
194
|
+
#
|
195
|
+
# ==== Options
|
196
|
+
# * +:device_list+ - an Array of Device to build the program for
|
197
|
+
# * +:user_data+ - a Pointer (or convertible to Pointer using to_ptr) to the memory area to pass to the callback
|
198
|
+
# * +:options+ - a String containing the options to use for the build
|
199
|
+
def build(options = { }, &block)
|
200
|
+
OpenCL.build_program(self, options, &block)
|
201
|
+
end
|
202
|
+
|
203
|
+
# Returns the Context the Program is associated to
|
204
|
+
def context
|
205
|
+
ptr = FFI::MemoryPointer.new( Context )
|
206
|
+
error = OpenCL.clGetProgramInfo(self, Program::CONTEXT, Context.size, ptr, nil)
|
207
|
+
OpenCL.error_check(error)
|
208
|
+
return OpenCL::Context::new( ptr.read_pointer )
|
209
|
+
end
|
210
|
+
|
211
|
+
##
|
212
|
+
# :method: num_devices()
|
213
|
+
# Returns the number of device this Program is associated with
|
214
|
+
|
215
|
+
##
|
216
|
+
# :method: reference_count()
|
217
|
+
# Returns the reference counter for this Program
|
218
|
+
%w( NUM_DEVICES REFERENCE_COUNT ).each { |prop|
|
219
|
+
eval OpenCL.get_info("Program", :cl_uint, prop)
|
220
|
+
}
|
221
|
+
|
222
|
+
# Returns the Array of Device the Program is associated with
|
223
|
+
def devices
|
224
|
+
n = self.num_devices
|
225
|
+
ptr2 = FFI::MemoryPointer.new( Device, n )
|
226
|
+
error = OpenCL.clGetProgramInfo(self, Program::DEVICES, Device.size*n, ptr2, nil)
|
227
|
+
OpenCL.error_check(error)
|
228
|
+
return ptr2.get_array_of_pointer(0, n).collect { |device_ptr|
|
229
|
+
OpenCL::Device.new(device_ptr)
|
230
|
+
}
|
231
|
+
end
|
232
|
+
|
233
|
+
# Returns the Kernel corresponding the the specified name in the Program
|
234
|
+
def create_kernel( name )
|
235
|
+
return OpenCL.create_kernel( self, name )
|
236
|
+
end
|
237
|
+
|
238
|
+
# Returns an Array of Kernel corresponding to the kernels defined inside the Program
|
239
|
+
def kernels
|
240
|
+
return OpenCL.create_kernels_in_program( self )
|
241
|
+
end
|
242
|
+
|
243
|
+
end
|
244
|
+
|
245
|
+
end
|