In addition to assuming that built-in functions are called with the correct number of arguments (bug 83603), the restrict pass also assumes that they are called with arguments of the expected types. When a built-in is declared with no prototype and called with arguments of the wrong type (such as memcpy (3, dst, "")) the pass crashes GCC with a SEGV.
The attached patch prevents invalid calls from being checked for -Wrestrict violations. It's meant to be applied in conjunction with the patch for bug 83603: https://gcc.gnu.org/ml/gcc-patches/2018-01/msg00005.html A separate fix is to have GCC issue a warning for declarations of built-ins with no prototype. I opened bug 83656 to track this change request to the C front end (C++ already warns for this). Another improvement is to also warn about calls with the wrong number or types of arguments when a built-in is declared without a prototype despite the first warning, similarly to what Clang does. I opened bug 83657 to add this warning. Martin
PR tree-optimization/83655 - ICE on an invalid call to memcpy declared with no prototype gcc/testsuite/ChangeLog: PR tree-optimization/83655 * gcc.dg/Wrestrict-5.c: New test. * c-c++-common/builtins.c: New test. gcc/ChangeLog: PR tree-optimization/83655 * gimple-ssa-warn-restrict.c (wrestrict_dom_walker::check_call): diff --git a/gcc/gimple-ssa-warn-restrict.c b/gcc/gimple-ssa-warn-restrict.c index ac545e4..47d4900 100644 --- a/gcc/gimple-ssa-warn-restrict.c +++ b/gcc/gimple-ssa-warn-restrict.c @@ -1688,7 +1688,17 @@ wrestrict_dom_walker::check_call (gcall *call) if (!dstwr && strfun) dstwr = size_one_node; - if (check_bounds_or_overlap (call, dst, src, dstwr, NULL_TREE)) + /* DST and SRC can be null for a call with an insufficient number + of arguments to a built-in function declared without a protype. */ + if (!dst || !src) + return; + + /* DST, SRC, or DSTWR can also have the wrong type in a call to + a function declared without a prototype. Avoid checking such + invalid calls. */ + if (TREE_CODE (TREE_TYPE (dst)) != POINTER_TYPE + || TREE_CODE (TREE_TYPE (src)) != POINTER_TYPE + || (dstwr && !INTEGRAL_TYPE_P (TREE_TYPE (dstwr)))) return; /* Avoid diagnosing the call again. */ diff --git a/gcc/testsuite/gcc.dg/Wrestrict-5.c b/gcc/testsuite/gcc.dg/Wrestrict-5.c new file mode 100644 index 0000000..4912ee5 --- /dev/null +++ b/gcc/testsuite/gcc.dg/Wrestrict-5.c @@ -0,0 +1,43 @@ +/* Test to verify that valid calls to common restrict-qualified built-in + functions declared with no prototype are checked for overlap, and that + invalid calls are ignored. + { dg-do compile } + { dg-options "-O2 -Wrestrict" } */ + +typedef __SIZE_TYPE__ size_t; + +#if __cplusplus +extern "C" { + +#define NO_PROTO ... +#else +#define NO_PROTO /* empty */ +#endif + +void* memcpy (); +char* strncpy (); + +#if __cplusplus +} /* extern "C" */ +#endif + +void test_memcpy_warn (char *d) +{ + memcpy (d, d + 2, 3); /* { dg-warning "accessing 3 bytes at offsets 0 and 2 overlaps 1 byte at offset 2" } */ +} + +void test_memcpy_nowarn (char *d) +{ + memcpy (d, d + 2, ""); +} + + +void test_strncpy_warn (char *d) +{ + strncpy (d + 1, d + 3, 5); /* { dg-warning "accessing 5 bytes at offsets 1 and 3 overlaps 2 bytes at offset 3" } */ +} + +void test_strncpy_nowarn (char *d) +{ + strncpy (d + 1, d + 3, ""); +} diff --git a/gcc/testsuite/c-c++-common/builtins.c b/gcc/testsuite/c-c++-common/builtins.c new file mode 100644 index 0000000..673fcad --- /dev/null +++ b/gcc/testsuite/c-c++-common/builtins.c @@ -0,0 +1,193 @@ +/* Test to verify that calls to common built-in functions declared + with no prototype do not cause an ICE. + { dg-do compile } + { dg-options "-O2 -Wall -Wextra" } + { dg-prune-output "warning" } */ + +typedef __SIZE_TYPE__ size_t; + +#if __cplusplus +extern "C" { + +#define NO_PROTO ... +#else +#define NO_PROTO /* empty */ +#endif + + /* Character classification built-ins from <ctype.h>. */ + int isalpha (NO_PROTO); + int isalnum (NO_PROTO); + int isalpha (NO_PROTO); + int iscntrl (NO_PROTO); + int isdigit (NO_PROTO); + int isgraph (NO_PROTO); + int islower (NO_PROTO); + int isprint (NO_PROTO); + int ispunct (NO_PROTO); + int isspace (NO_PROTO); + int isupper (NO_PROTO); + int isxdigit (NO_PROTO); + int tolower (NO_PROTO); + int toupper (NO_PROTO); + + /* Memory allocation built-ins from <stdlib.h>. */ + void* alloca (NO_PROTO); + void* aligned_alloc (NO_PROTO); + void* calloc (NO_PROTO); + void* malloc (NO_PROTO); + void* realloc (NO_PROTO); + + /* Raw memory built-ins from <string.h>. */ + void* memcpy (NO_PROTO); + void* memchr (NO_PROTO); + void* memmove (NO_PROTO); + void* mempcpy (NO_PROTO); + void* memset (NO_PROTO); + + /* String built-ins from <string.h>. */ + char* stpcpy (NO_PROTO); + char* stpncpy (NO_PROTO); + + char* strcat (NO_PROTO); + char* strcpy (NO_PROTO); + + char* strdup (NO_PROTO); + char* strndup (NO_PROTO); + + char* strncat (NO_PROTO); + char* strncpy (NO_PROTO); + + size_t strlen (NO_PROTO); + size_t strnlen (NO_PROTO); + + char* strchr (NO_PROTO); + int strcmp (NO_PROTO); + int strncmp (NO_PROTO); + + /* Input/output functions from <stdio.h>. */ + int puts (NO_PROTO); + int fputs (NO_PROTO); + + int scanf (NO_PROTO); + int fscanf (NO_PROTO); + int sscanf (NO_PROTO); + int vfscanf (NO_PROTO); + int vsscanf (NO_PROTO); + + int printf (NO_PROTO); + int fprintf (NO_PROTO); + int sprintf (NO_PROTO); + + int snprintf (NO_PROTO); + + int vprintf (NO_PROTO); + int vfprintf (NO_PROTO); + int vsprintf (NO_PROTO); + + int vsnprintf (NO_PROTO); + +#if __cplusplus +} +#endif + + +#define CONCAT(a, b) a ## b +#define UNIQ_NAME(func, id) CONCAT (test_ ## func ## _, id) + +#define TEST_FUNC(func, arglist) \ + __typeof__ (func arglist) \ + UNIQ_NAME (func, __COUNTER__) (void) { \ + return func arglist; \ + } + +#define T1(func) \ + TEST_FUNC (func, ()); \ + TEST_FUNC (func, (1)); \ + TEST_FUNC (func, ("")); \ + TEST_FUNC (func, ((void*)1)); \ + TEST_FUNC (func, (iarr)); \ + TEST_FUNC (func, (function)) + +#define T2(func) \ + TEST_FUNC (func, (1, 1)); \ + TEST_FUNC (func, (1, "")); \ + TEST_FUNC (func, (1, (void*)1)); \ + TEST_FUNC (func, (1, iarr)); \ + TEST_FUNC (func, (1, function)) + +#define T3(func) \ + TEST_FUNC (func, (1, 1, 1)); \ + TEST_FUNC (func, (1, 1, "")); \ + TEST_FUNC (func, (1, 1, (void*)1)); \ + TEST_FUNC (func, (1, 1, iarr)); \ + TEST_FUNC (func, (1, 1, function)) + +extern int iarr[]; +extern void function (void); + +T1 (isalpha); +T1 (isalnum); +T1 (isalpha); +T1 (iscntrl); +T1 (isdigit); +T1 (isgraph); +T1 (islower); +T1 (isprint); +T1 (ispunct); +T1 (isspace); +T1 (isupper); +T1 (isxdigit); +T1 (tolower); +T1 (toupper); + +T1 (alloca); +T2 (aligned_alloc); +T2 (malloc); +T2 (calloc); +T2 (realloc); + +T3 (memcpy); +T3 (memmove); +T3 (mempcpy); +T3 (memset); +T3 (memchr); + +T2 (stpcpy); +T3 (stpncpy); + +T2 (strcat); +T2 (strcpy); + +T1 (strdup); +T2 (strndup); + +T3 (strncat); +T3 (strncpy); + +T2 (strchr); +T2 (strcmp); +T3 (strncmp); + +T1 (strlen); +T2 (strnlen); + +T1 (puts); +T2 (fputs); + +T1 (scanf); +T2 (fscanf); +T2 (sscanf); +T2 (vfscanf); +T2 (vsscanf); + +T2 (printf); +T3 (fprintf); +T3 (sprintf); + +T3 (snprintf); + +T2 (vprintf); +T2 (vfprintf); +T2 (vsprintf); + +T3 (vsnprintf);