/* * Copyright (C) 2017 The Android Open Source Project * * 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 "VtsHalRenderscriptV1_0TargetTest.h" /* * Create a Blur intrinsic with scriptIntrinsicCreate, and call * scriptSetTimeZone to make sure it is not crashing. * * Calls: elementCreate, scriptIntrinsicCreate, scriptSetTimeZone */ TEST_F(RenderscriptHidlTest, IntrinsicTest) { // uint8 Element element = context->elementCreate(DataType::UNSIGNED_8, DataKind::USER, false, 1); EXPECT_NE(Element(0), element); Script script = context->scriptIntrinsicCreate(ScriptIntrinsicID::ID_BLUR, element); EXPECT_NE(Script(0), script); context->scriptSetTimeZone(script, "UTF-8"); } /* * Create a user script “struct_test”, and verified the setters and getters work * for the global variables. * * Calls: scriptCCreate, scriptGetVarV, scriptSetVarI, scriptSetVarJ, * scriptSetVarF, scriptSetVarD, elementCreate, typeCreate, * allocationCreateTyped, scriptSetVarObj, scriptSetVarV, scriptSetVarVE */ TEST_F(RenderscriptHidlTest, ScriptVarTest) { hidl_vec bitcode; bitcode.setToExternal((uint8_t*)bitCode, bitCodeLength); Script script = context->scriptCCreate("struct_test", "/data/local/tmp/", bitcode); ASSERT_NE(Script(0), script); // arg tests context->scriptSetVarI(script, mExportVarIdx_var_int, 100); int resultI = 0; context->scriptGetVarV(script, mExportVarIdx_var_int, sizeof(int), [&](const hidl_vec& _data){ resultI = *((int*)_data.data()); }); EXPECT_EQ(100, resultI); context->scriptSetVarJ(script, mExportVarIdx_var_long, 101l); int resultJ = 0; context->scriptGetVarV(script, mExportVarIdx_var_long, sizeof(long), [&](const hidl_vec& _data){ resultJ = *((long*)_data.data()); }); EXPECT_EQ(101l, resultJ); context->scriptSetVarF(script, mExportVarIdx_var_float, 102.0f); int resultF = 0.0f; context->scriptGetVarV(script, mExportVarIdx_var_float, sizeof(float), [&](const hidl_vec& _data){ resultF = *((float*)_data.data()); }); EXPECT_EQ(102.0f, resultF); context->scriptSetVarD(script, mExportVarIdx_var_double, 103.0); int resultD = 0.0; context->scriptGetVarV(script, mExportVarIdx_var_double, sizeof(double), [&](const hidl_vec& _data){ resultD = *((double*)_data.data()); }); EXPECT_EQ(103.0, resultD); // float1 Element element = context->elementCreate(DataType::FLOAT_32, DataKind::USER, false, 1); ASSERT_NE(Element(0), element); // 128 x float1 Type type = context->typeCreate(element, 128, 0, 0, false, false, YuvFormat::YUV_NONE); ASSERT_NE(Type(0), type); // 128 x float1 Allocation allocationIn = context->allocationCreateTyped(type, AllocationMipmapControl::NONE, (int)AllocationUsageType::SCRIPT, (Ptr)nullptr); ASSERT_NE(Allocation(0), allocationIn); Allocation allocationOut = Allocation(0); context->scriptSetVarObj(script, mExportVarIdx_var_allocation, (ObjectBase)allocationIn); context->scriptGetVarV(script, mExportVarIdx_var_allocation, sizeof(ObjectBase), [&](const hidl_vec& _data){ allocationOut = (Allocation) *((ObjectBase*)_data.data()); }); EXPECT_EQ(allocationOut, allocationIn); uint32_t valueV = 104u; hidl_vec _dataV; _dataV.setToExternal((uint8_t*)&valueV, sizeof(uint32_t)); context->scriptSetVarV(script, mExportVarIdx_var_uint32_t, _dataV); uint32_t resultV = 0u; context->scriptGetVarV(script, mExportVarIdx_var_uint32_t, sizeof(uint32_t), [&](const hidl_vec& _data){ resultV = *((uint32_t*)_data.data()); }); EXPECT_EQ(104u, resultV); std::vector dataVE = {1000, 1001}; std::vector dimsVE = {1}; std::vector outVE(2); hidl_vec _dataVE; hidl_vec _dimsVE; _dataVE.setToExternal((uint8_t*)dataVE.data(), dataVE.size()*sizeof(int)); _dimsVE.setToExternal((uint32_t*)dimsVE.data(), dimsVE.size()); // intx2 to represent point2 which is {int, int} Element elementVE = context->elementCreate(DataType::SIGNED_32, DataKind::USER, false, 2); ASSERT_NE(Element(0), elementVE); context->scriptSetVarVE(script, mExportVarIdx_var_point2, _dataVE, elementVE, _dimsVE); context->scriptGetVarV(script, mExportVarIdx_var_point2, 2*sizeof(int), [&](const hidl_vec& _data){ outVE = std::vector( (int*)_data.data(), (int*)_data.data() + 2); }); EXPECT_EQ(1000, outVE[0]); EXPECT_EQ(1001, outVE[1]); } /* * Create a user script “struct_test”, and input and output Allocations. * Verified the foreach launch correctly for the invoke kernel. * * Calls: scriptCCreate, scriptInvoke, scriptGetVarV, scriptInvokeV */ TEST_F(RenderscriptHidlTest, ScriptInvokeTest) { hidl_vec bitcode; bitcode.setToExternal((uint8_t*)bitCode, bitCodeLength); Script script = context->scriptCCreate("struct_test", "/data/local/tmp/", bitcode); ASSERT_NE(Script(0), script); // invoke test int resultI = 0; long resultJ = 0l; float resultF = 0.0f; double resultD = 0.0; uint32_t resultV = 0u; std::vector resultVE(2); context->scriptInvoke(script, mExportFuncIdx_function); context->scriptGetVarV(script, mExportVarIdx_var_int, sizeof(int), [&](const hidl_vec& _data){ resultI = *((int*)_data.data()); }); context->scriptGetVarV(script, mExportVarIdx_var_long, sizeof(long), [&](const hidl_vec& _data){ resultJ = *((long*)_data.data()); }); context->scriptGetVarV(script, mExportVarIdx_var_float, sizeof(float), [&](const hidl_vec& _data){ resultF = *((float*)_data.data()); }); context->scriptGetVarV(script, mExportVarIdx_var_double, sizeof(double), [&](const hidl_vec& _data){ resultD = *((double*)_data.data()); }); context->scriptGetVarV(script, mExportVarIdx_var_uint32_t, sizeof(uint32_t), [&](const hidl_vec& _data){ resultV = *((uint32_t*)_data.data()); }); context->scriptGetVarV(script, mExportVarIdx_var_point2, 2*sizeof(int), [&](const hidl_vec& _data){ resultVE = std::vector( (int*)_data.data(), (int*)_data.data() + 2); }); EXPECT_EQ(1, resultI); EXPECT_EQ(2l, resultJ); EXPECT_EQ(3.0f, resultF); EXPECT_EQ(4.0, resultD); EXPECT_EQ(5u, resultV); EXPECT_EQ(6, resultVE[0]); EXPECT_EQ(7, resultVE[1]); // invokeV test int functionV_arg = 5; int functionV_res = 0; hidl_vec functionV_data; functionV_data.setToExternal((uint8_t*)&functionV_arg, sizeof(int)); context->scriptInvokeV(script, mExportFuncIdx_functionV, functionV_data); context->scriptGetVarV(script, mExportVarIdx_var_int, sizeof(int), [&](const hidl_vec& _data){ functionV_res = *((int*)_data.data()); }); EXPECT_EQ(5, functionV_res); } /* * Create a user script “struct_test”, and input and output Allocations. * Verified the foreach launch correctly for the foreach kernel. * * Calls: scriptCCreate, elementCreate, typeCreate, allocationCreateTyped, * allocation1DWrite, scriptForEach, allocationRead */ TEST_F(RenderscriptHidlTest, ScriptForEachTest) { hidl_vec bitcode; bitcode.setToExternal((uint8_t*)bitCode, bitCodeLength); Script script = context->scriptCCreate("struct_test", "/data/local/tmp/", bitcode); ASSERT_NE(Script(0), script); // uint8_t Element element = context->elementCreate(DataType::UNSIGNED_8, DataKind::USER, false, 1); ASSERT_NE(Element(0), element); // 64 x uint8_t Type type = context->typeCreate(element, 64, 0, 0, false, false, YuvFormat::YUV_NONE); ASSERT_NE(Type(0), type); std::vector dataIn(64), dataOut(64), expected(64); std::generate(dataIn.begin(), dataIn.end(), [](){ static uint8_t val = 0; return val++; }); std::generate(expected.begin(), expected.end(), [](){ static uint8_t val = 1; return val++; }); hidl_vec _data; _data.setToExternal((uint8_t*)dataIn.data(), dataIn.size()); // 64 x float1 Allocation allocation = context->allocationCreateTyped(type, AllocationMipmapControl::NONE, (int)AllocationUsageType::SCRIPT, (Ptr)nullptr); ASSERT_NE(Allocation(0), allocation); Allocation vout = context->allocationCreateTyped(type, AllocationMipmapControl::NONE, (int)AllocationUsageType::SCRIPT, (Ptr)nullptr); ASSERT_NE(Allocation(0), vout); context->allocation1DWrite(allocation, 0, 0, (Size)dataIn.size(), _data); hidl_vec vains; vains.setToExternal(&allocation, 1); hidl_vec params; context->scriptForEach(script, mExportForEachIdx_increment, vains, vout, params, nullptr); context->allocationRead(vout, (Ptr)dataOut.data(), (Size)dataOut.size()*sizeof(uint8_t)); EXPECT_EQ(expected, dataOut); } /* * Create a user script “struct_test”, and input and output Allocations. * Verified the foreach launch correctly for the reduction kernel. * * Calls: scriptCCreate, elementCreate, typeCreate, allocationCreateTyped, * allocation1DWrite, scriptReduce, contextFinish, allocationRead */ TEST_F(RenderscriptHidlTest, ScriptReduceTest) { hidl_vec bitcode; bitcode.setToExternal((uint8_t*)bitCode, bitCodeLength); Script script = context->scriptCCreate("struct_test", "/data/local/tmp/", bitcode); ASSERT_NE(Script(0), script); // uint8_t Element element = context->elementCreate(DataType::SIGNED_32, DataKind::USER, false, 1); ASSERT_NE(Element(0), element); // 64 x uint8_t Type type = context->typeCreate(element, 64, 0, 0, false, false, YuvFormat::YUV_NONE); ASSERT_NE(Type(0), type); Type type2 = context->typeCreate(element, 1, 0, 0, false, false, YuvFormat::YUV_NONE); ASSERT_NE(Type(0), type2); std::vector dataIn(64), dataOut(1); std::generate(dataIn.begin(), dataIn.end(), [](){ static int val = 0; return val++; }); hidl_vec _data; _data.setToExternal((uint8_t*)dataIn.data(), dataIn.size()*sizeof(int)); // 64 x float1 Allocation allocation = context->allocationCreateTyped(type, AllocationMipmapControl::NONE, (int)AllocationUsageType::SCRIPT, (Ptr)nullptr); ASSERT_NE(Allocation(0), allocation); Allocation vaout = context->allocationCreateTyped(type2, AllocationMipmapControl::NONE, (int)AllocationUsageType::SCRIPT, (Ptr)nullptr); ASSERT_NE(Allocation(0), vaout); context->allocation1DWrite(allocation, 0, 0, (Size)dataIn.size(), _data); hidl_vec vains; vains.setToExternal(&allocation, 1); context->scriptReduce(script, mExportReduceIdx_summation, vains, vaout, nullptr); context->contextFinish(); context->allocationRead(vaout, (Ptr)dataOut.data(), (Size)dataOut.size()*sizeof(int)); // sum of 0, 1, 2, ..., 62, 63 int sum = 63*64/2; EXPECT_EQ(sum, dataOut[0]); } /* * This test creates an allocation and binds it to a data segment in the * RenderScript script, represented in the bitcode. * * Calls: scriptCCreate, elementCreate, typeCreate, allocationCreateTyped, * allocation1DWrite, scriptBindAllocation, scriptSetVarV, scriptBindAllocation, * allocationRead, scriptInvokeV, allocationRead */ TEST_F(RenderscriptHidlTest, ScriptBindTest) { hidl_vec bitcode; bitcode.setToExternal((uint8_t*)bitCode, bitCodeLength); Script script = context->scriptCCreate("struct_test", "/data/local/tmp/", bitcode); ASSERT_NE(Script(0), script); // in32 Element element = context->elementCreate(DataType::SIGNED_32, DataKind::USER, false, 1); ASSERT_NE(Element(0), element); // 64 x int32 Type type = context->typeCreate(element, 64, 0, 0, false, false, YuvFormat::YUV_NONE); ASSERT_NE(Type(0), type); // 64 x int32 Allocation allocation = context->allocationCreateTyped(type, AllocationMipmapControl::NONE, (int)AllocationUsageType::SCRIPT, (Ptr)nullptr); ASSERT_NE(Allocation(0), allocation); std::vector dataIn(64), dataOut(64), expected(64, 5); hidl_vec _data; _data.setToExternal((uint8_t*)dataIn.data(), dataIn.size()*sizeof(int)); context->allocation1DWrite(allocation, 0, 0, (Size)dataIn.size(), _data); context->scriptBindAllocation(script, allocation, mExportVarIdx_var_int_ptr); int dim = 64; hidl_vec _dim; _dim.setToExternal((uint8_t*)&dim, sizeof(int)); context->scriptInvokeV(script, mExportFuncIdx_setBuffer, _dim); context->allocationRead(allocation, (Ptr)dataOut.data(), (Size)dataOut.size()*sizeof(int)); EXPECT_EQ(expected, dataOut); } /* * This test groups together two RenderScript intrinsic kernels to run one after * the other asynchronously with respect to the client. The test configures * Blend and Blur, and links them together such that Blur will execute after * Blend and use its result. The test checks the data returned to make sure it * was changed after passing through the entire ScriptGroup. * * Calls: elementCreate, typeCreate, allocationCreateTyped, allocation2DWrite, * scriptIntrinsicCreate, scriptKernelIDCreate, scriptFieldIDCreate, * scriptGroupCreate, scriptGroupSetInput, scriptGroupSetOutput, * scriptGroupExecute, contextFinish, allocation2DRead */ TEST_F(RenderscriptHidlTest, ScriptGroupTest) { std::vector dataIn(256 * 256 * 4, 128), dataOut(256 * 256 * 4, 0), zeros(256 * 256 * 4, 0); hidl_vec _dataIn, _dataOut; _dataIn.setToExternal(dataIn.data(), dataIn.size()); _dataOut.setToExternal(dataOut.data(), dataOut.size()); // 256 x 256 YUV pixels Element element1 = context->elementCreate(DataType::UNSIGNED_8, DataKind::PIXEL_RGBA, true, 4); ASSERT_NE(Element(0), element1); Type type1 = context->typeCreate(element1, 256, 256, 0, false, false, YuvFormat::YUV_NONE); ASSERT_NE(Type(0), type1); Allocation allocation1 = context->allocationCreateTyped(type1, AllocationMipmapControl::NONE, (int)AllocationUsageType::SCRIPT, (Ptr)nullptr); ASSERT_NE(Allocation(0), allocation1); context->allocation2DWrite(allocation1, 0, 0, 0, AllocationCubemapFace::POSITIVE_X, 256, 256, _dataIn, 0); // 256 x 256 RGBA pixels Element element2 = context->elementCreate(DataType::UNSIGNED_8, DataKind::PIXEL_RGBA, true, 4); ASSERT_NE(Element(0), element2); Type type2 = context->typeCreate(element2, 256, 256, 0, false, false, YuvFormat::YUV_NONE); ASSERT_NE(Type(0), type2); Allocation allocation2 = context->allocationCreateTyped(type2, AllocationMipmapControl::NONE, (int)AllocationUsageType::SCRIPT, (Ptr)nullptr); ASSERT_NE(Allocation(0), allocation2); context->allocation2DWrite(allocation2, 0, 0, 0, AllocationCubemapFace::POSITIVE_X, 256, 256, _dataOut, 0); // create scripts Script blend = context->scriptIntrinsicCreate(ScriptIntrinsicID::ID_BLEND, element1); ASSERT_NE(Script(0), blend); ScriptKernelID blendKID = context->scriptKernelIDCreate(blend, 1, 3); ASSERT_NE(ScriptKernelID(0), blendKID); Script blur = context->scriptIntrinsicCreate(ScriptIntrinsicID::ID_BLUR, element2); ASSERT_NE(Script(0), blur); ScriptKernelID blurKID = context->scriptKernelIDCreate(blur, 0, 2); ASSERT_NE(ScriptKernelID(0), blurKID); ScriptFieldID blurFID = context->scriptFieldIDCreate(blur, 1); ASSERT_NE(ScriptFieldID(0), blurFID); // ScriptGroup hidl_vec kernels = {blendKID, blurKID}; hidl_vec srcK = {blendKID}; hidl_vec dstK = {ScriptKernelID(0)}; hidl_vec dstF = {blurFID}; hidl_vec types = {type2}; ScriptGroup scriptGroup = context->scriptGroupCreate(kernels, srcK, dstK, dstF, types); ASSERT_NE(ScriptGroup(0), scriptGroup); context->scriptGroupSetInput(scriptGroup, blendKID, allocation1); context->scriptGroupSetOutput(scriptGroup, blurKID, allocation2); context->scriptGroupExecute(scriptGroup); context->contextFinish(); // verify contents were changed context->allocation2DRead(allocation2, 0, 0, 0, AllocationCubemapFace::POSITIVE_X, 256, 256, (Ptr)dataOut.data(), (Size)dataOut.size(), 0); EXPECT_NE(zeros, dataOut); } /* * Similar to the ScriptGroup test, this test verifies the execution flow of * RenderScript kernels and invokables. * * Calls: scriptCCreate, elementCreate, typeCreate, allocationCreateTyped, * allocation1DWrite, scriptFieldIDCreate, scriptInvokeIDCreate, * invokeClosureCreate, closureCreate, closureSetGlobal, scriptGroup2Create, * scriptGroupExecute, allocationRead */ TEST_F(RenderscriptHidlTest, ScriptGroup2Test) { hidl_vec bitcode; bitcode.setToExternal((uint8_t*)bitCode, bitCodeLength); Script script = context->scriptCCreate("struct_test", "/data/local/tmp/", bitcode); ASSERT_NE(Script(0), script); std::vector dataIn(128, 128), dataOut(128, 0), expected(128, 7+1); hidl_vec _dataIn, _dataOut; _dataIn.setToExternal(dataIn.data(), dataIn.size()); // 256 x 256 YUV pixels Element element = context->elementCreate(DataType::UNSIGNED_8, DataKind::USER, false, 1); ASSERT_NE(Element(0), element); Type type = context->typeCreate(element, 128, 0, 0, false, false, YuvFormat::YUV_NONE); ASSERT_NE(Type(0), type); Allocation allocation = context->allocationCreateTyped(type, AllocationMipmapControl::NONE, (int)AllocationUsageType::SCRIPT, (Ptr)nullptr); ASSERT_NE(Allocation(0), allocation); context->allocation1DWrite(allocation, 0, 0, (Size)_dataIn.size(), _dataIn); ScriptFieldID fieldID = context->scriptFieldIDCreate(script, mExportVarIdx_var_allocation); ASSERT_NE(ScriptFieldID(0), fieldID); // invoke ScriptInvokeID invokeID = context->scriptInvokeIDCreate(script, mExportFuncIdx_setAllocation); ASSERT_NE(ScriptInvokeID(0), invokeID); int dim = 128; hidl_vec params; params.setToExternal((uint8_t*)&dim, sizeof(dim)); hidl_vec fieldIDS1 = {fieldID}; hidl_vec values1 = {int64_t(0)}; hidl_vec sizes1 = {int32_t(0)}; Closure closure1 = context->invokeClosureCreate(invokeID, params, fieldIDS1, values1, sizes1); ASSERT_NE(Closure(0), closure1); // kernel ScriptKernelID kernelID = context->scriptKernelIDCreate(script, mExportForEachIdx_increment, 3); ASSERT_NE(ScriptKernelID(0), kernelID); hidl_vec fieldIDS2 = {ScriptFieldID(0)}; hidl_vec values2 = {(int64_t)(intptr_t)allocation}; hidl_vec sizes2 = {-1 /* allocation */}; hidl_vec depClosures2 = {closure1}; hidl_vec depFieldIDS2 = {fieldID}; Closure closure2 = context->closureCreate(kernelID, allocation /* returnValue */, fieldIDS2, values2, sizes2, depClosures2, depFieldIDS2); ASSERT_NE(Closure(0), closure2); // set argument context->closureSetGlobal(closure1, fieldID, (int64_t)(intptr_t)allocation, -1 /* allocation */); // execute hidl_string name = "script_group_2_test"; hidl_string cacheDir = "/data/local/tmp"; hidl_vec closures = {closure1, closure2}; ScriptGroup2 scriptGroup2 = context->scriptGroup2Create(name, cacheDir, closures); ASSERT_NE(ScriptGroup2(0), scriptGroup2); context->scriptGroupExecute(scriptGroup2); context->allocationRead(allocation, (Ptr)dataOut.data(), (Size)dataOut.size()*sizeof(uint8_t)); EXPECT_EQ(expected, dataOut); } /* * Similar to the ScriptGroup test, this test verifies a single kernel can be * called by ScriptGroup with an unbound allocation specified before launch * * Calls: scriptCCreate, elementCreate, typeCreate, allocationCreateTyped, * allocation1DWrite, scriptKernelIDCreate, closureCreate, closureSetArg, * scriptGroup2Create, scriptGroupExecute, allocationRead */ TEST_F(RenderscriptHidlTest, ScriptGroup2KernelTest) { hidl_vec bitcode; bitcode.setToExternal((uint8_t*)bitCode, bitCodeLength); Script script = context->scriptCCreate("struct_test", "/data/local/tmp/", bitcode); ASSERT_NE(Script(0), script); std::vector dataIn(128, 128), dataOut(128, 0), expected(128, 128 + 1); hidl_vec _dataIn, _dataOut; _dataIn.setToExternal(dataIn.data(), dataIn.size()); // 256 x 256 YUV pixels Element element = context->elementCreate(DataType::UNSIGNED_8, DataKind::USER, false, 1); ASSERT_NE(Element(0), element); Type type = context->typeCreate(element, 128, 0, 0, false, false, YuvFormat::YUV_NONE); ASSERT_NE(Type(0), type); Allocation allocation = context->allocationCreateTyped(type, AllocationMipmapControl::NONE, (int)AllocationUsageType::SCRIPT, (Ptr)nullptr); ASSERT_NE(Allocation(0), allocation); context->allocation1DWrite(allocation, 0, 0, (Size)_dataIn.size(), _dataIn); // kernel ScriptKernelID kernelID = context->scriptKernelIDCreate(script, mExportForEachIdx_increment, 3); ASSERT_NE(ScriptKernelID(0), kernelID); hidl_vec fieldIDS = {ScriptFieldID(0)}; hidl_vec values = {int64_t(0)}; hidl_vec sizes = {int32_t(0)}; hidl_vec depClosures = {Closure(0)}; hidl_vec depFieldIDS = {ScriptFieldID(0)}; Closure closure = context->closureCreate(kernelID, allocation /* returnValue */, fieldIDS, values, sizes, depClosures, depFieldIDS); ASSERT_NE(Closure(0), closure); // set argument context->closureSetArg(closure, 0 /* first argument */, (Ptr)allocation, -1); // execute hidl_string name = "script_group_2_test"; hidl_string cacheDir = "/data/local/tmp"; hidl_vec closures = {closure}; ScriptGroup2 scriptGroup2 = context->scriptGroup2Create(name, cacheDir, closures); ASSERT_NE(ScriptGroup2(0), scriptGroup2); context->scriptGroupExecute(scriptGroup2); context->allocationRead(allocation, (Ptr)dataOut.data(), (Size)dataOut.size()*sizeof(uint8_t)); EXPECT_EQ(expected, dataOut); }