
As reported in https://github.com/llvm/llvm-project/pull/103714#issuecomment-2295769193. CSA crashes on trying to bind value to symbolic region with `void *`. This happens when such region gets passed as inline asm input and engine tries to bind `UnknownVal` to that region. Fix it by changing type from void to char before calling `GetElementZeroRegion`
55 lines
1.7 KiB
C++
55 lines
1.7 KiB
C++
// RUN: %clang_analyze_cc1 -triple=x86_64-unknown-unknown \
|
|
// RUN: -analyzer-checker debug.ExprInspection,core -Wno-error=invalid-gnu-asm-cast -w %s -verify
|
|
|
|
int clang_analyzer_eval(int);
|
|
void clang_analyzer_dump(int);
|
|
void clang_analyzer_dump_ptr(void *);
|
|
|
|
int global;
|
|
void testRValueOutput() {
|
|
int &ref = global;
|
|
ref = 1;
|
|
__asm__("" : "=r"(((int)(global)))); // don't crash on rvalue output operand
|
|
clang_analyzer_eval(global == 1); // expected-warning{{UNKNOWN}}
|
|
clang_analyzer_eval(ref == 1); // expected-warning{{UNKNOWN}}
|
|
}
|
|
|
|
void *MyMemcpy(void *d, const void *s, const int n) {
|
|
asm volatile (
|
|
"cld\n rep movsb\n"
|
|
:: "S" (s), "D" (d), "c" (n) : "memory"
|
|
);
|
|
return d;
|
|
}
|
|
|
|
void testInlineAsmMemcpy(void)
|
|
{
|
|
int a, b = 10, c;
|
|
MyMemcpy(&a, &b, sizeof(b));
|
|
c = a; // no-warning
|
|
}
|
|
|
|
void testInlineAsmMemcpyArray(void)
|
|
{
|
|
int a[10], b[10] = {}, c;
|
|
MyMemcpy(&a, &b, sizeof(b));
|
|
c = a[8]; // no-warning
|
|
}
|
|
|
|
void testInlineAsmMemcpyUninit(void)
|
|
{
|
|
int a[10], b[10] = {}, c;
|
|
MyMemcpy(&a[1], &b[1], sizeof(b) - sizeof(b[1]));
|
|
c = a[0]; // expected-warning{{Assigned value is garbage or undefined}}
|
|
}
|
|
|
|
void testAsmWithVoidPtrArgument()
|
|
{
|
|
extern void *globalVoidPtr;
|
|
clang_analyzer_dump(*(int *)globalVoidPtr); // expected-warning-re {{reg_${{[0-9]+}}<int Element{SymRegion{reg_${{[0-9]+}}<void * globalVoidPtr>},0 S64b,int}>}}
|
|
clang_analyzer_dump_ptr(globalVoidPtr); // expected-warning-re {{&SymRegion{reg_${{[0-9]+}}<void * globalVoidPtr>}}}
|
|
asm ("" : : "a"(globalVoidPtr)); // no crash
|
|
clang_analyzer_dump(*(int *)globalVoidPtr); // expected-warning {{Unknown}}
|
|
clang_analyzer_dump_ptr(globalVoidPtr); // expected-warning-re {{&SymRegion{reg_${{[0-9]+}}<void * globalVoidPtr>}}}
|
|
}
|