Home | History | Annotate | Download | only in lib6
      1 /*
      2  * Copyright (c) International Business Machines Corp., 2001
      3  * Copyright (c) 2017 Petr Vorel <pvorel (at) suse.cz>
      4  *
      5  * This program is free software; you can redistribute it and/or
      6  * modify it under the terms of the GNU General Public License as
      7  * published by the Free Software Foundation; either version 2 of
      8  * the License, or (at your option) any later version.
      9  *
     10  * This program is distributed in the hope that it would be useful,
     11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
     12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
     13  * GNU General Public License for more details.
     14  *
     15  * You should have received a copy of the GNU General Public License
     16  * along with this program. If not, see <http://www.gnu.org/licenses/>.
     17  *
     18  * Author: David L Stevens
     19  */
     20 
     21 /*
     22  * Description: Verify that in6 and sockaddr fields are present.
     23  */
     24 
     25 #include <errno.h>
     26 #include <arpa/inet.h>
     27 #include <netinet/in.h>
     28 #include <sys/socket.h>
     29 
     30 #include "tst_test.h"
     31 #include "tst_safe_macros.h"
     32 
     33 static struct {
     34 	char *addr;
     35 	int ismap;
     36 } maptab[] = {
     37 	{ "2002::1", 0 },
     38 	{ "::ffff:10.0.0.1", 1 },
     39 	{ "::fffe:10.0.0.1", 0 },
     40 	{ "::7fff:10.0.0.1", 0 },
     41 	{ "0:0:0:0:0:0:ffff:a001", 0 },
     42 	{ "0:0:1:0:0:0:ffff:a001", 0 },
     43 };
     44 
     45 static struct {
     46 	char *addr;
     47 } sstab[] = {
     48 	{ "2002::1" },
     49 	{ "10.0.0.1" },
     50 	{ "::ffff:10.0.0.1" },
     51 	{ "::1" },
     52 	{ "::" },
     53 };
     54 
     55 static void test_in6_addr(void);
     56 static void test_sockaddr_in6(void);
     57 static void test_global_in6_def(void);
     58 static void test_in6_is_addr_v4mapped(void);
     59 static void test_sockaddr_storage(void);
     60 
     61 static void (*testfunc[])(void) = { test_in6_addr,
     62 	test_sockaddr_in6, test_global_in6_def,
     63 	test_in6_is_addr_v4mapped, test_sockaddr_storage };
     64 
     65 /* struct in6_addr tests */
     66 static void test_in6_addr(void)
     67 {
     68 	uint8_t ui8 = 1;
     69 	struct in6_addr in6;
     70 
     71 	in6.s6_addr[0] = ui8;
     72 	tst_res(TINFO, "type of in6.s6_addr[0] is uint8_t");
     73 	if (sizeof(in6.s6_addr) != 16)
     74 		tst_res(TFAIL, "sizeof(in6.s6_addr) != 16");
     75 	else
     76 		tst_res(TPASS, "sizeof(in6.s6_addr) == 16");
     77 }
     78 
     79 /* struct sockaddr_in6 tests */
     80 static void test_sockaddr_in6(void)
     81 {
     82 	uint8_t ui8 = 1;
     83 	uint32_t ui16 = 2;
     84 	uint32_t ui32 = 3;
     85 	struct in6_addr in6;
     86 	struct sockaddr_in6 sin6;
     87 	int sd;
     88 
     89 	in6.s6_addr[0] = ui8;
     90 	sin6.sin6_family = AF_INET6;
     91 	sin6.sin6_port = ui16;
     92 	sin6.sin6_flowinfo = ui32;
     93 	sin6.sin6_addr = in6;
     94 	sin6.sin6_scope_id = ui32;
     95 
     96 	sd = SAFE_SOCKET(AF_INET6, SOCK_STREAM, 0);
     97 	bind(sd, (struct sockaddr *)&sin6, sizeof(sin6));
     98 	SAFE_CLOSE(sd);
     99 
    100 	tst_res(TPASS, "all sockaddr_in6 fields present and correct");
    101 }
    102 
    103 /* initializers and global in6 definitions tests */
    104 static void test_global_in6_def(void)
    105 {
    106 	struct in6_addr ina6 = IN6ADDR_ANY_INIT;
    107 	struct in6_addr inl6 = IN6ADDR_LOOPBACK_INIT;
    108 
    109 	tst_res(TINFO, "IN6ADDR_ANY_INIT present");
    110 	if (memcmp(&ina6, &in6addr_any, sizeof(ina6)) == 0)
    111 		tst_res(TINFO, "in6addr_any present and correct");
    112 	else {
    113 		tst_res(TFAIL, "in6addr_any incorrect value");
    114 		return;
    115 	}
    116 
    117 	tst_res(TINFO, "IN6ADDR_LOOPBACK_INIT present");
    118 	if (memcmp(&inl6, &in6addr_loopback, sizeof(inl6)) == 0)
    119 		tst_res(TINFO, "in6addr_loopback present and correct");
    120 	else {
    121 		tst_res(TFAIL, "in6addr_loopback incorrect value");
    122 		return;
    123 	}
    124 
    125 	if (inet_pton(AF_INET6, "::1", &inl6) <= 0)
    126 		tst_brk(TBROK | TERRNO, "inet_pton(\"::1\")");
    127 
    128 	if (memcmp(&inl6, &in6addr_loopback, sizeof(inl6)) == 0)
    129 		tst_res(TINFO, "in6addr_loopback in network byte order");
    130 	else {
    131 		tst_res(TFAIL, "in6addr_loopback has wrong byte order");
    132 		return;
    133 	}
    134 
    135 	tst_res(TPASS, "global in6 definitions tests succeed");
    136 }
    137 
    138 /* IN6_IS_ADDR_V4MAPPED tests */
    139 static void test_in6_is_addr_v4mapped(void)
    140 {
    141 	unsigned int i;
    142 	struct in6_addr in6;
    143 
    144 	for (i = 0; i < ARRAY_SIZE(maptab); ++i) {
    145 		if (inet_pton(AF_INET6, maptab[i].addr, &in6) <= 0)
    146 			tst_brk(TBROK | TERRNO,
    147 				"\"%s\" is not a valid IPv6 address",
    148 				maptab[i].addr);
    149 		TEST(IN6_IS_ADDR_V4MAPPED(in6.s6_addr));
    150 		if (maptab[i].ismap == TEST_RETURN)
    151 			tst_res(TINFO, "IN6_IS_ADDR_V4MAPPED(\"%s\") %ld",
    152 				maptab[i].addr, TEST_RETURN);
    153 		else {
    154 			tst_res(TFAIL, "IN6_IS_ADDR_V4MAPPED(\"%s\") %ld",
    155 				maptab[i].addr, TEST_RETURN);
    156 			return;
    157 		}
    158 	}
    159 
    160 	tst_res(TPASS, "IN6_IS_ADDR_V4MAPPED tests succeed");
    161 }
    162 
    163 /* sockaddr_storage tests */
    164 static void test_sockaddr_storage(void)
    165 {
    166 	unsigned int i;
    167 	struct sockaddr_storage ss;
    168 
    169 	if (sizeof(ss) <= sizeof(struct sockaddr_in) ||
    170 		sizeof(ss) <= sizeof(struct sockaddr_in6))
    171 		tst_brk(TBROK, "sockaddr_storage too small");
    172 
    173 	for (i = 0; i < ARRAY_SIZE(sstab); ++i) {
    174 		struct sockaddr_in *psin = (struct sockaddr_in *)&ss;
    175 		struct sockaddr_in6 *psin6 = (struct sockaddr_in6 *)&ss;
    176 		int rv;
    177 		uint8_t af;
    178 
    179 		af = psin->sin_family = AF_INET;
    180 		rv = inet_pton(AF_INET, sstab[i].addr, &psin->sin_addr);
    181 		if (rv == 0) {
    182 			af = psin6->sin6_family = AF_INET6;
    183 			rv = inet_pton(AF_INET6, sstab[i].addr,
    184 				&psin6->sin6_addr);
    185 		}
    186 		if (rv <= 0)
    187 			tst_brk(TBROK,
    188 				"\"%s\" is not a valid address", sstab[i].addr);
    189 		if (ss.ss_family == af)
    190 			tst_res(TINFO, "\"%s\" is AF_INET%s",
    191 				sstab[i].addr, af == AF_INET ? "" : "6");
    192 		else
    193 			tst_res(TFAIL, "\"%s\" ss_family (%d) != AF_INET%s",
    194 				sstab[i].addr, af, af == AF_INET ? "" : "6");
    195 	}
    196 
    197 	tst_res(TPASS, "sockaddr_storage tests succeed");
    198 }
    199 
    200 static void do_test(unsigned int i)
    201 {
    202 	testfunc[i]();
    203 }
    204 
    205 static struct tst_test test = {
    206 	.tcnt = ARRAY_SIZE(testfunc),
    207 	.test = do_test,
    208 };
    209