2015-02-18 14:11:32 +00:00
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2015 Michael Brown <mbrown@fensystems.co.uk>.
|
|
|
|
|
*
|
|
|
|
|
* 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.
|
2015-03-02 11:54:40 +00:00
|
|
|
*
|
|
|
|
|
* You can also choose to distribute this program under the terms of
|
|
|
|
|
* the Unmodified Binary Distribution Licence (as given in the file
|
|
|
|
|
* COPYING.UBDL), provided that you have satisfied its requirements.
|
2015-02-18 14:11:32 +00:00
|
|
|
*/
|
|
|
|
|
|
2015-03-02 11:54:40 +00:00
|
|
|
FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
|
2015-02-18 14:11:32 +00:00
|
|
|
|
|
|
|
|
/** @file
|
|
|
|
|
*
|
|
|
|
|
* IPv4 tests
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* Forcibly enable assertions */
|
|
|
|
|
#undef NDEBUG
|
|
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <string.h>
|
2025-06-10 13:37:31 +01:00
|
|
|
#include <assert.h>
|
2015-02-18 14:11:32 +00:00
|
|
|
#include <byteswap.h>
|
|
|
|
|
#include <ipxe/in.h>
|
2025-06-10 13:37:31 +01:00
|
|
|
#include <ipxe/ip.h>
|
2015-02-18 14:11:32 +00:00
|
|
|
#include <ipxe/test.h>
|
2025-06-10 13:37:31 +01:00
|
|
|
#include "netdev_test.h"
|
2015-02-18 14:11:32 +00:00
|
|
|
|
|
|
|
|
/** Define inline IPv4 address */
|
2015-07-21 15:26:52 +01:00
|
|
|
#define IPV4(a,b,c,d) \
|
|
|
|
|
htonl ( ( (a) << 24 ) | ( (b) << 16 ) | ( (c) << 8 ) | (d) )
|
2015-02-18 14:11:32 +00:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Report an inet_ntoa() test result
|
|
|
|
|
*
|
|
|
|
|
* @v addr IPv4 address
|
|
|
|
|
* @v text Expected textual representation
|
|
|
|
|
* @v file Test code file
|
|
|
|
|
* @v line Test code line
|
|
|
|
|
*/
|
|
|
|
|
static void inet_ntoa_okx ( uint32_t addr, const char *text, const char *file,
|
|
|
|
|
unsigned int line ) {
|
2015-07-21 15:26:52 +01:00
|
|
|
struct in_addr in = { .s_addr = addr };
|
2015-02-18 14:11:32 +00:00
|
|
|
char *actual;
|
|
|
|
|
|
|
|
|
|
/* Format address */
|
|
|
|
|
actual = inet_ntoa ( in );
|
|
|
|
|
DBG ( "inet_ntoa ( %d.%d.%d.%d ) = %s\n",
|
|
|
|
|
( ( ntohl ( addr ) >> 24 ) & 0xff ),
|
|
|
|
|
( ( ntohl ( addr ) >> 16 ) & 0xff ),
|
|
|
|
|
( ( ntohl ( addr ) >> 8 ) & 0xff ),
|
|
|
|
|
( ( ntohl ( addr ) >> 0 ) & 0xff ), actual );
|
|
|
|
|
okx ( strcmp ( actual, text ) == 0, file, line );
|
|
|
|
|
}
|
|
|
|
|
#define inet_ntoa_ok( addr, text ) \
|
|
|
|
|
inet_ntoa_okx ( addr, text, __FILE__, __LINE__ )
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Report an inet_aton() test result
|
|
|
|
|
*
|
|
|
|
|
* @v text Textual representation
|
|
|
|
|
* @v addr Expected IPv4 address
|
|
|
|
|
* @v file Test code file
|
|
|
|
|
* @v line Test code line
|
|
|
|
|
*/
|
|
|
|
|
static void inet_aton_okx ( const char *text, uint32_t addr, const char *file,
|
|
|
|
|
unsigned int line ) {
|
|
|
|
|
struct in_addr actual;
|
|
|
|
|
|
|
|
|
|
/* Parse address */
|
|
|
|
|
okx ( inet_aton ( text, &actual ) != 0, file, line );
|
|
|
|
|
DBG ( "inet_aton ( \"%s\" ) = %s\n", text, inet_ntoa ( actual ) );
|
2015-07-21 15:26:52 +01:00
|
|
|
okx ( actual.s_addr == addr, file, line );
|
2015-02-18 14:11:32 +00:00
|
|
|
};
|
|
|
|
|
#define inet_aton_ok( text, addr ) \
|
|
|
|
|
inet_aton_okx ( text, addr, __FILE__, __LINE__ )
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Report an inet_aton() failure test result
|
|
|
|
|
*
|
|
|
|
|
* @v text Textual representation
|
|
|
|
|
* @v file Test code file
|
|
|
|
|
* @v line Test code line
|
|
|
|
|
*/
|
|
|
|
|
static void inet_aton_fail_okx ( const char *text, const char *file,
|
|
|
|
|
unsigned int line ) {
|
|
|
|
|
struct in_addr actual;
|
|
|
|
|
|
|
|
|
|
/* Attempt to parse address */
|
|
|
|
|
okx ( inet_aton ( text, &actual ) == 0, file, line );
|
|
|
|
|
}
|
|
|
|
|
#define inet_aton_fail_ok( text ) \
|
|
|
|
|
inet_aton_fail_okx ( text, __FILE__, __LINE__ )
|
|
|
|
|
|
2025-06-10 13:37:31 +01:00
|
|
|
/**
|
|
|
|
|
* Report an ipv4_route() test result
|
|
|
|
|
*
|
|
|
|
|
* @v dest Destination address
|
|
|
|
|
* @v scope Destination scope test network device, or NULL
|
|
|
|
|
* @v next Expected next hop address (on success)
|
|
|
|
|
* @v egress Expected egress device, or NULL to expect failure
|
|
|
|
|
* @v src Expected source address (on success)
|
|
|
|
|
* @v bcast Expected broadcast packet (on success)
|
|
|
|
|
* @v file Test code file
|
|
|
|
|
* @v line Test code line
|
|
|
|
|
*/
|
|
|
|
|
static void ipv4_route_okx ( const char *dest, struct testnet *scope,
|
|
|
|
|
const char *next, struct testnet *egress,
|
|
|
|
|
const char *src, int bcast,
|
|
|
|
|
const char *file, unsigned int line ) {
|
|
|
|
|
struct ipv4_miniroute *miniroute;
|
|
|
|
|
struct in_addr in_dest;
|
|
|
|
|
struct in_addr in_src;
|
|
|
|
|
struct in_addr in_next;
|
|
|
|
|
struct in_addr actual;
|
|
|
|
|
unsigned int scope_id;
|
|
|
|
|
|
|
|
|
|
/* Sanity checks */
|
|
|
|
|
assert ( ( scope == NULL ) || ( scope->netdev != NULL ) );
|
|
|
|
|
assert ( ( egress == NULL ) == ( src == NULL ) );
|
|
|
|
|
|
|
|
|
|
/* Parse addresses */
|
|
|
|
|
okx ( inet_aton ( dest, &in_dest ) != 0, file, line );
|
|
|
|
|
if ( src )
|
|
|
|
|
okx ( inet_aton ( src, &in_src ) != 0, file, line );
|
|
|
|
|
if ( next ) {
|
|
|
|
|
okx ( inet_aton ( next, &in_next ) != 0, file, line );
|
|
|
|
|
} else {
|
|
|
|
|
in_next.s_addr = in_dest.s_addr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Perform routing */
|
|
|
|
|
actual.s_addr = in_dest.s_addr;
|
|
|
|
|
scope_id = ( scope ? scope->netdev->scope_id : 0 );
|
|
|
|
|
miniroute = ipv4_route ( scope_id, &actual );
|
|
|
|
|
|
|
|
|
|
/* Validate result */
|
|
|
|
|
if ( src ) {
|
|
|
|
|
|
|
|
|
|
/* Check that a route was found */
|
|
|
|
|
okx ( miniroute != NULL, file, line );
|
|
|
|
|
DBG ( "ipv4_route ( %s, %s ) = %s",
|
|
|
|
|
( scope ? scope->dev.name : "<any>" ), dest,
|
|
|
|
|
inet_ntoa ( actual ) );
|
|
|
|
|
DBG ( " from %s via %s\n",
|
|
|
|
|
inet_ntoa ( miniroute->address ), egress->dev.name );
|
|
|
|
|
|
|
|
|
|
/* Check that expected network device was used */
|
|
|
|
|
okx ( miniroute->netdev == egress->netdev, file, line );
|
|
|
|
|
|
|
|
|
|
/* Check that expected source address was used */
|
|
|
|
|
okx ( miniroute->address.s_addr == in_src.s_addr, file, line );
|
|
|
|
|
|
|
|
|
|
/* Check that expected next hop address was used */
|
|
|
|
|
okx ( actual.s_addr == in_next.s_addr, file, line );
|
|
|
|
|
|
|
|
|
|
/* Check that expected broadcast choice was used */
|
|
|
|
|
okx ( ( ! ( ( ~actual.s_addr ) & miniroute->hostmask.s_addr ) )
|
|
|
|
|
== ( !! bcast ), file, line );
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
|
|
/* Routing is expected to fail */
|
|
|
|
|
okx ( miniroute == NULL, file, line );
|
|
|
|
|
DBG ( "ipv4_route ( %s, %s ) = <unreachable>\n",
|
|
|
|
|
( scope ? scope->dev.name : "<any>" ), dest );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#define ipv4_route_ok( dest, scope, next, egress, src, bcast ) \
|
|
|
|
|
ipv4_route_okx ( dest, scope, next, egress, src, bcast, \
|
|
|
|
|
__FILE__, __LINE__ )
|
|
|
|
|
|
|
|
|
|
/** net0: Single address and gateway (DHCP assignment) */
|
|
|
|
|
TESTNET ( net0,
|
|
|
|
|
{ "dhcp/ip", "192.168.0.1" },
|
|
|
|
|
{ "dhcp/netmask", "255.255.255.0" },
|
|
|
|
|
{ "dhcp/gateway", "192.168.0.254" } );
|
|
|
|
|
|
|
|
|
|
/** net1: Single address and gateway (DHCP assignment) */
|
|
|
|
|
TESTNET ( net1,
|
|
|
|
|
{ "dhcp/ip", "192.168.0.2" },
|
|
|
|
|
{ "dhcp/netmask", "255.255.255.0" },
|
|
|
|
|
{ "dhcp/gateway", "192.168.0.254" } );
|
|
|
|
|
|
|
|
|
|
/** net2: Small /31 subnet mask */
|
|
|
|
|
TESTNET ( net2,
|
|
|
|
|
{ "ip", "10.31.31.0" },
|
|
|
|
|
{ "netmask", "255.255.255.254" },
|
|
|
|
|
{ "gateway", "10.31.31.1" } );
|
|
|
|
|
|
|
|
|
|
/** net3: Small /32 subnet mask */
|
|
|
|
|
TESTNET ( net3,
|
|
|
|
|
{ "ip", "10.32.32.32" },
|
|
|
|
|
{ "netmask", "255.255.255.255" },
|
|
|
|
|
{ "gateway", "192.168.32.254" } );
|
|
|
|
|
|
|
|
|
|
/** net4: Local subnet with no gateway */
|
|
|
|
|
TESTNET ( net4,
|
|
|
|
|
{ "ip", "192.168.86.1" },
|
|
|
|
|
{ "netmask", "255.255.240.0" } );
|
|
|
|
|
|
2015-02-18 14:11:32 +00:00
|
|
|
/**
|
|
|
|
|
* Perform IPv4 self-tests
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
static void ipv4_test_exec ( void ) {
|
|
|
|
|
|
|
|
|
|
/* Address testing macros */
|
2015-07-21 15:26:52 +01:00
|
|
|
ok ( IN_IS_CLASSA ( IPV4 ( 10, 0, 0, 1 ) ) );
|
|
|
|
|
ok ( ! IN_IS_CLASSB ( IPV4 ( 10, 0, 0, 1 ) ) );
|
|
|
|
|
ok ( ! IN_IS_CLASSC ( IPV4 ( 10, 0, 0, 1 ) ) );
|
|
|
|
|
ok ( ! IN_IS_CLASSA ( IPV4 ( 172, 16, 0, 1 ) ) );
|
|
|
|
|
ok ( IN_IS_CLASSB ( IPV4 ( 172, 16, 0, 1 ) ) );
|
|
|
|
|
ok ( ! IN_IS_CLASSC ( IPV4 ( 172, 16, 0, 1 ) ) );
|
|
|
|
|
ok ( ! IN_IS_CLASSA ( IPV4 ( 192, 168, 0, 1 ) ) );
|
|
|
|
|
ok ( ! IN_IS_CLASSB ( IPV4 ( 192, 168, 0, 1 ) ) );
|
|
|
|
|
ok ( IN_IS_CLASSC ( IPV4 ( 192, 168, 0, 1 ) ) );
|
|
|
|
|
ok ( ! IN_IS_MULTICAST ( IPV4 ( 127, 0, 0, 1 ) ) );
|
|
|
|
|
ok ( ! IN_IS_MULTICAST ( IPV4 ( 8, 8, 8, 8 ) ) );
|
|
|
|
|
ok ( ! IN_IS_MULTICAST ( IPV4 ( 0, 0, 0, 0 ) ) );
|
|
|
|
|
ok ( ! IN_IS_MULTICAST ( IPV4 ( 223, 0, 0, 1 ) ) );
|
|
|
|
|
ok ( ! IN_IS_MULTICAST ( IPV4 ( 240, 0, 0, 1 ) ) );
|
|
|
|
|
ok ( IN_IS_MULTICAST ( IPV4 ( 224, 0, 0, 1 ) ) );
|
|
|
|
|
ok ( IN_IS_MULTICAST ( IPV4 ( 231, 89, 0, 2 ) ) );
|
|
|
|
|
ok ( IN_IS_MULTICAST ( IPV4 ( 239, 6, 1, 17 ) ) );
|
2015-02-18 14:11:32 +00:00
|
|
|
|
|
|
|
|
/* inet_ntoa() tests */
|
|
|
|
|
inet_ntoa_ok ( IPV4 ( 127, 0, 0, 1 ), "127.0.0.1" );
|
|
|
|
|
inet_ntoa_ok ( IPV4 ( 0, 0, 0, 0 ), "0.0.0.0" );
|
|
|
|
|
inet_ntoa_ok ( IPV4 ( 255, 255, 255, 255 ), "255.255.255.255" );
|
|
|
|
|
inet_ntoa_ok ( IPV4 ( 212, 13, 204, 60 ), "212.13.204.60" );
|
|
|
|
|
|
|
|
|
|
/* inet_aton() tests */
|
|
|
|
|
inet_aton_ok ( "212.13.204.60", IPV4 ( 212, 13, 204, 60 ) );
|
|
|
|
|
inet_aton_ok ( "127.0.0.1", IPV4 ( 127, 0, 0, 1 ) );
|
|
|
|
|
|
|
|
|
|
/* inet_aton() failure tests */
|
|
|
|
|
inet_aton_fail_ok ( "256.0.0.1" ); /* Byte out of range */
|
|
|
|
|
inet_aton_fail_ok ( "212.13.204.60.1" ); /* Too long */
|
|
|
|
|
inet_aton_fail_ok ( "127.0.0" ); /* Too short */
|
2015-02-19 14:02:07 +00:00
|
|
|
inet_aton_fail_ok ( "1.2.3.a" ); /* Invalid characters */
|
|
|
|
|
inet_aton_fail_ok ( "127.0..1" ); /* Missing bytes */
|
2025-06-10 13:37:31 +01:00
|
|
|
|
|
|
|
|
/* Single address and gateway */
|
|
|
|
|
testnet_ok ( &net0 );
|
|
|
|
|
ipv4_route_ok ( "192.168.0.10", NULL,
|
|
|
|
|
"192.168.0.10", &net0, "192.168.0.1", 0 );
|
|
|
|
|
ipv4_route_ok ( "10.0.0.6", NULL,
|
|
|
|
|
"192.168.0.254", &net0, "192.168.0.1", 0 );
|
|
|
|
|
ipv4_route_ok ( "192.168.0.255", NULL,
|
|
|
|
|
"192.168.0.255", &net0, "192.168.0.1", 1 );
|
|
|
|
|
testnet_remove_ok ( &net0 );
|
|
|
|
|
|
|
|
|
|
/* Overridden DHCP-assigned address */
|
|
|
|
|
testnet_ok ( &net1 );
|
|
|
|
|
ipv4_route_ok ( "192.168.1.3", NULL,
|
|
|
|
|
"192.168.0.254", &net1, "192.168.0.2", 0 );
|
|
|
|
|
testnet_set_ok ( &net1, "ip", "192.168.1.2" );
|
|
|
|
|
ipv4_route_ok ( "192.168.1.3", NULL,
|
|
|
|
|
"192.168.1.3", &net1, "192.168.1.2", 0 );
|
|
|
|
|
testnet_remove_ok ( &net1 );
|
|
|
|
|
|
|
|
|
|
/* Small /31 subnet */
|
|
|
|
|
testnet_ok ( &net2 );
|
|
|
|
|
ipv4_route_ok ( "10.31.31.1", NULL,
|
|
|
|
|
"10.31.31.1", &net2, "10.31.31.0", 0 );
|
|
|
|
|
ipv4_route_ok ( "212.13.204.60", NULL,
|
|
|
|
|
"10.31.31.1", &net2, "10.31.31.0", 0 );
|
|
|
|
|
testnet_remove_ok ( &net2 );
|
|
|
|
|
|
|
|
|
|
/* Small /32 subnet */
|
|
|
|
|
testnet_ok ( &net3 );
|
|
|
|
|
ipv4_route_ok ( "10.32.32.31", NULL,
|
|
|
|
|
"192.168.32.254", &net3, "10.32.32.32", 0 );
|
|
|
|
|
ipv4_route_ok ( "8.8.8.8", NULL,
|
|
|
|
|
"192.168.32.254", &net3, "10.32.32.32", 0 );
|
|
|
|
|
testnet_remove_ok ( &net3 );
|
|
|
|
|
|
|
|
|
|
/* No gateway */
|
|
|
|
|
testnet_ok ( &net4 );
|
|
|
|
|
ipv4_route_ok ( "192.168.87.1", NULL,
|
|
|
|
|
"192.168.87.1", &net4, "192.168.86.1", 0 );
|
|
|
|
|
ipv4_route_ok ( "192.168.96.1", NULL, NULL, NULL, NULL, 0 );
|
|
|
|
|
testnet_remove_ok ( &net4 );
|
2025-06-05 16:49:42 +01:00
|
|
|
|
|
|
|
|
/* Multiple interfaces */
|
|
|
|
|
testnet_ok ( &net0 );
|
|
|
|
|
testnet_ok ( &net1 );
|
|
|
|
|
testnet_ok ( &net2 );
|
|
|
|
|
testnet_close_ok ( &net1 );
|
|
|
|
|
ipv4_route_ok ( "192.168.0.9", NULL,
|
|
|
|
|
"192.168.0.9", &net0, "192.168.0.1", 0 );
|
|
|
|
|
ipv4_route_ok ( "10.31.31.1", NULL,
|
|
|
|
|
"10.31.31.1", &net2, "10.31.31.0", 0 );
|
|
|
|
|
testnet_close_ok ( &net0 );
|
|
|
|
|
testnet_open_ok ( &net1 );
|
|
|
|
|
ipv4_route_ok ( "192.168.0.9", NULL,
|
|
|
|
|
"192.168.0.9", &net1, "192.168.0.2", 0 );
|
|
|
|
|
ipv4_route_ok ( "10.31.31.1", NULL,
|
|
|
|
|
"10.31.31.1", &net2, "10.31.31.0", 0 );
|
|
|
|
|
testnet_close_ok ( &net2 );
|
|
|
|
|
ipv4_route_ok ( "8.8.8.8", NULL,
|
|
|
|
|
"192.168.0.254", &net1, "192.168.0.2", 0 );
|
|
|
|
|
testnet_close_ok ( &net1 );
|
|
|
|
|
testnet_open_ok ( &net0 );
|
|
|
|
|
ipv4_route_ok ( "8.8.8.8", NULL,
|
|
|
|
|
"192.168.0.254", &net0, "192.168.0.1", 0 );
|
|
|
|
|
testnet_close_ok ( &net0 );
|
|
|
|
|
testnet_open_ok ( &net2 );
|
|
|
|
|
ipv4_route_ok ( "8.8.8.8", NULL,
|
|
|
|
|
"10.31.31.1", &net2, "10.31.31.0", 0 );
|
|
|
|
|
testnet_remove_ok ( &net2 );
|
|
|
|
|
testnet_remove_ok ( &net1 );
|
|
|
|
|
testnet_remove_ok ( &net0 );
|
2015-02-18 14:11:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** IPv4 self-test */
|
|
|
|
|
struct self_test ipv4_test __self_test = {
|
|
|
|
|
.name = "ipv4",
|
|
|
|
|
.exec = ipv4_test_exec,
|
|
|
|
|
};
|