dm9000的驱动分析
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
};
网卡驱动的数据接收发送流程?
初始化设备
writeb(NCR_RST, db);
/* dm9000的初始化,芯片厂商会支持 */
dm9000_init_dm9000(dev);
/* 启动发送队列 */
netif_start_queue(dev);
}
数据接收流程
(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++;
数据发送流程
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->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 设置
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 ...
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)
(email:wwwlllll@126.com)
)