summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMichael Adam <obnox@samba.org>2014-05-30 13:06:02 +0200
committerMichael Adam <obnox@samba.org>2014-06-01 10:03:05 +0200
commit97a65a132d1b14cbfd539f327f260237c1afff3d (patch)
tree0fc4d9989e99fbfa1f8879960be47503f1b4dde6
parentaf40930bebe0322d5876bda2aa36598e1bae2056 (diff)
downloadsocket_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>
-rw-r--r--src/socket_wrapper.c50
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;
}