forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			196 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			196 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
| // RUN: %clang_analyze_cc1 -analyzer-checker=core,unix.cstring,unix.Malloc,alpha.unix.cstring,debug.ExprInspection -analyzer-store=region -verify -analyzer-config eagerly-assume=false %s
 | |
| // RUN: %clang_analyze_cc1 -DUSE_BUILTINS -analyzer-checker=core,unix.cstring,unix.Malloc,alpha.unix.cstring,debug.ExprInspection -analyzer-store=region -verify -analyzer-config eagerly-assume=false %s
 | |
| // RUN: %clang_analyze_cc1 -DVARIANT -analyzer-checker=core,unix.cstring,alpha.unix.cstring,unix.Malloc,debug.ExprInspection -analyzer-store=region -verify -analyzer-config eagerly-assume=false %s
 | |
| // RUN: %clang_analyze_cc1 -DUSE_BUILTINS -DVARIANT -analyzer-checker=core,unix.cstring,alpha.unix.cstring,unix.Malloc,debug.ExprInspection -analyzer-store=region -verify -analyzer-config eagerly-assume=false %s
 | |
| // RUN: %clang_analyze_cc1 -DSUPPRESS_OUT_OF_BOUND -analyzer-checker=core,unix.cstring,unix.Malloc,alpha.unix.cstring.BufferOverlap,alpha.unix.cstring.NotNullTerminated,debug.ExprInspection -analyzer-store=region -verify -analyzer-config eagerly-assume=false %s
 | |
| 
 | |
| #include "Inputs/system-header-simulator-cxx.h"
 | |
| #include "Inputs/system-header-simulator-for-malloc.h"
 | |
| 
 | |
| // This provides us with four possible mempcpy() definitions.
 | |
| // See also comments in bstring.c.
 | |
| 
 | |
| #ifdef USE_BUILTINS
 | |
| #define BUILTIN(f) __builtin_##f
 | |
| #else /* USE_BUILTINS */
 | |
| #define BUILTIN(f) f
 | |
| #endif /* USE_BUILTINS */
 | |
| 
 | |
| #ifdef VARIANT
 | |
| 
 | |
| #define __mempcpy_chk BUILTIN(__mempcpy_chk)
 | |
| void *__mempcpy_chk(void *__restrict__ s1, const void *__restrict__ s2,
 | |
|                     size_t n, size_t destlen);
 | |
| 
 | |
| #define mempcpy(a,b,c) __mempcpy_chk(a,b,c,(size_t)-1)
 | |
| 
 | |
| #else /* VARIANT */
 | |
| 
 | |
| #define mempcpy BUILTIN(mempcpy)
 | |
| void *mempcpy(void *__restrict__ s1, const void *__restrict__ s2, size_t n);
 | |
| 
 | |
| #endif /* VARIANT */
 | |
| 
 | |
| void clang_analyzer_eval(int);
 | |
| 
 | |
| int *testStdCopyInvalidatesBuffer(std::vector<int> v) {
 | |
|   int n = v.size();
 | |
|   int *buf = (int *)malloc(n * sizeof(int));
 | |
| 
 | |
|   buf[0] = 66;
 | |
| 
 | |
|   // Call to copy should invalidate buf.
 | |
|   std::copy(v.begin(), v.end(), buf);
 | |
| 
 | |
|   int i = buf[0];
 | |
| 
 | |
|   clang_analyzer_eval(i == 66); // expected-warning {{UNKNOWN}}
 | |
| 
 | |
|   return buf;
 | |
| }
 | |
| 
 | |
| int *testStdCopyBackwardInvalidatesBuffer(std::vector<int> v) {
 | |
|   int n = v.size();
 | |
|   int *buf = (int *)malloc(n * sizeof(int));
 | |
|   
 | |
|   buf[0] = 66;
 | |
| 
 | |
|   // Call to copy_backward should invalidate buf.
 | |
|   std::copy_backward(v.begin(), v.end(), buf + n);
 | |
| 
 | |
|   int i = buf[0];
 | |
| 
 | |
|   clang_analyzer_eval(i == 66); // expected-warning {{UNKNOWN}}
 | |
| 
 | |
|   return buf;
 | |
| }
 | |
| 
 | |
| namespace pr34460 {
 | |
| short a;
 | |
| class b {
 | |
|   int c;
 | |
|   long g;
 | |
|   void d() {
 | |
|     int e = c;
 | |
|     f += e;
 | |
|     mempcpy(f, &a, g);
 | |
|   }
 | |
|   unsigned *f;
 | |
| };
 | |
| }
 | |
| 
 | |
| void *memset(void *dest, int ch, std::size_t count);
 | |
