Skip to content
This repository has been archived by the owner on Mar 21, 2024. It is now read-only.

Commit

Permalink
Implement cuda::any_resource_ref
Browse files Browse the repository at this point in the history
  • Loading branch information
miscco committed Aug 30, 2022
1 parent 04c7250 commit a90c51a
Show file tree
Hide file tree
Showing 4 changed files with 624 additions and 1 deletion.
Original file line number Diff line number Diff line change
@@ -0,0 +1,110 @@
//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//

// UNSUPPORTED: c++03, c++11

// cuda::resource, cuda::resource_with
#include <cuda/memory_resource>

#include <cuda/std/cstdint>

template <class T>
struct property_with_value {
using value_type = T;
};

template <class T>
struct property_without_value {};

template <class... Properties>
struct resource {
inline __host__ __device__ void* allocate(std::size_t, std::size_t) {
return &_val;
}

inline __host__ __device__ void deallocate(void* ptr, std::size_t,
std::size_t) {
// ensure that we did get the right inputs forwarded
_val = *static_cast<int*>(ptr);
}

inline __host__ __device__ bool operator==(const resource& other) {
return _val == other._val;
}
inline __host__ __device__ bool operator!=(const resource& other) {
return _val != other._val;
}

int _val = 0;

_LIBCUDACXX_TEMPLATE(class Property)
(requires !cuda::property_with_value<Property> &&
_CUDA_VSTD::_One_of<Property, Properties...>) //
inline __host__ __device__
friend void get_property(const resource&, Property) noexcept {}

_LIBCUDACXX_TEMPLATE(class Property)
(requires cuda::property_with_value<Property>&&
_CUDA_VSTD::_One_of<Property, Properties...>) //
inline __host__ __device__ //
friend typename Property::value_type
get_property(const resource& res, Property) noexcept {
return res._val;
}
};

namespace constructible {
using ref = cuda::any_resource_ref<property_with_value<int>,
property_with_value<double>,
property_without_value<std::size_t> >;

using matching_properties =
resource<property_with_value<double>, property_without_value<std::size_t>,
property_with_value<int> >;

using missing_stateful_property =
resource<property_with_value<int>, property_without_value<std::size_t> >;
using missing_stateless_property =
resource<property_with_value<int>, property_with_value<double> >;

using cuda::std::is_constructible;
static_assert(is_constructible<ref, matching_properties&>::value, "");
static_assert(!is_constructible<ref, missing_stateful_property&>::value, "");
static_assert(!is_constructible<ref, missing_stateless_property&>::value, "");

static_assert(is_constructible<ref, ref&>::value, "");

// Ensure we require a mutable valid reference and do not bind against rvalues
static_assert(!is_constructible<ref, matching_properties>::value, "");
static_assert(!is_constructible<ref, const matching_properties&>::value, "");

static_assert(cuda::std::is_copy_constructible<ref>::value, "");
static_assert(cuda::std::is_move_constructible<ref>::value, "");
} // namespace constructible

namespace assignable {
using ref = cuda::any_resource_ref<property_with_value<int>,
property_with_value<double>,
property_without_value<std::size_t> >;

using res = resource<property_with_value<int>, property_with_value<double>,
property_without_value<std::size_t> >;

using other_res =
resource<property_without_value<int>, property_with_value<int>,
property_with_value<double>, property_without_value<std::size_t> >;

using cuda::std::is_assignable;
static_assert(cuda::std::is_assignable<ref, res&>::value, "");
static_assert(cuda::std::is_assignable<ref, other_res&>::value, "");

static_assert(cuda::std::is_copy_assignable<ref>::value, "");
static_assert(cuda::std::is_move_assignable<ref>::value, "");
} // namespace assignable

int main(int, char**) { return 0; }
Original file line number Diff line number Diff line change
@@ -0,0 +1,112 @@
//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//

// UNSUPPORTED: c++03, c++11

// cuda::resource, cuda::resource_with
#include <cuda/memory_resource>

#include <cuda/std/cassert>
#include <cuda/std/cstdint>

template <class T>
struct property_with_value {
using value_type = T;
};

template <class T>
struct property_without_value {};

template <class... Properties>
struct resource {
inline __host__ __device__ void* allocate(std::size_t, std::size_t) {
return &_val;
}

inline __host__ __device__ void deallocate(void* ptr, std::size_t,
std::size_t) {
// ensure that we did get the right inputs forwarded
_val = *static_cast<int*>(ptr);
}

inline __host__ __device__ bool operator==(const resource& other) {
return _val == other._val;
}
inline __host__ __device__ bool operator!=(const resource& other) {
return _val != other._val;
}

int _val = 0;

_LIBCUDACXX_TEMPLATE(class Property)
(requires !cuda::property_with_value<Property> &&
_CUDA_VSTD::_One_of<Property, Properties...>) //
inline __host__ __device__
friend void get_property(const resource&, Property) noexcept {}

_LIBCUDACXX_TEMPLATE(class Property)
(requires cuda::property_with_value<Property>&&
_CUDA_VSTD::_One_of<Property, Properties...>) //
inline __host__ __device__ //
friend typename Property::value_type
get_property(const resource& res, Property) noexcept {
return res._val;
}
};

namespace equality {
using ref = cuda::any_resource_ref<property_with_value<int>,
property_with_value<double>,
property_without_value<std::size_t> >;

using pertubed_properties =
cuda::any_resource_ref<property_with_value<double>,
property_with_value<int>,
property_without_value<std::size_t> >;

using different_properties =
cuda::any_resource_ref<property_with_value<short>, property_with_value<int>,
property_without_value<std::size_t> >;

using res = resource<property_with_value<int>, property_with_value<double>,
property_without_value<std::size_t> >;
using other_res =
resource<property_with_value<double>, property_with_value<int>,
property_without_value<std::size_t> >;

__host__ __device__ void test_equality() {
res input{42};
res with_equal_value{42};
res with_different_value{1337};

assert(input == with_equal_value);
assert(input != with_different_value);

assert(ref{input} == ref{with_equal_value});
assert(ref{input} != ref{with_different_value});

// Should ignore pertubed properties
assert(ref{input} == pertubed_properties{with_equal_value});
assert(ref{input} != pertubed_properties{with_different_value});

// Should reject different resources
other_res other_with_matching_value{42};
other_res other_with_different_value{1337};
assert(ref{input} != ref{other_with_matching_value});
assert(ref{input} != ref{other_with_different_value});

assert(ref{input} != pertubed_properties{other_with_matching_value});
assert(ref{input} != pertubed_properties{other_with_matching_value});
}

} // namespace equality

int main(int, char**) {
equality::test_equality();
return 0;
}
Loading

0 comments on commit a90c51a

Please sign in to comment.