diff options
| author | Artem Belevich <tra@google.com> | 2018-09-21 17:29:33 +0000 |
|---|---|---|
| committer | Artem Belevich <tra@google.com> | 2018-09-21 17:29:33 +0000 |
| commit | 78929efb4dcbb2c8aeba46eee912a614d4ecebb9 (patch) | |
| tree | d8ae6e44b0267bf826d68a01e9ad339e984d9cc0 /clang/test | |
| parent | a3d0f409647a2051c1cd1546c48369f9f9ce3f27 (diff) | |
| download | bcm5719-llvm-78929efb4dcbb2c8aeba46eee912a614d4ecebb9.tar.gz bcm5719-llvm-78929efb4dcbb2c8aeba46eee912a614d4ecebb9.zip | |
[CUDA] Ignore uncallable functions when we check for usual deallocators.
Previously clang considered function variants from both sides of
compilation and that resulted in picking up wrong deallocation function.
Differential Revision: https://reviews.llvm.org/D51808
llvm-svn: 342749
Diffstat (limited to 'clang/test')
| -rw-r--r-- | clang/test/CodeGenCUDA/usual-deallocators.cu | 133 | ||||
| -rw-r--r-- | clang/test/SemaCUDA/call-host-fn-from-device.cu | 6 | ||||
| -rw-r--r-- | clang/test/SemaCUDA/usual-deallocators.cu | 95 |
3 files changed, 231 insertions, 3 deletions
diff --git a/clang/test/CodeGenCUDA/usual-deallocators.cu b/clang/test/CodeGenCUDA/usual-deallocators.cu new file mode 100644 index 00000000000..2d97c8c9f7d --- /dev/null +++ b/clang/test/CodeGenCUDA/usual-deallocators.cu @@ -0,0 +1,133 @@ +// RUN: %clang_cc1 %s --std=c++11 -triple nvptx-unknown-unknown -fcuda-is-device \ +// RUN: -emit-llvm -o - | FileCheck %s --check-prefixes=COMMON,DEVICE +// RUN: %clang_cc1 %s --std=c++11 -triple nvptx-unknown-unknown \ +// RUN: -emit-llvm -o - | FileCheck %s --check-prefixes=COMMON,HOST +// RUN: %clang_cc1 %s --std=c++17 -triple nvptx-unknown-unknown -fcuda-is-device \ +// RUN: -emit-llvm -o - | FileCheck %s --check-prefixes=COMMON,DEVICE +// RUN: %clang_cc1 %s --std=c++17 -triple nvptx-unknown-unknown \ +// RUN: -emit-llvm -o - | FileCheck %s --check-prefixes=COMMON,HOST + +#include "Inputs/cuda.h" +extern "C" __host__ void host_fn(); +extern "C" __device__ void dev_fn(); +extern "C" __host__ __device__ void hd_fn(); + +struct H1D1 { + __host__ void operator delete(void *) { host_fn(); }; + __device__ void operator delete(void *) { dev_fn(); }; +}; + +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; +} + +// Make sure we call the right variant of usual deallocator. +__host__ __device__ void tests_hd(void *t) { + // COMMON-LABEL: define linkonce_odr void @_Z7test_hdI4H1D1EvPv + // COMMON: call void @_ZN4H1D1dlEPv + test_hd<H1D1>(t); + // COMMON-LABEL: define linkonce_odr void @_Z7test_hdI4H1D2EvPv + // DEVICE: call void @_ZN4H1D2dlEPvj(i8* {{.*}}, i32 1) + // HOST: call void @_ZN4H1D2dlEPv(i8* {{.*}}) + test_hd<H1D2>(t); + // COMMON-LABEL: define linkonce_odr void @_Z7test_hdI4H2D1EvPv + // DEVICE: call void @_ZN4H2D1dlEPv(i8* {{.*}}) + // HOST: call void @_ZN4H2D1dlEPvj(i8* %3, i32 1) + test_hd<H2D1>(t); + // COMMON-LABEL: define linkonce_odr void @_Z7test_hdI4H2D2EvPv + // COMMON: call void @_ZN4H2D2dlEPvj(i8* {{.*}}, i32 1) + test_hd<H2D2>(t); + // COMMON-LABEL: define linkonce_odr void @_Z7test_hdI6H1D1D2EvPv + // COMMON: call void @_ZN6H1D1D2dlEPv(i8* %3) + test_hd<H1D1D2>(t); + // COMMON-LABEL: define linkonce_odr void @_Z7test_hdI6H1H2D1EvPv + // COMMON: call void @_ZN6H1H2D1dlEPv(i8* {{.*}}) + test_hd<H1H2D1>(t); + // COMMON-LABEL: define linkonce_odr void @_Z7test_hdI6H1H2D2EvPv + // DEVICE: call void @_ZN6H1H2D2dlEPvj(i8* {{.*}}, i32 1) + // HOST: call void @_ZN6H1H2D2dlEPv(i8* {{.*}}) + test_hd<H1H2D2>(t); + // COMMON-LABEL: define linkonce_odr void @_Z7test_hdI8H1H2D1D2EvPv + // COMMON: call void @_ZN8H1H2D1D2dlEPv(i8* {{.*}}) + test_hd<H1H2D1D2>(t); +} + +// Make sure we've picked deallocator for the correct side of compilation. + +// COMMON-LABEL: define linkonce_odr void @_ZN4H1D1dlEPv(i8*) +// DEVICE: call void @dev_fn() +// HOST: call void @host_fn() + +// DEVICE-LABEL: define linkonce_odr void @_ZN4H1D2dlEPvj(i8*, i32) +// DEVICE: call void @dev_fn() +// HOST-LABEL: define linkonce_odr void @_ZN4H1D2dlEPv(i8*) +// HOST: call void @host_fn() + +// DEVICE-LABEL: define linkonce_odr void @_ZN4H2D1dlEPv(i8*) +// DEVICE: call void @dev_fn() +// HOST-LABEL: define linkonce_odr void @_ZN4H2D1dlEPvj(i8*, i32) +// HOST: call void @host_fn() + +// COMMON-LABEL: define linkonce_odr void @_ZN4H2D2dlEPvj(i8*, i32) +// DEVICE: call void @dev_fn() +// HOST: call void @host_fn() + +// COMMON-LABEL: define linkonce_odr void @_ZN6H1D1D2dlEPv(i8*) +// DEVICE: call void @dev_fn() +// HOST: call void @host_fn() + +// COMMON-LABEL: define linkonce_odr void @_ZN6H1H2D1dlEPv(i8*) +// DEVICE: call void @dev_fn() +// HOST: call void @host_fn() + +// DEVICE-LABEL: define linkonce_odr void @_ZN6H1H2D2dlEPvj(i8*, i32) +// DEVICE: call void @dev_fn() +// HOST-LABEL: define linkonce_odr void @_ZN6H1H2D2dlEPv(i8*) +// HOST: call void @host_fn() + +// COMMON-LABEL: define linkonce_odr void @_ZN8H1H2D1D2dlEPv(i8*) +// DEVICE: call void @dev_fn() +// HOST: call void @host_fn() diff --git a/clang/test/SemaCUDA/call-host-fn-from-device.cu b/clang/test/SemaCUDA/call-host-fn-from-device.cu index c727dc8cbeb..acdd291b664 100644 --- a/clang/test/SemaCUDA/call-host-fn-from-device.cu +++ b/clang/test/SemaCUDA/call-host-fn-from-device.cu @@ -41,12 +41,12 @@ struct T { operator Dummy() { return Dummy(); } // expected-note@-1 {{'operator Dummy' declared here}} - __host__ void operator delete(void*); - __device__ void operator delete(void*, size_t); + __host__ void operator delete(void *) { host_fn(); }; + __device__ void operator delete(void*, __SIZE_TYPE__); }; struct U { - __device__ void operator delete(void*, size_t) = delete; + __device__ void operator delete(void*, __SIZE_TYPE__) = delete; __host__ __device__ void operator delete(void*); }; diff --git a/clang/test/SemaCUDA/usual-deallocators.cu b/clang/test/SemaCUDA/usual-deallocators.cu new file mode 100644 index 00000000000..a0238649c6d --- /dev/null +++ b/clang/test/SemaCUDA/usual-deallocators.cu @@ -0,0 +1,95 @@ +// 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); +} |

