| 
									
										
										
										
											2008-03-01 03:45:59 +03:00
										 |  |  | #include <sys/kmem.h>
 | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | #ifdef DEBUG_SUBSYSTEM
 | 
					
						
							|  |  |  | #undef DEBUG_SUBSYSTEM
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DEBUG_SUBSYSTEM S_KMEM
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Memory allocation interfaces | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #ifdef DEBUG_KMEM
 | 
					
						
							|  |  |  | /* Shim layer memory accounting */ | 
					
						
							| 
									
										
										
										
											2008-03-14 23:56:26 +03:00
										 |  |  | atomic64_t kmem_alloc_used; | 
					
						
							|  |  |  | unsigned long kmem_alloc_max = 0; | 
					
						
							|  |  |  | atomic64_t vmem_alloc_used; | 
					
						
							|  |  |  | unsigned long vmem_alloc_max = 0; | 
					
						
							|  |  |  | int kmem_warning_flag = 1; | 
					
						
							| 
									
										
										
										
											2008-03-14 22:04:41 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | EXPORT_SYMBOL(kmem_alloc_used); | 
					
						
							|  |  |  | EXPORT_SYMBOL(kmem_alloc_max); | 
					
						
							|  |  |  | EXPORT_SYMBOL(vmem_alloc_used); | 
					
						
							|  |  |  | EXPORT_SYMBOL(vmem_alloc_max); | 
					
						
							| 
									
										
										
										
											2008-03-14 23:56:26 +03:00
										 |  |  | EXPORT_SYMBOL(kmem_warning_flag); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int kmem_set_warning(int flag) { return (kmem_warning_flag = !!flag); } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | int kmem_set_warning(int flag) { return 0; } | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-03-14 23:56:26 +03:00
										 |  |  | EXPORT_SYMBOL(kmem_set_warning); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Slab allocation interfaces | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * While the linux slab implementation was inspired by solaris they | 
					
						
							|  |  |  |  * have made some changes to the API which complicates this shim | 
					
						
							|  |  |  |  * layer.  For one thing the same symbol names are used with different | 
					
						
							|  |  |  |  * arguments for the prototypes.  To deal with this we must use the | 
					
						
							|  |  |  |  * preprocessor to re-order arguments.  Happily for us standard C says, | 
					
						
							|  |  |  |  * "Macro's appearing in their own expansion are not reexpanded" so | 
					
						
							|  |  |  |  * this does not result in an infinite recursion.  Additionally the | 
					
						
							|  |  |  |  * function pointers registered by solarias differ from those used | 
					
						
							|  |  |  |  * by linux so a lookup and mapping from linux style callback to a | 
					
						
							|  |  |  |  * solaris style callback is needed.  There is some overhead in this | 
					
						
							|  |  |  |  * operation which isn't horibile but it needs to be kept in mind. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | typedef struct kmem_cache_cb { | 
					
						
							|  |  |  |         struct list_head    kcc_list; | 
					
						
							|  |  |  |         kmem_cache_t *      kcc_cache; | 
					
						
							|  |  |  |         kmem_constructor_t  kcc_constructor; | 
					
						
							|  |  |  |         kmem_destructor_t   kcc_destructor; | 
					
						
							|  |  |  |         kmem_reclaim_t      kcc_reclaim; | 
					
						
							|  |  |  |         void *              kcc_private; | 
					
						
							|  |  |  |         void *              kcc_vmp; | 
					
						
							|  |  |  | } kmem_cache_cb_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static spinlock_t kmem_cache_cb_lock = SPIN_LOCK_UNLOCKED; | 
					
						
							|  |  |  | static LIST_HEAD(kmem_cache_cb_list); | 
					
						
							|  |  |  | static struct shrinker *kmem_cache_shrinker; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Function must be called while holding the kmem_cache_cb_lock
 | 
					
						
							|  |  |  |  * Because kmem_cache_t is an opaque datatype we're forced to | 
					
						
							|  |  |  |  * match pointers to identify specific cache entires. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static kmem_cache_cb_t * | 
					
						
							|  |  |  | kmem_cache_find_cache_cb(kmem_cache_t *cache) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |         kmem_cache_cb_t *kcc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         list_for_each_entry(kcc, &kmem_cache_cb_list, kcc_list) | 
					
						
							|  |  |  | 		if (cache == kcc->kcc_cache) | 
					
						
							|  |  |  |                         return kcc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static kmem_cache_cb_t * | 
					
						
							|  |  |  | kmem_cache_add_cache_cb(kmem_cache_t *cache, | 
					
						
							|  |  |  | 			kmem_constructor_t constructor, | 
					
						
							|  |  |  |                         kmem_destructor_t destructor, | 
					
						
							|  |  |  | 			kmem_reclaim_t reclaim, | 
					
						
							|  |  |  |                         void *priv, void *vmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |         kmem_cache_cb_t *kcc; | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	unsigned long flags; | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         kcc = (kmem_cache_cb_t *)kmalloc(sizeof(*kcc), GFP_KERNEL); | 
					
						
							|  |  |  |         if (kcc) { | 
					
						
							|  |  |  | 		kcc->kcc_cache = cache; | 
					
						
							|  |  |  |                 kcc->kcc_constructor = constructor; | 
					
						
							|  |  |  |                 kcc->kcc_destructor = destructor; | 
					
						
							|  |  |  |                 kcc->kcc_reclaim = reclaim; | 
					
						
							|  |  |  |                 kcc->kcc_private = priv; | 
					
						
							|  |  |  |                 kcc->kcc_vmp = vmp; | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 		spin_lock_irqsave(&kmem_cache_cb_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |                 list_add(&kcc->kcc_list, &kmem_cache_cb_list); | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 		spin_unlock_irqrestore(&kmem_cache_cb_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return kcc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | kmem_cache_remove_cache_cb(kmem_cache_cb_t *kcc) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spin_lock_irqsave(&kmem_cache_cb_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |         list_del(&kcc->kcc_list); | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_unlock_irqrestore(&kmem_cache_cb_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |        if (kcc) | 
					
						
							|  |  |  |               kfree(kcc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | kmem_cache_generic_constructor(void *ptr, kmem_cache_t *cache, unsigned long flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |         kmem_cache_cb_t *kcc; | 
					
						
							| 
									
										
										
										
											2008-04-16 00:53:36 +04:00
										 |  |  | 	kmem_constructor_t constructor; | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	unsigned long irqflags; | 
					
						
							| 
									
										
										
										
											2008-04-16 00:53:36 +04:00
										 |  |  | 	void *private; | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_lock_irqsave(&kmem_cache_cb_lock, irqflags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /* Callback list must be in sync with linux slab caches */ | 
					
						
							|  |  |  |         kcc = kmem_cache_find_cache_cb(cache); | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  |         ASSERT(kcc); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-16 00:53:36 +04:00
										 |  |  | 	constructor = kcc->kcc_constructor; | 
					
						
							|  |  |  | 	private = kcc->kcc_private; | 
					
						
							| 
									
										
										
										
											2008-04-03 20:33:31 +04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_unlock_irqrestore(&kmem_cache_cb_lock, irqflags); | 
					
						
							| 
									
										
										
										
											2008-04-16 00:53:36 +04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (constructor) | 
					
						
							|  |  |  | 		constructor(ptr, private, (int)flags); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 	/* Linux constructor has no return code, silently eat it */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | kmem_cache_generic_destructor(void *ptr, kmem_cache_t *cache, unsigned long flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |         kmem_cache_cb_t *kcc; | 
					
						
							| 
									
										
										
										
											2008-04-16 00:53:36 +04:00
										 |  |  |         kmem_destructor_t destructor; | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	unsigned long irqflags; | 
					
						
							| 
									
										
										
										
											2008-04-16 00:53:36 +04:00
										 |  |  | 	void *private; | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_lock_irqsave(&kmem_cache_cb_lock, irqflags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /* Callback list must be in sync with linux slab caches */ | 
					
						
							|  |  |  |         kcc = kmem_cache_find_cache_cb(cache); | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | 	ASSERT(kcc); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-16 00:53:36 +04:00
										 |  |  | 	destructor = kcc->kcc_destructor; | 
					
						
							|  |  |  | 	private = kcc->kcc_private; | 
					
						
							| 
									
										
										
										
											2008-04-03 20:33:31 +04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_unlock_irqrestore(&kmem_cache_cb_lock, irqflags); | 
					
						
							| 
									
										
										
										
											2008-04-16 00:53:36 +04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Solaris destructor takes no flags, silently eat them */ | 
					
						
							|  |  |  | 	if (destructor) | 
					
						
							|  |  |  | 		destructor(ptr, private); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* XXX - Arguments are ignored */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | kmem_cache_generic_shrinker(int nr_to_scan, unsigned int gfp_mask) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |         kmem_cache_cb_t *kcc; | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	unsigned long flags; | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |         int total = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Under linux a shrinker is not tightly coupled with a slab
 | 
					
						
							|  |  |  | 	 * cache.  In fact linux always systematically trys calling all | 
					
						
							|  |  |  | 	 * registered shrinker callbacks until its target reclamation level | 
					
						
							|  |  |  | 	 * is reached.  Because of this we only register one shrinker | 
					
						
							|  |  |  | 	 * function in the shim layer for all slab caches.  And we always | 
					
						
							|  |  |  | 	 * attempt to shrink all caches when this generic shrinker is called. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_lock_irqsave(&kmem_cache_cb_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         list_for_each_entry(kcc, &kmem_cache_cb_list, kcc_list) { | 
					
						
							|  |  |  | 	        /* Under linux the desired number and gfp type of objects
 | 
					
						
							|  |  |  | 		 * is passed to the reclaiming function as a sugested reclaim | 
					
						
							|  |  |  | 		 * target.  I do not pass these args on because reclaim | 
					
						
							|  |  |  | 		 * policy is entirely up to the owner under solaris.  We only | 
					
						
							|  |  |  | 		 * pass on the pre-registered private data. | 
					
						
							|  |  |  |                  */ | 
					
						
							|  |  |  | 		if (kcc->kcc_reclaim) | 
					
						
							|  |  |  |                         kcc->kcc_reclaim(kcc->kcc_private); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	        total += 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Under linux we should return the remaining number of entires in
 | 
					
						
							|  |  |  | 	 * the cache.  Unfortunately, I don't see an easy way to safely | 
					
						
							|  |  |  | 	 * emulate this behavior so I'm returning one entry per cache which | 
					
						
							|  |  |  | 	 * was registered with the generic shrinker.  This should fake out | 
					
						
							|  |  |  | 	 * the linux VM when it attempts to shrink caches. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_unlock_irqrestore(&kmem_cache_cb_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 	return total; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Ensure the __kmem_cache_create/__kmem_cache_destroy macros are
 | 
					
						
							|  |  |  |  * removed here to prevent a recursive substitution, we want to call | 
					
						
							|  |  |  |  * the native linux version. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #undef kmem_cache_create
 | 
					
						
							|  |  |  | #undef kmem_cache_destroy
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | kmem_cache_t * | 
					
						
							|  |  |  | __kmem_cache_create(char *name, size_t size, size_t align, | 
					
						
							| 
									
										
										
										
											2008-02-27 22:09:51 +03:00
										 |  |  |         kmem_constructor_t constructor, | 
					
						
							|  |  |  | 	kmem_destructor_t destructor, | 
					
						
							|  |  |  | 	kmem_reclaim_t reclaim, | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |         void *priv, void *vmp, int flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |         kmem_cache_t *cache; | 
					
						
							|  |  |  |         kmem_cache_cb_t *kcc; | 
					
						
							|  |  |  | 	int shrinker_flag = 0; | 
					
						
							| 
									
										
										
										
											2008-03-14 23:56:26 +03:00
										 |  |  | 	char *cache_name; | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | 	ENTRY; | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  |         /* XXX: - Option currently unsupported by shim layer */ | 
					
						
							|  |  |  |         ASSERT(!vmp); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-14 23:56:26 +03:00
										 |  |  | 	cache_name = kzalloc(strlen(name) + 1, GFP_KERNEL); | 
					
						
							|  |  |  | 	if (cache_name == NULL) | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | 		RETURN(NULL); | 
					
						
							| 
									
										
										
										
											2008-03-14 23:56:26 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	strcpy(cache_name, name); | 
					
						
							|  |  |  |         cache = kmem_cache_create(cache_name, size, align, flags, | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |                                   kmem_cache_generic_constructor, | 
					
						
							|  |  |  |                                   kmem_cache_generic_destructor); | 
					
						
							|  |  |  | 	if (cache == NULL) | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  |                 RETURN(NULL); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /* Register shared shrinker function on initial cache create */ | 
					
						
							|  |  |  | 	spin_lock(&kmem_cache_cb_lock); | 
					
						
							|  |  |  | 	if (list_empty(&kmem_cache_cb_list)) { | 
					
						
							|  |  |  |                 kmem_cache_shrinker = set_shrinker(KMC_DEFAULT_SEEKS, | 
					
						
							|  |  |  |                                                  kmem_cache_generic_shrinker); | 
					
						
							|  |  |  |                 if (kmem_cache_shrinker == NULL) { | 
					
						
							|  |  |  |                         kmem_cache_destroy(cache); | 
					
						
							|  |  |  | 			spin_unlock(&kmem_cache_cb_lock); | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  |                         RETURN(NULL); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 	spin_unlock(&kmem_cache_cb_lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         kcc = kmem_cache_add_cache_cb(cache, constructor, destructor, | 
					
						
							|  |  |  |                                       reclaim, priv, vmp); | 
					
						
							|  |  |  |         if (kcc == NULL) { | 
					
						
							|  |  |  | 		if (shrinker_flag) /* New shrinker registered must be removed */ | 
					
						
							|  |  |  | 			remove_shrinker(kmem_cache_shrinker); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 kmem_cache_destroy(cache); | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  |                 RETURN(NULL); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  |         RETURN(cache); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2008-02-27 22:09:51 +03:00
										 |  |  | EXPORT_SYMBOL(__kmem_cache_create); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-19 02:20:30 +03:00
										 |  |  | /* Return code provided despite Solaris's void return.  There should be no
 | 
					
						
							|  |  |  |  * harm here since the Solaris versions will ignore it anyway. */ | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | __kmem_cache_destroy(kmem_cache_t *cache) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |         kmem_cache_cb_t *kcc; | 
					
						
							| 
									
										
										
										
											2008-03-14 23:56:26 +03:00
										 |  |  | 	char *name; | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	unsigned long flags; | 
					
						
							| 
									
										
										
										
											2008-03-19 02:20:30 +03:00
										 |  |  | 	int rc; | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | 	ENTRY; | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_lock_irqsave(&kmem_cache_cb_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |         kcc = kmem_cache_find_cache_cb(cache); | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_unlock_irqrestore(&kmem_cache_cb_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |         if (kcc == NULL) | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  |                 RETURN(-EINVAL); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-14 23:56:26 +03:00
										 |  |  | 	name = (char *)kmem_cache_name(cache); | 
					
						
							| 
									
										
										
										
											2008-03-19 02:20:30 +03:00
										 |  |  |         rc = kmem_cache_destroy(cache); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  |         kmem_cache_remove_cache_cb(kcc); | 
					
						
							| 
									
										
										
										
											2008-03-14 23:56:26 +03:00
										 |  |  | 	kfree(name); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Unregister generic shrinker on removal of all caches */ | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_lock_irqsave(&kmem_cache_cb_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 	if (list_empty(&kmem_cache_cb_list)) | 
					
						
							|  |  |  |                 remove_shrinker(kmem_cache_shrinker); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-16 20:37:51 +04:00
										 |  |  | 	spin_unlock_irqrestore(&kmem_cache_cb_lock, flags); | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | 	RETURN(rc); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2008-02-27 22:09:51 +03:00
										 |  |  | EXPORT_SYMBOL(__kmem_cache_destroy); | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-27 22:09:51 +03:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | __kmem_reap(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ENTRY; | 
					
						
							| 
									
										
										
										
											2008-02-27 22:09:51 +03:00
										 |  |  | 	/* Since there's no easy hook in to linux to force all the registered
 | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | 	 * shrinkers to run we just run the ones registered for this shim */ | 
					
						
							|  |  |  | 	kmem_cache_generic_shrinker(KMC_REAP_CHUNK, GFP_KERNEL); | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | 	EXIT; | 
					
						
							| 
									
										
										
										
											2008-02-26 23:36:04 +03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2008-02-27 22:09:51 +03:00
										 |  |  | EXPORT_SYMBOL(__kmem_reap); | 
					
						
							| 
									
										
										
										
											2008-03-18 07:56:43 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | kmem_init(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  |         ENTRY; | 
					
						
							| 
									
										
										
										
											2008-03-18 07:56:43 +03:00
										 |  |  | #ifdef DEBUG_KMEM
 | 
					
						
							|  |  |  | 	atomic64_set(&kmem_alloc_used, 0); | 
					
						
							|  |  |  | 	atomic64_set(&vmem_alloc_used, 0); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | 	RETURN(0); | 
					
						
							| 
									
										
										
										
											2008-03-18 07:56:43 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | kmem_fini(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | 	ENTRY; | 
					
						
							| 
									
										
										
										
											2008-03-18 07:56:43 +03:00
										 |  |  | #ifdef DEBUG_KMEM
 | 
					
						
							|  |  |  |         if (atomic64_read(&kmem_alloc_used) != 0) | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  |                 CWARN("kmem leaked %ld/%ld bytes\n", | 
					
						
							| 
									
										
										
										
											2008-03-18 07:56:43 +03:00
										 |  |  |                        atomic_read(&kmem_alloc_used), kmem_alloc_max); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (atomic64_read(&vmem_alloc_used) != 0) | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  |                 CWARN("vmem leaked %ld/%ld bytes\n", | 
					
						
							| 
									
										
										
										
											2008-03-18 07:56:43 +03:00
										 |  |  |                        atomic_read(&vmem_alloc_used), vmem_alloc_max); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-04-21 21:29:47 +04:00
										 |  |  | 	EXIT; | 
					
						
							| 
									
										
										
										
											2008-03-18 07:56:43 +03:00
										 |  |  | } |