Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp693575ybl; Fri, 10 Jan 2020 05:17:02 -0800 (PST) X-Google-Smtp-Source: APXvYqyNvnX337AOKdLpjmpnL4omNfaBXzBof6ZTDhvnG54sn4b49d1ySbvbxqKkhmVa0oCQa9Yn X-Received: by 2002:aca:5303:: with SMTP id h3mr1999261oib.109.1578662222077; Fri, 10 Jan 2020 05:17:02 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1578662222; cv=none; d=google.com; s=arc-20160816; b=s0D1UN57GfCYOUwGKPMlDuxrc8LdAP7JUDxoIScQMEIKjHsL53CyoGmIUjK6obC3yb 6rACNz/AhD6QNIBfOLKAPPfoJsQ9zJuV+kw+DA1dxhX+sCJaTCOh+vy1caGjpTB2roaQ fQF9fcUw1jehE95BIOkDffV/B6lp9iu3LyCCVQbqShbpqr4ksJvbDnNCsMZgKt25kkWK hZS33+wVGw68C7525ZMgej+mN8KWvKVW1/eadjePprKNrZhVymum+is2KRzaGDDPlGHF 3S6SX4G1Xs3R7VG7lwpDj3r0PCwWcFmORq8vH/eJxCiRuvAL/UZ28B7nPj8YCdvhcsf9 XNkA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :message-id:date:subject:cc:to:from; bh=uyyAjBpbOj2VFcjTvVgjoAe+DnWuUNVRaSKp1aY7qqE=; b=jhybqfa/jGRKoT5rLv3X6/zddja9v4eGq5O0Whxs4/PzQ1EZPLv8/W+7LJWMjWxypn 6kK9mEXkA7xguOtQlIIaW0u5OVq317PKsFXCAL9kZWATWohIp+vaWfCk9WCdVnyT+tI2 GePZZx1ZZbANXLdfj9wVfETi5IswhjHzCJ57c9ZnW127tvFbEeOpTPovfNTZs0pNzZ/4 5xL5Aagvm/lR27WD37U/peuzwgojD+Zz7ixKu3zhabvslRHCilDgAGXvLk6ROmqYgpnO ortomFPvReW7aLdi53BGnzKMLpHdzCGSmlIrHrvy1AqJiQ1JvvkP2hPEfyeH9RKXVRWE T1sg== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=alibaba.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id s68si1175497oih.275.2020.01.10.05.16.50; Fri, 10 Jan 2020 05:17:02 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=alibaba.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727610AbgAJNPl (ORCPT + 99 others); Fri, 10 Jan 2020 08:15:41 -0500 Received: from out30-42.freemail.mail.aliyun.com ([115.124.30.42]:55687 "EHLO out30-42.freemail.mail.aliyun.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727428AbgAJNPk (ORCPT ); Fri, 10 Jan 2020 08:15:40 -0500 X-Alimail-AntiSpam: AC=PASS;BC=-1|-1;BR=01201311R171e4;CH=green;DM=||false|;DS=||;FP=0|-1|-1|-1|0|-1|-1|-1;HT=e01e07486;MF=wenyang@linux.alibaba.com;NM=1;PH=DS;RN=12;SR=0;TI=SMTPD_---0TnKZRe1_1578662128; Received: from localhost(mailfrom:wenyang@linux.alibaba.com fp:SMTPD_---0TnKZRe1_1578662128) by smtp.aliyun-inc.com(127.0.0.1); Fri, 10 Jan 2020 21:15:37 +0800 From: Wen Yang To: Julia Lawall Cc: Wen Yang , Julia Lawall , Gilles Muller , Nicolas Palix , Michal Marek , Matthias Maennich , Greg Kroah-Hartman , Masahiro Yamada , Thomas Gleixner , cocci@systeme.lip6.fr, linux-kernel@vger.kernel.org Subject: [PATCH v3] coccinelle: semantic patch to check for inappropriate do_div() calls Date: Fri, 10 Jan 2020 21:15:26 +0800 Message-Id: <20200110131526.60180-1-wenyang@linux.alibaba.com> X-Mailer: git-send-email 2.23.0 MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org do_div() does a 64-by-32 division. When the divisor is unsigned long, u64, or s64, do_div() truncates it to 32 bits, this means it can test non-zero and be truncated to zero for division. This semantic patch is inspired by Mateusz Guzik's patch: commit b0ab99e7736a ("sched: Fix possible divide by zero in avg_atom() calculation") Signed-off-by: Wen Yang Cc: Julia Lawall Cc: Gilles Muller Cc: Nicolas Palix Cc: Michal Marek Cc: Matthias Maennich Cc: Greg Kroah-Hartman Cc: Masahiro Yamada Cc: Thomas Gleixner Cc: cocci@systeme.lip6.fr Cc: linux-kernel@vger.kernel.org --- v3: - also filter out safe consts for context mode. - cleanup code. v2: - add a special case for constants and checking whether the value is obviously safe and no warning is needed. - fix 'WARNING:' twice in each case. - extend the warning to say "consider using div64_xxx instead". scripts/coccinelle/misc/do_div.cocci | 155 +++++++++++++++++++++++++++ 1 file changed, 155 insertions(+) create mode 100644 scripts/coccinelle/misc/do_div.cocci diff --git a/scripts/coccinelle/misc/do_div.cocci b/scripts/coccinelle/misc/do_div.cocci new file mode 100644 index 000000000000..79db083c5208 --- /dev/null +++ b/scripts/coccinelle/misc/do_div.cocci @@ -0,0 +1,155 @@ +// SPDX-License-Identifier: GPL-2.0-only +/// do_div() does a 64-by-32 division. +/// When the divisor is long, unsigned long, u64, or s64, +/// do_div() truncates it to 32 bits, this means it can test +/// non-zero and be truncated to 0 for division on 64bit platforms. +/// +//# This makes an effort to find those inappropriate do_div() calls. +// +// Confidence: Moderate +// Copyright: (C) 2020 Wen Yang, Alibaba. +// Comments: +// Options: --no-includes --include-headers + +virtual context +virtual org +virtual report + +@initialize:python@ +@@ + +def get_digit_type_and_value(str): + is_digit = False + value = 0 + + try: + if (str.isdigit()): + is_digit = True + value = int(str, 0) + elif (str.upper().endswith('ULL')): + is_digit = True + value = int(str[:-3], 0) + elif (str.upper().endswith('LL')): + is_digit = True + value = int(str[:-2], 0) + elif (str.upper().endswith('UL')): + is_digit = True + value = int(str[:-2], 0) + elif (str.upper().endswith('L')): + is_digit = True + value = int(str[:-1], 0) + elif (str.upper().endswith('U')): + is_digit = True + value = int(str[:-1], 0) + except Exception as e: + print('Error:',e) + is_digit = False + value = 0 + finally: + return is_digit, value + +def filter_out_safe_constants(str): + is_digit, value = get_digit_type_and_value(str) + if (is_digit): + if (value >= 0x100000000): + return True + else: + return False + else: + return True + +def construct_warnings(suggested_fun): + msg="WARNING: do_div() does a 64-by-32 division, please consider using %s instead." + return msg % suggested_fun + +@depends on context@ +expression f; +long l: script:python() { filter_out_safe_constants(l) }; +unsigned long ul : script:python() { filter_out_safe_constants(ul) }; +u64 ul64 : script:python() { filter_out_safe_constants(ul64) }; +s64 sl64 : script:python() { filter_out_safe_constants(sl64) }; + +@@ +( +* do_div(f, l); +| +* do_div(f, ul); +| +* do_div(f, ul64); +| +* do_div(f, sl64); +) + +@r depends on (org || report)@ +expression f; +position p; +long l: script:python() { filter_out_safe_constants(l) }; +unsigned long ul : script:python() { filter_out_safe_constants(ul) }; +u64 ul64 : script:python() { filter_out_safe_constants(ul64) }; +s64 sl64 : script:python() { filter_out_safe_constants(sl64) }; +@@ +( +do_div@p(f, l); +| +do_div@p(f, ul); +| +do_div@p(f, ul64); +| +do_div@p(f, sl64); +) + +@script:python depends on org@ +p << r.p; +ul << r.ul; +@@ + +coccilib.org.print_todo(p[0], construct_warnings("div64_ul")) + +@script:python depends on org@ +p << r.p; +l << r.l; +@@ + +coccilib.org.print_todo(p[0], construct_warnings("div64_long")) + +@script:python depends on org@ +p << r.p; +ul64 << r.ul64; +@@ + +coccilib.org.print_todo(p[0], construct_warnings("div64_u64")) + +@script:python depends on org@ +p << r.p; +sl64 << r.sl64; +@@ + +coccilib.org.print_todo(p[0], construct_warnings("div64_s64")) + +@script:python depends on report@ +p << r.p; +ul << r.ul; +@@ + +coccilib.report.print_report(p[0], construct_warnings("div64_ul")) + +@script:python depends on report@ +p << r.p; +l << r.l; +@@ + +coccilib.report.print_report(p[0], construct_warnings("div64_long")) + +@script:python depends on report@ +p << r.p; +sl64 << r.sl64; +@@ + +coccilib.report.print_report(p[0], construct_warnings("div64_s64")) + +@script:python depends on report@ +p << r.p; +ul64 << r.ul64; +@@ + +coccilib.report.print_report(p[0], construct_warnings("div64_u64")) -- 2.23.0