#define _GNU_SOURCE #include "helpers.h" #include #include #include #include #include #include "params.h" void hexdump_limited_dlog(const uint8_t *data, size_t size, size_t limit) { size_t k; bool bcut = false; if (size > limit) { size = limit; bcut = true; } if (!size) return; for (k = 0; k < size; k++) DLOG("%02X ", data[k]); DLOG(bcut ? "... : " : ": "); for (k = 0; k < size; k++) DLOG("%c", data[k] >= 0x20 && data[k] <= 0x7F ? (char)data[k] : '.'); if (bcut) DLOG(" ..."); } char *strncasestr(const char *s, const char *find, size_t slen) { char c, sc; size_t len; if ((c = *find++) != '\0') { len = strlen(find); do { do { if (slen-- < 1 || (sc = *s++) == '\0') return NULL; } while (toupper(c) != toupper(sc)); if (len > slen) return NULL; } while (strncasecmp(s, find, len) != 0); s--; } return (char *)s; } bool load_file(const char *filename, void *buffer, size_t *buffer_size) { FILE *F; F = fopen(filename, "rb"); if (!F) return false; *buffer_size = fread(buffer, 1, *buffer_size, F); if (ferror(F)) { fclose(F); return false; } fclose(F); return true; } bool load_file_nonempty(const char *filename, void *buffer, size_t *buffer_size) { bool b = load_file(filename, buffer, buffer_size); return b && *buffer_size; } bool save_file(const char *filename, const void *buffer, size_t buffer_size) { FILE *F; F = fopen(filename, "wb"); if (!F) return false; fwrite(buffer, 1, buffer_size, F); if (ferror(F)) { fclose(F); return false; } fclose(F); return true; } bool append_to_list_file(const char *filename, const char *s) { FILE *F = fopen(filename,"at"); if (!F) return false; bool bOK = fprintf(F,"%s\n",s)>0; fclose(F); return bOK; } void ntop46(const struct sockaddr *sa, char *str, size_t len) { if (!len) return; *str = 0; switch (sa->sa_family) { case AF_INET: inet_ntop(sa->sa_family, &((struct sockaddr_in*)sa)->sin_addr, str, len); break; case AF_INET6: inet_ntop(sa->sa_family, &((struct sockaddr_in6*)sa)->sin6_addr, str, len); break; default: snprintf(str, len, "UNKNOWN_FAMILY_%d", sa->sa_family); } } void ntop46_port(const struct sockaddr *sa, char *str, size_t len) { char ip[40]; ntop46(sa, ip, sizeof(ip)); switch (sa->sa_family) { case AF_INET: snprintf(str, len, "%s:%u", ip, ntohs(((struct sockaddr_in*)sa)->sin_port)); break; case AF_INET6: snprintf(str, len, "[%s]:%u", ip, ntohs(((struct sockaddr_in6*)sa)->sin6_port)); break; default: snprintf(str, len, "%s", ip); } } void print_sockaddr(const struct sockaddr *sa) { char ip_port[48]; ntop46_port(sa, ip_port, sizeof(ip_port)); printf("%s", ip_port); } bool pton4_port(const char *s, struct sockaddr_in *sa) { char ip[16],*p; size_t l; unsigned int u; p = strchr(s,':'); if (!p) return false; l = p-s; if (l<7 || l>15) return false; memcpy(ip,s,l); ip[l]=0; p++; sa->sin_family = AF_INET; if (inet_pton(AF_INET,ip,&sa->sin_addr)!=1 || sscanf(p,"%u",&u)!=1 || !u || u>0xFFFF) return false; sa->sin_port = htons((uint16_t)u); return true; } bool pton6_port(const char *s, struct sockaddr_in6 *sa) { char ip[40],*p; size_t l; unsigned int u; if (*s++!='[') return false; p = strchr(s,']'); if (!p || p[1]!=':') return false; l = p-s; if (l<2 || l>39) return false; p+=2; memcpy(ip,s,l); ip[l]=0; sa->sin6_family = AF_INET6; if (inet_pton(AF_INET6,ip,&sa->sin6_addr)!=1 || sscanf(p,"%u",&u)!=1 || !u || u>0xFFFF) return false; sa->sin6_port = htons((uint16_t)u); sa->sin6_flowinfo = 0; sa->sin6_scope_id = 0; return true; } void dbgprint_socket_buffers(int fd) { if (params.debug) { int v; socklen_t sz; sz = sizeof(int); if (!getsockopt(fd, SOL_SOCKET, SO_RCVBUF, &v, &sz)) DLOG("fd=%d SO_RCVBUF=%d\n", fd, v); sz = sizeof(int); if (!getsockopt(fd, SOL_SOCKET, SO_SNDBUF, &v, &sz)) DLOG("fd=%d SO_SNDBUF=%d\n", fd, v); } } bool set_socket_buffers(int fd, int rcvbuf, int sndbuf) { DLOG("set_socket_buffers fd=%d rcvbuf=%d sndbuf=%d\n", fd, rcvbuf, sndbuf); if (rcvbuf && setsockopt(fd, SOL_SOCKET, SO_RCVBUF, &rcvbuf, sizeof(int)) < 0) { DLOG_PERROR("setsockopt (SO_RCVBUF)"); close(fd); return false; } if (sndbuf && setsockopt(fd, SOL_SOCKET, SO_SNDBUF, &sndbuf, sizeof(int)) < 0) { DLOG_PERROR("setsockopt (SO_SNDBUF)"); close(fd); return false; } dbgprint_socket_buffers(fd); return true; } uint64_t pntoh64(const void *p) { return (uint64_t)*((const uint8_t *)(p)+0) << 56 | (uint64_t)*((const uint8_t *)(p)+1) << 48 | (uint64_t)*((const uint8_t *)(p)+2) << 40 | (uint64_t)*((const uint8_t *)(p)+3) << 32 | (uint64_t)*((const uint8_t *)(p)+4) << 24 | (uint64_t)*((const uint8_t *)(p)+5) << 16 | (uint64_t)*((const uint8_t *)(p)+6) << 8 | (uint64_t)*((const uint8_t *)(p)+7) << 0; } void phton64(uint8_t *p, uint64_t v) { p[0] = (uint8_t)(v >> 56); p[1] = (uint8_t)(v >> 48); p[2] = (uint8_t)(v >> 40); p[3] = (uint8_t)(v >> 32); p[4] = (uint8_t)(v >> 24); p[5] = (uint8_t)(v >> 16); p[6] = (uint8_t)(v >> 8); p[7] = (uint8_t)(v >> 0); } bool seq_within(uint32_t s, uint32_t s1, uint32_t s2) { return (s2>=s1 && s>=s1 && s<=s2) || (s2=s1)); } bool ipv6_addr_is_zero(const struct in6_addr *a) { return !memcmp(a,"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00",16); } #define INVALID_HEX_DIGIT ((uint8_t)-1) static inline uint8_t parse_hex_digit(char c) { return (c>='0' && c<='9') ? c-'0' : (c>='a' && c<='f') ? c-'a'+0xA : (c>='A' && c<='F') ? c-'A'+0xA : INVALID_HEX_DIGIT; } static inline bool parse_hex_byte(const char *s, uint8_t *pbyte) { uint8_t u,l; u = parse_hex_digit(s[0]); l = parse_hex_digit(s[1]); if (u==INVALID_HEX_DIGIT || l==INVALID_HEX_DIGIT) { *pbyte=0; return false; } else { *pbyte=(u<<4) | l; return true; } } bool parse_hex_str(const char *s, uint8_t *pbuf, size_t *size) { uint8_t *pe = pbuf+*size; *size=0; while(pbufpatsize ? patsize : bufsize; memcpy(buf,pattern,size); buf += size; bufsize -= size; } } int fprint_localtime(FILE *F) { struct tm t; time_t now; time(&now); localtime_r(&now,&t); return fprintf(F, "%02d.%02d.%04d %02d:%02d:%02d", t.tm_mday, t.tm_mon + 1, t.tm_year + 1900, t.tm_hour, t.tm_min, t.tm_sec); } time_t file_mod_time(const char *filename) { struct stat st; return stat(filename,&st)==-1 ? 0 : st.st_mtime; } bool pf_in_range(uint16_t port, const port_filter *pf) { return port && (((!pf->from && !pf->to) || (port>=pf->from && port<=pf->to)) ^ pf->neg); } bool pf_parse(const char *s, port_filter *pf) { unsigned int v1,v2; if (!s) return false; if (*s=='~') { pf->neg=true; s++; } else pf->neg=false; if (sscanf(s,"%u-%u",&v1,&v2)==2) { if (!v1 || v1>65535 || v2>65535 || v1>v2) return false; pf->from=(uint16_t)v1; pf->to=(uint16_t)v2; } else if (sscanf(s,"%u",&v1)==1) { if (!v1 || v1>65535) return false; pf->to=pf->from=(uint16_t)v1; } else return false; return true; }