diff options
Diffstat (limited to 'lib/objpool.c')
| -rw-r--r-- | lib/objpool.c | 100 | 
1 files changed, 0 insertions, 100 deletions
| diff --git a/lib/objpool.c b/lib/objpool.c index cfdc02420884..f696308fc026 100644 --- a/lib/objpool.c +++ b/lib/objpool.c @@ -152,106 +152,6 @@ int objpool_init(struct objpool_head *pool, int nr_objs, int object_size,  }  EXPORT_SYMBOL_GPL(objpool_init); -/* adding object to slot, abort if the slot was already full */ -static inline int -objpool_try_add_slot(void *obj, struct objpool_head *pool, int cpu) -{ -	struct objpool_slot *slot = pool->cpu_slots[cpu]; -	uint32_t head, tail; - -	/* loading tail and head as a local snapshot, tail first */ -	tail = READ_ONCE(slot->tail); - -	do { -		head = READ_ONCE(slot->head); -		/* fault caught: something must be wrong */ -		WARN_ON_ONCE(tail - head > pool->nr_objs); -	} while (!try_cmpxchg_acquire(&slot->tail, &tail, tail + 1)); - -	/* now the tail position is reserved for the given obj */ -	WRITE_ONCE(slot->entries[tail & slot->mask], obj); -	/* update sequence to make this obj available for pop() */ -	smp_store_release(&slot->last, tail + 1); - -	return 0; -} - -/* reclaim an object to object pool */ -int objpool_push(void *obj, struct objpool_head *pool) -{ -	unsigned long flags; -	int rc; - -	/* disable local irq to avoid preemption & interruption */ -	raw_local_irq_save(flags); -	rc = objpool_try_add_slot(obj, pool, raw_smp_processor_id()); -	raw_local_irq_restore(flags); - -	return rc; -} -EXPORT_SYMBOL_GPL(objpool_push); - -/* try to retrieve object from slot */ -static inline void *objpool_try_get_slot(struct objpool_head *pool, int cpu) -{ -	struct objpool_slot *slot = pool->cpu_slots[cpu]; -	/* load head snapshot, other cpus may change it */ -	uint32_t head = smp_load_acquire(&slot->head); - -	while (head != READ_ONCE(slot->last)) { -		void *obj; - -		/* -		 * data visibility of 'last' and 'head' could be out of -		 * order since memory updating of 'last' and 'head' are -		 * performed in push() and pop() independently -		 * -		 * before any retrieving attempts, pop() must guarantee -		 * 'last' is behind 'head', that is to say, there must -		 * be available objects in slot, which could be ensured -		 * by condition 'last != head && last - head <= nr_objs' -		 * that is equivalent to 'last - head - 1 < nr_objs' as -		 * 'last' and 'head' are both unsigned int32 -		 */ -		if (READ_ONCE(slot->last) - head - 1 >= pool->nr_objs) { -			head = READ_ONCE(slot->head); -			continue; -		} - -		/* obj must be retrieved before moving forward head */ -		obj = READ_ONCE(slot->entries[head & slot->mask]); - -		/* move head forward to mark it's consumption */ -		if (try_cmpxchg_release(&slot->head, &head, head + 1)) -			return obj; -	} - -	return NULL; -} - -/* allocate an object from object pool */ -void *objpool_pop(struct objpool_head *pool) -{ -	void *obj = NULL; -	unsigned long flags; -	int i, cpu; - -	/* disable local irq to avoid preemption & interruption */ -	raw_local_irq_save(flags); - -	cpu = raw_smp_processor_id(); -	for (i = 0; i < num_possible_cpus(); i++) { -		obj = objpool_try_get_slot(pool, cpu); -		if (obj) -			break; -		cpu = cpumask_next_wrap(cpu, cpu_possible_mask, -1, 1); -	} -	raw_local_irq_restore(flags); - -	return obj; -} -EXPORT_SYMBOL_GPL(objpool_pop); -  /* release whole objpool forcely */  void objpool_free(struct objpool_head *pool)  { | 
