diff options
| author | Michael Adam <obnox@samba.org> | 2014-05-30 13:06:02 +0200 |
|---|---|---|
| committer | Michael Adam <obnox@samba.org> | 2014-06-01 10:03:05 +0200 |
| commit | 97a65a132d1b14cbfd539f327f260237c1afff3d (patch) | |
| tree | 0fc4d9989e99fbfa1f8879960be47503f1b4dde6 /src | |
| parent | af40930bebe0322d5876bda2aa36598e1bae2056 (diff) | |
| download | socket_wrapper-97a65a132d1b14cbfd539f327f260237c1afff3d.tar.gz socket_wrapper-97a65a132d1b14cbfd539f327f260237c1afff3d.tar.xz socket_wrapper-97a65a132d1b14cbfd539f327f260237c1afff3d.zip | |
swrap: extend input checks in swrap_bind()
Not only check family, but depending on family, also check the length.
Signed-off-by: Michael Adam <obnox@samba.org>
Reviewed-by: Andreas Schneider <asn@samba.org>
Diffstat (limited to 'src')
| -rw-r--r-- | src/socket_wrapper.c | 50 |
1 files changed, 48 insertions, 2 deletions
diff --git a/src/socket_wrapper.c b/src/socket_wrapper.c index 621a3e7..0303b21 100644 --- a/src/socket_wrapper.c +++ b/src/socket_wrapper.c @@ -2725,13 +2725,59 @@ static int swrap_bind(int s, const struct sockaddr *myaddr, socklen_t addrlen) int ret; struct sockaddr_un un_addr; struct socket_info *si = find_socket_info(s); + int bind_error = 0; if (!si) { return libc_bind(s, myaddr, addrlen); } - if (si->family != myaddr->sa_family) { - errno = EAFNOSUPPORT; + switch (si->family) { + case AF_INET: { + const struct sockaddr_in *sin; + if (addrlen < sizeof(struct sockaddr_in)) { + bind_error = EINVAL; + break; + } + + sin = (struct sockaddr_in *)myaddr; + + if (sin->sin_family != AF_INET) { + bind_error = EAFNOSUPPORT; + } + + /* special case for AF_UNSPEC */ + if (sin->sin_family == AF_UNSPEC && + (sin->sin_addr.s_addr == htonl(INADDR_ANY))) + { + bind_error = 0; + } + + break; + } +#ifdef HAVE_IPV6 + case AF_INET6: { + const struct sockaddr_in6 *sin6; + if (addrlen < sizeof(struct sockaddr_in6)) { + bind_error = EINVAL; + break; + } + + sin6 = (struct sockaddr_in6 *)myaddr; + + if (sin6->sin6_family != AF_INET6) { + bind_error = EAFNOSUPPORT; + } + + break; + } +#endif + default: + bind_error = EINVAL; + break; + } + + if (bind_error != 0) { + errno = bind_error; return -1; } |
