2018-09-22 01:29:33 +08:00
|
|
|
// 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);
|
|
|
|
}
|
2021-01-15 08:05:33 +08:00
|
|
|
|
|
|
|
// 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;
|