/* See e.g. https://en.cppreference.com/w/c/string/byte/strcat */ #include "analyzer-decls.h" char *strcat (char *dest, const char *src); #define NULL ((void *)0) char * test_passthrough (char *dest, const char *src) { return strcat (dest, src); } char * test_null_dest (const char *src) { return strcat (NULL, src); /* { dg-warning "use of NULL where non-null expected" } */ } char * test_null_src (char *dest) { return strcat (dest, NULL); /* { dg-warning "use of NULL where non-null expected" } */ } char * test_uninit_dest (const char *src) { char dest[10]; return strcat (dest, src); /* { dg-warning "use of uninitialized value 'dest\\\[0\\\]'" } */ } char * test_uninit_src (char *dest) { const char src[10]; return strcat (dest, src); /* { dg-warning "use of uninitialized value 'src\\\[0\\\]'" } */ } char * test_dest_not_terminated (char *src) { char dest[3] = "foo"; return strcat (dest, src); /* { dg-warning "stack-based buffer over-read" } */ /* { dg-message "while looking for null terminator for argument 1 \\('&dest'\\) of 'strcat'" "" { target *-*-* } .-1 } */ } char * test_src_not_terminated (char *dest) { const char src[3] = "foo"; return strcat (dest, src); /* { dg-warning "stack-based buffer over-read" } */ /* { dg-message "while looking for null terminator for argument 2 \\('&src'\\) of 'strcat'" "" { target *-*-* } .-1 } */ } char * __attribute__((noinline)) call_strcat (char *dest, const char *src) { return strcat (dest, src); } void test_concrete_valid_static_size (void) { char buf[16]; char *p1 = __builtin_strcpy (buf, "abc"); char *p2 = call_strcat (buf, "def"); __analyzer_eval (p1 == buf); /* { dg-warning "TRUE" } */ __analyzer_eval (p2 == buf); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[0] == 'a'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[1] == 'b'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[2] == 'c'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[3] == 'd'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[4] == 'e'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[5] == 'f'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[6] == '\0'); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf) == 6); /* { dg-warning "TRUE" } */ } void test_concrete_valid_static_size_2 (void) { char buf[16]; char *p1 = __builtin_strcpy (buf, "abc"); char *p2 = call_strcat (buf, "def"); char *p3 = call_strcat (buf, "ghi"); __analyzer_eval (p1 == buf); /* { dg-warning "TRUE" } */ __analyzer_eval (p2 == buf); /* { dg-warning "TRUE" } */ __analyzer_eval (p3 == buf); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[0] == 'a'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[1] == 'b'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[2] == 'c'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[3] == 'd'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[4] == 'e'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[5] == 'f'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[6] == 'g'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[7] == 'h'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[8] == 'i'); /* { dg-warning "TRUE" } */ __analyzer_eval (buf[9] == '\0'); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf) == 9); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf + 1) == 8); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf + 2) == 7); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf + 3) == 6); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf + 4) == 5); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf + 5) == 4); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf + 6) == 3); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf + 7) == 2); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf + 8) == 1); /* { dg-warning "TRUE" } */ __analyzer_eval (__builtin_strlen (buf + 9) == 0); /* { dg-warning "TRUE" } */ } char * __attribute__((noinline)) call_strcat_invalid (char *dest, const char *src) { return strcat (dest, src); /* { dg-warning "stack-based buffer overflow" } */ } void test_concrete_invalid_static_size (void) { char buf[3]; buf[0] = '\0'; call_strcat_invalid (buf, "abc"); } void test_concrete_symbolic (const char *suffix) { char buf[10]; buf[0] = '\0'; call_strcat (buf, suffix); } /* TODO: - "The behavior is undefined if the strings overlap." */