Received: by 2002:a25:1506:0:0:0:0:0 with SMTP id 6csp886000ybv; Fri, 7 Feb 2020 10:15:34 -0800 (PST) X-Google-Smtp-Source: APXvYqyTaqN1ZUej2Z6eTduqdhm9VZHZF/P1UfBshZaTeETRY4uIdBKxz9i1iYSMhlAdN89i39cu X-Received: by 2002:a9d:624e:: with SMTP id i14mr481444otk.371.1581099334297; Fri, 07 Feb 2020 10:15:34 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1581099334; cv=none; d=google.com; s=arc-20160816; b=TkyqMe3pSk6cqQ1wiQbMl8Yu8Qi9xRxahG5JvbAt8IHP/0CwOG4pJG4KCrSQFzB6oB 6+Ja7gyxWcOCRr+Nb3VlPZtwsPjeTqP1OMszzwRQnTAM8QawTWuKMgOjna9kIj6SO3J6 jCpa3u2LOdeQ6q0vz601LMFTYHegRDNOQaab3RrK66mraOKtagPUKgemKta7xwxwvcgI 3RjbL1Vg1M6wC9EUl3GO+10C28cGxyyDIFGiSJaYYXLFq8+RFGAMTUz2OJELRK5g8GRb zf0K/VCGmYT4fge+X/KbPLLUuwqeEenmf75ceaS2yAdFPAX7zR4I3nRShsrZAD3YkCau 7/yw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:references:in-reply-to:message-id:date :subject:cc:to:from; bh=naS3/hepLEymLMzZW1DVJjlznn2C2ajjfuREDjYtEDU=; b=O6UyXyG4BqDRq9VdO6PV1udcyONUwe7ShIKjkosVDcdbFor30L9ZE3x4VJNbRjGMsz tO/gGta8qcoiIVnXWoiDXcI8Kk1Nf+QcFOXRxCPSaUwtUIQ8RowhDGfulrhHOyRAQWx+ IIMnaHLkTFw4SKZUHwugxoLPBRlrFDKtYkvRb/mlAZPhc2iaAgZyeLLNcaetUKWBAe1I JZbIdEbYXOSkln3Jjid/VKMqqKRiPWqvj+JOTiGtbuKZfU0rm8RkISR2OAgxwHzh7tVt HxTJQr6ELmhPqHYeSGyYKYzeAAVuWaxVuHG6EXHqhDIC8QeAMMTXgx1kESpGeKuUyyY7 jhPg== 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 e15si26553otq.237.2020.02.07.10.15.22; Fri, 07 Feb 2020 10:15:34 -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 Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727387AbgBGSOM (ORCPT + 99 others); Fri, 7 Feb 2020 13:14:12 -0500 Received: from mx2.suse.de ([195.135.220.15]:39464 "EHLO mx2.suse.de" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727234AbgBGSOL (ORCPT ); Fri, 7 Feb 2020 13:14:11 -0500 X-Virus-Scanned: by amavisd-new at test-mx.suse.de Received: from relay2.suse.de (unknown [195.135.220.254]) by mx2.suse.de (Postfix) with ESMTP id 8D5A8ADB3; Fri, 7 Feb 2020 18:14:09 +0000 (UTC) From: Davidlohr Bueso To: akpm@linux-foundation.org Cc: dave@stgolabs.net, linux-kernel@vger.kernel.org, mcgrof@kernel.org, broonie@kernel.org, alex.williamson@redhat.com, Davidlohr Bueso Subject: [PATCH 3/5] regmap: optimize sync() and drop() regcache callbacks Date: Fri, 7 Feb 2020 10:03:03 -0800 Message-Id: <20200207180305.11092-4-dave@stgolabs.net> X-Mailer: git-send-email 2.16.4 In-Reply-To: <20200207180305.11092-1-dave@stgolabs.net> References: <20200207180305.11092-1-dave@stgolabs.net> Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org At a higher memory footprint (two additional pointers per node) we can get branchless O(1) tree iterators which can optimize in-order tree walks/traversals. For example, on my laptop looking at the rbtree debugfs entries: before: 60 nodes, 165 registers, average 2 registers, used 4036 bytes after: 60 nodes, 165 registers, average 2 registers, used 5004 bytes Cc: broonie@kernel.org Signed-off-by: Davidlohr Bueso --- drivers/base/regmap/regcache-rbtree.c | 62 +++++++++++++++++++---------------- 1 file changed, 34 insertions(+), 28 deletions(-) diff --git a/drivers/base/regmap/regcache-rbtree.c b/drivers/base/regmap/regcache-rbtree.c index cfa29dc89bbf..d55f6b6c87b4 100644 --- a/drivers/base/regmap/regcache-rbtree.c +++ b/drivers/base/regmap/regcache-rbtree.c @@ -8,7 +8,7 @@ #include #include -#include +#include #include #include @@ -28,11 +28,11 @@ struct regcache_rbtree_node { /* number of registers available in the block */ unsigned int blklen; /* the actual rbtree node holding this block */ - struct rb_node node; + struct llrb_node node; }; struct regcache_rbtree_ctx { - struct rb_root root; + struct llrb_root root; struct regcache_rbtree_node *cached_rbnode; }; @@ -75,9 +75,9 @@ static struct regcache_rbtree_node *regcache_rbtree_lookup(struct regmap *map, return rbnode; } - node = rbtree_ctx->root.rb_node; + node = rbtree_ctx->root.rb_root.rb_node; while (node) { - rbnode = rb_entry(node, struct regcache_rbtree_node, node); + rbnode = llrb_entry(node, struct regcache_rbtree_node, node); regcache_rbtree_get_base_top_reg(map, rbnode, &base_reg, &top_reg); if (reg >= base_reg && reg <= top_reg) { @@ -93,18 +93,20 @@ static struct regcache_rbtree_node *regcache_rbtree_lookup(struct regmap *map, return NULL; } -static int regcache_rbtree_insert(struct regmap *map, struct rb_root *root, +static int regcache_rbtree_insert(struct regmap *map, struct llrb_root *root, struct regcache_rbtree_node *rbnode) { struct rb_node **new, *parent; + struct llrb_node *prev = NULL; struct regcache_rbtree_node *rbnode_tmp; unsigned int base_reg_tmp, top_reg_tmp; unsigned int base_reg; parent = NULL; - new = &root->rb_node; + new = &root->rb_root.rb_node; while (*new) { - rbnode_tmp = rb_entry(*new, struct regcache_rbtree_node, node); + rbnode_tmp = llrb_entry(*new, + struct regcache_rbtree_node, node); /* base and top registers of the current rbnode */ regcache_rbtree_get_base_top_reg(map, rbnode_tmp, &base_reg_tmp, &top_reg_tmp); @@ -115,15 +117,16 @@ static int regcache_rbtree_insert(struct regmap *map, struct rb_root *root, if (base_reg >= base_reg_tmp && base_reg <= top_reg_tmp) return 0; - else if (base_reg > top_reg_tmp) + else if (base_reg > top_reg_tmp) { + prev = llrb_from_rb(parent); new = &((*new)->rb_right); - else if (base_reg < base_reg_tmp) + } else if (base_reg < base_reg_tmp) new = &((*new)->rb_left); } /* insert the node into the rbtree */ - rb_link_node(&rbnode->node, parent, new); - rb_insert_color(&rbnode->node, root); + rb_link_node(&rbnode->node.rb_node, parent, new); + llrb_insert(root, &rbnode->node, prev); return 1; } @@ -134,7 +137,7 @@ static int rbtree_show(struct seq_file *s, void *ignored) struct regmap *map = s->private; struct regcache_rbtree_ctx *rbtree_ctx = map->cache; struct regcache_rbtree_node *n; - struct rb_node *node; + struct llrb_node *node; unsigned int base, top; size_t mem_size; int nodes = 0; @@ -145,8 +148,8 @@ static int rbtree_show(struct seq_file *s, void *ignored) mem_size = sizeof(*rbtree_ctx); - for (node = rb_first(&rbtree_ctx->root); node != NULL; - node = rb_next(node)) { + for (node = llrb_first(&rbtree_ctx->root); node != NULL; + node = llrb_next(node)) { n = rb_entry(node, struct regcache_rbtree_node, node); mem_size += sizeof(*n); mem_size += (n->blklen * map->cache_word_size); @@ -192,7 +195,7 @@ static int regcache_rbtree_init(struct regmap *map) return -ENOMEM; rbtree_ctx = map->cache; - rbtree_ctx->root = RB_ROOT; + rbtree_ctx->root = LLRB_ROOT; rbtree_ctx->cached_rbnode = NULL; for (i = 0; i < map->num_reg_defaults; i++) { @@ -212,7 +215,7 @@ static int regcache_rbtree_init(struct regmap *map) static int regcache_rbtree_exit(struct regmap *map) { - struct rb_node *next; + struct llrb_node *next; struct regcache_rbtree_ctx *rbtree_ctx; struct regcache_rbtree_node *rbtree_node; @@ -222,11 +225,11 @@ static int regcache_rbtree_exit(struct regmap *map) return 0; /* free up the rbtree */ - next = rb_first(&rbtree_ctx->root); + next = llrb_first(&rbtree_ctx->root); while (next) { rbtree_node = rb_entry(next, struct regcache_rbtree_node, node); - next = rb_next(&rbtree_node->node); - rb_erase(&rbtree_node->node, &rbtree_ctx->root); + next = llrb_next(&rbtree_node->node); + llrb_erase(&rbtree_node->node, &rbtree_ctx->root); kfree(rbtree_node->cache_present); kfree(rbtree_node->block); kfree(rbtree_node); @@ -400,10 +403,10 @@ static int regcache_rbtree_write(struct regmap *map, unsigned int reg, max = reg + max_dist; /* look for an adjacent register to the one we are about to add */ - node = rbtree_ctx->root.rb_node; + node = rbtree_ctx->root.rb_root.rb_node; while (node) { - rbnode_tmp = rb_entry(node, struct regcache_rbtree_node, - node); + rbnode_tmp = llrb_entry(node, + struct regcache_rbtree_node, node); regcache_rbtree_get_base_top_reg(map, rbnode_tmp, &base_reg, &top_reg); @@ -466,15 +469,17 @@ static int regcache_rbtree_sync(struct regmap *map, unsigned int min, unsigned int max) { struct regcache_rbtree_ctx *rbtree_ctx; - struct rb_node *node; + struct llrb_node *node; struct regcache_rbtree_node *rbnode; unsigned int base_reg, top_reg; unsigned int start, end; int ret; rbtree_ctx = map->cache; - for (node = rb_first(&rbtree_ctx->root); node; node = rb_next(node)) { - rbnode = rb_entry(node, struct regcache_rbtree_node, node); + for (node = llrb_first(&rbtree_ctx->root); node; + node = llrb_next(node)) { + rbnode = rb_entry(node, + struct regcache_rbtree_node, node); regcache_rbtree_get_base_top_reg(map, rbnode, &base_reg, &top_reg); @@ -508,12 +513,13 @@ static int regcache_rbtree_drop(struct regmap *map, unsigned int min, { struct regcache_rbtree_ctx *rbtree_ctx; struct regcache_rbtree_node *rbnode; - struct rb_node *node; + struct llrb_node *node; unsigned int base_reg, top_reg; unsigned int start, end; rbtree_ctx = map->cache; - for (node = rb_first(&rbtree_ctx->root); node; node = rb_next(node)) { + for (node = llrb_first(&rbtree_ctx->root); node; + node = llrb_next(node)) { rbnode = rb_entry(node, struct regcache_rbtree_node, node); regcache_rbtree_get_base_top_reg(map, rbnode, &base_reg, -- 2.16.4