C++程序  |  228行  |  5.6 KB

/*
 *
 *   Copyright (c) International Business Machines  Corp., 2001
 *   Author: David L Stevens
 *
 *   This program is free software;  you can redistribute it and/or modify
 *   it under the terms of the GNU General Public License as published by
 *   the Free Software Foundation; either version 2 of the License, or
 *   (at your option) any later version.
 *
 *   This program is distributed in the hope that it will be useful,
 *   but WITHOUT ANY WARRANTY;  without even the implied warranty of
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See
 *   the GNU General Public License for more details.
 *
 *   You should have received a copy of the GNU General Public License
 *   along with this program;  if not, write to the Free Software Foundation,
 *   Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 */
/*
 *   Description:
 *     Verify that in6 and sockaddr fields are present. Most of these are
 *     "PASS" if they just compile.
 */

#include <stdio.h>
#include <unistd.h>
#include <errno.h>

#include <netinet/in.h>
#include <arpa/inet.h>

#include "test.h"

static struct {
	char *addr;
	int ismap;
} maptab[] = {
	{ "2002::1", 0 },
	{ "::ffff:10.0.0.1", 1 },
	{ "::fffe:10.0.0.1", 0 },
	{ "::7fff:10.0.0.1", 0 },
	{ "0:0:0:0:0:0:ffff:0a001", 0 },
	{ "0:0:1:0:0:0:ffff:0a001", 0 },
};

#define MAPSIZE (sizeof(maptab)/sizeof(maptab[0]))

static struct {
	char *addr;
} sstab[] = {
	{ "2002::1" },
	{ "10.0.0.1" },
	{ "::ffff:10.0.0.1" },
	{ "::1" },
	{ "::" },
};

#define SSSIZE (sizeof(sstab)/sizeof(sstab[0]))

static void setup(void);
static void test_in6_addr(void);
static void test_sockaddr_in6(void);
static void test_global_in6_def(void);
static void test_in6_is_addr_v4mapped(void);
static void test_sockaddr_storage(void);

static void (*testfunc[])(void) = { test_in6_addr,
	test_sockaddr_in6, test_global_in6_def,
	test_in6_is_addr_v4mapped, test_sockaddr_storage };

char *TCID = "in6_01";
int TST_TOTAL = ARRAY_SIZE(testfunc);

int main(int argc, char *argv[])
{
	int lc;
	int i;

	tst_parse_opts(argc, argv, NULL, NULL);

	setup();

	for (lc = 0; TEST_LOOPING(lc); lc++) {
		tst_count = 0;

		for (i = 0; i < TST_TOTAL; i++)
			(*testfunc[i])();
	}

	tst_exit();
}

static void setup(void)
{
	TEST_PAUSE;
}

/* struct in6_addr tests */
static void test_in6_addr(void)
{
	uint8_t ui8 = 1;
	struct in6_addr in6;

	in6.s6_addr[0] = ui8;
	tst_resm(TINFO, "type of in6.s6_addr[0] is uint8_t");
	if (sizeof(in6.s6_addr) != 16)
		tst_resm(TFAIL, "sizeof(in6.s6_addr) != 16");
	else
		tst_resm(TPASS, "sizeof(in6.s6_addr) == 16");
}

/* struct sockaddr_in6 tests */
static void test_sockaddr_in6(void)
{
	uint8_t ui8 = 1;
	uint32_t ui16 = 2;
	uint32_t ui32 = 3;
	struct in6_addr in6;
	struct sockaddr_in6 sin6;

	in6.s6_addr[0] = ui8;
	sin6.sin6_family = AF_INET6;
	sin6.sin6_port = ui16;
	sin6.sin6_flowinfo = ui32;
	sin6.sin6_addr = in6;
	sin6.sin6_scope_id = ui32;
	tst_resm(TPASS, "all sockaddr_in6 fields present and correct");
}

