diff options
Diffstat (limited to 'drivers/net/wireless/b43/main.c')
-rw-r--r-- | drivers/net/wireless/b43/main.c | 65 |
1 files changed, 23 insertions, 42 deletions
diff --git a/drivers/net/wireless/b43/main.c b/drivers/net/wireless/b43/main.c index 79c25d6385ce..d4a356b11636 100644 --- a/drivers/net/wireless/b43/main.c +++ b/drivers/net/wireless/b43/main.c | |||
@@ -3068,30 +3068,22 @@ static const u16 b43_qos_shm_offsets[] = { | |||
3068 | [3] = B43_QOS_BACKGROUND, | 3068 | [3] = B43_QOS_BACKGROUND, |
3069 | }; | 3069 | }; |
3070 | 3070 | ||
3071 | /* Update the QOS parameters in hardware. */ | 3071 | /* Update all QOS parameters in hardware. */ |
3072 | static void b43_qos_update(struct b43_wldev *dev) | 3072 | static void b43_qos_upload_all(struct b43_wldev *dev) |
3073 | { | 3073 | { |
3074 | struct b43_wl *wl = dev->wl; | 3074 | struct b43_wl *wl = dev->wl; |
3075 | struct b43_qos_params *params; | 3075 | struct b43_qos_params *params; |
3076 | unsigned long flags; | ||
3077 | unsigned int i; | 3076 | unsigned int i; |
3078 | 3077 | ||
3079 | BUILD_BUG_ON(ARRAY_SIZE(b43_qos_shm_offsets) != | 3078 | BUILD_BUG_ON(ARRAY_SIZE(b43_qos_shm_offsets) != |
3080 | ARRAY_SIZE(wl->qos_params)); | 3079 | ARRAY_SIZE(wl->qos_params)); |
3081 | 3080 | ||
3082 | b43_mac_suspend(dev); | 3081 | b43_mac_suspend(dev); |
3083 | spin_lock_irqsave(&wl->irq_lock, flags); | ||
3084 | |||
3085 | for (i = 0; i < ARRAY_SIZE(wl->qos_params); i++) { | 3082 | for (i = 0; i < ARRAY_SIZE(wl->qos_params); i++) { |
3086 | params = &(wl->qos_params[i]); | 3083 | params = &(wl->qos_params[i]); |
3087 | if (params->need_hw_update) { | 3084 | b43_qos_params_upload(dev, &(params->p), |
3088 | b43_qos_params_upload(dev, &(params->p), | 3085 | b43_qos_shm_offsets[i]); |
3089 | b43_qos_shm_offsets[i]); | ||
3090 | params->need_hw_update = 0; | ||
3091 | } | ||
3092 | } | 3086 | } |
3093 | |||
3094 | spin_unlock_irqrestore(&wl->irq_lock, flags); | ||
3095 | b43_mac_enable(dev); | 3087 | b43_mac_enable(dev); |
3096 | } | 3088 | } |
3097 | 3089 | ||
@@ -3136,20 +3128,14 @@ static void b43_qos_clear(struct b43_wl *wl) | |||
3136 | default: | 3128 | default: |
3137 | B43_WARN_ON(1); | 3129 | B43_WARN_ON(1); |
3138 | } | 3130 | } |
3139 | params->need_hw_update = 1; | ||
3140 | } | 3131 | } |
3141 | } | 3132 | } |
3142 | 3133 | ||
3143 | /* Initialize the core's QOS capabilities */ | 3134 | /* Initialize the core's QOS capabilities */ |
3144 | static void b43_qos_init(struct b43_wldev *dev) | 3135 | static void b43_qos_init(struct b43_wldev *dev) |
3145 | { | 3136 | { |
3146 | struct b43_wl *wl = dev->wl; | ||
3147 | unsigned int i; | ||
3148 | |||
3149 | /* Upload the current QOS parameters. */ | 3137 | /* Upload the current QOS parameters. */ |
3150 | for (i = 0; i < ARRAY_SIZE(wl->qos_params); i++) | 3138 | b43_qos_upload_all(dev); |
3151 | wl->qos_params[i].need_hw_update = 1; | ||
3152 | b43_qos_update(dev); | ||
3153 | 3139 | ||
3154 | /* Enable QOS support. */ | 3140 | /* Enable QOS support. */ |
3155 | b43_hf_write(dev, b43_hf_read(dev) | B43_HF_EDCF); | 3141 | b43_hf_write(dev, b43_hf_read(dev) | B43_HF_EDCF); |
@@ -3158,25 +3144,13 @@ static void b43_qos_init(struct b43_wldev *dev) | |||
3158 | | B43_MMIO_IFSCTL_USE_EDCF); | 3144 | | B43_MMIO_IFSCTL_USE_EDCF); |
3159 | } | 3145 | } |
3160 | 3146 | ||
3161 | static void b43_qos_update_work(struct work_struct *work) | ||
3162 | { | ||
3163 | struct b43_wl *wl = container_of(work, struct b43_wl, qos_update_work); | ||
3164 | struct b43_wldev *dev; | ||
3165 | |||
3166 | mutex_lock(&wl->mutex); | ||
3167 | dev = wl->current_dev; | ||
3168 | if (likely(dev && (b43_status(dev) >= B43_STAT_INITIALIZED))) | ||
3169 | b43_qos_update(dev); | ||
3170 | mutex_unlock(&wl->mutex); | ||
3171 | } | ||
3172 | |||
3173 | static int b43_op_conf_tx(struct ieee80211_hw *hw, u16 _queue, | 3147 | static int b43_op_conf_tx(struct ieee80211_hw *hw, u16 _queue, |
3174 | const struct ieee80211_tx_queue_params *params) | 3148 | const struct ieee80211_tx_queue_params *params) |
3175 | { | 3149 | { |
3176 | struct b43_wl *wl = hw_to_b43_wl(hw); | 3150 | struct b43_wl *wl = hw_to_b43_wl(hw); |
3177 | unsigned long flags; | 3151 | struct b43_wldev *dev; |
3178 | unsigned int queue = (unsigned int)_queue; | 3152 | unsigned int queue = (unsigned int)_queue; |
3179 | struct b43_qos_params *p; | 3153 | int err = -ENODEV; |
3180 | 3154 | ||
3181 | if (queue >= ARRAY_SIZE(wl->qos_params)) { | 3155 | if (queue >= ARRAY_SIZE(wl->qos_params)) { |
3182 | /* Queue not available or don't support setting | 3156 | /* Queue not available or don't support setting |
@@ -3184,16 +3158,25 @@ static int b43_op_conf_tx(struct ieee80211_hw *hw, u16 _queue, | |||
3184 | * confuse mac80211. */ | 3158 | * confuse mac80211. */ |
3185 | return 0; | 3159 | return 0; |
3186 | } | 3160 | } |
3161 | BUILD_BUG_ON(ARRAY_SIZE(b43_qos_shm_offsets) != | ||
3162 | ARRAY_SIZE(wl->qos_params)); | ||
3187 | 3163 | ||
3188 | spin_lock_irqsave(&wl->irq_lock, flags); | 3164 | mutex_lock(&wl->mutex); |
3189 | p = &(wl->qos_params[queue]); | 3165 | dev = wl->current_dev; |
3190 | memcpy(&(p->p), params, sizeof(p->p)); | 3166 | if (unlikely(!dev || (b43_status(dev) < B43_STAT_INITIALIZED))) |
3191 | p->need_hw_update = 1; | 3167 | goto out_unlock; |
3192 | spin_unlock_irqrestore(&wl->irq_lock, flags); | ||
3193 | 3168 | ||
3194 | queue_work(hw->workqueue, &wl->qos_update_work); | 3169 | memcpy(&(wl->qos_params[queue].p), params, sizeof(*params)); |
3170 | b43_mac_suspend(dev); | ||
3171 | b43_qos_params_upload(dev, &(wl->qos_params[queue].p), | ||
3172 | b43_qos_shm_offsets[queue]); | ||
3173 | b43_mac_enable(dev); | ||
3174 | err = 0; | ||
3195 | 3175 | ||
3196 | return 0; | 3176 | out_unlock: |
3177 | mutex_unlock(&wl->mutex); | ||
3178 | |||
3179 | return err; | ||
3197 | } | 3180 | } |
3198 | 3181 | ||
3199 | static int b43_op_get_tx_stats(struct ieee80211_hw *hw, | 3182 | static int b43_op_get_tx_stats(struct ieee80211_hw *hw, |
@@ -4220,7 +4203,6 @@ static void b43_op_stop(struct ieee80211_hw *hw) | |||
4220 | struct b43_wldev *dev = wl->current_dev; | 4203 | struct b43_wldev *dev = wl->current_dev; |
4221 | 4204 | ||
4222 | b43_rfkill_exit(dev); | 4205 | b43_rfkill_exit(dev); |
4223 | cancel_work_sync(&(wl->qos_update_work)); | ||
4224 | cancel_work_sync(&(wl->beacon_update_trigger)); | 4206 | cancel_work_sync(&(wl->beacon_update_trigger)); |
4225 | 4207 | ||
4226 | mutex_lock(&wl->mutex); | 4208 | mutex_lock(&wl->mutex); |
@@ -4619,7 +4601,6 @@ static int b43_wireless_init(struct ssb_device *dev) | |||
4619 | spin_lock_init(&wl->shm_lock); | 4601 | spin_lock_init(&wl->shm_lock); |
4620 | mutex_init(&wl->mutex); | 4602 | mutex_init(&wl->mutex); |
4621 | INIT_LIST_HEAD(&wl->devlist); | 4603 | INIT_LIST_HEAD(&wl->devlist); |
4622 | INIT_WORK(&wl->qos_update_work, b43_qos_update_work); | ||
4623 | INIT_WORK(&wl->beacon_update_trigger, b43_beacon_update_trigger_work); | 4604 | INIT_WORK(&wl->beacon_update_trigger, b43_beacon_update_trigger_work); |
4624 | INIT_WORK(&wl->txpower_adjust_work, b43_phy_txpower_adjust_work); | 4605 | INIT_WORK(&wl->txpower_adjust_work, b43_phy_txpower_adjust_work); |
4625 | 4606 | ||