2 * Copyright (c) 1996, 2003 VIA Networking Technologies, Inc.
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.
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.
18 * Purpose: Handle USB interrupt endpoint
27 * 04-02-2004 Jerry Chen: Initial release
36 static const u8 fallback_rate0
[5][5] = {
37 {RATE_18M
, RATE_18M
, RATE_12M
, RATE_12M
, RATE_12M
},
38 {RATE_24M
, RATE_24M
, RATE_18M
, RATE_12M
, RATE_12M
},
39 {RATE_36M
, RATE_36M
, RATE_24M
, RATE_18M
, RATE_18M
},
40 {RATE_48M
, RATE_48M
, RATE_36M
, RATE_24M
, RATE_24M
},
41 {RATE_54M
, RATE_54M
, RATE_48M
, RATE_36M
, RATE_36M
}
44 static const u8 fallback_rate1
[5][5] = {
45 {RATE_18M
, RATE_18M
, RATE_12M
, RATE_6M
, RATE_6M
},
46 {RATE_24M
, RATE_24M
, RATE_18M
, RATE_6M
, RATE_6M
},
47 {RATE_36M
, RATE_36M
, RATE_24M
, RATE_12M
, RATE_12M
},
48 {RATE_48M
, RATE_48M
, RATE_24M
, RATE_12M
, RATE_12M
},
49 {RATE_54M
, RATE_54M
, RATE_36M
, RATE_18M
, RATE_18M
}
52 void vnt_int_start_interrupt(struct vnt_private
*priv
)
57 dev_dbg(&priv
->usb
->dev
, "---->Interrupt Polling Thread\n");
59 spin_lock_irqsave(&priv
->lock
, flags
);
61 status
= vnt_start_interrupt_urb(priv
);
63 spin_unlock_irqrestore(&priv
->lock
, flags
);
66 static int vnt_int_report_rate(struct vnt_private
*priv
, u8 pkt_no
, u8 tsr
)
68 struct vnt_usb_send_context
*context
;
69 struct ieee80211_tx_info
*info
;
70 struct ieee80211_rate
*rate
;
71 u8 tx_retry
= (tsr
& 0xf0) >> 4;
74 if (pkt_no
>= priv
->num_tx_context
)
77 context
= priv
->tx_context
[pkt_no
];
82 info
= IEEE80211_SKB_CB(context
->skb
);
83 idx
= info
->control
.rates
[0].idx
;
85 if (context
->fb_option
&& !(tsr
& (TSR_TMO
| TSR_RETRYTMO
))) {
89 rate
= ieee80211_get_tx_rate(priv
->hw
, info
);
90 tx_rate
= rate
->hw_value
- RATE_18M
;
95 if (context
->fb_option
== AUTO_FB_0
)
96 tx_rate
= fallback_rate0
[tx_rate
][retry
];
97 else if (context
->fb_option
== AUTO_FB_1
)
98 tx_rate
= fallback_rate1
[tx_rate
][retry
];
100 if (info
->band
== NL80211_BAND_5GHZ
)
101 idx
= tx_rate
- RATE_6M
;
106 ieee80211_tx_info_clear_status(info
);
108 info
->status
.rates
[0].count
= tx_retry
;
110 if (!(tsr
& (TSR_TMO
| TSR_RETRYTMO
))) {
111 info
->status
.rates
[0].idx
= idx
;
112 info
->flags
|= IEEE80211_TX_STAT_ACK
;
115 ieee80211_tx_status_irqsafe(priv
->hw
, context
->skb
);
117 context
->in_use
= false;
122 void vnt_int_process_data(struct vnt_private
*priv
)
124 struct vnt_interrupt_data
*int_data
;
125 struct ieee80211_low_level_stats
*low_stats
= &priv
->low_stats
;
127 dev_dbg(&priv
->usb
->dev
, "---->s_nsInterruptProcessData\n");
129 int_data
= (struct vnt_interrupt_data
*)priv
->int_buf
.data_buf
;
131 if (int_data
->tsr0
& TSR_VALID
)
132 vnt_int_report_rate(priv
, int_data
->pkt0
, int_data
->tsr0
);
134 if (int_data
->tsr1
& TSR_VALID
)
135 vnt_int_report_rate(priv
, int_data
->pkt1
, int_data
->tsr1
);
137 if (int_data
->tsr2
& TSR_VALID
)
138 vnt_int_report_rate(priv
, int_data
->pkt2
, int_data
->tsr2
);
140 if (int_data
->tsr3
& TSR_VALID
)
141 vnt_int_report_rate(priv
, int_data
->pkt3
, int_data
->tsr3
);
143 if (int_data
->isr0
!= 0) {
144 if (int_data
->isr0
& ISR_BNTX
&&
145 priv
->op_mode
== NL80211_IFTYPE_AP
)
146 vnt_schedule_command(priv
, WLAN_CMD_BECON_SEND
);
148 if (int_data
->isr0
& ISR_TBTT
&&
149 priv
->hw
->conf
.flags
& IEEE80211_CONF_PS
) {
150 if (!priv
->wake_up_count
)
151 priv
->wake_up_count
=
152 priv
->hw
->conf
.listen_interval
;
154 --priv
->wake_up_count
;
156 /* Turn on wake up to listen next beacon */
157 if (priv
->wake_up_count
== 1)
158 vnt_schedule_command(priv
,
159 WLAN_CMD_TBTT_WAKEUP
);
161 priv
->current_tsf
= le64_to_cpu(int_data
->tsf
);
163 low_stats
->dot11RTSSuccessCount
+= int_data
->rts_success
;
164 low_stats
->dot11RTSFailureCount
+= int_data
->rts_fail
;
165 low_stats
->dot11ACKFailureCount
+= int_data
->ack_fail
;
166 low_stats
->dot11FCSErrorCount
+= int_data
->fcs_err
;
169 priv
->int_buf
.in_use
= false;