Kernel与用户进程通信
测试IPv6 ready logo rfc 3315的时候,遇到一个问题,要求在收到ICMPv6 RA的时候,DHCPv6 Client要发Solicit消息。在平常的应用中,都是启动DHCPv6 Client后,client每隔一段时间就会发送Solicit,而并不是要在收到RA的时候发送。于是就修改了下kernel和client的代码,使得在kernel收到RA消息时,发送消息给DHCPv6 Client,使client发送Solicit。
思路为:使用netlink_kernel_create建立消息机制。在DHCPv6 Client初始化的时候,新建类型为NETLINK_RTK_ICMP6_RA 的netlink socket,监听kernel的消息,并且把自己的pid发送给kernel, kernel 收到消息后,存储pid为全局变量。当kernel收到RA包后,将发送消息给pid的用户进程。用户进程收到消息,发送solicit。
Kernel中接收ICMPv6 RA的代码在linux-3.10/net/ipv6/ndisc.c。
Kernel:
- 初始化IPC
在ndisc_init()中加上进程通信的socket。
//create netlink socket
struct netlink_kernel_cfg cfg = {
.input = nl_icmp_input,
};
printk("%s: before netlink_kernel_create\n", __func__);
icmp6_ra_sock = netlink_kernel_create(&init_net, NETLINK_RTK_ICMP6_RA, &cfg);
if (icmp6_ra_sock == NULL) {
printk(KERN_ERR "Netlink[Kernel] Cannot create netlink socket for ipv6 ra.\n");
return -EAFNOSUPPORT;
}
printk("Netlink[Kernel] create netlink socket for ipv6 ra ok.\n");
其中NETLINK_RTK_ICMP6_RA 为在netlink.h中定义的一个类型。nl_icmp_input用于接收用户层发来的信息。
2. 接收DHCPv6 Client发来的pid 信息
static void nl_icmp_input(struct sk_buff *__skb)
{
struct sk_buff *skb;
struct nlmsghdr *nlh;
char str[100];
struct completion cmpl;
int i=10;
skb = skb_get (__skb);
if(skb->len >= NLMSG_SPACE(0))
{
nlh = nlmsg_hdr(skb);
memcpy(str, NLMSG_DATA(nlh), sizeof(str));
printk("Message received:%s\n",str) ;
DHCP6C_PID = nlh->nlmsg_pid;
printk("DHCP PID = %d:%s\n", DHCP6C_PID) ;
kfree_skb(skb);
}
return;
}
3. 收到RA后,发消息给用户进程
在ndisc_router_discovery中,确认RA包ok后,发送消息。
if (icmp6_ra_sock) {
printk("%s: ra socket is not null\n", __func__);
payloadLen = NLMSG_SPACE(payloadLen);
/*Alloc skb ,this check helps to call the fucntion from interrupt context */
if(in_atomic())
{
skb_ra = alloc_skb(payloadLen, GFP_ATOMIC);
}
else
{
skb_ra = alloc_skb(payloadLen, GFP_KERNEL);
}
if(!skb_ra)
{
printk(KERN_ERR "failed to alloc skb in %s",__FUNCTION__);
return;
}
nl_msgHdr = (struct nlmsghdr *)skb_ra->data;
nl_msgHdr->nlmsg_type = MESSAGE_ICMP6_RA;
nl_msgHdr->nlmsg_pid=0;/*from kernel */
nl_msgHdr->nlmsg_len = payloadLen;
nl_msgHdr->nlmsg_flags =0;
NETLINK_CB(skb_ra).portid = 0; /*from kernel */
skb_ra->len = payloadLen;
err = netlink_unicast(icmp6_ra_sock, skb_ra, DHCP6C_PID, MSG_DONTWAIT); //send to dhcp6c
if (err < 0)
printk("%s: nfnetlink_unicast icmpv6 ra socket fail\n", __func__);
else
printk("%s: nfnetlink_unicast icmpv6 ra socket ok\n", __func__);
}
用户层:
- Dhcp6c 初始化,并发送pid给kernel
int init_ra_monitor_fd()
{
int ret = 0;
struct sockaddr_nl src_addr, dest_addr;
struct nlmsghdr *nlh = NULL;
struct iovec iov;
struct msghdr msg;
int state_smg = 0;
int pid = getpid();
if ((rasock = socket(AF_NETLINK, SOCK_RAW, NETLINK_RTK_ICMP6_RA)) < 0)
{
printf("Could not open netlink socket for kernel monitor\n");
return 1;
}
else
{
printf("kernelMonitorFd=%d\n", rasock);
}
memset(&src_addr, 0, sizeof(src_addr));
src_addr.nl_family = AF_NETLINK;
src_addr.nl_pid = pid;
src_addr.nl_groups = 0; // multicast
if (bind(rasock, (struct sockaddr *)&src_addr, sizeof(src_addr)) < 0)
{
printf("Could not bind netlink socket for kernel monitor\n");
close(rasock);
rasock = -1;
return 1;
}
/* Tell kernel dhcp6c pid, kernel will send messages to dhcp6c when it receive ICMPv6 RA */
/* Fix IPv6 Ready logo rfc3315 */
memset(&dest_addr, 0, sizeof(dest_addr));
dest_addr.nl_family = AF_NETLINK;
dest_addr.nl_pid = 0;
dest_addr.nl_groups = 0;
nlh = (struct nlmsghdr *)malloc(NLMSG_SPACE(MAX_PAYLOAD));
if(!nlh){
printf("malloc nlmsghdr error!\n");
close(rasock);
return -1;
}
nlh->nlmsg_len = NLMSG_SPACE(MAX_PAYLOAD);
nlh->nlmsg_pid = pid;
nlh->nlmsg_flags = 0;
sprintf(NLMSG_DATA(nlh), "%d", pid);
iov.iov_base = (void *)nlh;
iov.iov_len = NLMSG_SPACE(MAX_PAYLOAD);
memset(&msg, 0, sizeof(msg));
msg.msg_name = (void *)&dest_addr;
msg.msg_namelen = sizeof(dest_addr);
msg.msg_iov = &iov;
msg.msg_iovlen = 1;
printf("state_smg\n");
state_smg = sendmsg(rasock, &msg, 0);
if(state_smg == -1)
{
printf("get error sendmsg = %s\n",strerror(errno));
}
}
2. 等待接收消息
// add to read sockets
if (rasock >= 0) {
FD_SET(rasock, &r);
maxsock = (rasock > maxsock) ? rasock : maxsock;
}
//wait
ret = select(maxsock + 1, &r, NULL, NULL, w);
if (FD_ISSET(rasock, &r)) {
printf("%s: have receive the ra message\n", __func__);
dhcp6_ra_recv(); //send solicit
}
Kernel与用户进程通信的更多相关文章
- 图文详解 Android Binder跨进程通信机制 原理
图文详解 Android Binder跨进程通信机制 原理 目录 目录 1. Binder到底是什么? 中文即 粘合剂,意思为粘合了两个不同的进程 网上有很多对Binder的定义,但都说不清楚:Bin ...
- Linux间的进程通信;以及子进程的创建
"-----第六天-----------------------------------------------------------------------------" .版 ...
- Android随笔之——跨进程通信(一) Activity篇
在Android应用开发中,我们会碰到跨进程通信的情况,例如:你用QQ通讯录打电话的时候会调用系统的拨号应用.某些新闻客户端可以将新闻分享到QQ.微信等应用,这些都是跨进程通信的情况.简而言之,就是一 ...
- linux之间进程通信
进程间通信方式: 同主机进程间数据交换机制: pipe(无名管道) / fifo(有名管道)/ message queue(消息队列)和共享内存. 必备基础: f ...
- MINIX3 进程通信分析
MINIX3 进程通信分析 6.1MINIX3 进程通信概要 MINIX3 的进程通信是 MINIX3 内核部分最重要的一个部件,我个人认为其实这 是内核中的“内核”,怎么来理解这个概念呢?其实 MI ...
- Linux系统编程@进程通信(一)
进程间通信概述 需要进程通信的原因: 数据传输 资源共享 通知事件 进程控制 Linux进程间通信(IPC)发展由来 Unix进程间通信 基于System V进程间通信(System V:UNIX系统 ...
- [转]WINDOW进程通信的几种方式
windows进程通信的几种方式 1 文件映射 文件映射(Memory-Mapped Files)能使进程把文件内容当作进程地址区间一块内存那样来对待.因此,进程不必使用文件I/O操作,只需简单的指针 ...
- Unix 进程通信基本概念
一:通信分为两类: 控制信息的传递: 低级通信 大批量数据的传递: 高级通信 二:基本的通信方式 (a)主从式通信: 通信的双方存在一种隶属关系, 其中主进程是通信过程的控制者,而从进程是通信过程的从 ...
- [置顶] 简单解析linux下进程通信方法
linux下的进程通信手段基本上是从Unix平台上的进程通信手段继承而来的.而对Unix发展做出重大贡献的两大主力AT&T的贝尔实验室及BSD(加州大学伯克利分校的伯克利软件发布中心)在进程间 ...
随机推荐
- iOS-UI控件之UITableView(三)- 自定义不等高的cell
Storyboard_不等高 对比自定义等高cell,需要几个额外的步骤(iOS8开始才支持) 添加子控件和contentView之间的间距约束 设置tableViewCell的真实行高和估算行高 / ...
- SQL SERVER的数据类型
1.SQL SERVER的数据类型 数据类弄是数据的一种属性,表示数据所表示信息的类型.任何一种计算机语言都定义了自己的数据类型.当然,不同的程序语言都具有不同的特点,所定义的数据类型的各类和名称都或 ...
- 关于Qt模态框总汇
转载请注明出处:http://www.cnblogs.com/dachen408/p/7285710.html 父窗体为QMainWindow: 当子窗体为: 1.QWidget,需要设置 this- ...
- java规范与标准?
所谓规范,即指由很多人同时遵守的行为或理论. java的规范并不是指其中一种,而是有很多种,比如java编码规范,java命名规范,java虚拟机规范等等,甚至于一个编码规范都有很多种,不同的公司.组 ...
- UI开发复杂度度量
1)要素的个数: 2)要素布局和渲染的复杂度: 3)交互的复杂度. 本质上分为两种:要素的复杂度和联系的复杂度. 联系包含要素间布局的联系与交互的联系,已经和外部上下文的联系.
- 使用python划分数据集
无论是训练机器学习或是深度学习,第一步当然是先划分数据集啦,今天小白整理了一些划分数据集的方法,希望大佬们多多指教啊,嘻嘻~ 首先看一下数据集的样子,flower_data文件夹下有四个文件夹,每个文 ...
- lua 函数练习
--逻辑表达式 --1+2+3+...+n function fun1(n) ,n do sum = sum + i end return sum end -- 计算奇数和 function fun2 ...
- ArrayList集合的特点和几种遍历方法
public class temp { public static void main(String[] args)throws Exception { ArrayList 在定义时长度为空 ,在新增 ...
- CAD参数绘制角度标注(com接口)
主要用到函数说明: _DMxDrawX::DrawDimAngular 绘制一个角度标注.详细说明如下: 参数 说明 DOUBLE dAngleVertexX 角度标注的顶点的X值 DOUBLE dA ...
- ansible基础配置
1.基础配置 1.1.环境 主机配置 ansible版本:2.7.4 控制端:centos7.4,IP:192.168.1.213,主机名:operation 被控制端: centos6.5,IP:1 ...