/* See LICENSE file for copyright and license details. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "sock.h" #include "util.h" int sock_get_ips(const char *host, const char* port) { struct addrinfo hints = { .ai_flags = AI_NUMERICSERV, .ai_family = AF_UNSPEC, .ai_socktype = SOCK_STREAM, }; struct addrinfo *ai, *p; int ret, insock = 0; if ((ret = getaddrinfo(host, port, &hints, &ai))) { die("getaddrinfo: %s", gai_strerror(ret)); } for (p = ai; p; p = p->ai_next) { if ((insock = socket(p->ai_family, p->ai_socktype, p->ai_protocol)) < 0) { continue; } if (setsockopt(insock, SOL_SOCKET, SO_REUSEADDR, &(int){1}, sizeof(int)) < 0) { die("setsockopt:"); } if (bind(insock, p->ai_addr, p->ai_addrlen) < 0) { /* bind failed, close the insock and retry */ if (close(insock) < 0) { die("close:"); } continue; } break; } freeaddrinfo(ai); if (!p) { /* we exhaustet the addrinfo-list and found no connection */ if (errno == EACCES) { die("You need to run as root or have " "CAP_NET_BIND_SERVICE set to bind to " "privileged ports"); } else { die("bind:"); } } if (listen(insock, SOMAXCONN) < 0) { die("listen:"); } return insock; } int sock_get_uds(const char *udsname, uid_t uid, gid_t gid) { struct sockaddr_un addr = { .sun_family = AF_UNIX, }; size_t udsnamelen; int insock, sockmode = S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH; if ((insock = socket(AF_UNIX, SOCK_STREAM, 0)) < 0) { die("socket:"); } if ((udsnamelen = strlen(udsname)) > sizeof(addr.sun_path) - 1) { die("UNIX-domain socket name truncated"); } memcpy(addr.sun_path, udsname, udsnamelen + 1); if (bind(insock, (const struct sockaddr *)&addr, sizeof(addr)) < 0) { die("bind '%s':", udsname); } if (listen(insock, SOMAXCONN) < 0) { sock_rem_uds(udsname); die("listen:"); } if (chmod(udsname, sockmode) < 0) { sock_rem_uds(udsname); die("chmod '%s':", udsname); } if (chown(udsname, uid, gid) < 0) { sock_rem_uds(udsname); die("chown '%s':", udsname); } return insock; } void sock_rem_uds(const char *udsname) { if (unlink(udsname) < 0) { die("unlink '%s':", udsname); } } int sock_set_timeout(int fd, int sec) { struct timeval tv; tv.tv_sec = sec; tv.tv_usec = 0; if (setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv)) < 0 || setsockopt(fd, SOL_SOCKET, SO_SNDTIMEO, &tv, sizeof(tv)) < 0) { warn("setsockopt:"); return 1; } return 0; } int sock_set_nonblocking(int fd) { int flags; if ((flags = fcntl(fd, F_GETFL, 0)) < 0) { warn("fcntl:"); return 1; } flags |= O_NONBLOCK; if (fcntl(fd, F_SETFL, flags) < 0) { warn("fcntl:"); return 1; } return 0; } int sock_get_inaddr_str(const struct sockaddr_storage *in_sa, char *str, size_t len) { switch (in_sa->ss_family) { case AF_INET: if (!inet_ntop(AF_INET, &(((struct sockaddr_in *)in_sa)->sin_addr), str, len)) { warn("inet_ntop:"); return 1; } break; case AF_INET6: if (!inet_ntop(AF_INET6, &(((struct sockaddr_in6 *)in_sa)->sin6_addr), str, len)) { warn("inet_ntop:"); return 1; } break; case AF_UNIX: snprintf(str, len, "uds"); break; default: snprintf(str, len, "-"); } return 0; } int sock_same_addr(const struct sockaddr_storage *sa1, const struct sockaddr_storage *sa2) { /* return early if address-families don't match */ if (sa1->ss_family != sa2->ss_family) { return 0; } switch (sa1->ss_family) { case AF_INET6: return memcmp(((struct sockaddr_in6 *)sa1)->sin6_addr.s6_addr, ((struct sockaddr_in6 *)sa2)->sin6_addr.s6_addr, sizeof(((struct sockaddr_in6 *)sa1)->sin6_addr.s6_addr)); case AF_INET: return ntohl(((struct sockaddr_in *)sa1)->sin_addr.s_addr) == ntohl(((struct sockaddr_in *)sa2)->sin_addr.s_addr); default: /* AF_UNIX */ return strcmp(((struct sockaddr_un *)sa1)->sun_path, ((struct sockaddr_un *)sa2)->sun_path) == 0; } }