- 论坛徽章:
- 0
|
从下面的代码来看,e100的包发送似乎是同步的,也就是发一个就等待一个完成(见循环执行e100_exec_cmd),浪费大量CPU时间。可是按CB List的构成,CPU 似乎可以提交transmit descriptor,然后启动CU(comand unit),处理整个CB链表。 事实上,e100接收过程就是这样做的,但是发送过程为什么没有这样做呢?- static int e100_exec_cb(struct nic *nic, struct sk_buff *skb,
- void (*cb_prepare)(struct nic *, struct cb *, struct sk_buff *))
- {
- struct cb *cb;
- unsigned long flags;
- int err = 0;
- spin_lock_irqsave(&nic->cb_lock, flags);
- if (unlikely(!nic->cbs_avail)) {
- err = -ENOMEM;
- goto err_unlock;
- }
- cb = nic->cb_to_use;
- nic->cb_to_use = cb->next;
- nic->cbs_avail--;
- cb->skb = skb;
- if (unlikely(!nic->cbs_avail))
- err = -ENOSPC;
- cb_prepare(nic, cb, skb);
- /* Order is important otherwise we'll be in a race with h/w:
- * set S-bit in current first, then clear S-bit in previous. */
- cb->command |= cpu_to_le16(cb_s);
- wmb();
- cb->prev->command &= cpu_to_le16(~cb_s);
- while (nic->cb_to_send != nic->cb_to_use) {
- if (unlikely(e100_exec_cmd(nic, nic->cuc_cmd,
- nic->cb_to_send->dma_addr))) {
- /* Ok, here's where things get sticky. It's
- * possible that we can't schedule the command
- * because the controller is too busy, so
- * let's just queue the command and try again
- * when another command is scheduled. */
- if (err == -ENOSPC) {
- //request a reset
- schedule_work(&nic->tx_timeout_task);
- }
- break;
- } else {
- nic->cuc_cmd = cuc_resume;
- nic->cb_to_send = nic->cb_to_send->next;
- }
- }
- err_unlock:
- spin_unlock_irqrestore(&nic->cb_lock, flags);
- return err;
- }
- static int e100_exec_cmd(struct nic *nic, u8 cmd, dma_addr_t dma_addr)
- {
- unsigned long flags;
- unsigned int i;
- int err = 0;
- spin_lock_irqsave(&nic->cmd_lock, flags);
- /* Previous command is accepted when SCB clears */
- for (i = 0; i < E100_WAIT_SCB_TIMEOUT; i++) {
- if (likely(!ioread8(&nic->csr->scb.cmd_lo)))
- break;
- cpu_relax();
- if (unlikely(i > E100_WAIT_SCB_FAST))
- udelay(5);
- }
- if (unlikely(i == E100_WAIT_SCB_TIMEOUT)) {
- err = -EAGAIN;
- goto err_unlock;
- }
- if (unlikely(cmd != cuc_resume))
- iowrite32(dma_addr, &nic->csr->scb.gen_ptr);
- iowrite8(cmd, &nic->csr->scb.cmd_lo);
- err_unlock:
- spin_unlock_irqrestore(&nic->cmd_lock, flags);
- return err;
- }
复制代码 |
|