inet_ntop причина ошибки сегментации - PullRequest
0 голосов
/ 07 октября 2019

У меня проблема. У меня есть код, где мне нужно преобразовать доменное имя в IP. Для этого я использую функцию getaddrinfo (). Но кажется, что когда я вызываю другую функцию (ПОСЛЕ GETADDRINFO ()), моя программа падает (ошибка возврата getaddrinfo и генерируется ошибка сегментации). Я обнаружил, что проблема заключается в вызове функции inet_ntop () после вызова getaddrinfo (). Но я не знаю, почему происходит такое странное поведение ... Вот мой код:

#include<stdio.h>   //scanf , printf
#include<string.h>  //strtok
#include<stdlib.h>  //realloc
#include<sys/socket.h>  //socket
#include<netinet/in.h> //sockaddr_in
#include<arpa/inet.h>   //getsockname
#include<netdb.h>   //hostent
#include<unistd.h>  //close
#include <getopt.h> //getopt


#define DEFAULT_SOCKET_PROTOCOL 0
#define SERVER_PORT 43
#define IPv4_MAX_LENGTH 16
#define IPv6_MAX_LENGTH 40
#define DOMAIN_NAME_MAX_LENGTH 254
#define MAX_ADDRES_COUNT 3


typedef struct argument{
    char address[DOMAIN_NAME_MAX_LENGTH];
    char IPv4[IPv4_MAX_LENGTH];
    char IPv6[IPv6_MAX_LENGTH];
    int address_family;
    struct argument *next;
} Argument;

int addres_to_ip(char * hostname , char* ip);
int get_IP_addresses(Argument **args);


int main(int argc , char *argv[])
{
    Argument *args = NULL;  
    args = malloc(sizeof(Argument));    

    strcpy(args->address, "google.com");

    get_IP_addresses(&args);
    return 0;
}

int get_IP_addresses(Argument **args){

    char addrstr[DOMAIN_NAME_MAX_LENGTH];
    char tmp_ip[DOMAIN_NAME_MAX_LENGTH];
    strcpy(tmp_ip, (*args)->address);
    struct addrinfo tmp_ai, *new_ai;

    tmp_ai.ai_flags = AI_CANONNAME;
    tmp_ai.ai_family = AF_INET;
    tmp_ai.ai_socktype = SOCK_STREAM;

    printf("a\n");
    if(getaddrinfo (tmp_ip, NULL, &tmp_ai, &new_ai) != 0){
        printf("CHYBA: gettaddrinfo()\n");
    }

    while (new_ai)
    {
      inet_ntop (AF_INET, new_ai->ai_addr->sa_data, addrstr, DOMAIN_NAME_MAX_LENGTH);
      new_ai = new_ai->ai_next;
    }

    return 0;
}

Если вы удалите

inet_ntop (AF_INET, new_ai-> ai_addr-> sa_data, addrstr, DOMAIN_NAME_MAX_LENGTH);

это не вызовет ошибку сегмента. Но зачем это давать ?? Пожалуйста, помогите, спасибо!

(PS: я пытался упростить мой код, насколько смог. Это причина, по которой он не дает никаких полезных результатов)

1 Ответ

0 голосов
/ 08 октября 2019

… выдает ошибку сегмента. Но почему он это дает?

getaddrinfo(…, &new_ai) возвращается с ошибкой, и в этом случае вы не можете полагаться на установленный new_ai.

Причина возврата ошибкичто вы забыли инициализировать tmp_ai.ai_protocol. Этого бы не произошло, если бы вы использовали инициализатор с определением tmp_ai:

    struct addrinfo tmp_ai = { .ai_flags = AI_CANONNAME,
                               .ai_family = AF_INET,
                               .ai_socktype = SOCK_STREAM }, *new_ai;
...