/* initializers and global in6 definitions tests */
static void test_global_in6_def(void)
{
	struct in6_addr ina6 = IN6ADDR_ANY_INIT;
	struct in6_addr inl6 = IN6ADDR_LOOPBACK_INIT;

	tst_resm(TINFO, "IN6ADDR_ANY_INIT present");
	if (memcmp(&ina6, &in6addr_any, sizeof(ina6)) == 0) {
		tst_resm(TINFO, "in6addr_any present and correct");
	} else {
		tst_resm(TFAIL, "in6addr_any incorrect value");
		return;
	}

	tst_resm(TINFO, "IN6ADDR_LOOPBACK_INIT present");
	if (memcmp(&inl6, &in6addr_loopback, sizeof(inl6)) == 0) {
		tst_resm(TINFO, "in6addr_loopback present and correct");
	} else {
		tst_resm(TFAIL, "in6addr_loopback incorrect value");
		return;
	}
	if (inet_pton(AF_INET6, "::1", &inl6) <= 0)
		tst_brkm(TBROK | TERRNO, NULL, "inet_pton(\"::1\")");
	if (memcmp(&inl6, &in6addr_loopback, sizeof(inl6)) == 0) {
		tst_resm(TINFO, "in6addr_loopback in network byte order");
	} else {
		tst_resm(TFAIL, "in6addr_loopback has wrong byte order");
		return;
	}

	tst_resm(TPASS, "global in6 definitions tests succeed");
}

/* IN6_IS_ADDR_V4MAPPED tests */
static void test_in6_is_addr_v4mapped(void)
{
	unsigned int i;
	struct in6_addr in6;

	for (i = 0; i < MAPSIZE; ++i) {
		if (inet_pton(AF_INET6, maptab[i].addr, &in6) <= 0) {
			tst_brkm(TBROK | TERRNO, NULL,
				"\"%s\" is not a valid IPv6 address",
				maptab[i].addr);
		}
		TEST(IN6_IS_ADDR_V4MAPPED(in6.s6_addr));
		if (TEST_RETURN == maptab[i].ismap) {
			tst_resm(TINFO, "IN6_IS_ADDR_V4MAPPED(\"%s\") %ld",
				maptab[i].addr, TEST_RETURN);
		} else {
			tst_resm(TFAIL, "IN6_IS_ADDR_V4MAPPED(\"%s\") %ld",
				maptab[i].addr, TEST_RETURN);
			return;
		}
	}

	tst_resm(TPASS, "IN6_IS_ADDR_V4MAPPED tests succeed");
}

/* sockaddr_storage tests */
static void test_sockaddr_storage(void)
{
	unsigned int i;
	struct sockaddr_storage ss;

	if (sizeof(ss) <= sizeof(struct sockaddr_in) ||
		sizeof(ss) <= sizeof(struct sockaddr_in6))
		tst_brkm(TBROK, NULL, "sockaddr_storage too small");

	for (i = 0; i < SSSIZE; ++i) {
		struct sockaddr_in *psin = (struct sockaddr_in *)&ss;
		struct sockaddr_in6 *psin6 = (struct sockaddr_in6 *)&ss;
		int rv;
		uint8_t af;

		af = psin->sin_family = AF_INET;
		rv = inet_pton(AF_INET, sstab[i].addr, &psin->sin_addr);
		if (rv == 0) {
			af = psin6->sin6_family = AF_INET6;
			rv = inet_pton(AF_INET6, sstab[i].addr,
				&psin6->sin6_addr);
		}
		if (rv <= 0) {
			tst_brkm(TBROK, NULL,
				"\"%s\" is not a valid address", sstab[i].addr);
		}
		if (ss.ss_family == af) {
			tst_resm(TINFO, "\"%s\" is AF_INET%s",
				sstab[i].addr, af == AF_INET ? "" : "6");
		} else {
			tst_resm(TFAIL, "\"%s\" ss_family (%d) != AF_INET%s",
				sstab[i].addr, af, af == AF_INET ? "" : "6");
		}
	}

	tst_resm(TPASS, "sockaddr_storage tests succeed");
}