检查这是否为空

时间:2016-03-15 03:03:30

标签: c struct arguments

我使用这行代码作为代码段的最后一部分,在Linux上打印接口的IP地址。

printf ("%s\n", inet_ntoa (((struct sockaddr_in *) &ifr.ifr_addr)->sin_addr));

这很好用,但是如果我不输入接口名作为参数,我就会出现分段错误。

有没有办法检查这段代码:

inet_ntoa (((struct sockaddr_in *) &ifr.ifr_addr)->sin_addr)

是否提供了参数?感谢。

以下是上下文的完整源代码。

/*
*  This program is free software: you can redistribute it and/or modify
*  it under the terms of the GNU General Public License as published by
*  the Free Software Foundation, either version 3 of the License, or
*  (at your option) any later version.
*
*  This program is distributed in the hope that it will be useful,
*  but WITHOUT ANY WARRANTY; without even the implied warranty of
*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
*  GNU General Public License for more details.
*
*  You should have received a copy of the GNU General Public License
*  along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

/********************************************************************
* Description:
* Author: John Cartwright <>
* Created at: Wed Jan 13 11:44:29 AEDT 2016
* Computer: ubuntu
* System: Linux 4.3.0-5-generic on x86_64
*
* Copyright (c) 2016 John Cartwright,,,  All rights reserved.
*
********************************************************************/

#include <linux/kernel.h>       /* for struct sysinfo */
#include <sys/sysinfo.h>        /* For the sysinfo struct. */
#include <stdio.h>
#include <string.h>     /* for strncpy */
#include <unistd.h>     /* for close */
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <netinet/in.h>
#include <net/if.h>
#include <stdlib.h> /* For the system() function. */
#include <arpa/inet.h>

#include "info.h"

#define BUF 0x05

int main (int argc, char **argv)
{
    int fd;
    struct ifreq ifr;
    char *myarg1 = argv[2];
    char hostname[128];
    char *iface = myarg1;
    unsigned char *mac;
    char command[50];

    if (argv[1] == NULL)
    {
        printf ("********************************************\n");
        printf ("*          Simple system information.      *\n");
        printf ("*                                          *\n");
        printf ("* IP info: --ip <IFACE>                    *\n");
        printf ("* Print ARP table: --arp                   *\n");
        printf ("* Print sys info: --system                 *\n");
        printf ("********************************************\n");
        printf ("\nThe PID of this program was: %i\n", getpid ());
    }

    /* Getting information about the routing table. */

    if (argc > 1 && strncmp (argv[1], "--arp", BUF) == 0)
    {
        printf ("This is the routing table.\n");
        char *arg[] = { "cat", "/proc/net/arp", NULL };
        execvp (arg[0], arg);
    }

    if (argc > 1 && strncmp (argv[1], "--system", BUF) == 0)
    {
        information();
    }


    if (argc > 1 && strncmp (argv[1], "--hostname", BUF) == 0)
    {
        gethostname(hostname, sizeof hostname);
        printf("Hostname: %s\n", hostname);
    }

    if (argc > 1 && strncmp (argv[1], "--ip", BUF) == 0)
    {
        // This code from: http://www.binarytides.com/c-program-to-get-mac-address-from-interface-name-on-linux/
        FILE *f;
        char line[100] , *p , *c;
        char myip;

        f = fopen("/proc/net/route" , "r");

        while(fgets(line , 100 , f))
        {
            p = strtok(line , " \t");
            c = strtok(NULL , " \t");

            if(p!=NULL && c!=NULL)
            {
                if(strcmp(c , "00000000") == 0)
                {
                    printf("Default interface is : %s \n" , p);
                    break;
                }
            }
        }

        fd = socket(AF_INET, SOCK_DGRAM, 0);

        ifr.ifr_addr.sa_family = AF_INET;
        strncpy(ifr.ifr_name , iface , IFNAMSIZ-1);

        ioctl(fd, SIOCGIFHWADDR, &ifr);

        close(fd);

        mac = (unsigned char *)ifr.ifr_hwaddr.sa_data;

        //display mac address
        printf("Mac : %.2x:%.2x:%.2x:%.2x:%.2x:%.2x\n" , mac[0], mac[1], mac[2], \
        mac[3], mac[4], mac[5]);

        fd = socket (AF_INET, SOCK_DGRAM, 0);
        /* I want to get an IPv4 IP address */
        ifr.ifr_addr.sa_family = AF_INET;

        /* I want IP address attached to the specified interface. */
        strncpy (ifr.ifr_name, myarg1, IFNAMSIZ - 1);

        ioctl (fd, SIOCGIFADDR, &ifr);

        close (fd);

        // End binarytides code.

        /* display result */
        printf ("IP information.\n");
        printf("Gateway IP:\n");

        strcpy( command, "/sbin/ip route | awk '/default/ { print $3 }'" );
        system(command);

        printf("IP address:\n");
        printf ("%s\n", inet_ntoa (((struct sockaddr_in *) &ifr.ifr_addr)->sin_addr));
    }

    return 0;
}

2 个答案:

答案 0 :(得分:0)

我怀疑问题是sin_add为NULL,但被解除引用,这就是导致段错误的原因。

从病房开始向后工作,就像M.M建议的那样。

答案 1 :(得分:0)

你的问题应该是我怎么知道命令名后至少有2个参数

argc > 1更改为argc >= 3。那么参数是您的程序名称"--ip"和接口名称?

if (argc >= 3 && strncmp (argv[1], "--ip", BUF) == 0)

否则,您可以programname --ip执行该程序,该程序将与SIGSEGV一起崩溃。这不是你要问的吗?