From 391839fb138e00c7a99ab2c81f00f5aeb3be71c5 Mon Sep 17 00:00:00 2001 From: HiGarfield Date: Sat, 12 Feb 2022 03:10:58 +0800 Subject: [PATCH] use getaddrinfo() to parse address simplify implementation and support hostname as address Note: do not use static linking in compilation after this commit, since getaddrinfo() causes problems with static linking. Ref: [1] https://stackoverflow.com/questions/2725255/create-statically-linked-binary-that-uses-getaddrinfo [2] https://www.linuxquestions.org/questions/programming-9/glibc-warning-concerning-use-of-getaddrinfo-in-static-library-734169 --- common.cpp | 116 ++++++++++------------------------------------------- common.h | 2 + 2 files changed, 23 insertions(+), 95 deletions(-) diff --git a/common.cpp b/common.cpp index 138d41e..688bc3b 100644 --- a/common.cpp +++ b/common.cpp @@ -18,116 +18,42 @@ int force_socket_buf=0; int address_t::from_str(char *str) { clear(); - - char ip_addr_str[100];u32_t port; - mylog(log_info,"parsing address: %s\n",str); - int is_ipv6=0; - if(sscanf(str, "[%[^]]]:%u", ip_addr_str,&port)==2) + char addr_str[256], port_str[6], drop[2]; + mylog(log_info, "parsing address: %s\n", str); + if ((sscanf(str, "[%45[^]]]:%5[0-9]%1s", addr_str, port_str, drop) != 2 && + sscanf(str, "%255[^:]:%5[0-9]%1s", addr_str, port_str, drop) != 2) || + strtoul(port_str, NULL, 10) > 65535) { - mylog(log_info,"its an ipv6 adress\n"); - inner.ipv6.sin6_family=AF_INET6; - is_ipv6=1; - } - else if(sscanf(str, "%[^:]:%u", ip_addr_str,&port)==2) - { - mylog(log_info,"its an ipv4 adress\n"); - inner.ipv4.sin_family=AF_INET; - } - else - { - mylog(log_error,"failed to parse\n"); + mylog(log_error, "failed to parse: %s\n", str); myexit(-1); } - mylog(log_info,"ip_address is {%s}, port is {%u}\n",ip_addr_str,port); - - if(port>65535) + struct addrinfo *res; + int ret = getaddrinfo(addr_str, port_str, NULL, &res); + if (ret) { - mylog(log_error,"invalid port: %d\n",port); + mylog(log_error, "failed to parse: %s, error code: %d\n", str, ret); myexit(-1); } - - int ret=-100; - if(is_ipv6) - { - ret=inet_pton(AF_INET6, ip_addr_str,&(inner.ipv6.sin6_addr)); - inner.ipv6.sin6_port=htons(port); - if(ret==0) // 0 if address type doesnt match - { - mylog(log_error,"ip_addr %s is not an ipv6 address, %d\n",ip_addr_str,ret); - myexit(-1); - } - else if(ret==1) // inet_pton returns 1 on success - { - //okay - } - else - { - mylog(log_error,"ip_addr %s is invalid, %d\n",ip_addr_str,ret); - myexit(-1); - } - } - else - { - ret=inet_pton(AF_INET, ip_addr_str,&(inner.ipv4.sin_addr)); - inner.ipv4.sin_port=htons(port); - - if(ret==0) - { - mylog(log_error,"ip_addr %s is not an ipv4 address, %d\n",ip_addr_str,ret); - myexit(-1); - } - else if(ret==1) - { - //okay - } - else - { - mylog(log_error,"ip_addr %s is invalid, %d\n",ip_addr_str,ret); - myexit(-1); - } - } - + memcpy(&inner, res->ai_addr, res->ai_addrlen); + freeaddrinfo(res); return 0; } -int address_t::from_str_ip_only(char * str) +int address_t::from_str_ip_only(char *str) { clear(); - - u32_t type; - - if(strchr(str,':')==NULL) - type=AF_INET; - else - type=AF_INET6; - - ((sockaddr*)&inner)->sa_family=type; - - int ret; - if(type==AF_INET) + struct addrinfo hints, *res; + memset(&hints, 0, sizeof(hints)); + hints.ai_flags = AI_NUMERICHOST; + int ret = getaddrinfo(str, NULL, &hints, &res); + if (ret) { - ret=inet_pton(type, str,&inner.ipv4.sin_addr); - } - else - { - ret=inet_pton(type, str,&inner.ipv6.sin6_addr); - } - - if(ret==0) // 0 if address type doesnt match - { - mylog(log_error,"confusion in parsing %s, %d\n",str,ret); - myexit(-1); - } - else if(ret==1) // inet_pton returns 1 on success - { - //okay - } - else - { - mylog(log_error,"ip_addr %s is invalid, %d\n",str,ret); + mylog(log_error, "invalid addr: %s, error code: %d\n", str, ret); myexit(-1); } + memcpy(&inner, res->ai_addr, res->ai_addrlen); + freeaddrinfo(res); return 0; } diff --git a/common.h b/common.h index d01e5ca..8c4b4e7 100644 --- a/common.h +++ b/common.h @@ -68,6 +68,7 @@ const int is_udp2raw_mp=0; #if defined(__MINGW32__) #include #include +#include typedef unsigned char u_int8_t; typedef unsigned short u_int16_t; typedef unsigned int u_int32_t; @@ -78,6 +79,7 @@ typedef int socklen_t; #include #include #include +#include #endif