Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1754450Ab0AECPe (ORCPT ); Mon, 4 Jan 2010 21:15:34 -0500 Received: (majordomo@vger.kernel.org) by vger.kernel.org id S1754357Ab0AECPc (ORCPT ); Mon, 4 Jan 2010 21:15:32 -0500 Received: from one.firstfloor.org ([213.235.205.2]:43058 "EHLO one.firstfloor.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1753909Ab0AECP3 (ORCPT ); Mon, 4 Jan 2010 21:15:29 -0500 From: Andi Kleen References: <20100105315.789846878@firstfloor.org> In-Reply-To: <20100105315.789846878@firstfloor.org> To: paulmck@linux.vnet.ibm.com, ebiederm@xmission.com, paulmck@linux.vnet.ibm.com, akpm@linux-foundation.org, linux-kernel@vger.kernel.org Subject: [PATCH] [1/9] Add rcustring ADT for RCU protected strings v2 Message-Id: <20100105021526.17EF2B17C2@basil.firstfloor.org> Date: Tue, 5 Jan 2010 03:15:26 +0100 (CET) Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 6189 Lines: 195 Add a little ADT for RCU protected strings. RCU is a convenient way to manage modifications to read-often-write-seldom strings. Add some helper functions to make this more straight forward. Used by follow-on patches to implement RCU protected sysctl strings. * General rules: * Reader has to use rcu_read_lock() and not sleep while accessing the string, * or alternatively get a copy with access_rcu_string() * Writer needs an own lock against each other. * Each modification should allocate a new string first and free the old * one with free_rcu_string() * In writers use rcu_assign_pointer to publicize the updated string to * global readers. * The size passed to access_rcu_string() must be the same as passed * to alloc_rcu_string() and be known in advance. Don't use strlen()! * * For sysctls also see proc_rcu_string() as a convenient wrapper v2: Use rcu_dereference correctly Cc: paulmck@linux.vnet.ibm.com Signed-off-by: Andi Kleen --- include/linux/rcustring.h | 20 ++++++++ lib/Makefile | 3 - lib/rcustring.c | 115 ++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 137 insertions(+), 1 deletion(-) Index: linux-2.6.33-rc2-ak/include/linux/rcustring.h =================================================================== --- /dev/null +++ linux-2.6.33-rc2-ak/include/linux/rcustring.h @@ -0,0 +1,20 @@ +#ifndef _RCUSTRING_H +#define _RCUSTRING_H 1 + +#include +#include + +/* + * Simple wrapper to manage strings by RCU. + */ + +extern char *alloc_rcu_string(int size, gfp_t gfp); +extern void free_rcu_string(const char *string); + +/* + * size must be the same as alloc_rcu_string, don't + * use strlen on str! + */ +extern char *access_rcu_string(char **str, int size, gfp_t gfp); + +#endif Index: linux-2.6.33-rc2-ak/lib/rcustring.c =================================================================== --- /dev/null +++ linux-2.6.33-rc2-ak/lib/rcustring.c @@ -0,0 +1,115 @@ +/* + * Manage strings by Read-Copy-Update. This is useful for global strings + * that change only very rarely, but are read often. + * + * Author: Andi Kleen + * + * General rules: + * Reader has to use rcu_read_lock() and not sleep while accessing the string, + * or alternatively get a copy with access_rcu_string() + * Writer needs an own lock against each other. + * Each modification should allocate a new string first and free the old + * one with free_rcu_string() + * In writers use rcu_assign_pointer to publicize the updated string to + * global readers. + * The size passed to access_rcu_string() must be the same as passed + * to alloc_rcu_string() and be known in advance. Don't use strlen()! + * The pointer passed to access_rcu_string must be to the variable modified + * by the writer. + * + * For sysctls also see proc_rcu_string() as a convenient wrapper + * + * Typical example: + * #define MAX_GLOBAL_SIZE ... + * char *global = "default"; + * + * Rare writer: + * char *old, *new; + * DECLARE_MUTEX(writer_lock); + * mutex_lock(&writer_lock); + * new = alloc_rcu_string(MAX_GLOBAL_SIZE, GFP_KERNEL); + * if (!new) { + * mutex_unlock(&writer_lock); + * return -ENOMEM; + * } + * strlcpy(new, new_value, MAX_GLOBAL_SIZE); + * old = global; + * rcu_assign_pointer(global, new); + * mutex_unlock(&writer_lock); + * free_rcu_string(old); + * + * Sleepy reader: + * char *str = access_rcu_string(&global, MAX_GLOBAL_SIZE, GFP_KERNEL); + * if (!str) + * return -ENOMEM; + * ... use str while sleeping ... + * kfree(string); + * + * Non sleepy reader: + * rcu_read_lock(); + * str = rcu_dereference(&global); + * ... use str without sleeping ... + * rcu_read_unlock(); + * + * Note this code could be relatively easily generalized for other kinds + * of non-atomic data, but this initial version only handles strings. + * Only need to change the strlcpy() below to memcpy() + */ +#include +#include +#include +#include +#include +#include + +struct rcu_string { + struct rcu_head rcu; + char str[0]; +}; + +char *alloc_rcu_string(int size, gfp_t gfp) +{ + struct rcu_string *rs = kmalloc(sizeof(struct rcu_string) + size, gfp); + if (!rs) + return NULL; + return rs->str; +} +EXPORT_SYMBOL(alloc_rcu_string); + +static void do_free_rcu_string(struct rcu_head *h) +{ + kfree(container_of(h, struct rcu_string, rcu)); +} + +static inline struct rcu_string *str_to_rcustr(const char *str) +{ + /* + * Opencoded container_of because the strict type checking + * in normal container_of cannot deal with char str[0] vs char *str. + */ + return (struct rcu_string *)(str - offsetof(struct rcu_string, str)); +} + +void free_rcu_string(const char *str) +{ + struct rcu_string *rs = str_to_rcustr(str); + call_rcu(&rs->rcu, do_free_rcu_string); +} +EXPORT_SYMBOL(free_rcu_string); + +/* + * Get a local private copy of a RCU protected string. + * Mostly useful to get a string that is stable while sleeping. + * Caller must free returned string. + */ +char *access_rcu_string(char **str, int size, gfp_t gfp) +{ + char *copy = kmalloc(size, gfp); + if (!str) + return NULL; + rcu_read_lock(); + strlcpy(copy, rcu_dereference(*str), size); + rcu_read_unlock(); + return copy; +} +EXPORT_SYMBOL(access_rcu_string); Index: linux-2.6.33-rc2-ak/lib/Makefile =================================================================== --- linux-2.6.33-rc2-ak.orig/lib/Makefile +++ linux-2.6.33-rc2-ak/lib/Makefile @@ -12,7 +12,8 @@ lib-y := ctype.o string.o vsprintf.o cmd idr.o int_sqrt.o extable.o prio_tree.o \ sha1.o irq_regs.o reciprocal_div.o argv_split.o \ proportions.o prio_heap.o ratelimit.o show_mem.o \ - is_single_threaded.o plist.o decompress.o flex_array.o + is_single_threaded.o plist.o decompress.o flex_array.o \ + rcustring.o lib-$(CONFIG_MMU) += ioremap.o lib-$(CONFIG_SMP) += cpumask.o -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/