diff options
Diffstat (limited to 'net/mac80211/rc80211_multirate_pid.h')
-rw-r--r-- | net/mac80211/rc80211_multirate_pid.h | 301 |
1 files changed, 301 insertions, 0 deletions
diff --git a/net/mac80211/rc80211_multirate_pid.h b/net/mac80211/rc80211_multirate_pid.h new file mode 100644 index 000000000000..9f6b473f8821 --- /dev/null +++ b/net/mac80211/rc80211_multirate_pid.h @@ -0,0 +1,301 @@ +/* + * Copyright 2007, Mattias Nissler <mattias.nissler@gmx.de> + * Copyright 2007, Stefano Brivio <stefano.brivio@polimi.it> + * Copyright 2009, Digi International <support@digi.com> + * + * 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. + */ + +/* + * This is a modified version of the 802.11 PID algorithm. The standard + * version uses the same rate for all the retries of the same frame. The code + * was modified to perform the first few retries at the same rate and then + * reduce the data rate for each successive retry until we either run out of retries + * or successfully transmit the frame. This seems to provide better performance + * on error prone networks. + */ + +#ifndef RC80211_MULTIRATE_PID_H +#define RC80211_MULTIRATE_PID_H + +/* Sampling period for measuring percentage of failed frames in ms. */ +#define RC80211_MULTIRATE_PID_INTERVAL 125 + +/* Exponential averaging smoothness (used for I part of PID controller) */ +#define RC80211_MULTIRATE_PID_SMOOTHING_SHIFT 3 +#define RC80211_MULTIRATE_PID_SMOOTHING (1 << RC80211_MULTIRATE_PID_SMOOTHING_SHIFT) + +/* Sharpening factor (used for D part of PID controller) */ +#define RC80211_MULTIRATE_PID_SHARPENING_FACTOR 0 +#define RC80211_MULTIRATE_PID_SHARPENING_DURATION 0 + +/* Fixed point arithmetic shifting amount. */ +#define RC80211_MULTIRATE_PID_ARITH_SHIFT 8 + +/* Fixed point arithmetic factor. */ +#define RC80211_MULTIRATE_PID_ARITH_FACTOR (1 << RC80211_MULTIRATE_PID_ARITH_SHIFT) + +/* Proportional PID component coefficient. */ +#define RC80211_MULTIRATE_PID_COEFF_P 15 +/* Integral PID component coefficient. */ +#define RC80211_MULTIRATE_PID_COEFF_I 9 +/* Derivative PID component coefficient. */ +#define RC80211_MULTIRATE_PID_COEFF_D 15 + +/* 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. + */ +#define RC80211_MULTIRATE_PID_TARGET_PF 14 + +/* Rate behaviour normalization quantity over time. */ +#define RC80211_MULTIRATE_PID_NORM_OFFSET 3 + +/* Push high rates right after loading. */ +#define RC80211_MULTIRATE_PID_FAST_START 0 + +/* Arithmetic right shift for positive and negative values for ISO C. */ +#define RC80211_MULTIRATE_PID_DO_ARITH_RIGHT_SHIFT(x, y) \ + ((x) < 0 ? -((-(x)) >> (y)) : (x) >> (y)) + +enum rc_multirate_pid_event_type { + RC80211_MULTIRATE_PID_EVENT_TYPE_TX_STATUS, + RC80211_MULTIRATE_PID_EVENT_TYPE_RATE_CHANGE, + RC80211_MULTIRATE_PID_EVENT_TYPE_TX_RATE, + RC80211_MULTIRATE_PID_EVENT_TYPE_PF_SAMPLE, +}; + +union rc_multirate_pid_event_data { + /* RC80211_MULTIRATE_PID_EVENT_TX_STATUS */ + struct { + u32 flags; + struct ieee80211_tx_info tx_status; + }; + /* RC80211_MULTIRATE_PID_EVENT_TYPE_RATE_CHANGE */ + /* RC80211_MULTIRATE_PID_EVENT_TYPE_TX_RATE */ + struct { + int index; + int rate; + }; + /* RC80211_MULTIRATE_PID_EVENT_TYPE_PF_SAMPLE */ + struct { + s32 pf_sample; + s32 prop_err; + s32 int_err; + s32 der_err; + }; +}; + +struct rc_multirate_pid_event { + /* The time when the event occured */ + unsigned long timestamp; + + /* Event ID number */ + unsigned int id; + + /* Type of event */ + enum rc_multirate_pid_event_type type; + + /* type specific data */ + union rc_multirate_pid_event_data data; +}; + +/* Size of the event ring buffer. */ +#define RC80211_MULTIRATE_PID_EVENT_RING_SIZE 32 + +struct rc_multirate_pid_event_buffer { + /* Counter that generates event IDs */ + unsigned int ev_count; + + /* Ring buffer of events */ + struct rc_multirate_pid_event + ring[RC80211_MULTIRATE_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_multirate_pid_events_file_info { + /* The event buffer we read */ + struct rc_multirate_pid_event_buffer *events; + + /* The entry we have should read next */ + unsigned int next_entry; +}; + +/** + * struct rc_multirate_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) + */ +struct rc_multirate_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; +}; + +void rate_control_multirate_pid_event_tx_status(struct + rc_multirate_pid_event_buffer + *buf, + struct ieee80211_tx_info + *stat); + +void rate_control_multirate_pid_event_rate_change(struct + rc_multirate_pid_event_buffer + *buf, int index, + int rate); + +void rate_control_multirate_pid_event_tx_rate(struct + rc_multirate_pid_event_buffer + *buf, int index, int rate); + +void rate_control_multirate_pid_event_pf_sample(struct + rc_multirate_pid_event_buffer + *buf, s32 pf_sample, + s32 prop_err, s32 int_err, + s32 der_err); + +void rate_control_multirate_pid_add_sta_debugfs(void *priv, void *priv_sta, + struct dentry *dir); + +void rate_control_multirate_pid_remove_sta_debugfs(void *priv, + void *priv_sta); + +struct rc_multirate_pid_sta_info { + unsigned long last_change; + unsigned long last_sample; + + u32 tx_num_failed; + u32 tx_num_xmit; + + int txrate_idx; + + /* Average failed frames percentage error (i.e. actual vs. target + * percentage), scaled by RC80211_MULTIRATE_PID_SMOOTHING. This value is computed + * using using an exponential weighted average technique: + * + * (RC80211_MULTIRATE_PID_SMOOTHING - 1) * err_avg_old + err + * err_avg = ------------------------------------------ + * RC80211_MULTIRATE_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 RC80211_MULTIRATE_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 * RC80211_MULTIRATE_PID_SMOOTHING + * err_avg_old_scaled = err_avg_old * RC80211_MULTIRATE_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 + * RC80211_MULTIRATE_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_multirate_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_multirate_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_multirate_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; + + /* Sharpening factor and duration. */ + unsigned int sharpen_factor; + unsigned int sharpen_duration; + + /* Normalization offset. */ + unsigned int norm_offset; + + /* Rates information. */ + struct rc_multirate_pid_rateinfo *rinfo; + + /* Index of the last used rate. */ + int oldrate; + +#ifdef CONFIG_MAC80211_DEBUGFS + /* Debugfs entries created for the parameters above. */ + struct rc_multirate_pid_debugfs_entries dentries; +#endif +}; + +#endif /* RC80211_MULTIRATE_PID_H */ |