由于内核版本的变更,net_device的结构体变化很大,struct net_device 的结构在新版本内核中发生了变化,删去了quota成员;将(open,stop,set_config,hard_start_xmit等)封装进了struct net_device_ops;将(poll,weight等)封装进了 struct napi_struct;将(hard_header,hard_header_cache,rebuild等)封装进了struct header_ops。并对部分函数名和参数进行了变动。
netif_rx_complete和netif_rx_schedule属于poll机制中的函数,接口名称已经变为napi_complete和napi_schedule。
在改源程序的过程中总结了一些,记下来(以后估计会忘了):
1,在原net_devide结构体中有open、stop等函数,可以直接调用,但是变更之后,这些操作被封装在net_device_ops中,并且是const的只读类型,下面的操作便会出错:
dev->open = dys_dev_open;
dev->stop = dys_dev_stop;
dev->set_config = dys_dev_set_config;
dev->hard_start_xmit = dys_dev_tx;
dev->do_ioctl = dys_dev_ioctl;
dev->get_stats = dys_dev_get_stats;
dev->tx_timeout = dys_dev_tx_timeout;
要使用这些操作必须建立如下结构体初始化:
static const struct net_device_ops dys_netdev_ops = {
.ndo_open = dys_dev_open,
.ndo_stop = dys_dev_stop,
.ndo_start_xmit = dys_dev_tx,
.ndo_do_ioctl = dys_dev_ioctl,
.ndo_get_stats = dys_dev_get_stats,
.ndo_set_config = dys_dev_set_config,
.ndo_tx_timeout = dys_dev_tx_timeout,
};
并在需要使用到open和stop等操作的时候
dev->netdev_ops = &dys_netdev_ops;
2,在对结构体中的成员变量进行强制转换时,格式:
(unsigned char *)(skb->end)
skb->end是sk_buff_data_t类型, skb->head为unsigned char*类型,进行双目操作时(比如-)需要进行强制类型转换,其实类型是一样的,只是typdef欢乐一个别名:
typedef unsigned char *sk_buff_data_t;
3,顺便普及一下define和typedef的区别:
#define是预处理指令,在编译预处理时进行简单的替换,不作正确性检查,不关含义是否正确照样带入,只有在编译已被展开的源程序时才会发现可能的错误并报错。例如:
#define PI 3.1415926
程序中的:area=PI*r*r 会替换为3.1415926*r*r
如果你把#define语句中的数字9 写成字母g 预处理也照样带入。
typedef:在C/C++语言中,typedef常用来定义一个标识符及关键字的别名,它是语言编译过程的一部分,但它并不实际分配内存空间。
#typedef与#define的区别
从以上的概念便也能基本清楚,typedef只是为了增加可读性而为标识符另起的新名称(仅仅只是个别名),而#define原本在C中是为了定义常量, 到了C++,const、enum、inline的出现使它也渐渐成为了起别名的工具。有时很容易搞不清楚与typedef两者到底该用哪个好,如 #define INT int这样的语句,用typedef一样可以完成,用哪个好呢?我主张用typedef,因为在早期的许多C编译器中这条语句是非法的,只是现今的编译器 又做了扩充。为了尽可能地兼容,一般都遵循#define定义“可读”的常量以及一些宏语句的任务,而typedef则常用来定义关键字、冗长的类型的别 名。
宏定义只是简单的字符串代换(原地扩展),而typedef则不是原地扩展,它的新名字具有一定的封装性,以致于新命名的标识符具有更易定义变量的功能。请看上面第一大点代码的第三行:
typedef (int*) pINT;以及下面这行:#define pINT2 int*
效果相同?实则不同!实践中见差别:pINT a,b;的效果同int *a; int *b;表示定义了两个整型指针变量。而pINT2 a,b;的效果同int *a, b;表示定义了一个整型指针变量a和整型变量b。
注意:两者还有一个行尾;号的区别哦!
netif_rx_complete和netif_rx_schedule属于poll机制中的函数,接口名称已经变为napi_complete和napi_schedule。
在改源程序的过程中总结了一些,记下来(以后估计会忘了):
1,在原net_devide结构体中有open、stop等函数,可以直接调用,但是变更之后,这些操作被封装在net_device_ops中,并且是const的只读类型,下面的操作便会出错:
dev->open = dys_dev_open;
dev->stop = dys_dev_stop;
dev->set_config = dys_dev_set_config;
dev->hard_start_xmit = dys_dev_tx;
dev->do_ioctl = dys_dev_ioctl;
dev->get_stats = dys_dev_get_stats;
dev->tx_timeout = dys_dev_tx_timeout;
要使用这些操作必须建立如下结构体初始化:
static const struct net_device_ops dys_netdev_ops = {
.ndo_open = dys_dev_open,
.ndo_stop = dys_dev_stop,
.ndo_start_xmit = dys_dev_tx,
.ndo_do_ioctl = dys_dev_ioctl,
.ndo_get_stats = dys_dev_get_stats,
.ndo_set_config = dys_dev_set_config,
.ndo_tx_timeout = dys_dev_tx_timeout,
};
并在需要使用到open和stop等操作的时候
dev->netdev_ops = &dys_netdev_ops;
2,在对结构体中的成员变量进行强制转换时,格式:
(unsigned char *)(skb->end)
skb->end是sk_buff_data_t类型, skb->head为unsigned char*类型,进行双目操作时(比如-)需要进行强制类型转换,其实类型是一样的,只是typdef欢乐一个别名:
typedef unsigned char *sk_buff_data_t;
3,顺便普及一下define和typedef的区别:
#define是预处理指令,在编译预处理时进行简单的替换,不作正确性检查,不关含义是否正确照样带入,只有在编译已被展开的源程序时才会发现可能的错误并报错。例如:
#define PI 3.1415926
程序中的:area=PI*r*r 会替换为3.1415926*r*r
如果你把#define语句中的数字9 写成字母g 预处理也照样带入。
typedef:在C/C++语言中,typedef常用来定义一个标识符及关键字的别名,它是语言编译过程的一部分,但它并不实际分配内存空间。
#typedef与#define的区别
从以上的概念便也能基本清楚,typedef只是为了增加可读性而为标识符另起的新名称(仅仅只是个别名),而#define原本在C中是为了定义常量, 到了C++,const、enum、inline的出现使它也渐渐成为了起别名的工具。有时很容易搞不清楚与typedef两者到底该用哪个好,如 #define INT int这样的语句,用typedef一样可以完成,用哪个好呢?我主张用typedef,因为在早期的许多C编译器中这条语句是非法的,只是现今的编译器 又做了扩充。为了尽可能地兼容,一般都遵循#define定义“可读”的常量以及一些宏语句的任务,而typedef则常用来定义关键字、冗长的类型的别 名。
宏定义只是简单的字符串代换(原地扩展),而typedef则不是原地扩展,它的新名字具有一定的封装性,以致于新命名的标识符具有更易定义变量的功能。请看上面第一大点代码的第三行:
typedef (int*) pINT;以及下面这行:#define pINT2 int*
效果相同?实则不同!实践中见差别:pINT a,b;的效果同int *a; int *b;表示定义了两个整型指针变量。而pINT2 a,b;的效果同int *a, b;表示定义了一个整型指针变量a和整型变量b。
注意:两者还有一个行尾;号的区别哦!