By default, clang assumes that all trailing array objects could be a FAM. So, an array of undefined size, size 0, size 1, or even size 42 is considered as FAMs for optimizations at least. One needs to override the default behavior by supplying the `-fstrict-flex-arrays=<N>` flag, with `N > 0` value to reduce the set of FAM candidates. Value `3` is the most restrictive and `0` is the most permissive on this scale. 0: all trailing arrays are FAMs 1: only incomplete, zero and one-element arrays are FAMs 2: only incomplete, zero-element arrays are FAMs 3: only incomplete arrays are FAMs If the user is happy with consdering single-element arrays as FAMs, they just need to remove the `consider-single-element-arrays-as-flexible-array-members` from the command line. Otherwise, if they don't want to recognize such cases as FAMs, they should specify `-fstrict-flex-arrays` anyway, which will be picked up by CSA. Any use of the deprecated analyzer-config value will trigger a warning explaining what to use instead. The `-analyzer-config-help` is updated accordingly. Depends on D138657 Reviewed By: xazax.hun Differential Revision: https://reviews.llvm.org/D138659
135 lines
5.1 KiB
C
135 lines
5.1 KiB
C
// -fstrict-flex-arrays=2 means that only undefined or zero element arrays are considered as FAMs.
|
|
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c90 \
|
|
// RUN: -fstrict-flex-arrays=2
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c99 \
|
|
// RUN: -fstrict-flex-arrays=2
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c11 \
|
|
// RUN: -fstrict-flex-arrays=2
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c17 \
|
|
// RUN: -fstrict-flex-arrays=2
|
|
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c++98 -x c++ \
|
|
// RUN: -fstrict-flex-arrays=2
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c++03 -x c++ \
|
|
// RUN: -fstrict-flex-arrays=2
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c++11 -x c++ \
|
|
// RUN: -fstrict-flex-arrays=2
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c++14 -x c++ \
|
|
// RUN: -fstrict-flex-arrays=2
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c++17 -x c++ \
|
|
// RUN: -fstrict-flex-arrays=2
|
|
|
|
// By default, -fstrict-flex-arrays=0, which means that even single element arrays are considered as FAMs.
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c17 \
|
|
// RUN: -DSINGLE_ELEMENT_FAMS
|
|
// RUN: %clang_analyze_cc1 -triple x86_64-linux-gnu -analyzer-checker=core,unix,debug.ExprInspection %s -verify -std=c++17 -x c++ \
|
|
// RUN: -DSINGLE_ELEMENT_FAMS
|
|
|
|
typedef __typeof(sizeof(int)) size_t;
|
|
size_t clang_analyzer_getExtent(void *);
|
|
void clang_analyzer_dump(size_t);
|
|
|
|
void *alloca(size_t size);
|
|
void *malloc(size_t size);
|
|
void free(void *ptr);
|
|
|
|
void test_incomplete_array_fam(void) {
|
|
typedef struct FAM {
|
|
char c;
|
|
int data[];
|
|
} FAM;
|
|
|
|
FAM fam;
|
|
clang_analyzer_dump(clang_analyzer_getExtent(&fam));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(fam.data));
|
|
// expected-warning@-2 {{4 S64b}}
|
|
// expected-warning@-2 {{Unknown}}
|
|
|
|
FAM *p = (FAM *)alloca(sizeof(FAM));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(p));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(p->data));
|
|
// expected-warning@-2 {{4 U64b}}
|
|
// expected-warning@-2 {{Unknown}}
|
|
|
|
FAM *q = (FAM *)malloc(sizeof(FAM));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(q));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(q->data));
|
|
// expected-warning@-2 {{4 U64b}}
|
|
// expected-warning@-2 {{Unknown}}
|
|
free(q);
|
|
}
|
|
|
|
void test_zero_length_array_fam(void) {
|
|
typedef struct FAM {
|
|
char c;
|
|
int data[0];
|
|
} FAM;
|
|
|
|
FAM fam;
|
|
clang_analyzer_dump(clang_analyzer_getExtent(&fam));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(fam.data));
|
|
// expected-warning@-2 {{4 S64b}}
|
|
// expected-warning@-2 {{Unknown}}
|
|
|
|
FAM *p = (FAM *)alloca(sizeof(FAM));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(p));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(p->data));
|
|
// expected-warning@-2 {{4 U64b}}
|
|
// expected-warning@-2 {{Unknown}}
|
|
|
|
FAM *q = (FAM *)malloc(sizeof(FAM));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(q));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(q->data));
|
|
// expected-warning@-2 {{4 U64b}}
|
|
// expected-warning@-2 {{Unknown}}
|
|
free(q);
|
|
}
|
|
|
|
void test_single_element_array_possible_fam(void) {
|
|
typedef struct FAM {
|
|
char c;
|
|
int data[1];
|
|
} FAM;
|
|
|
|
#ifdef SINGLE_ELEMENT_FAMS
|
|
FAM likely_fam;
|
|
clang_analyzer_dump(clang_analyzer_getExtent(&likely_fam));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(likely_fam.data));
|
|
// expected-warning@-2 {{8 S64b}}
|
|
// expected-warning@-2 {{Unknown}}
|
|
|
|
FAM *p = (FAM *)alloca(sizeof(FAM));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(p));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(p->data));
|
|
// expected-warning@-2 {{8 U64b}}
|
|
// expected-warning@-2 {{Unknown}}
|
|
|
|
FAM *q = (FAM *)malloc(sizeof(FAM));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(q));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(q->data));
|
|
// expected-warning@-2 {{8 U64b}}
|
|
// expected-warning@-2 {{Unknown}}
|
|
free(q);
|
|
#else
|
|
FAM likely_fam;
|
|
clang_analyzer_dump(clang_analyzer_getExtent(&likely_fam));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(likely_fam.data));
|
|
// expected-warning@-2 {{8 S64b}}
|
|
// expected-warning@-2 {{4 S64b}}
|
|
|
|
FAM *p = (FAM *)alloca(sizeof(FAM));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(p));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(p->data));
|
|
// expected-warning@-2 {{8 U64b}}
|
|
// expected-warning@-2 {{4 S64b}}
|
|
|
|
FAM *q = (FAM *)malloc(sizeof(FAM));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(q));
|
|
clang_analyzer_dump(clang_analyzer_getExtent(q->data));
|
|
// expected-warning@-2 {{8 U64b}}
|
|
// expected-warning@-2 {{4 S64b}}
|
|
free(q);
|
|
#endif
|
|
}
|