Files
OpenCL-CTS/test_conformance/spirv_new/test_spirv_14.cpp
Ben Ashbaugh bc5f6cdfe8 add SPIR-V 1.4 testing for OpPtrEqual, OpPtrNotEqual, OpPtrDiff (#2054)
This PR adds targeted testing for the SPIR-V 1.4 instructions
OpPtrEqual, OpPtrNotEqual, and OpPtrDiff.
2024-10-29 09:36:26 -07:00

289 lines
10 KiB
C++

//
// Copyright (c) 2024 The Khronos Group Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
#include "testBase.h"
#include "spirvInfo.hpp"
#include "types.hpp"
#include <string>
#include <vector>
static int test_image_operand_helper(cl_device_id deviceID, cl_context context,
cl_command_queue queue, bool signExtend)
{
const char* filename = signExtend ? "spv1.4/image_operand_signextend"
: "spv1.4/image_operand_zeroextend";
cl_image_format image_format = {
CL_RGBA,
signExtend ? CL_SIGNED_INT8 : CL_UNSIGNED_INT8,
};
cl_int error = CL_SUCCESS;
std::vector<cl_uchar> imgData({ 0x1, 0x80, 0xFF, 0x0 });
std::vector<cl_uint> expected;
for (auto v : imgData)
{
if (signExtend)
{
expected.push_back((cl_int)(cl_char)v);
}
else
{
expected.push_back(v);
}
}
clProgramWrapper prog;
error = get_program_with_il(prog, deviceID, context, filename);
SPIRV_CHECK_ERROR(error, "Failed to compile spv program");
clKernelWrapper kernel = clCreateKernel(prog, "read_image_test", &error);
SPIRV_CHECK_ERROR(error, "Failed to create spv kernel");
std::vector<cl_uint> h_dst({ 0, 0, 0, 0 });
clMemWrapper dst =
clCreateBuffer(context, CL_MEM_COPY_HOST_PTR,
h_dst.size() * sizeof(cl_uint), h_dst.data(), &error);
SPIRV_CHECK_ERROR(error, "Failed to create dst buffer");
clMemWrapper src =
clCreateImage2D(context, CL_MEM_READ_ONLY | CL_MEM_COPY_HOST_PTR,
&image_format, 1, 1, 0, imgData.data(), &error);
SPIRV_CHECK_ERROR(error, "Failed to create src image");
error |= clSetKernelArg(kernel, 0, sizeof(dst), &dst);
error |= clSetKernelArg(kernel, 1, sizeof(src), &src);
SPIRV_CHECK_ERROR(error, "Failed to set kernel args");
size_t global = 1;
error = clEnqueueNDRangeKernel(queue, kernel, 1, NULL, &global, NULL, 0,
NULL, NULL);
SPIRV_CHECK_ERROR(error, "Failed to enqueue kernel");
error = clEnqueueReadBuffer(queue, dst, CL_TRUE, 0,
h_dst.size() * sizeof(cl_uint), h_dst.data(), 0,
NULL, NULL);
SPIRV_CHECK_ERROR(error, "Unable to read destination buffer");
if (h_dst != expected)
{
log_error("Mismatch! Got: %u, %u, %u, %u, Wanted: %u, %u, %u, %u\n",
h_dst[0], h_dst[1], h_dst[2], h_dst[3], expected[0],
expected[1], expected[2], expected[3]);
return TEST_FAIL;
}
return TEST_PASS;
}
TEST_SPIRV_FUNC(spirv14_image_operand_signextend)
{
if (!is_spirv_version_supported(deviceID, "SPIR-V_1.4"))
{
log_info("SPIR-V 1.4 not supported; skipping tests.\n");
return TEST_SKIPPED_ITSELF;
}
return test_image_operand_helper(deviceID, context, queue, true);
}
TEST_SPIRV_FUNC(spirv14_image_operand_zeroextend)
{
if (!is_spirv_version_supported(deviceID, "SPIR-V_1.4"))
{
log_info("SPIR-V 1.4 not supported; skipping tests.\n");
return TEST_SKIPPED_ITSELF;
}
return test_image_operand_helper(deviceID, context, queue, false);
}
static int test_loop_control_helper(cl_device_id deviceID, cl_context context,
cl_command_queue queue,
const char* filename)
{
const int count = 10;
const int value = 5;
cl_int error = CL_SUCCESS;
clProgramWrapper prog;
std::string full_filename = "spv1.4/" + std::string(filename);
error = get_program_with_il(prog, deviceID, context, full_filename.c_str());
SPIRV_CHECK_ERROR(error, "Failed to compile spv program");
clKernelWrapper kernel = clCreateKernel(prog, "loop_control_test", &error);
SPIRV_CHECK_ERROR(error, "Failed to create spv kernel");
int h_dst = 0;
clMemWrapper dst = clCreateBuffer(context, CL_MEM_COPY_HOST_PTR,
sizeof(h_dst), &h_dst, &error);
SPIRV_CHECK_ERROR(error, "Failed to create dst buffer");
error |= clSetKernelArg(kernel, 0, sizeof(dst), &dst);
error |= clSetKernelArg(kernel, 1, sizeof(count), &count);
error |= clSetKernelArg(kernel, 2, sizeof(value), &value);
SPIRV_CHECK_ERROR(error, "Failed to set kernel args");
size_t global = 1;
error = clEnqueueNDRangeKernel(queue, kernel, 1, NULL, &global, NULL, 0,
NULL, NULL);
SPIRV_CHECK_ERROR(error, "Failed to enqueue kernel");
error = clEnqueueReadBuffer(queue, dst, CL_TRUE, 0, sizeof(h_dst), &h_dst,
0, NULL, NULL);
SPIRV_CHECK_ERROR(error, "Unable to read destination buffer");
if (h_dst != count * value)
{
log_error("Mismatch! Got: %i, Wanted: %i\n", h_dst, count * value);
return TEST_FAIL;
}
return TEST_PASS;
}
TEST_SPIRV_FUNC(spirv14_loop_control_miniterations)
{
if (!is_spirv_version_supported(deviceID, "SPIR-V_1.4"))
{
log_info("SPIR-V 1.4 not supported; skipping tests.\n");
return TEST_SKIPPED_ITSELF;
}
return test_loop_control_helper(deviceID, context, queue,
"loop_control_miniterations");
}
TEST_SPIRV_FUNC(spirv14_loop_control_maxiterations)
{
if (!is_spirv_version_supported(deviceID, "SPIR-V_1.4"))
{
log_info("SPIR-V 1.4 not supported; skipping tests.\n");
return TEST_SKIPPED_ITSELF;
}
return test_loop_control_helper(deviceID, context, queue,
"loop_control_maxiterations");
}
TEST_SPIRV_FUNC(spirv14_loop_control_iterationmultiple)
{
if (!is_spirv_version_supported(deviceID, "SPIR-V_1.4"))
{
log_info("SPIR-V 1.4 not supported; skipping tests.\n");
return TEST_SKIPPED_ITSELF;
}
return test_loop_control_helper(deviceID, context, queue,
"loop_control_iterationmultiple");
}
TEST_SPIRV_FUNC(spirv14_loop_control_peelcount)
{
if (!is_spirv_version_supported(deviceID, "SPIR-V_1.4"))
{
log_info("SPIR-V 1.4 not supported; skipping tests.\n");
return TEST_SKIPPED_ITSELF;
}
return test_loop_control_helper(deviceID, context, queue,
"loop_control_peelcount");
}
TEST_SPIRV_FUNC(spirv14_loop_control_partialcount)
{
if (!is_spirv_version_supported(deviceID, "SPIR-V_1.4"))
{
log_info("SPIR-V 1.4 not supported; skipping tests.\n");
return TEST_SKIPPED_ITSELF;
}
return test_loop_control_helper(deviceID, context, queue,
"loop_control_partialcount");
}
TEST_SPIRV_FUNC(spirv14_ptrops)
{
if (!is_spirv_version_supported(deviceID, "SPIR-V_1.4"))
{
log_info("SPIR-V 1.4 not supported; skipping tests.\n");
return TEST_SKIPPED_ITSELF;
}
cl_int error = CL_SUCCESS;
clProgramWrapper prog;
error = get_program_with_il(prog, deviceID, context, "spv1.4/ptrops");
SPIRV_CHECK_ERROR(error, "Failed to compile spv program");
clKernelWrapper kernel = clCreateKernel(prog, "ptrops_test", &error);
SPIRV_CHECK_ERROR(error, "Failed to create spv kernel");
std::vector<cl_int> results(3);
clMemWrapper dst =
clCreateBuffer(context, CL_MEM_READ_WRITE,
results.size() * sizeof(cl_int), NULL, &error);
SPIRV_CHECK_ERROR(error, "Failed to create dst buffer");
clMemWrapper tst = clCreateBuffer(context, CL_MEM_READ_WRITE,
sizeof(cl_int), NULL, &error);
SPIRV_CHECK_ERROR(error, "Failed to create tst buffer");
// Test with different pointers:
error |= clSetKernelArg(kernel, 0, sizeof(dst), &dst);
error |= clSetKernelArg(kernel, 1, sizeof(tst), &tst);
SPIRV_CHECK_ERROR(error, "Failed to set kernel args");
size_t global = 1;
error = clEnqueueNDRangeKernel(queue, kernel, 1, NULL, &global, NULL, 0,
NULL, NULL);
SPIRV_CHECK_ERROR(error, "Failed to enqueue kernel");
error = clEnqueueReadBuffer(queue, dst, CL_TRUE, 0,
results.size() * sizeof(cl_int), results.data(),
0, NULL, NULL);
SPIRV_CHECK_ERROR(error, "Unable to read destination buffer");
if (results[0] != (dst == tst) || results[1] != (dst != tst)
|| results[2] == 0 /* dst - tst */)
{
log_error(
"Results mismatch with different pointers! Got: %i, %i, %i\n",
results[0], results[1], results[2]);
return TEST_FAIL;
}
// Test with equal pointers:
error |= clSetKernelArg(kernel, 0, sizeof(dst), &dst);
error |= clSetKernelArg(kernel, 1, sizeof(dst), &dst);
SPIRV_CHECK_ERROR(error, "Failed to set kernel args");
error = clEnqueueNDRangeKernel(queue, kernel, 1, NULL, &global, NULL, 0,
NULL, NULL);
SPIRV_CHECK_ERROR(error, "Failed to enqueue kernel");
error = clEnqueueReadBuffer(queue, dst, CL_TRUE, 0,
results.size() * sizeof(cl_int), results.data(),
0, NULL, NULL);
SPIRV_CHECK_ERROR(error, "Unable to read destination buffer");
if (results[0] != (dst == dst) || results[1] != (dst != dst)
|| results[2] != 0 /* dst - dst */)
{
log_error("Results mismatch with equal pointers! Got: %i, %i, %i\n",
results[0], results[1], results[2]);
return TEST_FAIL;
}
return TEST_PASS;
}