// RUN: %clang_analyze_cc1 -verify %s \ // RUN: -analyzer-checker=alpha.unix.cstring.BufferOverlap // // RUN: %clang_analyze_cc1 -verify %s -DUSE_BUILTINS \ // RUN: -analyzer-checker=alpha.unix.cstring.BufferOverlap // // RUN: %clang_analyze_cc1 -verify %s -DVARIANT \ // RUN: -analyzer-checker=alpha.unix.cstring.BufferOverlap // // RUN: %clang_analyze_cc1 -verify %s -DVARIANT -DUSE_BUILTINS \ // RUN: -analyzer-checker=alpha.unix.cstring.BufferOverlap // This provides us with four possible sprintf() definitions. #ifdef USE_BUILTINS #define BUILTIN(f) __builtin_##f #else /* USE_BUILTINS */ #define BUILTIN(f) f #endif /* USE_BUILTINS */ typedef typeof(sizeof(int)) size_t; #ifdef VARIANT #define __sprintf_chk BUILTIN(__sprintf_chk) #define __snprintf_chk BUILTIN(__snprintf_chk) int __sprintf_chk (char * __restrict str, int flag, size_t os, const char * __restrict fmt, ...); int __snprintf_chk (char * __restrict str, size_t len, int flag, size_t os, const char * __restrict fmt, ...); #define sprintf(str, ...) __sprintf_chk(str, 0, __builtin_object_size(str, 0), __VA_ARGS__) #define snprintf(str, len, ...) __snprintf_chk(str, len, 0, __builtin_object_size(str, 0), __VA_ARGS__) #else /* VARIANT */ #define sprintf BUILTIN(sprintf) int sprintf(char *restrict buffer, const char *restrict format, ... ); int snprintf(char *restrict buffer, size_t bufsz, const char *restrict format, ... ); #endif /* VARIANT */ void test_sprintf1() { char a[4] = {0}; sprintf(a, "%d/%s", 1, a); // expected-warning{{Arguments must not be overlapping buffers}} } void test_sprintf2() { char a[4] = {0}; sprintf(a, "%s", a); // expected-warning{{Arguments must not be overlapping buffers}} } void test_sprintf3() { char a[4] = {0}; sprintf(a, "%s/%s", a, a); // expected-warning{{Arguments must not be overlapping buffers}} } void test_sprintf4() { char a[4] = {0}; sprintf(a, "%d", 42); // no-warning } void test_sprintf5() { char a[4] = {0}; char b[4] = {0}; sprintf(a, "%s", b); // no-warning } void test_snprintf1() { char a[4] = {0}; snprintf(a, sizeof(a), "%d/%s", 1, a); // expected-warning{{Arguments must not be overlapping buffers}} } void test_snprintf2() { char a[4] = {0}; snprintf(a+1, sizeof(a)-1, "%d/%s", 1, a); // expected-warning{{Arguments must not be overlapping buffers}} } void test_snprintf3() { char a[4] = {0}; snprintf(a, sizeof(a), "%s", a); // expected-warning{{Arguments must not be overlapping buffers}} } void test_snprintf4() { char a[4] = {0}; snprintf(a, sizeof(a), "%s/%s", a, a); // expected-warning{{Arguments must not be overlapping buffers}} } void test_snprintf5() { char a[4] = {0}; snprintf(a, sizeof(a), "%d", 42); // no-warning } void test_snprintf6() { char a[4] = {0}; char b[4] = {0}; snprintf(a, sizeof(a), "%s", b); // no-warning }