struct nlmsghdr struct msghdr

我们从一个实际的数据包发送的例子入手,来看看其发送的具体流程,以及过程中涉及到的相关数据结构。在我们的虚拟机上发送icmp回显请求包,ping另一台主机172.16.48.1。我们使用系统调用sendto发送这个icmp包。     ssize_t sendto(int s, const void *buf, size_t len, int flags,              
摘要由CSDN通过智能技术生成

我们从一个实际的数据包发送的例子入手,来看看其发送的具体流程,以及过程中涉及到的相关数据结构。在我们的虚拟机上发送icmp回显请求包,ping另一台主机172.16.48.1。我们使用系统调用sendto发送这个icmp包。

     ssize_t sendto(int s, const void *buf, size_t len, int flags,

                        const struct sockaddr *to, socklen_t tolen);

    系统调用sendto最终调用内核函数 asmlinkage long sys_sendto(int fd, void __user * buff, size_t len, unsigned flags, struct sockaddr __user *addr, int addr_len)

    sys_sendto构建一个结构体struct msghdr,用于接收来自应用层的数据包 ,下面是结构体struct msghdr的定义:

        struct msghdr {

            void            *msg_name;//存数据包的目的地址,网络包指向sockaddr_in

                                                   //向内核发数据时,指向sockaddr_nl

            int             msg_namelen;//地址长度

            struct iovec    *msg_iov;

            __kerne

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,您想了解如何通过RTM_GETROUTE获取路由表。RTM_GETROUTE是Linux内核中用于获取路由表的一种Netlink消息类型。要使用它,您需要使用Socket编程接口中的Netlink Socket,向内核发送RTM_GETROUTE消息,并解析内核返回的路由表信息。 以下是获取路由表的基本步骤: 1. 创建Netlink Socket: ```c int netlink_fd; struct sockaddr_nl src_addr; memset(&src_addr, 0, sizeof(src_addr)); src_addr.nl_family = AF_NETLINK; if ((netlink_fd = socket(AF_NETLINK, SOCK_RAW, NETLINK_ROUTE)) < 0) { perror("socket"); exit(EXIT_FAILURE); } if (bind(netlink_fd, (struct sockaddr *)&src_addr, sizeof(src_addr)) < 0) { perror("bind"); close(netlink_fd); exit(EXIT_FAILURE); } ``` 2. 构造RTM_GETROUTE消息: ```c struct { struct nlmsghdr nlh; struct rtmsg rt; } req; memset(&req, 0, sizeof(req)); req.nlh.nlmsg_len = NLMSG_LENGTH(sizeof(struct rtmsg)); req.nlh.nlmsg_type = RTM_GETROUTE; req.nlh.nlmsg_flags = NLM_F_REQUEST | NLM_F_DUMP; req.nlh.nlmsg_seq = 1; req.rt.rtm_family = AF_INET; req.rt.rtm_table = RT_TABLE_MAIN; req.rt.rtm_protocol = RTPROT_UNSPEC; ``` 3. 发送RTM_GETROUTE消息并接收内核返回的消息: ```c struct msghdr msg; struct iovec iov; memset(&msg, 0, sizeof(msg)); memset(&iov, 0, sizeof(iov)); iov.iov_base = &req; iov.iov_len = req.nlh.nlmsg_len; msg.msg_name = &src_addr; msg.msg_namelen = sizeof(src_addr); msg.msg_iov = &iov; msg.msg_iovlen = 1; if (sendmsg(netlink_fd, &msg, 0) < 0) { perror("sendmsg"); close(netlink_fd); exit(EXIT_FAILURE); } char buf[4096]; struct nlmsghdr *nlh; while (1) { int len = recv(netlink_fd, buf, sizeof(buf), 0); if (len < 0) { perror("recv"); close(netlink_fd); exit(EXIT_FAILURE); } for (nlh = (struct nlmsghdr *)buf; NLMSG_OK(nlh, len); nlh = NLMSG_NEXT(nlh, len)) { // 处理内核返回的消息 } if (nlh->nlmsg_type == NLMSG_DONE) break; } ``` 4. 解析内核返回的路由表信息: ```c struct rtmsg *rtm; struct rtattr *rta; int rt_len; rtm = (struct rtmsg *) NLMSG_DATA(nlh); rt_len = nlh->nlmsg_len - NLMSG_LENGTH(sizeof(struct rtmsg)); for (rta = RTM_RTA(rtm); RTA_OK(rta, rt_len); rta = RTA_NEXT(rta, rt_len)) { switch (rta->rta_type) { case RTA_DST: // 目的地址 break; case RTA_GATEWAY: // 网关地址 break; case RTA_OIF: // 输出接口 break; case RTA_PRIORITY: // 优先级 break; // 其他属性... } } ``` 以上是基本的流程,您可以根据具体需求修改RTM_GETROUTE消息的参数,以获取特定的路由表信息。希望能够帮到您!

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值