2005-04-16 15:20:36 -07:00
|
|
|
/*
|
|
|
|
* include/linux/idr.h
|
|
|
|
*
|
|
|
|
* 2002-10-18 written by Jim Houston jim.houston@ccur.com
|
|
|
|
* Copyright (C) 2002 by Concurrent Computer Corporation
|
|
|
|
* Distributed under the GNU GPL license version 2.
|
|
|
|
*
|
|
|
|
* Small id to pointer translation service avoiding fixed sized
|
|
|
|
* tables.
|
|
|
|
*/
|
2005-11-08 17:14:08 +01:00
|
|
|
|
|
|
|
#ifndef __IDR_H__
|
|
|
|
#define __IDR_H__
|
|
|
|
|
2005-04-16 15:20:36 -07:00
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/bitops.h>
|
2008-04-29 01:03:13 -07:00
|
|
|
#include <linux/init.h>
|
2008-07-25 01:47:57 -07:00
|
|
|
#include <linux/rcupdate.h>
|
2005-04-16 15:20:36 -07:00
|
|
|
|
|
|
|
#if BITS_PER_LONG == 32
|
|
|
|
# define IDR_BITS 5
|
|
|
|
# define IDR_FULL 0xfffffffful
|
|
|
|
/* We can only use two of the bits in the top level because there is
|
|
|
|
only one possible bit in the top level (5 bits * 7 levels = 35
|
|
|
|
bits, but you only use 31 bits in the id). */
|
|
|
|
# define TOP_LEVEL_FULL (IDR_FULL >> 30)
|
|
|
|
#elif BITS_PER_LONG == 64
|
|
|
|
# define IDR_BITS 6
|
|
|
|
# define IDR_FULL 0xfffffffffffffffful
|
|
|
|
/* We can only use two of the bits in the top level because there is
|
|
|
|
only one possible bit in the top level (6 bits * 6 levels = 36
|
|
|
|
bits, but you only use 31 bits in the id). */
|
|
|
|
# define TOP_LEVEL_FULL (IDR_FULL >> 62)
|
|
|
|
#else
|
|
|
|
# error "BITS_PER_LONG is not 32 or 64"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define IDR_SIZE (1 << IDR_BITS)
|
|
|
|
#define IDR_MASK ((1 << IDR_BITS)-1)
|
|
|
|
|
2012-10-04 17:13:15 -07:00
|
|
|
#define MAX_IDR_SHIFT (sizeof(int)*8 - 1)
|
|
|
|
#define MAX_IDR_BIT (1U << MAX_IDR_SHIFT)
|
|
|
|
#define MAX_IDR_MASK (MAX_IDR_BIT - 1)
|
2005-04-16 15:20:36 -07:00
|
|
|
|
|
|
|
/* Leave the possibility of an incomplete final layer */
|
2012-10-04 17:13:15 -07:00
|
|
|
#define MAX_IDR_LEVEL ((MAX_IDR_SHIFT + IDR_BITS - 1) / IDR_BITS)
|
2005-04-16 15:20:36 -07:00
|
|
|
|
|
|
|
/* Number of id_layer structs to leave in free list */
|
2012-10-04 17:13:15 -07:00
|
|
|
#define MAX_IDR_FREE (MAX_IDR_LEVEL * 2)
|
2005-04-16 15:20:36 -07:00
|
|
|
|
|
|
|
struct idr_layer {
|
|
|
|
unsigned long bitmap; /* A zero bit means "space here" */
|
2010-02-26 14:53:26 +01:00
|
|
|
struct idr_layer __rcu *ary[1<<IDR_BITS];
|
2005-04-16 15:20:36 -07:00
|
|
|
int count; /* When zero, we can release it */
|
2008-12-01 13:14:02 -08:00
|
|
|
int layer; /* distance from leaf */
|
2008-07-25 01:47:57 -07:00
|
|
|
struct rcu_head rcu_head;
|
2005-04-16 15:20:36 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct idr {
|
2010-02-26 14:53:26 +01:00
|
|
|
struct idr_layer __rcu *top;
|
2005-04-16 15:20:36 -07:00
|
|
|
struct idr_layer *id_free;
|
2008-12-01 13:14:02 -08:00
|
|
|
int layers; /* only valid without concurrent changes */
|
2005-04-16 15:20:36 -07:00
|
|
|
int id_free_cnt;
|
|
|
|
spinlock_t lock;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define IDR_INIT(name) \
|
|
|
|
{ \
|
|
|
|
.top = NULL, \
|
|
|
|
.id_free = NULL, \
|
|
|
|
.layers = 0, \
|
|
|
|
.id_free_cnt = 0, \
|
2006-07-03 00:24:34 -07:00
|
|
|
.lock = __SPIN_LOCK_UNLOCKED(name.lock), \
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
|
|
|
#define DEFINE_IDR(name) struct idr name = IDR_INIT(name)
|
|
|
|
|
2008-07-25 01:47:59 -07:00
|
|
|
/* Actions to be taken after a call to _idr_sub_alloc */
|
|
|
|
#define IDR_NEED_TO_GROW -2
|
|
|
|
#define IDR_NOMORE_SPACE -3
|
|
|
|
|
|
|
|
#define _idr_rc_to_errno(rc) ((rc) == -1 ? -EAGAIN : -ENOSPC)
|
|
|
|
|
2008-07-25 01:48:01 -07:00
|
|
|
/**
|
2010-10-26 14:19:08 -07:00
|
|
|
* DOC: idr sync
|
2008-07-25 01:48:01 -07:00
|
|
|
* idr synchronization (stolen from radix-tree.h)
|
|
|
|
*
|
|
|
|
* idr_find() is able to be called locklessly, using RCU. The caller must
|
|
|
|
* ensure calls to this function are made within rcu_read_lock() regions.
|
|
|
|
* Other readers (lock-free or otherwise) and modifications may be running
|
|
|
|
* concurrently.
|
|
|
|
*
|
|
|
|
* It is still required that the caller manage the synchronization and
|
|
|
|
* lifetimes of the items. So if RCU lock-free lookups are used, typically
|
|
|
|
* this would mean that the items have their own locks, or are amenable to
|
|
|
|
* lock-free access; and that the items are freed by RCU (or only freed after
|
|
|
|
* having been deleted from the idr tree *and* a synchronize_rcu() grace
|
|
|
|
* period).
|
|
|
|
*/
|
|
|
|
|
2005-04-16 15:20:36 -07:00
|
|
|
/*
|
|
|
|
* This is what we export.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void *idr_find(struct idr *idp, int id);
|
2005-10-21 03:18:50 -04:00
|
|
|
int idr_pre_get(struct idr *idp, gfp_t gfp_mask);
|
2005-04-16 15:20:36 -07:00
|
|
|
int idr_get_new(struct idr *idp, void *ptr, int *id);
|
|
|
|
int idr_get_new_above(struct idr *idp, void *ptr, int starting_id, int *id);
|
2007-07-15 23:37:24 -07:00
|
|
|
int idr_for_each(struct idr *idp,
|
|
|
|
int (*fn)(int id, void *p, void *data), void *data);
|
cgroup: CSS ID support
Patch for Per-CSS(Cgroup Subsys State) ID and private hierarchy code.
This patch attaches unique ID to each css and provides following.
- css_lookup(subsys, id)
returns pointer to struct cgroup_subysys_state of id.
- css_get_next(subsys, id, rootid, depth, foundid)
returns the next css under "root" by scanning
When cgroup_subsys->use_id is set, an id for css is maintained.
The cgroup framework only parepares
- css_id of root css for subsys
- id is automatically attached at creation of css.
- id is *not* freed automatically. Because the cgroup framework
don't know lifetime of cgroup_subsys_state.
free_css_id() function is provided. This must be called by subsys.
There are several reasons to develop this.
- Saving space .... For example, memcg's swap_cgroup is array of
pointers to cgroup. But it is not necessary to be very fast.
By replacing pointers(8bytes per ent) to ID (2byes per ent), we can
reduce much amount of memory usage.
- Scanning without lock.
CSS_ID provides "scan id under this ROOT" function. By this, scanning
css under root can be written without locks.
ex)
do {
rcu_read_lock();
next = cgroup_get_next(subsys, id, root, &found);
/* check sanity of next here */
css_tryget();
rcu_read_unlock();
id = found + 1
} while(...)
Characteristics:
- Each css has unique ID under subsys.
- Lifetime of ID is controlled by subsys.
- css ID contains "ID" and "Depth in hierarchy" and stack of hierarchy
- Allowed ID is 1-65535, ID 0 is UNUSED ID.
Design Choices:
- scan-by-ID v.s. scan-by-tree-walk.
As /proc's pid scan does, scan-by-ID is robust when scanning is done
by following kind of routine.
scan -> rest a while(release a lock) -> conitunue from interrupted
memcg's hierarchical reclaim does this.
- When subsys->use_id is set, # of css in the system is limited to
65535.
[bharata@linux.vnet.ibm.com: remove rcu_read_lock() from css_get_next()]
Signed-off-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Acked-by: Paul Menage <menage@google.com>
Cc: Li Zefan <lizf@cn.fujitsu.com>
Cc: Balbir Singh <balbir@in.ibm.com>
Cc: Daisuke Nishimura <nishimura@mxp.nes.nec.co.jp>
Signed-off-by: Bharata B Rao <bharata@linux.vnet.ibm.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2009-04-02 16:57:25 -07:00
|
|
|
void *idr_get_next(struct idr *idp, int *nextid);
|
2006-06-26 00:27:19 -07:00
|
|
|
void *idr_replace(struct idr *idp, void *ptr, int id);
|
2005-04-16 15:20:36 -07:00
|
|
|
void idr_remove(struct idr *idp, int id);
|
2005-10-23 12:57:18 -07:00
|
|
|
void idr_destroy(struct idr *idp);
|
2005-04-16 15:20:36 -07:00
|
|
|
void idr_init(struct idr *idp);
|
2005-11-08 17:14:08 +01:00
|
|
|
|
2013-02-27 17:03:50 -08:00
|
|
|
void __idr_remove_all(struct idr *idp); /* don't use */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* idr_remove_all - remove all ids from the given idr tree
|
|
|
|
* @idp: idr handle
|
|
|
|
*
|
|
|
|
* If you're trying to destroy @idp, calling idr_destroy() is enough.
|
|
|
|
* This is going away. Don't use.
|
|
|
|
*/
|
|
|
|
static inline void __deprecated idr_remove_all(struct idr *idp)
|
|
|
|
{
|
|
|
|
__idr_remove_all(idp);
|
|
|
|
}
|
2007-06-14 03:45:13 +09:00
|
|
|
|
|
|
|
/*
|
|
|
|
* IDA - IDR based id allocator, use when translation from id to
|
|
|
|
* pointer isn't necessary.
|
2010-09-16 01:30:19 +09:00
|
|
|
*
|
|
|
|
* IDA_BITMAP_LONGS is calculated to be one less to accommodate
|
|
|
|
* ida_bitmap->nr_busy so that the whole struct fits in 128 bytes.
|
2007-06-14 03:45:13 +09:00
|
|
|
*/
|
|
|
|
#define IDA_CHUNK_SIZE 128 /* 128 bytes per chunk */
|
2010-09-16 01:30:19 +09:00
|
|
|
#define IDA_BITMAP_LONGS (IDA_CHUNK_SIZE / sizeof(long) - 1)
|
|
|
|
#define IDA_BITMAP_BITS (IDA_BITMAP_LONGS * sizeof(long) * 8)
|
2007-06-14 03:45:13 +09:00
|
|
|
|
|
|
|
struct ida_bitmap {
|
|
|
|
long nr_busy;
|
|
|
|
unsigned long bitmap[IDA_BITMAP_LONGS];
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ida {
|
|
|
|
struct idr idr;
|
|
|
|
struct ida_bitmap *free_bitmap;
|
|
|
|
};
|
|
|
|
|
2011-07-17 21:25:03 +02:00
|
|
|
#define IDA_INIT(name) { .idr = IDR_INIT((name).idr), .free_bitmap = NULL, }
|
2007-06-14 03:45:13 +09:00
|
|
|
#define DEFINE_IDA(name) struct ida name = IDA_INIT(name)
|
|
|
|
|
|
|
|
int ida_pre_get(struct ida *ida, gfp_t gfp_mask);
|
|
|
|
int ida_get_new_above(struct ida *ida, int starting_id, int *p_id);
|
|
|
|
int ida_get_new(struct ida *ida, int *p_id);
|
|
|
|
void ida_remove(struct ida *ida, int id);
|
|
|
|
void ida_destroy(struct ida *ida);
|
|
|
|
void ida_init(struct ida *ida);
|
|
|
|
|
2011-08-03 16:21:06 -07:00
|
|
|
int ida_simple_get(struct ida *ida, unsigned int start, unsigned int end,
|
|
|
|
gfp_t gfp_mask);
|
|
|
|
void ida_simple_remove(struct ida *ida, unsigned int id);
|
|
|
|
|
2008-04-29 01:03:13 -07:00
|
|
|
void __init idr_init_cache(void);
|
|
|
|
|
2011-07-20 14:59:37 +02:00
|
|
|
/**
|
|
|
|
* idr_for_each_entry - iterate over an idr's elements of a given type
|
|
|
|
* @idp: idr handle
|
|
|
|
* @entry: the type * to use as cursor
|
|
|
|
* @id: id entry's key
|
|
|
|
*/
|
|
|
|
#define idr_for_each_entry(idp, entry, id) \
|
|
|
|
for (id = 0, entry = (typeof(entry))idr_get_next((idp), &(id)); \
|
|
|
|
entry != NULL; \
|
|
|
|
++id, entry = (typeof(entry))idr_get_next((idp), &(id)))
|
|
|
|
|
2005-11-08 17:14:08 +01:00
|
|
|
#endif /* __IDR_H__ */
|