Received: by 2002:a25:f815:0:0:0:0:0 with SMTP id u21csp2790892ybd; Thu, 27 Jun 2019 20:02:47 -0700 (PDT) X-Google-Smtp-Source: APXvYqxY1UHAcCOBhbM7Yog61lNnBLzK5oXX6k3NSJWJoxmMk7u+jGCivu8ab+kGPk8q7136xkX6 X-Received: by 2002:a17:902:8546:: with SMTP id d6mr8472633plo.207.1561690967823; Thu, 27 Jun 2019 20:02:47 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1561690967; cv=none; d=google.com; s=arc-20160816; b=h+Wlk3brssOsliWetOHthaoPOyM4+Ns3YthiJCBaw8tcgRkWujGtYw9uaL5StbYy75 F3wwGPUmg6eX/vLyl5Q1cmMk7CDoV6DlZQA+XGX9wX7vC4AiIr+BZsMgOw3/t4e5FCk+ fZksC2oR5zf0QlzhoP5XIOtqCkd9KxpZIVxBMXwPNvPYzKhmzCaVTLNKt83O1lUTcm75 HIMCDPcsHuYR8x7FhiMr7aqRka912SbYITm47KgByiyJIt3JCpkzqruY+g+mBDmoqmVq EWWhlWv5D0MBpg7Mff6D/PQjg1M65kQS2AmgEumr/xgS4sCfiWB1he0CHQ5+9Dc2mUx2 9lYg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:message-id:date:subject:cc:to:from; bh=SIDMMBqJJqmqfnOoVd+09SYMCWyDJWozzL85lRLYvxs=; b=XESwFk3zmP1Mi9E5EyVBJpKsItxdrNzjWdqlsW1H358spHhRQ/UenKcxtyjjzHoyke Wx/L2k6jgzeJQ47Pz8i0VX6fgQXwz7tPMSwRseQDzGOzEfV53E6B3vWj6yEVJPZd4d+Q L8SmzG05BwbT4aWdvFALV7gvFMlzpxymaFUWMjpal46oEm00/A9xzFjBhXC5kAIEFPsQ y7eorvq6n9B2nsMuvsKP8IKYJfSGHtcttLNtE8zPp5D17od9x7Xj+V1niNE2Myb1+LTL 0WD9igK2hFt+PrQXzqMAuP1SzLtMFBdQ0YzZPx6Pzq+wcBIBHvQAyufvsjzQt7Y5+cxN NVSA== 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 Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id m5si792184plt.167.2019.06.27.20.02.31; Thu, 27 Jun 2019 20:02:47 -0700 (PDT) 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 Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726816AbfF1DCF (ORCPT + 99 others); Thu, 27 Jun 2019 23:02:05 -0400 Received: from mxhk.zte.com.cn ([63.217.80.70]:8216 "EHLO mxhk.zte.com.cn" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726542AbfF1DCE (ORCPT ); Thu, 27 Jun 2019 23:02:04 -0400 Received: from mse-fl2.zte.com.cn (unknown [10.30.14.239]) by Forcepoint Email with ESMTPS id DFF94594B33C4A312078; Fri, 28 Jun 2019 11:02:01 +0800 (CST) Received: from notes_smtp.zte.com.cn ([10.30.1.239]) by mse-fl2.zte.com.cn with ESMTP id x5S30C0k068092; Fri, 28 Jun 2019 11:00:12 +0800 (GMT-8) (envelope-from wen.yang99@zte.com.cn) Received: from fox-host8.localdomain ([10.74.120.8]) by szsmtp06.zte.com.cn (Lotus Domino Release 8.5.3FP6) with ESMTP id 2019062811004795-1800625 ; Fri, 28 Jun 2019 11:00:47 +0800 From: Wen Yang To: linux-kernel@vger.kernel.org Cc: wang.yi59@zte.com.cn, Wen Yang , Julia Lawall , Gilles Muller , Nicolas Palix , Michal Marek , Masahiro Yamada , Markus Elfring , cocci@systeme.lip6.fr Subject: [PATCH v2] coccinelle: semantic code search for missing of_node_put Date: Fri, 28 Jun 2019 10:58:52 +0800 Message-Id: <1561690732-20694-1-git-send-email-wen.yang99@zte.com.cn> X-Mailer: git-send-email 1.8.3.1 X-MIMETrack: Itemize by SMTP Server on SZSMTP06/server/zte_ltd(Release 8.5.3FP6|November 21, 2013) at 2019-06-28 11:00:48, Serialize by Router on notes_smtp/zte_ltd(Release 9.0.1FP7|August 17, 2016) at 2019-06-28 11:00:18, Serialize complete at 2019-06-28 11:00:18 X-MAIL: mse-fl2.zte.com.cn x5S30C0k068092 Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org There are functions which increment a reference counter for a device node. These functions belong to a programming interface for the management of information from device trees. The counter must be decremented after the last usage of a device node. We find these functions by using the following SmPL: @initialize:ocaml@ @@ let relevant_str = "use of_node_put() on it when done" let contains s1 s2 = let re = Str.regexp_string s2 in try ignore (Str.search_forward re s1 0); true with Not_found -> false let relevant_functions = ref [] let add_function f c = if not (List.mem f !relevant_functions) then begin let s = String.concat " " ( (List.map String.lowercase_ascii (List.filter (function x -> Str.string_match (Str.regexp "[a-zA-Z_\\(\\)][-a-zA-Z0-9_\\(\\)]*$") x 0) (Str.split (Str.regexp "[ .;\t\n]+") c)))) in if contains s relevant_str then Printf.printf "Found relevant function: %s\n" f; relevant_functions := f :: !relevant_functions; end @r@ identifier fn; comments c; type T = struct device_node *; @@ T@c fn(...) { ... } @script:ocaml@ f << r.fn; c << r.c; @@ let (cb,cm,ca) = List.hd c in let c = String.concat " " cb in add_function f c And this patch also looks for places where an of_node_put() call is on some paths but not on others. Suggested-by: Julia Lawall Signed-off-by: Wen Yang Cc: Julia Lawall Cc: Gilles Muller Cc: Nicolas Palix Cc: Michal Marek Cc: Masahiro Yamada Cc: Markus Elfring Cc: cocci@systeme.lip6.fr --- v2: improve the commit description and delete duplicate code. scripts/coccinelle/free/of_node_put.cocci | 143 ++++++++++++++++++++++++++++++ 1 file changed, 143 insertions(+) create mode 100644 scripts/coccinelle/free/of_node_put.cocci diff --git a/scripts/coccinelle/free/of_node_put.cocci b/scripts/coccinelle/free/of_node_put.cocci new file mode 100644 index 0000000..81a9f54 --- /dev/null +++ b/scripts/coccinelle/free/of_node_put.cocci @@ -0,0 +1,143 @@ +// SPDX-License-Identifier: GPL-2.0 +/// Find missing of_node_put +/// +// Confidence: Moderate +// Copyright: (C) 2018-2019 Wen Yang, ZTE. +// Comments: +// Options: --no-includes --include-headers + +virtual report +virtual org + +@initialize:python@ +@@ + +seen = set() + +def add_if_not_present (p1, p2): + if (p1, p2) not in seen: + seen.add((p1, p2)) + return True + return False + +def display_report(p1, p2): + if add_if_not_present(p1[0].line, p2[0].line): + coccilib.report.print_report(p2[0], + "ERROR: missing of_node_put; acquired a node pointer with refcount incremented on line " + + p1[0].line + + ", but without a corresponding object release within this function.") + +def display_org(p1, p2): + cocci.print_main("acquired a node pointer with refcount incremented", p1) + cocci.print_secs("needed of_node_put", p2) + +@r1 exists@ +local idexpression struct device_node *x; +expression e, e1; +position p1, p2; +statement S; +type T; +@@ + +x = @p1\(of_find_all_nodes\| + of_get_cpu_node\| + of_get_parent\| + of_get_next_parent\| + of_get_next_child\| + of_get_next_cpu_node\| + of_get_compatible_child\| + of_get_child_by_name\| + of_find_node_opts_by_path\| + of_find_node_by_name\| + of_find_node_by_type\| + of_find_compatible_node\| + of_find_node_with_property\| + of_find_matching_node_and_match\| + of_find_node_by_phandle\| + of_parse_phandle\| + of_find_next_cache_node\| + of_get_next_available_child\)(...); +... +if (x == NULL || ...) S +... when != e = (T)x + when != true x == NULL + when != of_node_put(x) + when != of_get_next_parent(x) + when != of_find_matching_node(x, ...) + when != if (x) { ... return x; } + when != v4l2_async_notifier_add_fwnode_subdev(<...x...>) + when != e1 = of_fwnode_handle(x) +( +if (x) { ... when forall + of_node_put(x) ... } +| +return x; +| +return of_fwnode_handle(x); +| +return@p2 ...; +) + +@script:python depends on report && r1@ +p1 << r1.p1; +p2 << r1.p2; +@@ + +display_report(p1, p2) + +@script:python depends on org && r1@ +p1 << r1.p1; +p2 << r1.p2; +@@ + +display_org(p1, p2) + +@r2 exists@ +local idexpression struct device_node *x; +expression e, e1; +position p1, p2; +identifier f; +statement S; +type T; +@@ + +( +x = f@p1(...); +... when != e = (T)x + when != true x == NULL + when != of_node_put(x) + when != of_get_next_parent(x) + when != of_find_matching_node(x, ...) + when != if (x) { ... return x; } + when != v4l2_async_notifier_add_fwnode_subdev(<...x...>) + when != e1 = of_fwnode_handle(x) +( +if (x) { ... when forall + of_node_put(x) ... } +| +return x; +| +return of_fwnode_handle(x); +| +return@p2 ...; +) +& +x = f(...) +... +if (<+...x...+>) S +... +of_node_put(x); +) +@script:python depends on report && r2@ +p1 << r2.p1; +p2 << r2.p2; +@@ + +display_report(p1, p2) + +@script:python depends on org && r2@ +p1 << r2.p1; +p2 << r2.p2; +@@ + +display_org(p1, p2) -- 2.9.5