2019-09-13 14:02:15 +08:00
// RUN: %clang_cc1 -flax-vector-conversions=all -triple x86_64-apple-darwin10 -fsyntax-only -verify %s
// RUN: %clang_cc1 -flax-vector-conversions=all -triple x86_64-apple-darwin10 -fsyntax-only -verify -std=c++98 %s
// RUN: %clang_cc1 -flax-vector-conversions=all -triple x86_64-apple-darwin10 -fsyntax-only -verify -std=c++11 %s
// RUN: %clang_cc1 -flax-vector-conversions=integer -triple x86_64-apple-darwin10 -fsyntax-only -verify %s -DNO_LAX_FLOAT
// RUN: %clang_cc1 -flax-vector-conversions=none -triple x86_64-apple-darwin10 -fsyntax-only -verify %s -DNO_LAX_FLOAT -DNO_LAX_INT
2015-11-18 04:25:05 +08:00
2010-05-19 06:42:18 +08:00
typedef char char16 __attribute__ ( ( __vector_size__ ( 16 ) ) ) ;
typedef long long longlong16 __attribute__ ( ( __vector_size__ ( 16 ) ) ) ;
typedef char char16_e __attribute__ ( ( __ext_vector_type__ ( 16 ) ) ) ;
typedef long long longlong16_e __attribute__ ( ( __ext_vector_type__ ( 2 ) ) ) ;
// Test overloading and function calls with vector types.
2019-09-13 14:02:15 +08:00
void f0 ( char16 ) ; // expected-note 0+{{candidate}}
2010-05-19 06:42:18 +08:00
void f0_test ( char16 c16 , longlong16 ll16 , char16_e c16e , longlong16_e ll16e ) {
f0 ( c16 ) ;
f0 ( ll16 ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_INT
// expected-error@-2 {{no matching function}}
# endif
2010-05-19 06:42:18 +08:00
f0 ( c16e ) ;
f0 ( ll16e ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_INT
// expected-error@-2 {{no matching function}}
# endif
2010-05-19 06:42:18 +08:00
}
2018-11-16 11:00:00 +08:00
int & f1 ( char16 ) ;
float & f1 ( longlong16 ) ;
2010-05-19 06:42:18 +08:00
void f1_test ( char16 c16 , longlong16 ll16 , char16_e c16e , longlong16_e ll16e ) {
int & ir1 = f1 ( c16 ) ;
float & fr1 = f1 ( ll16 ) ;
2018-11-16 11:00:00 +08:00
int & ir2 = f1 ( c16e ) ;
float & fr2 = f1 ( ll16e ) ;
2010-05-19 06:42:18 +08:00
}
2019-09-13 14:02:15 +08:00
void f2 ( char16_e ) ; // expected-note 0+{{candidate}}
2010-05-19 06:42:18 +08:00
void f2_test ( char16 c16 , longlong16 ll16 , char16_e c16e , longlong16_e ll16e ) {
f2 ( c16 ) ;
f2 ( ll16 ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_INT
// expected-error@-2 {{no matching function}}
# endif
2010-05-19 06:42:18 +08:00
f2 ( c16e ) ;
f2 ( ll16e ) ; // expected-error{{no matching function}}
f2 ( ' a ' ) ;
f2 ( 17 ) ;
}
// Test the conditional operator with vector types.
2013-09-14 04:43:08 +08:00
void conditional ( bool Cond , char16 c16 , longlong16 ll16 , char16_e c16e ,
2010-05-19 06:42:18 +08:00
longlong16_e ll16e ) {
// Conditional operators with the same type.
__typeof__ ( Cond ? c16 : c16 ) * c16p1 = & c16 ;
__typeof__ ( Cond ? ll16 : ll16 ) * ll16p1 = & ll16 ;
__typeof__ ( Cond ? c16e : c16e ) * c16ep1 = & c16e ;
__typeof__ ( Cond ? ll16e : ll16e ) * ll16ep1 = & ll16e ;
// Conditional operators with similar types.
__typeof__ ( Cond ? c16 : c16e ) * c16ep2 = & c16e ;
__typeof__ ( Cond ? c16e : c16 ) * c16ep3 = & c16e ;
__typeof__ ( Cond ? ll16 : ll16e ) * ll16ep2 = & ll16e ;
__typeof__ ( Cond ? ll16e : ll16 ) * ll16ep3 = & ll16e ;
2010-08-26 08:42:16 +08:00
// Conditional operators with compatible types under -flax-vector-conversions (default)
( void ) ( Cond ? c16 : ll16 ) ;
( void ) ( Cond ? ll16e : c16e ) ;
( void ) ( Cond ? ll16e : c16 ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_INT
// expected-error@-4 {{cannot convert}}
// expected-error@-4 {{cannot convert}}
// expected-error@-4 {{cannot convert}}
# endif
2010-05-19 06:42:18 +08:00
}
// Test C++ cast'ing of vector types.
void casts ( longlong16 ll16 , longlong16_e ll16e ) {
// C-style casts.
( void ) ( char16 ) ll16 ;
( void ) ( char16_e ) ll16 ;
( void ) ( longlong16 ) ll16 ;
( void ) ( longlong16_e ) ll16 ;
( void ) ( char16 ) ll16e ;
( void ) ( char16_e ) ll16e ;
( void ) ( longlong16 ) ll16e ;
( void ) ( longlong16_e ) ll16e ;
// Function-style casts.
( void ) char16 ( ll16 ) ;
( void ) char16_e ( ll16 ) ;
( void ) longlong16 ( ll16 ) ;
( void ) longlong16_e ( ll16 ) ;
( void ) char16 ( ll16e ) ;
( void ) char16_e ( ll16e ) ;
( void ) longlong16 ( ll16e ) ;
( void ) longlong16_e ( ll16e ) ;
// static_cast
( void ) static_cast < char16 > ( ll16 ) ;
( void ) static_cast < char16_e > ( ll16 ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_INT
// expected-error@-3 {{not allowed}}
// expected-error@-3 {{not allowed}}
# endif
2010-05-19 06:42:18 +08:00
( void ) static_cast < longlong16 > ( ll16 ) ;
( void ) static_cast < longlong16_e > ( ll16 ) ;
( void ) static_cast < char16 > ( ll16e ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_INT
// expected-error@-2 {{not allowed}}
# endif
2014-04-26 04:41:38 +08:00
( void ) static_cast < char16_e > ( ll16e ) ; // expected-error{{static_cast from 'longlong16_e' (vector of 2 'long long' values) to 'char16_e' (vector of 16 'char' values) is not allowed}}
2010-05-19 06:42:18 +08:00
( void ) static_cast < longlong16 > ( ll16e ) ;
( void ) static_cast < longlong16_e > ( ll16e ) ;
// reinterpret_cast
( void ) reinterpret_cast < char16 > ( ll16 ) ;
( void ) reinterpret_cast < char16_e > ( ll16 ) ;
( void ) reinterpret_cast < longlong16 > ( ll16 ) ;
( void ) reinterpret_cast < longlong16_e > ( ll16 ) ;
( void ) reinterpret_cast < char16 > ( ll16e ) ;
( void ) reinterpret_cast < char16_e > ( ll16e ) ;
( void ) reinterpret_cast < longlong16 > ( ll16e ) ;
( void ) reinterpret_cast < longlong16_e > ( ll16e ) ;
}
2010-05-19 07:05:44 +08:00
template < typename T >
2015-11-18 04:25:05 +08:00
struct convertible_to { // expected-note 3 {{candidate function (the implicit copy assignment operator) not viable}}
# if __cplusplus >= 201103L // C++11 or later
// expected-note@-2 3 {{candidate function (the implicit move assignment operator) not viable}}
# endif
2010-05-19 07:05:44 +08:00
operator T ( ) const ;
} ;
2013-09-14 04:43:08 +08:00
void test_implicit_conversions ( bool Cond , char16 c16 , longlong16 ll16 ,
2010-05-19 11:21:00 +08:00
char16_e c16e , longlong16_e ll16e ,
2013-09-14 04:43:08 +08:00
convertible_to < char16 > to_c16 ,
convertible_to < longlong16 > to_ll16 ,
convertible_to < char16_e > to_c16e ,
2010-05-19 11:21:00 +08:00
convertible_to < longlong16_e > to_ll16e ,
convertible_to < char16 & > rto_c16 ,
convertible_to < char16_e & > rto_c16e ) {
2010-05-19 07:05:44 +08:00
f0 ( to_c16 ) ;
f0 ( to_ll16 ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_INT
// expected-error@-2 {{no matching function}}
# endif
2010-05-19 07:05:44 +08:00
f0 ( to_c16e ) ;
f0 ( to_ll16e ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_INT
// expected-error@-2 {{no matching function}}
# endif
2010-05-19 07:05:44 +08:00
f2 ( to_c16 ) ;
f2 ( to_ll16 ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_INT
// expected-error@-2 {{no matching function}}
# endif
2010-05-19 07:05:44 +08:00
f2 ( to_c16e ) ;
f2 ( to_ll16e ) ; // expected-error{{no matching function}}
2010-05-19 11:21:00 +08:00
( void ) ( c16 = = c16e ) ;
( void ) ( c16 = = to_c16 ) ;
( void ) + to_c16 ;
( void ) - to_c16 ;
( void ) ~ to_c16 ;
( void ) ( to_c16 = = to_c16e ) ;
( void ) ( to_c16 ! = to_c16e ) ;
( void ) ( to_c16 < to_c16e ) ;
( void ) ( to_c16 < = to_c16e ) ;
( void ) ( to_c16 > to_c16e ) ;
( void ) ( to_c16 > = to_c16e ) ;
( void ) ( to_c16 + to_c16 ) ;
( void ) ( to_c16 - to_c16 ) ;
( void ) ( to_c16 * to_c16 ) ;
( void ) ( to_c16 / to_c16 ) ;
( void ) ( rto_c16 = to_c16 ) ; // expected-error{{no viable overloaded '='}}
( void ) ( rto_c16 + = to_c16 ) ;
( void ) ( rto_c16 - = to_c16 ) ;
( void ) ( rto_c16 * = to_c16 ) ;
( void ) ( rto_c16 / = to_c16 ) ;
( void ) + to_c16e ;
( void ) - to_c16e ;
( void ) ~ to_c16e ;
( void ) ( to_c16e = = to_c16e ) ;
( void ) ( to_c16e ! = to_c16e ) ;
( void ) ( to_c16e < to_c16e ) ;
( void ) ( to_c16e < = to_c16e ) ;
( void ) ( to_c16e > to_c16e ) ;
( void ) ( to_c16e > = to_c16e ) ;
( void ) ( to_c16e + to_c16 ) ;
( void ) ( to_c16e - to_c16 ) ;
( void ) ( to_c16e * to_c16 ) ;
( void ) ( to_c16e / to_c16 ) ;
( void ) ( rto_c16e = to_c16 ) ; // expected-error{{no viable overloaded '='}}
( void ) ( rto_c16e + = to_c16 ) ;
( void ) ( rto_c16e - = to_c16 ) ;
( void ) ( rto_c16e * = to_c16 ) ;
( void ) ( rto_c16e / = to_c16 ) ;
( void ) + to_c16 ;
( void ) - to_c16 ;
( void ) ~ to_c16 ;
( void ) ( to_c16 = = to_c16e ) ;
( void ) ( to_c16 ! = to_c16e ) ;
( void ) ( to_c16 < to_c16e ) ;
( void ) ( to_c16 < = to_c16e ) ;
( void ) ( to_c16 > to_c16e ) ;
( void ) ( to_c16 > = to_c16e ) ;
( void ) ( to_c16 + to_c16e ) ;
( void ) ( to_c16 - to_c16e ) ;
( void ) ( to_c16 * to_c16e ) ;
( void ) ( to_c16 / to_c16e ) ;
( void ) ( rto_c16 = c16e ) ; // expected-error{{no viable overloaded '='}}
2011-06-24 02:10:35 +08:00
( void ) ( rto_c16 + = to_c16e ) ;
( void ) ( rto_c16 - = to_c16e ) ;
( void ) ( rto_c16 * = to_c16e ) ;
( void ) ( rto_c16 / = to_c16e ) ;
2010-05-19 11:21:00 +08:00
( void ) ( Cond ? to_c16 : to_c16e ) ;
( void ) ( Cond ? to_ll16e : to_ll16 ) ;
2013-09-14 04:43:08 +08:00
2018-04-06 23:14:32 +08:00
// These 2 are convertible with -flax-vector-conversions (default)
2010-08-26 08:42:16 +08:00
( void ) ( Cond ? to_c16 : to_ll16 ) ;
( void ) ( Cond ? to_c16e : to_ll16e ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_INT
// expected-error@-3 {{cannot convert}}
// expected-error@-3 {{cannot convert}}
# endif
2010-05-19 07:05:44 +08:00
}
2010-06-23 00:52:27 +08:00
typedef float fltx2 __attribute__ ( ( __vector_size__ ( 8 ) ) ) ;
typedef float fltx4 __attribute__ ( ( __vector_size__ ( 16 ) ) ) ;
typedef double dblx2 __attribute__ ( ( __vector_size__ ( 16 ) ) ) ;
typedef double dblx4 __attribute__ ( ( __vector_size__ ( 32 ) ) ) ;
2014-04-26 04:41:38 +08:00
void accept_fltx2 ( fltx2 ) ; // expected-note{{candidate function not viable: no known conversion from 'double' to 'fltx2' (vector of 2 'float' values) for 1st argument}}
2010-06-23 00:52:27 +08:00
void accept_fltx4 ( fltx4 ) ;
void accept_dblx2 ( dblx2 ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_FLOAT
// expected-note@-3 {{no known conversion}}
// expected-note@-3 {{no known conversion}}
# endif
2010-06-23 00:52:27 +08:00
void accept_dblx4 ( dblx4 ) ;
2014-04-26 04:41:38 +08:00
void accept_bool ( bool ) ; // expected-note{{candidate function not viable: no known conversion from 'fltx2' (vector of 2 'float' values) to 'bool' for 1st argument}}
2010-06-23 00:52:27 +08:00
void test ( fltx2 fltx2_val , fltx4 fltx4_val , dblx2 dblx2_val , dblx4 dblx4_val ) {
// Exact matches
accept_fltx2 ( fltx2_val ) ;
accept_fltx4 ( fltx4_val ) ;
accept_dblx2 ( dblx2_val ) ;
accept_dblx4 ( dblx4_val ) ;
// Same-size conversions
accept_fltx4 ( dblx2_val ) ;
accept_dblx2 ( fltx4_val ) ;
2019-09-13 14:02:15 +08:00
# ifdef NO_LAX_FLOAT
// expected-error@-3 {{no matching function}}
// expected-error@-3 {{no matching function}}
# endif
2010-06-23 00:52:27 +08:00
// Conversion to bool.
accept_bool ( fltx2_val ) ; // expected-error{{no matching function for call to 'accept_bool'}}
// Scalar-to-vector conversions.
accept_fltx2 ( 1.0 ) ; // expected-error{{no matching function for call to 'accept_fltx2'}}
}
2011-10-28 11:31:48 +08:00
typedef int intx4 __attribute__ ( ( __vector_size__ ( 16 ) ) ) ;
typedef int inte4 __attribute__ ( ( __ext_vector_type__ ( 4 ) ) ) ;
2019-09-13 14:02:15 +08:00
typedef float flte4 __attribute__ ( ( __ext_vector_type__ ( 4 ) ) ) ;
2011-10-28 11:31:48 +08:00
2019-09-13 14:02:15 +08:00
void test_mixed_vector_types ( fltx4 f , intx4 n , flte4 g , inte4 m ) {
2011-10-28 11:31:48 +08:00
( void ) ( f = = g ) ;
( void ) ( g ! = f ) ;
( void ) ( f < = g ) ;
( void ) ( g > = f ) ;
( void ) ( f < g ) ;
( void ) ( g > f ) ;
( void ) ( + g ) ;
( void ) ( - g ) ;
( void ) ( f + g ) ;
( void ) ( f - g ) ;
( void ) ( f * g ) ;
( void ) ( f / g ) ;
( void ) ( f = g ) ;
( void ) ( f + = g ) ;
( void ) ( f - = g ) ;
( void ) ( f * = g ) ;
( void ) ( f / = g ) ;
( void ) ( n = = m ) ;
( void ) ( m ! = n ) ;
( void ) ( n < = m ) ;
( void ) ( m > = n ) ;
( void ) ( n < m ) ;
( void ) ( m > n ) ;
( void ) ( + m ) ;
( void ) ( - m ) ;
( void ) ( ~ m ) ;
( void ) ( n + m ) ;
( void ) ( n - m ) ;
( void ) ( n * m ) ;
( void ) ( n / m ) ;
( void ) ( n % m ) ;
( void ) ( n = m ) ;
( void ) ( n + = m ) ;
( void ) ( n - = m ) ;
( void ) ( n * = m ) ;
( void ) ( n / = m ) ;
}
2012-09-10 22:57:06 +08:00
template < typename T > void test_pseudo_dtor_tmpl ( T * ptr ) {
ptr - > ~ T ( ) ;
( * ptr ) . ~ T ( ) ;
}
void test_pseudo_dtor ( fltx4 * f ) {
f - > ~ fltx4 ( ) ;
( * f ) . ~ fltx4 ( ) ;
test_pseudo_dtor_tmpl ( f ) ;
}
2013-06-18 05:09:57 +08:00
// PR16204
typedef __attribute__ ( ( ext_vector_type ( 4 ) ) ) int vi4 ;
const int & reference_to_vec_element = vi4 ( 1 ) . x ;
2013-09-14 04:43:08 +08:00
// PR12649
typedef bool bad __attribute__ ( ( __vector_size__ ( 16 ) ) ) ; // expected-error {{invalid vector element type 'bool'}}
2018-07-14 03:46:04 +08:00
namespace Templates {
template < typename Elt , unsigned Size >
struct TemplateVectorType {
2019-09-13 14:02:15 +08:00
typedef Elt __attribute__ ( ( __vector_size__ ( Size ) ) ) type ; // #1
2018-07-14 03:46:04 +08:00
} ;
template < int N , typename T >
struct PR15730 {
typedef T __attribute__ ( ( vector_size ( N * sizeof ( T ) ) ) ) type ;
2019-09-13 14:02:15 +08:00
typedef T __attribute__ ( ( vector_size ( 8192 ) ) ) type2 ; // #2
typedef T __attribute__ ( ( vector_size ( 3 ) ) ) type3 ; // #3
2018-07-14 03:46:04 +08:00
} ;
void Init ( ) {
const TemplateVectorType < float , 32 > : : type Works = { } ;
const TemplateVectorType < int , 32 > : : type Works2 = { } ;
2019-09-13 14:02:15 +08:00
// expected-error@#1 {{invalid vector element type 'bool'}}
2018-07-14 03:46:04 +08:00
// expected-note@+1 {{in instantiation of template class 'Templates::TemplateVectorType<bool, 32>' requested here}}
const TemplateVectorType < bool , 32 > : : type NoBool ;
2019-09-13 14:02:15 +08:00
// expected-error@#1 {{invalid vector element type 'int __attribute__((ext_vector_type(4)))' (vector of 4 'int' values)}}
2018-07-14 03:46:04 +08:00
// expected-note@+1 {{in instantiation of template class 'Templates::TemplateVectorType<int __attribute__((ext_vector_type(4))), 32>' requested here}}
const TemplateVectorType < vi4 , 32 > : : type NoComplex ;
2019-09-13 14:02:15 +08:00
// expected-error@#1 {{vector size not an integral multiple of component size}}
2018-07-14 03:46:04 +08:00
// expected-note@+1 {{in instantiation of template class 'Templates::TemplateVectorType<int, 33>' requested here}}
const TemplateVectorType < int , 33 > : : type BadSize ;
2019-09-13 14:02:15 +08:00
// expected-error@#1 {{vector size too large}}
2018-07-14 03:46:04 +08:00
// expected-note@+1 {{in instantiation of template class 'Templates::TemplateVectorType<int, 8192>' requested here}}
const TemplateVectorType < int , 8192 > : : type TooLarge ;
2019-09-13 14:02:15 +08:00
// expected-error@#1 {{zero vector size}}
2018-07-14 03:46:04 +08:00
// expected-note@+1 {{in instantiation of template class 'Templates::TemplateVectorType<int, 0>' requested here}}
const TemplateVectorType < int , 0 > : : type Zero ;
2019-09-13 14:02:15 +08:00
// expected-error@#2 {{vector size too large}}
// expected-error@#3 {{vector size not an integral multiple of component size}}
2018-07-14 03:46:04 +08:00
// expected-note@+1 {{in instantiation of template class 'Templates::PR15730<8, int>' requested here}}
const PR15730 < 8 , int > : : type PR15730_1 = { } ;
2019-09-13 14:02:15 +08:00
// expected-error@#2 {{vector size too large}}
2018-07-14 03:46:04 +08:00
// expected-note@+1 {{in instantiation of template class 'Templates::PR15730<8, char>' requested here}}
const PR15730 < 8 , char > : : type2 PR15730_2 = { } ;
}
} // namespace Templates
2019-08-02 19:19:35 +08:00
typedef int inte2 __attribute__ ( ( __ext_vector_type__ ( 2 ) ) ) ;
void test_vector_literal ( inte4 res ) {
inte2 a = ( inte2 ) ( 1 , 2 ) ; //expected-warning{{expression result unused}}
inte4 b = ( inte4 ) ( a , a ) ; //expected-error{{C-style cast from vector 'inte2' (vector of 2 'int' values) to vector 'inte4' (vector of 4 'int' values) of different size}} //expected-warning{{expression result unused}}
}
2019-08-15 06:57:50 +08:00
typedef __attribute__ ( ( __ext_vector_type__ ( 4 ) ) ) float vector_float4 ;
typedef __attribute__ ( ( __ext_vector_type__ ( 4 ) ) ) int vector_int4 ;
namespace swizzle_template_confusion {
template < typename T > struct xyzw { } ;
vector_int4 foo123 ( vector_float4 & A , vector_float4 & B ) {
return A . xyzw < B . x & & B . y > A . y ; // OK, not a template-id
}
}
namespace swizzle_typo_correction {
template < typename T > struct xyzv { } ;
vector_int4 foo123 ( vector_float4 & A , vector_float4 & B ) {
return A . xyzw < B . x & & B . y > A . y ; // OK, not a typo for 'xyzv'
}
}