From patchwork Thu Apr 27 18:54:25 2017 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Sandhya Bankar X-Patchwork-Id: 9705675 Return-Path: Received: from mail.wl.linuxfoundation.org (pdx-wl-mail.web.codeaurora.org [172.30.200.125]) by pdx-korg-patchwork.web.codeaurora.org (Postfix) with ESMTP id 2C3FA603F7 for ; Sat, 29 Apr 2017 07:35:49 +0000 (UTC) Received: from mail.wl.linuxfoundation.org (localhost [127.0.0.1]) by mail.wl.linuxfoundation.org (Postfix) with ESMTP id 1963828668 for ; Sat, 29 Apr 2017 07:35:49 +0000 (UTC) Received: by mail.wl.linuxfoundation.org (Postfix, from userid 486) id 0DDE12868E; Sat, 29 Apr 2017 07:35:49 +0000 (UTC) X-Spam-Checker-Version: SpamAssassin 3.3.1 (2010-03-16) on pdx-wl-mail.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-5.7 required=2.0 tests=BAYES_00, DATE_IN_PAST_24_48, DKIM_SIGNED,DKIM_VALID,DKIM_VALID_AU,FREEMAIL_FROM,RCVD_IN_DNSWL_HI autolearn=ham version=3.3.1 Received: from vger.kernel.org (vger.kernel.org [209.132.180.67]) by mail.wl.linuxfoundation.org (Postfix) with ESMTP id 5228928681 for ; Sat, 29 Apr 2017 07:35:48 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1035801AbdD2H26 (ORCPT ); Sat, 29 Apr 2017 03:28:58 -0400 Received: from mail-pf0-f193.google.com ([209.85.192.193]:33923 "EHLO mail-pf0-f193.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S936047AbdD2H24 (ORCPT ); Sat, 29 Apr 2017 03:28:56 -0400 Received: by mail-pf0-f193.google.com with SMTP id g23so20546374pfj.1; Sat, 29 Apr 2017 00:28:56 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=from:date:to:subject:message-id:references:mime-version :content-disposition:in-reply-to:user-agent; bh=Nzt26q6UkolKJU1XoJ93uQBK4gw76cCULGc3lD0wEEA=; b=PIB3qC2brAcA7k8IuJHxa2Oy+pX5d1y+zbWQymO7ygoXxioKamgJRlO+4MnItmOOXe yvF2nbRFjY23dHqy58txhQXgWaGw1xTRn7xzWGQV/i1SZRBlm3hrgfS5sl03gB3ZNAx8 x+I1nraYtoBYBlVtuRxcxEXo82gFr3WWh9yak6Wkzns0xS1irZQRQTqlSLrf8LoCmiFh wgyDC7wtD+YU2hYLBoAhwfv0p2zeSScw6tZKaV0BfjLQS5pTMh6IUibovmHL2uIXchs6 WOtesL7LTZjkFH5XTOoNdDgzkKtrk77JOt+gXSMmaXoC2jTGW+h2lkza1X973k7vVOz5 S1fw== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:date:to:subject:message-id:references :mime-version:content-disposition:in-reply-to:user-agent; bh=Nzt26q6UkolKJU1XoJ93uQBK4gw76cCULGc3lD0wEEA=; b=tEweWc7K6eMV7JbUyi4mALklcpRxxIXmqU++xllIfy8Rsfc9uTL1+Hy8PM0ObyHMUE BKFYQ76FF3EuzXi0K4YujsdQbNZuc1+wTxdCQQzwfrWVbb+wwnqyPCHShWNRBsgCY4PH Lub/kW3eOlbh3vhh0TkQWMQUvGGln8HNLO21bv/5VZff+StJkbu/7XZuogGpqIq0IGo8 wN2xjmY6YNM4FM/i1zeRhIGbHgQnIzTQ/7PmwDabaWETtYlrat+DmGCxVkw2JKEtdJW5 V6crpzA0V8MA/wY+bSSbR38oSdkW6/xvxSUp1Tv0WsDhr2Egf4q7x1tq3XQUBv5Id8kF Mk0w== X-Gm-Message-State: AN3rC/5SI+FxrszLHKSaUJafZ4HK68AuwRemZ1sbcS37QJoOqHVGKHS5 NAyivlnF2vKF1w== X-Received: by 10.84.210.10 with SMTP id z10mr20306353plh.173.1493450936077; Sat, 29 Apr 2017 00:28:56 -0700 (PDT) Received: from localhost.localdomain ([223.229.187.11]) by smtp.gmail.com with ESMTPSA id h7sm18047030pfc.99.2017.04.29.00.28.51 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Sat, 29 Apr 2017 00:28:55 -0700 (PDT) From: Matthew Wilcox X-Google-Original-From: Matthew Wilcox Date: Fri, 28 Apr 2017 00:24:25 +0530 To: linux-fsdevel@vger.kernel.org, linux-kernel@vger.kernel.org, viro@zeniv.linux.org.uk, mawmilcox@microsoft.com, keescook@chromium.org, adobriyan@gmail.com, re.emese@gmail.com, riel@surriel.com Subject: [PATCH 01/13] idr: Add ability to set/clear tags Message-ID: References: MIME-Version: 1.0 Content-Disposition: inline In-Reply-To: User-Agent: Mutt/1.5.21 (2010-09-15) Sender: linux-fsdevel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-fsdevel@vger.kernel.org X-Virus-Scanned: ClamAV using ClamSMTP Now that the IDR uses the radix tree, we can expose the radix tree tags to users of the IDR. A few spots in the radix tree needed to be changed to cope with the fact that the IDR can have NULL pointers with tags set. One of the more notable changes is that IDR_FREE really is special -- an index which is out of range of the current tree height is free. Signed-off-by: Matthew Wilcox --- include/linux/idr.h | 57 +++++++++++++++++++++++++++++++++++-- lib/radix-tree.c | 31 +++++++++----------- tools/testing/radix-tree/idr-test.c | 27 ++++++++++++++++++ 3 files changed, 95 insertions(+), 20 deletions(-) diff --git a/include/linux/idr.h b/include/linux/idr.h index bf70b3e..7eb4432 100644 --- a/include/linux/idr.h +++ b/include/linux/idr.h @@ -22,8 +22,8 @@ struct idr { }; /* - * The IDR API does not expose the tagging functionality of the radix tree - * to users. Use tag 0 to track whether a node has free space below it. + * Reserve one of the radix tree tags to track whether a node has free + * space below it. */ #define IDR_FREE 0 @@ -93,6 +93,59 @@ static inline void *idr_remove(struct idr *idr, int id) return radix_tree_delete_item(&idr->idr_rt, id, NULL); } +/** + * idr_tag_set - Set a tag on an entry + * @idr: IDR pointer + * @id: ID of entry to tag + * @tag: Tag index to set + * + * If there is an entry at @id in this IDR, set a tag on it and return + * the address of the entry. If @id is outside the range of the IDR, + * return NULL. This API does not allow you to set IDR_FREE on an entry; + * use idr_remove() for that. The implementation does not currently check + * that IDR_FREE is clear, so it is possible to set a tag on a free entry. + * This is not recommended and may change in the future. + */ +static inline void *idr_tag_set(struct idr *idr, int id, unsigned int tag) +{ + BUG_ON(tag == IDR_FREE); + return radix_tree_tag_set(&idr->idr_rt, id, tag); +} + +/** + * idr_tag_clear - Clear a tag on an entry + * @idr: IDR pointer + * @id: ID of entry to tag + * @tag: Tag index to clear + * + * If there is an entry at @id in this IDR, clear its tag and return + * the address of the entry. If @id is outside the range of the IDR, + * return NULL. This API does not allow you to clear IDR_FREE on an entry; + * use idr_alloc() for that. The implementation does not currently check + * that IDR_FREE is clear, so it is possible to clear a tag on a free entry. + * This is not recommended and may change in the future. + */ +static inline void *idr_tag_clear(struct idr *idr, int id, unsigned int tag) +{ + BUG_ON(tag == IDR_FREE); + return radix_tree_tag_clear(&idr->idr_rt, id, tag); +} + +/** + * idr_tag_get - Return whether a particular entry has a tag set + * @idr: IDR pointer + * @id: ID of entry to check + * @tag: Tag index to check + * + * Returns true/false depending whether @tag is set on this ID. Unlike + * idr_tag_set() or idr_tag_clear(), you can use the IDR_FREE tag value, + * as it can be useful to know whether a particular ID has been allocated. + */ +static inline bool idr_tag_get(const struct idr *idr, int id, unsigned int tag) +{ + return radix_tree_tag_get(&idr->idr_rt, id, tag); +} + static inline void idr_init(struct idr *idr) { INIT_RADIX_TREE(&idr->idr_rt, IDR_RT_MARKER); diff --git a/lib/radix-tree.c b/lib/radix-tree.c index 691a9ad..6723384 100644 --- a/lib/radix-tree.c +++ b/lib/radix-tree.c @@ -638,18 +638,17 @@ static int radix_tree_extend(struct radix_tree_root *root, gfp_t gfp, if (!node) return -ENOMEM; + /* Propagate the aggregated tag info to the new child */ + for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++) { + if (root_tag_get(root, tag)) + tag_set(node, tag, 0); + } if (is_idr(root)) { all_tag_set(node, IDR_FREE); if (!root_tag_get(root, IDR_FREE)) { tag_clear(node, IDR_FREE, 0); root_tag_set(root, IDR_FREE); } - } else { - /* Propagate the aggregated tag info to the new child */ - for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++) { - if (root_tag_get(root, tag)) - tag_set(node, tag, 0); - } } BUG_ON(shift > BITS_PER_LONG); @@ -1434,8 +1433,6 @@ void *radix_tree_tag_set(struct radix_tree_root *root, parent = entry_to_node(node); offset = radix_tree_descend(parent, &node, index); - BUG_ON(!node); - if (!tag_get(parent, tag, offset)) tag_set(parent, tag, offset); } @@ -1489,7 +1486,7 @@ static void node_tag_clear(struct radix_tree_root *root, * Clear the search tag (which must be < RADIX_TREE_MAX_TAGS) * corresponding to @index in the radix tree. If this causes * the leaf node to have no tags set then clear the tag in the - * next-to-leaf node, etc. + * parent node, etc. * * Returns the address of the tagged item on success, else NULL. ie: * has the same return value and semantics as radix_tree_lookup(). @@ -1512,8 +1509,7 @@ void *radix_tree_tag_clear(struct radix_tree_root *root, offset = radix_tree_descend(parent, &node, index); } - if (node) - node_tag_clear(root, parent, tag, offset); + node_tag_clear(root, parent, tag, offset); return node; } @@ -1552,11 +1548,11 @@ int radix_tree_tag_get(const struct radix_tree_root *root, struct radix_tree_node *node, *parent; unsigned long maxindex; - if (!root_tag_get(root, tag)) - return 0; - radix_tree_load_root(root, &node, &maxindex); if (index > maxindex) + return is_idr(root) && (tag == IDR_FREE); + + if (!root_tag_get(root, tag)) return 0; while (radix_tree_is_internal_node(node)) { @@ -1783,7 +1779,7 @@ void __rcu **radix_tree_next_chunk(const struct radix_tree_root *root, child = rcu_dereference_raw(node->slots[offset]); } - if (!child) + if (!child && !is_idr(root)) goto restart; if (child == RADIX_TREE_RETRY) break; @@ -1994,11 +1990,10 @@ static bool __radix_tree_delete(struct radix_tree_root *root, unsigned offset = get_slot_offset(node, slot); int tag; + for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++) + node_tag_clear(root, node, tag, offset); if (is_idr(root)) node_tag_set(root, node, IDR_FREE, offset); - else - for (tag = 0; tag < RADIX_TREE_MAX_TAGS; tag++) - node_tag_clear(root, node, tag, offset); replace_slot(slot, NULL, node, -1, exceptional); return node && delete_node(root, node, NULL, NULL); diff --git a/tools/testing/radix-tree/idr-test.c b/tools/testing/radix-tree/idr-test.c index 30cd0b2..fd94bee 100644 --- a/tools/testing/radix-tree/idr-test.c +++ b/tools/testing/radix-tree/idr-test.c @@ -135,6 +135,32 @@ void idr_null_test(void) assert(idr_is_empty(&idr)); } +#define IDR_TEST 1 + +void idr_tag_test(void) +{ + unsigned int i; + DEFINE_IDR(idr); + + for (i = 0; i < 100; i++) { + assert(idr_alloc(&idr, NULL, 0, 0, GFP_KERNEL) == i); + if (i % 7 == 0) + idr_tag_set(&idr, i, IDR_TEST); + } + + for (i = 0; i < 100; i += 14) { + assert(idr_tag_get(&idr, i, IDR_TEST)); + idr_tag_clear(&idr, i, IDR_TEST); + } + + for (i = 0; i < 100; i++) { + assert(idr_tag_get(&idr, i, IDR_TEST) == (i % 14 == 7)); + } + + idr_for_each(&idr, item_idr_free, &idr); + idr_destroy(&idr); +} + void idr_nowait_test(void) { unsigned int i; @@ -225,6 +251,7 @@ void idr_checks(void) idr_replace_test(); idr_alloc_test(); idr_null_test(); + idr_tag_test(); idr_nowait_test(); idr_get_next_test(); }