Commit c3baa3de authored by Christoph Hellwig's avatar Christoph Hellwig Committed by Linus Torvalds

[PATCH] idr.c: remove stale comment

The generation counters were removed from the idr code.  Update idr.c
comments, now uses doc-book style API documentation.
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 901294f1
/* /*
* linux/kernel/id.c
*
* 2002-10-18 written by Jim Houston jim.houston@ccur.com * 2002-10-18 written by Jim Houston jim.houston@ccur.com
* Copyright (C) 2002 by Concurrent Computer Corporation * Copyright (C) 2002 by Concurrent Computer Corporation
* Distributed under the GNU GPL license version 2. * Distributed under the GNU GPL license version 2.
* *
* Modified by George Anzinger to reuse immediately and to use
* find bit instructions. Also removed _irq on spinlocks.
*
* Small id to pointer translation service. * Small id to pointer translation service.
* *
* It uses a radix tree like structure as a sparse array indexed * It uses a radix tree like structure as a sparse array indexed
* by the id to obtain the pointer. The bitmap makes allocating * by the id to obtain the pointer. The bitmap makes allocating
* a new id quick. * a new id quick.
*
* Modified by George Anzinger to reuse immediately and to use
* find bit instructions. Also removed _irq on spinlocks.
* So here is what this bit of code does:
* You call it to allocate an id (an int) an associate with that id a * You call it to allocate an id (an int) an associate with that id a
* pointer or what ever, we treat it as a (void *). You can pass this * pointer or what ever, we treat it as a (void *). You can pass this
* id to a user for him to pass back at a later time. You then pass * id to a user for him to pass back at a later time. You then pass
...@@ -26,78 +22,8 @@ ...@@ -26,78 +22,8 @@
* don't need to go to the memory "store" during an id allocate, just * don't need to go to the memory "store" during an id allocate, just
* so you don't need to be too concerned about locking and conflicts * so you don't need to be too concerned about locking and conflicts
* with the slab allocator. * with the slab allocator.
* What you need to do is, since we don't keep the counter as part of
* id / ptr pair, to keep a copy of it in the pointed to structure
* (or else where) so that when you ask for a ptr you can varify that
* the returned ptr is correct by comparing the id it contains with the one
* you asked for. In other words, we only did half the reuse protection.
* Since the code depends on your code doing this check, we ignore high
* order bits in the id, not just the count, but bits that would, if used,
* index outside of the allocated ids. In other words, if the largest id
* currently allocated is 32 a look up will only look at the low 5 bits of
* the id. Since you will want to keep this id in the structure anyway
* (if for no other reason than to be able to eliminate the id when the
* structure is found in some other way) this seems reasonable. If you
* really think otherwise, the code to check these bits here, it is just
* disabled with a #if 0.
* So here are the complete details:
* include <linux/idr.h>
* void idr_init(struct idr *idp)
* This function is use to set up the handle (idp) that you will pass
* to the rest of the functions. The structure is defined in the
* header.
* int idr_pre_get(struct idr *idp, unsigned gfp_mask)
* This function should be called prior to locking and calling the
* following function. It pre allocates enough memory to satisfy the
* worst possible allocation. Unless gfp_mask is GFP_ATOMIC, it can
* sleep, so must not be called with any spinlocks held. If the system is
* REALLY out of memory this function returns 0, other wise 1.
* int idr_get_new(struct idr *idp, void *ptr, int *id);
* This is the allocate id function. It should be called with any
* required locks. In fact, in the SMP case, you MUST lock prior to
* calling this function to avoid possible out of memory problems.
* If memory is required, it will return -EAGAIN, you should unlock
* and go back to the idr_pre_get() call. If the idr is full, it
* will return a -ENOSPC. ptr is the pointer you want associated
* with the id. The value is returned in the "id" field. idr_get_new()
* returns a value in the range 0 ... 0x7fffffff
* int idr_get_new_above(struct idr *idp, void *ptr, int start_id, int *id);
* Like idr_get_new(), but the returned id is guaranteed to be at or
* above start_id.
* void *idr_find(struct idr *idp, int id);
* returns the "ptr", given the id. A NULL return indicates that the
* id is not valid (or you passed NULL in the idr_get_new(), shame on
* you). This function must be called with a spinlock that prevents
* calling either idr_get_new() or idr_remove() or idr_find() while it
* is working.
* void idr_remove(struct idr *idp, int id);
* removes the given id, freeing that slot and any memory that may
* now be unused. See idr_find() for locking restrictions.
* int idr_full(struct idr *idp);
* Returns true if the idr is full and false if not.
*/ */
#ifndef TEST // to test in user space... #ifndef TEST // to test in user space...
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/init.h> #include <linux/init.h>
...@@ -106,11 +32,8 @@ ...@@ -106,11 +32,8 @@
#include <linux/string.h> #include <linux/string.h>
#include <linux/idr.h> #include <linux/idr.h>
static kmem_cache_t *idr_layer_cache; static kmem_cache_t *idr_layer_cache;
static struct idr_layer *alloc_layer(struct idr *idp) static struct idr_layer *alloc_layer(struct idr *idp)
{ {
struct idr_layer *p; struct idr_layer *p;
...@@ -137,6 +60,18 @@ static void free_layer(struct idr *idp, struct idr_layer *p) ...@@ -137,6 +60,18 @@ static void free_layer(struct idr *idp, struct idr_layer *p)
spin_unlock(&idp->lock); spin_unlock(&idp->lock);
} }
/**
* idr_pre_get - reserver resources for idr allocation
* @idp: idr handle
* @gfp_mask: memory allocation flags
*
* This function should be called prior to locking and calling the
* following function. It preallocates enough memory to satisfy
* the worst possible allocation.
*
* If the system is REALLY out of memory this function returns 0,
* otherwise 1.
*/
int idr_pre_get(struct idr *idp, unsigned gfp_mask) int idr_pre_get(struct idr *idp, unsigned gfp_mask)
{ {
while (idp->id_free_cnt < IDR_FREE_MAX) { while (idp->id_free_cnt < IDR_FREE_MAX) {
...@@ -271,6 +206,22 @@ static int idr_get_new_above_int(struct idr *idp, void *ptr, int starting_id) ...@@ -271,6 +206,22 @@ static int idr_get_new_above_int(struct idr *idp, void *ptr, int starting_id)
return(v); return(v);
} }
/**
* idr_get_new_above - allocate new idr entry above a start id
* @idp: idr handle
* @ptr: pointer you want associated with the ide
* @start_id: id to start search at
* @id: pointer to the allocated handle
*
* This is the allocate id function. It should be called with any
* required locks.
*
* If memory is required, it will return -EAGAIN, you should unlock
* and go back to the idr_pre_get() call. If the idr is full, it will
* return -ENOSPC.
*
* @id returns a value in the range 0 ... 0x7fffffff
*/
int idr_get_new_above(struct idr *idp, void *ptr, int starting_id, int *id) int idr_get_new_above(struct idr *idp, void *ptr, int starting_id, int *id)
{ {
int rv; int rv;
...@@ -290,6 +241,21 @@ int idr_get_new_above(struct idr *idp, void *ptr, int starting_id, int *id) ...@@ -290,6 +241,21 @@ int idr_get_new_above(struct idr *idp, void *ptr, int starting_id, int *id)
} }
EXPORT_SYMBOL(idr_get_new_above); EXPORT_SYMBOL(idr_get_new_above);
/**
* idr_get_new - allocate new idr entry
* @idp: idr handle
* @ptr: pointer you want associated with the ide
* @id: pointer to the allocated handle
*
* This is the allocate id function. It should be called with any
* required locks.
*
* If memory is required, it will return -EAGAIN, you should unlock
* and go back to the idr_pre_get() call. If the idr is full, it will
* return -ENOSPC.
*
* @id returns a value in the range 0 ... 0x7fffffff
*/
int idr_get_new(struct idr *idp, void *ptr, int *id) int idr_get_new(struct idr *idp, void *ptr, int *id)
{ {
int rv; int rv;
...@@ -338,6 +304,11 @@ static void sub_remove(struct idr *idp, int shift, int id) ...@@ -338,6 +304,11 @@ static void sub_remove(struct idr *idp, int shift, int id)
} }
} }
/**
* idr_remove - remove the given id and free it's slot
* idp: idr handle
* id: uniqueue key
*/
void idr_remove(struct idr *idp, int id) void idr_remove(struct idr *idp, int id)
{ {
struct idr_layer *p; struct idr_layer *p;
...@@ -365,6 +336,17 @@ void idr_remove(struct idr *idp, int id) ...@@ -365,6 +336,17 @@ void idr_remove(struct idr *idp, int id)
} }
EXPORT_SYMBOL(idr_remove); EXPORT_SYMBOL(idr_remove);
/**
* idr_find - return pointer for given id
* @idp: idr handle
* @id: lookup key
*
* Return the pointer given the id it has been registered with. A %NULL
* return indicates that @id is not valid or you passed %NULL in
* idr_get_new().
*
* The caller must serialize idr_find() vs idr_get_new() and idr_remove().
*/
void *idr_find(struct idr *idp, int id) void *idr_find(struct idr *idp, int id)
{ {
int n; int n;
...@@ -372,17 +354,13 @@ void *idr_find(struct idr *idp, int id) ...@@ -372,17 +354,13 @@ void *idr_find(struct idr *idp, int id)
n = idp->layers * IDR_BITS; n = idp->layers * IDR_BITS;
p = idp->top; p = idp->top;
#if 0
/*
* This tests to see if bits outside the current tree are
* present. If so, tain't one of ours!
*/
if ( unlikely( (id & ~(~0 << MAX_ID_SHIFT)) >> (n + IDR_BITS)))
return NULL;
#endif
/* Mask off upper bits we don't use for the search. */ /* Mask off upper bits we don't use for the search. */
id &= MAX_ID_MASK; id &= MAX_ID_MASK;
if (id >= (1 << n))
return NULL;
while (n > 0 && p) { while (n > 0 && p) {
n -= IDR_BITS; n -= IDR_BITS;
p = p->ary[(id >> n) & IDR_MASK]; p = p->ary[(id >> n) & IDR_MASK];
...@@ -405,6 +383,13 @@ static int init_id_cache(void) ...@@ -405,6 +383,13 @@ static int init_id_cache(void)
return 0; return 0;
} }
/**
* idr_init - initialize idr handle
* @idp: idr handle
*
* This function is use to set up the handle (@idp) that you will pass
* to the rest of the functions.
*/
void idr_init(struct idr *idp) void idr_init(struct idr *idp)
{ {
init_id_cache(); init_id_cache();
...@@ -412,4 +397,3 @@ void idr_init(struct idr *idp) ...@@ -412,4 +397,3 @@ void idr_init(struct idr *idp)
spin_lock_init(&idp->lock); spin_lock_init(&idp->lock);
} }
EXPORT_SYMBOL(idr_init); EXPORT_SYMBOL(idr_init);
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment