Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1751186AbWHXMDW (ORCPT ); Thu, 24 Aug 2006 08:03:22 -0400 Received: (majordomo@vger.kernel.org) by vger.kernel.org id S1751191AbWHXMDW (ORCPT ); Thu, 24 Aug 2006 08:03:22 -0400 Received: from mailhub.sw.ru ([195.214.233.200]:37466 "EHLO relay.sw.ru") by vger.kernel.org with ESMTP id S1751186AbWHXMDV (ORCPT ); Thu, 24 Aug 2006 08:03:21 -0400 Message-ID: <44ED9633.7090504@sw.ru> Date: Thu, 24 Aug 2006 16:06:11 +0400 From: Kirill Korotaev User-Agent: Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.7.13) Gecko/20060417 X-Accept-Language: en-us, en, ru MIME-Version: 1.0 To: Andrew Morton CC: Linux Kernel Mailing List , Alan Cox , Christoph Hellwig , Pavel Emelianov , Andrey Savochkin , devel@openvz.org, Rik van Riel , Andi Kleen , Greg KH , Oleg Nesterov , Matt Helsley , Rohit Seth , Chandra Seetharaman Subject: Re: [PATCH 2/6] BC: beancounters core (API) References: <44EC31FB.2050002@sw.ru> <44EC35EB.1030000@sw.ru> <20060823094202.ff3a5573.akpm@osdl.org> In-Reply-To: <20060823094202.ff3a5573.akpm@osdl.org> Content-Type: text/plain; charset=us-ascii; format=flowed Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 3303 Lines: 124 Andrew Morton wrote: [... snip ...] >>+#define bc_charge_locked(bc, r, v, s) (0) >>> +#define bc_charge(bc, r, v) (0) > >akpm:/home/akpm> cat t.c >void foo(void) >{ > (0); >} >akpm:/home/akpm> gcc -c -Wall t.c >t.c: In function 'foo': >t.c:4: warning: statement with no effect these functions return value should always be checked (!). i.e. it is never called like: ub_charge(bc, r, v); >>+struct beancounter *beancounter_findcreate(uid_t uid, int mask) >>+{ >>+ struct beancounter *new_bc, *bc; >>+ unsigned long flags; >>+ struct hlist_head *slot; >>+ struct hlist_node *pos; >>+ >>+ slot = &bc_hash[bc_hash_fun(uid)]; >>+ new_bc = NULL; >>+ >>+retry: >>+ spin_lock_irqsave(&bc_hash_lock, flags); >>+ hlist_for_each_entry (bc, pos, slot, hash) >>+ if (bc->bc_id == uid) >>+ break; >>+ >>+ if (pos != NULL) { >>+ get_beancounter(bc); >>+ spin_unlock_irqrestore(&bc_hash_lock, flags); >>+ >>+ if (new_bc != NULL) >>+ kmem_cache_free(bc_cachep, new_bc); >>+ return bc; >>+ } >>+ >>+ if (!(mask & BC_ALLOC)) >>+ goto out_unlock; >>+ >>+ if (new_bc != NULL) >>+ goto out_install; >>+ >>+ spin_unlock_irqrestore(&bc_hash_lock, flags); >>+ >>+ new_bc = kmem_cache_alloc(bc_cachep, >>+ mask & BC_ALLOC_ATOMIC ? GFP_ATOMIC : GFP_KERNEL); >>+ if (new_bc == NULL) >>+ goto out; >>+ >>+ memcpy(new_bc, &default_beancounter, sizeof(*new_bc)); >>+ init_beancounter_struct(new_bc, uid); >>+ goto retry; >>+ >>+out_install: >>+ hlist_add_head(&new_bc->hash, slot); >>+out_unlock: >>+ spin_unlock_irqrestore(&bc_hash_lock, flags); >>+out: >>+ return new_bc; >>+} > > > Can remove the global bc_hash_lock and make the locking per-hash-bucket. it is not performance critical path IMHO. this lock is taken on container create/change/destroy/user interfaces only. >>+static inline void verify_held(struct beancounter *bc) >>+{ >>+ int i; >>+ >>+ for (i = 0; i < BC_RESOURCES; i++) >>+ if (bc->bc_parms[i].held != 0) >>+ bc_print_resource_warning(bc, i, >>+ "resource is held on put", 0, 0); >>+} >>+ >>+void __put_beancounter(struct beancounter *bc) >>+{ >>+ unsigned long flags; >>+ >>+ /* equivalent to atomic_dec_and_lock_irqsave() */ >>+ local_irq_save(flags); >>+ if (likely(!atomic_dec_and_lock(&bc->bc_refcount, &bc_hash_lock))) { >>+ local_irq_restore(flags); >>+ if (unlikely(atomic_read(&bc->bc_refcount) < 0)) >>+ printk(KERN_ERR "BC: Bad refcount: bc=%p, " >>+ "luid=%d, ref=%d\n", >>+ bc, bc->bc_id, >>+ atomic_read(&bc->bc_refcount)); >>+ return; >>+ } >>+ >>+ BUG_ON(bc == &init_bc); >>+ verify_held(bc); >>+ hlist_del(&bc->hash); >>+ spin_unlock_irqrestore(&bc_hash_lock, flags); >>+ kmem_cache_free(bc_cachep, bc); >>+} > > I wonder if it's safe and worthwhile to optimise away the local_irq_save(): > > if (atomic_dec_and_test(&bc->bc_refcount)) { > spin_lock_irqsave(&bc_hash_lock, flags); > if (atomic_read(&bc->bc_refcount) == 0) { > free it put_beancounter can happen from IRQ context. so we need something like atomic_dec_and_lock_irqsave() Oleg Nesterov proposed more details. Thanks, Kirill - 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/