netlink ARP table

#include <netinet/ip.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <syslog.h>
#include <fcntl.h>
#include <net/if_arp.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <string.h>
#include <errno.h>
#include <time.h>
#include <sys/uio.h>
#include <bits/sockaddr.h>
#include <asm/types.h>
#include <linux/rtnetlink.h>

FILE *ip4to6;     
char ipv4[INET_ADDRSTRLEN];
char ipv6[INET6_ADDRSTRLEN];
unsigned long long lladdr;

int injectmacipv4()
{
    int fd = socket(PF_NETLINK, SOCK_DGRAM, NETLINK_ROUTE);
    struct sockaddr_nl la;
    memset(&la, 0, sizeof(la));
    la.nl_family = AF_NETLINK;
    la.nl_pad = 0;
    la.nl_pid = getpid();
    la.nl_groups = 0;
    
    bind(fd, (struct sockaddr*) &la, sizeof(la));

    struct {
        struct nlmsghdr n;
        struct ndmsg r;
        char   buf[1024];
    } req2;
    memset(&req2, 0, sizeof(req2));
    int ndmsglen = NLMSG_LENGTH(sizeof(struct ndmsg));

// create the IP attribute            
    struct rtattr *dstattr;
    dstattr = (struct rtattr *) req2.buf;
    dstattr->rta_type = NDA_DST;
    dstattr->rta_len = sizeof(struct rtattr) + 4;
    int intip;
    inet_pton(AF_INET, ipv4, ((char *)dstattr) + sizeof(struct rtattr));
    ndmsglen += dstattr->rta_len;

// create the MAC attribute   
    //struct rtattr *dstattr;
    dstattr = (struct rtattr *) req2.buf + 2;
    dstattr->rta_type = NDA_LLADDR;
    dstattr->rta_len = sizeof(struct rtattr) + 6;
    memcpy(((char *)dstattr) + sizeof(struct rtattr), &lladdr, 6);
    ndmsglen += dstattr->rta_len;

// fill in the netlink message header
    req2.n.nlmsg_len = ndmsglen; //48;
    req2.n.nlmsg_type = RTM_NEWNEIGH;
    req2.n.nlmsg_flags = NLM_F_REQUEST|NLM_F_CREATE|NLM_F_EXCL; //1537
    req2.n.nlmsg_seq;
    req2.n.nlmsg_pid;

// fill in the netlink message NEWNEIGH
    req2.r.ndm_family = AF_INET;
    req2.r.ndm_pad1;
    req2.r.ndm_pad2;
    req2.r.ndm_ifindex = 2; // /sys/class/net/eth2/ifindex
    req2.r.ndm_state = NUD_PERMANENT;
    req2.r.ndm_flags;
    req2.r.ndm_type;

// debug stuff about the packet to be send via NETLINK   
/*
    printf("The sent packet:\n\n");
    printf("NLMSG HEADER:\nnlmsg_len = %i\nnlmsg_type = %i\nnlmsg_flags = %i\nnlmsg_seq = %i\nnlmsg_pid = %i\n\n", req2.n.nlmsg_len, req2.n.nlmsg_type, req2.n.nlmsg_flags, req2.n.nlmsg_seq, req2.n.nlmsg_pid);
    printf("NLMSG MESSAGE:\nndm_family = %i\nndm_pad1 = %i\nndm_pad2 = %i\nndm_ifindex = %i\nndm_state = %i\nndm_flags = %i\nndm_type = %i\n\n", req2.r.ndm_family, req2.r.ndm_pad1, req2.r.ndm_pad2, req2.r.ndm_ifindex, req2.r.ndm_state, req2.r.ndm_flags, req2.r.ndm_type);
    printf("BUF:\n");
    int bufteller = 0;
    while(bufteller < 256)
    {
        printf ("%i ", req2.buf[bufteller]);
        bufteller++;
    }
    printf("\n\n");
 */

    int status2;
    status2 = send(fd, &req2, req2.n.nlmsg_len, 0);
    close (fd);
    return 1;
}

