// // Copyright (c) 2017 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 "harness/compat.h" #include "harness/imageHelpers.h" #include #include #include #include #include #include #include using test_function_t = int (*)(cl_device_id, cl_context, cl_command_queue, cl_mem_flags, cl_mem_flags, cl_mem_object_type, const cl_image_format *); static int test_arrayimagecopy_single_format( cl_device_id device, cl_context context, cl_command_queue queue, cl_mem_flags buffer_flags, cl_mem_flags image_flags, cl_mem_object_type image_type, const cl_image_format *format) { std::unique_ptr bufptr{ nullptr, free }, imgptr{ nullptr, free }; clMemWrapper buffer, image; int img_width = 512; int img_height = 512; int img_depth = (image_type == CL_MEM_OBJECT_IMAGE3D) ? 32 : 1; size_t elem_size; size_t buffer_size; cl_int err; cl_event copyevent; RandomSeed seed(gRandomSeed); log_info("Testing %s %s\n", GetChannelOrderName(format->image_channel_order), GetChannelTypeName(format->image_channel_data_type)); elem_size = get_pixel_size(format); buffer_size = sizeof(cl_uchar) * elem_size * img_width * img_height * img_depth; if (image_flags & CL_MEM_USE_HOST_PTR || image_flags & CL_MEM_COPY_HOST_PTR) { imgptr.reset(static_cast( create_random_data(kUChar, seed, buffer_size))); } bufptr.reset( static_cast(create_random_data(kUChar, seed, buffer_size))); if (CL_MEM_OBJECT_IMAGE2D == image_type) { image = create_image_2d(context, image_flags, format, img_width, img_height, 0, imgptr.get(), &err); } else { image = create_image_3d(context, image_flags, format, img_width, img_height, img_depth, 0, 0, imgptr.get(), &err); } test_error(err, "create_image_xd failed"); if (buffer_flags & CL_MEM_USE_HOST_PTR || buffer_flags & CL_MEM_COPY_HOST_PTR) { buffer = clCreateBuffer(context, buffer_flags, buffer_size, bufptr.get(), &err); test_error(err, "clCreateBuffer failed"); } else { buffer = clCreateBuffer(context, buffer_flags, buffer_size, nullptr, &err); test_error(err, "clCreateBuffer failed"); err = clEnqueueWriteBuffer(queue, buffer, CL_TRUE, 0, buffer_size, bufptr.get(), 0, nullptr, nullptr); test_error(err, "clEnqueueWriteBuffer failed"); } size_t origin[3] = { 0, 0, 0 }, region[3] = { img_width, img_height, img_depth }; err = clEnqueueCopyBufferToImage(queue, buffer, image, 0, origin, region, 0, nullptr, ©event); test_error(err, "clEnqueueCopyImageToBuffer failed"); imgptr.reset(static_cast(malloc(buffer_size))); err = clEnqueueReadImage(queue, image, CL_TRUE, origin, region, 0, 0, imgptr.get(), 1, ©event, nullptr); test_error(err, "clEnqueueReadImage failed"); err = clReleaseEvent(copyevent); test_error(err, "clReleaseEvent failed"); if (memcmp(bufptr.get(), imgptr.get(), buffer_size) != 0) { log_error("ERROR: Results did not validate!\n"); auto inchar = static_cast(bufptr.get()); auto outchar = static_cast(imgptr.get()); int failuresPrinted = 0; for (int i = 0; i < (int)buffer_size; i += (int)elem_size) { if (memcmp(&inchar[i], &outchar[i], elem_size) != 0) { log_error("%d(0x%x) -> actual [", i, i); for (int j = 0; j < (int)elem_size; j++) log_error("0x%02x ", inchar[i + j]); log_error("] != expected ["); for (int j = 0; j < (int)elem_size; j++) log_error("0x%02x ", outchar[i + j]); log_error("]\n"); failuresPrinted++; } if (failuresPrinted > 5) { log_error("Not printing further failures...\n"); break; } } err = -1; } if (err) log_error( "ARRAY to IMAGE copy test failed for image_channel_order=0x%lx and " "image_channel_data_type=0x%lx\n", (unsigned long)format->image_channel_order, (unsigned long)format->image_channel_data_type); return err; } static int test_arrayimagecommon(cl_device_id device, cl_context context, cl_command_queue queue, cl_mem_flags buffer_flags, cl_mem_flags image_flags, cl_mem_object_type image_type, test_function_t test_function) { cl_int err; cl_uint num_formats; err = clGetSupportedImageFormats(context, image_flags, image_type, 0, nullptr, &num_formats); test_error(err, "clGetSupportedImageFormats failed"); std::vector formats(num_formats); err = clGetSupportedImageFormats(context, image_flags, image_type, num_formats, formats.data(), nullptr); test_error(err, "clGetSupportedImageFormats failed"); for (const auto &format : formats) { err |= test_function(device, context, queue, buffer_flags, image_flags, image_type, &format); } if (err) log_error("ARRAY to IMAGE%s copy test failed\n", convert_image_type_to_string(image_type)); else log_info("ARRAY to IMAGE%s copy test passed\n", convert_image_type_to_string(image_type)); return err; } REGISTER_TEST(arrayimagecopy) { PASSIVE_REQUIRE_IMAGE_SUPPORT(device) return test_arrayimagecommon(device, context, queue, CL_MEM_READ_WRITE, CL_MEM_READ_WRITE, CL_MEM_OBJECT_IMAGE2D, test_arrayimagecopy_single_format); } REGISTER_TEST(arrayimagecopy3d) { PASSIVE_REQUIRE_3D_IMAGE_SUPPORT(device) return test_arrayimagecommon(device, context, queue, CL_MEM_READ_WRITE, CL_MEM_READ_ONLY, CL_MEM_OBJECT_IMAGE3D, test_arrayimagecopy_single_format); }