| /* |
| * Airtime policy configuration |
| * Copyright (c) 2018-2019, Toke Høiland-Jørgensen <toke@toke.dk> |
| * |
| * This software may be distributed under the terms of the BSD license. |
| * See README for more details. |
| */ |
| |
| #include "utils/includes.h" |
| |
| #include "utils/common.h" |
| #include "utils/eloop.h" |
| #include "hostapd.h" |
| #include "ap_drv_ops.h" |
| #include "sta_info.h" |
| #include "airtime_policy.h" |
| |
| /* Idea: |
| * Two modes of airtime enforcement: |
| * 1. Static weights: specify weights per MAC address with a per-BSS default |
| * 2. Per-BSS limits: Dynamically calculate weights of backlogged stations to |
| * enforce relative total shares between BSSes. |
| * |
| * - Periodic per-station callback to update queue status. |
| * |
| * Copy accounting_sta_update_stats() to get TXQ info and airtime weights and |
| * keep them updated in sta_info. |
| * |
| * - Separate periodic per-bss (or per-iface?) callback to update weights. |
| * |
| * Just need to loop through all interfaces, count sum the active stations (or |
| * should the per-STA callback just adjust that for the BSS?) and calculate new |
| * weights. |
| */ |
| |
| static int get_airtime_policy_update_timeout(struct hostapd_iface *iface, |
| unsigned int *sec, |
| unsigned int *usec) |
| { |
| unsigned int update_int = iface->conf->airtime_update_interval; |
| |
| if (!update_int) { |
| wpa_printf(MSG_ERROR, |
| "Airtime policy: Invalid airtime policy update interval %u", |
| update_int); |
| return -1; |
| } |
| |
| *sec = update_int / 1000; |
| *usec = (update_int % 1000) * 1000; |
| |
| return 0; |
| } |
| |
| |
| static void set_new_backlog_time(struct hostapd_data *hapd, |
| struct sta_info *sta, |
| struct os_reltime *now) |
| { |
| sta->backlogged_until = *now; |
| sta->backlogged_until.usec += hapd->iconf->airtime_update_interval * |
| AIRTIME_BACKLOG_EXPIRY_FACTOR; |
| while (sta->backlogged_until.usec >= 1000000) { |
| sta->backlogged_until.sec++; |
| sta->backlogged_until.usec -= 1000000; |
| } |
| } |
| |
| |
| static void count_backlogged_sta(struct hostapd_data *hapd) |
| { |
| struct sta_info *sta; |
| struct hostap_sta_driver_data data = {}; |
| unsigned int num_backlogged = 0; |
| struct os_reltime now; |
| |
| os_get_reltime(&now); |
| |
| for (sta = hapd->sta_list; sta; sta = sta->next) { |
| if (hostapd_drv_read_sta_data(hapd, &data, sta->addr)) |
| continue; |
| #ifdef CONFIG_TESTING_OPTIONS |
| if (hapd->force_backlog_bytes) |
| data.backlog_bytes = 1; |
| #endif /* CONFIG_TESTING_OPTIONS */ |
| |
| if (data.backlog_bytes > 0) |
| set_new_backlog_time(hapd, sta, &now); |
| if (os_reltime_before(&now, &sta->backlogged_until)) |
| num_backlogged++; |
| } |
| hapd->num_backlogged_sta = num_backlogged; |
| } |
| |
| |
| static int sta_set_airtime_weight(struct hostapd_data *hapd, |
| struct sta_info *sta, |
| unsigned int weight) |
| { |
| int ret = 0; |
| |
| if (weight != sta->airtime_weight && |
| (ret = hostapd_sta_set_airtime_weight(hapd, sta->addr, weight))) |
| return ret; |
| |
| sta->airtime_weight = weight; |
| return ret; |
| } |
| |
| |
| static void set_sta_weights(struct hostapd_data *hapd, unsigned int weight) |
| { |
| struct sta_info *sta; |
| |
| for (sta = hapd->sta_list; sta; sta = sta->next) |
| sta_set_airtime_weight(hapd, sta, weight); |
| } |
| |
| |
| static unsigned int get_airtime_quantum(unsigned int max_wt) |
| { |
| unsigned int quantum = AIRTIME_QUANTUM_TARGET / max_wt; |
| |
| if (quantum < AIRTIME_QUANTUM_MIN) |
| quantum = AIRTIME_QUANTUM_MIN; |
| else if (quantum > AIRTIME_QUANTUM_MAX) |
| quantum = AIRTIME_QUANTUM_MAX; |
| |
| return quantum; |
| } |
| |
| |
| static void update_airtime_weights(void *eloop_data, void *user_data) |
| { |
| struct hostapd_iface *iface = eloop_data; |
| struct hostapd_data *bss; |
| unsigned int sec, usec; |
| unsigned int num_sta_min = 0, num_sta_prod = 1, num_sta_sum = 0, |
| wt_sum = 0; |
| unsigned int quantum; |
| bool all_div_min = true; |
| bool apply_limit = iface->conf->airtime_mode == AIRTIME_MODE_DYNAMIC; |
| int wt, num_bss = 0, max_wt = 0; |
| size_t i; |
| |
| for (i = 0; i < iface->num_bss; i++) { |
| bss = iface->bss[i]; |
| if (!bss->started || !bss->conf->airtime_weight) |
| continue; |
| |
| count_backlogged_sta(bss); |
| if (!bss->num_backlogged_sta) |
| continue; |
| |
| if (!num_sta_min || bss->num_backlogged_sta < num_sta_min) |
| num_sta_min = bss->num_backlogged_sta; |
| |
| num_sta_prod *= bss->num_backlogged_sta; |
| num_sta_sum += bss->num_backlogged_sta; |
| wt_sum += bss->conf->airtime_weight; |
| num_bss++; |
| } |
| |
| if (num_sta_min) { |
| for (i = 0; i < iface->num_bss; i++) { |
| bss = iface->bss[i]; |
| if (!bss->started || !bss->conf->airtime_weight) |
| continue; |
| |
| /* Check if we can divide all sta numbers by the |
| * smallest number to keep weights as small as possible. |
| * This is a lazy way to avoid having to factor |
| * integers. */ |
| if (bss->num_backlogged_sta && |
| bss->num_backlogged_sta % num_sta_min > 0) |
| all_div_min = false; |
| |
| /* If we're in LIMIT mode, we only apply the weight |
| * scaling when the BSS(es) marked as limited would a |
| * larger share than the relative BSS weights indicates |
| * it should. */ |
| if (!apply_limit && bss->conf->airtime_limit) { |
| if (bss->num_backlogged_sta * wt_sum > |
| bss->conf->airtime_weight * num_sta_sum) |
| apply_limit = true; |
| } |
| } |
| if (all_div_min) |
| num_sta_prod /= num_sta_min; |
| } |
| |
| for (i = 0; i < iface->num_bss; i++) { |
| bss = iface->bss[i]; |
| if (!bss->started || !bss->conf->airtime_weight) |
| continue; |
| |
| /* We only set the calculated weight if the BSS has active |
| * stations and there are other active interfaces as well - |
| * otherwise we just set a unit weight. This ensures that |
| * the weights are set reasonably when stations transition from |
| * inactive to active. */ |
| if (apply_limit && bss->num_backlogged_sta && num_bss > 1) |
| wt = bss->conf->airtime_weight * num_sta_prod / |
| bss->num_backlogged_sta; |
| else |
| wt = 1; |
| |
| bss->airtime_weight = wt; |
| if (wt > max_wt) |
| max_wt = wt; |
| } |
| |
| quantum = get_airtime_quantum(max_wt); |
| |
| for (i = 0; i < iface->num_bss; i++) { |
| bss = iface->bss[i]; |
| if (!bss->started || !bss->conf->airtime_weight) |
| continue; |
| set_sta_weights(bss, bss->airtime_weight * quantum); |
| } |
| |
| if (get_airtime_policy_update_timeout(iface, &sec, &usec) < 0) |
| return; |
| |
| eloop_register_timeout(sec, usec, update_airtime_weights, iface, |
| NULL); |
| } |
| |
| |
| static int get_weight_for_sta(struct hostapd_data *hapd, const u8 *sta) |
| { |
| struct airtime_sta_weight *wt; |
| |
| wt = hapd->conf->airtime_weight_list; |
| while (wt && os_memcmp(wt->addr, sta, ETH_ALEN) != 0) |
| wt = wt->next; |
| |
| return wt ? wt->weight : hapd->conf->airtime_weight; |
| } |
| |
| |
| int airtime_policy_new_sta(struct hostapd_data *hapd, struct sta_info *sta) |
| { |
| unsigned int weight; |
| |
| if (hapd->iconf->airtime_mode == AIRTIME_MODE_STATIC) { |
| weight = get_weight_for_sta(hapd, sta->addr); |
| if (weight) |
| return sta_set_airtime_weight(hapd, sta, weight); |
| } |
| return 0; |
| } |
| |
| |
| int airtime_policy_update_init(struct hostapd_iface *iface) |
| { |
| unsigned int sec, usec; |
| |
| if (iface->conf->airtime_mode < AIRTIME_MODE_DYNAMIC) |
| return 0; |
| |
| if (get_airtime_policy_update_timeout(iface, &sec, &usec) < 0) |
| return -1; |
| |
| eloop_register_timeout(sec, usec, update_airtime_weights, iface, NULL); |
| return 0; |
| } |
| |
| |
| void airtime_policy_update_deinit(struct hostapd_iface *iface) |
| { |
| eloop_cancel_timeout(update_airtime_weights, iface, NULL); |
| } |