【Linux高级驱动】如何分析并移植网卡驱动

时间:2023-03-09 05:39:27
【Linux高级驱动】如何分析并移植网卡驱动

dm9000的驱动分析

m9000_init
 platform_driver_register();
 db);
 db);
 );
  ;
 id_val ;
 id_val ;
 /* 获取芯片型号 */
 id_val = ior(db, DM9000_CHIPR);
 ether_setup(ndev);
 /*设置操作方法*/
 ndev->netdev_ops  = &dm9000_netdev_ops;
 /*注册网络设备*/
 register_netdev(ndev);   //register_chrdev

cs8900a网卡驱动分析

);
 /*识别芯片*/
 /*操作方法的设置*/
 dev->netdev_ops = &net_ops;
 /*注册网络设备*/
 register_netdev(dev);

static const struct net_device_ops dm9000_netdev_ops = {
 .ndo_open    = dm9000_open,   //必须的
 .ndo_stop    = dm9000_stop,   //必须的
 .ndo_start_xmit   = dm9000_start_xmit, //必须的
 .ndo_tx_timeout   = dm9000_timeout,  //必须的
 .ndo_set_multicast_list = dm9000_hash_table,
 .ndo_do_ioctl   = dm9000_ioctl,
 .ndo_change_mtu   = eth_change_mtu,
 .ndo_validate_addr  = eth_validate_addr,
 .ndo_set_mac_address = eth_mac_addr,
#ifdef CONFIG_NET_POLL_CONTROLLER
 .ndo_poll_controller = dm9000_poll_controller,
#endif
};

网卡驱动的数据接收发送流程?

初始化设备

dm9000_open();
    writeb(NCR_RST, db);
  /* dm9000的初始化,芯片厂商会支持 */
  dm9000_init_dm9000(dev);
  
  /* 启动发送队列 */
  netif_start_queue(dev);
}

数据接收流程

dm9000_interrupt())   );  );
   (db->inblk)(db->io_data, rdptr, RxLen);  //dm9000_inblk_16bit   //读真正的有效数据(MAC头,TCP头,IP头,网络数据)
   dev->stats.rx_bytes += RxLen;
   /* Pass to upper layer,去掉MAC头 */
   skb->protocol = eth_type_trans(skb, dev);
   /* 将数据上报到上层 */
   netif_rx(skb);
   dev->stats.rx_packets++;

数据发送流程

sk_buff
dm9000_start_xmit
 );
  )
    dm9000_send_packet(dev, db);
     /*启动发送:数据发送完成,产生中断*/
     iow(dm, DM9000_TCR, TCR_TXREQ); /* Cleared after TX complete */

netif_wake_queue(dev);
 /* Re-enable interrupt mask */
 iow(db, DM9000_IMR, db->imr_all);
 /* Restore previous register address */
 writeb(reg_save, db->io_addr);  //恢复为自动增加

怎么写网卡驱动

1.cs89x0.c

1.1 分配一个net_device结构体

alloc_etherdev

1.2 设置

    dev->open       = net_open;
    dev->stop       = net_close;
    dev->tx_timeout     = net_timeout;
    dev->watchdog_timeo = HZ;
    dev->hard_start_xmit    = net_send_packet;
    dev->get_stats      = net_get_stats;
    dev->set_multicast_list = set_multicast_list;
    dev->set_mac_address    = set_mac_address;

1.3 注册

register_netdev

2. DM9000.c

2.1 分配一个net_device结构体
    ndev = alloc_etherdev(sizeof (struct board_info));
2.2 设置

    ether_setup(ndev);
    ndev->open       = &dm9000_open;
    ndev->hard_start_xmit    = &dm9000_start_xmit;
    ndev->tx_timeout         = &dm9000_timeout;
    ndev->watchdog_timeo = msecs_to_jiffies(watchdog);
    ndev->stop       = &dm9000_stop;
    ndev->get_stats      = &dm9000_get_stats;
    ndev->set_multicast_list = &dm9000_hash_table;

2.3 注册
    ret = register_netdev(ndev);

任何设备的核心都是收发数据

1. 发数据:
   上层要发送数据时,构造一个sk_buff,然后调用net_device的hard_start_xmit来发送

2. 收数据:
   网卡收到数据后,发生中断
   在中断服务程序里:
   从硬件上读出数据,然后构造一个sk_buff,上报:
a. 分配一个sk_buff结构体:
    dev_alloc_skb
b. 使用硬件上得到数据填充这个结构体
c. 上报:netif_rx

测试方法

1. 编译/安装驱动 farsight_net_1.c

   ifconfig fs_net0 up
   ifconfig fs_net0 ...
   ping ... 成功,证明ping自己的话,不经过硬件
   ping ... 多次调用fsnet_hard_start_tx 
   PING ... (...) data bytes
   fsnet_hard_start_tx 
   fsnet_hard_start_tx    
   再次ifconfig发现fs_net0的rx/tx都是0

2. 编译/安装驱动 farsight_net_2.c: 添加统计信息

3. 编译/安装驱动 farsight_net_3.c: 设MAC地址
   ifconfig fs_net0
   ifconfig         可以看到MAC地址

4. 编译/安装驱动 farsight_net_4.c: 构造ping的返回包
   ifconfig fs_net0 up
   ifconfig fs_net0 192.188.1.1
   ping 192.188.1.2                成功

怎么移植网卡驱动

网卡基本上都是内存接口(ram-like)
    1. 根据原理图确定访问地址, 在驱动里修改相应项
    2. 为了能通过这些地址访问网卡,对于2410还要设置memory controller
   比如设置位宽、时间参数
    3. 根据原理图确定中断号, 在驱动里修改相应项(包括中断号、中断触发方式(高/低有效))

@成鹏致远

(blogs:http://lcw.cnblogs.com)

(emailwwwlllll@126.com)