首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >如何从getifaddr读取子网掩码

如何从getifaddr读取子网掩码
EN

Stack Overflow用户
提问于 2018-07-03 13:05:39
回答 1查看 978关注 0票数 0

使用getifaddrs()函数,我想提取CIDR格式的网络ip和掩码(即24而不是255.255.255.0),这与其他问题不同。但我在面具部分遇到了困难。

例如:

192.168.0.114/24

到目前为止,这就是我一直在做的事情。但我一定是读错了网络掩码。这是完全错误的。

代码语言:javascript
复制
#include <stdio.h>
#include <sys/types.h>
#include <ifaddrs.h>
#include <netinet/in.h>
#include <string.h>
#include <arpa/inet.h>

unsigned int cidrMask(unsigned int n) {
    unsigned int count = 0;
    while (n) {
        count += n & 1;
        n >>= 1;
    }
    return 32-count;
}


int main (int argc, const char * argv[]) {
    struct ifaddrs * ifAddrStruct=NULL;
    struct ifaddrs * ifa=NULL;
    void * tmpAddrPtr=NULL;
    unsigned int tmpMask;

    getifaddrs(&ifAddrStruct);

    for (ifa = ifAddrStruct; ifa != NULL; ifa = ifa->ifa_next) {
        if (!ifa->ifa_addr) {
            continue;
        }

        if (ifa->ifa_addr->sa_family == AF_INET) { // check it is IP4
            // is a valid IP4 Address
            tmpAddrPtr=&((struct sockaddr_in *)ifa->ifa_addr)->sin_addr;
            memcpy(&tmpMask, &(*(struct sockaddr_in *)&ifa->ifa_netmask).sin_addr, 4);

            char addressBuffer[INET_ADDRSTRLEN];
            inet_ntop(AF_INET, tmpAddrPtr, addressBuffer, INET_ADDRSTRLEN);
            printf("%s IP Address %s, mask=%u, %u\n", ifa->ifa_name, addressBuffer, ifa->ifa_netmask, cidrMask(tmpMask));
                printf("tmpmask=%d\n", tmpMask);
        }
    }
    if (ifAddrStruct!=NULL) freeifaddrs(ifAddrStruct);
    return 0;
}

所以网络掩码的输出是完全错误的。

代码语言:javascript
复制
lo IP Address 127.0.0.1, mask=3106544572, 22
tmpmask=21982
enp0s25 IP Address 192.168.0.114, mask=3106544756, 22
tmpmask=21982
docker0 IP Address 172.17.0.1, mask=3106544940, 22
tmpmask=21982

如何读取网络掩码并生成正确的输出?

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2018-07-03 15:51:44

您对tmpMask的赋值是错误的,因为您的类型转换ifa_netmask是错误的,所以您正在通过无效的指针访问它的sin_addr成员。

您将获取ifa_netmask的地址并将其类型转换为sockaddr_in*,但您需要改为类型转换其值,就像对ifa_addr所做的那样

代码语言:javascript
复制
memcpy(&tmpMask, &(((struct sockaddr_in *)(ifa->ifa_netmask))->sin_addr.s_addr), 4); 

这可以通过删除memcpy()来简化

代码语言:javascript
复制
tmpMask = ((struct sockaddr_in *)(ifa->ifa_netmask))->sin_addr.s_addr;

当您打印ifa_netmask时,您将打印出它的原始指针值(使用%u,这对于指针来说是未定义的行为)。您应该像格式化ifa_addr一样将ifa_netmask格式化为人类可读的字符串,因为它们都是指向sockaddr_in结构的指针:

代码语言:javascript
复制
char addressBuffer[INET_ADDRSTRLEN];
char maskBuffer[INET_ADDRSTRLEN];

tmpAddrPtr = &((struct sockaddr_in *)(ifa->ifa_addr))->sin_addr;
inet_ntop(AF_INET, tmpAddrPtr, addressBuffer, INET_ADDRSTRLEN);

tmpAddrPtr = &((struct sockaddr_in *)(ifa->ifa_netmask))->sin_addr;
inet_ntop(AF_INET, tmpAddrPtr, maskBuffer, INET_ADDRSTRLEN);

tmpMask = ((struct sockaddr_in *)(ifa->ifa_netmask))->sin_addr.s_addr;

printf("%s IP Address %s, Mask %s, %u\n", ifa->ifa_name, addressBuffer, maskBuffer, cidrMask(tmpMask));
printf("tmpmask=%u\n", tmpMask);
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/51146999

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档