xmit_linux.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535
  1. /******************************************************************************
  2. *
  3. * Copyright(c) 2007 - 2012 Realtek Corporation. All rights reserved.
  4. *
  5. * This program is free software; you can redistribute it and/or modify it
  6. * under the terms of version 2 of the GNU General Public License as
  7. * published by the Free Software Foundation.
  8. *
  9. * This program is distributed in the hope that it will be useful, but WITHOUT
  10. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  11. * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  12. * more details.
  13. *
  14. * You should have received a copy of the GNU General Public License along with
  15. * this program; if not, write to the Free Software Foundation, Inc.,
  16. * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA
  17. *
  18. *
  19. ******************************************************************************/
  20. #define _XMIT_OSDEP_C_
  21. #include <drv_types.h>
  22. #define DBG_DUMP_OS_QUEUE_CTL 0
  23. uint rtw_remainder_len(struct pkt_file *pfile)
  24. {
  25. return pfile->buf_len - ((SIZE_PTR)(pfile->cur_addr) - (SIZE_PTR)(pfile->buf_start));
  26. }
  27. void _rtw_open_pktfile(_pkt *pktptr, struct pkt_file *pfile)
  28. {
  29. pfile->pkt = pktptr;
  30. pfile->cur_addr = pfile->buf_start = pktptr->data;
  31. pfile->pkt_len = pfile->buf_len = pktptr->len;
  32. pfile->cur_buffer = pfile->buf_start ;
  33. }
  34. uint _rtw_pktfile_read(struct pkt_file *pfile, u8 *rmem, uint rlen)
  35. {
  36. uint len = 0;
  37. len = rtw_remainder_len(pfile);
  38. len = (rlen > len) ? len : rlen;
  39. if (rmem)
  40. skb_copy_bits(pfile->pkt, pfile->buf_len - pfile->pkt_len, rmem, len);
  41. pfile->cur_addr += len;
  42. pfile->pkt_len -= len;
  43. return len;
  44. }
  45. sint rtw_endofpktfile(struct pkt_file *pfile)
  46. {
  47. if (pfile->pkt_len == 0) {
  48. return _TRUE;
  49. }
  50. return _FALSE;
  51. }
  52. void rtw_set_tx_chksum_offload(_pkt *pkt, struct pkt_attrib *pattrib)
  53. {
  54. #ifdef CONFIG_TCP_CSUM_OFFLOAD_TX
  55. struct sk_buff *skb = (struct sk_buff *)pkt;
  56. pattrib->hw_tcp_csum = 0;
  57. if (skb->ip_summed == CHECKSUM_PARTIAL) {
  58. if (skb_shinfo(skb)->nr_frags == 0) {
  59. const struct iphdr *ip = ip_hdr(skb);
  60. if (ip->protocol == IPPROTO_TCP) {
  61. /* TCP checksum offload by HW */
  62. RTW_INFO("CHECKSUM_PARTIAL TCP\n");
  63. pattrib->hw_tcp_csum = 1;
  64. /* skb_checksum_help(skb); */
  65. } else if (ip->protocol == IPPROTO_UDP) {
  66. /* RTW_INFO("CHECKSUM_PARTIAL UDP\n"); */
  67. #if 1
  68. skb_checksum_help(skb);
  69. #else
  70. /* Set UDP checksum = 0 to skip checksum check */
  71. struct udphdr *udp = skb_transport_header(skb);
  72. udp->check = 0;
  73. #endif
  74. } else {
  75. RTW_INFO("%s-%d TCP CSUM offload Error!!\n", __FUNCTION__, __LINE__);
  76. WARN_ON(1); /* we need a WARN() */
  77. }
  78. } else { /* IP fragmentation case */
  79. RTW_INFO("%s-%d nr_frags != 0, using skb_checksum_help(skb);!!\n", __FUNCTION__, __LINE__);
  80. skb_checksum_help(skb);
  81. }
  82. }
  83. #endif
  84. }
  85. int rtw_os_xmit_resource_alloc(_adapter *padapter, struct xmit_buf *pxmitbuf, u32 alloc_sz, u8 flag)
  86. {
  87. if (alloc_sz > 0) {
  88. #ifdef CONFIG_USE_USB_BUFFER_ALLOC_TX
  89. struct dvobj_priv *pdvobjpriv = adapter_to_dvobj(padapter);
  90. struct usb_device *pusbd = pdvobjpriv->pusbdev;
  91. pxmitbuf->pallocated_buf = rtw_usb_buffer_alloc(pusbd, (size_t)alloc_sz, &pxmitbuf->dma_transfer_addr);
  92. pxmitbuf->pbuf = pxmitbuf->pallocated_buf;
  93. if (pxmitbuf->pallocated_buf == NULL)
  94. return _FAIL;
  95. #else /* CONFIG_USE_USB_BUFFER_ALLOC_TX */
  96. pxmitbuf->pallocated_buf = rtw_zmalloc(alloc_sz);
  97. if (pxmitbuf->pallocated_buf == NULL)
  98. return _FAIL;
  99. pxmitbuf->pbuf = (u8 *)N_BYTE_ALIGMENT((SIZE_PTR)(pxmitbuf->pallocated_buf), XMITBUF_ALIGN_SZ);
  100. #endif /* CONFIG_USE_USB_BUFFER_ALLOC_TX */
  101. }
  102. if (flag) {
  103. #ifdef CONFIG_USB_HCI
  104. int i;
  105. for (i = 0; i < 8; i++) {
  106. pxmitbuf->pxmit_urb[i] = usb_alloc_urb(0, GFP_KERNEL);
  107. if (pxmitbuf->pxmit_urb[i] == NULL) {
  108. RTW_INFO("pxmitbuf->pxmit_urb[i]==NULL");
  109. return _FAIL;
  110. }
  111. }
  112. #endif
  113. }
  114. return _SUCCESS;
  115. }
  116. void rtw_os_xmit_resource_free(_adapter *padapter, struct xmit_buf *pxmitbuf, u32 free_sz, u8 flag)
  117. {
  118. if (flag) {
  119. #ifdef CONFIG_USB_HCI
  120. int i;
  121. for (i = 0; i < 8; i++) {
  122. if (pxmitbuf->pxmit_urb[i]) {
  123. /* usb_kill_urb(pxmitbuf->pxmit_urb[i]); */
  124. usb_free_urb(pxmitbuf->pxmit_urb[i]);
  125. }
  126. }
  127. #endif
  128. }
  129. if (free_sz > 0) {
  130. #ifdef CONFIG_USE_USB_BUFFER_ALLOC_TX
  131. struct dvobj_priv *pdvobjpriv = adapter_to_dvobj(padapter);
  132. struct usb_device *pusbd = pdvobjpriv->pusbdev;
  133. rtw_usb_buffer_free(pusbd, (size_t)free_sz, pxmitbuf->pallocated_buf, pxmitbuf->dma_transfer_addr);
  134. pxmitbuf->pallocated_buf = NULL;
  135. pxmitbuf->dma_transfer_addr = 0;
  136. #else /* CONFIG_USE_USB_BUFFER_ALLOC_TX */
  137. if (pxmitbuf->pallocated_buf)
  138. rtw_mfree(pxmitbuf->pallocated_buf, free_sz);
  139. #endif /* CONFIG_USE_USB_BUFFER_ALLOC_TX */
  140. }
  141. }
  142. void dump_os_queue(void *sel, _adapter *padapter)
  143. {
  144. struct net_device *ndev = padapter->pnetdev;
  145. #if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
  146. int i;
  147. for (i = 0; i < 4; i++) {
  148. RTW_PRINT_SEL(sel, "os_queue[%d]:%s\n"
  149. , i, __netif_subqueue_stopped(ndev, i) ? "stopped" : "waked");
  150. }
  151. #else
  152. RTW_PRINT_SEL(sel, "os_queue:%s\n"
  153. , netif_queue_stopped(ndev) ? "stopped" : "waked");
  154. #endif
  155. }
  156. #define WMM_XMIT_THRESHOLD (NR_XMITFRAME*2/5)
  157. static inline bool rtw_os_need_wake_queue(_adapter *padapter, u16 qidx)
  158. {
  159. #if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
  160. struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
  161. if (padapter->registrypriv.wifi_spec) {
  162. if (pxmitpriv->hwxmits[qidx].accnt < WMM_XMIT_THRESHOLD)
  163. return _TRUE;
  164. } else {
  165. #ifdef CONFIG_MCC_MODE
  166. if (MCC_EN(padapter)) {
  167. if (rtw_hal_check_mcc_status(padapter, MCC_STATUS_DOING_MCC)
  168. && MCC_STOP(padapter))
  169. return _FALSE;
  170. }
  171. #endif /* CONFIG_MCC_MODE */
  172. return _TRUE;
  173. }
  174. return _FALSE;
  175. #else
  176. #ifdef CONFIG_MCC_MODE
  177. if (MCC_EN(padapter)) {
  178. if (rtw_hal_check_mcc_status(padapter, MCC_STATUS_DOING_MCC)
  179. && MCC_STOP(padapter))
  180. return _FALSE;
  181. }
  182. #endif /* CONFIG_MCC_MODE */
  183. return _TRUE;
  184. #endif
  185. }
  186. static inline bool rtw_os_need_stop_queue(_adapter *padapter, u16 qidx)
  187. {
  188. struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
  189. #if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
  190. if (padapter->registrypriv.wifi_spec) {
  191. /* No free space for Tx, tx_worker is too slow */
  192. if (pxmitpriv->hwxmits[qidx].accnt > WMM_XMIT_THRESHOLD)
  193. return _TRUE;
  194. } else {
  195. if (pxmitpriv->free_xmitframe_cnt <= 4)
  196. return _TRUE;
  197. }
  198. #else
  199. if (pxmitpriv->free_xmitframe_cnt <= 4)
  200. return _TRUE;
  201. #endif
  202. return _FALSE;
  203. }
  204. void rtw_os_pkt_complete(_adapter *padapter, _pkt *pkt)
  205. {
  206. #if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
  207. u16 qidx;
  208. qidx = skb_get_queue_mapping(pkt);
  209. if (rtw_os_need_wake_queue(padapter, qidx)) {
  210. if (DBG_DUMP_OS_QUEUE_CTL)
  211. RTW_INFO(FUNC_ADPT_FMT": netif_wake_subqueue[%d]\n", FUNC_ADPT_ARG(padapter), qidx);
  212. netif_wake_subqueue(padapter->pnetdev, qidx);
  213. }
  214. #else
  215. if (rtw_os_need_wake_queue(padapter, 0)) {
  216. if (DBG_DUMP_OS_QUEUE_CTL)
  217. RTW_INFO(FUNC_ADPT_FMT": netif_wake_queue\n", FUNC_ADPT_ARG(padapter));
  218. netif_wake_queue(padapter->pnetdev);
  219. }
  220. #endif
  221. rtw_skb_free(pkt);
  222. }
  223. void rtw_os_xmit_complete(_adapter *padapter, struct xmit_frame *pxframe)
  224. {
  225. if (pxframe->pkt)
  226. rtw_os_pkt_complete(padapter, pxframe->pkt);
  227. pxframe->pkt = NULL;
  228. }
  229. void rtw_os_xmit_schedule(_adapter *padapter)
  230. {
  231. #if defined(CONFIG_SDIO_HCI) || defined(CONFIG_GSPI_HCI)
  232. _adapter *pri_adapter = GET_PRIMARY_ADAPTER(padapter);
  233. if (!padapter)
  234. return;
  235. if (_rtw_queue_empty(&padapter->xmitpriv.pending_xmitbuf_queue) == _FALSE)
  236. _rtw_up_sema(&pri_adapter->xmitpriv.xmit_sema);
  237. #else
  238. _irqL irqL;
  239. struct xmit_priv *pxmitpriv;
  240. if (!padapter)
  241. return;
  242. pxmitpriv = &padapter->xmitpriv;
  243. _enter_critical_bh(&pxmitpriv->lock, &irqL);
  244. if (rtw_txframes_pending(padapter))
  245. tasklet_hi_schedule(&pxmitpriv->xmit_tasklet);
  246. _exit_critical_bh(&pxmitpriv->lock, &irqL);
  247. #if defined(CONFIG_PCI_HCI) && defined(CONFIG_XMIT_THREAD_MODE)
  248. if (_rtw_queue_empty(&padapter->xmitpriv.pending_xmitbuf_queue) == _FALSE)
  249. _rtw_up_sema(&padapter->xmitpriv.xmit_sema);
  250. #endif
  251. #endif
  252. }
  253. static bool rtw_check_xmit_resource(_adapter *padapter, _pkt *pkt)
  254. {
  255. bool busy = _FALSE;
  256. struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
  257. #if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
  258. u16 qidx;
  259. qidx = skb_get_queue_mapping(pkt);
  260. if (rtw_os_need_stop_queue(padapter, qidx)) {
  261. if (DBG_DUMP_OS_QUEUE_CTL)
  262. RTW_INFO(FUNC_ADPT_FMT": netif_stop_subqueue[%d]\n", FUNC_ADPT_ARG(padapter), qidx);
  263. netif_stop_subqueue(padapter->pnetdev, qidx);
  264. busy = _TRUE;
  265. }
  266. #else
  267. if (rtw_os_need_stop_queue(padapter, 0)) {
  268. if (DBG_DUMP_OS_QUEUE_CTL)
  269. RTW_INFO(FUNC_ADPT_FMT": netif_stop_queue\n", FUNC_ADPT_ARG(padapter));
  270. rtw_netif_stop_queue(padapter->pnetdev);
  271. busy = _TRUE;
  272. }
  273. #endif
  274. return busy;
  275. }
  276. void rtw_os_wake_queue_at_free_stainfo(_adapter *padapter, int *qcnt_freed)
  277. {
  278. #if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
  279. int i;
  280. for (i = 0; i < 4; i++) {
  281. if (qcnt_freed[i] == 0)
  282. continue;
  283. if (rtw_os_need_wake_queue(padapter, i)) {
  284. if (DBG_DUMP_OS_QUEUE_CTL)
  285. RTW_INFO(FUNC_ADPT_FMT": netif_wake_subqueue[%d]\n", FUNC_ADPT_ARG(padapter), i);
  286. netif_wake_subqueue(padapter->pnetdev, i);
  287. }
  288. }
  289. #else
  290. if (qcnt_freed[0] || qcnt_freed[1] || qcnt_freed[2] || qcnt_freed[3]) {
  291. if (rtw_os_need_wake_queue(padapter, 0)) {
  292. if (DBG_DUMP_OS_QUEUE_CTL)
  293. RTW_INFO(FUNC_ADPT_FMT": netif_wake_queue\n", FUNC_ADPT_ARG(padapter));
  294. netif_wake_queue(padapter->pnetdev);
  295. }
  296. }
  297. #endif
  298. }
  299. #ifdef CONFIG_TX_MCAST2UNI
  300. int rtw_mlcst2unicst(_adapter *padapter, struct sk_buff *skb)
  301. {
  302. struct sta_priv *pstapriv = &padapter->stapriv;
  303. struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
  304. _irqL irqL;
  305. _list *phead, *plist;
  306. struct sk_buff *newskb;
  307. struct sta_info *psta = NULL;
  308. u8 chk_alive_num = 0;
  309. char chk_alive_list[NUM_STA];
  310. u8 bc_addr[6] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
  311. u8 null_addr[6] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
  312. int i;
  313. s32 res;
  314. DBG_COUNTER(padapter->tx_logs.os_tx_m2u);
  315. _enter_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  316. phead = &pstapriv->asoc_list;
  317. plist = get_next(phead);
  318. /* free sta asoc_queue */
  319. while ((rtw_end_of_queue_search(phead, plist)) == _FALSE) {
  320. int stainfo_offset;
  321. psta = LIST_CONTAINOR(plist, struct sta_info, asoc_list);
  322. plist = get_next(plist);
  323. stainfo_offset = rtw_stainfo_offset(pstapriv, psta);
  324. if (stainfo_offset_valid(stainfo_offset))
  325. chk_alive_list[chk_alive_num++] = stainfo_offset;
  326. }
  327. _exit_critical_bh(&pstapriv->asoc_list_lock, &irqL);
  328. for (i = 0; i < chk_alive_num; i++) {
  329. psta = rtw_get_stainfo_by_offset(pstapriv, chk_alive_list[i]);
  330. if (!(psta->state & _FW_LINKED)) {
  331. DBG_COUNTER(padapter->tx_logs.os_tx_m2u_ignore_fw_linked);
  332. continue;
  333. }
  334. /* avoid come from STA1 and send back STA1 */
  335. if (_rtw_memcmp(psta->hwaddr, &skb->data[6], 6) == _TRUE
  336. || _rtw_memcmp(psta->hwaddr, null_addr, 6) == _TRUE
  337. || _rtw_memcmp(psta->hwaddr, bc_addr, 6) == _TRUE
  338. ) {
  339. DBG_COUNTER(padapter->tx_logs.os_tx_m2u_ignore_self);
  340. continue;
  341. }
  342. DBG_COUNTER(padapter->tx_logs.os_tx_m2u_entry);
  343. newskb = rtw_skb_copy(skb);
  344. if (newskb) {
  345. _rtw_memcpy(newskb->data, psta->hwaddr, 6);
  346. res = rtw_xmit(padapter, &newskb);
  347. if (res < 0) {
  348. DBG_COUNTER(padapter->tx_logs.os_tx_m2u_entry_err_xmit);
  349. RTW_INFO("%s()-%d: rtw_xmit() return error! res=%d\n", __FUNCTION__, __LINE__, res);
  350. pxmitpriv->tx_drop++;
  351. rtw_skb_free(newskb);
  352. }
  353. } else {
  354. DBG_COUNTER(padapter->tx_logs.os_tx_m2u_entry_err_skb);
  355. RTW_INFO("%s-%d: rtw_skb_copy() failed!\n", __FUNCTION__, __LINE__);
  356. pxmitpriv->tx_drop++;
  357. /* rtw_skb_free(skb); */
  358. return _FALSE; /* Caller shall tx this multicast frame via normal way. */
  359. }
  360. }
  361. rtw_skb_free(skb);
  362. return _TRUE;
  363. }
  364. #endif /* CONFIG_TX_MCAST2UNI */
  365. int _rtw_xmit_entry(_pkt *pkt, _nic_hdl pnetdev)
  366. {
  367. _adapter *padapter = (_adapter *)rtw_netdev_priv(pnetdev);
  368. struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
  369. #ifdef CONFIG_TX_MCAST2UNI
  370. struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
  371. extern int rtw_mc2u_disable;
  372. #endif /* CONFIG_TX_MCAST2UNI */
  373. s32 res = 0;
  374. #if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
  375. u16 queue;
  376. #endif
  377. if (padapter->registrypriv.mp_mode) {
  378. RTW_INFO("MP_TX_DROP_OS_FRAME\n");
  379. goto drop_packet;
  380. }
  381. DBG_COUNTER(padapter->tx_logs.os_tx);
  382. if (rtw_if_up(padapter) == _FALSE) {
  383. DBG_COUNTER(padapter->tx_logs.os_tx_err_up);
  384. #ifdef DBG_TX_DROP_FRAME
  385. RTW_INFO("DBG_TX_DROP_FRAME %s if_up fail\n", __FUNCTION__);
  386. #endif
  387. goto drop_packet;
  388. }
  389. rtw_check_xmit_resource(padapter, pkt);
  390. #ifdef CONFIG_TX_MCAST2UNI
  391. if (!rtw_mc2u_disable
  392. && check_fwstate(pmlmepriv, WIFI_AP_STATE) == _TRUE
  393. && (IP_MCAST_MAC(pkt->data)
  394. || ICMPV6_MCAST_MAC(pkt->data)
  395. #ifdef CONFIG_TX_BCAST2UNI
  396. || is_broadcast_mac_addr(pkt->data)
  397. #endif
  398. )
  399. && (padapter->registrypriv.wifi_spec == 0)
  400. ) {
  401. if (pxmitpriv->free_xmitframe_cnt > (NR_XMITFRAME / 4)) {
  402. res = rtw_mlcst2unicst(padapter, pkt);
  403. if (res == _TRUE)
  404. goto exit;
  405. } else {
  406. /* RTW_INFO("Stop M2U(%d, %d)! ", pxmitpriv->free_xmitframe_cnt, pxmitpriv->free_xmitbuf_cnt); */
  407. /* RTW_INFO("!m2u ); */
  408. DBG_COUNTER(padapter->tx_logs.os_tx_m2u_stop);
  409. }
  410. }
  411. #endif /* CONFIG_TX_MCAST2UNI */
  412. res = rtw_xmit(padapter, &pkt);
  413. if (res < 0) {
  414. #ifdef DBG_TX_DROP_FRAME
  415. RTW_INFO("DBG_TX_DROP_FRAME %s rtw_xmit fail\n", __FUNCTION__);
  416. #endif
  417. goto drop_packet;
  418. }
  419. goto exit;
  420. drop_packet:
  421. pxmitpriv->tx_drop++;
  422. rtw_os_pkt_complete(padapter, pkt);
  423. exit:
  424. return 0;
  425. }
  426. int rtw_xmit_entry(_pkt *pkt, _nic_hdl pnetdev)
  427. {
  428. _adapter *padapter = (_adapter *)rtw_netdev_priv(pnetdev);
  429. struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
  430. int ret = 0;
  431. if (pkt) {
  432. if (check_fwstate(pmlmepriv, WIFI_MONITOR_STATE) == _TRUE) {
  433. #if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 24))
  434. rtw_monitor_xmit_entry((struct sk_buff *)pkt, pnetdev);
  435. #endif
  436. }
  437. else {
  438. rtw_mstat_update(MSTAT_TYPE_SKB, MSTAT_ALLOC_SUCCESS, pkt->truesize);
  439. ret = _rtw_xmit_entry(pkt, pnetdev);
  440. }
  441. }
  442. return ret;
  443. }