2017-12-21 07:45:38 +08:00
|
|
|
/*
|
2021-05-17 02:51:16 +08:00
|
|
|
* Copyright (C) 2018-2021 Intel Corporation
|
2017-12-21 07:45:38 +08:00
|
|
|
*
|
2018-09-18 15:11:08 +08:00
|
|
|
* SPDX-License-Identifier: MIT
|
2017-12-21 07:45:38 +08:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2020-02-24 05:44:01 +08:00
|
|
|
#include "shared/source/compiler_interface/compiler_interface.h"
|
|
|
|
#include "shared/source/helpers/file_io.h"
|
2021-01-21 20:10:13 +08:00
|
|
|
#include "shared/test/common/helpers/test_files.h"
|
2020-02-24 17:22:30 +08:00
|
|
|
|
2020-02-23 05:50:57 +08:00
|
|
|
#include "opencl/source/context/context.h"
|
2020-02-23 22:20:22 +08:00
|
|
|
#include "opencl/test/unit_test/helpers/kernel_binary_helper.h"
|
2019-02-27 18:39:32 +08:00
|
|
|
|
|
|
|
#include "cl_api_tests.h"
|
2017-12-21 07:45:38 +08:00
|
|
|
|
2019-03-26 18:59:46 +08:00
|
|
|
using namespace NEO;
|
2017-12-21 07:45:38 +08:00
|
|
|
|
|
|
|
typedef api_tests clCompileProgramTests;
|
|
|
|
|
|
|
|
namespace ULT {
|
|
|
|
|
2018-09-12 21:33:37 +08:00
|
|
|
TEST_F(clCompileProgramTests, GivenKernelAsSingleSourceWhenCompilingProgramThenSuccessIsReturned) {
|
2017-12-21 07:45:38 +08:00
|
|
|
cl_program pProgram = nullptr;
|
|
|
|
size_t sourceSize = 0;
|
|
|
|
std::string testFile;
|
|
|
|
|
|
|
|
KernelBinaryHelper kbHelper("copybuffer", false);
|
|
|
|
testFile.append(clFiles);
|
|
|
|
testFile.append("copybuffer.cl");
|
|
|
|
|
2019-08-29 21:10:51 +08:00
|
|
|
auto pSource = loadDataFromFile(
|
2017-12-21 07:45:38 +08:00
|
|
|
testFile.c_str(),
|
2019-08-29 21:10:51 +08:00
|
|
|
sourceSize);
|
2017-12-21 07:45:38 +08:00
|
|
|
|
|
|
|
ASSERT_NE(0u, sourceSize);
|
|
|
|
ASSERT_NE(nullptr, pSource);
|
|
|
|
|
2019-08-29 21:10:51 +08:00
|
|
|
const char *sources[1] = {pSource.get()};
|
2017-12-21 07:45:38 +08:00
|
|
|
pProgram = clCreateProgramWithSource(
|
|
|
|
pContext,
|
|
|
|
1,
|
2019-08-29 21:10:51 +08:00
|
|
|
sources,
|
2017-12-21 07:45:38 +08:00
|
|
|
&sourceSize,
|
|
|
|
&retVal);
|
|
|
|
|
|
|
|
EXPECT_NE(nullptr, pProgram);
|
|
|
|
ASSERT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
2020-04-24 18:58:39 +08:00
|
|
|
1,
|
|
|
|
&testedClDevice,
|
2017-12-21 07:45:38 +08:00
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
ASSERT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clReleaseProgram(pProgram);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
}
|
|
|
|
|
2018-09-12 21:33:37 +08:00
|
|
|
TEST_F(clCompileProgramTests, GivenKernelAsSourceWithHeaderWhenCompilingProgramThenSuccessIsReturned) {
|
2017-12-21 07:45:38 +08:00
|
|
|
cl_program pProgram = nullptr;
|
|
|
|
cl_program pHeader = nullptr;
|
|
|
|
size_t sourceSize = 0;
|
|
|
|
std::string testFile;
|
|
|
|
const char *simpleHeaderName = "simple_header.h";
|
|
|
|
|
|
|
|
testFile.append(clFiles);
|
|
|
|
testFile.append("/copybuffer_with_header.cl");
|
|
|
|
|
2019-08-29 21:10:51 +08:00
|
|
|
auto pSource = loadDataFromFile(
|
2017-12-21 07:45:38 +08:00
|
|
|
testFile.c_str(),
|
2019-08-29 21:10:51 +08:00
|
|
|
sourceSize);
|
2017-12-21 07:45:38 +08:00
|
|
|
|
|
|
|
ASSERT_NE(0u, sourceSize);
|
|
|
|
ASSERT_NE(nullptr, pSource);
|
|
|
|
|
2019-08-29 21:10:51 +08:00
|
|
|
const char *sources[1] = {pSource.get()};
|
2017-12-21 07:45:38 +08:00
|
|
|
pProgram = clCreateProgramWithSource(
|
|
|
|
pContext,
|
|
|
|
1,
|
2019-08-29 21:10:51 +08:00
|
|
|
sources,
|
2017-12-21 07:45:38 +08:00
|
|
|
&sourceSize,
|
|
|
|
&retVal);
|
|
|
|
|
|
|
|
EXPECT_NE(nullptr, pProgram);
|
|
|
|
ASSERT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
testFile.clear();
|
|
|
|
testFile.append(clFiles);
|
|
|
|
testFile.append("simple_header.h");
|
2019-08-29 21:10:51 +08:00
|
|
|
auto pHeaderSource = loadDataFromFile(
|
2017-12-21 07:45:38 +08:00
|
|
|
testFile.c_str(),
|
2019-08-29 21:10:51 +08:00
|
|
|
sourceSize);
|
2017-12-21 07:45:38 +08:00
|
|
|
|
|
|
|
ASSERT_NE(0u, sourceSize);
|
|
|
|
ASSERT_NE(nullptr, pHeaderSource);
|
|
|
|
|
2019-08-29 21:10:51 +08:00
|
|
|
const char *headerSources[1] = {pHeaderSource.get()};
|
2017-12-21 07:45:38 +08:00
|
|
|
pHeader = clCreateProgramWithSource(
|
|
|
|
pContext,
|
|
|
|
1,
|
2019-08-29 21:10:51 +08:00
|
|
|
headerSources,
|
2017-12-21 07:45:38 +08:00
|
|
|
&sourceSize,
|
|
|
|
&retVal);
|
|
|
|
|
|
|
|
EXPECT_NE(nullptr, pHeader);
|
|
|
|
ASSERT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
2020-04-24 18:58:39 +08:00
|
|
|
1,
|
|
|
|
&testedClDevice,
|
2017-12-21 07:45:38 +08:00
|
|
|
nullptr,
|
|
|
|
1,
|
|
|
|
&pHeader,
|
|
|
|
&simpleHeaderName,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
ASSERT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clReleaseProgram(pProgram);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clReleaseProgram(pHeader);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
}
|
|
|
|
|
2018-09-12 21:33:37 +08:00
|
|
|
TEST_F(clCompileProgramTests, GivenNullProgramWhenCompilingProgramThenInvalidProgramErrorIsReturned) {
|
2017-12-21 07:45:38 +08:00
|
|
|
retVal = clCompileProgram(
|
|
|
|
nullptr,
|
|
|
|
1,
|
|
|
|
nullptr,
|
|
|
|
"",
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
EXPECT_EQ(CL_INVALID_PROGRAM, retVal);
|
|
|
|
}
|
2020-10-27 19:43:48 +08:00
|
|
|
|
|
|
|
TEST_F(clCompileProgramTests, GivenInvalidCallbackInputWhenCompileProgramThenInvalidValueErrorIsReturned) {
|
|
|
|
cl_program pProgram = nullptr;
|
|
|
|
size_t sourceSize = 0;
|
|
|
|
std::string testFile;
|
|
|
|
|
|
|
|
testFile.append(clFiles);
|
|
|
|
testFile.append("copybuffer.cl");
|
|
|
|
auto pSource = loadDataFromFile(
|
|
|
|
testFile.c_str(),
|
|
|
|
sourceSize);
|
|
|
|
|
|
|
|
ASSERT_NE(0u, sourceSize);
|
|
|
|
ASSERT_NE(nullptr, pSource);
|
|
|
|
|
|
|
|
const char *sources[1] = {pSource.get()};
|
|
|
|
pProgram = clCreateProgramWithSource(
|
|
|
|
pContext,
|
|
|
|
1,
|
|
|
|
sources,
|
|
|
|
&sourceSize,
|
|
|
|
&retVal);
|
|
|
|
|
|
|
|
EXPECT_NE(nullptr, pProgram);
|
|
|
|
ASSERT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
|
|
|
1,
|
|
|
|
&testedClDevice,
|
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
&retVal);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_INVALID_VALUE, retVal);
|
|
|
|
|
|
|
|
retVal = clReleaseProgram(pProgram);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(clCompileProgramTests, GivenValidCallbackInputWhenLinkProgramThenCallbackIsInvoked) {
|
|
|
|
cl_program pProgram = nullptr;
|
|
|
|
size_t sourceSize = 0;
|
|
|
|
std::string testFile;
|
|
|
|
|
|
|
|
testFile.append(clFiles);
|
|
|
|
testFile.append("copybuffer.cl");
|
|
|
|
auto pSource = loadDataFromFile(
|
|
|
|
testFile.c_str(),
|
|
|
|
sourceSize);
|
|
|
|
|
|
|
|
ASSERT_NE(0u, sourceSize);
|
|
|
|
ASSERT_NE(nullptr, pSource);
|
|
|
|
|
|
|
|
const char *sources[1] = {pSource.get()};
|
|
|
|
pProgram = clCreateProgramWithSource(
|
|
|
|
pContext,
|
|
|
|
1,
|
|
|
|
sources,
|
|
|
|
&sourceSize,
|
|
|
|
&retVal);
|
|
|
|
|
|
|
|
EXPECT_NE(nullptr, pProgram);
|
|
|
|
ASSERT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
char userData = 0;
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
|
|
|
1,
|
|
|
|
&testedClDevice,
|
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
notifyFuncProgram,
|
|
|
|
&userData);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
EXPECT_EQ('a', userData);
|
|
|
|
|
|
|
|
retVal = clReleaseProgram(pProgram);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
}
|
|
|
|
|
2020-11-23 21:51:25 +08:00
|
|
|
TEST(clCompileProgramTest, givenProgramWhenCompilingForInvalidDevicesInputThenInvalidDeviceErrorIsReturned) {
|
2020-10-28 17:27:14 +08:00
|
|
|
cl_program pProgram = nullptr;
|
|
|
|
std::unique_ptr<char[]> pSource = nullptr;
|
|
|
|
size_t sourceSize = 0;
|
|
|
|
std::string testFile;
|
|
|
|
|
|
|
|
KernelBinaryHelper kbHelper("CopyBuffer_simd16");
|
|
|
|
|
|
|
|
testFile.append(clFiles);
|
|
|
|
testFile.append("CopyBuffer_simd16.cl");
|
|
|
|
|
|
|
|
pSource = loadDataFromFile(
|
|
|
|
testFile.c_str(),
|
|
|
|
sourceSize);
|
|
|
|
|
|
|
|
ASSERT_NE(0u, sourceSize);
|
|
|
|
ASSERT_NE(nullptr, pSource);
|
|
|
|
|
|
|
|
const char *sources[1] = {pSource.get()};
|
|
|
|
|
|
|
|
MockUnrestrictiveContextMultiGPU context;
|
|
|
|
cl_int retVal = CL_INVALID_PROGRAM;
|
|
|
|
|
|
|
|
pProgram = clCreateProgramWithSource(
|
|
|
|
&context,
|
|
|
|
1,
|
|
|
|
sources,
|
|
|
|
&sourceSize,
|
|
|
|
&retVal);
|
|
|
|
|
|
|
|
EXPECT_NE(nullptr, pProgram);
|
|
|
|
ASSERT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
MockContext mockContext;
|
|
|
|
cl_device_id nullDeviceInput[] = {context.getDevice(0), nullptr};
|
|
|
|
cl_device_id notAssociatedDeviceInput[] = {mockContext.getDevice(0)};
|
|
|
|
cl_device_id validDeviceInput[] = {context.getDevice(0)};
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
|
|
|
0,
|
|
|
|
validDeviceInput,
|
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_INVALID_VALUE, retVal);
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
|
|
|
1,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_INVALID_VALUE, retVal);
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
|
|
|
2,
|
|
|
|
nullDeviceInput,
|
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_INVALID_DEVICE, retVal);
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
|
|
|
1,
|
|
|
|
notAssociatedDeviceInput,
|
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_INVALID_DEVICE, retVal);
|
|
|
|
|
|
|
|
retVal = clReleaseProgram(pProgram);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
}
|
2020-11-23 21:51:25 +08:00
|
|
|
|
|
|
|
TEST(clCompileProgramTest, givenMultiDeviceProgramWithCreatedKernelWhenCompilingThenInvalidOperationErrorIsReturned) {
|
|
|
|
|
|
|
|
MockSpecializedContext context;
|
|
|
|
cl_program pProgram = nullptr;
|
|
|
|
size_t sourceSize = 0;
|
|
|
|
cl_int retVal = CL_INVALID_PROGRAM;
|
|
|
|
std::string testFile;
|
|
|
|
|
|
|
|
testFile.append(clFiles);
|
|
|
|
testFile.append("copybuffer.cl");
|
|
|
|
auto pSource = loadDataFromFile(
|
|
|
|
testFile.c_str(),
|
|
|
|
sourceSize);
|
|
|
|
|
|
|
|
ASSERT_NE(0u, sourceSize);
|
|
|
|
ASSERT_NE(nullptr, pSource);
|
|
|
|
|
|
|
|
const char *sources[1] = {pSource.get()};
|
|
|
|
pProgram = clCreateProgramWithSource(
|
|
|
|
&context,
|
|
|
|
1,
|
|
|
|
sources,
|
|
|
|
&sourceSize,
|
|
|
|
&retVal);
|
|
|
|
|
|
|
|
EXPECT_NE(nullptr, pProgram);
|
|
|
|
ASSERT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
cl_device_id firstSubDevice = context.pSubDevice0;
|
|
|
|
cl_device_id secondSubDevice = context.pSubDevice1;
|
|
|
|
|
|
|
|
retVal = clBuildProgram(
|
|
|
|
pProgram,
|
|
|
|
1,
|
|
|
|
&firstSubDevice,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
auto kernel = clCreateKernel(pProgram, "fullCopy", &retVal);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
|
|
|
1,
|
|
|
|
&secondSubDevice,
|
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_INVALID_OPERATION, retVal);
|
|
|
|
|
|
|
|
retVal = clReleaseKernel(kernel);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
|
|
|
1,
|
|
|
|
&secondSubDevice,
|
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clReleaseProgram(pProgram);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(clCompileProgramTest, givenMultiDeviceProgramWithCreatedKernelsWhenCompilingThenInvalidOperationErrorIsReturned) {
|
|
|
|
|
|
|
|
MockSpecializedContext context;
|
|
|
|
cl_program pProgram = nullptr;
|
|
|
|
size_t sourceSize = 0;
|
|
|
|
cl_int retVal = CL_INVALID_PROGRAM;
|
|
|
|
std::string testFile;
|
|
|
|
|
|
|
|
testFile.append(clFiles);
|
|
|
|
testFile.append("copybuffer.cl");
|
|
|
|
auto pSource = loadDataFromFile(
|
|
|
|
testFile.c_str(),
|
|
|
|
sourceSize);
|
|
|
|
|
|
|
|
ASSERT_NE(0u, sourceSize);
|
|
|
|
ASSERT_NE(nullptr, pSource);
|
|
|
|
|
|
|
|
const char *sources[1] = {pSource.get()};
|
|
|
|
pProgram = clCreateProgramWithSource(
|
|
|
|
&context,
|
|
|
|
1,
|
|
|
|
sources,
|
|
|
|
&sourceSize,
|
|
|
|
&retVal);
|
|
|
|
|
|
|
|
EXPECT_NE(nullptr, pProgram);
|
|
|
|
ASSERT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
cl_device_id firstSubDevice = context.pSubDevice0;
|
|
|
|
cl_device_id secondSubDevice = context.pSubDevice1;
|
|
|
|
|
|
|
|
retVal = clBuildProgram(
|
|
|
|
pProgram,
|
|
|
|
1,
|
|
|
|
&firstSubDevice,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
size_t numKernels = 0;
|
|
|
|
retVal = clGetProgramInfo(pProgram, CL_PROGRAM_NUM_KERNELS, sizeof(numKernels), &numKernels, nullptr);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
auto kernels = std::make_unique<cl_kernel[]>(numKernels);
|
|
|
|
|
|
|
|
retVal = clCreateKernelsInProgram(pProgram, static_cast<cl_uint>(numKernels), kernels.get(), nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
|
|
|
1,
|
|
|
|
&secondSubDevice,
|
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_INVALID_OPERATION, retVal);
|
|
|
|
|
|
|
|
for (auto i = 0u; i < numKernels; i++) {
|
|
|
|
retVal = clReleaseKernel(kernels[i]);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
}
|
|
|
|
|
|
|
|
retVal = clCompileProgram(
|
|
|
|
pProgram,
|
|
|
|
1,
|
|
|
|
&secondSubDevice,
|
|
|
|
nullptr,
|
|
|
|
0,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
|
|
|
|
retVal = clReleaseProgram(pProgram);
|
|
|
|
EXPECT_EQ(CL_SUCCESS, retVal);
|
|
|
|
}
|
2017-12-21 07:45:38 +08:00
|
|
|
} // namespace ULT
|