vcard介绍
vcard是一个虚拟网卡,和具体的硬件无关,它没有任何的实际意义,但是通过编写这样一个虚拟网卡,我们可以很好的研究内核中网络设备层的代码逻辑。
这篇笔记通过vcard虚拟网卡展示了网络设备的分配、注册、注销和销毁这一完整过程。
网络设备的分配与注册
为了简便,在模块插入内核时完成网络设备的分配与注册。
// 网卡名字
#define DEV_NAME "vcard"
// 网卡设备
static struct net_device *vcard_device = NULL;
static int install_net_device(void)
{
int ret;
// 分配网卡对象
vcard_device = alloc_netdev(0, DEV_NAME, vcard_setup);
if (!vcard_device) {
printk(TAG "alloc_netdev failed\n");
return -ENOMEM;
}
// 将网卡注册到系统中
ret = register_netdev(vcard_device);
if (ret) {
printk(TAG "register_netdev failed\n");
goto free;
}
printk(TAG "after register_netdev: reg_state=%d\n", vcard_device->reg_state);
return 0;
free:
free_netdev(vcard_device);
vcard_device = NULL;
return ret;
}
分配时提供了setup()回调,该接口必须提供,驱动程序这样可以完成一些初始化,vcard_setup()的实现如下:
// init()回调
static int vcard_init(struct net_device *dev)
{
printk(TAG "in init: reg_state=%d\n", dev->reg_state);
return 0;
}
// uninit()回调
static void vcard_uninit(struct net_device *dev)
{
printk(TAG "in uninit: reg_state=%d\n", dev->reg_state);
}
// destructor()回调
void vcard_destructor(struct net_device *dev)
{
printk(TAG "in destructor: reg_state=%d\n", dev->reg_state);
}
// 实验电脑上是3.10.0内核,所以struct net_device的某些数据成员发生了变化,但思想是不变的
static struct net_device_ops vcard_device_ops = {
.ndo_init = &vcard_init,
.ndo_uninit = &vcard_uninit,
};
// 初始化网卡
static void vcard_setup(struct net_device *dev)
{
dev->netdev_ops = &vcard_device_ops;
dev->destructor = &vcard_destructor;
printk(TAG "in setup: reg_state=%d\n", dev->reg_state);
}
这里实际上什么也没有做,就是提供了三个回调来监控网卡设备的生命周期。
网络设备的注销与销毁
在模块从内核移除时完成网络设备的注销与销毁。
static void uninstall_net_device(void)
{
if (vcard_device) {
unregister_netdev(vcard_device);
free_netdev(vcard_device);
vcard_device = NULL;
}
}
Makefile
obj-m := vcard.o
KDIR := /lib/modules/$(shell uname -r)/build/
vcard:
make -C $(KDIR) M=$(PWD) modules
clean:
rm -f *.o *.ko *.order *.symvers
实验输出
当模块插入内核时,输出如下:
当模块从内核移除时,输出如下:
从上面可以清晰的看到在分配、注册、注销和清除过程中,网络设备的注册状态的变化,以及几个回调被调用的时机。完整的代码在这里。