Received: by 10.223.176.5 with SMTP id f5csp1820854wra; Wed, 31 Jan 2018 12:07:03 -0800 (PST) X-Google-Smtp-Source: AH8x2264f8WOmygrIAiDcjX3Mt/mRBPtuCJxQWWntAJBfrYkEIK8iK+WLAFA151e+17kl91PoGq5 X-Received: by 10.101.100.208 with SMTP id t16mr20016162pgv.19.1517429223749; Wed, 31 Jan 2018 12:07:03 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1517429223; cv=none; d=google.com; s=arc-20160816; b=WQAP5WMtsGhf0CIcV1hSUGA4L964KUy3RC9jMkTLTA+zawtFywUBjvMNJfsMumHL1Z 9wYu0qv2hKmE9+O2hzTRVLztZYp+cgbtjBaO1SebS8M7wM9sUbDTiVHKseSID0U/Hf8Z pOEqejrLglHIMAgryyksDhPTzb6mGJRM6CJ6RoDji8Gms8vPDjT0uBjH5UwP49TeO5qD XLxGcYvQNjdodhLuMKj0aB33p3hgPPGfgBK85bkBgt7qW2rqGbn1Mjl4n6NDugd3NJyW jwUOicp/qWRGu5m7i4grCl0TDD8LjNXQ/xn268pUmoiLU1FSdsBZ5omK02yjA0ieyXb9 2DSg== 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 :dkim-signature:arc-authentication-results; bh=DK1FmItf/nZPyZz83i5zWBl5wHfLB38nLb+xX1q/LaU=; b=cwe4nsFwr8dPUGo31MSgrjm7l84iRWn/H7hRChwp+6DcSNvorr7V7uB13uGGdt8akp UWee9oE+qQrXJEzbUxz6zQTxpqIVyvYIdpyALsfKB1+DlsI8uIUjt6ENvpnyMa/tykkj /00qV0uXAzo+4W2C8vtvudbKruEqJOFq4BTa5qfVcqYVNIRVrVisk7c39NBrEmJ+j88x 4sz1vGi/XmSXfiqkNIS+/6/UOch69QbpQirQC6/2dTu68cvgsYP7OtnXsWoZZg/ms9YO OwJ+0u6aaOYtBcc3N42/T3he2hZtxB0uQS35TQ1EXLrFZkGaLOx/pcPymxfGb5BE8C7/ uM/g== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=uoL0TSVb; 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=pass (p=NONE sp=NONE dis=NONE) header.from=gmail.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id s192si2085517pgc.347.2018.01.31.12.06.48; Wed, 31 Jan 2018 12:07:03 -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; dkim=pass header.i=@gmail.com header.s=20161025 header.b=uoL0TSVb; 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=pass (p=NONE sp=NONE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1751476AbeAaUGY (ORCPT + 99 others); Wed, 31 Jan 2018 15:06:24 -0500 Received: from mail-pf0-f195.google.com ([209.85.192.195]:37825 "EHLO mail-pf0-f195.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1751326AbeAaUGX (ORCPT ); Wed, 31 Jan 2018 15:06:23 -0500 Received: by mail-pf0-f195.google.com with SMTP id p1so13785645pfh.4; Wed, 31 Jan 2018 12:06:22 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=from:to:cc:subject:date:message-id; bh=DK1FmItf/nZPyZz83i5zWBl5wHfLB38nLb+xX1q/LaU=; b=uoL0TSVbYNevfz02X8VXx3GBvWM0HeFASbSgaH4MDpC6a866Xoeli2VQEZVaVzcKx2 i72fu5aDJXBJumqRW0snJ6YLw3JiFH6NdycpMcbCoKG8GIXwUYA4V5n8lQkDDfUtlac/ xxLb3lr7iRKY/XeXIkpMJNjf0tWXlh9UZOvvvNYVHJUwzDMblGStCgFteqn/DWwm07E0 KYUt9sDMOAqLgocr187MmyQsUJfudbrOaYaqinG37KBIuxZWDyQOA5k+pS0P0KGoWXl9 9/aBzTT4kftJkapXBjgXH5tYDOZCs+ce6n9mHb3MHKlkPDZkHSmordTvRh1MGuq4RoQ5 OAuw== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id; bh=DK1FmItf/nZPyZz83i5zWBl5wHfLB38nLb+xX1q/LaU=; b=kNwscT6gduT9ZvKhPzjxnwZgoIu//ujqxt/4ylX763gWIXuYeQu9sfHnLLUmoqxqhc tzLtFtg1X/WTPoHvBkH5pxoM38fOepc/gQO9GrwfjPmStV/OVwbLYh0Ew3+Conn3hrOs kBfH6FjMm2t2LUHW1szH09g/aasLCbtkwEa44MxMTvkYioeoUBejdsyrfOt10Wt/zoYv IVbOEPZmA7A+pGgZZMcA/EKdGVTzP1QOG6Q4XGuYItaVZX+CpanHWylgJwPJGZKQpPPG mOrQOsxEY1c0kbMBJHIK4kwJmTFYUCBDcREJ8Yu/oU1Sst/YAuJuEmz9i7TxNn7Z0DJT fJoQ== X-Gm-Message-State: AKwxytfU2nEx2w4+Z+8WGYJDRKqGrzDRHNEZz9c7jbzoULybXzygdasW fIyxO1NAglkWDBnhYtmvBPU= X-Received: by 10.99.153.1 with SMTP id d1mr17355392pge.190.1517429182655; Wed, 31 Jan 2018 12:06:22 -0800 (PST) Received: from localhost.localdomain (c-73-93-215-6.hsd1.ca.comcast.net. [73.93.215.6]) by smtp.gmail.com with ESMTPSA id o1sm41705447pfa.101.2018.01.31.12.06.21 (version=TLS1_2 cipher=ECDHE-RSA-AES128-SHA bits=128/128); Wed, 31 Jan 2018 12:06:22 -0800 (PST) From: frowand.list@gmail.com To: Rob Herring , cpandya@codeaurora.org Cc: devicetree@vger.kernel.org, linux-kernel@vger.kernel.org Subject: [PATCH] of: cache phandle nodes to decrease cost of of_find_node_by_phandle() Date: Wed, 31 Jan 2018 12:05:42 -0800 Message-Id: <1517429142-25727-1-git-send-email-frowand.list@gmail.com> X-Mailer: git-send-email 1.9.1 Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Frank Rowand Create a cache of the nodes that contain a phandle property. Use this cache to find the node for a given phandle value instead of scanning the devicetree to find the node. If the phandle value is not found in the cache, of_find_node_by_phandle() will fall back to the tree scan algorithm. The cache is initialized in of_core_init(). The cache is freed via a late_initcall_sync(). Signed-off-by: Frank Rowand --- Some of_find_by_phandle() calls may occur before the cache is initialized or after it is freed. For example, for the qualcomm qcom-apq8074-dragonboard, 11 calls occur before the initialization and 80 occur after the cache is freed (out of 516 total calls.) drivers/of/base.c | 85 ++++++++++++++++++++++++++++++++++++++++++++++--- drivers/of/of_private.h | 5 +++ drivers/of/resolver.c | 21 ------------ 3 files changed, 86 insertions(+), 25 deletions(-) diff --git a/drivers/of/base.c b/drivers/of/base.c index 26618ba8f92a..c3091d0e391f 100644 --- a/drivers/of/base.c +++ b/drivers/of/base.c @@ -95,10 +95,14 @@ int __weak of_node_to_nid(struct device_node *np) } #endif +static void of_populate_phandle_cache(void); + void __init of_core_init(void) { struct device_node *np; + of_populate_phandle_cache(); + /* Create the kset, and register existing nodes */ mutex_lock(&of_mutex); of_kset = kset_create_and_add("devicetree", NULL, firmware_kobj); @@ -990,6 +994,72 @@ int of_modalias_node(struct device_node *node, char *modalias, int len) } EXPORT_SYMBOL_GPL(of_modalias_node); +phandle live_tree_max_phandle(void) +{ + struct device_node *node; + phandle max_phandle; + unsigned long flags; + + raw_spin_lock_irqsave(&devtree_lock, flags); + max_phandle = 0; + for_each_of_allnodes(node) { + if (node->phandle != OF_PHANDLE_ILLEGAL && + node->phandle > max_phandle) + max_phandle = node->phandle; + } + raw_spin_unlock_irqrestore(&devtree_lock, flags); + + return max_phandle; +} + +static struct device_node **phandle_cache; +static u32 max_phandle_cache; + +static int __init of_free_phandle_cache(void) +{ + max_phandle_cache = 0; + kfree(phandle_cache); + phandle_cache = NULL; + + return 0; +} +late_initcall_sync(of_free_phandle_cache); + +static void of_populate_phandle_cache(void) +{ + unsigned long flags; + phandle max_phandle; + u32 nodes = 0; + struct device_node *np; + + if (phandle_cache) + return; + + max_phandle = live_tree_max_phandle(); + + raw_spin_lock_irqsave(&devtree_lock, flags); + + for_each_of_allnodes(np) + nodes++; + + /* sanity cap for malformed tree */ + if (max_phandle > nodes) + max_phandle = nodes; + + phandle_cache = kzalloc((max_phandle + 1) * sizeof(*phandle_cache), + GFP_KERNEL); + + for_each_of_allnodes(np) + if (np->phandle != OF_PHANDLE_ILLEGAL && + np->phandle <= max_phandle && + np->phandle) + phandle_cache[np->phandle] = np; + + max_phandle_cache = max_phandle; + + raw_spin_unlock_irqrestore(&devtree_lock, flags); +} + /** * of_find_node_by_phandle - Find a node given a phandle * @handle: phandle of the node to find @@ -999,16 +1069,23 @@ int of_modalias_node(struct device_node *node, char *modalias, int len) */ struct device_node *of_find_node_by_phandle(phandle handle) { - struct device_node *np; + struct device_node *np = NULL; unsigned long flags; if (!handle) return NULL; raw_spin_lock_irqsave(&devtree_lock, flags); - for_each_of_allnodes(np) - if (np->phandle == handle) - break; + + if (handle <= max_phandle_cache) + np = phandle_cache[handle]; + + if (!np || np->phandle != handle) { + for_each_of_allnodes(np) + if (np->phandle == handle) + break; + } + of_node_get(np); raw_spin_unlock_irqrestore(&devtree_lock, flags); return np; diff --git a/drivers/of/of_private.h b/drivers/of/of_private.h index 92a9a3687446..77005978d60a 100644 --- a/drivers/of/of_private.h +++ b/drivers/of/of_private.h @@ -135,6 +135,11 @@ extern void __of_update_property_sysfs(struct device_node *np, extern void __of_sysfs_remove_bin_file(struct device_node *np, struct property *prop); +/* illegal phandle value (set when unresolved) */ +#define OF_PHANDLE_ILLEGAL 0xdeadbeef + +extern phandle live_tree_max_phandle(void); + /* iterators for transactions, used for overlays */ /* forward iterator */ #define for_each_transaction_entry(_oft, _te) \ diff --git a/drivers/of/resolver.c b/drivers/of/resolver.c index cfaeef5f6cb1..0384ce8fdc3b 100644 --- a/drivers/of/resolver.c +++ b/drivers/of/resolver.c @@ -22,27 +22,6 @@ #include "of_private.h" -/* illegal phandle value (set when unresolved) */ -#define OF_PHANDLE_ILLEGAL 0xdeadbeef - -static phandle live_tree_max_phandle(void) -{ - struct device_node *node; - phandle phandle; - unsigned long flags; - - raw_spin_lock_irqsave(&devtree_lock, flags); - phandle = 0; - for_each_of_allnodes(node) { - if (node->phandle != OF_PHANDLE_ILLEGAL && - node->phandle > phandle) - phandle = node->phandle; - } - raw_spin_unlock_irqrestore(&devtree_lock, flags); - - return phandle; -} - static void adjust_overlay_phandles(struct device_node *overlay, int phandle_delta) { -- Frank Rowand