int getmacipv6()
{
    int fd = socket(PF_NETLINK, SOCK_DGRAM, NETLINK_ROUTE);

    struct sockaddr_nl la;
    memset(&la, 0, sizeof(la));
    la.nl_family = AF_NETLINK;
    la.nl_pad = 0;
    la.nl_pid = getpid();
    la.nl_groups = 0;

    bind(fd, (struct sockaddr*) &la, sizeof(la));

//send request via netlink, the message is in req 
    struct {
        struct nlmsghdr n;
        struct ndmsg r;
    } req;
    
    memset(&req, 0, sizeof(req));
    // fill in the netlink message header
    req.n.nlmsg_len = NLMSG_LENGTH(sizeof(struct ndmsg));
    req.n.nlmsg_type = RTM_GETNEIGH;
    req.n.nlmsg_flags = NLM_F_REQUEST | NLM_F_DUMP;
    req.n.nlmsg_seq;
    req.n.nlmsg_pid;// = getpid();

    // fill in the netlink message GETNEIGH
    req.r.ndm_family = AF_INET6;
    req.r.ndm_pad1;
    req.r.ndm_pad2;
    req.r.ndm_ifindex;
    req.r.ndm_state;// = NUD_PERMANENT;
    req.r.ndm_flags;
    req.r.ndm_type;

    int status;

    status = send(fd, &req, req.n.nlmsg_len, 0);
    if (status < 0)    
            printf("RFC 5549: Problem sending to netlink\n");

    //receive data and put it in buf
    char buf[256];
    char *p;
    struct nlmsghdr *nlp;
    memset(buf, 0, sizeof(buf));
    p = buf;
    int nll = 0;
    struct ndmsg *ndp;
    struct rtattr *ndap;
    int ndl;
 
    while(1)
    {
            status = recv(fd, p, sizeof(buf) - nll, 0);
            if (status < 0)
                    printf("RFC 5549: Problem receiving from netlink\n");
            close(fd);
            
            nlp = (struct nlmsghdr *) p;
            if(nlp->nlmsg_type == NLMSG_DONE)
            {
                    break;
            }
            while(1)
            {
                    ndp = (struct ndmsg *) NLMSG_DATA(nlp);     // get RTNETLINK message header
                    ndap = (struct rtattr *) RTM_RTA(ndp);      // get start of attributes
                    ndl = RTM_PAYLOAD(nlp);                             
                    while(1)
                    {
                        if (ndap->rta_type == NDA_DST)
                        {
                            char dstp[INET6_ADDRSTRLEN];
                            memset(dstp, 0, sizeof(dstp));
                            //long long *intippoint = RTA_DATA(ndap);
                            //long long intip = *intippoint;                          
                            
                            //printf("intip = %lli\n", intip);
                            inet_ntop(AF_INET6, RTA_DATA(ndap), dstp, INET6_ADDRSTRLEN);                     
                            //printf("dstp = %s, ipv6 = %s\n", dstp, ipv6);
                            
                            if (strcmp(dstp, ipv6) == 0)
                            {
                                ndap = RTA_NEXT(ndap, ndl);
                                long long *macpoint = RTA_DATA(ndap);
                                lladdr = *macpoint;
                                //printf("lladdr = %llx\n", lladdr);
                                return 2;
                            }  
                             
                        }
                        ndap = RTA_NEXT(ndap, ndl);
                        if (RTA_OK(ndap, ndl) != 1)
                        {
                             break;
                        }  
                        
                    }
                    p += status;
                    nll += status;
                    nlp = NLMSG_NEXT(nlp, nll);
                    if (NLMSG_OK(nlp, nll) != 1)
                    {
                        break;
                    }
            } 
    }
    
    return 1;
}

