Linux 内核态 Socket TCP 编程


1.内核态 socket API

内核态socket编程的过程和用户态下的socket编程流程一样,但是接口不同。Kernel提供了一组内核态的socket API,基本上在用户态的sockt API在内核中都有对应的API。 在net/socket.c中可以看到如下导出符号:

  1. EXPORT_SYMBOL(kernel_sendmsg);    
  2. EXPORT_SYMBOL(kernel_recvmsg);    
  3. EXPORT_SYMBOL(sock_create_kern);    
  4. EXPORT_SYMBOL(sock_release);    
  5. EXPORT_SYMBOL(kernel_bind);    
  6. EXPORT_SYMBOL(kernel_listen);    
  7. EXPORT_SYMBOL(kernel_accept);    
  8. EXPORT_SYMBOL(kernel_connect);    
  9. EXPORT_SYMBOL(kernel_getsockname);    
  10. EXPORT_SYMBOL(kernel_getpeername);    
  11. EXPORT_SYMBOL(kernel_getsockopt);    
  12. EXPORT_SYMBOL(kernel_setsockopt);    
  13. EXPORT_SYMBOL(kernel_sendpage);    
  14. EXPORT_SYMBOL(kernel_sock_ioctl);    
  15. EXPORT_SYMBOL(kernel_sock_shutdown);   

下面以两个内核模块间利用socket进行通信为例来讲述内核态下socket编程。

server内核模块功能是:建立套接字,绑定端口,监听端口,等待client的连接,并接收client发来的数据。

client内核模块功能是:建立套接字,和server建立连接,并向server发送数据。

2. server模块

  1. /*server.c*/  
  2. #include<linux/in.h>  
  3. #include<linux/inet.h>  
  4. #include<linux/socket.h>  
  5. #include<net/sock.h>  
  6.   
  7. #include<linux/init.h>  
  8. #include<linux/module.h>  
  9.   
  10. int myserver(void){  
  11.   
  12.         struct socket *sock,*client_sock;  
  13.         struct sockaddr_in s_addr;  
  14.         unsigned short portnum=0x8888;  
  15.         int ret=0;  
  16.   
  17.         memset(&s_addr,0,sizeof(s_addr));  
  18.         s_addr.sin_family=AF_INET;  
  19.         s_addr.sin_port=htons(portnum);  
  20.         s_addr.sin_addr.s_addr=htonl(INADDR_ANY);  
  21.   
  22.   
  23.         sock=(struct socket *)kmalloc(sizeof(struct socket),GFP_KERNEL);  
  24.         client_sock=(struct socket *)kmalloc(sizeof(struct socket),GFP_KERNEL);  
  25.   
  26.         /*create a socket*/  
  27.         ret=sock_create_kern(AF_INET, SOCK_STREAM,0,&sock);  
  28.         if(ret){  
  29.                 printk("server:socket_create error!\n");  
  30.         }  
  31.         printk("server:socket_create ok!\n");  
  32.   
  33.         /*bind the socket*/  
  34.         ret=sock->ops->bind(sock,(struct sockaddr *)&s_addr,sizeof(struct sockaddr_in));  
  35.         if(ret<0){  
  36.                 printk("server: bind error\n");  
  37.                 return ret;  
  38.         }  
  39.         printk("server:bind ok!\n");  
  40.   
  41.         /*listen*/  
  42.         ret=sock->ops->listen(sock,10);  
  43.         if(ret<0){  
  44.                 printk("server: listen error\n");  
  45.                 return ret;  
  46.         }  
  47.         printk("server:listen ok!\n");  
  48.   
  49.         ret=sock->ops->accept(sock,client_sock,10);if(ret<0){  
  50.                 printk("server:accept error!\n");  
  51.                 return ret;  
  52.         }  
  53.   
  54.         printk("server: accept ok, Connection Established\n");  
  55.   
  56.         /*kmalloc a receive buffer*/  
  57.         char *recvbuf=NULL;  
  58.         recvbuf=kmalloc(1024,GFP_KERNEL);  
  59.         if(recvbuf==NULL){  
  60.                 printk("server: recvbuf kmalloc error!\n");  
  61.                 return -1;  
  62.         }  
  63.         memset(recvbuf, 0, sizeof(recvbuf));  
  64.   
  65.         /*receive message from client*/  
  66.         struct kvec vec;  
  67.         struct msghdr msg;  
  68.         memset(&vec,0,sizeof(vec));  
  69.         memset(&msg,0,sizeof(msg));  
  70.         vec.iov_base=recvbuf;  
  71.         vec.iov_len=1024;  
  72.         ret=kernel_recvmsg(client_sock,&msg,&vec,1,1024,0); /*receive message*/  
  73.         printk("receive message:\n %s\n",recvbuf);  
  74.           
  75.     /*release socket*/  
  76.     sock_release(sock);  
  77.         sock_release(client_sock);  
  78.         return ret;  
  79. }  
  80.   
  81. static int server_init(void){  
  82.         printk("server init:\n");  
  83.         return (myserver());  
  84. }  
  85.   
  86. static void server_exit(void){  
  87.         printk("good bye\n");  
  88. }  
  89.   
  90. module_init(server_init);  
  91. module_exit(server_exit);  
  92.   
  93. MODULE_LICENSE("GPL");  

