llvm-project/clang/test/Sema/tautological-pointer-comparison.c
Nikita Popov 1295aa2e81
[Clang] Add -fwrapv-pointer flag (#122486)
GCC supports three flags related to overflow behavior:
 * `-fwrapv`: Makes signed integer overflow well-defined.
 * `-fwrapv-pointer`: Makes pointer overflow well-defined.
* `-fno-strict-overflow`: Implies `-fwrapv -fwrapv-pointer`, making both
signed integer overflow and pointer overflow well-defined.

Clang currently only supports `-fno-strict-overflow` and `-fwrapv`, but
not `-fwrapv-pointer`.

This PR proposes to introduce `-fwrapv-pointer` and adjust the semantics
of `-fwrapv` to match GCC.

This allows signed integer overflow and pointer overflow to be
controlled independently, while `-fno-strict-overflow` still exists to
control both at the same time (and that option is consistent across GCC
and Clang).
2025-01-28 09:57:00 +01:00

78 lines
2.3 KiB
C

// RUN: %clang_cc1 -fsyntax-only -verify %s
// RUN: %clang_cc1 -fsyntax-only -fwrapv-pointer -verify=fwrapv %s
// fwrapv-no-diagnostics
int add_ptr_idx_ult_ptr(const char *ptr, unsigned index) {
return ptr + index < ptr; // expected-warning {{pointer comparison always evaluates to false}}
}
int add_idx_ptr_ult_ptr(const char *ptr, unsigned index) {
return index + ptr < ptr; // expected-warning {{pointer comparison always evaluates to false}}
}
int ptr_ugt_add_ptr_idx(const char *ptr, unsigned index) {
return ptr > ptr + index; // expected-warning {{pointer comparison always evaluates to false}}
}
int ptr_ugt_add_idx_ptr(const char *ptr, unsigned index) {
return ptr > index + ptr; // expected-warning {{pointer comparison always evaluates to false}}
}
int add_ptr_idx_uge_ptr(const char *ptr, unsigned index) {
return ptr + index >= ptr; // expected-warning {{pointer comparison always evaluates to true}}
}
int add_idx_ptr_uge_ptr(const char *ptr, unsigned index) {
return index + ptr >= ptr; // expected-warning {{pointer comparison always evaluates to true}}
}
int ptr_ule_add_ptr_idx(const char *ptr, unsigned index) {
return ptr <= ptr + index; // expected-warning {{pointer comparison always evaluates to true}}
}
int ptr_ule_add_idx_ptr(const char *ptr, unsigned index) {
return ptr <= index + ptr; // expected-warning {{pointer comparison always evaluates to true}}
}
int add_ptr_idx_ult_ptr_array(unsigned index) {
char ptr[10];
return ptr + index < ptr; // expected-warning {{pointer comparison always evaluates to false}}
}
// Negative tests with wrong predicate.
int add_ptr_idx_ule_ptr(const char *ptr, unsigned index) {
return ptr + index <= ptr;
}
int add_ptr_idx_ugt_ptr(const char *ptr, unsigned index) {
return ptr + index > ptr;
}
int ptr_uge_add_idx_ptr(const char *ptr, unsigned index) {
return ptr >= index + ptr;
}
int ptr_ult_add_idx_ptr(const char *ptr, unsigned index) {
return ptr < index + ptr;
}
// Negative test with signed index.
int add_ptr_idx_ult_ptr_signed(const char *ptr, int index) {
return ptr + index < ptr;
}
// Negative test with unrelated pointers.
int add_ptr_idx_ult_ptr2(const char *ptr, const char *ptr2, unsigned index) {
return ptr + index < ptr2;
}
// Negative test with non-pointer operands.
int add_ptr_idx_ult_ptr_not_pointer(unsigned ptr, unsigned index) {
return ptr + index < ptr;
}