llvm-project/clang/test/Analysis/unix-fns-o_creat.c
Alejandro Álvarez Ayllón 37c19f9a35
[analyzer] UnixAPIMisuseChecker Get O_CREAT from preprocessor (#81855)
Now calling `open` with the `O_CREAT` flag and no mode parameter will
raise an issue in any system that defines `O_CREAT`.

The value for this flag is obtained after the full source code has been
parsed, leveraging `checkASTDecl`.
Hence, any `#define` or `#undefine` of `O_CREAT` following an `open` may
alter the results. Nevertheless, since redefining reserved identifiers
is UB, this is probably ok.
2024-02-20 10:25:01 +01:00

40 lines
1.3 KiB
C

// RUN: %clang_analyze_cc1 -verify -analyzer-checker=core,unix.API -analyzer-output=text %s
// Verify that the UnixAPIChecker finds the missing mode value regardless
// of the particular values of these macros, particularly O_CREAT.
#define O_RDONLY 0x2000
#define O_WRONLY 0x8000
#define O_CREAT 0x0002
extern int open(const char *path, int flags, ...);
void missing_mode_1(const char *path) {
(void)open(path, O_CREAT); // expected-warning{{Call to 'open' requires a 3rd argument when the 'O_CREAT' flag is set}} \
expected-note{{Call to 'open' requires a 3rd argument when the 'O_CREAT' flag is set}}
}
extern int some_flag;
void missing_mode_2(const char *path) {
int mode = O_WRONLY;
if (some_flag) { // expected-note {{Assuming 'some_flag' is not equal to 0}} \
expected-note {{Taking true branch}}
mode |= O_CREAT;
}
(void)open(path, mode); // expected-warning{{Call to 'open' requires a 3rd argument when the 'O_CREAT' flag is set}} \
expected-note{{Call to 'open' requires a 3rd argument when the 'O_CREAT' flag is set}}
}
void no_creat(const char* path) {
int mode = O_RDONLY;
(void)open(path, mode); // ok
}
void mode_is_there(const char *path) {
int mode = O_WRONLY;
if (some_flag) {
mode |= O_CREAT;
}
(void)open(path, mode, 0770); // ok
}