blob: 34e3e03ba9d5dbe1f573597fe93786c24589ec5c [file] [log] [blame]
/* Test wrong usage of C23 nullptr. */
/* { dg-do compile } */
/* { dg-options "-std=c2x -pedantic-errors -Wall -Wextra -Wno-unused-variable" } */
typedef __typeof__(nullptr) nullptr_t;
void g (nullptr_t); /* { dg-message "expected .nullptr_t. but argument is of type .int." } */
nullptr_t cmp (void);
void
test1 (int *p)
{
(void) (p > nullptr); /* { dg-error "ordered comparison" } */
(void) (p >= nullptr); /* { dg-error "ordered comparison" } */
(void) (p < nullptr); /* { dg-error "ordered comparison" } */
(void) (p <= nullptr); /* { dg-error "ordered comparison" } */
(void) (nullptr == 1); /* { dg-error "invalid operands" } */
(void) (1 == nullptr); /* { dg-error "invalid operands" } */
(void) (nullptr != 1); /* { dg-error "invalid operands" } */
(void) (1 != nullptr); /* { dg-error "invalid operands" } */
(void) (1 > nullptr); /* { dg-error "invalid operands" } */
/* "(nullptr_t)nullptr" has type nullptr_t but isn't an NPC. */
(void) ((nullptr_t)nullptr == p); /* { dg-error "invalid operands" } */
(void) ((nullptr_t)nullptr != p); /* { dg-error "invalid operands" } */
(void) (p == (nullptr_t)nullptr); /* { dg-error "invalid operands" } */
(void) (p != (nullptr_t)nullptr); /* { dg-error "invalid operands" } */
(void) (cmp () == p); /* { dg-error "invalid operands" } */
(void) (cmp () != p); /* { dg-error "invalid operands" } */
(void) (p == cmp ()); /* { dg-error "invalid operands" } */
(void) (p != cmp ()); /* { dg-error "invalid operands" } */
/* "(void *)nullptr" is not an NPC, either. */
(void) ((void *)nullptr == cmp ()); /* { dg-error "invalid operands" } */
(void) ((void *)nullptr != cmp ()); /* { dg-error "invalid operands" } */
(void) (cmp () == (void *)nullptr); /* { dg-error "invalid operands" } */
(void) (cmp () != (void *)nullptr); /* { dg-error "invalid operands" } */
}
void
test2 (void)
{
const nullptr_t nptr = nullptr;
int p = nullptr; /* { dg-error "incompatible types" } */
float d = nullptr; /* { dg-error "incompatible types" } */
char arr[10] = { nullptr }; /* { dg-error "incompatible types" } */
/* No type other than nullptr_t shall be converted to nullptr_t. */
const nullptr_t n = 0; /* { dg-error "invalid initializer" } */
+(nullptr_t) 0; /* { dg-error "conversion from .int. to .nullptr_t." } */
g (0); /* { dg-error "incompatible type" } */
int i = 42 + nullptr; /* { dg-error "invalid operands" } */
/* The assignment of an object of type nullptr_t with a value of another
type, even if the value is a null pointer constant, is a constraint
violation. */
nullptr_t m;
m = 0; /* { dg-error "incompatible types" } */
(void) m;
nullptr_t o = 0; /* { dg-error "invalid initializer" } */
switch (nullptr); /* { dg-error "switch quantity not an integer" } */
}
/* If a second or third operand of type nullptr_t is used that is not a null
pointer constant and the other operand is not a pointer or does not have
itself nullptr_t, a constraint is violated even if that other operand is
a null pointer constant such as 0. */
void
test3 (_Bool b, int i)
{
const nullptr_t nptr = nullptr;
__auto_type a1 = b ? nptr : i; /* { dg-error "type mismatch" } */
__auto_type a2 = b ? i : nptr; /* { dg-error "type mismatch" } */
__auto_type a3 = b ? nptr : 0; /* { dg-error "type mismatch" } */
__auto_type a4 = b ? 0 : nptr; /* { dg-error "type mismatch" } */
__auto_type a5 = b ? 0 : nullptr; /* { dg-error "type mismatch" } */
__auto_type a6 = b ? nullptr : 0; /* { dg-error "type mismatch" } */
}