/****************************************************************************** * * Copyright(c) 2007 - 2012 Realtek Corporation. All rights reserved. * * This program is free software; you can redistribute it and/or modify it * under the terms of version 2 of the GNU General Public License as * published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License along with * this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA * * ******************************************************************************/ #ifndef _RTW_RECV_H_ #define _RTW_RECV_H_ #include #include #define NR_RECVFRAME 256 #define RXFRAME_ALIGN 8 #define RXFRAME_ALIGN_SZ (1<signal_stat_timer, \ (recvpriv)->signal_stat_sampling_interval) struct sta_recv_priv { spinlock_t lock; int option; struct __queue defrag_q; /* keeping the fragment frame until defrag */ struct stainfo_rxcache rxcache; }; struct recv_buf { struct list_head list; spinlock_t recvbuf_lock; u32 ref_cnt; struct adapter *adapter; u8 *pbuf; u8 *pallocated_buf; u32 len; u8 *phead; u8 *pdata; u8 *ptail; u8 *pend; struct urb *purb; dma_addr_t dma_transfer_addr; /* (in) dma addr for transfer_buffer */ u32 alloc_sz; u8 irp_pending; int transfer_len; struct sk_buff *pskb; u8 reuse; }; /* head -----> data -----> payload tail -----> end -----> len = (unsigned int )(tail - data); */ struct recv_frame { struct list_head list; struct sk_buff *pkt; struct sk_buff *pkt_newalloc; struct adapter *adapter; u8 fragcnt; int frame_tag; struct rx_pkt_attrib attrib; uint len; u8 *rx_head; u8 *rx_data; u8 *rx_tail; u8 *rx_end; void *precvbuf; struct sta_info *psta; /* for A-MPDU Rx reordering buffer control */ struct recv_reorder_ctrl *preorder_ctrl; }; struct recv_frame *_rtw_alloc_recvframe(struct __queue *pfree_recv_queue); struct recv_frame *rtw_alloc_recvframe(struct __queue *pfree_recv_queue); void rtw_init_recvframe(struct recv_frame *precvframe, struct recv_priv *precvpriv); int rtw_free_recvframe(struct recv_frame *precvframe, struct __queue *pfree_recv_queue); #define rtw_dequeue_recvframe(queue) rtw_alloc_recvframe(queue) int _rtw_enqueue_recvframe(struct recv_frame *precvframe, struct __queue *queue); int rtw_enqueue_recvframe(struct recv_frame *precvframe, struct __queue *queue); void rtw_free_recvframe_queue(struct __queue *pframequeue, struct __queue *pfree_recv_queue); u32 rtw_free_uc_swdec_pending_queue(struct adapter *adapter); int rtw_enqueue_recvbuf_to_head(struct recv_buf *buf, struct __queue *queue); int rtw_enqueue_recvbuf(struct recv_buf *precvbuf, struct __queue *queue); struct recv_buf *rtw_dequeue_recvbuf(struct __queue *queue); void rtw_reordering_ctrl_timeout_handler(void *pcontext); static inline u8 *get_rxmem(struct recv_frame *precvframe) { /* always return rx_head... */ if (precvframe == NULL) return NULL; return precvframe->rx_head; } static inline u8 *get_rx_status(struct recv_frame *precvframe) { return get_rxmem(precvframe); } static inline u8 *get_recvframe_data(struct recv_frame *precvframe) { /* always return rx_data */ if (precvframe == NULL) return NULL; return precvframe->rx_data; } static inline u8 *recvframe_push(struct recv_frame *precvframe, int sz) { /* append data before rx_data */ /* add data to the start of recv_frame * * This function extends the used data area of the recv_frame at the buffer * start. rx_data must be still larger than rx_head, after pushing. */ if (precvframe == NULL) return NULL; precvframe->rx_data -= sz ; if (precvframe->rx_data < precvframe->rx_head) { precvframe->rx_data += sz; return NULL; } precvframe->len += sz; return precvframe->rx_data; } static inline u8 *recvframe_pull(struct recv_frame *precvframe, int sz) { /* rx_data += sz; move rx_data sz bytes hereafter */ /* used for extract sz bytes from rx_data, update rx_data and return * the updated rx_data to the caller */ if (precvframe == NULL) return NULL; precvframe->rx_data += sz; if (precvframe->rx_data > precvframe->rx_tail) { precvframe->rx_data -= sz; return NULL; } precvframe->len -= sz; return precvframe->rx_data; } static inline u8 *recvframe_put(struct recv_frame *precvframe, int sz) { /* used for append sz bytes from ptr to rx_tail, update rx_tail * and return the updated rx_tail to the caller */ /* after putting, rx_tail must be still larger than rx_end. */ if (precvframe == NULL) return NULL; precvframe->rx_tail += sz; if (precvframe->rx_tail > precvframe->rx_end) { precvframe->rx_tail -= sz; return NULL; } precvframe->len += sz; return precvframe->rx_tail; } static inline u8 *recvframe_pull_tail(struct recv_frame *precvframe, int sz) { /* rmv data from rx_tail (by yitsen) */ /* used for extract sz bytes from rx_end, update rx_end and return * the updated rx_end to the caller */ /* after pulling, rx_end must be still larger than rx_data. */ if (precvframe == NULL) return NULL; precvframe->rx_tail -= sz; if (precvframe->rx_tail < precvframe->rx_data) { precvframe->rx_tail += sz; return NULL; } precvframe->len -= sz; return precvframe->rx_tail; } static inline unsigned char *get_rxbuf_desc(struct recv_frame *precvframe) { unsigned char *buf_desc; if (precvframe == NULL) return NULL; return buf_desc; } static inline struct recv_frame *rxmem_to_recvframe(u8 *rxmem) { /* due to the design of 2048 bytes alignment of recv_frame, * we can reference the struct recv_frame */ /* from any given member of recv_frame. */ /* rxmem indicates the any member/address in recv_frame */ return (struct recv_frame *)(((size_t)rxmem >> RXFRAME_ALIGN) << RXFRAME_ALIGN); } static inline struct recv_frame *pkt_to_recvframe(struct sk_buff *pkt) { u8 *buf_star; struct recv_frame *precv_frame; precv_frame = rxmem_to_recvframe((unsigned char *)buf_star); return precv_frame; } static inline u8 *pkt_to_recvmem(struct sk_buff *pkt) { /* return the rx_head */ struct recv_frame *precv_frame = pkt_to_recvframe(pkt); return precv_frame->rx_head; } static inline u8 *pkt_to_recvdata(struct sk_buff *pkt) { /* return the rx_data */ struct recv_frame *precv_frame = pkt_to_recvframe(pkt); return precv_frame->rx_data; } static inline int get_recvframe_len(struct recv_frame *precvframe) { return precvframe->len; } static inline s32 translate_percentage_to_dbm(u32 sig_stren_index) { s32 power; /* in dBm. */ /* Translate to dBm (x=0.5y-95). */ power = (s32)((sig_stren_index + 1) >> 1); power -= 95; return power; } struct sta_info; void _rtw_init_sta_recv_priv(struct sta_recv_priv *psta_recvpriv); void mgt_dispatcher(struct adapter *padapter, struct recv_frame *precv_frame); #endif