2 * carl9170 firmware - used by the ar9170 wireless device
4 * DMA descriptor handling functions
6 * Copyright (c) 2000-2005 ZyDAS Technology Corporation
7 * Copyright (c) 2007-2009 Atheros Communications, Inc.
8 * Copyright 2009 Johannes Berg <johannes@sipsolutions.net>
9 * Copyright 2009-2011 Christian Lamparter <chunkeey@googlemail.com>
11 * This program is free software; you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License as published by
13 * the Free Software Foundation; either version 2 of the License, or
14 * (at your option) any later version.
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
21 * You should have received a copy of the GNU General Public License along
22 * with this program; If not, see <http://www.gnu.org/licenses/>.
29 struct ar9170_dma_memory dma_mem __in_section(sram);
31 static void copy_dma_desc(struct dma_desc *dst,
34 memcpy(dst, src, sizeof(struct dma_desc));
37 static void clear_descriptor(struct dma_desc *d)
39 d->status = AR9170_OWN_BITS_SW;
48 static void fill_descriptor(struct dma_desc *d, uint16_t size, uint8_t *data)
50 d->status = AR9170_OWN_BITS_SW;
59 static void init_queue(struct dma_queue *q, struct dma_desc *d)
61 q->head = q->terminator = d;
65 * - Init up_queue, down_queue, tx_queue[5], rx_queue.
66 * - Setup descriptors and data buffer address.
67 * - Ring descriptors rx_queue and down_queue by dma_reclaim().
69 * NOTE: LastAddr tempary point (same) to nextAddr after initialize.
70 * Because LastAddr is don't care in function dma_reclaim().
72 void dma_init_descriptors(void)
76 for (i = 0; i < ARRAY_SIZE(dma_mem.terminator); i++)
77 clear_descriptor(&dma_mem.terminator[i]);
79 /* Assign terminators to DMA queues */
81 init_queue(&fw.pta.up_queue, &dma_mem.terminator[i++]);
82 init_queue(&fw.pta.down_queue, &dma_mem.terminator[i++]);
83 for (j = 0; j < __AR9170_NUM_TX_QUEUES; j++)
84 init_queue(&fw.wlan.tx_queue[j], &dma_mem.terminator[i++]);
85 init_queue(&fw.wlan.tx_retry, &dma_mem.terminator[i++]);
86 init_queue(&fw.wlan.rx_queue, &dma_mem.terminator[i++]);
87 fw.usb.int_desc = &dma_mem.terminator[i++];
88 fw.wlan.fw_desc = &dma_mem.terminator[i++];
90 for (j = 0; j < CARL9170_INTF_NUM; j++)
91 init_queue(&fw.wlan.cab_queue[j], &dma_mem.terminator[i++]);
93 BUG_ON(AR9170_TERMINATOR_NUMBER != i);
95 DBG("Blocks:%d [tx:%d, rx:%d] Terminators:%d/%d\n",
96 AR9170_BLOCK_NUMBER, AR9170_TX_BLOCK_NUMBER,
97 AR9170_RX_BLOCK_NUMBER, AR9170_TERMINATOR_NUMBER, i);
99 /* Init descriptors and memory blocks */
100 for (i = 0; i < AR9170_BLOCK_NUMBER; i++) {
101 fill_descriptor(&dma_mem.block[i], AR9170_BLOCK_SIZE, dma_mem.data[i].data);
103 if (i < AR9170_TX_BLOCK_NUMBER)
104 dma_reclaim(&fw.pta.down_queue, &dma_mem.block[i]);
106 dma_reclaim(&fw.wlan.rx_queue, &dma_mem.block[i]);
109 /* Set DMA address registers */
110 set(AR9170_PTA_REG_DN_DMA_ADDRH, (uint32_t) fw.pta.down_queue.head >> 16);
111 set(AR9170_PTA_REG_DN_DMA_ADDRL, (uint32_t) fw.pta.down_queue.head & 0xffff);
112 set(AR9170_PTA_REG_UP_DMA_ADDRH, (uint32_t) fw.pta.up_queue.head >> 16);
113 set(AR9170_PTA_REG_UP_DMA_ADDRL, (uint32_t) fw.pta.up_queue.head & 0xffff);
115 for (i = 0; i < __AR9170_NUM_TX_QUEUES; i++)
116 set_wlan_txq_dma_addr(i, (uint32_t) fw.wlan.tx_queue[i].head);
118 set(AR9170_MAC_REG_DMA_RXQ_ADDR, (uint32_t) fw.wlan.rx_queue.head);
119 fw.usb.int_desc->dataSize = AR9170_BLOCK_SIZE;
120 fw.usb.int_desc->dataAddr = (void *) &dma_mem.reserved.rsp;
122 memset(DESC_PAYLOAD(fw.usb.int_desc), 0xff,
123 AR9170_INT_MAGIC_HEADER_SIZE);
124 memset(DESC_PAYLOAD_OFF(fw.usb.int_desc, AR9170_INT_MAGIC_HEADER_SIZE),
125 0, AR9170_BLOCK_SIZE - AR9170_INT_MAGIC_HEADER_SIZE);
127 /* rsp is now available for use */
128 fw.usb.int_desc_available = 1;
130 memset(DESC_PAYLOAD(fw.wlan.fw_desc), 0, 128);
131 fw.wlan.fw_desc_available = 1;
137 * Exchange the terminator and the first descriptor of the packet
138 * for hardware ascy...
140 void dma_reclaim(struct dma_queue *q, struct dma_desc *desc)
142 struct dma_desc *tmpDesc, *last;
143 struct dma_desc tdesc;
145 /* 1. Set OWN bit to HW for all TDs to be added, clear ctrl and size */
147 last = desc->lastAddr;
150 tmpDesc->status = AR9170_OWN_BITS_HW;
152 tmpDesc->totalLen = 0;
153 tmpDesc->dataSize = AR9170_BLOCK_SIZE;
155 /* TODO : Exception handle */
157 tmpDesc->lastAddr = tmpDesc;
162 tmpDesc = tmpDesc->nextAddr;
165 /* 2. Next address of Last TD to be added = first TD */
166 tmpDesc->nextAddr = desc;
168 /* Link first TD to self */
169 desc->lastAddr = q->terminator;
171 /* 3. Copy first TD to be added to TTD */
172 copy_dma_desc(&tdesc, desc);
174 /* 4. Initialize new terminator */
175 clear_descriptor(desc);
177 /* 5. Copy TTD to last TD */
179 copy_dma_desc((void *)q->terminator, (void *)&tdesc);
180 q->terminator->status |= AR9170_OWN_BITS_HW;
182 /* Update terminator pointer */
183 q->terminator = desc;
187 * Put a complete packet into the tail of the Queue q.
188 * Exchange the terminator and the first descriptor of the packet
189 * for hardware ascy...
191 void dma_put(struct dma_queue *q, struct dma_desc *desc)
193 struct dma_desc *tmpDesc;
194 struct dma_desc tdesc;
199 /* update totalLen */
200 tmpDesc->totalLen = desc->totalLen;
202 /* 1. Set OWN bit to HW for all TDs to be added */
203 tmpDesc->status = AR9170_OWN_BITS_HW;
204 /* TODO : Exception handle */
206 tmpDesc->lastAddr = desc->lastAddr;
208 if (desc->lastAddr == tmpDesc)
211 tmpDesc = tmpDesc->nextAddr;
214 /* 2. Next address of Last TD to be added = first TD */
215 desc->lastAddr->nextAddr = desc;
217 /* If there is only one descriptor, update pointer of last descriptor */
218 if (desc->lastAddr == desc)
219 desc->lastAddr = q->terminator;
221 /* 3. Copy first TD to be added to TTD */
222 copy_dma_desc(&tdesc, desc);
224 /* 4. Initialize new terminator */
225 clear_descriptor(desc);
227 /* 5. Copy TTD to last TD */
228 tdesc.status &= (~AR9170_OWN_BITS);
229 copy_dma_desc((void *)q->terminator, (void *)&tdesc);
230 q->terminator->status |= AR9170_OWN_BITS_HW;
232 /* Update terminator pointer */
233 q->terminator = desc;
236 struct dma_desc *dma_unlink_head(struct dma_queue *queue)
238 struct dma_desc *desc;
240 if (queue_empty(queue))
245 queue->head = desc->lastAddr->nextAddr;
247 /* poison nextAddr address */
248 desc->lastAddr->nextAddr = desc->lastAddr;
249 desc->lastAddr->lastAddr = desc->lastAddr;