Fix inet_addr/inet_aton/inet_network.
Rewrite inet_addr and inet_network in terms of inet_aton, and reimplement that to include all the missing error checks. Bug: http://b/24754503 Change-Id: I5dfa971c87201968985a0894df419f0fbf54768a
This commit is contained in:
		| @@ -104,6 +104,7 @@ libc_bionic_ndk_src_files := \ | ||||
|     bionic/accept.cpp \ | ||||
|     bionic/accept4.cpp \ | ||||
|     bionic/access.cpp \ | ||||
|     bionic/arpa_inet.cpp \ | ||||
|     bionic/assert.cpp \ | ||||
|     bionic/atof.cpp \ | ||||
|     bionic/bionic_systrace.cpp \ | ||||
| @@ -395,11 +396,9 @@ libc_upstream_openbsd_ndk_src_files := \ | ||||
|     upstream-openbsd/lib/libc/locale/wctomb.c \ | ||||
|     upstream-openbsd/lib/libc/net/htonl.c \ | ||||
|     upstream-openbsd/lib/libc/net/htons.c \ | ||||
|     upstream-openbsd/lib/libc/net/inet_addr.c \ | ||||
|     upstream-openbsd/lib/libc/net/inet_lnaof.c \ | ||||
|     upstream-openbsd/lib/libc/net/inet_makeaddr.c \ | ||||
|     upstream-openbsd/lib/libc/net/inet_netof.c \ | ||||
|     upstream-openbsd/lib/libc/net/inet_network.c \ | ||||
|     upstream-openbsd/lib/libc/net/inet_ntoa.c \ | ||||
|     upstream-openbsd/lib/libc/net/inet_ntop.c \ | ||||
|     upstream-openbsd/lib/libc/net/inet_pton.c \ | ||||
|   | ||||
							
								
								
									
										71
									
								
								libc/bionic/arpa_inet.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								libc/bionic/arpa_inet.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| /* | ||||
|  * Copyright (C) 2015 The Android Open Source Project | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *      http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  */ | ||||
|  | ||||
| #include <arpa/inet.h> | ||||
| #include <netinet/in.h> | ||||
| #include <stdlib.h> | ||||
|  | ||||
| #include "private/ErrnoRestorer.h" | ||||
|  | ||||
| // The difference between inet_network(3) and inet_addr(3) is that | ||||
| // inet_network uses host order and inet_addr network order. | ||||
| in_addr_t inet_network(const char* cp) { | ||||
|   in_addr_t network_order = inet_addr(cp); | ||||
|   return ntohl(network_order); | ||||
| } | ||||
|  | ||||
| in_addr_t inet_addr(const char* cp) { | ||||
|   in_addr addr; | ||||
|   return inet_aton(cp, &addr) ? addr.s_addr : INADDR_NONE; | ||||
| } | ||||
|  | ||||
| int inet_aton(const char* cp, in_addr* addr) { | ||||
|   ErrnoRestorer errno_restorer; | ||||
|  | ||||
|   unsigned long parts[4]; | ||||
|   size_t i; | ||||
|   for (i = 0; i < 4; ++i) { | ||||
|     char* end; | ||||
|     parts[i] = strtoul(cp, &end, 0); | ||||
|     if (end == cp || (*end != '.' && *end != '\0')) return 0; | ||||
|     if (*end == '\0') break; | ||||
|     cp = end + 1; | ||||
|   } | ||||
|  | ||||
|   uint32_t result = 0; | ||||
|   if (i == 0) { | ||||
|     // a (a 32-bit). | ||||
|     if (parts[0] > 0xffffffff) return 0; | ||||
|     result = parts[0]; | ||||
|   } else if (i == 1) { | ||||
|     // a.b (b 24-bit). | ||||
|     if (parts[0] > 0xff || parts[1] > 0xffffff) return 0; | ||||
|     result = (parts[0] << 24) | parts[1]; | ||||
|   } else if (i == 2) { | ||||
|     // a.b.c (c 16-bit). | ||||
|     if (parts[0] > 0xff || parts[1] > 0xff || parts[2] > 0xffff) return 0; | ||||
|     result = (parts[0] << 24) | (parts[1] << 16) | parts[2]; | ||||
|   } else if (i == 3) { | ||||
|     // a.b.c.d (d 8-bit). | ||||
|     if (parts[0] > 0xff || parts[1] > 0xff || parts[2] > 0xff || parts[3] > 0xff) return 0; | ||||
|     result = (parts[0] << 24) | (parts[1] << 16) | (parts[2] << 8) | parts[3]; | ||||
|   } else { | ||||
|     return 0; | ||||
|   } | ||||
|  | ||||
|   if (addr != nullptr) addr->s_addr = htonl(result); | ||||
|   return 1; | ||||
| } | ||||
| @@ -1,175 +0,0 @@ | ||||
| /*	$OpenBSD: inet_addr.c,v 1.10 2013/11/24 23:51:28 deraadt Exp $	*/ | ||||
|  | ||||
| /* | ||||
|  * ++Copyright++ 1983, 1990, 1993 | ||||
|  * - | ||||
|  * Copyright (c) 1983, 1990, 1993 | ||||
|  *    The Regents of the University of California.  All rights reserved. | ||||
|  *  | ||||
|  * Redistribution and use in source and binary forms, with or without | ||||
|  * modification, are permitted provided that the following conditions | ||||
|  * are met: | ||||
|  * 1. Redistributions of source code must retain the above copyright | ||||
|  *    notice, this list of conditions and the following disclaimer. | ||||
|  * 2. Redistributions in binary form must reproduce the above copyright | ||||
|  *    notice, this list of conditions and the following disclaimer in the | ||||
|  *    documentation and/or other materials provided with the distribution. | ||||
|  * 3. Neither the name of the University nor the names of its contributors | ||||
|  *    may be used to endorse or promote products derived from this software | ||||
|  *    without specific prior written permission. | ||||
|  *  | ||||
|  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND | ||||
|  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
|  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | ||||
|  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE | ||||
|  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
|  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS | ||||
|  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) | ||||
|  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT | ||||
|  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY | ||||
|  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | ||||
|  * SUCH DAMAGE. | ||||
|  * - | ||||
|  * Portions Copyright (c) 1993 by Digital Equipment Corporation. | ||||
|  *  | ||||
|  * Permission to use, copy, modify, and distribute this software for any | ||||
|  * purpose with or without fee is hereby granted, provided that the above | ||||
|  * copyright notice and this permission notice appear in all copies, and that | ||||
|  * the name of Digital Equipment Corporation not be used in advertising or | ||||
|  * publicity pertaining to distribution of the document or software without | ||||
|  * specific, written prior permission. | ||||
|  *  | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS" AND DIGITAL EQUIPMENT CORP. DISCLAIMS ALL | ||||
|  * WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES | ||||
|  * OF MERCHANTABILITY AND FITNESS.   IN NO EVENT SHALL DIGITAL EQUIPMENT | ||||
|  * CORPORATION BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL | ||||
|  * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR | ||||
|  * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS | ||||
|  * ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS | ||||
|  * SOFTWARE. | ||||
|  * - | ||||
|  * --Copyright-- | ||||
|  */ | ||||
|  | ||||
| #include <sys/types.h> | ||||
| #include <sys/param.h> | ||||
| #include <netinet/in.h> | ||||
| #include <arpa/inet.h> | ||||
| #include <ctype.h> | ||||
|  | ||||
| /* | ||||
|  * Ascii internet address interpretation routine. | ||||
|  * The value returned is in network order. | ||||
|  */ | ||||
| in_addr_t | ||||
| inet_addr(const char *cp) | ||||
| { | ||||
| 	struct in_addr val; | ||||
|  | ||||
| 	if (inet_aton(cp, &val)) | ||||
| 		return (val.s_addr); | ||||
| 	return (INADDR_NONE); | ||||
| } | ||||
|  | ||||
| /*  | ||||
|  * Check whether "cp" is a valid ascii representation | ||||
|  * of an Internet address and convert to a binary address. | ||||
|  * Returns 1 if the address is valid, 0 if not. | ||||
|  * This replaces inet_addr, the return value from which | ||||
|  * cannot distinguish between failure and a local broadcast address. | ||||
|  */ | ||||
| int | ||||
| inet_aton(const char *cp, struct in_addr *addr) | ||||
| { | ||||
| 	in_addr_t val; | ||||
| 	int base, n; | ||||
| 	char c; | ||||
| 	u_int parts[4]; | ||||
| 	u_int *pp = parts; | ||||
|  | ||||
| 	c = *cp; | ||||
| 	for (;;) { | ||||
| 		/* | ||||
| 		 * Collect number up to ``.''. | ||||
| 		 * Values are specified as for C: | ||||
| 		 * 0x=hex, 0=octal, isdigit=decimal. | ||||
| 		 */ | ||||
| 		if (!isdigit((unsigned char)c)) | ||||
| 			return (0); | ||||
| 		val = 0; base = 10; | ||||
| 		if (c == '0') { | ||||
| 			c = *++cp; | ||||
| 			if (c == 'x' || c == 'X') | ||||
| 				base = 16, c = *++cp; | ||||
| 			else | ||||
| 				base = 8; | ||||
| 		} | ||||
| 		for (;;) { | ||||
| 			if (isascii((unsigned char)c) && | ||||
| 			    isdigit((unsigned char)c)) { | ||||
| 				val = (val * base) + (c - '0'); | ||||
| 				c = *++cp; | ||||
| 			} else if (base == 16 && | ||||
| 			    isascii((unsigned char)c) && | ||||
| 			    isxdigit((unsigned char)c)) { | ||||
| 				val = (val << 4) | | ||||
| 				    (c + 10 - (islower((unsigned char)c) ? 'a' : 'A')); | ||||
| 				c = *++cp; | ||||
| 			} else | ||||
| 				break; | ||||
| 		} | ||||
| 		if (c == '.') { | ||||
| 			/* | ||||
| 			 * Internet format: | ||||
| 			 *	a.b.c.d | ||||
| 			 *	a.b.c	(with c treated as 16 bits) | ||||
| 			 *	a.b	(with b treated as 24 bits) | ||||
| 			 */ | ||||
| 			if (pp >= parts + 3) | ||||
| 				return (0); | ||||
| 			*pp++ = val; | ||||
| 			c = *++cp; | ||||
| 		} else | ||||
| 			break; | ||||
| 	} | ||||
| 	/* | ||||
| 	 * Check for trailing characters. | ||||
| 	 */ | ||||
| 	if (c != '\0' && | ||||
| 	    (!isascii((unsigned char)c) || !isspace((unsigned char)c))) | ||||
| 		return (0); | ||||
| 	/* | ||||
| 	 * Concoct the address according to | ||||
| 	 * the number of parts specified. | ||||
| 	 */ | ||||
| 	n = pp - parts + 1; | ||||
| 	switch (n) { | ||||
|  | ||||
| 	case 0: | ||||
| 		return (0);		/* initial nondigit */ | ||||
|  | ||||
| 	case 1:				/* a -- 32 bits */ | ||||
| 		break; | ||||
|  | ||||
| 	case 2:				/* a.b -- 8.24 bits */ | ||||
| 		if ((val > 0xffffff) || (parts[0] > 0xff)) | ||||
| 			return (0); | ||||
| 		val |= parts[0] << 24; | ||||
| 		break; | ||||
|  | ||||
| 	case 3:				/* a.b.c -- 8.8.16 bits */ | ||||
| 		if ((val > 0xffff) || (parts[0] > 0xff) || (parts[1] > 0xff)) | ||||
| 			return (0); | ||||
| 		val |= (parts[0] << 24) | (parts[1] << 16); | ||||
| 		break; | ||||
|  | ||||
| 	case 4:				/* a.b.c.d -- 8.8.8.8 bits */ | ||||
| 		if ((val > 0xff) || (parts[0] > 0xff) || (parts[1] > 0xff) || (parts[2] > 0xff)) | ||||
| 			return (0); | ||||
| 		val |= (parts[0] << 24) | (parts[1] << 16) | (parts[2] << 8); | ||||
| 		break; | ||||
| 	} | ||||
| 	if (addr) | ||||
| 		addr->s_addr = htonl(val); | ||||
| 	return (1); | ||||
| } | ||||
| @@ -1,84 +0,0 @@ | ||||
| /*	$OpenBSD: inet_network.c,v 1.11 2013/11/25 17:29:19 deraadt Exp $ */ | ||||
| /* | ||||
|  * Copyright (c) 1983, 1993 | ||||
|  *	The Regents of the University of California.  All rights reserved. | ||||
|  * | ||||
|  * Redistribution and use in source and binary forms, with or without | ||||
|  * modification, are permitted provided that the following conditions | ||||
|  * are met: | ||||
|  * 1. Redistributions of source code must retain the above copyright | ||||
|  *    notice, this list of conditions and the following disclaimer. | ||||
|  * 2. Redistributions in binary form must reproduce the above copyright | ||||
|  *    notice, this list of conditions and the following disclaimer in the | ||||
|  *    documentation and/or other materials provided with the distribution. | ||||
|  * 3. Neither the name of the University nor the names of its contributors | ||||
|  *    may be used to endorse or promote products derived from this software | ||||
|  *    without specific prior written permission. | ||||
|  * | ||||
|  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND | ||||
|  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
|  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | ||||
|  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE | ||||
|  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
|  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS | ||||
|  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) | ||||
|  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT | ||||
|  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY | ||||
|  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | ||||
|  * SUCH DAMAGE. | ||||
|  */ | ||||
|  | ||||
| #include <sys/types.h> | ||||
| #include <netinet/in.h> | ||||
| #include <arpa/inet.h> | ||||
| #include <ctype.h> | ||||
|  | ||||
| /* | ||||
|  * Internet network address interpretation routine. | ||||
|  * The library routines call this routine to interpret | ||||
|  * network numbers. | ||||
|  */ | ||||
| in_addr_t | ||||
| inet_network(const char *cp) | ||||
| { | ||||
| 	in_addr_t val, base, n; | ||||
| 	u_char c; | ||||
| 	in_addr_t parts[4], *pp = parts; | ||||
| 	int i; | ||||
|  | ||||
| again: | ||||
| 	val = 0; base = 10; | ||||
| 	if (*cp == '0') | ||||
| 		base = 8, cp++; | ||||
| 	if (*cp == 'x' || *cp == 'X') | ||||
| 		base = 16, cp++; | ||||
| 	while ((c = *cp)) { | ||||
| 		if (isdigit(c)) { | ||||
| 			val = (val * base) + (c - '0'); | ||||
| 			cp++; | ||||
| 			continue; | ||||
| 		} | ||||
| 		if (base == 16 && isxdigit(c)) { | ||||
| 			val = (val << 4) + (c + 10 - (islower(c) ? 'a' : 'A')); | ||||
| 			cp++; | ||||
| 			continue; | ||||
| 		} | ||||
| 		break; | ||||
| 	} | ||||
| 	if (*cp == '.') { | ||||
| 		if (pp >= parts + 3) | ||||
| 			return (INADDR_NONE); | ||||
| 		*pp++ = val, cp++; | ||||
| 		goto again; | ||||
| 	} | ||||
| 	if (*cp && !isspace(*cp)) | ||||
| 		return (INADDR_NONE); | ||||
| 	*pp++ = val; | ||||
| 	n = pp - parts; | ||||
| 	for (val = 0, i = 0; i < 4; i++) { | ||||
| 		val <<= 8; | ||||
| 		if (i < n) | ||||
| 			val |= parts[i] & 0xff; | ||||
| 	} | ||||
| 	return (val); | ||||
| } | ||||
| @@ -24,8 +24,81 @@ TEST(arpa_inet, inet_addr) { | ||||
|  | ||||
| TEST(arpa_inet, inet_aton) { | ||||
|   in_addr a; | ||||
|   ASSERT_EQ(1, inet_aton("127.0.0.1", &a)); | ||||
|  | ||||
|   // a.b.c.d | ||||
|   a.s_addr = 0; | ||||
|   ASSERT_EQ(1, inet_aton("127.1.2.3", &a)); | ||||
|   ASSERT_EQ((htonl)(0x7f010203), a.s_addr); | ||||
|  | ||||
|   // a.b.c | ||||
|   a.s_addr = 0; | ||||
|   ASSERT_EQ(1, inet_aton("127.1.2", &a)); | ||||
|   ASSERT_EQ((htonl)(0x7f010002), a.s_addr); | ||||
|  | ||||
|   // a.b | ||||
|   a.s_addr = 0; | ||||
|   ASSERT_EQ(1, inet_aton("127.1", &a)); | ||||
|   ASSERT_EQ((htonl)(0x7f000001), a.s_addr); | ||||
|  | ||||
|   // a | ||||
|   a.s_addr = 0; | ||||
|   ASSERT_EQ(1, inet_aton("0x7f000001", &a)); | ||||
|   ASSERT_EQ((htonl)(0x7f000001), a.s_addr); | ||||
|  | ||||
|   // Hex (0x) and mixed-case hex digits. | ||||
|   a.s_addr = 0; | ||||
|   ASSERT_EQ(1, inet_aton("0xFf.0.0.1", &a)); | ||||
|   ASSERT_EQ((htonl)(0xff000001), a.s_addr); | ||||
|  | ||||
|   // Hex (0X) and mixed-case hex digits. | ||||
|   a.s_addr = 0; | ||||
|   ASSERT_EQ(1, inet_aton("0XfF.0.0.1", &a)); | ||||
|   ASSERT_EQ((htonl)(0xff000001), a.s_addr); | ||||
|  | ||||
|   // Octal. | ||||
|   a.s_addr = 0; | ||||
|   ASSERT_EQ(1, inet_aton("0177.0.0.1", &a)); | ||||
|   ASSERT_EQ((htonl)(0x7f000001), a.s_addr); | ||||
|  | ||||
|   a.s_addr = 0; | ||||
|   ASSERT_EQ(1, inet_aton("036", &a)); | ||||
|   ASSERT_EQ((htonl)(036U), a.s_addr); | ||||
| } | ||||
|  | ||||
| TEST(arpa_inet, inet_aton_nullptr) { | ||||
|   ASSERT_EQ(0, inet_aton("", nullptr)); | ||||
|   ASSERT_EQ(1, inet_aton("127.0.0.1", nullptr)); | ||||
| } | ||||
|  | ||||
| TEST(arpa_inet, inet_aton_invalid) { | ||||
|   ASSERT_EQ(0, inet_aton("", nullptr)); // Empty. | ||||
|   ASSERT_EQ(0, inet_aton("x", nullptr)); // Leading junk. | ||||
|   ASSERT_EQ(0, inet_aton("127.0.0.1x", nullptr)); // Trailing junk. | ||||
|   ASSERT_EQ(0, inet_aton("09.0.0.1", nullptr)); // Invalid octal. | ||||
|   ASSERT_EQ(0, inet_aton("0xg.0.0.1", nullptr)); // Invalid hex. | ||||
|  | ||||
|   ASSERT_EQ(0, inet_aton("1.2.3.4.5", nullptr)); // Too many dots. | ||||
|   ASSERT_EQ(0, inet_aton("1.2.3.4.", nullptr)); // Trailing dot. | ||||
|  | ||||
|   // Out of range a.b.c.d form. | ||||
|   ASSERT_EQ(0, inet_aton("999.0.0.1", nullptr)); | ||||
|   ASSERT_EQ(0, inet_aton("0.999.0.1", nullptr)); | ||||
|   ASSERT_EQ(0, inet_aton("0.0.999.1", nullptr)); | ||||
|   ASSERT_EQ(0, inet_aton("0.0.0.999", nullptr)); | ||||
|  | ||||
|   // Out of range a.b.c form. | ||||
|   ASSERT_EQ(0, inet_aton("256.0.0", nullptr)); | ||||
|   ASSERT_EQ(0, inet_aton("0.256.0", nullptr)); | ||||
|   ASSERT_EQ(0, inet_aton("0.0.0x10000", nullptr)); | ||||
|  | ||||
|   // Out of range a.b form. | ||||
|   ASSERT_EQ(0, inet_aton("256.0", nullptr)); | ||||
|   ASSERT_EQ(0, inet_aton("0.0x1000000", nullptr)); | ||||
|  | ||||
|   // Out of range a form. | ||||
|   ASSERT_EQ(0, inet_aton("0x100000000", nullptr)); | ||||
|  | ||||
|   ASSERT_EQ(0, inet_aton("0400.0.0.1", nullptr)); // Out of range octal. | ||||
| } | ||||
|  | ||||
| TEST(arpa_inet, inet_lnaof) { | ||||
| @@ -45,6 +118,8 @@ TEST(arpa_inet, inet_netof) { | ||||
|  | ||||
| TEST(arpa_inet, inet_network) { | ||||
|   ASSERT_EQ(0x7f000001U, inet_network("127.0.0.1")); | ||||
|   ASSERT_EQ(0x7fU, inet_network("0x7f")); | ||||
|   ASSERT_EQ(~0U, inet_network("")); | ||||
| } | ||||
|  | ||||
| TEST(arpa_inet, inet_ntoa) { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Elliott Hughes
					Elliott Hughes