int readfile(int teller)
{
    ip4to6 = fopen("/usr/local/sbin/ip4to6","r");
    char line[100];
    while (1)
    {   
        int teller2 = 0;
        while (teller2 <= teller)
        {
                fgets(line, 100, ip4to6);
                teller2++;
        }
        char *word;
        word = strtok(line, " ");
        memcpy(ipv6, word, INET6_ADDRSTRLEN);
        word = strtok(NULL, " ");
        word[strlen(word) - 1] = word[strlen(word)];    // remove the last \n in the ipv4 address
        memcpy(ipv4, word, INET_ADDRSTRLEN);
        break;
    }   
    
    fclose(ip4to6);
    return 1;
}

int countlines()
{
    ip4to6 = fopen("/usr/local/sbin/ip4to6","r");
    char c;
    int teller = 0;
    while ((c = fgetc(ip4to6)) != EOF)
    {
        if (c == '\n')
        {
            teller++;
        }
    }
    fclose(ip4to6);
    return teller;
}

void main()
{
    while(1)
    {
        int lines = countlines();
        int teller = 0;
        while (teller < lines)
        {
            lladdr = 0;
            readfile(teller);
            getmacipv6();
           // printf("\n");
            if (lladdr != 0)
            {
                injectmacipv4();
            }
            teller++;
        }
        sleep(5);
    }
}
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Linux Netlink 是一种用于内核与用户空间进行通信的机制。它提供了一种可靠的、高效的跨进程通信方式,用于在内核和用户空间之间传递网络相关的信息和事件。 Netlink 机制通过一组专用的 socket 进行通信,允许用户空间程序发送请求到内核,以获取网络配置、状态和事件等信息,并可以控制网络设备的行为。同时,内核也可以通过 Netlink 通知用户空间应用程序有关网络相关的事件,例如连接建立、断开、路由更新等。 Netlink 消息由一个头部和一个可选的负载组成。头部包含了消息的类型、标志和序列号等信息,而负载则是具体的数据内容。不同的消息类型对应着不同的操作,例如查询配置信息、添加/删除路由、修改网络接口属性等。 Netlink 通信的优势之一是其灵活性和可扩展性。通过定义不同的消息类型,可以实现各种网络管理的功能。例如,可以使用 Netlink获取网络接口的状态和统计信息,监控网络连接的建立和断开,甚至控制网络设备的驱动程序。 此外,Netlink 还支持一种多播通信方式,允许多个用户空间程序同时订阅网络相关的事件。这样,当有事件发生时,内核可以将通知发送给所有订阅者,实现实时的网络状态监控和响应。 总的来说,Linux Netlink 是一种用于内核与用户空间之间进行网络相关信息传递和交互的机制。它提供了灵活、可扩展的通信方式,用于实现网络管理和监控等功能。 ### 回答2: Linux Netlink是用于在Linux内核和用户空间之间进行通信的一种机制。它可以让用户空间程序与内核进行信息交互,如获取系统状态信息、配置网络接口等。 Netlink套接字是通过调用socket()函数创建的,它允许进程使用Netlink协议与内核通信。Netlink协议有多个不同的消息类型,每个类型对应不同的功能。比如,NETLINK_ROUTE用于配置和查询路由信息,NETLINK_INET_DIAG用于获取网络连接的状态信息。 在应用程序中,可以使用套接字的send()和recv()函数发送和接收消息。消息的格式由Netlink通信协议规定,一般包括一个消息头和一个NLMSG_PAYLOAD部分,后者可以包含消息所携带的数据。 对于内核模块来说,可以使用netlink_kernel_create()函数创建一个Netlink套接字,并通过netlink_kernel_recvmsg()函数接收和处理来自用户空间的消息。 通过Netlink机制,用户空间程序可以实时监控内核的状态变化,比如网络接口的变更、路由的变动等。同时,内核也可以通过Netlink向用户空间发送事件通知,以便及时更新相关信息。 总的来说,Linux Netlink提供了一个灵活、高效的通信机制,有效地实现了内核和用户空间之间的信息交互。它在很多情况下被广泛应用,如网络管理工具、性能监控工具等。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值