武汉linux驱动培训转让,【亚嵌武汉中心】Linux下网络设备驱动编程(二)

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼

5,关闭网络设备

static int tg3_close(struct net_device *dev)

{

//停止网卡传输包

netif_stop_queue(dev);

netif_carrier_off(tp->dev);

//去除定时器

del_timer_sync(&tp->timer);

//释放收包和发包的缓冲区

tg3_free_rings(tp);

//释放中断

free_irq(dev->irq, dev);

}

[NextPage]

6,硬件处理数据包发送

static int tg3_start_xmit(struct sk_buff *skb, struct net_device *dev)

{

len = (skb->len - skb->data_len);

//以DMA方式向网卡物理设备传输包。如果是wireless的话,需要根据802.11协议及硬

件的规范从新填充

//硬件帧头,然后提交给硬件发送。

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

tp->tx_buffers[entry].skb = skb;

pci_unmap_addr_set(&tp->tx_buffers[entry], mapping, mapping);

//硬件发送

tg3_set_txd(tp, entry, mapping, len, base_flags, mss_and_is_end);

//记录发包开始时间

dev->trans_start = jiffies;

}

7,中断处理收包,发包

static void tg3_interrupt(int irq, void *dev_id, struct pt_regs *regs)

{

//如果要收包

tg3_rx(tp);

//如果要发包

tg3_tx(tp);

}

8,发包

static void tg3_tx(struct tg3 *tp)

{

struct tx_ring_info *ri = &tp->tx_buffers[sw_idx];

struct sk_buff *skb = ri->skb;

//以DMA方式向网卡传输包完毕

pci_unmap_single(tp->pdev, pci_unmap_addr(ri, mapping),

(skb->len - skb->data_len), PCI_DMA_TODEVICE);

ri->skb = NULL;

dev_kfree_skb_irq(skb);

}

9,收包

static int tg3_rx(struct tg3 *tp, int budget)

{

struct sk_buff *copy_skb;

//分配一个包

copy_skb = dev_alloc_skb(len + 2);

copy_skb->dev = tp->dev;

//修改包头空间

skb_reserve(copy_skb, 2);

//加入数据到包中

skb_put(copy_skb, len);

//以DMA方式从网卡传输回数据

pci_dma_sync_single(tp->pdev, dma_addr, len, PCI_DMA_FROMDEVICE);

memcpy(copy_skb->data, skb->data, len);

skb = copy_skb;

//解析包的协议

skb->protocol = eth_type_trans(skb, tp->dev);

//把包送到协议层

netif_rx(skb);

//记录收包时间

tp->dev->last_rx = jiffies;

}

10, 读取包的网卡收发包的状态,统计数据

static struct net_device_stats *tg3_get_stats(struct net_device *dev)

{

//从硬件相关的寄存器读取数据,累加

//stats->rx_packets, stats->tx_packets, stats->rx_bytes, stats->tx_bytes等

}

11, 用户的ioctl命令系统调用

static int tg3_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)

{

struct mii_ioctl_data *data = (struct mii_ioctl_data *)&ifr->ifr_data;

switch(cmd) {

//ethtool程序命令的调用

case SIOCETHTOOL:

return tg3_ethtool_ioctl(dev, (void *) ifr->ifr_data);

//mii程序命令的调用

case SIOCGMIIREG: {

err = tg3_readphy(tp, data->reg_num & 0x1f, &mii_regval)

data->val_out = mii_regval;

return err;

}

……

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值