向内核注册一个新的协议族

 说到注册新的协议族(下文用family描述),我们需要再次描述一下数组net_families,这个数组共有NPROTO(32)项,其每一项都固定分配给一个family使用。比如,AF_INET(因特网协议)固定占用net_families[2],如果net_families[2]== NULL,则表示当前内核没有AF_INET模块,不支持因特网协议。net_families的每一项是一个结构体指针,指向一个描述family的结构体: struct net_proto_family { int family; int (*create) (struct socket * sock, int protocol); short authentication; short encryption; short encrypt_net; struct module *owner; }; 成员变量family用于描述family类型,成员函数create指向该family的创建函数,用于创建一个socket。 net_families数组当前总共被使用掉了28项(包括第0项保留,其中我们熟悉的UNIX域协议占第1项,因特网协议占第2项),还有 27,28,29,30项未使用。所以,我们可以利用这四个空项编译新的模块,向内核注释新的family。 我们现在的目标是克隆一个AF_INET,通过抄写,调试代码,熟悉其实现原理,直至能够提出精简,优化方案。所以,我们定义: #define MY_PF_INET 30 使用第30项,重新写一个AF_INET,下面是模块的初始代码: #include #include #include #include "my_inet.h" static int myinet_create(struct socket *sock, int protocol) { printk(KERN_INFO "register myinet family successed!\n"); return -ESOCKTNOSUPPORT; } static struct net_proto_family myinet_family_ops = { .family = MY_PF_INET, .create = myinet_create, .owner = THIS_MODULE, }; static int __init myinet_init(void) { (void)sock_register( &myinet_family_ops ); return 0; } static void __exit myinet_exit(void) { } module_init( myinet_init ) module_exit( myinet_exit ) 模块初始化函数myinet_init的主要作用是把myinet_family_ops的地址赋给net_families数组的第30项,让内核看到该family的存在。函数sock_register完成这个功能。insmod之后,内核的日志输出: NET: Registered protocol family 30 表示注册成功。 成功之后,我们测试一下其功能,写一个简单的应用程序: #include #include int main() { socket( 30, 1, 1 ); return 0; } socket系统调用的执行流程: 1、调用sys_socket (net/socket.c 1180行) 2、调用__create_socket (net/socket.c 1069行) 3、调用net_families[family]->create,即我们的myinet_create函数,由于当前没有实现,只简单返回不支持,但我们可以看到日志输出,表示确行执行到了这里。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值