3.client模块 

  1. /*client.c*/  
  2. #include<linux/in.h>  
  3. #include<linux/inet.h>  
  4. #include<linux/socket.h>  
  5. #include<net/sock.h>  
  6.   
  7. #include<linux/init.h>  
  8. #include<linux/module.h>  
  9.   
  10. int myclient(void){  
  11.         struct socket *sock;  
  12.         struct sockaddr_in s_addr;  
  13.         unsigned short portnum=0x8888;  
  14.         int ret=0;  
  15.   
  16.         memset(&s_addr,0,sizeof(s_addr));  
  17.         s_addr.sin_family=AF_INET;  
  18.         s_addr.sin_port=htons(portnum);  
  19.        
  20.         s_addr.sin_addr.s_addr=in_aton("192.168.209.134"); /*server ip is 192.168.209.134*/  
  21.         sock=(struct socket *)kmalloc(sizeof(struct socket),GFP_KERNEL);  
  22.   
  23.         /*create a socket*/  
  24.         ret=sock_create_kern(AF_INET, SOCK_STREAM,0,&sock);  
  25.         if(ret<0){  
  26.                 printk("client:socket create error!\n");  
  27.                 return ret;  
  28.         }  
  29.         printk("client: socket create ok!\n");  
  30.   
  31.         /*connect server*/  
  32.         ret=sock->ops->connect(sock,(struct sockaddr *)&s_addr, sizeof(s_addr),0);  
  33.         if(ret!=0){  
  34.                 printk("client:connect error!\n");  
  35.                 return ret;  
  36.         }  
  37.         printk("client:connect ok!\n");  
  38.   
  39.         /*kmalloc sendbuf*/  
  40.         char *sendbuf=NULL;  
  41.         sendbuf=kmalloc(1024,GFP_KERNEL);  
  42.         if(sendbuf==NULL){  
  43.                 printk("client: sendbuf kmalloc error!\n");  
  44.                 return -1;  
  45.         }  
  46.         memset(sendbuf,1,1024);          
  47.           
  48.     struct kvec vec;  
  49.         struct msghdr msg;  
  50.   
  51.         vec.iov_base=sendbuf;  
  52.         vec.iov_len=1024;  
  53.   
  54.         memset(&msg,0,sizeof(msg));  
  55.   
  56.         ret=kernel_sendmsg(sock,&msg,&vec,1,1024); /*send message */  
  57.         if(ret<0){  
  58.                 printk("client: kernel_sendmsg error!\n");  
  59.                 return ret;  
  60.         }else if(ret!=1024){  
  61.                 printk("client: ret!=1024");  
  62.         }  
  63.         printk("client:send ok!\n");  
  64.   
  65.         return ret;  
  66. }  
  67.   
  68. static int client_init(void){  
  69.         printk("client:init\n");  
  70.         return (myclient());  
  71. }  
  72.   
  73. static void client_exit(void){  
  74.         printk("client exit!\n");  
  75. }  
  76.   
  77. module_init(client_init);  
  78. module_exit(client_exit);  
  79. MODULE_LICENSE("GPL");  
  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值