blob: e3b698ae73d38836989696c6a0fe955e343ab9ca [file]
/* 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."
*/