The OpenCL C 1.2 language provides an expressive variant of the C language with which to program heterogeneous architectures. There is already a significant body of code in the wild written in the OpenCL C language - both in open source and proprietary software. This document explains how the OpenCL C language is mapped onto an implementation of the Vulkan standard for high-performance graphics and compute.
The following subjects are covered:
- Which SPIR-V features are used.
- How the Vulkan API makes use of the Vulkan variant of SPIR-V produced.
- How OpenCL C language constructs are mapped down onto Vulkan's variant of SPIR-V.
- Restrictions on the OpenCL C language as is to be consumed by a Vulkan implementation.
The sampler map is deprecated. It is no longer necessary to specify a sampler map for literal samplers. Refer to the bindings generated for the literal samplers in descriptor map. The -samplermap option is still accepted, but support will be removed at later date. A single sampler binding is generated for each unique literal sampler in the program.
The SPIR-V as produced from the OpenCL C language can make use of the following additional extensions:
- SPV_KHR_variable_pointers - to enable the support of more expressive pointers that the OpenCL C language can make use of.
- SPV_KHR_storage_buffer_storage_class - required by SPV_KHR_variable_pointers, to enable use of the StorageBuffer storage class.
The SPIR-V as produced from the OpenCL C language can make use of the following capabilities:
Shader
as we are targeting the OpenCL C language at a Vulkan implementation.VariablePointersStorageBuffer
, from the SPV_KHR_variable_pointers extension.VariablePointers
, from the SPV_KHR_variable_pointers extension.- Note: the compiler attempts to add the minimal variable pointers capability required.
Int8
if char or uchar types (or composites of them) are used.Int16
if short or ushort types (or composites of them) are used.Int64
if long or ulong types (or composites of them) are used.Float16
if the half type (or composites of it) is used.- Note: this requires enabling the cl_khr_fp16 extension in the source.
Float64
if the double type (or composites of it) is used.ImageStorageWriteWithoutFormat
if write_only images are used.ImageQuery
if any image query is used.Image1D
if a write_only image is used.Sampled1D
if a read_only image is used.
A Vulkan implementation that is to consume the SPIR-V produced from the OpenCL C language must conform to the following the rules:
- If the short/ushort types are used in the OpenCL C:
- The
shaderInt16
field ofVkPhysicalDeviceFeatures
must be set to true.
- The
- If images are used in the OpenCL C:
- The
shaderStorageImageReadWithoutFormat
field ofVkPhysicalDeviceFeatures
must be set to true. - The
shaderStorageImageWriteWithoutFormat
field ofVkPhysicalDeviceFeatures
must be set to true.
- The
- The implementation must support extensions VK_KHR_storage_buffer_storage_class and
VK_KHR_variable_pointers:
- A call to
vkCreateDevice()
where theppEnabledExtensionNames
field ofVkDeviceCreateInfo
contains extension strings "VK_KHR_storage_buffer_storage_class" and "VK_KHR_variable_pointers" must succeed.
- A call to
In order to pass 8- or 16-bit types in the shader interface, Vulkan requires the appropriate bits are set in VkPhysicalDevice8BitStorageFeaturesKHR for 8-bit types or VkPhysicalDevice16BitStorageFeaturesKHR (or the appropriate Vulkan 1.1 or 1.2 feature structs). By default clspv assumes all feature bits are enabled, but provides options to disallow 8- or 16-bit interfaces. -no-8bit-storage reflects 8-bit storage features and -no-16bit-storage reflects 16-bit storage features. Each option can take the following values (can be specified in a comma-separated list or multiple times):
ssbo
: Represents the storage buffer feature bit.ubo
: Represents the uniform and storage buffer feature bit.pushconstant
: Represents the push constant feature bit.
For example, if your device only supports storageBuffer16BitAccess (and no 8-bit interfaces), pass the following on the command line:
-no-16bit-storage=ubo,pushconstant -no-8bit-storage=ssbo,ubo,pushconstant
OpenCL C kernel argument types are mapped to Vulkan descriptor types in the following way:
- If the argument to the kernel is a read only image, the matching Vulkan
descriptor set type is
VK_DESCRIPTOR_TYPE_SAMPLED_IMAGE
. - If the argument to the kernel is a write only image, the matching Vulkan
descriptor set type is
VK_DESCRIPTOR_TYPE_STORAGE_IMAGE
. - If the argument to the kernel is a sampler, the matching Vulkan
descriptor set type is
VK_DESCRIPTOR_TYPE_SAMPLER
. - If the argument to the kernel is a constant or global pointer type, the
matching Vulkan descriptor set type is
VK_DESCRIPTOR_TYPE_STORAGE_BUFFER
. If option -constant-args-ubo' is used and the kernel has constant pointer types, setVK_DESCRIPTOR_TYPE_UNIFORM_BUFFER
. - If the argument to the kernel is a plain-old-data type, the matching Vulkan
descriptor set type is
VK_DESCRIPTOR_TYPE_STORAGE_BUFFER
by default. If the option-pod-ubo
is used the descriptor set type isVK_DESCRIPTOR_TYPE_UNIFORM_BUFFER
. If the option-pod-pushconstant
is used the arg is instead passed via push constants.
Note: -pod-ubo
and -pod-pushconstant
are exclusive options.
Note: By default, all plain-old-data kernel arguments are collected into a
single structure to be passed in to the compute shader as a single resource. If
-cluster-pod-kernel-args=0
is specified, each plain-old-data argument will be
passed in a via a unique resource.
Note: -pod-pushconstant
cannot be specified with -cluster-pod-kernel-args=0
.
Some OpenCL C language features that are not natively expressible in Vulkan's variant of SPIR-V, require a subtle mapping to how Vulkan SPIR-V represents the corresponding functionality.
An additional preprocessor macro VULKAN
is set, to allow developers to guard
OpenCL C functionality based on whether the Vulkan API is being targeted or not.
This value is set to 100, to match Vulkan version 1.0.
OpenCL C language kernels take the form:
void kernel foo(global int* a, global float* b, uint c, float2 local* d);
SPIR-V tracks OpenCL C language kernels using OpEntryPoint
opcodes that denote
the entry-points where an API interacts with a compute kernel.
Vulkan's variant of SPIR-V requires that the entry-points be void
return
functions, and that they take no arguments.
To pass data into Vulkan SPIR-V shaders, OpVariable
s are declared outside of
the functions, and decorated with DescriptorSet
and Binding
decorations, to
denote that the shaders can interact with their data.
The default way to map an OpenCL C language kernel to a Vulkan SPIR-V compute shader is as follows:
- If a sampler map file is specified, all literal samplers use descriptor set 0.
- By default, all kernels in the translation unit use the same descriptor set
number, either 0, 1, or 2. (The particular value depends on whether
a sampler map is used, and how
__constant
variables are mapped.) This is new default behaviour.- Use option
-distinct-kernel-descriptor-sets
to get the old behaviour, where each kernel is assigned its own descriptor set number, such that the first kernel has descriptor set 0, and each subsequent kernel is an increment of 1 from the previous.
- Use option
- Except for pointer-to-local arguments, each kernel argument
is assigned a descriptor binding in that kernel's
corresponding
DescriptorSet
. - If the argument to the kernel is a
global
orconstant
pointer, it is placed into a SPIR-VOpTypeStruct
that is decorated withBlock
, and anOpVariable
of this structure type is created and decorated with the correspondingDescriptorSet
andBinding
, using theStorageBuffer
storage class. - If the argument to the kernel is a plain-old-data type, it is placed into a
SPIR-V
OpTypeStruct
that is decorated withBlock
, and anOpVariable
of this structure type is created and decorated with the correspondingDescriptorSet
andBinding
, using theStorageBuffer
storage class. - If the argument to the kernel is an image or sampler, an
OpVariable
of theOpTypeImage
orOpTypeSampler
type is created and decorated with the correspondingDescriptorSet
andBinding
, using theUniformConstant
storage class. - If the argument to the kernel is a pointer to type T in
__local
storage, then no descriptor is generated. Instead, that argument is mapped to a variable inWorkgroup
storage class, of type array-of-T. The array size is specified by an integer specialization constant. The specialization ID is reported in the descriptor map file, generated via the-descriptormap
option.
The shaders produced use the GLSL450 memory model. As such, there is an assumption of no aliasing by default. The compiler does not generate Aliased decorations currently. Users should be aware of this and ensure they are not relying on aliasing.
The compiler can report the descriptor set and bindings used for samplers
in the sampler map and for the kernel arguments, and also array sizing information for
pointer-to-local arguments.
Use option -descriptormap
to name a file that should contain the mapping information.
Example:
clspv foo.cl -descriptormap=foomap.csv
The descriptor map is a text file with comma-separated values.
Consider this example:
// First kernel in the translation unit, and no sampler map is used.
kernel void foo(global int* a, float f, global float* b, uint c) {...}
It generates the following descriptor map:
kernel_decl,foo
kernel,foo,arg,a,argOrdinal,0,descriptorSet,0,binding,0,offset,0,argKind,buffer
kernel,foo,arg,f,argOrdinal,1,descriptorSet,0,binding,1,offset,0,argKind,pod,argSize,4
kernel,foo,arg,b,argOrdinal,2,descriptorSet,0,binding,2,offset,0,argKind,buffer
kernel,foo,arg,c,argOrdinal,3,descriptorSet,0,binding,3,offset,0,argKind,pod,argSize,4
spec_constant,workgroup_size_x,spec_id,0
spec_constant,workgroup_size_y,spec_id,1
spec_constant,workgroup_size_z,spec_id,2
The kernels in the module module are declared as follows:
kernel_decl
- kernel name
For kernel arguments of types pointer-to-global, pointer-to-constant, and plain-old-data types, the fields are:
kernel
to indicate a kernel argument- kernel name
arg
to indicate a kernel argument- argument name
argOrdinal
to indicate a kernel argument ordinal position field- the argument's 0-based position in the kernel's parameter list
descriptorSet
- the DescriptorSet value
binding
- the Binding value
offset
- The byte offset inside the storage buffer where you should write the argument value. This will always be zero, unless you cluster plain-old-data kernel arguments. (See below.)
argKind
- a string describing the kind of argument, one of:
buffer
- OpenCL bufferbuffer_ubo
- OpenCL constant buffer. Sent in a uniform buffer.pod
- Plain Old Data, e.g. a scalar, vector, or structure. Sent in a storage buffer.pod_ubo
- Plain Old Data, e.g. a scalar, vector, or structure. Sent in a uniform buffer.pod_pushconstant
- Plain Old Data, e.g. a scalar, vector or structure. Sent in push constants.ro_image
- Read-only imagewo_image
- Write-only imagesampler
- Sampler
argSize
- only present for plain-old-data kernel arguments.
Module-wide specialization constants are specified as follows:
spec_constant
to describe the use of a module-wide specialization constant- specialization constant type (e.g x dimension of WorkgroupSize)
spec_id
- the SpecId value
Consider this example, which uses pointer-to-local arguments:
kernel void foo(local float* L, global float* A, local float4 *L2) {...}
It generates the following descriptor map:
kernel_decl,foo
kernel,foo,arg,L,argOrdinal,0,argKind,local,arrayElemSize,4,arrayNumElemSpecId,3
kernel,foo,arg,A,argOrdinal,1,descriptorSet,0,binding,0,offset,0,argKind,buffer
kernel,foo,arg,L2,argOrdinal,2,argKind,local,arrayElemSize,16,arrayNumElemSpecId,4
spec_constant,workgroup_size_x,spec_id,0
spec_constant,workgroup_size_y,spec_id,1
spec_constant,workgroup_size_z,spec_id,2
The kernels in the module module are declared as follows:
kernel_decl
- kernel name
For kernel arguments of type pointer-to-local, the fields are:
kernel
to indicate a kernel argument- kernel name
arg
to indicate a kernel argument- argument name
argOrdinal
to indicate a kernel argument ordinal position field- the argument's 0-based position in the kernel's parameter list
argKind
local
to indicate a pointer-to-local argumentarrayElemSize
- the number of bytes in each element of the array
arrayNumElemSpecId
- the specialization constant ID used to specify the number of elements to
allocate for the array in
Workgroup
storage. Specifically, it is theSpecId
decoration on the integer constant that specficies the array size. (This number is always at least 3 so that specialization IDs 0, 1, and 2 can be use for the workgroup size dimensions alongx
,y
, andz
.)
Module-wide specialization constants are specified as follows:
spec_constant
to describe the use of a module-wide specialization constant- specialization constant type (e.g x dimension of WorkgroupSize)
spec_id
- the SpecId value
Notes: Each pointer-to-local argument is assigned its own array type and specialization constant to size the array. Unless you override the array size specialization constant at pipeline creation time, the array will only have one element.
If a sampler map is used, then samplers use descriptor set 0 and kernel descriptor
set numbers start at 1. For example, if the sampler map file is mysamplermap
containing:
CLK_NORMALIZED_COORDS_FALSE | CLK_ADDRESS_CLAMP_TO_EDGE | CLK_FILTER_NEAREST,
CLK_NORMALIZED_COORDS_TRUE | CLK_ADDRESS_CLAMP_TO_EDGE | CLK_FILTER_LINEAR
Then compiling with:
clspv foo.cl -samplermap=mysamplermap -descriptormap=mydescriptormap
Then mydescriptormap
will contain:
sampler,18,samplerExpr,"CLK_ADDRESS_CLAMP_TO_EDGE|CLK_FILTER_NEAREST|CLK_NORMALIZED_COORDS_FALSE",descriptorSet,0,binding,0
sampler,35,samplerExpr,"CLK_ADDRESS_CLAMP_TO_EDGE|CLK_FILTER_LINEAR|CLK_NORMALIZED_COORDS_TRUE",descriptorSet,0,binding,1
kernel,foo,arg,a,argOrdinal,0,descriptorSet,1,binding,0,offset,0,argKind,buffer
kernel,foo,arg,f,argOrdinal,1,descriptorSet,1,binding,1,offset,0,argKind,pod,argSize,4
kernel,foo,arg,b,argOrdinal,2,descriptorSet,1,binding,2,offset,0,argKind,buffer
kernel,foo,arg,c,argOrdinal,3,descriptorSet,1,binding,3,offset,0,argKind,pod,argSize,4
Normally plain-old-data arguments are passed into the kernel via a storage buffer.
Use option -pod-ubo
to pass these parameters in via a uniform buffer. These can
be faster to read in the shader.
When option -pod-ubo
is used, the descriptor map list the argKind
of a plain-old-data
argument as pod_ubo
rather than the default of pod
.
Normally plain-old-data arguments are passed into the kernel via a storage buffer.
Use the option -pod-pushconstant
to pass these parameters in via push constants. The
option -cluster-pod-kernel-args=0
cannot be specified. Push constants are intended
to provide a fast read path and should be faster to access than a buffer.
When the option -pod-pushconstant
is used, the descriptor map lists the argKind
of
plain-old-data arguments as pod_pushconstant
rather than the default of pod
. There
is no descriptorset
or binding
information for push constants.
Note: Vulkan implementations have limited push constant storage (default is 128B).
clspv provides the option -max-pushconstant-size
to specify (in bytes) the
implementation limit for push constants. This is validated at the start of the compile.
Note: Module scope push constanst are currently incompatible with plain-old-data arguments sent as push constants.
TODO(alan-baker): See #529 for the overall plan to address this.
The descriptor map entry for kernel arguments will not contain descriptorSet
or
binding
entries. For example, an integer arg f
to kernel foo
might look like:
kernel,foo,arg,f,argOrdinal,1,offset,0,argKind,pod_pushconstant,argSize,4
Normally pointer-to-constant kernel arguments are passed into the kernel via a storage
buffer. Use option -constant-args-ubo
to pass these parameters in via a uniform buffer.
Uniform buffers can be faster to read in the shader.
The compiler will generate an error if the layout of the buffer does not satisfy the Standard Uniform Buffer Layout rules of the Vulkan specification (see section 15.5.4).
Descriptors can be scarce. So the compiler also has an option
-cluster-pod-kernel-args
which can be used to reduce the number of descriptors.
When the option is used:
- All plain-old-data (POD) kernel arguments are collected into a single struct and passed into the compute shader via a single storage buffer resource.
- The binding numbers are assigned as previously, except:
- Binding numbers for non-POD arguments are assigned as if there were no POD arguments.
- The binding number for the struct containing the POD arguments is one more than the highest non-POD argument.
By default this option is enabled. To disable this behavior, pass
-cluster-pod-kernel-args=0
to the compiler.
For example:
// First kernel in the translation unit, and no sampler map is used.
void kernel foo(global int* a, float f, global float* b, uint c);
In the default case, the bindings are:
a
is mapped to a storage buffer with descriptor set 0, binding 0b
is mapped to a storage buffer with descriptor set 0, binding 1f
andc
are POD arguments, so they are mapped to the first and second members of a struct, and that struct is mapped to a storage buffer with descriptor set 0 and binding 2
If -cluster-pod-kernel-args=0
is used:
a
is mapped to a storage buffer with descriptor set 0, binding 0f
is mapped to a storage buffer with descriptor set 0, binding 1b
is mapped to a storage buffer with descriptor set 0, binding 2c
is mapped to a storage buffer with descriptor set 0, binding 3
That is, compiling as follows:
clspv foo.cl -descriptormap=myclusteredmap
will produce the following in myclusteredmap
:
kernel,foo,arg,a,argOrdinal,0,descriptorSet,0,binding,0,offset,0,argKind,buffer
kernel,foo,arg,b,argOrdinal,2,descriptorSet,0,binding,1,offset,0,argKind,buffer
kernel,foo,arg,f,argOrdinal,1,descriptorSet,0,binding,2,offset,0,argKind,pod,argSize,4
kernel,foo,arg,c,argOrdinal,3,descriptorSet,0,binding,2,offset,4,argKind,pod,argSize,4
If foo
were the second kernel in the translation unit, then its arguments
would also use descriptor set 0.
If foo
were the second kernel in the translation unit and option
-distinct-kernel-descriptor-sets
is used, then its arguments would
use descriptor set 1.
Compiling with the same sampler map from before:
clspv foo.cl -descriptormap=myclusteredmap -samplermap=mysamplermap
produces the following descriptor map:
sampler,18,samplerExpr,"CLK_ADDRESS_CLAMP_TO_EDGE|CLK_FILTER_NEAREST|CLK_NORMALIZED_COORDS_FALSE",descriptorSet,0,binding,0
sampler,35,samplerExpr,"CLK_ADDRESS_CLAMP_TO_EDGE|CLK_FILTER_LINEAR|CLK_NORMALIZED_COORDS_TRUE",descriptorSet,0,binding,1
kernel,foo,arg,a,argOrdinal,0,descriptorSet,1,binding,0,offset,0,argKind,buffer
kernel,foo,arg,b,argOrdinal,2,descriptorSet,1,binding,1,offset,0,argKind,buffer
kernel,foo,arg,f,argOrdinal,1,descriptorSet,1,binding,2,offset,0,argKind,pod,argSize,4
kernel,foo,arg,c,argOrdinal,3,descriptorSet,1,binding,2,offset,4,argKind,pod,argSize,4
TODO(dneto): Give an example using images.
The descriptor map includes entries for specialization constants that are used module-wide. The following specialization constants are currently generated:
workgroup\_size\_x
: The x-dimension of workgroup size.workgroup\_size\_y
: The y-dimension of workgroup size.workgroup\_size\_z
: The z-dimension of workgroup size.work_dim
: The work dimensions.global\_offset\_x
: The x-dimension of global offset.global\_offset\_y
: The y-dimension of global offset.global\_offset\_z
: The z-dimension of global offset.
By default, each module-scope variable in __constant
address space is mapped to
a SPIR-V variable in Private address space, with an intializer. This works only
for simple scenarios, where:
- The variable is small, so it's reasonable to fit in a single invocations private registers, and
- The variable is only read, and in particular its address is not taken.
In more general cases, use compiler option -module-constants-in-storage-buffer
. In this case:
- All module-scope constants are collected into a single SPIR-V storage buffer variable in its own descriptor set.
- The intialization data are written to the descriptor map, and the host program must fill the buffer with that data before the kernel executes.
Consider this example kernel a.cl
:
typedef struct {
char c;
uint a;
float f;
} Foo;
__constant Foo ppp[3] = {{'a', 0x1234abcd, 1.0}, {'b', 0xffffffff, 1.5}, {0}};
kernel void foo(global uint* A, uint i) { *A = ppp[i].a; }
Compiling as follows:
clspv a.cl -descriptormap=map -module-constants-in-storage-buffer
Produces the following in file map
:
constant,descriptorSet,1,binding,0,hexbytes,61000000cdab34120000803f62000000ffffffff0000c03f000000000000000000000000
kernel,foo,arg,A,argOrdinal,0,descriptorSet,0,binding,0,offset,0,argKind,buffer
kernel,foo,arg,i,argOrdinal,1,descriptorSet,0,binding,1,offset,0,argKind,pod
The initialization data are in the line starting with constant
, and its fields are:
constant
to indicate constant initialization datadescriptorSet
- the DescriptorSet value
binding
- the Binding value
kind
buffer
to indicate the use of a storage bufferhexbytes
to indicate the next field is the data, as a sequence of bytes in hexadecimal- a sequence of bytes expressed in hexadecimal notation, presented in order from lowest address to highest address.
Take a closer look at the hexadecimal bytes in the example. They are:
61
: ASCII character 'a'000000
: zero padding to satisfy alignment for the 32-bit integer value that followscdab3412
: the integer value0x1234abcd
in little-endian format0000803f
: the float value 1.062
: ASCII character 'b'000000
: zero padding to satisfy alignment for the 32-bit integer value that followsffffffff
: the integer value0xffffffff
0000c03f
: the float value 1.5000000000000000000000000
: 12 zero bytes representing the zero-initialized third Foo value.
Some features, when enabled, require values to be passed by the application via push constants. For each value requested by clspv, an entry will be present in the descriptor map. The format is:
pushconstant
to indicate the entry is a push constantname
to indicate the name of the requested push constant follows- the name of the push constant requested
offset
to indicate the offset at which the push constant is expected follows- the offset within push constants at which the entry is expected
size
to indicate that the total size follows- the total size of the push constant that will be used.
Here is a list of the push constants currently supported:
global_offset
: the 3D global offset used byget_global_offset()
and in global ID calculations. A vector of 3 32-bit integer values. Lower dimensions come first in memory.enqueued_local_size
: the 3D local work size returned byget_enqueued_local_size()
. A vector of 3 32-bit integer values. Lower dimensions come first in memory.global_size
: the 3D global size of the NDRange as returned byget_global_size()
. A vector of 3 32-bit integer values. Lower dimensions come first in memory. Only required when non-uniform NDRanges are supported.num_workgroups
: the 3D number of work groups in the NDRange as returned byget_num_groups()
. A vector of 3 32-bit integer values. Lower dimensions come first in memory. Only required when non-uniform NDRanges are supported.region_offset
: the sum of the global ID offset into the NDRange for this uniform region and the global offset of the NDRange. A vector of 3 32-bit integer values. Lower dimensions come first in memory. Only required when non-uniform NDRanges are supported.region_group_offset
: the 3D group ID offset into the NDRange for this region. A vector of 3 32-bit integer values. Lower dimensions come first in memory. Only required when non-uniform NDRanges are supported.
The following attributes are ignored in the OpenCL C source, and thus have no functional impact on the produced SPIR-V:
__attribute__((work_group_size_hint(X, Y, Z)))
__attribute__((packed))
__attribute__ ((endian(host)))
__attribute__ ((endian(device)))
__attribute__((vec_type_hint(<typen>)))
The __attribute__((reqd_work_group_size(X, Y, Z)))
kernel attribute specifies
the work-group size that must be used with that kernel.
The OpenCL C language allows the work-group size to be set just before executing
the kernel on the device, at clEnqueueNDRangeKernel()
time.
Vulkan requires that the work group size be specified no later than when the
VkPipeline
is created, which in OpenCL terms corresponds to when the
cl_kernel
is created.
To allow for the maximum flexibility to developers who are used to specifying
the work-group size in the host API and not in the device-side kernel
language, we can use specialization constants to allow for setting the work-group
size at VkPipeline
creation time.
If the reqd_work_group_size
attribute is used in the OpenCL C source, then that
attribute will specify the work-group size that must be used.
Otherwise, the Vulkan SPIR-V produced by the compiler will contain specialization
constants as follows:
- The x dimension of the work-group size is stored in a specialization
constant that is decorated with the
SpecId
, whose value defaults to 1. - The y dimension of the work-group size is stored in a specialization
constant that is decorated with the
SpecId
, whose value defaults to 1. - The z dimension of the work-group size is stored in a specialization
constant that is decorated with the
SpecId
, whose value defaults to 1.
The ids for the work-group size are recorded in the descriptor map.
If a compilation unit contains multiple kernels, then either:
- All kernels should have a
reqd_work_group_size
attribute, or - No kernels should have a
reqd_work_group_size
attribute. In this case work group sizes would be set via specialization constants for the pipeline as described above.
Signed integer types are mapped down onto their unsigned equivalents in SPIR-V as produced from OpenCL C.
Signed integer modulus (%
) operations, where either argument to the modulus is
a negative integer, will result in an undefined result.
OpenCL C language built-in functions are mapped, where possible, onto their GLSL
4.5 built-in equivalents.
For example, the OpenCL C language built-in function tan()
is mapped onto
GLSL's built-in function tan()
.
The OpenCL C built-in sign()
function does not differentiate between a signed
and unsigned 0.0 input value, nor does it return 0.0 if the input value is a
NaN.
The OpenCL C built-in mad24()
and mul24()
functions do not perform their
operations using 24-bit integers. Instead, they use 32-bit integers, and thus
have no performance-improving characteristics over normal 32-bit integer
arithmetic.
The OpenCL C work-item functions map to Vulkan SPIR-V as follows:
get_work_dim()
is mapped to thework_dim
spec constant when-work-dim
is enabled (on by default), otherwise it always returns 3.get_global_size()
is implemented by multiplying the result fromget_local_size()
by the result fromget_num_groups()
.get_global_id()
is mapped to a SPIR-V variable decorated withGlobalInvocationId
. The global offset is added to that variable when-global-offset
or-global-offset-push-constant
is enabled.get_local_size()
is mapped to a SPIR-V variable decorated withWorkgroupSize
.get_local_id()
is mapped to a SPIR-V variable decorated withLocalInvocationId
.get_num_groups()
is mapped to a SPIR-V variable decorated withNumWorkgroups
.get_group_id()
is mapped to a SPIR-V variable decorated withWorkgroupId
.get_global_offset()
is mapped to theglobal_offset
spec constant or push constant when-global-offset
or-global-offset-push-constant
is enabled, otherwise it always returns 0. Spec constants are used unless-global-offset-push-constant
is specified or the language is set to OpenCL C++ or OpenCL 2.0.
Some OpenCL C language features that have no expressible equivalents in Vulkan's variant of SPIR-V are restricted.
OpenCL C language kernels must not be called from other kernels.
Pointers of type half
must not be used as kernel arguments.
Booleans are an abstract type - they have no known compile-time size.
Using a boolean type as the argument to the sizeof()
operator will result in
an undefined value.
The boolean type must not be used to form global
, or constant
variables,
nor be used within a struct or union type in the global
, or constant
address
spaces.
The char
, char2
, char3
, uchar
, uchar2
, and uchar3
types
can be used. To disable general support for these types, use -int8=0
.
The double
, double2
, double3
and double4
types must not be used.
The image1d_buffer_t
type must not be used.
Any sampler_t
's must be passed in via a kernel argument, or the sampler
must be in the sampler map (see the -samplemap command line argument).
The event_t
type must not be used.
Pointers are an abstract type - they have no known compile-time size.
Using a pointer type as the argument to the sizeof()
operator will result in
an undefined value.
Pointer-to-integer casts must not be used.
Integer-to-pointer casts must not be used.
Pointers must not be compared for equality or inequality.
Vectors of 8 and 16 elements must not be used.
Recursively defined struct types must not be used.
Since pointers have no known compile-time size, the pointer-sized types
size_t
, ptrdiff_t
, uintptr_t
, and intptr_t
do not represent types that
are the same size as a pointer.
Instead, those types are mapped to 32-bit integer types.
For any OpenCL C language built-in functions that are mapped onto their GLSL 4.5 built-in equivalents, the precision requirements of the OpenCL C language built-ins are not necessarily honoured.
The atomic_xchg()
built-in function that takes a floating-point argument
must not be used.
The convert_<type>_rte()
, convert_<type>_rtz()
, convert_<type>_rtp()
,
convert_<type>_rtn()
, convert_<type>_sat()
, convert_<type>_sat_rte()
,
convert_<type>_sat_rtz()
, convert_<type>_sat_rtp()
, and
convert_<type>_sat_rtn()
built-in functions must not be used.
The cbrt()
, cospi()
, erf()
, erfc()
, expm1()
, fdim()
,
hypot()
, ilogb()
, lgamma()
, lgamma_r()
, log1p()
, logb()
, maxmag()
,
minmag()
, nan()
, nextafter()
, pown()
, remainder()
, remquo()
,
rint()
, rootn()
, sincos()
, sinpi()
, tanpi()
, and tgamma()
built-in
functions must not be used.
The add_sat()
, hadd()
, mad_sat()
, rhadd()
and sub_sat()
built-in
functions must not be used.
The islessgreater()
, isfinite()
, isnormal()
, isordered()
and
isunordered()
built-in functions must not be used.
The vload<size>()
, vstore<size>()
, vstore_half_rtp()
, vstore_half_rtn()
,
vstore_half<size>_rtp()
, vstore_half<size>_rtn()
, vstorea_half<size>_rtp()
, and vstorea_half<size>_rtn()
built-in functions must not be used.
The vload_half()
, vload_half<size>()
, vstore_half()
, vstore_half_rte()
,
vstore_half_rtz()
, vstore_half<size>()
, vstore_half<size>_rte()
,
vstore_half<size>_rtz()
, and vloada_half<size>()
built-in functions
are only allowed to use the global
and constant
address spaces.
Note: When 16-bit storage support is not assumed, both vload_half
and
vstore_half
assume the pointers are aligned to 4 bytes, not 2 bytes.
See issue 6.
Builtin functions
vstorea_half2()
,
vstorea_half4()
,
vstorea_half2_rtz()
,
vstorea_half4_rtz()
,
vstorea_half2_rte()
,
and
vstorea_half4_rte()
built-in functions
have implementations for global, local, and private
address spaces.
The vstore_half_rte()
, vstore_half_rtz()
, vstore_half<size>_rte()
,
vstore_half<size>_rtz()
, vstorea_half<size>_rte()
, and
vstorea_half<size>_rtz()
built-in functions are not guaranteed to round the
result correctly if the destination address was not declared as a half*
on the
kernel entry point.
The async_work_group_copy()
, async_work_group_strided_copy()
,
wait_group_events()
, and prefetch()
built-in functions must not be used.
The shuffle()
, shuffle2()
and vec_step()
built-in functions must not
be used.
The printf()
built-in function must not be used.
The get_image_channel_data_type()
and get_image_channel_order()
built-in functions must not be used.