mirror of https://gitee.com/openkylin/linux.git
256 lines
5.4 KiB
C
256 lines
5.4 KiB
C
|
/*
|
||
|
Copyright (C) 2004 - 2007 rt2x00 SourceForge Project
|
||
|
<http://rt2x00.serialmonkey.com>
|
||
|
|
||
|
This program is free software; you can redistribute it and/or modify
|
||
|
it under the terms of the GNU General Public License as published by
|
||
|
the Free Software Foundation; either version 2 of the License, or
|
||
|
(at your option) any later version.
|
||
|
|
||
|
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.,
|
||
|
59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
Module: rt2x00
|
||
|
Abstract: rt2x00 ring datastructures and routines
|
||
|
*/
|
||
|
|
||
|
#ifndef RT2X00RING_H
|
||
|
#define RT2X00RING_H
|
||
|
|
||
|
/*
|
||
|
* data_desc
|
||
|
* Each data entry also contains a descriptor which is used by the
|
||
|
* device to determine what should be done with the packet and
|
||
|
* what the current status is.
|
||
|
* This structure is greatly simplified, but the descriptors
|
||
|
* are basically a list of little endian 32 bit values.
|
||
|
* Make the array by default 1 word big, this will allow us
|
||
|
* to use sizeof() correctly.
|
||
|
*/
|
||
|
struct data_desc {
|
||
|
__le32 word[1];
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* data_entry_desc
|
||
|
* Summary of information that should be written into the
|
||
|
* descriptor for sending a TX frame.
|
||
|
*/
|
||
|
struct data_entry_desc {
|
||
|
unsigned long flags;
|
||
|
#define ENTRY_TXDONE 1
|
||
|
#define ENTRY_TXD_RTS_FRAME 2
|
||
|
#define ENTRY_TXD_OFDM_RATE 3
|
||
|
#define ENTRY_TXD_MORE_FRAG 4
|
||
|
#define ENTRY_TXD_REQ_TIMESTAMP 5
|
||
|
#define ENTRY_TXD_BURST 6
|
||
|
|
||
|
/*
|
||
|
* Queue ID. ID's 0-4 are data TX rings
|
||
|
*/
|
||
|
int queue;
|
||
|
#define QUEUE_MGMT 13
|
||
|
#define QUEUE_RX 14
|
||
|
#define QUEUE_OTHER 15
|
||
|
|
||
|
/*
|
||
|
* PLCP values.
|
||
|
*/
|
||
|
u16 length_high;
|
||
|
u16 length_low;
|
||
|
u16 signal;
|
||
|
u16 service;
|
||
|
|
||
|
/*
|
||
|
* Timing information
|
||
|
*/
|
||
|
int aifs;
|
||
|
int ifs;
|
||
|
int cw_min;
|
||
|
int cw_max;
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* data_entry
|
||
|
* The data ring is a list of data entries.
|
||
|
* Each entry holds a reference to the descriptor
|
||
|
* and the data buffer. For TX rings the reference to the
|
||
|
* sk_buff of the packet being transmitted is also stored here.
|
||
|
*/
|
||
|
struct data_entry {
|
||
|
/*
|
||
|
* Status flags
|
||
|
*/
|
||
|
unsigned long flags;
|
||
|
#define ENTRY_OWNER_NIC 1
|
||
|
|
||
|
/*
|
||
|
* Ring we belong to.
|
||
|
*/
|
||
|
struct data_ring *ring;
|
||
|
|
||
|
/*
|
||
|
* sk_buff for the packet which is being transmitted
|
||
|
* in this entry (Only used with TX related rings).
|
||
|
*/
|
||
|
struct sk_buff *skb;
|
||
|
|
||
|
/*
|
||
|
* Store a ieee80211_tx_status structure in each
|
||
|
* ring entry, this will optimize the txdone
|
||
|
* handler.
|
||
|
*/
|
||
|
struct ieee80211_tx_status tx_status;
|
||
|
|
||
|
/*
|
||
|
* private pointer specific to driver.
|
||
|
*/
|
||
|
void *priv;
|
||
|
|
||
|
/*
|
||
|
* Data address for this entry.
|
||
|
*/
|
||
|
void *data_addr;
|
||
|
dma_addr_t data_dma;
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* data_ring
|
||
|
* Data rings are used by the device to send and receive packets.
|
||
|
* The data_addr is the base address of the data memory.
|
||
|
* To determine at which point in the ring we are,
|
||
|
* have to use the rt2x00_ring_index_*() functions.
|
||
|
*/
|
||
|
struct data_ring {
|
||
|
/*
|
||
|
* Pointer to main rt2x00dev structure where this
|
||
|
* ring belongs to.
|
||
|
*/
|
||
|
struct rt2x00_dev *rt2x00dev;
|
||
|
|
||
|
/*
|
||
|
* Base address for the device specific data entries.
|
||
|
*/
|
||
|
struct data_entry *entry;
|
||
|
|
||
|
/*
|
||
|
* TX queue statistic info.
|
||
|
*/
|
||
|
struct ieee80211_tx_queue_stats_data stats;
|
||
|
|
||
|
/*
|
||
|
* TX Queue parameters.
|
||
|
*/
|
||
|
struct ieee80211_tx_queue_params tx_params;
|
||
|
|
||
|
/*
|
||
|
* Base address for data ring.
|
||
|
*/
|
||
|
dma_addr_t data_dma;
|
||
|
void *data_addr;
|
||
|
|
||
|
/*
|
||
|
* Index variables.
|
||
|
*/
|
||
|
u16 index;
|
||
|
u16 index_done;
|
||
|
|
||
|
/*
|
||
|
* Size of packet and descriptor in bytes.
|
||
|
*/
|
||
|
u16 data_size;
|
||
|
u16 desc_size;
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* Handlers to determine the address of the current device specific
|
||
|
* data entry, where either index or index_done points to.
|
||
|
*/
|
||
|
static inline struct data_entry *rt2x00_get_data_entry(struct data_ring *ring)
|
||
|
{
|
||
|
return &ring->entry[ring->index];
|
||
|
}
|
||
|
|
||
|
static inline struct data_entry *rt2x00_get_data_entry_done(struct data_ring
|
||
|
*ring)
|
||
|
{
|
||
|
return &ring->entry[ring->index_done];
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Total ring memory
|
||
|
*/
|
||
|
static inline int rt2x00_get_ring_size(struct data_ring *ring)
|
||
|
{
|
||
|
return ring->stats.limit * (ring->desc_size + ring->data_size);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Ring index manipulation functions.
|
||
|
*/
|
||
|
static inline void rt2x00_ring_index_inc(struct data_ring *ring)
|
||
|
{
|
||
|
ring->index++;
|
||
|
if (ring->index >= ring->stats.limit)
|
||
|
ring->index = 0;
|
||
|
ring->stats.len++;
|
||
|
}
|
||
|
|
||
|
static inline void rt2x00_ring_index_done_inc(struct data_ring *ring)
|
||
|
{
|
||
|
ring->index_done++;
|
||
|
if (ring->index_done >= ring->stats.limit)
|
||
|
ring->index_done = 0;
|
||
|
ring->stats.len--;
|
||
|
ring->stats.count++;
|
||
|
}
|
||
|
|
||
|
static inline void rt2x00_ring_index_clear(struct data_ring *ring)
|
||
|
{
|
||
|
ring->index = 0;
|
||
|
ring->index_done = 0;
|
||
|
ring->stats.len = 0;
|
||
|
ring->stats.count = 0;
|
||
|
}
|
||
|
|
||
|
static inline int rt2x00_ring_empty(struct data_ring *ring)
|
||
|
{
|
||
|
return ring->stats.len == 0;
|
||
|
}
|
||
|
|
||
|
static inline int rt2x00_ring_full(struct data_ring *ring)
|
||
|
{
|
||
|
return ring->stats.len == ring->stats.limit;
|
||
|
}
|
||
|
|
||
|
static inline int rt2x00_ring_free(struct data_ring *ring)
|
||
|
{
|
||
|
return ring->stats.limit - ring->stats.len;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* TX/RX Descriptor access functions.
|
||
|
*/
|
||
|
static inline void rt2x00_desc_read(struct data_desc *desc,
|
||
|
const u8 word, u32 *value)
|
||
|
{
|
||
|
*value = le32_to_cpu(desc->word[word]);
|
||
|
}
|
||
|
|
||
|
static inline void rt2x00_desc_write(struct data_desc *desc,
|
||
|
const u8 word, const u32 value)
|
||
|
{
|
||
|
desc->word[word] = cpu_to_le32(value);
|
||
|
}
|
||
|
|
||
|
#endif /* RT2X00RING_H */
|