// RUN: %clang_cc1 %s --std=c++11 -triple nvptx-unknown-unknown -fcuda-is-device \ // RUN: -emit-llvm -o /dev/null -verify -verify-ignore-unexpected=note // Note: This test won't work with -fsyntax-only, because some of these errors // are emitted during codegen. #include "Inputs/cuda.h" extern "C" void host_fn() {} // expected-note@-1 7 {{'host_fn' declared here}} struct Dummy {}; struct S { S() {} // expected-note@-1 2 {{'S' declared here}} ~S() { host_fn(); } // expected-note@-1 {{'~S' declared here}} int x; }; struct T { __host__ __device__ void hd() { host_fn(); } // expected-error@-1 {{reference to __host__ function 'host_fn' in __host__ __device__ function}} // No error; this is (implicitly) inline and is never called, so isn't // codegen'ed. __host__ __device__ void hd2() { host_fn(); } __host__ __device__ void hd3(); void h() {} // expected-note@-1 {{'h' declared here}} void operator+(); // expected-note@-1 {{'operator+' declared here}} void operator-(const T&) {} // expected-note@-1 {{'operator-' declared here}} operator Dummy() { return Dummy(); } // expected-note@-1 {{'operator Dummy' declared here}} __host__ void operator delete(void*); __device__ void operator delete(void*, size_t); }; struct U { __device__ void operator delete(void*, size_t) = delete; __host__ __device__ void operator delete(void*); }; __host__ __device__ void T::hd3() { host_fn(); // expected-error@-1 {{reference to __host__ function 'host_fn' in __host__ __device__ function}} } template __host__ __device__ void hd2() { host_fn(); } // expected-error@-1 {{reference to __host__ function 'host_fn' in __host__ __device__ function}} __global__ void kernel() { hd2(); } __host__ __device__ void hd() { host_fn(); } // expected-error@-1 {{reference to __host__ function 'host_fn' in __host__ __device__ function}} template __host__ __device__ void hd3() { host_fn(); } // expected-error@-1 {{reference to __host__ function 'host_fn' in __host__ __device__ function}} __device__ void device_fn() { hd3(); } // No error because this is never instantiated. template __host__ __device__ void hd4() { host_fn(); } __host__ __device__ void local_var() { S s; // expected-error@-1 {{reference to __host__ function 'S' in __host__ __device__ function}} } __host__ __device__ void placement_new(char *ptr) { ::new(ptr) S(); // expected-error@-1 {{reference to __host__ function 'S' in __host__ __device__ function}} } __host__ __device__ void explicit_destructor(S *s) { s->~S(); // expected-error@-1 {{reference to __host__ function '~S' in __host__ __device__ function}} } __host__ __device__ void class_specific_delete(T *t, U *u) { delete t; // ok, call sized device delete even though host has preferable non-sized version delete u; // ok, call non-sized HD delete rather than sized D delete } __host__ __device__ void hd_member_fn() { T t; // Necessary to trigger an error on T::hd. It's (implicitly) inline, so // isn't codegen'ed until we call it. t.hd(); } __host__ __device__ void h_member_fn() { T t; t.h(); // expected-error@-1 {{reference to __host__ function 'h' in __host__ __device__ function}} } __host__ __device__ void fn_ptr() { auto* ptr = &host_fn; // expected-error@-1 {{reference to __host__ function 'host_fn' in __host__ __device__ function}} } template __host__ __device__ void fn_ptr_template() { auto* ptr = &host_fn; // Not an error because the template isn't instantiated. } __host__ __device__ void unaryOp() { T t; (void) +t; // expected-error {{reference to __host__ function 'operator+' in __host__ __device__ function}} } __host__ __device__ void binaryOp() { T t; (void) (t - t); // expected-error {{reference to __host__ function 'operator-' in __host__ __device__ function}} } __host__ __device__ void implicitConversion() { T t; Dummy d = t; // expected-error {{reference to __host__ function 'operator Dummy' in __host__ __device__ function}} } template struct TmplStruct { template __host__ __device__ void fn() {} }; template <> template <> __host__ __device__ void TmplStruct::fn() { host_fn(); } // expected-error@-1 {{reference to __host__ function 'host_fn' in __host__ __device__ function}} __device__ void double_specialization() { TmplStruct().fn(); }