1 //
2 // Copyright (c) 2017 The Khronos Group Inc.
3 //
4 // Licensed under the Apache License, Version 2.0 (the "License");
5 // you may not use this file except in compliance with the License.
6 // You may obtain a copy of the License at
7 //
8 // http://www.apache.org/licenses/LICENSE-2.0
9 //
10 // Unless required by applicable law or agreed to in writing, software
11 // distributed under the License is distributed on an "AS IS" BASIS,
12 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 // See the License for the specific language governing permissions and
14 // limitations under the License.
15 //
16 #include <stdio.h>
17 #if defined(__APPLE__)
18 #include <OpenCL/opencl.h>
19 #include <OpenCL/cl_platform.h>
20 #else
21 #include <CL/opencl.h>
22 #include <CL/cl_platform.h>
23 #endif
24 #include "testBase.h"
25 #include "harness/typeWrappers.h"
26 #include "harness/testHarness.h"
27 #include "procs.h"
28
29
30 enum { SUCCESS, FAILURE };
31 typedef enum { NON_NULL_PATH, ADDROF_NULL_PATH, NULL_PATH } test_type;
32
33 #define NITEMS 4096
34
35 /* places the comparison result of value of the src ptr against 0 into each element of the output
36 * array, to allow testing that the kernel actually _gets_ the NULL value */
37 const char *kernel_string_long =
38 "kernel void test_kernel(global float *src, global long *dst)\n"
39 "{\n"
40 " uint tid = get_global_id(0);\n"
41 " dst[tid] = (long)(src != 0);\n"
42 "}\n";
43
44 // For gIsEmbedded
45 const char *kernel_string =
46 "kernel void test_kernel(global float *src, global int *dst)\n"
47 "{\n"
48 " uint tid = get_global_id(0);\n"
49 " dst[tid] = (int)(src != 0);\n"
50 "}\n";
51
52
53 /*
54 * The guts of the test:
55 * call setKernelArgs with a regular buffer, &NULL, or NULL depending on
56 * the value of 'test_type'
57 */
test_setargs_and_execution(cl_command_queue queue,cl_kernel kernel,cl_mem test_buf,cl_mem result_buf,test_type type)58 static int test_setargs_and_execution(cl_command_queue queue, cl_kernel kernel,
59 cl_mem test_buf, cl_mem result_buf, test_type type)
60 {
61 unsigned int test_success = 0;
62
63 unsigned int i;
64 cl_int status;
65 const char *typestr;
66
67 if (type == NON_NULL_PATH) {
68 status = clSetKernelArg(kernel, 0, sizeof(cl_mem), &test_buf);
69 typestr = "non-NULL";
70 } else if (type == ADDROF_NULL_PATH) {
71 test_buf = NULL;
72 status = clSetKernelArg(kernel, 0, sizeof(cl_mem), &test_buf);
73 typestr = "&NULL";
74 } else if (type == NULL_PATH) {
75 status = clSetKernelArg(kernel, 0, sizeof(cl_mem), NULL);
76 typestr = "NULL";
77 }
78
79 log_info("Testing setKernelArgs with %s buffer.\n", typestr);
80
81 if (status != CL_SUCCESS) {
82 log_error("clSetKernelArg failed with status: %d\n", status);
83 return FAILURE; // no point in continuing *this* test
84 }
85
86 size_t global = NITEMS;
87 status = clEnqueueNDRangeKernel(queue, kernel, 1, NULL, &global,
88 NULL, 0, NULL, NULL);
89 test_error(status, "NDRangeKernel failed.");
90
91 if (gIsEmbedded)
92 {
93 cl_int* host_result = (cl_int*)malloc(NITEMS*sizeof(cl_int));
94 status = clEnqueueReadBuffer(queue, result_buf, CL_TRUE, 0,
95 sizeof(cl_int)*NITEMS, host_result, 0, NULL, NULL);
96 test_error(status, "ReadBuffer failed.");
97 // in the non-null case, we expect NONZERO values:
98 if (type == NON_NULL_PATH) {
99 for (i=0; i<NITEMS; i++) {
100 if (host_result[i] == 0) {
101 log_error("failure: item %d in the result buffer was unexpectedly NULL.\n", i);
102 test_success = FAILURE; break;
103 }
104 }
105
106 } else if (type == ADDROF_NULL_PATH || type == NULL_PATH) {
107 for (i=0; i<NITEMS; i++) {
108 if (host_result[i] != 0) {
109 log_error("failure: item %d in the result buffer was unexpectedly non-NULL.\n", i);
110 test_success = FAILURE; break;
111 }
112 }
113 }
114 free(host_result);
115 }
116 else
117 {
118 cl_long* host_result = (cl_long*)malloc(NITEMS*sizeof(cl_long));
119 status = clEnqueueReadBuffer(queue, result_buf, CL_TRUE, 0,
120 sizeof(cl_long)*NITEMS, host_result, 0, NULL, NULL);
121 test_error(status, "ReadBuffer failed.");
122 // in the non-null case, we expect NONZERO values:
123 if (type == NON_NULL_PATH) {
124 for (i=0; i<NITEMS; i++) {
125 if (host_result[i] == 0) {
126 log_error("failure: item %d in the result buffer was unexpectedly NULL.\n", i);
127 test_success = FAILURE; break;
128 }
129 }
130 } else if (type == ADDROF_NULL_PATH || type == NULL_PATH) {
131 for (i=0; i<NITEMS; i++) {
132 if (host_result[i] != 0) {
133 log_error("failure: item %d in the result buffer was unexpectedly non-NULL.\n", i);
134 test_success = FAILURE; break;
135 }
136 }
137 }
138 free(host_result);
139 }
140
141 if (test_success == SUCCESS) {
142 log_info("\t%s ok.\n", typestr);
143 }
144
145 return test_success;
146 }
147
test_null_buffer_arg(cl_device_id device,cl_context context,cl_command_queue queue,int num_elements)148 int test_null_buffer_arg(cl_device_id device, cl_context context,
149 cl_command_queue queue, int num_elements)
150 {
151 unsigned int test_success = 0;
152 unsigned int i;
153 unsigned int buffer_size;
154 cl_int status;
155 cl_program program;
156 cl_kernel kernel;
157
158 // prep kernel:
159 if (gIsEmbedded)
160 status = create_single_kernel_helper(context, &program, &kernel, 1,
161 &kernel_string, "test_kernel");
162 else
163 status = create_single_kernel_helper(
164 context, &program, &kernel, 1, &kernel_string_long, "test_kernel");
165
166 test_error(status, "Unable to create kernel");
167
168 cl_mem dev_src = clCreateBuffer(context, CL_MEM_READ_ONLY, NITEMS*sizeof(cl_float),
169 NULL, NULL);
170
171 if (gIsEmbedded)
172 buffer_size = NITEMS*sizeof(cl_int);
173 else
174 buffer_size = NITEMS*sizeof(cl_long);
175
176 cl_mem dev_dst = clCreateBuffer(context, CL_MEM_WRITE_ONLY, buffer_size,
177 NULL, NULL);
178
179 // set the destination buffer normally:
180 status = clSetKernelArg(kernel, 1, sizeof(cl_mem), &dev_dst);
181 test_error(status, "SetKernelArg failed.");
182
183 //
184 // we test three cases:
185 //
186 // - typical case, used everyday: non-null buffer
187 // - the case of src as &NULL (the spec-compliance test)
188 // - the case of src as NULL (the backwards-compatibility test, Apple only)
189 //
190
191 test_success = test_setargs_and_execution(queue, kernel, dev_src, dev_dst, NON_NULL_PATH);
192 test_success |= test_setargs_and_execution(queue, kernel, dev_src, dev_dst, ADDROF_NULL_PATH);
193
194 #ifdef __APPLE__
195 test_success |= test_setargs_and_execution(queue, kernel, dev_src, dev_dst, NULL_PATH);
196 #endif
197
198 // clean up:
199 if (dev_src) clReleaseMemObject(dev_src);
200 clReleaseMemObject(dev_dst);
201 clReleaseKernel(kernel);
202 clReleaseProgram(program);
203
204 return test_success;
205 }
206