| /* 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" } */ |
| } |