From patchwork Mon Aug 13 21:29:28 2018 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Martin Sebor X-Patchwork-Id: 957286 Return-Path: X-Original-To: incoming@patchwork.ozlabs.org Delivered-To: patchwork-incoming@bilbo.ozlabs.org Authentication-Results: ozlabs.org; spf=pass (mailfrom) smtp.mailfrom=gcc.gnu.org (client-ip=209.132.180.131; helo=sourceware.org; envelope-from=gcc-patches-return-483600-incoming=patchwork.ozlabs.org@gcc.gnu.org; receiver=) Authentication-Results: ozlabs.org; dmarc=pass (p=none dis=none) header.from=gmail.com Authentication-Results: ozlabs.org; dkim=pass (1024-bit key; unprotected) header.d=gcc.gnu.org header.i=@gcc.gnu.org header.b="dyIfgvY9"; dkim=pass (2048-bit key; unprotected) header.d=gmail.com header.i=@gmail.com header.b="l83bCtkP"; dkim-atps=neutral Received: from sourceware.org (server1.sourceware.org [209.132.180.131]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by ozlabs.org (Postfix) with ESMTPS id 41q8565GJwz9rxx for ; Tue, 14 Aug 2018 07:29:46 +1000 (AEST) DomainKey-Signature: a=rsa-sha1; c=nofws; d=gcc.gnu.org; h=list-id :list-unsubscribe:list-archive:list-post:list-help:sender :subject:to:references:from:message-id:date:mime-version :in-reply-to:content-type; q=dns; s=default; b=iXTzWCL5N8CaLdYSQ a2v6GRAIA+CGHrrQysI2gV/hzTYSsRbmniiv6THC1MQsizdF6CnnYknTsOsgHtDE znvgeXejvdHzmuBB4bDsCRtJEsh5Zc+4/bxussPKIb53bacZ8Bax9YXoTWNTcglA AnzHcnzlhORHRYo9Z3B3dQG6yc= DKIM-Signature: v=1; a=rsa-sha1; c=relaxed; d=gcc.gnu.org; h=list-id :list-unsubscribe:list-archive:list-post:list-help:sender :subject:to:references:from:message-id:date:mime-version :in-reply-to:content-type; s=default; bh=MK3DWfy177E/0r1KD8SOAp7 2gHU=; b=dyIfgvY9LZ6TRMS4eWEMMIRyzCPJeipUw0fTuN9KAhbAVG5PmVwpT3e 1M7g6Ik2TTLcbQlRmQy1E2+DMyE29/19C6XMRSKm24wxY0RiGUUOC1slFnrUrTJ5 B2s1JhSahwkOwkj9cWvn/ap6Y7H701xj8DwmiOLfMxNQ/Htk5iA4= Received: (qmail 3447 invoked by alias); 13 Aug 2018 21:29:37 -0000 Mailing-List: contact gcc-patches-help@gcc.gnu.org; run by ezmlm Precedence: bulk List-Id: List-Unsubscribe: List-Archive: List-Post: List-Help: Sender: gcc-patches-owner@gcc.gnu.org Delivered-To: mailing list gcc-patches@gcc.gnu.org Received: (qmail 3350 invoked by uid 89); 13 Aug 2018 21:29:36 -0000 Authentication-Results: sourceware.org; auth=none X-Spam-SWARE-Status: No, score=-25.0 required=5.0 tests=AWL, BAYES_00, FREEMAIL_FROM, GIT_PATCH_0, GIT_PATCH_1, GIT_PATCH_2, GIT_PATCH_3, RCVD_IN_DNSWL_NONE, SPF_PASS autolearn=ham version=3.3.2 spammy=suggests X-HELO: mail-qt0-f179.google.com Received: from mail-qt0-f179.google.com (HELO mail-qt0-f179.google.com) (209.85.216.179) by sourceware.org (qpsmtpd/0.93/v0.84-503-g423c35a) with ESMTP; Mon, 13 Aug 2018 21:29:33 +0000 Received: by mail-qt0-f179.google.com with SMTP id y5-v6so19046843qti.12 for ; Mon, 13 Aug 2018 14:29:32 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=subject:to:references:from:message-id:date:user-agent:mime-version :in-reply-to; bh=C1VrI2dCxM27KW8YkPYe/SxW3dPyLy6Aw3gjC3Mq258=; b=l83bCtkPmubDp1qic+RxCcZQQwaphs0wPXS2d5On2st3X8A34ZYnC1u20VA+ybyWWB 3NNRMBcCAkIj4dJi3NH0ljrpVyPX6kO2aNUhrsCsZMrulkP1OeEC6E4W7GuTWL0alDbO KiKFkC9aim/JOgTePPuaj9gMwg38wctaMW6fzQfgwejNRiOXZm4HAMKHIoDIDFpNjHNY g9iVFEbrj8A6ot1HhIlB6oNJGJxuUFBvWyj4l/a6ht9L4c+se46De6ZKdDk+cuVI7vmL z1N47q6qNt83QFZUGVHWjyzWNSWGYoL7iym3awsnIDpwoEYlqMr5BEwneLkHbU08tLRI +Q0A== Received: from localhost.localdomain (97-118-124-30.hlrn.qwest.net. [97.118.124.30]) by smtp.gmail.com with ESMTPSA id 140-v6sm10998118qkm.18.2018.08.13.14.29.29 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Mon, 13 Aug 2018 14:29:30 -0700 (PDT) Subject: [PATCH 6/6] detect unterminated const arrays in strnlen calls (PR 86552) To: Gcc Patch List , Jeff Law References: From: Martin Sebor Message-ID: <90953771-af77-b1a6-2215-fc48a07cebd6@gmail.com> Date: Mon, 13 Aug 2018 15:29:28 -0600 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:45.0) Gecko/20100101 Thunderbird/45.8.0 MIME-Version: 1.0 In-Reply-To: X-IsSubscribed: yes The attached changes implement the detection of past-the-end reads by strncpy due to unterminated arguments and excessive bounds. PR tree-optimization/86552 - missing warning for reading past the end of non-string arrays gcc/ChangeLog: * builtins.c (expand_builtin_strnlen): Detect, avoid expanding, and diagnose unterminated arrays. gcc/testsuite/ChangeLog: * gcc.dg/warn-strnlen-no-nul.c: New. diff --git a/gcc/builtins.c b/gcc/builtins.c index 2f493d3..46df2ea 100644 --- a/gcc/builtins.c +++ b/gcc/builtins.c @@ -582,11 +582,16 @@ warn_string_no_nul (location_t loc, tree exp, tree fndecl, tree nonstr) /* If EXP refers to an unterminated constant character array return the declaration of the object of which the array is a member or - element. Otherwise return null. */ + element and if SIZE is not null, set *SIZE to the size of + the unterminated array and set *EXACT if the size is exact or + clear it otherwise. Otherwise return null. */ tree -unterminated_array (tree exp) +unterminated_array (tree exp, tree *size /* = NULL */, bool *exact /* = NULL */) { + /* Offset from the beginning of the array or null. */ + tree off = NULL_TREE; + if (TREE_CODE (exp) == SSA_NAME) { gimple *stmt = SSA_NAME_DEF_STMT (exp); @@ -595,18 +600,43 @@ unterminated_array (tree exp) tree rhs1 = gimple_assign_rhs1 (stmt); tree_code code = gimple_assign_rhs_code (stmt); - if (code == ADDR_EXPR - && TREE_CODE (TREE_OPERAND (rhs1, 0)) == ARRAY_REF) - rhs1 = rhs1; - else if (code != POINTER_PLUS_EXPR) + if ((code == ADDR_EXPR + && TREE_CODE (TREE_OPERAND (rhs1, 0)) == ARRAY_REF) + || code == POINTER_PLUS_EXPR) + { + /* Store the index or offset. */ + off = gimple_assign_rhs2 (stmt); + exp = rhs1; + } + else return NULL_TREE; - - exp = rhs1; } tree nonstr; - if (c_strlen (exp, 1, &nonstr) && nonstr) - return nonstr; + tree len = c_strlen (exp, 1, &nonstr); + if (len && nonstr) + { + if (size) + { + if (off) + { + if (TREE_CODE (off) == INTEGER_CST) + { + /* Subtract the offset from the size of the array. */ + *exact = true; + off = fold_convert (ssizetype, off); + len = fold_build2 (MINUS_EXPR, ssizetype, len, off); + } + else + *exact = false; + } + else + *exact = true; + + *size = len; + } + return nonstr; + } return NULL_TREE; } @@ -3068,7 +3098,8 @@ expand_builtin_strnlen (tree exp, rtx target, machine_mode target_mode) tree maxobjsize = max_object_size (); tree func = get_callee_fndecl (exp); - tree len = c_strlen (src, 0); + tree nonstr; + tree len = c_strlen (src, 0, &nonstr); if (TREE_CODE (bound) == INTEGER_CST) { @@ -3080,8 +3111,41 @@ expand_builtin_strnlen (tree exp, rtx target, machine_mode target_mode) exp, func, bound, maxobjsize)) TREE_NO_WARNING (exp) = true; + bool exact = true; if (!len || TREE_CODE (len) != INTEGER_CST) - return NULL_RTX; + { + /* Clear EXACT if LEN may be less than SRC suggests, + such as in + strnlen (&a[i], sizeof a) + where the value of i is unknown. Unless i's value is + zero, the call is unsafe because the bound is greater. */ + nonstr = unterminated_array (src, &len, &exact); + if (!nonstr) + return NULL_RTX; + } + + if (nonstr + && !TREE_NO_WARNING (exp) + && (tree_int_cst_lt (len, bound) + || !exact)) + { + location_t warnloc + = expansion_point_location_if_in_system_header (loc); + + if (warning_at (warnloc, OPT_Wstringop_overflow_, + exact + ? G_("%K%qD specified bound %E exceeds the size %E " + "of unterminated array") + : G_("%K%qD specified bound %E may exceed the size " + "of at most %E of unterminated array"), + exp, func, bound, len)) + { + inform (DECL_SOURCE_LOCATION (nonstr), + "referenced argument declared here"); + TREE_NO_WARNING (exp) = true; + return NULL_RTX; + } + } len = fold_convert_loc (loc, size_type_node, len); len = fold_build2_loc (loc, MIN_EXPR, size_type_node, len, bound); @@ -3107,6 +3171,18 @@ expand_builtin_strnlen (tree exp, rtx target, machine_mode target_mode) if (!len || TREE_CODE (len) != INTEGER_CST) return NULL_RTX; + if (!TREE_NO_WARNING (exp) + && wi::ltu_p (wi::to_wide (len), min) + && warning_at (loc, OPT_Wstringop_overflow_, + "%K%qD specified bound [%wu, %wu] " + "exceeds the size %E of unterminated array", + exp, func, min.to_uhwi (), max.to_uhwi (), len)) + { + inform (DECL_SOURCE_LOCATION (nonstr), + "referenced argument declared here"); + TREE_NO_WARNING (exp) = true; + } + if (wi::gtu_p (min, wi::to_wide (len))) return expand_expr (len, target, target_mode, EXPAND_NORMAL); diff --git a/gcc/builtins.h b/gcc/builtins.h index f722dd8..c55fa6b 100644 --- a/gcc/builtins.h +++ b/gcc/builtins.h @@ -103,7 +103,7 @@ extern bool target_char_cst_p (tree t, char *p); extern internal_fn associated_internal_fn (tree); extern internal_fn replacement_internal_fn (gcall *); -extern tree unterminated_array (tree); +extern tree unterminated_array (tree, tree * = NULL, bool * = NULL); extern void warn_string_no_nul (location_t, tree, tree, tree); extern tree max_object_size (); diff --git a/gcc/testsuite/gcc.dg/warn-strnlen-no-nul.c b/gcc/testsuite/gcc.dg/warn-strnlen-no-nul.c new file mode 100644 index 0000000..09a527e --- /dev/null +++ b/gcc/testsuite/gcc.dg/warn-strnlen-no-nul.c @@ -0,0 +1,356 @@ +/* PR tree-optimization/86552 - missing warning for reading past the end + of non-string arrays + { dg-do compile } + { dg-options "-O2 -Wall -ftrack-macro-expansion=0" } */ + +typedef __SIZE_TYPE__ size_t; +extern size_t strnlen (const char*, size_t); + +const char a[5] = "12345"; /* { dg-message "declared here" } */ +enum { asz = sizeof a }; + +int v0 = 0; +int v1 = 1; + +void sink (int, ...); + +#define CONCAT(a, b) a ## b +#define CAT(a, b) CONCAT(a, b) + +#define T(str, n) \ + __attribute__ ((noipa)) \ + void CAT (test_, __LINE__) (void) { \ + int i0 = 0, i1 = i0 + 1, i2 = i1 + 1, i3 = i2 + 1; \ + sink (strnlen (str, n), i0, i1, i2, i3); \ + } typedef void dummy_type + +T (a, asz); +T (a, asz - 1); +T (a, asz - 2); +T (a, asz - 5); +T (&a[0], asz); +T (&a[0] + 1, asz); /* { dg-warning "specified bound 5 exceeds the size 4 of unterminated array" } */ +T (&a[1], asz); /* { dg-warning "specified bound 5 exceeds the size 4 of unterminated array" } */ +T (&a[1], asz - 1); +T (&a[v0], asz); /* { dg-warning "specified bound 5 may exceed the size of at most 5 of unterminated array" } */ +T (&a[v0] + 1, asz); /* { dg-warning "specified bound 5 may exceed the size of at most 5 of unterminated array" } */ + +T (a, asz + 1); /* { dg-warning "specified bound 6 exceeds the size 5 " } */ +T (&a[0], asz + 1); /* { dg-warning "unterminated" } */ +T (&a[0] + 1, asz - 1); +T (&a[0] + 1, asz + 1); /* { dg-warning "unterminated" } */ +T (&a[1], asz + 1); /* { dg-warning "unterminated" } */ +T (&a[v0], asz + 1); /* { dg-warning "unterminated" } */ +T (&a[v0] + 1, asz + 1); /* { dg-warning "unterminated" } */ + + +const char b[][5] = { /* { dg-message "declared here" } */ + "12", "123", "1234", "54321" +}; +enum { bsz = sizeof b[0] }; + +T (b[0], bsz); +T (b[1], bsz); +T (b[2], bsz); +T (b[3], bsz); + +T (b[0], bsz - 1); +T (b[1], bsz - 1); +T (b[2], bsz - 1); +T (b[3], bsz - 1); + +T (b[0], bsz + 1); +T (b[1], bsz + 1); +T (b[2], bsz + 1); +T (b[3], bsz + 1); /* { dg-warning "unterminated" } */ + +T (b[i0], bsz); +T (b[i1], bsz); +T (b[i2], bsz); +T (b[i3], bsz); + +T (b[i0], bsz + 1); +T (b[i1], bsz + 1); +T (b[i2], bsz + 1); +T (b[i3], bsz + 1); /* { dg-warning "unterminated" } */ + +T (b[v0], bsz); +T (b[v0], bsz + 1); + +T (&b[i2][i1], bsz); +T (&b[i2][i1] + i1, bsz); +T (&b[i2][v0], bsz); +T (&b[i2][i1] + v0, bsz); + +T (&b[i2][i1], bsz + 1); +T (&b[i2][i1] + i1, bsz + 1); +T (&b[i2][v0], bsz + 1); +T (&b[i2][i1] + v0, bsz + 1); + +T (&b[2][1], bsz); +T (&b[2][1] + i1, bsz); +T (&b[2][i0], bsz); +T (&b[2][1] + i0, bsz); +T (&b[2][1] + v0, bsz); +T (&b[2][v0], bsz); + +T (&b[2][1], bsz + 1); +T (&b[2][1] + i1, bsz + 1); +T (&b[2][i0], bsz + 1); +T (&b[2][1] + i0, bsz + 1); +T (&b[2][1] + v0, bsz + 1); +T (&b[2][v0], bsz + 1); + +T (&b[3][1], bsz); /* { dg-warning "unterminated" } */ +T (&b[3][1], bsz - 1); +T (&b[3][1] + 1, bsz); /* { dg-warning "unterminated" } */ +T (&b[3][1] + 1, bsz - 1); /* { dg-warning "unterminated" } */ +T (&b[3][1] + 1, bsz - 2); +T (&b[3][1] + i1, bsz); /* { dg-warning "unterminated" } */ +T (&b[3][1] + i1, bsz - i1); /* { dg-warning "unterminated" } */ +T (&b[3][1] + i1, bsz - i2); +T (&b[3][v0], bsz); +T (&b[3][1] + v0, bsz); /* { dg-warning "specified bound 5 may exceed the size of at most 4 of unterminated array" } */ +T (&b[3][v0] + v1, bsz); /* { dg-warning "specified bound 5 may exceed the size of at most 4 of unterminated array" "pr?????" { xfail *-*-* } } */ + +T (&b[3][1], bsz + 1); /* { dg-warning "unterminated" } */ +T (&b[3][1] + 1, bsz + 1); /* { dg-warning "unterminated" } */ +T (&b[3][1] + i1, bsz + 1); /* { dg-warning "unterminated" } */ +T (&b[3][v0], bsz + 1); /* { dg-warning "unterminated" "pr86936" { xfail *-*-* } } */ +T (&b[3][1] + v0, bsz + 1); /* { dg-warning "unterminated" } */ +T (&b[3][v0] + v1, bsz + 1); /* { dg-warning "unterminated" "pr86936" { xfail *-*-* } } */ + +T (&b[i3][i1], bsz); /* { dg-warning "unterminated" } */ +T (&b[i3][i1] + 1, bsz); /* { dg-warning "unterminated" } */ +T (&b[i3][i1] + i1, bsz); /* { dg-warning "specified bound 5 exceeds the size 3 of unterminated array" } */ +T (&b[i3][v0], bsz); +T (&b[i3][i1] + v0, bsz); /* { dg-warning "specified bound 5 may exceed the size of at most 4 of unterminated array" } */ +T (&b[i3][v0] + v1, bsz); + +T (&b[i3][i1], bsz + 1); /* { dg-warning "unterminated" } */ +T (&b[i3][i1] + 1, bsz + 1); /* { dg-warning "unterminated" } */ +T (&b[i3][i1] + i1, bsz + 1); /* { dg-warning "unterminated" } */ +T (&b[i3][v0], bsz + 1); /* { dg-warning "unterminated" "pr86919" { xfail *-*-* } } */ +T (&b[i3][i1] + v0, bsz + 1); /* { dg-warning "unterminated" } */ +T (&b[i3][v0] + v1, bsz + 1); /* { dg-warning "unterminated" "pr86919" { xfail *-*-* } } */ + +T (v0 ? "" : b[0], bsz); +T (v0 ? "" : b[1], bsz); +T (v0 ? "" : b[2], bsz); +T (v0 ? "" : b[3], bsz); +T (v0 ? b[0] : "", bsz); +T (v0 ? b[1] : "", bsz); +T (v0 ? b[2] : "", bsz); +T (v0 ? b[3] : "", bsz); + +T (v0 ? "" : b[0], bsz + 1); +T (v0 ? "" : b[1], bsz + 1); +T (v0 ? "" : b[2], bsz + 1); +T (v0 ? "" : b[3], bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ +T (v0 ? b[0] : "", bsz + 1); +T (v0 ? b[1] : "", bsz + 1); +T (v0 ? b[2] : "", bsz + 1); +T (v0 ? b[3] : "", bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ + +T (v0 ? "" : b[i0], bsz); +T (v0 ? "" : b[i1], bsz); +T (v0 ? "" : b[i2], bsz); +T (v0 ? "" : b[i3], bsz); +T (v0 ? b[i0] : "", bsz); +T (v0 ? b[i1] : "", bsz); +T (v0 ? b[i2] : "", bsz); +T (v0 ? b[i3] : "", bsz); + +T (v0 ? "" : b[i0], bsz + 1); +T (v0 ? "" : b[i1], bsz + 1); +T (v0 ? "" : b[i2], bsz + 1); +T (v0 ? "" : b[i3], bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ +T (v0 ? b[i0] : "", bsz + 1); +T (v0 ? b[i1] : "", bsz + 1); +T (v0 ? b[i2] : "", bsz + 1); +T (v0 ? b[i3] : "", bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ + +T (v0 ? "1234" : b[3], bsz); +T (v0 ? "1234" : b[i3], bsz); +T (v0 ? b[3] : "1234", bsz); +T (v0 ? b[i3] : "1234", bsz); + +T (v0 ? a : b[3], bsz); +T (v0 ? b[0] : b[2], bsz); +T (v0 ? b[2] : b[3], bsz); +T (v0 ? b[3] : b[2], bsz); + +T (v0 ? "1234" : b[3], bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ +T (v0 ? "1234" : b[i3], bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ +T (v0 ? b[3] : "1234", bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ +T (v0 ? b[i3] : "1234", bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ + +T (v0 ? a : b[3], bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ +T (v0 ? b[0] : b[2], bsz + 1); +T (v0 ? b[2] : b[3], bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ +T (v0 ? b[3] : b[2], bsz + 1); /* { dg-warning "unterminated" "pr86937" { xfail *-*-* } } */ + +struct A { char a[5], b[5]; }; + +const struct A s = { "1234", "12345" }; + +T (s.a, asz); +T (&s.a[0], asz); +T (&s.a[0] + 1, asz); +T (&s.a[0] + v0, asz); +T (&s.a[1], asz); +T (&s.a[1] + 1, asz); +T (&s.a[1] + v0, asz); + +T (&s.a[i0], asz); +T (&s.a[i0] + i1, asz); +T (&s.a[i0] + v0, asz); +T (&s.a[i1], asz); +T (&s.a[i1] + i1, asz); +T (&s.a[i1] + v0, asz); + +T (s.a, asz + 1); +T (&s.a[0], asz + 1); +T (&s.a[0] + 1, asz + 1); +T (&s.a[0] + v0, asz + 1); +T (&s.a[1], asz + 1); +T (&s.a[1] + 1, asz + 1); +T (&s.a[1] + v0, asz + 1); + +T (&s.a[i0], asz + 1); +T (&s.a[i0] + i1, asz + 1); +T (&s.a[i0] + v0, asz + 1); +T (&s.a[i1], asz + 1); +T (&s.a[i1] + i1, asz + 1); +T (&s.a[i1] + v0, asz + 1); + +T (s.b, bsz); +T (&s.b[0], bsz); +T (&s.b[0] + 1, bsz); /* { dg-warning "unterminated" } */ +T (&s.b[0] + v0, bsz); /* { dg-warning "unterminated" } */ +T (&s.b[1], bsz); /* { dg-warning "unterminated" } */ +T (&s.b[1] + 1, bsz); /* { dg-warning "unterminated" } */ +T (&s.b[1] + v0, bsz); /* { dg-warning "unterminated" } */ + +T (&s.b[i0], bsz); +T (&s.b[i0] + i1, bsz); /* { dg-warning "unterminated" } */ +T (&s.b[i0] + v0, bsz); /* { dg-warning "unterminated" } */ +T (&s.b[i1], bsz); /* { dg-warning "unterminated" } */ +T (&s.b[i1] + i1, bsz); /* { dg-warning "unterminated" } */ +T (&s.b[i1] + v0, bsz); /* { dg-warning "unterminated" } */ + +T (s.b, bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[0], bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[0] + 1, bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[0] + v0, bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[1], bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[1] + 1, bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[1] + v0, bsz + 1); /* { dg-warning "unterminated" } */ + +T (&s.b[i0], bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[i0] + i1, bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[i0] + v0, bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[i1], bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[i1] + i1, bsz + 1); /* { dg-warning "unterminated" } */ +T (&s.b[i1] + v0, bsz + 1); /* { dg-warning "unterminated" } */ + +struct B { struct A a[2]; }; + +const struct B ba[] = { + { { { "123", "12345" }, { "12345", "123" } } }, + { { { "12345", "123" }, { "123", "12345" } } }, + { { { "1", "12" }, { "123", "1234" } } }, + { { { "123", "1234" }, { "12345", "12" } } } +}; + +T (ba[0].a[0].a, asz + 1); +T (&ba[0].a[0].a[0], asz + 1); +T (&ba[0].a[0].a[0] + 1, asz + 1); +T (&ba[0].a[0].a[0] + v0, asz + 1); +T (&ba[0].a[0].a[1], asz + 1); +T (&ba[0].a[0].a[1] + 1, asz + 1); +T (&ba[0].a[0].a[1] + v0, asz + 1); + +T (ba[0].a[0].b, bsz); +T (&ba[0].a[0].b[0], bsz); +T (&ba[0].a[0].b[0] + 1, bsz); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[0] + 1, bsz - 1); +T (&ba[0].a[0].b[0] + v0, bsz); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[1], bsz); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[1], bsz - 1); +T (&ba[0].a[0].b[1] + 1, bsz - 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[1] + 1, bsz - 2); +T (&ba[0].a[0].b[1] + 1, bsz); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[1] + v0, bsz); /* { dg-warning "unterminated" } */ + +T (ba[0].a[0].b, bsz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[0], bsz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[0] + 1, bsz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[0] + v0, bsz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[1], bsz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[1] + 1, bsz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[0].b[1] + v0, bsz + 1); /* { dg-warning "unterminated" } */ + +T (ba[0].a[1].a, asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[1].a[0], asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[1].a[0] + 1, asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[1].a[0] + v0, asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[1].a[1], asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[1].a[1] + 1, asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[0].a[1].a[1] + v0, asz + 1); /* { dg-warning "unterminated" } */ + +T (ba[0].a[1].b, bsz + 1); +T (&ba[0].a[1].b[0], bsz + 1); +T (&ba[0].a[1].b[0] + 1, bsz + 1); +T (&ba[0].a[1].b[0] + v0, bsz + 1); +T (&ba[0].a[1].b[1], bsz + 1); +T (&ba[0].a[1].b[1] + 1, bsz + 1); +T (&ba[0].a[1].b[1] + v0, bsz + 1); + +T (ba[1].a[0].a, asz); +T (&ba[1].a[0].a[0], asz); +T (&ba[1].a[0].a[0] + 1, asz); /* { dg-warning "unterminated" } */ +T (&ba[1].a[0].a[0] + v0, asz); /* { dg-warning "unterminated" } */ +T (&ba[1].a[0].a[1], asz); /* { dg-warning "unterminated" } */ +T (&ba[1].a[0].a[1] + 1, asz); /* { dg-warning "unterminated" } */ +T (&ba[1].a[0].a[1] + v0, asz); /* { dg-warning "unterminated" } */ + +T (ba[1].a[0].a, asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[1].a[0].a[0], asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[1].a[0].a[0] + 1, asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[1].a[0].a[0] + v0, asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[1].a[0].a[1], asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[1].a[0].a[1] + 1, asz + 1); /* { dg-warning "unterminated" } */ +T (&ba[1].a[0].a[1] + v0, asz + 1); /* { dg-warning "unterminated" } */ + +T (ba[1].a[0].b, bsz); +T (&ba[1].a[0].b[0], bsz); +T (&ba[1].a[0].b[0] + 1, bsz); +T (&ba[1].a[0].b[0] + v0, bsz); +T (&ba[1].a[0].b[1], bsz); +T (&ba[1].a[0].b[1] + 1, bsz); +T (&ba[1].a[0].b[1] + v0, bsz); + +T (ba[1].a[1].a, asz); +T (&ba[1].a[1].a[0], asz); +T (&ba[1].a[1].a[0] + 1, asz); +T (&ba[1].a[1].a[0] + v0, asz); +T (&ba[1].a[1].a[1], asz); +T (&ba[1].a[1].a[1] + 1, asz); +T (&ba[1].a[1].a[1] + v0, asz); + +T (ba[1].a[1].b, bsz); +T (&ba[1].a[1].b[0], bsz); +T (&ba[1].a[1].b[0] + 1, bsz); /* { dg-warning "unterminated" } */ +T (&ba[1].a[1].b[0] + 1, bsz - 1); +T (&ba[1].a[1].b[0] + v0, bsz); /* { dg-warning "unterminated" } */ +T (&ba[1].a[1].b[1], bsz); /* { dg-warning "unterminated" } */ +T (&ba[1].a[1].b[1], bsz - 1); +T (&ba[1].a[1].b[1] + 1, bsz); /* { dg-warning "unterminated" } */ +T (&ba[1].a[1].b[1] + 1, bsz - 1); /* { dg-warning "unterminated" } */ +T (&ba[1].a[1].b[1] + 1, bsz - 2); +T (&ba[1].a[1].b[1] + 1, bsz - i2); +T (&ba[1].a[1].b[1] + v0, bsz); /* { dg-warning "unterminated" } */ + +/* Prune out warnings with no location (pr?????). + { dg-prune-output "cc1:" } */ diff --git a/gcc/tree-ssa-strlen.c b/gcc/tree-ssa-strlen.c index 1d813b4..b747c35 100644 --- a/gcc/tree-ssa-strlen.c +++ b/gcc/tree-ssa-strlen.c @@ -336,8 +336,12 @@ get_stridx (tree exp) return idx; } - s = string_constant (exp, &o); + /* Set if EXP refers to a constant array that is not a nul-terminated + string, otherwise clear. */ + tree nonstr; + s = string_constant (exp, &o, &nonstr); if (s != NULL_TREE + && nonstr == NULL_TREE && (o == NULL_TREE || tree_fits_shwi_p (o)) && TREE_STRING_LENGTH (s) > 0) {