| namespace memset_non_pod {
 | |
| class Base {
 | |
| public:
 | |
|   int b_mem;
 | |
|   Base() : b_mem(1) {}
 | |
| };
 | |
| 
 | |
| class Derived : public Base {
 | |
| public:
 | |
|   int d_mem;
 | |
|   Derived() : d_mem(2) {}
 | |
| };
 | |
| 
 | |
| void memset1_inheritance() {
 | |
|   Derived d;
 | |
|   memset(&d, 0, sizeof(Derived));
 | |
|   clang_analyzer_eval(d.b_mem == 0); // expected-warning{{TRUE}}
 | |
|   clang_analyzer_eval(d.d_mem == 0); // expected-warning{{TRUE}}
 | |
| }
 | |
| 
 | |
| #ifdef SUPPRESS_OUT_OF_BOUND
 | |
| void memset2_inheritance_field() {
 | |
|   Derived d;
 | |
|   memset(&d.d_mem, 0, sizeof(Derived));
 | |
|   clang_analyzer_eval(d.b_mem == 0); // expected-warning{{UNKNOWN}}
 | |
|   clang_analyzer_eval(d.d_mem == 0); // expected-warning{{UNKNOWN}}
 | |
| }
 | |
| 
 | |
| void memset3_inheritance_field() {
 | |
|   Derived d;
 | |
|   memset(&d.b_mem, 0, sizeof(Derived));
 | |
|   clang_analyzer_eval(d.b_mem == 0); // expected-warning{{TRUE}}
 | |
|   clang_analyzer_eval(d.d_mem == 0); // expected-warning{{TRUE}}
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void memset4_array_nonpod_object() {
 | |
|   Derived array[10];
 | |
|   clang_analyzer_eval(array[1].b_mem == 1); // expected-warning{{UNKNOWN}}
 | |
|   clang_analyzer_eval(array[1].d_mem == 2); // expected-warning{{UNKNOWN}}
 | |
|   memset(&array[1], 0, sizeof(Derived));
 | |
|   clang_analyzer_eval(array[1].b_mem == 0); // expected-warning{{UNKNOWN}}
 | |
|   clang_analyzer_eval(array[1].d_mem == 0); // expected-warning{{UNKNOWN}}
 | |
| }
 | |
| 
 | |
| void memset5_array_nonpod_object() {
 | |
|   Derived array[10];
 | |
|   clang_analyzer_eval(array[1].b_mem == 1); // expected-warning{{UNKNOWN}}
 | |
|   clang_analyzer_eval(array[1].d_mem == 2); // expected-warning{{UNKNOWN}}
 | |
|   memset(array, 0, sizeof(array));
 | |
|   clang_analyzer_eval(array[1].b_mem == 0); // expected-warning{{TRUE}}
 | |
|   clang_analyzer_eval(array[1].d_mem == 0); // expected-warning{{TRUE}}
 | |
| }
 | |
| 
 | |
| void memset6_new_array_nonpod_object() {
 | |
|   Derived *array = new Derived[10];
 | |
|   clang_analyzer_eval(array[2].b_mem == 1); // expected-warning{{UNKNOWN}}
 | |
|   clang_analyzer_eval(array[2].d_mem == 2); // expected-warning{{UNKNOWN}}
 | |
|   memset(array, 0, 10 * sizeof(Derived));
 | |
|   clang_analyzer_eval(array[2].b_mem == 0); // expected-warning{{TRUE}}
 | |
|   clang_analyzer_eval(array[2].d_mem == 0); // expected-warning{{TRUE}}
 | |
|   delete[] array;
 | |
| }
 | |
| 
 | |
| void memset7_placement_new() {
 | |
|   Derived *d = new Derived();
 | |
|   clang_analyzer_eval(d->b_mem == 1); // expected-warning{{TRUE}}
 | |
|   clang_analyzer_eval(d->d_mem == 2); // expected-warning{{TRUE}}
 | |
| 
 | |
|   memset(d, 0, sizeof(Derived));
 | |
|   clang_analyzer_eval(d->b_mem == 0); // expected-warning{{TRUE}}
 | |
|   clang_analyzer_eval(d->d_mem == 0); // expected-warning{{TRUE}}
 | |
| 
 | |
|   Derived *d1 = new (d) Derived();
 | |
|   clang_analyzer_eval(d1->b_mem == 1); // expected-warning{{TRUE}}
 | |
|   clang_analyzer_eval(d1->d_mem == 2); // expected-warning{{TRUE}}
 | |
| 
 | |
|   memset(d1, 0, sizeof(Derived));
 | |
|   clang_analyzer_eval(d->b_mem == 0); // expected-warning{{TRUE}}
 | |
|   clang_analyzer_eval(d->d_mem == 0); // expected-warning{{TRUE}}
 | |
| }
 | |
| 
 | |
| class BaseVirtual {
 | |
| public:
 | |
|   int b_mem;
 | |
|   virtual int get() { return 1; }
 | |
| };
 | |
| 
 | |
| class DerivedVirtual : public BaseVirtual {
 | |
| public:
 | |
|   int d_mem;
 | |
| };
 | |
| 
 | |
| #ifdef SUPPRESS_OUT_OF_BOUND
 | |
| void memset8_virtual_inheritance_field() {
 | |
|   DerivedVirtual d;
 | |
|   memset(&d.b_mem, 0, sizeof(Derived));
 | |
|   clang_analyzer_eval(d.b_mem == 0); // expected-warning{{UNKNOWN}}
 | |
|   clang_analyzer_eval(d.d_mem == 0); // expected-warning{{UNKNOWN}}
 | |
| }
 | |
| #endif
 | |
| } // namespace memset_non_pod
 | |
| 
 | |
| #ifdef SUPPRESS_OUT_OF_BOUND
 | |
| void memset1_new_array() {
 | |
|   int *array = new int[10];
 | |
|   memset(array, 0, 10 * sizeof(int));
 | |
|   clang_analyzer_eval(array[2] == 0); // expected-warning{{TRUE}}
 | |
|   memset(array + 1, 'a', 10 * sizeof(9));
 | |
|   clang_analyzer_eval(array[2] == 0); // expected-warning{{UNKNOWN}}
 | |
|   delete[] array;
 | |
| }
 | |
| #endif
 |