blob: 2ef70e4701f69af603c3c55ad3bb8e1f7a542e60 [file] [log] [blame]
Forest Bond92b96792009-06-13 07:38:31 -04001/*
2 * Copyright (c) 1996, 2003 VIA Networking Technologies, Inc.
3 * All rights reserved.
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 *
20 * File: int.c
21 *
22 * Purpose: Handle USB interrupt endpoint
23 *
24 * Author: Jerry Chen
25 *
26 * Date: Apr. 2, 2004
27 *
28 * Functions:
29 *
30 * Revision History:
31 * 04-02-2004 Jerry Chen: Initial release
32 *
33 */
34
Forest Bond92b96792009-06-13 07:38:31 -040035#include "int.h"
Forest Bond92b96792009-06-13 07:38:31 -040036#include "mac.h"
Forest Bond92b96792009-06-13 07:38:31 -040037#include "power.h"
Forest Bond92b96792009-06-13 07:38:31 -040038#include "usbpipe.h"
Forest Bond92b96792009-06-13 07:38:31 -040039
Malcolm Priestley71d764a2014-07-05 19:24:27 +010040static const u8 fallback_rate0[5][5] = {
41 {RATE_18M, RATE_18M, RATE_12M, RATE_12M, RATE_12M},
42 {RATE_24M, RATE_24M, RATE_18M, RATE_12M, RATE_12M},
43 {RATE_36M, RATE_36M, RATE_24M, RATE_18M, RATE_18M},
44 {RATE_48M, RATE_48M, RATE_36M, RATE_24M, RATE_24M},
45 {RATE_54M, RATE_54M, RATE_48M, RATE_36M, RATE_36M}
46};
47
48static const u8 fallback_rate1[5][5] = {
49 {RATE_18M, RATE_18M, RATE_12M, RATE_6M, RATE_6M},
50 {RATE_24M, RATE_24M, RATE_18M, RATE_6M, RATE_6M},
51 {RATE_36M, RATE_36M, RATE_24M, RATE_12M, RATE_12M},
52 {RATE_48M, RATE_48M, RATE_24M, RATE_12M, RATE_12M},
53 {RATE_54M, RATE_54M, RATE_36M, RATE_18M, RATE_18M}
54};
55
Malcolm Priestley62001932014-07-15 19:54:35 +010056void vnt_int_start_interrupt(struct vnt_private *priv)
Forest Bond92b96792009-06-13 07:38:31 -040057{
Malcolm Priestley0b596b22014-05-24 14:36:57 +010058 unsigned long flags;
Malcolm Priestley526fed32014-07-15 19:54:32 +010059 int status;
Forest Bond92b96792009-06-13 07:38:31 -040060
Malcolm Priestley50730602014-07-15 19:54:33 +010061 dev_dbg(&priv->usb->dev, "---->Interrupt Polling Thread\n");
Forest Bond92b96792009-06-13 07:38:31 -040062
Malcolm Priestley526fed32014-07-15 19:54:32 +010063 spin_lock_irqsave(&priv->lock, flags);
Malcolm Priestley749627f2014-02-17 21:24:33 +000064
Malcolm Priestleyb0bc5722014-07-15 19:54:37 +010065 status = vnt_start_interrupt_urb(priv);
Malcolm Priestley749627f2014-02-17 21:24:33 +000066
Malcolm Priestley526fed32014-07-15 19:54:32 +010067 spin_unlock_irqrestore(&priv->lock, flags);
Daniel Kenji Toyamaff8041b2010-03-30 00:28:19 +110068}
Forest Bond92b96792009-06-13 07:38:31 -040069
Malcolm Priestley71d764a2014-07-05 19:24:27 +010070static int vnt_int_report_rate(struct vnt_private *priv, u8 pkt_no, u8 tsr)
71{
72 struct vnt_usb_send_context *context;
73 struct ieee80211_tx_info *info;
74 struct ieee80211_rate *rate;
75 u8 tx_retry = (tsr & 0xf0) >> 4;
76 s8 idx;
77
Malcolm Priestley03b7e352014-07-19 12:30:04 +010078 if (pkt_no >= priv->num_tx_context)
Malcolm Priestley71d764a2014-07-05 19:24:27 +010079 return -EINVAL;
80
Malcolm Priestleyf7e4a8f2014-07-18 06:36:16 +010081 context = priv->tx_context[pkt_no];
Malcolm Priestley71d764a2014-07-05 19:24:27 +010082
83 if (!context->skb)
84 return -EINVAL;
85
86 info = IEEE80211_SKB_CB(context->skb);
87 idx = info->control.rates[0].idx;
88
89 if (context->fb_option && !(tsr & (TSR_TMO | TSR_RETRYTMO))) {
90 u8 tx_rate;
91 u8 retry = tx_retry;
92
93 rate = ieee80211_get_tx_rate(priv->hw, info);
94 tx_rate = rate->hw_value - RATE_18M;
95
96 if (retry > 4)
97 retry = 4;
98
99 if (context->fb_option == AUTO_FB_0)
100 tx_rate = fallback_rate0[tx_rate][retry];
101 else if (context->fb_option == AUTO_FB_1)
102 tx_rate = fallback_rate1[tx_rate][retry];
103
104 if (info->band == IEEE80211_BAND_5GHZ)
105 idx = tx_rate - RATE_6M;
106 else
107 idx = tx_rate;
108 }
109
110 ieee80211_tx_info_clear_status(info);
111
112 info->status.rates[0].count = tx_retry;
113
114 if (!(tsr & (TSR_TMO | TSR_RETRYTMO))) {
115 info->status.rates[0].idx = idx;
116 info->flags |= IEEE80211_TX_STAT_ACK;
117 }
118
119 ieee80211_tx_status_irqsafe(priv->hw, context->skb);
120
121 context->in_use = false;
122
123 return 0;
124}
125
Malcolm Priestleye360d2b2014-07-15 19:54:34 +0100126void vnt_int_process_data(struct vnt_private *priv)
Forest Bond92b96792009-06-13 07:38:31 -0400127{
Malcolm Priestley210098a2014-02-16 19:10:18 +0000128 struct vnt_interrupt_data *int_data;
Malcolm Priestley17863842014-06-29 20:39:54 +0100129 struct ieee80211_low_level_stats *low_stats = &priv->low_stats;
Forest Bond92b96792009-06-13 07:38:31 -0400130
Malcolm Priestley50730602014-07-15 19:54:33 +0100131 dev_dbg(&priv->usb->dev, "---->s_nsInterruptProcessData\n");
Forest Bond92b96792009-06-13 07:38:31 -0400132
Malcolm Priestleyf764e002014-02-19 18:39:09 +0000133 int_data = (struct vnt_interrupt_data *)priv->int_buf.data_buf;
Malcolm Priestley210098a2014-02-16 19:10:18 +0000134
Malcolm Priestley71d764a2014-07-05 19:24:27 +0100135 if (int_data->tsr0 & TSR_VALID)
136 vnt_int_report_rate(priv, int_data->pkt0, int_data->tsr0);
Malcolm Priestley210098a2014-02-16 19:10:18 +0000137
Malcolm Priestley71d764a2014-07-05 19:24:27 +0100138 if (int_data->tsr1 & TSR_VALID)
139 vnt_int_report_rate(priv, int_data->pkt1, int_data->tsr1);
Malcolm Priestley210098a2014-02-16 19:10:18 +0000140
Malcolm Priestley71d764a2014-07-05 19:24:27 +0100141 if (int_data->tsr2 & TSR_VALID)
142 vnt_int_report_rate(priv, int_data->pkt2, int_data->tsr2);
Malcolm Priestley210098a2014-02-16 19:10:18 +0000143
Malcolm Priestley71d764a2014-07-05 19:24:27 +0100144 if (int_data->tsr3 & TSR_VALID)
145 vnt_int_report_rate(priv, int_data->pkt3, int_data->tsr3);
Malcolm Priestley210098a2014-02-16 19:10:18 +0000146
147 if (int_data->isr0 != 0) {
Malcolm Priestleyc2246422014-06-25 21:14:32 +0100148 if (int_data->isr0 & ISR_BNTX &&
149 priv->op_mode == NL80211_IFTYPE_AP)
Malcolm Priestley57981a62014-07-13 10:42:50 +0100150 vnt_schedule_command(priv, WLAN_CMD_BECON_SEND);
Malcolm Priestley210098a2014-02-16 19:10:18 +0000151
152 if (int_data->isr0 & ISR_TBTT) {
Malcolm Priestleyc2246422014-06-25 21:14:32 +0100153 if (priv->hw->conf.flags & IEEE80211_CONF_PS)
Malcolm Priestley57981a62014-07-13 10:42:50 +0100154 vnt_schedule_command(priv,
155 WLAN_CMD_TBTT_WAKEUP);
Daniel Kenji Toyamaff8041b2010-03-30 00:28:19 +1100156 }
Malcolm Priestley113f0b92014-07-19 12:30:13 +0100157 priv->current_tsf = le64_to_cpu(int_data->tsf);
Malcolm Priestley17863842014-06-29 20:39:54 +0100158
159 low_stats->dot11RTSSuccessCount += int_data->rts_success;
160 low_stats->dot11RTSFailureCount += int_data->rts_fail;
161 low_stats->dot11ACKFailureCount += int_data->ack_fail;
162 low_stats->dot11FCSErrorCount += int_data->fcs_err;
Daniel Kenji Toyamaff8041b2010-03-30 00:28:19 +1100163 }
Malcolm Priestley210098a2014-02-16 19:10:18 +0000164
Malcolm Priestleyf764e002014-02-19 18:39:09 +0000165 priv->int_buf.in_use = false;
Forest Bond92b96792009-06-13 07:38:31 -0400166}