Received: by 2002:a25:ad19:0:0:0:0:0 with SMTP id y25csp245409ybi; Mon, 15 Jul 2019 20:09:24 -0700 (PDT) X-Google-Smtp-Source: APXvYqyv0Ww4X4ywEFt6TMGWGedPagQGdU9m+DXgH/Br62uGwDo7uAbfZo4i7ctbo/jfhwBMrwiS X-Received: by 2002:a17:90a:cb01:: with SMTP id z1mr32810491pjt.93.1563246564170; Mon, 15 Jul 2019 20:09:24 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1563246564; cv=none; d=google.com; s=arc-20160816; b=vcF+AqP9OGTNyNL7Vy9UPqNGBTzjqQyGzZrxkxKjHyDScwMGFRMIWwgtmNiXsRh4Pk 9sCdCNmRuipy2LlWwDKhXmJCYySJk+JyqfsniRifEOKecsD1R7914+2lZMN1eVjC1lcS BiErVaHRb9NQP+GoUqd7HZvv6ntq2e3GNYExTja3InuGKWk1TYOMNqR0zMjc1xAldQ1e eVwBryJpUCHtOesWwWgYLM897we2SETE2TMXDMUKWAHaNqCRJYWpRkc4gPwCP/k2gde/ 0+/8gBxE4k4sMZ8cXXY2sjqOMdlWmcpXYQCb6NXA7UmUnudVj5zMYNIbyi17gO5cMsek Gtvg== 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=JgvtdFbAq6bRLuc7Kw18aAw7V83x9QyXh8OujJ3hDqk=; b=elMbf185qOywWMgjIsQDi11XHSz4qJDr4qudNNI7ij6qE7YBnbxxDhqBNlSBtvB184 xc7AGnYQwQMlUMkQO/o/akgPfoft344lARHUrhraQVCyd3mJOCzeZBDmfRYrAGgmdIS4 zSPKhANHpWmpssIx5IyIlcO67paBWRTBpx7LBTy3qbY75jRrRoeI1UE8qKkUEDdBTYK/ gW+wALl1g2v6Nu9qBWXdheyQ/zd/9BNT7EIJ3NhIsyiMbYHjIkCUevQf7iKzu0Ea5xc+ KRhXd2HIa+P88pmsYWThmdjYwBQ8K9pVeBStS1zRtTQ5ZhG2vdatm1Y4vZhdfd8dp4P6 AgWg== 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 f186si17971439pgc.438.2019.07.15.20.09.07; Mon, 15 Jul 2019 20:09:24 -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 S1730833AbfGPDIM (ORCPT + 99 others); Mon, 15 Jul 2019 23:08:12 -0400 Received: from mx7.zte.com.cn ([202.103.147.169]:46754 "EHLO mxct.zte.com.cn" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1728256AbfGPDIL (ORCPT ); Mon, 15 Jul 2019 23:08:11 -0400 Received: from mse-fl2.zte.com.cn (unknown [10.30.14.239]) by Forcepoint Email with ESMTPS id 6A39F8A7F9E7858D8137; Tue, 16 Jul 2019 11:08:08 +0800 (CST) Received: from notes_smtp.zte.com.cn ([10.30.1.239]) by mse-fl2.zte.com.cn with ESMTP id x6G37mQK057592; Tue, 16 Jul 2019 11:07:48 +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 2019071611075612-2367600 ; Tue, 16 Jul 2019 11:07:56 +0800 From: Wen Yang To: linux-kernel@vger.kernel.org Cc: xue.zhihong@zte.com.cn, wang.yi59@zte.com.cn, cheng.shengyu@zte.com.cn, Wen Yang , Julia Lawall , Gilles Muller , Nicolas Palix , Michal Marek , Masahiro Yamada , Wen Yang , Markus Elfring , cocci@systeme.lip6.fr Subject: [PATCH v3] coccinelle: semantic code search for missing of_node_put Date: Tue, 16 Jul 2019 11:05:47 +0800 Message-Id: <1563246347-7803-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-07-16 11:07:56, Serialize by Router on notes_smtp/zte_ltd(Release 9.0.1FP7|August 17, 2016) at 2019-07-16 11:07:51, Serialize complete at 2019-07-16 11:07:51 X-MAIL: mse-fl2.zte.com.cn x6G37mQK057592 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 script: @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 Then copy the function names found by the above script to the r_miss_put rule. This rule checks for missing of_node_put. And this patch also looks for places where an of_node_put() call is on some paths but not on others (implemented by the r_miss_put_ext rule). Finally, this patch finds use-after-free issues for a node. (implemented by the r_use_after_put rule) 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: Wen Yang Cc: Markus Elfring Cc: cocci@systeme.lip6.fr --- v3: delete the global set, add a rule that checks for use-after-free. v2: improve the commit description and delete duplicate code. scripts/coccinelle/free/of_node_put.cocci | 192 ++++++++++++++++++++++++++++++ 1 file changed, 192 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..cda43fa --- /dev/null +++ b/scripts/coccinelle/free/of_node_put.cocci @@ -0,0 +1,192 @@ +// 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@ +@@ + +report_miss_prefix = "ERROR: missing of_node_put; acquired a node pointer with refcount incremented on line " +report_miss_suffix = ", but without a corresponding object release within this function." +org_miss_main = "acquired a node pointer with refcount incremented" +org_miss_sec = "needed of_node_put" +report_use_after_put = "ERROR: use-after-free; reference preceded by of_node_put on line " +org_use_after_put_main = "of_node_put" +org_use_after_put_sec = "reference" + +@r_miss_put exists@ +local idexpression struct device_node *x; +expression e, e1; +position p1, p2; +statement S; +type T, T1; +@@ + +* 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 != 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 (T1)x; +| + return of_fwnode_handle(x); +| +* return@p2 ...; +) + +@script:python depends on report && r_miss_put@ +p1 << r_miss_put.p1; +p2 << r_miss_put.p2; +@@ + +coccilib.report.print_report(p2[0], report_miss_prefix + p1[0].line + report_miss_suffix) + +@script:python depends on org && r_miss_put@ +p1 << r_miss_put.p1; +p2 << r_miss_put.p2; +@@ + +cocci.print_main(org_miss_main, p1) +cocci.print_secs(org_miss_sec, p2) + +@r_miss_put_ext exists@ +local idexpression struct device_node *x; +expression e, e1; +position p1 != r_miss_put.p1, p2 != r_miss_put.p2; +identifier f; +statement S; +type T, T1; +@@ + +( +* 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 (T1)x; +| + return of_fwnode_handle(x); +| +* return@p2 ...; +) +& +x = f(...) +... +if (<+...x...+>) S +... +of_node_put(x); +) + +@script:python depends on report && r_miss_put_ext@ +p1 << r_miss_put_ext.p1; +p2 << r_miss_put_ext.p2; +@@ + +coccilib.report.print_report(p2[0], report_miss_prefix + p1[0].line + report_miss_suffix) + +@script:python depends on org && r_miss_put_ext@ +p1 << r_miss_put_ext.p1; +p2 << r_miss_put_ext.p2; +@@ +cocci.print_main(org_miss_main, p1) +cocci.print_secs(org_miss_sec, p2) + +@r_put@ +expression E; +position p1; +@@ + +* of_node_put@p1(E) + +@r_use_after_put exists@ +expression r_put.E, subE<=r_put.E; +constant char [] c; +expression E1; +iterator iter; +identifier f; +statement S; +type T; +position r_put.p1, p2; +@@ + +* of_node_put@p1(E,...) +... +( + iter(...,subE,...) S +| + subE = (T)E1 +| + &(T)subE +| + f(...,c,...,(T)E,...) +| + E == (T)E1 +| + E != (T)E1 +| + E1 == (T)E +| + E1 != (T)E +| + !E +| + (T)E || ... +| +* (T)E@p2 +) + +@script:python depends on r_use_after_put && report@ +p1 << r_put.p1; +p2 << r_use_after_put.p2; +@@ + +coccilib.report.print_report(p2[0], report_use_after_put + p1[0].line) + +@script:python depends on r_use_after_put && org@ +p1 << r_put.p1; +p2 << r_use_after_put.p2; +@@ + +cocci.print_main(org_use_after_put_main, p1) +cocci.print_secs(org_use_after_put_sec, p2) -- 2.9.5