Ejemplo n.º 1
0
static int rtl8180_tx(struct ieee80211_hw *dev, struct sk_buff *skb)
{
	struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
	struct rtl8180_priv *priv = dev->priv;
	struct rtl8180_tx_ring *ring;
	struct rtl8180_tx_desc *entry;
	unsigned long flags;
	unsigned int idx, prio;
	dma_addr_t mapping;
	u32 tx_flags;
	u8 rc_flags;
	u16 plcp_len = 0;
	__le16 rts_duration = 0;

	prio = skb_get_queue_mapping(skb);
	ring = &priv->tx_ring[prio];

	mapping = pci_map_single(priv->pdev, skb->data,
				 skb->len, PCI_DMA_TODEVICE);

	tx_flags = RTL818X_TX_DESC_FLAG_OWN | RTL818X_TX_DESC_FLAG_FS |
		   RTL818X_TX_DESC_FLAG_LS |
		   (ieee80211_get_tx_rate(dev, info)->hw_value << 24) |
		   skb->len;

	if (priv->r8185)
		tx_flags |= RTL818X_TX_DESC_FLAG_DMA |
			    RTL818X_TX_DESC_FLAG_NO_ENC;

	rc_flags = info->control.rates[0].flags;
	if (rc_flags & IEEE80211_TX_RC_USE_RTS_CTS) {
		tx_flags |= RTL818X_TX_DESC_FLAG_RTS;
		tx_flags |= ieee80211_get_rts_cts_rate(dev, info)->hw_value << 19;
	} else if (rc_flags & IEEE80211_TX_RC_USE_CTS_PROTECT) {
		tx_flags |= RTL818X_TX_DESC_FLAG_CTS;
		tx_flags |= ieee80211_get_rts_cts_rate(dev, info)->hw_value << 19;
	}

	if (rc_flags & IEEE80211_TX_RC_USE_RTS_CTS)
		rts_duration = ieee80211_rts_duration(dev, priv->vif, skb->len,
						      info);

	if (!priv->r8185) {
		unsigned int remainder;

		plcp_len = DIV_ROUND_UP(16 * (skb->len + 4),
				(ieee80211_get_tx_rate(dev, info)->bitrate * 2) / 10);
		remainder = (16 * (skb->len + 4)) %
			    ((ieee80211_get_tx_rate(dev, info)->bitrate * 2) / 10);
		if (remainder > 0 && remainder <= 6)
			plcp_len |= 1 << 15;
	}

	spin_lock_irqsave(&priv->lock, flags);
	idx = (ring->idx + skb_queue_len(&ring->queue)) % ring->entries;
	entry = &ring->desc[idx];

	entry->rts_duration = rts_duration;
	entry->plcp_len = cpu_to_le16(plcp_len);
	entry->tx_buf = cpu_to_le32(mapping);
	entry->frame_len = cpu_to_le32(skb->len);
	entry->flags2 = info->control.rates[1].idx >= 0 ?
		ieee80211_get_alt_retry_rate(dev, info, 0)->bitrate << 4 : 0;
	entry->retry_limit = info->control.rates[0].count;
	entry->flags = cpu_to_le32(tx_flags);
	__skb_queue_tail(&ring->queue, skb);
	if (ring->entries - skb_queue_len(&ring->queue) < 2)
		ieee80211_stop_queue(dev, skb_get_queue_mapping(skb));
	spin_unlock_irqrestore(&priv->lock, flags);

	rtl818x_iowrite8(priv, &priv->map->TX_DMA_POLLING, (1 << (prio + 4)));

	return 0;
}
Ejemplo n.º 2
0
static void rt2x00queue_create_tx_descriptor(struct queue_entry *entry,
        struct txentry_desc *txdesc)
{
    struct rt2x00_dev *rt2x00dev = entry->queue->rt2x00dev;
    struct ieee80211_tx_info *tx_info = IEEE80211_SKB_CB(entry->skb);
    struct rt2x00_intf *intf = vif_to_intf(tx_info->control.vif);
    struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)entry->skb->data;
    struct ieee80211_rate *rate =
        ieee80211_get_tx_rate(rt2x00dev->hw, tx_info);
    const struct rt2x00_rate *hwrate;
    unsigned int data_length;
    unsigned int duration;
    unsigned int residual;
    unsigned long irqflags;

    memset(txdesc, 0, sizeof(*txdesc));

    /*
     * Initialize information from queue
     */
    txdesc->queue = entry->queue->qid;
    txdesc->cw_min = entry->queue->cw_min;
    txdesc->cw_max = entry->queue->cw_max;
    txdesc->aifs = entry->queue->aifs;

    /* Data length should be extended with 4 bytes for CRC */
    data_length = entry->skb->len + 4;

    /*
     * Check whether this frame is to be acked.
     */
    if (!(tx_info->flags & IEEE80211_TX_CTL_NO_ACK))
        __set_bit(ENTRY_TXD_ACK, &txdesc->flags);

    /*
     * Check if this is a RTS/CTS frame
     */
    if (ieee80211_is_rts(hdr->frame_control) ||
            ieee80211_is_cts(hdr->frame_control)) {
        __set_bit(ENTRY_TXD_BURST, &txdesc->flags);
        if (ieee80211_is_rts(hdr->frame_control))
            __set_bit(ENTRY_TXD_RTS_FRAME, &txdesc->flags);
        else
            __set_bit(ENTRY_TXD_CTS_FRAME, &txdesc->flags);
        if (tx_info->control.rts_cts_rate_idx >= 0)
            rate =
                ieee80211_get_rts_cts_rate(rt2x00dev->hw, tx_info);
    }

    /*
     * Determine retry information.
     */
    txdesc->retry_limit = tx_info->control.retry_limit;
    if (tx_info->flags & IEEE80211_TX_CTL_LONG_RETRY_LIMIT)
        __set_bit(ENTRY_TXD_RETRY_MODE, &txdesc->flags);

    /*
     * Check if more fragments are pending
     */
    if (ieee80211_has_morefrags(hdr->frame_control)) {
        __set_bit(ENTRY_TXD_BURST, &txdesc->flags);
        __set_bit(ENTRY_TXD_MORE_FRAG, &txdesc->flags);
    }

    /*
     * Beacons and probe responses require the tsf timestamp
     * to be inserted into the frame.
     */
    if (ieee80211_is_beacon(hdr->frame_control) ||
            ieee80211_is_probe_resp(hdr->frame_control))
        __set_bit(ENTRY_TXD_REQ_TIMESTAMP, &txdesc->flags);

    /*
     * Determine with what IFS priority this frame should be send.
     * Set ifs to IFS_SIFS when the this is not the first fragment,
     * or this fragment came after RTS/CTS.
     */
    if (test_bit(ENTRY_TXD_RTS_FRAME, &txdesc->flags)) {
        txdesc->ifs = IFS_SIFS;
    } else if (tx_info->flags & IEEE80211_TX_CTL_FIRST_FRAGMENT) {
        __set_bit(ENTRY_TXD_FIRST_FRAGMENT, &txdesc->flags);
        txdesc->ifs = IFS_BACKOFF;
    } else {
        txdesc->ifs = IFS_SIFS;
    }

    /*
     * Hardware should insert sequence counter.
     * FIXME: We insert a software sequence counter first for
     * hardware that doesn't support hardware sequence counting.
     *
     * This is wrong because beacons are not getting sequence
     * numbers assigned properly.
     *
     * A secondary problem exists for drivers that cannot toggle
     * sequence counting per-frame, since those will override the
     * sequence counter given by mac80211.
     */
    if (tx_info->flags & IEEE80211_TX_CTL_ASSIGN_SEQ) {
        spin_lock_irqsave(&intf->seqlock, irqflags);

        if (test_bit(ENTRY_TXD_FIRST_FRAGMENT, &txdesc->flags))
            intf->seqno += 0x10;
        hdr->seq_ctrl &= cpu_to_le16(IEEE80211_SCTL_FRAG);
        hdr->seq_ctrl |= cpu_to_le16(intf->seqno);

        spin_unlock_irqrestore(&intf->seqlock, irqflags);

        __set_bit(ENTRY_TXD_GENERATE_SEQ, &txdesc->flags);
    }

    /*
     * PLCP setup
     * Length calculation depends on OFDM/CCK rate.
     */
    hwrate = rt2x00_get_rate(rate->hw_value);
    txdesc->signal = hwrate->plcp;
    txdesc->service = 0x04;

    if (hwrate->flags & DEV_RATE_OFDM) {
        __set_bit(ENTRY_TXD_OFDM_RATE, &txdesc->flags);

        txdesc->length_high = (data_length >> 6) & 0x3f;
        txdesc->length_low = data_length & 0x3f;
    } else {