#define _GNU_SOURCE #include "helpers.h" #include #include #include #include #include #include #include #define UNIQ_SORT \ { \ int i, j, u; \ for (i = j = 0; j < ct; i++) \ { \ u = pu[j++]; \ for (; j < ct && pu[j] == u; j++); \ pu[i] = u; \ } \ return i; \ } int unique_size_t(size_t *pu, int ct) UNIQ_SORT int unique_ssize_t(ssize_t *pu, int ct) UNIQ_SORT static int cmp_size_t(const void * a, const void * b) { return *(size_t*)a < *(size_t*)b ? -1 : *(size_t*)a > *(size_t*)b; } void qsort_size_t(size_t *array, int ct) { qsort(array,ct,sizeof(*array),cmp_size_t); } static int cmp_ssize_t(const void * a, const void * b) { return *(ssize_t*)a < *(ssize_t*)b ? -1 : *(ssize_t*)a > *(ssize_t*)b; } void qsort_ssize_t(ssize_t *array, int ct) { qsort(array,ct,sizeof(*array),cmp_ssize_t); } int str_index(const char **strs, int count, const char *str) { for(int i=0;i= s && (*p == '\n' || *p == '\r'); p--) *p = '\0'; } void replace_char(char *s, char from, char to) { for(;*s;s++) if (*s==from) *s=to; } 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; } static inline bool is_letter(char c) { return (c>='a' && c<='z') || (c>='A' && c<='Z'); } static inline bool is_digit(char c) { return c>='0' && c<='9'; } bool is_identifier(const char *p) { if (*p!='_' && !is_letter(*p)) return false; for(++p;*p;p++) if (!is_letter(*p) && !is_digit(*p) && *p!='_') return false; return true; } bool load_file(const char *filename, off_t offset, void *buffer, size_t *buffer_size) { FILE *F; F = fopen(filename, "rb"); if (!F) return false; if (offset) { if (-1 == lseek(fileno(F), offset, SEEK_SET)) { fclose(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, off_t offset, void *buffer, size_t *buffer_size) { bool b = load_file(filename, offset, 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 expand_bits(void *target, const void *source, unsigned int source_bitlen, unsigned int target_bytelen) { unsigned int target_bitlen = target_bytelen<<3; unsigned int bitlen = target_bitlen>3; if ((target_bytelen-bytelen)>=1) memset(target+bytelen,0,target_bytelen-bytelen); memcpy(target,source,bytelen); if ((bitlen &= 7)) ((uint8_t*)target)[bytelen] = ((uint8_t*)source)[bytelen] & (~((1 << (8-bitlen)) - 1)); } // " [fd00::1]" => "fd00::1" // "[fd00::1]:8000" => "fd00::1" // "127.0.0.1" => "127.0.0.1" // " 127.0.0.1:8000" => "127.0.0.1" // " vk.com:8000" => "vk.com" // return value: true - host is ip addr bool strip_host_to_ip(char *host) { size_t l; char *h,*p; uint8_t addr[16]; for (h = host ; *h==' ' || *h=='\t' ; h++); l = strlen(h); if (l>=2) { if (*h=='[') { // ipv6 ? for (p=++h ; *p && *p!=']' ; p++); if (*p==']') { l = p-h; memmove(host,h,l); host[l]=0; return inet_pton(AF_INET6, host, addr)>0; } } else { if (inet_pton(AF_INET6, h, addr)>0) { // ipv6 ? if (host!=h) { l = strlen(h); memmove(host,h,l); host[l]=0; } return true; } else { // ipv4 ? for (p=h ; *p && *p!=':' ; p++); l = p-h; if (host!=h) memmove(host,h,l); host[l]=0; return inet_pton(AF_INET, host, addr)>0; } } } return false; } void ntopa46(const struct in_addr *ip, const struct in6_addr *ip6,char *str, size_t len) { if (!len) return; *str = 0; if (ip) inet_ntop(AF_INET, ip, str, len); else if (ip6) inet_ntop(AF_INET6, ip6, str, len); else snprintf(str, len, "UNKNOWN_FAMILY"); } void ntop46(const struct sockaddr *sa, char *str, size_t len) { ntopa46(sa->sa_family==AF_INET ? &((struct sockaddr_in*)sa)->sin_addr : NULL, sa->sa_family==AF_INET6 ? &((struct sockaddr_in6*)sa)->sin6_addr : NULL, str, len); } 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; } uint16_t saport(const struct sockaddr *sa) { return ntohs(sa->sa_family==AF_INET ? ((struct sockaddr_in*)sa)->sin_port : sa->sa_family==AF_INET6 ? ((struct sockaddr_in6*)sa)->sin6_port : 0); } bool sa_has_addr(const struct sockaddr *sa) { switch(sa->sa_family) { case AF_INET: return ((struct sockaddr_in*)sa)->sin_addr.s_addr!=INADDR_ANY; case AF_INET6: return memcmp(((struct sockaddr_in6*)sa)->sin6_addr.s6_addr, "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00", 16); default: return false; } } 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); } uint16_t pntoh16(const uint8_t *p) { return ((uint16_t)p[0] << 8) | (uint16_t)p[1]; } void phton16(uint8_t *p, uint16_t v) { p[0] = (uint8_t)(v >> 8); p[1] = v & 0xFF; } uint32_t pntoh24(const uint8_t *p) { return ((uint32_t)p[0] << 16) | ((uint32_t)p[1] << 8) | (uint32_t)p[2]; } void phton24(uint8_t *p, uint32_t v) { p[0] = (uint8_t)(v>>16); p[1] = (uint8_t)(v>>8); p[2] = (uint8_t)v; } uint32_t pntoh32(const uint8_t *p) { return ((uint32_t)p[0] << 24) | ((uint32_t)p[1] << 16) | ((uint32_t)p[2] << 8) | (uint32_t)p[3]; } void phton32(uint8_t *p, uint32_t v) { p[0] = (uint8_t)(v>>24); p[1] = (uint8_t)(v>>16); p[2] = (uint8_t)(v>>8); p[3] = (uint8_t)v; } uint64_t pntoh64(const uint8_t *p) { return ((uint64_t)p[0] << 56) | ((uint64_t)p[1] << 48) | ((uint64_t)p[2] << 40) | ((uint64_t)p[3] << 32) | ((uint64_t)p[4] << 24) | ((uint64_t)p[5] << 16) | ((uint64_t)p[6] << 8) | p[7]; } 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; } #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(pbufsize ? size : bufsize; memcpy(buf,pattern+offset,size); buf += size; bufsize -= size; } while (bufsize) { size = bufsize>patsize ? 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); } bool file_size(const char *filename, off_t *size) { struct stat st; if (stat(filename,&st)==-1) return false; *size = st.st_size; return true; } time_t file_mod_time(const char *filename) { struct stat st; return stat(filename,&st)==-1 ? 0 : st.st_mtime; } bool file_mod_signature(const char *filename, file_mod_sig *ms) { struct stat st; if (stat(filename,&st)==-1) { FILE_MOD_RESET(ms); return false; } ms->mod_time=st.st_mtime; ms->size=st.st_size; return true; } bool file_open_test(const char *filename, int flags) { int fd = open(filename,flags); if (fd>=0) { close(fd); return true; } return false; } 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; char c; if (!s) return false; if (*s=='*' && s[1]==0) { pf->from=1; pf->to=0xFFFF; return true; } if (*s=='~') { pf->neg=true; s++; } else pf->neg=false; if (sscanf(s,"%u-%u%c",&v1,&v2,&c)==2) { if (v1>65535 || v2>65535 || v1>v2) return false; pf->from=(uint16_t)v1; pf->to=(uint16_t)v2; } else if (sscanf(s,"%u%c",&v1,&c)==1) { if (v1>65535) return false; pf->to=pf->from=(uint16_t)v1; } else return false; // deny all case if (!pf->from && !pf->to) pf->neg=true; return true; } bool pf_is_empty(const port_filter *pf) { return !pf->neg && !pf->from && !pf->to; } bool packet_pos_parse(const char *s, struct packet_pos *pos) { if (*s!='n' && *s!='d' && *s!='s' && *s!='b' && *s!='x' && *s!='a') return false; pos->mode=*s; if (pos->mode=='x' || pos->mode=='a') { pos->pos=0; return true; } return sscanf(s+1,"%u",&pos->pos)==1; } bool packet_range_parse(const char *s, struct packet_range *range) { const char *p; range->upper_cutoff = false; if (*s=='-' || *s=='<') { range->from = PACKET_POS_ALWAYS; range->upper_cutoff = *s=='<'; } else { if (!packet_pos_parse(s,&range->from)) return false; if (range->from.mode=='x') { range->to = range->from; return true; } if (!(p = strchr(s,'-'))) p = strchr(s,'<'); if (p) { s = p; range->upper_cutoff = *s=='<'; } else { if (range->from.mode=='a') { range->to = range->from; return true; } return false; } } s++; if (*s) { return packet_pos_parse(s,&range->to); } else { range->to = PACKET_POS_ALWAYS; return true; } } void fill_random_bytes(uint8_t *p,size_t sz) { size_t k,sz16 = sz>>1; for(k=0;k= 128) memset(a->s6_addr,0xFF,16); else { uint8_t n = plen >> 3; memset(a->s6_addr,0xFF,n); memset(a->s6_addr+n,0x00,16-n); a->s6_addr[n] = (uint8_t)(0xFF00 >> (plen & 7)); } } struct in6_addr ip6_mask[129]; void mask_from_preflen6_prepare(void) { for (int plen=0;plen<=128;plen++) mask_from_preflen6_make(plen, ip6_mask+plen); } #if defined(__GNUC__) && !defined(__llvm__) __attribute__((optimize ("no-strict-aliasing"))) #endif void ip6_and(const struct in6_addr * restrict a, const struct in6_addr * restrict b, struct in6_addr * restrict result) { // int128 requires 16-bit alignment. in struct sockaddr_in6.sin6_addr is 8-byte aligned. // it causes segfault on x64 arch with latest compiler. it can cause misalign slowdown on other archs // use 64-bit AND ((uint64_t*)result->s6_addr)[0] = ((uint64_t*)a->s6_addr)[0] & ((uint64_t*)b->s6_addr)[0]; ((uint64_t*)result->s6_addr)[1] = ((uint64_t*)a->s6_addr)[1] & ((uint64_t*)b->s6_addr)[1]; } void str_cidr4(char *s, size_t s_len, const struct cidr4 *cidr) { char s_ip[16]; *s_ip=0; inet_ntop(AF_INET, &cidr->addr, s_ip, sizeof(s_ip)); snprintf(s,s_len,cidr->preflen<32 ? "%s/%u" : "%s", s_ip, cidr->preflen); } void print_cidr4(const struct cidr4 *cidr) { char s[19]; str_cidr4(s,sizeof(s),cidr); printf("%s",s); } void str_cidr6(char *s, size_t s_len, const struct cidr6 *cidr) { char s_ip[40]; *s_ip=0; inet_ntop(AF_INET6, &cidr->addr, s_ip, sizeof(s_ip)); snprintf(s,s_len,cidr->preflen<128 ? "%s/%u" : "%s", s_ip, cidr->preflen); } void print_cidr6(const struct cidr6 *cidr) { char s[44]; str_cidr6(s,sizeof(s),cidr); printf("%s",s); } bool parse_cidr4(char *s, struct cidr4 *cidr) { char *p,d; bool b; unsigned int plen; if ((p = strchr(s, '/'))) { if (sscanf(p + 1, "%u", &plen)!=1 || plen>32) return false; cidr->preflen = (uint8_t)plen; d=*p; *p=0; // backup char } else cidr->preflen = 32; b = (inet_pton(AF_INET, s, &cidr->addr)==1); if (p) *p=d; // restore char return b; } bool parse_cidr6(char *s, struct cidr6 *cidr) { char *p,d; bool b; unsigned int plen; if ((p = strchr(s, '/'))) { if (sscanf(p + 1, "%u", &plen)!=1 || plen>128) return false; cidr->preflen = (uint8_t)plen; d=*p; *p=0; // backup char } else cidr->preflen = 128; b = (inet_pton(AF_INET6, s, &cidr->addr)==1); if (p) *p=d; // restore char return b; } bool parse_int16(const char *p, int16_t *v) { if (*p == '+' || *p == '-' || *p >= '0' && *p <= '9') { int i = atoi(p); *v = (int16_t)i; return *v == i; // check overflow } return false; }