| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright 2007, Mattias Nissler <mattias.nissler@gmx.de> | 
					
						
							| 
									
										
										
										
											2007-12-23 04:40:32 +01:00
										 |  |  |  * Copyright 2007, Stefano Brivio <stefano.brivio@polimi.it> | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU General Public License version 2 as | 
					
						
							|  |  |  |  * published by the Free Software Foundation. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef RC80211_PID_H
 | 
					
						
							|  |  |  | #define RC80211_PID_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-23 17:49:00 +01:00
										 |  |  | /* Sampling period for measuring percentage of failed frames in ms. */ | 
					
						
							|  |  |  | #define RC_PID_INTERVAL			125
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Exponential averaging smoothness (used for I part of PID controller) */ | 
					
						
							| 
									
										
										
										
											2007-12-23 04:44:56 +01:00
										 |  |  | #define RC_PID_SMOOTHING_SHIFT		3
 | 
					
						
							|  |  |  | #define RC_PID_SMOOTHING		(1 << RC_PID_SMOOTHING_SHIFT)
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Sharpening factor (used for D part of PID controller) */ | 
					
						
							| 
									
										
										
										
											2007-12-23 04:44:56 +01:00
										 |  |  | #define RC_PID_SHARPENING_FACTOR	0
 | 
					
						
							|  |  |  | #define RC_PID_SHARPENING_DURATION	0
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Fixed point arithmetic shifting amount. */ | 
					
						
							| 
									
										
										
										
											2007-12-23 04:44:56 +01:00
										 |  |  | #define RC_PID_ARITH_SHIFT		8
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Proportional PID component coefficient. */ | 
					
						
							| 
									
										
										
										
											2007-12-23 04:44:56 +01:00
										 |  |  | #define RC_PID_COEFF_P			15
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | /* Integral PID component coefficient. */ | 
					
						
							| 
									
										
										
										
											2007-12-23 04:44:56 +01:00
										 |  |  | #define RC_PID_COEFF_I			9
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | /* Derivative PID component coefficient. */ | 
					
						
							| 
									
										
										
										
											2007-12-23 04:44:56 +01:00
										 |  |  | #define RC_PID_COEFF_D			15
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Target failed frames rate for the PID controller. NB: This effectively gives
 | 
					
						
							|  |  |  |  * maximum failed frames percentage we're willing to accept. If the wireless | 
					
						
							|  |  |  |  * link quality is good, the controller will fail to adjust failed frames | 
					
						
							|  |  |  |  * percentage to the target. This is intentional. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2007-12-23 04:44:56 +01:00
										 |  |  | #define RC_PID_TARGET_PF		14
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Rate behaviour normalization quantity over time. */ | 
					
						
							| 
									
										
										
										
											2007-12-23 04:44:56 +01:00
										 |  |  | #define RC_PID_NORM_OFFSET		3
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Push high rates right after loading. */ | 
					
						
							| 
									
										
										
										
											2007-12-23 04:44:56 +01:00
										 |  |  | #define RC_PID_FAST_START		0
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Arithmetic right shift for positive and negative values for ISO C. */ | 
					
						
							|  |  |  | #define RC_PID_DO_ARITH_RIGHT_SHIFT(x, y) \
 | 
					
						
							| 
									
										
										
										
											2008-10-07 12:04:32 +02:00
										 |  |  | 	((x) < 0 ? -((-(x)) >> (y)) : (x) >> (y)) | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | enum rc_pid_event_type { | 
					
						
							|  |  |  | 	RC_PID_EVENT_TYPE_TX_STATUS, | 
					
						
							|  |  |  | 	RC_PID_EVENT_TYPE_RATE_CHANGE, | 
					
						
							|  |  |  | 	RC_PID_EVENT_TYPE_TX_RATE, | 
					
						
							|  |  |  | 	RC_PID_EVENT_TYPE_PF_SAMPLE, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | union rc_pid_event_data { | 
					
						
							|  |  |  | 	/* RC_PID_EVENT_TX_STATUS */ | 
					
						
							|  |  |  | 	struct { | 
					
						
							| 
									
										
										
										
											2008-10-21 12:40:02 +02:00
										 |  |  | 		u32 flags; | 
					
						
							| 
									
										
										
										
											2008-05-15 12:55:29 +02:00
										 |  |  | 		struct ieee80211_tx_info tx_status; | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 	}; | 
					
						
							|  |  |  | 	/* RC_PID_EVENT_TYPE_RATE_CHANGE */ | 
					
						
							|  |  |  | 	/* RC_PID_EVENT_TYPE_TX_RATE */ | 
					
						
							|  |  |  | 	struct { | 
					
						
							|  |  |  | 		int index; | 
					
						
							|  |  |  | 		int rate; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	/* RC_PID_EVENT_TYPE_PF_SAMPLE */ | 
					
						
							|  |  |  | 	struct { | 
					
						
							|  |  |  | 		s32 pf_sample; | 
					
						
							|  |  |  | 		s32 prop_err; | 
					
						
							|  |  |  | 		s32 int_err; | 
					
						
							|  |  |  | 		s32 der_err; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct rc_pid_event { | 
					
						
							| 
									
										
										
										
											2011-03-30 22:57:33 -03:00
										 |  |  | 	/* The time when the event occurred */ | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 	unsigned long timestamp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Event ID number */ | 
					
						
							|  |  |  | 	unsigned int id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Type of event */ | 
					
						
							|  |  |  | 	enum rc_pid_event_type type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* type specific data */ | 
					
						
							|  |  |  | 	union rc_pid_event_data data; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Size of the event ring buffer. */ | 
					
						
							|  |  |  | #define RC_PID_EVENT_RING_SIZE 32
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct rc_pid_event_buffer { | 
					
						
							|  |  |  | 	/* Counter that generates event IDs */ | 
					
						
							|  |  |  | 	unsigned int ev_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Ring buffer of events */ | 
					
						
							|  |  |  | 	struct rc_pid_event ring[RC_PID_EVENT_RING_SIZE]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Index to the entry in events_buf to be reused */ | 
					
						
							|  |  |  | 	unsigned int next_entry; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Lock that guards against concurrent access to this buffer struct */ | 
					
						
							|  |  |  | 	spinlock_t lock; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Wait queue for poll/select and blocking I/O */ | 
					
						
							|  |  |  | 	wait_queue_head_t waitqueue; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct rc_pid_events_file_info { | 
					
						
							|  |  |  | 	/* The event buffer we read */ | 
					
						
							|  |  |  | 	struct rc_pid_event_buffer *events; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* The entry we have should read next */ | 
					
						
							|  |  |  | 	unsigned int next_entry; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-23 04:40:32 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * struct rc_pid_debugfs_entries - tunable parameters | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Algorithm parameters, tunable via debugfs. | 
					
						
							|  |  |  |  * @target: target percentage for failed frames | 
					
						
							|  |  |  |  * @sampling_period: error sampling interval in milliseconds | 
					
						
							|  |  |  |  * @coeff_p: absolute value of the proportional coefficient | 
					
						
							|  |  |  |  * @coeff_i: absolute value of the integral coefficient | 
					
						
							|  |  |  |  * @coeff_d: absolute value of the derivative coefficient | 
					
						
							|  |  |  |  * @smoothing_shift: absolute value of the integral smoothing factor (i.e. | 
					
						
							|  |  |  |  *	amount of smoothing introduced by the exponential moving average) | 
					
						
							|  |  |  |  * @sharpen_factor: absolute value of the derivative sharpening factor (i.e. | 
					
						
							|  |  |  |  *	amount of emphasis given to the derivative term after low activity | 
					
						
							|  |  |  |  *	events) | 
					
						
							|  |  |  |  * @sharpen_duration: duration of the sharpening effect after the detected low | 
					
						
							|  |  |  |  *	activity event, relative to sampling_period | 
					
						
							|  |  |  |  * @norm_offset: amount of normalization periodically performed on the learnt | 
					
						
							|  |  |  |  *	rate behaviour values (lower means we should trust more what we learnt | 
					
						
							|  |  |  |  *	about behaviour of rates, higher means we should trust more the natural | 
					
						
							|  |  |  |  *	ordering of rates) | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2007-12-20 13:27:26 +01:00
										 |  |  | struct rc_pid_debugfs_entries { | 
					
						
							|  |  |  | 	struct dentry *target; | 
					
						
							|  |  |  | 	struct dentry *sampling_period; | 
					
						
							|  |  |  | 	struct dentry *coeff_p; | 
					
						
							|  |  |  | 	struct dentry *coeff_i; | 
					
						
							|  |  |  | 	struct dentry *coeff_d; | 
					
						
							|  |  |  | 	struct dentry *smoothing_shift; | 
					
						
							|  |  |  | 	struct dentry *sharpen_factor; | 
					
						
							|  |  |  | 	struct dentry *sharpen_duration; | 
					
						
							|  |  |  | 	struct dentry *norm_offset; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | void rate_control_pid_event_tx_status(struct rc_pid_event_buffer *buf, | 
					
						
							| 
									
										
										
										
											2008-05-15 12:55:29 +02:00
										 |  |  | 				      struct ieee80211_tx_info *stat); | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void rate_control_pid_event_rate_change(struct rc_pid_event_buffer *buf, | 
					
						
							|  |  |  | 					       int index, int rate); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void rate_control_pid_event_tx_rate(struct rc_pid_event_buffer *buf, | 
					
						
							|  |  |  | 					   int index, int rate); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void rate_control_pid_event_pf_sample(struct rc_pid_event_buffer *buf, | 
					
						
							|  |  |  | 					     s32 pf_sample, s32 prop_err, | 
					
						
							|  |  |  | 					     s32 int_err, s32 der_err); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void rate_control_pid_add_sta_debugfs(void *priv, void *priv_sta, | 
					
						
							|  |  |  | 					     struct dentry *dir); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void rate_control_pid_remove_sta_debugfs(void *priv, void *priv_sta); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct rc_pid_sta_info { | 
					
						
							|  |  |  | 	unsigned long last_change; | 
					
						
							|  |  |  | 	unsigned long last_sample; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	u32 tx_num_failed; | 
					
						
							|  |  |  | 	u32 tx_num_xmit; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-09-11 03:04:36 +02:00
										 |  |  | 	int txrate_idx; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 	/* Average failed frames percentage error (i.e. actual vs. target
 | 
					
						
							|  |  |  | 	 * percentage), scaled by RC_PID_SMOOTHING. This value is computed | 
					
						
							|  |  |  | 	 * using using an exponential weighted average technique: | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 *           (RC_PID_SMOOTHING - 1) * err_avg_old + err | 
					
						
							|  |  |  | 	 * err_avg = ------------------------------------------ | 
					
						
							|  |  |  | 	 *                       RC_PID_SMOOTHING | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * where err_avg is the new approximation, err_avg_old the previous one | 
					
						
							|  |  |  | 	 * and err is the error w.r.t. to the current failed frames percentage | 
					
						
							|  |  |  | 	 * sample. Note that the bigger RC_PID_SMOOTHING the more weight is | 
					
						
							|  |  |  | 	 * given to the previous estimate, resulting in smoother behavior (i.e. | 
					
						
							|  |  |  | 	 * corresponding to a longer integration window). | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * For computation, we actually don't use the above formula, but this | 
					
						
							|  |  |  | 	 * one: | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * err_avg_scaled = err_avg_old_scaled - err_avg_old + err | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * where: | 
					
						
							|  |  |  | 	 * 	err_avg_scaled = err * RC_PID_SMOOTHING | 
					
						
							|  |  |  | 	 * 	err_avg_old_scaled = err_avg_old * RC_PID_SMOOTHING | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * This avoids floating point numbers and the per_failed_old value can | 
					
						
							|  |  |  | 	 * easily be obtained by shifting per_failed_old_scaled right by | 
					
						
							|  |  |  | 	 * RC_PID_SMOOTHING_SHIFT. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	s32 err_avg_sc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Last framed failes percentage sample. */ | 
					
						
							|  |  |  | 	u32 last_pf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Sharpening needed. */ | 
					
						
							|  |  |  | 	u8 sharp_cnt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_MAC80211_DEBUGFS
 | 
					
						
							|  |  |  | 	/* Event buffer */ | 
					
						
							|  |  |  | 	struct rc_pid_event_buffer events; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Events debugfs file entry */ | 
					
						
							|  |  |  | 	struct dentry *events_entry; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Algorithm parameters. We keep them on a per-algorithm approach, so they can
 | 
					
						
							|  |  |  |  * be tuned individually for each interface. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct rc_pid_rateinfo { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Map sorted rates to rates in ieee80211_hw_mode. */ | 
					
						
							|  |  |  | 	int index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Map rates in ieee80211_hw_mode to sorted rates. */ | 
					
						
							|  |  |  | 	int rev_index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Did we do any measurement on this rate? */ | 
					
						
							|  |  |  | 	bool valid; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Comparison with the lowest rate. */ | 
					
						
							|  |  |  | 	int diff; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct rc_pid_info { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* The failed frames percentage target. */ | 
					
						
							|  |  |  | 	unsigned int target; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Rate at which failed frames percentage is sampled in 0.001s. */ | 
					
						
							|  |  |  | 	unsigned int sampling_period; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* P, I and D coefficients. */ | 
					
						
							|  |  |  | 	int coeff_p; | 
					
						
							|  |  |  | 	int coeff_i; | 
					
						
							|  |  |  | 	int coeff_d; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Exponential averaging shift. */ | 
					
						
							|  |  |  | 	unsigned int smoothing_shift; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-20 13:27:26 +01:00
										 |  |  | 	/* Sharpening factor and duration. */ | 
					
						
							|  |  |  | 	unsigned int sharpen_factor; | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | 	unsigned int sharpen_duration; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Normalization offset. */ | 
					
						
							|  |  |  | 	unsigned int norm_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Rates information. */ | 
					
						
							|  |  |  | 	struct rc_pid_rateinfo *rinfo; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Index of the last used rate. */ | 
					
						
							|  |  |  | 	int oldrate; | 
					
						
							| 
									
										
										
										
											2007-12-20 13:27:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef CONFIG_MAC80211_DEBUGFS
 | 
					
						
							|  |  |  | 	/* Debugfs entries created for the parameters above. */ | 
					
						
							|  |  |  | 	struct rc_pid_debugfs_entries dentries; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2007-12-19 01:27:18 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* RC80211_PID_H */
 |