X-Git-Url: http://git.onelab.eu/?a=blobdiff_plain;f=lib%2Fsocket-util.c;h=07687ba383b9ec1de11aac0dcb683f4b337bfa57;hb=d98fa5032eaf49b073058dbf390283c9a71001e1;hp=249b6f002b19e2303c9b054bc91b04af50b94c3d;hpb=2c360fbb2777fba6d35599e4b53287e2ecb26fa9;p=sliver-openvswitch.git diff --git a/lib/socket-util.c b/lib/socket-util.c index 249b6f002..07687ba38 100644 --- a/lib/socket-util.c +++ b/lib/socket-util.c @@ -19,6 +19,7 @@ #include #include #include +#include #include #include #include @@ -30,9 +31,18 @@ #include #include #include +#include "dynamic-string.h" #include "fatal-signal.h" +#include "packets.h" #include "util.h" #include "vlog.h" +#if AF_PACKET && __linux__ +#include +#endif +#ifdef HAVE_NETLINK +#include "netlink-protocol.h" +#include "netlink-socket.h" +#endif VLOG_DEFINE_THIS_MODULE(socket_util); @@ -544,9 +554,7 @@ exit: return error; } -/* Opens a non-blocking IPv4 socket of the specified 'style', binds to - * 'target', and listens for incoming connections. 'target' should be a string - * in the format "[][:]": +/* Parses 'target', which should be a string in the format "[][:]": * * - If 'default_port' is -1, then is required. Otherwise, if * is omitted, then 'default_port' is used instead. @@ -556,106 +564,127 @@ exit: * * - If is omitted then the IP address is wildcarded. * - * 'style' should be SOCK_STREAM (for TCP) or SOCK_DGRAM (for UDP). - * - * For TCP, the socket will have SO_REUSEADDR turned on. - * - * On success, returns a non-negative file descriptor. On failure, returns a - * negative errno value. - * - * If 'sinp' is non-null, then on success the bound address is stored into - * '*sinp'. */ -int -inet_open_passive(int style, const char *target_, int default_port, - struct sockaddr_in *sinp) + * If successful, stores the address into '*sinp' and returns true; otherwise + * zeros '*sinp' and returns false. */ +bool +inet_parse_passive(const char *target_, uint16_t default_port, + struct sockaddr_in *sinp) { char *target = xstrdup(target_); char *string_ptr = target; - struct sockaddr_in sin; const char *host_name; const char *port_string; - int fd = 0, error, port; - unsigned int yes = 1; + bool ok = false; + int port; /* Address defaults. */ - memset(&sin, 0, sizeof sin); - sin.sin_family = AF_INET; - sin.sin_addr.s_addr = htonl(INADDR_ANY); - sin.sin_port = htons(default_port); + memset(sinp, 0, sizeof *sinp); + sinp->sin_family = AF_INET; + sinp->sin_addr.s_addr = htonl(INADDR_ANY); + sinp->sin_port = htons(default_port); /* Parse optional port number. */ port_string = strsep(&string_ptr, ":"); if (port_string && str_to_int(port_string, 10, &port)) { - sin.sin_port = htons(port); + sinp->sin_port = htons(port); } else if (default_port < 0) { VLOG_ERR("%s: port number must be specified", target_); - error = EAFNOSUPPORT; goto exit; } /* Parse optional bind IP. */ host_name = strsep(&string_ptr, ":"); - if (host_name && host_name[0]) { - error = lookup_ip(host_name, &sin.sin_addr); - if (error) { - goto exit; - } + if (host_name && host_name[0] && lookup_ip(host_name, &sinp->sin_addr)) { + goto exit; + } + + ok = true; + +exit: + if (!ok) { + memset(sinp, 0, sizeof *sinp); + } + free(target); + return ok; +} + + +/* Opens a non-blocking IPv4 socket of the specified 'style', binds to + * 'target', and listens for incoming connections. Parses 'target' in the same + * way was inet_parse_passive(). + * + * 'style' should be SOCK_STREAM (for TCP) or SOCK_DGRAM (for UDP). + * + * For TCP, the socket will have SO_REUSEADDR turned on. + * + * On success, returns a non-negative file descriptor. On failure, returns a + * negative errno value. + * + * If 'sinp' is non-null, then on success the bound address is stored into + * '*sinp'. */ +int +inet_open_passive(int style, const char *target, int default_port, + struct sockaddr_in *sinp) +{ + struct sockaddr_in sin; + int fd = 0, error; + unsigned int yes = 1; + + if (!inet_parse_passive(target, default_port, &sin)) { + return EAFNOSUPPORT; } /* Create non-blocking socket, set SO_REUSEADDR. */ fd = socket(AF_INET, style, 0); if (fd < 0) { error = errno; - VLOG_ERR("%s: socket: %s", target_, strerror(error)); - goto exit; + VLOG_ERR("%s: socket: %s", target, strerror(error)); + return error; } error = set_nonblocking(fd); if (error) { - goto exit_close; + goto error; } if (style == SOCK_STREAM && setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof yes) < 0) { error = errno; - VLOG_ERR("%s: setsockopt(SO_REUSEADDR): %s", target_, strerror(error)); - goto exit_close; + VLOG_ERR("%s: setsockopt(SO_REUSEADDR): %s", target, strerror(error)); + goto error; } /* Bind. */ if (bind(fd, (struct sockaddr *) &sin, sizeof sin) < 0) { error = errno; - VLOG_ERR("%s: bind: %s", target_, strerror(error)); - goto exit_close; + VLOG_ERR("%s: bind: %s", target, strerror(error)); + goto error; } /* Listen. */ if (listen(fd, 10) < 0) { error = errno; - VLOG_ERR("%s: listen: %s", target_, strerror(error)); - goto exit_close; + VLOG_ERR("%s: listen: %s", target, strerror(error)); + goto error; } if (sinp) { socklen_t sin_len = sizeof sin; if (getsockname(fd, (struct sockaddr *) &sin, &sin_len) < 0){ error = errno; - VLOG_ERR("%s: getsockname: %s", target_, strerror(error)); - goto exit_close; + VLOG_ERR("%s: getsockname: %s", target, strerror(error)); + goto error; } if (sin.sin_family != AF_INET || sin_len != sizeof sin) { - VLOG_ERR("%s: getsockname: invalid socket name", target_); - goto exit_close; + VLOG_ERR("%s: getsockname: invalid socket name", target); + goto error; } *sinp = sin; } - error = 0; - goto exit; + return fd; -exit_close: +error: close(fd); -exit: - free(target); - return error ? -error : fd; + return error; } /* Returns a readable and writable fd for /dev/null, if successful, otherwise @@ -784,3 +813,148 @@ xpipe(int fds[2]) VLOG_FATAL("failed to create pipe (%s)", strerror(errno)); } } + +static int +getsockopt_int(int fd, int level, int optname, int *valuep) +{ + socklen_t len = sizeof *valuep; + + return (getsockopt(fd, level, optname, valuep, &len) ? errno + : len == sizeof *valuep ? 0 + : EINVAL); +} + +static void +describe_sockaddr(struct ds *string, int fd, + int (*getaddr)(int, struct sockaddr *, socklen_t *)) +{ + struct sockaddr_storage ss; + socklen_t len = sizeof ss; + + if (!getaddr(fd, (struct sockaddr *) &ss, &len)) { + if (ss.ss_family == AF_INET) { + struct sockaddr_in sin; + + memcpy(&sin, &ss, sizeof sin); + ds_put_format(string, IP_FMT":%"PRIu16, + IP_ARGS(&sin.sin_addr.s_addr), ntohs(sin.sin_port)); + } else if (ss.ss_family == AF_UNIX) { + struct sockaddr_un sun; + const char *null; + size_t maxlen; + + memcpy(&sun, &ss, sizeof sun); + maxlen = len - offsetof(struct sockaddr_un, sun_path); + null = memchr(sun.sun_path, '\0', maxlen); + ds_put_buffer(string, sun.sun_path, + null ? null - sun.sun_path : maxlen); + } +#ifdef HAVE_NETLINK + else if (ss.ss_family == AF_NETLINK) { + int protocol; + +/* SO_PROTOCOL was introduced in 2.6.32. Support it regardless of the version + * of the Linux kernel headers in use at build time. */ +#ifndef SO_PROTOCOL +#define SO_PROTOCOL 38 +#endif + + if (!getsockopt_int(fd, SOL_SOCKET, SO_PROTOCOL, &protocol)) { + switch (protocol) { + case NETLINK_ROUTE: + ds_put_cstr(string, "NETLINK_ROUTE"); + break; + + case NETLINK_GENERIC: + ds_put_cstr(string, "NETLINK_GENERIC"); + break; + + default: + ds_put_format(string, "AF_NETLINK family %d", protocol); + break; + } + } else { + ds_put_cstr(string, "AF_NETLINK"); + } + } +#endif +#if AF_PACKET && __linux__ + else if (ss.ss_family == AF_PACKET) { + struct sockaddr_ll sll; + + memcpy(&sll, &ss, sizeof sll); + ds_put_cstr(string, "AF_PACKET"); + if (sll.sll_ifindex) { + char name[IFNAMSIZ]; + + if (if_indextoname(sll.sll_ifindex, name)) { + ds_put_format(string, "(%s)", name); + } else { + ds_put_format(string, "(ifindex=%d)", sll.sll_ifindex); + } + } + if (sll.sll_protocol) { + ds_put_format(string, "(protocol=0x%"PRIu16")", + ntohs(sll.sll_protocol)); + } + } +#endif + else if (ss.ss_family == AF_UNSPEC) { + ds_put_cstr(string, "AF_UNSPEC"); + } else { + ds_put_format(string, "AF_%d", (int) ss.ss_family); + } + } +} + + +#ifdef __linux__ +static void +put_fd_filename(struct ds *string, int fd) +{ + char buf[1024]; + char *linkname; + int n; + + linkname = xasprintf("/proc/self/fd/%d", fd); + n = readlink(linkname, buf, sizeof buf); + if (n > 0) { + ds_put_char(string, ' '); + ds_put_buffer(string, buf, n); + if (n > sizeof buf) { + ds_put_cstr(string, "..."); + } + } + free(linkname); +} +#endif + +/* Returns a malloc()'d string describing 'fd', for use in logging. */ +char * +describe_fd(int fd) +{ + struct ds string; + struct stat s; + + ds_init(&string); + if (fstat(fd, &s)) { + ds_put_format(&string, "fstat failed (%s)", strerror(errno)); + } else if (S_ISSOCK(s.st_mode)) { + describe_sockaddr(&string, fd, getsockname); + ds_put_cstr(&string, "<->"); + describe_sockaddr(&string, fd, getpeername); + } else { + ds_put_cstr(&string, (isatty(fd) ? "tty" + : S_ISDIR(s.st_mode) ? "directory" + : S_ISCHR(s.st_mode) ? "character device" + : S_ISBLK(s.st_mode) ? "block device" + : S_ISREG(s.st_mode) ? "file" + : S_ISFIFO(s.st_mode) ? "FIFO" + : S_ISLNK(s.st_mode) ? "symbolic link" + : "unknown")); +#ifdef __linux__ + put_fd_filename(&string, fd); +#endif + } + return ds_steal_cstr(&string); +}