llvm-for-llvmta/tools/clang/test/SemaCUDA/usual-deallocators.cu

105 lines
3.5 KiB
Plaintext

// RUN: %clang_cc1 %s --std=c++11 -triple nvptx-unknown-unknown -fcuda-is-device \
// RUN: -emit-llvm -o /dev/null -verify=device
// RUN: %clang_cc1 %s --std=c++11 -triple nvptx-unknown-unknown \
// RUN: -emit-llvm -o /dev/null -verify=host
// RUN: %clang_cc1 %s --std=c++17 -triple nvptx-unknown-unknown -fcuda-is-device \
// RUN: -emit-llvm -o /dev/null -verify=device
// RUN: %clang_cc1 %s --std=c++17 -triple nvptx-unknown-unknown \
// RUN: -emit-llvm -o /dev/null -verify=host
#include "Inputs/cuda.h"
extern __host__ void host_fn();
extern __device__ void dev_fn();
extern __host__ __device__ void hd_fn();
struct H1D1 {
__host__ void operator delete(void *) { host_fn(); };
__device__ void operator delete(void *) { dev_fn(); };
};
struct h1D1 {
__host__ void operator delete(void *) = delete;
// host-note@-1 {{'operator delete' has been explicitly marked deleted here}}
__device__ void operator delete(void *) { dev_fn(); };
};
struct H1d1 {
__host__ void operator delete(void *) { host_fn(); };
__device__ void operator delete(void *) = delete;
// device-note@-1 {{'operator delete' has been explicitly marked deleted here}}
};
struct H1D2 {
__host__ void operator delete(void *) { host_fn(); };
__device__ void operator delete(void *, __SIZE_TYPE__) { dev_fn(); };
};
struct H2D1 {
__host__ void operator delete(void *, __SIZE_TYPE__) { host_fn(); };
__device__ void operator delete(void *) { dev_fn(); };
};
struct H2D2 {
__host__ void operator delete(void *, __SIZE_TYPE__) { host_fn(); };
__device__ void operator delete(void *, __SIZE_TYPE__) { dev_fn(); };
};
struct H1D1D2 {
__host__ void operator delete(void *) { host_fn(); };
__device__ void operator delete(void *) { dev_fn(); };
__device__ void operator delete(void *, __SIZE_TYPE__) { dev_fn(); };
};
struct H1H2D1 {
__host__ void operator delete(void *) { host_fn(); };
__host__ void operator delete(void *, __SIZE_TYPE__) { host_fn(); };
__device__ void operator delete(void *) { dev_fn(); };
};
struct H1H2D2 {
__host__ void operator delete(void *) { host_fn(); };
__host__ void operator delete(void *, __SIZE_TYPE__) { host_fn(); };
__device__ void operator delete(void *, __SIZE_TYPE__) { dev_fn(); };
};
struct H1H2D1D2 {
__host__ void operator delete(void *) { host_fn(); };
__host__ void operator delete(void *, __SIZE_TYPE__) { host_fn(); };
__device__ void operator delete(void *) { dev_fn(); };
__device__ void operator delete(void *, __SIZE_TYPE__) { dev_fn(); };
};
template <typename T>
__host__ __device__ void test_hd(void *p) {
T *t = (T *)p;
delete t;
// host-error@-1 {{attempt to use a deleted function}}
// device-error@-2 {{attempt to use a deleted function}}
}
__host__ __device__ void tests_hd(void *t) {
test_hd<H1D1>(t);
test_hd<h1D1>(t);
// host-note@-1 {{in instantiation of function template specialization 'test_hd<h1D1>' requested here}}
test_hd<H1d1>(t);
// device-note@-1 {{in instantiation of function template specialization 'test_hd<H1d1>' requested here}}
test_hd<H1D2>(t);
test_hd<H2D1>(t);
test_hd<H2D2>(t);
test_hd<H1D1D2>(t);
test_hd<H1H2D1>(t);
test_hd<H1H2D1>(t);
test_hd<H1H2D2>(t);
test_hd<H1H2D1D2>(t);
}
// This should produce no errors. Defaulted destructor should be treated as HD,
// which allows referencing host-only `operator delete` with a deferred
// diagnostics that would fire if we ever attempt to codegen it on device..
struct H {
virtual ~H() = default;
static void operator delete(void *) {}
};
H h;