| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Floating proportions | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Copyright (C) 2007 Red Hat, Inc., Peter Zijlstra <pzijlstr@redhat.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Description: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The floating proportion is a time derivative with an exponentially decaying | 
					
						
							|  |  |  |  * history: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   p_{j} = \Sum_{i=0} (dx_{j}/dt_{-i}) / 2^(1+i) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Where j is an element from {prop_local}, x_{j} is j's number of events, | 
					
						
							|  |  |  |  * and i the time period over which the differential is taken. So d/dt_{-i} is | 
					
						
							|  |  |  |  * the differential over the i-th last period. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The decaying history gives smooth transitions. The time differential carries | 
					
						
							|  |  |  |  * the notion of speed. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The denominator is 2^(1+i) because we want the series to be normalised, ie. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   \Sum_{i=0} 1/2^(1+i) = 1 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Further more, if we measure time (t) in the same events as x; so that: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   t = \Sum_{j} x_{j} | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * we get that: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   \Sum_{j} p_{j} = 1 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Writing this in an iterative fashion we get (dropping the 'd's): | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   if (++x_{j}, ++t > period) | 
					
						
							|  |  |  |  *     t /= 2; | 
					
						
							|  |  |  |  *     for_each (j) | 
					
						
							|  |  |  |  *       x_{j} /= 2; | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * so that: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   p_{j} = x_{j} / t; | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * We optimize away the '/= 2' for the global time delta by noting that: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   if (++t > period) t /= 2: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Can be approximated by: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   period/2 + (++t % period/2) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * [ Furthermore, when we choose period to be 2^n it can be written in terms of | 
					
						
							|  |  |  |  *   binary operations and wraparound artefacts disappear. ] | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Also note that this yields a natural counter of the elapsed periods: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   c = t / (period/2) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * [ Its monotonic increasing property can be applied to mitigate the wrap- | 
					
						
							|  |  |  |  *   around issue. ] | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This allows us to do away with the loop over all prop_locals on each period | 
					
						
							|  |  |  |  * expiration. By remembering the period count under which it was last accessed | 
					
						
							|  |  |  |  * as c_{j}, we can obtain the number of 'missed' cycles from: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   c - c_{j} | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * We can then lazily catch up to the global period count every time we are | 
					
						
							|  |  |  |  * going to use x_{j}, by doing: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   x_{j} /= 2^(c - c_{j}), c_{j} = c | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/proportions.h>
 | 
					
						
							|  |  |  | #include <linux/rcupdate.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int prop_descriptor_init(struct prop_descriptor *pd, int shift) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (shift > PROP_MAX_SHIFT) | 
					
						
							|  |  |  | 		shift = PROP_MAX_SHIFT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pd->index = 0; | 
					
						
							|  |  |  | 	pd->pg[0].shift = shift; | 
					
						
							|  |  |  | 	mutex_init(&pd->mutex); | 
					
						
							| 
									
										
										
										
											2008-12-26 15:08:55 +01:00
										 |  |  | 	err = percpu_counter_init(&pd->pg[0].events, 0); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | 	if (err) | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-26 15:08:55 +01:00
										 |  |  | 	err = percpu_counter_init(&pd->pg[1].events, 0); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | 	if (err) | 
					
						
							|  |  |  | 		percpu_counter_destroy(&pd->pg[0].events); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * We have two copies, and flip between them to make it seem like an atomic | 
					
						
							|  |  |  |  * update. The update is not really atomic wrt the events counter, but | 
					
						
							|  |  |  |  * it is internally consistent with the bit layout depending on shift. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * We copy the events count, move the bits around and flip the index. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void prop_change_shift(struct prop_descriptor *pd, int shift) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int index; | 
					
						
							|  |  |  | 	int offset; | 
					
						
							|  |  |  | 	u64 events; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (shift > PROP_MAX_SHIFT) | 
					
						
							|  |  |  | 		shift = PROP_MAX_SHIFT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mutex_lock(&pd->mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	index = pd->index ^ 1; | 
					
						
							|  |  |  | 	offset = pd->pg[pd->index].shift - shift; | 
					
						
							|  |  |  | 	if (!offset) | 
					
						
							|  |  |  | 		goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pd->pg[index].shift = shift; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	local_irq_save(flags); | 
					
						
							|  |  |  | 	events = percpu_counter_sum(&pd->pg[pd->index].events); | 
					
						
							|  |  |  | 	if (offset < 0) | 
					
						
							|  |  |  | 		events <<= -offset; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		events >>= offset; | 
					
						
							|  |  |  | 	percpu_counter_set(&pd->pg[index].events, events); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * ensure the new pg is fully written before the switch | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	smp_wmb(); | 
					
						
							|  |  |  | 	pd->index = index; | 
					
						
							|  |  |  | 	local_irq_restore(flags); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	synchronize_rcu(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	mutex_unlock(&pd->mutex); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * wrap the access to the data in an rcu_read_lock() section; | 
					
						
							|  |  |  |  * this is used to track the active references. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct prop_global *prop_get_global(struct prop_descriptor *pd) | 
					
						
							| 
									
										
										
										
											2009-01-06 14:40:50 -08:00
										 |  |  | __acquires(RCU) | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | { | 
					
						
							|  |  |  | 	int index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rcu_read_lock(); | 
					
						
							|  |  |  | 	index = pd->index; | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * match the wmb from vcd_flip() | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	smp_rmb(); | 
					
						
							|  |  |  | 	return &pd->pg[index]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void prop_put_global(struct prop_descriptor *pd, struct prop_global *pg) | 
					
						
							| 
									
										
										
										
											2009-01-06 14:40:50 -08:00
										 |  |  | __releases(RCU) | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | { | 
					
						
							|  |  |  | 	rcu_read_unlock(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | prop_adjust_shift(int *pl_shift, unsigned long *pl_period, int new_shift) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int offset = *pl_shift - new_shift; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!offset) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (offset < 0) | 
					
						
							|  |  |  | 		*pl_period <<= -offset; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		*pl_period >>= offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	*pl_shift = new_shift; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * PERCPU | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-22 14:03:29 -08:00
										 |  |  | #define PROP_BATCH (8*(1+ilog2(nr_cpu_ids)))
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | int prop_local_init_percpu(struct prop_local_percpu *pl) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-25 16:43:30 +02:00
										 |  |  | 	raw_spin_lock_init(&pl->lock); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | 	pl->shift = 0; | 
					
						
							|  |  |  | 	pl->period = 0; | 
					
						
							| 
									
										
										
										
											2008-12-26 15:08:55 +01:00
										 |  |  | 	return percpu_counter_init(&pl->events, 0); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void prop_local_destroy_percpu(struct prop_local_percpu *pl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	percpu_counter_destroy(&pl->events); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Catch up with missed period expirations. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   until (c_{j} == c) | 
					
						
							|  |  |  |  *     x_{j} -= x_{j}/2; | 
					
						
							|  |  |  |  *     c_{j}++; | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static | 
					
						
							|  |  |  | void prop_norm_percpu(struct prop_global *pg, struct prop_local_percpu *pl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned long period = 1UL << (pg->shift - 1); | 
					
						
							|  |  |  | 	unsigned long period_mask = ~(period - 1); | 
					
						
							|  |  |  | 	unsigned long global_period; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	global_period = percpu_counter_read(&pg->events); | 
					
						
							|  |  |  | 	global_period &= period_mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Fast path - check if the local and global period count still match | 
					
						
							|  |  |  | 	 * outside of the lock. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (pl->period == global_period) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-25 16:43:30 +02:00
										 |  |  | 	raw_spin_lock_irqsave(&pl->lock, flags); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | 	prop_adjust_shift(&pl->shift, &pl->period, pg->shift); | 
					
						
							| 
									
										
										
										
											2007-12-22 14:03:29 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * For each missed period, we half the local counter. | 
					
						
							|  |  |  | 	 * basically: | 
					
						
							|  |  |  | 	 *   pl->events >> (global_period - pl->period); | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2007-12-22 14:03:29 -08:00
										 |  |  | 	period = (global_period - pl->period) >> (pg->shift - 1); | 
					
						
							|  |  |  | 	if (period < BITS_PER_LONG) { | 
					
						
							|  |  |  | 		s64 val = percpu_counter_read(&pl->events); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (val < (nr_cpu_ids * PROP_BATCH)) | 
					
						
							|  |  |  | 			val = percpu_counter_sum(&pl->events); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		__percpu_counter_add(&pl->events, -val + (val >> period), | 
					
						
							|  |  |  | 					PROP_BATCH); | 
					
						
							|  |  |  | 	} else | 
					
						
							|  |  |  | 		percpu_counter_set(&pl->events, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | 	pl->period = global_period; | 
					
						
							| 
									
										
										
										
											2009-07-25 16:43:30 +02:00
										 |  |  | 	raw_spin_unlock_irqrestore(&pl->lock, flags); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  *   ++x_{j}, ++t | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void __prop_inc_percpu(struct prop_descriptor *pd, struct prop_local_percpu *pl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct prop_global *pg = prop_get_global(pd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	prop_norm_percpu(pg, pl); | 
					
						
							| 
									
										
										
										
											2007-12-22 14:03:29 -08:00
										 |  |  | 	__percpu_counter_add(&pl->events, 1, PROP_BATCH); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | 	percpu_counter_add(&pg->events, 1); | 
					
						
							|  |  |  | 	prop_put_global(pd, pg); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-30 00:54:36 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * identical to __prop_inc_percpu, except that it limits this pl's fraction to | 
					
						
							|  |  |  |  * @frac/PROP_FRAC_BASE by ignoring events when this limit has been exceeded. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void __prop_inc_percpu_max(struct prop_descriptor *pd, | 
					
						
							|  |  |  | 			   struct prop_local_percpu *pl, long frac) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct prop_global *pg = prop_get_global(pd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	prop_norm_percpu(pg, pl); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (unlikely(frac != PROP_FRAC_BASE)) { | 
					
						
							|  |  |  | 		unsigned long period_2 = 1UL << (pg->shift - 1); | 
					
						
							|  |  |  | 		unsigned long counter_mask = period_2 - 1; | 
					
						
							|  |  |  | 		unsigned long global_count; | 
					
						
							|  |  |  | 		long numerator, denominator; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		numerator = percpu_counter_read_positive(&pl->events); | 
					
						
							|  |  |  | 		global_count = percpu_counter_read(&pg->events); | 
					
						
							|  |  |  | 		denominator = period_2 + (global_count & counter_mask); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (numerator > ((denominator * frac) >> PROP_FRAC_SHIFT)) | 
					
						
							|  |  |  | 			goto out_put; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	percpu_counter_add(&pl->events, 1); | 
					
						
							|  |  |  | 	percpu_counter_add(&pg->events, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | out_put: | 
					
						
							|  |  |  | 	prop_put_global(pd, pg); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Obtain a fraction of this proportion | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   p_{j} = x_{j} / (period/2 + t % period/2) | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void prop_fraction_percpu(struct prop_descriptor *pd, | 
					
						
							|  |  |  | 		struct prop_local_percpu *pl, | 
					
						
							|  |  |  | 		long *numerator, long *denominator) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct prop_global *pg = prop_get_global(pd); | 
					
						
							|  |  |  | 	unsigned long period_2 = 1UL << (pg->shift - 1); | 
					
						
							|  |  |  | 	unsigned long counter_mask = period_2 - 1; | 
					
						
							|  |  |  | 	unsigned long global_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	prop_norm_percpu(pg, pl); | 
					
						
							|  |  |  | 	*numerator = percpu_counter_read_positive(&pl->events); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	global_count = percpu_counter_read(&pg->events); | 
					
						
							|  |  |  | 	*denominator = period_2 + (global_count & counter_mask); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	prop_put_global(pd, pg); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * SINGLE | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int prop_local_init_single(struct prop_local_single *pl) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-25 16:43:30 +02:00
										 |  |  | 	raw_spin_lock_init(&pl->lock); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | 	pl->shift = 0; | 
					
						
							|  |  |  | 	pl->period = 0; | 
					
						
							|  |  |  | 	pl->events = 0; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void prop_local_destroy_single(struct prop_local_single *pl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Catch up with missed period expirations. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static | 
					
						
							|  |  |  | void prop_norm_single(struct prop_global *pg, struct prop_local_single *pl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned long period = 1UL << (pg->shift - 1); | 
					
						
							|  |  |  | 	unsigned long period_mask = ~(period - 1); | 
					
						
							|  |  |  | 	unsigned long global_period; | 
					
						
							|  |  |  | 	unsigned long flags; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	global_period = percpu_counter_read(&pg->events); | 
					
						
							|  |  |  | 	global_period &= period_mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Fast path - check if the local and global period count still match | 
					
						
							|  |  |  | 	 * outside of the lock. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (pl->period == global_period) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-25 16:43:30 +02:00
										 |  |  | 	raw_spin_lock_irqsave(&pl->lock, flags); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | 	prop_adjust_shift(&pl->shift, &pl->period, pg->shift); | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * For each missed period, we half the local counter. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	period = (global_period - pl->period) >> (pg->shift - 1); | 
					
						
							|  |  |  | 	if (likely(period < BITS_PER_LONG)) | 
					
						
							|  |  |  | 		pl->events >>= period; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		pl->events = 0; | 
					
						
							|  |  |  | 	pl->period = global_period; | 
					
						
							| 
									
										
										
										
											2009-07-25 16:43:30 +02:00
										 |  |  | 	raw_spin_unlock_irqrestore(&pl->lock, flags); | 
					
						
							| 
									
										
										
										
											2007-10-16 23:25:49 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  *   ++x_{j}, ++t | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void __prop_inc_single(struct prop_descriptor *pd, struct prop_local_single *pl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct prop_global *pg = prop_get_global(pd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	prop_norm_single(pg, pl); | 
					
						
							|  |  |  | 	pl->events++; | 
					
						
							|  |  |  | 	percpu_counter_add(&pg->events, 1); | 
					
						
							|  |  |  | 	prop_put_global(pd, pg); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Obtain a fraction of this proportion | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   p_{j} = x_{j} / (period/2 + t % period/2) | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void prop_fraction_single(struct prop_descriptor *pd, | 
					
						
							|  |  |  | 	       	struct prop_local_single *pl, | 
					
						
							|  |  |  | 		long *numerator, long *denominator) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct prop_global *pg = prop_get_global(pd); | 
					
						
							|  |  |  | 	unsigned long period_2 = 1UL << (pg->shift - 1); | 
					
						
							|  |  |  | 	unsigned long counter_mask = period_2 - 1; | 
					
						
							|  |  |  | 	unsigned long global_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	prop_norm_single(pg, pl); | 
					
						
							|  |  |  | 	*numerator = pl->events; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	global_count = percpu_counter_read(&pg->events); | 
					
						
							|  |  |  | 	*denominator = period_2 + (global_count & counter_mask); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	prop_put_global(pd, pg); | 
					
						
							|  |  |  | } |