2014-09-13 02:44:36 +08:00
// RUN: %clang_cc1 %s -fsyntax-only -verify -pedantic -Wstrlcpy-strlcat-size -Wno-string-plus-int -triple=i686-apple-darwin9
2008-05-25 12:43:38 +08:00
// This test needs to set the target because it uses __builtin_ia32_vec_ext_v4si
2007-12-20 08:26:33 +08:00
int test1 ( float a , int b ) {
2013-08-17 08:46:16 +08:00
return __builtin_isless ( a , b ) ; // expected-note {{declared here}}
2007-12-20 08:26:33 +08:00
}
int test2 ( int a , int b ) {
return __builtin_islessequal ( a , b ) ; // expected-error {{floating point type}}
}
int test3 ( double a , float b ) {
return __builtin_isless ( a , b ) ;
}
int test4 ( int * a , double b ) {
return __builtin_islessequal ( a , b ) ; // expected-error {{floating point type}}
}
int test5 ( float a , long double b ) {
return __builtin_isless ( a , b , b ) ; // expected-error {{too many arguments}}
}
int test6 ( float a , long double b ) {
return __builtin_islessequal ( a ) ; // expected-error {{too few arguments}}
}
# define CFSTR __builtin___CFStringMakeConstantString
2009-05-08 14:58:22 +08:00
void test7 ( ) {
2009-12-31 06:10:22 +08:00
const void * X ;
2010-09-08 03:38:13 +08:00
X = CFSTR ( " \242 " ) ; // expected-warning {{input conversion stopped}}
2011-03-16 05:18:52 +08:00
X = CFSTR ( " \0 " ) ; // no-warning
2011-12-15 08:38:15 +08:00
X = CFSTR ( 242 ) ; // expected-error {{CFString literal is not a string constant}} expected-warning {{incompatible integer to pointer conversion}}
2009-12-31 06:10:22 +08:00
X = CFSTR ( " foo " , " bar " ) ; // expected-error {{too many arguments to function call}}
2007-12-20 08:26:33 +08:00
}
2008-05-06 06:18:14 +08:00
2009-05-08 14:58:22 +08:00
// atomics.
2009-07-22 08:43:08 +08:00
void test9 ( short v ) {
2009-05-08 14:58:22 +08:00
unsigned i , old ;
2010-10-15 16:48:51 +08:00
2009-05-08 14:58:22 +08:00
old = __sync_fetch_and_add ( ) ; // expected-error {{too few arguments to function call}}
old = __sync_fetch_and_add ( & old ) ; // expected-error {{too few arguments to function call}}
2012-11-02 09:40:23 +08:00
old = __sync_fetch_and_add ( ( unsigned * ) 0 , 42 i ) ; // expected-warning {{imaginary constants are a GNU extension}}
2010-07-10 02:59:35 +08:00
// PR7600: Pointers are implicitly casted to integers and back.
void * old_ptr = __sync_val_compare_and_swap ( ( void * * ) 0 , 0 , 0 ) ;
2010-07-18 15:23:17 +08:00
// Ensure the return type is correct even when implicit casts are stripped
// away. This triggers an assertion while checking the comparison otherwise.
if ( __sync_fetch_and_add ( & old , 1 ) = = 1 ) {
}
2009-05-08 14:58:22 +08:00
}
2009-09-21 11:09:59 +08:00
2012-09-22 17:05:22 +08:00
// overloaded atomics should be declared only once.
void test9_1 ( volatile int * ptr , int val ) {
2018-06-19 22:53:20 +08:00
__sync_fetch_and_add_4 ( ptr , val ) ;
}
void test9_2 ( volatile int * ptr , int val ) {
__sync_fetch_and_add ( ptr , val ) ;
}
void test9_3 ( volatile int * ptr , int val ) {
__sync_fetch_and_add_4 ( ptr , val ) ;
__sync_fetch_and_add ( ptr , val ) ;
__sync_fetch_and_add ( ptr , val ) ;
__sync_fetch_and_add_4 ( ptr , val ) ;
__sync_fetch_and_add_4 ( ptr , val ) ;
}
void test9_4 ( volatile int * ptr , int val ) {
// expected-warning@+1 {{the semantics of this intrinsic changed with GCC version 4.4 - the newer semantics are provided here}}
__sync_fetch_and_nand ( ptr , val ) ;
}
// rdar://7236819
void test10 ( void ) __attribute__ ( ( noreturn ) ) ;
void test10 ( void ) {
__asm__ ( " int3 " ) ;
__builtin_unreachable ( ) ;
// No warning about falling off the end of a noreturn function.
}
void test11 ( int X ) {
switch ( X ) {
case __builtin_eh_return_data_regno ( 0 ) : // constant foldable.
break ;
}
__builtin_eh_return_data_regno ( X ) ; // expected-error {{argument to '__builtin_eh_return_data_regno' must be a constant integer}}
}
// PR5062
void test12 ( void ) __attribute__ ( ( __noreturn__ ) ) ;
void test12 ( void ) {
__builtin_trap ( ) ; // no warning because trap is noreturn.
}
void test_unknown_builtin ( int a , int b ) {
__builtin_isles ( a , b ) ; / / expected - error { { use of unknown builtin } } \
// expected-note{{did you mean '__builtin_isless'?}}
}
int test13 ( ) {
__builtin_eh_return ( 0 , 0 ) ; // no warning, eh_return never returns.
}
// <rdar://problem/8228293>
void test14 ( ) {
int old ;
old = __sync_fetch_and_min ( ( volatile int * ) & old , 1 ) ;
}
// <rdar://problem/8336581>
void test15 ( const char * s ) {
__builtin_printf ( " string is %s \n " , s ) ;
}
// PR7885
int test16 ( ) {
return __builtin_constant_p ( ) + / / expected - error { { too few arguments } }
__builtin_constant_p ( 1 , 2 ) ; // expected-error {{too many arguments}}
}
2018-12-01 07:41:18 +08:00
// __builtin_constant_p cannot resolve non-constants as a file scoped array.
int expr ;
char y [ __builtin_constant_p ( expr ) ? - 1 : 1 ] ; // no warning, the builtin is false.
// no warning, the builtin is false.
struct foo { int a ; } ;
struct foo x = ( struct foo ) { __builtin_constant_p ( 42 ) ? 37 : 927 } ;
2018-06-19 22:53:20 +08:00
const int test17_n = 0 ;
const char test17_c [ ] = { 1 , 2 , 3 , 0 } ;
const char test17_d [ ] = { 1 , 2 , 3 , 4 } ;
typedef int __attribute__ ( ( vector_size ( 16 ) ) ) IntVector ;
struct Aggregate { int n ; char c ; } ;
enum Enum { EnumValue1 , EnumValue2 } ;
typedef __typeof ( sizeof ( int ) ) size_t ;
size_t strlen ( const char * ) ;
void test17 ( ) {
# define ASSERT(...) { int arr[(__VA_ARGS__) ? 1 : -1]; }
# define T(...) ASSERT(__builtin_constant_p(__VA_ARGS__))
# define F(...) ASSERT(!__builtin_constant_p(__VA_ARGS__))
// __builtin_constant_p returns 1 if the argument folds to:
// - an arithmetic constant with value which is known at compile time
T ( test17_n ) ;
T ( & test17_c [ 3 ] - test17_c ) ;
T ( 3 i + 5 ) ; // expected-warning {{imaginary constant}}
T ( 4.2 * 7.6 ) ;
T ( EnumValue1 ) ;
T ( ( enum Enum ) ( int ) EnumValue2 ) ;
// - the address of the first character of a string literal, losslessly cast
// to any type
T ( " string literal " ) ;
T ( ( double * ) " string literal " ) ;
T ( " string literal " + 0 ) ;
T ( ( long ) " string literal " ) ;
// ... and otherwise returns 0.
F ( " string literal " + 1 ) ;
F ( & test17_n ) ;
F ( test17_c ) ;
F ( & test17_c ) ;
F ( & test17_d ) ;
F ( ( struct Aggregate ) { 0 , 1 } ) ;
F ( ( IntVector ) { 0 , 1 , 2 , 3 } ) ;
2018-12-01 07:41:18 +08:00
F ( test17 ) ;
2018-06-19 22:53:20 +08:00
// Ensure that a technique used in glibc is handled correctly.
# define OPT(...) (__builtin_constant_p(__VA_ARGS__) && strlen(__VA_ARGS__) < 4)
// FIXME: These are incorrectly treated as ICEs because strlen is treated as
// a builtin.
ASSERT ( OPT ( " abc " ) ) ;
ASSERT ( ! OPT ( " abcd " ) ) ;
// In these cases, the strlen is non-constant, but the __builtin_constant_p
// is 0: the array size is not an ICE but is foldable.
ASSERT ( ! OPT ( test17_c ) ) ; // expected-warning {{folded}}
ASSERT ( ! OPT ( & test17_c [ 0 ] ) ) ; // expected-warning {{folded}}
ASSERT ( ! OPT ( ( char * ) test17_c ) ) ; // expected-warning {{folded}}
ASSERT ( ! OPT ( test17_d ) ) ; // expected-warning {{folded}}
ASSERT ( ! OPT ( & test17_d [ 0 ] ) ) ; // expected-warning {{folded}}
ASSERT ( ! OPT ( ( char * ) test17_d ) ) ; // expected-warning {{folded}}
# undef OPT
# undef T
# undef F
}
void test18 ( ) {
char src [ 1024 ] ;
char dst [ 2048 ] ;
size_t result ;
void * ptr ;
ptr = __builtin___memccpy_chk ( dst , src , ' \037 ' , sizeof ( src ) , sizeof ( dst ) ) ;
result = __builtin___strlcpy_chk ( dst , src , sizeof ( dst ) , sizeof ( dst ) ) ;
result = __builtin___strlcat_chk ( dst , src , sizeof ( dst ) , sizeof ( dst ) ) ;
ptr = __builtin___memccpy_chk ( dst , src , ' \037 ' , sizeof ( src ) ) ; // expected-error {{too few arguments to function call}}
ptr = __builtin___strlcpy_chk ( dst , src , sizeof ( dst ) , sizeof ( dst ) ) ; // expected-warning {{incompatible integer to pointer conversion}}
ptr = __builtin___strlcat_chk ( dst , src , sizeof ( dst ) , sizeof ( dst ) ) ; // expected-warning {{incompatible integer to pointer conversion}}
}
void no_ms_builtins ( ) {
__assume ( 1 ) ; // expected-warning {{implicit declaration}}
__noop ( 1 ) ; // expected-warning {{implicit declaration}}
__debugbreak ( ) ; // expected-warning {{implicit declaration}}
}
void unavailable ( ) {
__builtin_operator_new ( 0 ) ; // expected-error {{'__builtin_operator_new' is only available in C++}}
__builtin_operator_delete ( 0 ) ; // expected-error {{'__builtin_operator_delete' is only available in C++}}
}
// rdar://18259539
size_t strlcpy ( char * restrict dst , const char * restrict src , size_t size ) ;
size_t strlcat ( char * restrict dst , const char * restrict src , size_t size ) ;
void Test19 ( void )
{
static char b [ 40 ] ;
static char buf [ 20 ] ;
strlcpy ( buf , b , sizeof ( b ) ) ; / / expected - warning { { size argument in ' strlcpy ' call appears to be size of the source ; expected the size of the destination } } \ \
// expected-note {{change size argument to be the size of the destination}}
__builtin___strlcpy_chk ( buf , b , sizeof ( b ) , __builtin_object_size ( buf , 0 ) ) ; / / expected - warning { { size argument in ' __builtin___strlcpy_chk ' call appears to be size of the source ; expected the size of the destination } } \
/ / expected - note { { change size argument to be the size of the destination } } \
2019-03-19 03:23:45 +08:00
// expected-warning {{'strlcpy' will always overflow; destination buffer has size 20, but size argument is 40}}
2018-06-19 22:53:20 +08:00
strlcat ( buf , b , sizeof ( b ) ) ; / / expected - warning { { size argument in ' strlcat ' call appears to be size of the source ; expected the size of the destination } } \
// expected-note {{change size argument to be the size of the destination}}
__builtin___strlcat_chk ( buf , b , sizeof ( b ) , __builtin_object_size ( buf , 0 ) ) ; / / expected - warning { { size argument in ' __builtin___strlcat_chk ' call appears to be size of the source ; expected the size of the destination } } \
/ / expected - note { { change size argument to be the size of the destination } } \
2019-03-19 03:23:45 +08:00
// expected-warning {{'strlcat' will always overflow; destination buffer has size 20, but size argument is 40}}
2018-06-19 22:53:20 +08:00
}
// rdar://11076881
char * Test20 ( char * p , const char * in , unsigned n )
{
static char buf [ 10 ] ;
2019-03-19 03:23:45 +08:00
__builtin___memcpy_chk ( & buf [ 6 ] , in , 5 , __builtin_object_size ( & buf [ 6 ] , 0 ) ) ; // expected-warning {{'memcpy' will always overflow; destination buffer has size 4, but size argument is 5}}
2018-06-19 22:53:20 +08:00
__builtin___memcpy_chk ( p , " abcde " , n , __builtin_object_size ( p , 0 ) ) ;
__builtin___memcpy_chk ( & buf [ 5 ] , " abcde " , 5 , __builtin_object_size ( & buf [ 5 ] , 0 ) ) ;
__builtin___memcpy_chk ( & buf [ 5 ] , " abcde " , n , __builtin_object_size ( & buf [ 5 ] , 0 ) ) ;
2019-03-19 03:23:45 +08:00
__builtin___memcpy_chk ( & buf [ 6 ] , " abcde " , 5 , __builtin_object_size ( & buf [ 6 ] , 0 ) ) ; // expected-warning {{'memcpy' will always overflow; destination buffer has size 4, but size argument is 5}}
2018-06-19 22:53:20 +08:00
return buf ;
}
[Clang] Add __builtin_launder
Summary:
This patch adds `__builtin_launder`, which is required to implement `std::launder`. Additionally GCC provides `__builtin_launder`, so thing brings Clang in-line with GCC.
I'm not exactly sure what magic `__builtin_launder` requires, but based on previous discussions this patch applies a `@llvm.invariant.group.barrier`. As noted in previous discussions, this may not be enough to correctly handle vtables.
Reviewers: rnk, majnemer, rsmith
Reviewed By: rsmith
Subscribers: kristina, Romain-Geissler-1A, erichkeane, amharc, jroelofs, cfe-commits, Prazek
Differential Revision: https://reviews.llvm.org/D40218
llvm-svn: 349195
2018-12-15 05:11:28 +08:00
typedef void ( fn_t ) ( int ) ;
void test_builtin_launder ( char * p , void * vp , const void * cvp ,
const volatile int * ip , float * restrict fp ,
fn_t * fn ) {
__builtin_launder ( ) ; // expected-error {{too few arguments to function call, expected 1, have 0}}
__builtin_launder ( p , p ) ; // expected-error {{too many arguments to function call, expected 1, have 2}}
int x ;
__builtin_launder ( x ) ; // expected-error {{non-pointer argument to '__builtin_launder' is not allowed}}
char * d = __builtin_launder ( p ) ;
__builtin_launder ( vp ) ; // expected-error {{void pointer argument to '__builtin_launder' is not allowed}}
__builtin_launder ( cvp ) ; // expected-error {{void pointer argument to '__builtin_launder' is not allowed}}
const volatile int * id = __builtin_launder ( ip ) ;
int * id2 = __builtin_launder ( ip ) ; // expected-warning {{discards qualifiers}}
float * fd = __builtin_launder ( fp ) ;
__builtin_launder ( fn ) ; // expected-error {{function pointer argument to '__builtin_launder' is not allowed}}
}
2018-06-19 22:53:20 +08:00
void test21 ( const int * ptr ) {
__sync_fetch_and_add ( ptr , 1 ) ; // expected-error{{address argument to atomic builtin cannot be const-qualified ('const int *' invalid)}}
__atomic_fetch_add ( ptr , 1 , 0 ) ; // expected-error {{address argument to atomic operation must be a pointer to non-const type ('const int *' invalid)}}
}
2018-06-19 22:59:11 +08:00
void test22 ( void ) {
( void ) __builtin_signbit ( ) ; // expected-error{{too few arguments to function call, expected 1, have 0}}
( void ) __builtin_signbit ( 1.0 , 2.0 , 3.0 ) ; // expected-error{{too many arguments to function call, expected 1, have 3}}
( void ) __builtin_signbit ( 1 ) ; // expected-error {{floating point classification requires argument of floating point type (passed in 'int')}}
( void ) __builtin_signbit ( 1.0 ) ;
( void ) __builtin_signbit ( 1.0f ) ;
( void ) __builtin_signbit ( 1.0 L ) ;
( void ) __builtin_signbitf ( ) ; // expected-error{{too few arguments to function call, expected 1, have 0}}
( void ) __builtin_signbitf ( 1.0 , 2.0 , 3.0 ) ; // expected-error{{too many arguments to function call, expected 1, have 3}}
( void ) __builtin_signbitf ( 1 ) ;
( void ) __builtin_signbitf ( 1.0 ) ;
( void ) __builtin_signbitf ( 1.0f ) ;
( void ) __builtin_signbitf ( 1.0 L ) ;
( void ) __builtin_signbitl ( ) ; // expected-error{{too few arguments to function call, expected 1, have 0}}
( void ) __builtin_signbitl ( 1.0 , 2.0 , 3.0 ) ; // expected-error{{too many arguments to function call, expected 1, have 3}}
( void ) __builtin_signbitl ( 1 ) ;
( void ) __builtin_signbitl ( 1.0 ) ;
( void ) __builtin_signbitl ( 1.0f ) ;
( void ) __builtin_signbitl ( 1.0 L ) ;
}
2018-09-07 01:19:33 +08:00
// rdar://43909200
# define memcpy(x,y,z) __builtin___memcpy_chk(x,y,z, __builtin_object_size(x,0))
# define my_memcpy(x,y,z) __builtin___memcpy_chk(x,y,z, __builtin_object_size(x,0))
void test23 ( ) {
char src [ 1024 ] ;
char buf [ 10 ] ;
memcpy ( buf , src , 11 ) ; // expected-warning{{'memcpy' will always overflow; destination buffer has size 10, but size argument is 11}}
2019-03-19 03:23:45 +08:00
my_memcpy ( buf , src , 11 ) ; // expected-warning{{'memcpy' will always overflow; destination buffer has size 10, but size argument is 11}}
2018-09-07 01:19:33 +08:00
}
2019-04-24 10:23:30 +08:00
// Test that __builtin_is_constant_evaluated() is not allowed in C
int test_cxx_builtin ( ) {
// expected-error@+1 {{use of unknown builtin '__builtin_is_constant_evaluated'}}
return __builtin_is_constant_evaluated ( ) ;
}