fe39c423bb
[ROCm/hip-tests commit: 646eaa43ea]
275 строки
10 KiB
C++
275 строки
10 KiB
C++
/*
|
|
Copyright (c) 2022 Advanced Micro Devices, Inc. All rights reserved.
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
THE SOFTWARE.
|
|
*/
|
|
|
|
/*
|
|
hipMalloc3DArray API test scenarios
|
|
1. Basic Functionality
|
|
2. Negative Scenarios
|
|
3. Allocating Small and big chunk data
|
|
4. Multithreaded scenario
|
|
*/
|
|
|
|
#include <array>
|
|
#include <hip_test_common.hh>
|
|
#include "hipArrayCommon.hh"
|
|
|
|
static constexpr auto ARRAY_SIZE{4};
|
|
static constexpr auto BIG_ARRAY_SIZE{100};
|
|
static constexpr auto ARRAY_LOOP{100};
|
|
|
|
/*
|
|
* This API verifies memory allocations for small and
|
|
* bigger chunks of data.
|
|
* Two scenarios are verified in this API
|
|
* 1. SmallArray: Allocates ARRAY_SIZE in a loop and
|
|
* releases the memory and verifies the meminfo.
|
|
* 2. BigArray: Allocates BIG_ARRAY_SIZE in a loop and
|
|
* releases the memory and verifies the meminfo
|
|
*
|
|
* In both cases, the memory info before allocation and
|
|
* after releasing the memory should be the same
|
|
*
|
|
*/
|
|
static void Malloc3DArray_DiffSizes(int gpu) {
|
|
HIP_CHECK_THREAD(hipSetDevice(gpu));
|
|
const int size = GENERATE(ARRAY_SIZE, BIG_ARRAY_SIZE);
|
|
int width{size}, height{size}, depth{size};
|
|
hipChannelFormatDesc channelDesc = hipCreateChannelDesc<float>();
|
|
std::array<hipArray_t, ARRAY_LOOP> arr;
|
|
size_t pavail, avail;
|
|
HIP_CHECK_THREAD(hipMemGetInfo(&pavail, nullptr));
|
|
|
|
for (int i = 0; i < ARRAY_LOOP; i++) {
|
|
HIP_CHECK_THREAD(hipMalloc3DArray(&arr[i], &channelDesc, make_hipExtent(width, height, depth),
|
|
hipArrayDefault));
|
|
}
|
|
for (int i = 0; i < ARRAY_LOOP; i++) {
|
|
HIP_CHECK_THREAD(hipFreeArray(arr[i]));
|
|
}
|
|
|
|
HIP_CHECK_THREAD(hipMemGetInfo(&avail, nullptr));
|
|
REQUIRE_THREAD(pavail == avail);
|
|
}
|
|
|
|
/*
|
|
* Verifies the negative scenarios of hipMalloc3DArray API
|
|
*/
|
|
TEST_CASE("Unit_hipMalloc3DArray_Negative") {
|
|
constexpr int width{ARRAY_SIZE}, height{ARRAY_SIZE}, depth{ARRAY_SIZE};
|
|
hipChannelFormatDesc channelDesc = hipCreateChannelDesc<float>();
|
|
hipArray* arr;
|
|
#if HT_NVIDIA
|
|
SECTION("NullPointer to Array") {
|
|
REQUIRE(hipMalloc3DArray(nullptr, &channelDesc, make_hipExtent(width, height, depth),
|
|
hipArrayDefault) != hipSuccess);
|
|
}
|
|
|
|
SECTION("NullPointer to Channel Descriptor") {
|
|
REQUIRE(hipMalloc3DArray(&arr, nullptr, make_hipExtent(width, height, depth),
|
|
hipArrayDefault) != hipSuccess);
|
|
}
|
|
#endif
|
|
SECTION("Width 0 in hipExtent") {
|
|
REQUIRE(hipMalloc3DArray(&arr, &channelDesc, make_hipExtent(0, height, width),
|
|
hipArrayDefault) != hipSuccess);
|
|
}
|
|
|
|
SECTION("Height 0 in hipExtent") {
|
|
REQUIRE(hipMalloc3DArray(&arr, &channelDesc, make_hipExtent(width, 0, width),
|
|
hipArrayDefault) != hipSuccess);
|
|
}
|
|
|
|
SECTION("Invalid Flag") {
|
|
REQUIRE(hipMalloc3DArray(&arr, &channelDesc, make_hipExtent(width, height, depth), 100) !=
|
|
hipSuccess);
|
|
}
|
|
|
|
SECTION("Width,Height & Depth 0 in hipExtent") {
|
|
REQUIRE(hipMalloc3DArray(&arr, &channelDesc, make_hipExtent(0, 0, 0), hipArrayDefault) !=
|
|
hipSuccess);
|
|
}
|
|
|
|
SECTION("Max int values to extent") {
|
|
REQUIRE(hipMalloc3DArray(
|
|
&arr, &channelDesc,
|
|
make_hipExtent(std::numeric_limits<int>::max(), std::numeric_limits<int>::max(),
|
|
std::numeric_limits<int>::max()),
|
|
hipArrayDefault) != hipSuccess);
|
|
}
|
|
}
|
|
|
|
TEST_CASE("Unit_hipMalloc3DArray_DiffSizes") {
|
|
Malloc3DArray_DiffSizes(0);
|
|
HIP_CHECK_THREAD_FINALIZE();
|
|
}
|
|
|
|
/*
|
|
This testcase verifies the hipMalloc3DArray API in multithreaded
|
|
scenario by launching threads in parallel on multiple GPUs
|
|
and verifies the hipMalloc3DArray API with small and big chunks data
|
|
*/
|
|
TEST_CASE("Unit_hipMalloc3DArray_MultiThread") {
|
|
std::vector<std::thread> threadlist;
|
|
int devCnt = 0;
|
|
devCnt = HipTest::getDeviceCount();
|
|
const auto pavail = getFreeMem();
|
|
for (int i = 0; i < devCnt; i++) {
|
|
threadlist.push_back(std::thread(Malloc3DArray_DiffSizes, i));
|
|
}
|
|
|
|
for (auto& t : threadlist) {
|
|
t.join();
|
|
}
|
|
HIP_CHECK_THREAD_FINALIZE();
|
|
const auto avail = getFreeMem();
|
|
|
|
if (pavail != avail) {
|
|
WARN("Memory leak of hipMalloc3D API in multithreaded scenario");
|
|
REQUIRE(false);
|
|
}
|
|
}
|
|
|
|
namespace {
|
|
void checkArrayIsExpected(hipArray_t array, const hipChannelFormatDesc& expected_desc,
|
|
const hipExtent& expected_extent, const unsigned int expected_flags) {
|
|
// hipArrayGetInfo doesn't currently exist (EXSWCPHIPT-87)
|
|
#if HT_AMD
|
|
std::ignore = array;
|
|
std::ignore = expected_desc;
|
|
std::ignore = expected_extent;
|
|
std::ignore = expected_flags;
|
|
#else
|
|
cudaChannelFormatDesc queried_desc;
|
|
cudaExtent queried_extent;
|
|
unsigned int queried_flags;
|
|
|
|
cudaArrayGetInfo(&queried_desc, &queried_extent, &queried_flags, array);
|
|
|
|
REQUIRE(expected_desc.x == queried_desc.x);
|
|
REQUIRE(expected_desc.y == queried_desc.y);
|
|
REQUIRE(expected_desc.z == queried_desc.z);
|
|
REQUIRE(expected_desc.f == queried_desc.f);
|
|
|
|
REQUIRE(expected_extent.width == queried_extent.width);
|
|
REQUIRE(expected_extent.height == queried_extent.height);
|
|
REQUIRE(expected_extent.depth == queried_extent.depth);
|
|
|
|
REQUIRE(expected_flags == queried_flags);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
TEMPLATE_TEST_CASE("Unit_hipMalloc3DArray_happy", "", char, uchar2, uint2, int4, short4, float,
|
|
float2, float4) {
|
|
hipArray_t array;
|
|
const auto desc = hipCreateChannelDesc<TestType>();
|
|
#if HT_AMD
|
|
const unsigned int flags = hipArrayDefault;
|
|
#else
|
|
const unsigned int flags = GENERATE(hipArrayDefault, hipArraySurfaceLoadStore);
|
|
#endif
|
|
constexpr size_t size = 64;
|
|
hipExtent extent;
|
|
|
|
SECTION("1D Array") {
|
|
extent = make_hipExtent(size, 0, 0);
|
|
}
|
|
SECTION("2D Array") {
|
|
extent = make_hipExtent(size, size, 0);
|
|
}
|
|
SECTION("3D Array") {
|
|
extent = make_hipExtent(size, size, size);
|
|
}
|
|
|
|
HIP_CHECK(hipMalloc3DArray(&array, &desc, extent, flags));
|
|
checkArrayIsExpected(array, desc, extent, flags);
|
|
HIP_CHECK(hipFreeArray(array));
|
|
}
|
|
|
|
TEMPLATE_TEST_CASE("Unit_hipMalloc3DArray_MaxTexture", "", int, uint4, short, ushort2,
|
|
unsigned char, float, float4) {
|
|
#if HT_AMD
|
|
HipTest::HIP_SKIP_TEST("EXSWCPHIPT-97");
|
|
return;
|
|
#endif
|
|
|
|
hipArray_t array;
|
|
const hipChannelFormatDesc desc = hipCreateChannelDesc<TestType>();
|
|
#if HT_AMD
|
|
const unsigned int flag = hipArrayDefault;
|
|
#else
|
|
const unsigned int flag = GENERATE(hipArrayDefault, hipArraySurfaceLoadStore);
|
|
#endif
|
|
if (flag == hipArraySurfaceLoadStore) {
|
|
HipTest::HIP_SKIP_TEST("EXSWCPHIPT-58");
|
|
return;
|
|
}
|
|
CAPTURE(flag);
|
|
const Sizes sizes(flag);
|
|
CAPTURE(sizes.max1D, sizes.max2D, sizes.max3D);
|
|
|
|
const size_t s = 64;
|
|
SECTION("Happy") {
|
|
// stored in a vector so some values can be ifdef'd out
|
|
std::vector<hipExtent> extentsToTest{
|
|
make_hipExtent(sizes.max1D, 0, 0), // 1D max
|
|
make_hipExtent(sizes.max2D[0], s, 0), // 2D max width
|
|
make_hipExtent(s, sizes.max2D[1], 0), // 2D max height
|
|
make_hipExtent(sizes.max2D[0], sizes.max2D[1], 0), // 2D max
|
|
make_hipExtent(sizes.max3D[0], s, s), // 3D max width
|
|
make_hipExtent(s, sizes.max3D[1], s), // 3D max height
|
|
make_hipExtent(s, s, sizes.max3D[2]), // 3D max depth
|
|
make_hipExtent(s, sizes.max3D[1], sizes.max3D[2]), // 3D max height and depth
|
|
make_hipExtent(sizes.max3D[0], s, sizes.max3D[2]), // 3D max width and depth
|
|
make_hipExtent(sizes.max3D[0], sizes.max3D[1], s), // 3D max width and height
|
|
make_hipExtent(sizes.max3D[0], sizes.max3D[1], sizes.max3D[2]) // 3D max
|
|
};
|
|
const auto extent =
|
|
GENERATE_COPY(from_range(std::begin(extentsToTest), std::end(extentsToTest)));
|
|
CAPTURE(extent.width, extent.height, extent.depth);
|
|
auto maxArrayCreateError = hipMalloc3DArray(&array, &desc, extent, flag);
|
|
// this can try to alloc many GB of memory, so out of memory is acceptable
|
|
if (maxArrayCreateError == hipErrorOutOfMemory) return;
|
|
HIP_CHECK(maxArrayCreateError);
|
|
checkArrayIsExpected(array, desc, extent, flag);
|
|
HIP_CHECK(hipFreeArray(array));
|
|
}
|
|
SECTION("Negative") {
|
|
std::vector<hipExtent> extentsToTest {
|
|
make_hipExtent(sizes.max1D + 1, 0, 0), // 1D max
|
|
make_hipExtent(sizes.max2D[0] + 1, s, 0), // 2D max width
|
|
make_hipExtent(s, sizes.max2D[1] + 1, 0), // 2D max height
|
|
make_hipExtent(sizes.max2D[0] + 1, sizes.max2D[1] + 1, 0), // 2D max
|
|
make_hipExtent(sizes.max3D[0] + 1, s, s), // 3D max width
|
|
make_hipExtent(s, sizes.max3D[1] + 1, s), // 3D max height
|
|
#if !HT_NVIDIA // leads to hipSuccess on NVIDIA
|
|
make_hipExtent(s, s, sizes.max3D[2] + 1), // 3D max depth
|
|
#endif
|
|
make_hipExtent(s, sizes.max3D[1] + 1, sizes.max3D[2] + 1), // 3D max height and depth
|
|
make_hipExtent(sizes.max3D[0] + 1, s, sizes.max3D[2] + 1), // 3D max width and depth
|
|
make_hipExtent(sizes.max3D[0] + 1, sizes.max3D[1] + 1, s), // 3D max width and height
|
|
make_hipExtent(sizes.max3D[0] + 1, sizes.max3D[1] + 1, sizes.max3D[2] + 1) // 3D max
|
|
};
|
|
const auto extent =
|
|
GENERATE_COPY(from_range(std::begin(extentsToTest), std::end(extentsToTest)));
|
|
CAPTURE(extent.width, extent.height, extent.depth);
|
|
HIP_CHECK_ERROR(hipMalloc3DArray(&array, &desc, extent, flag), hipErrorInvalidValue);
|
|
}
|
|
}
|