mirror of
https://github.com/ipxe/ipxe
synced 2026-01-21 18:30:56 +03:00
[test] Add infrastructure for test network devices
Signed-off-by: Michael Brown <mcb30@ipxe.org>
This commit is contained in:
218
src/tests/netdev_test.c
Normal file
218
src/tests/netdev_test.c
Normal file
@@ -0,0 +1,218 @@
|
||||
/*
|
||||
* Copyright (C) 2025 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 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.
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
|
||||
|
||||
/** @file
|
||||
*
|
||||
* Network device tests
|
||||
*
|
||||
*/
|
||||
|
||||
/* Forcibly enable assertions */
|
||||
#undef NDEBUG
|
||||
|
||||
#include <string.h>
|
||||
#include <stdio.h>
|
||||
#include <ipxe/netdevice.h>
|
||||
#include <ipxe/ethernet.h>
|
||||
#include <ipxe/test.h>
|
||||
#include "netdev_test.h"
|
||||
|
||||
/**
|
||||
* Open network device
|
||||
*
|
||||
* @v netdev Network device
|
||||
* @ret rc Return status code
|
||||
*/
|
||||
static int testnet_open ( struct net_device *netdev __unused ) {
|
||||
|
||||
/* Do nothing, successfully */
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Close network device
|
||||
*
|
||||
* @v netdev Network device
|
||||
*/
|
||||
static void testnet_close ( struct net_device *netdev __unused ) {
|
||||
|
||||
/* Do nothing */
|
||||
}
|
||||
|
||||
/**
|
||||
* Transmit packet
|
||||
*
|
||||
* @v netdev Network device
|
||||
* @v iobuf I/O buffer
|
||||
* @ret rc Return status code
|
||||
*/
|
||||
static int testnet_transmit ( struct net_device *netdev,
|
||||
struct io_buffer *iobuf ) {
|
||||
|
||||
/* Complete immediately */
|
||||
netdev_tx_complete ( netdev, iobuf );
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Poll for completed and received packets
|
||||
*
|
||||
* @v netdev Network device
|
||||
*/
|
||||
static void testnet_poll ( struct net_device *netdev __unused ) {
|
||||
|
||||
/* Do nothing */
|
||||
}
|
||||
|
||||
/** Test network device operations */
|
||||
static struct net_device_operations testnet_operations = {
|
||||
.open = testnet_open,
|
||||
.close = testnet_close,
|
||||
.transmit = testnet_transmit,
|
||||
.poll = testnet_poll,
|
||||
};
|
||||
|
||||
/**
|
||||
* Report a network device creation test result
|
||||
*
|
||||
* @v testnet Test network device
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
void testnet_okx ( struct testnet *testnet, const char *file,
|
||||
unsigned int line ) {
|
||||
struct testnet_setting *testset;
|
||||
unsigned int i;
|
||||
|
||||
/* Allocate device */
|
||||
testnet->netdev = alloc_etherdev ( 0 );
|
||||
okx ( testnet->netdev != NULL, file, line );
|
||||
netdev_init ( testnet->netdev, &testnet_operations );
|
||||
testnet->netdev->dev = &testnet->dev;
|
||||
snprintf ( testnet->netdev->name, sizeof ( testnet->netdev->name ),
|
||||
"%s", testnet->dev.name );
|
||||
|
||||
/* Register device */
|
||||
okx ( register_netdev ( testnet->netdev ) == 0, file, line );
|
||||
|
||||
/* Open device */
|
||||
testnet_open_okx ( testnet, file, line );
|
||||
|
||||
/* Apply initial settings */
|
||||
for ( i = 0 ; i < testnet->count ; i++ ) {
|
||||
testset = &testnet->testset[i];
|
||||
testnet_set_okx ( testnet, testset->name, testset->value,
|
||||
file, line );
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Report a network device opening test result
|
||||
*
|
||||
* @v testnet Test network device
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
void testnet_open_okx ( struct testnet *testnet, const char *file,
|
||||
unsigned int line ) {
|
||||
|
||||
/* Sanity check */
|
||||
okx ( testnet->netdev != NULL, file, line );
|
||||
|
||||
/* Open device */
|
||||
okx ( netdev_open ( testnet->netdev ) == 0, file, line );
|
||||
}
|
||||
|
||||
/**
|
||||
* Report a network device setting test result
|
||||
*
|
||||
* @v testnet Test network device
|
||||
* @v name Setting name (relative to network device's settings)
|
||||
* @v value Setting value
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
void testnet_set_okx ( struct testnet *testnet, const char *name,
|
||||
const char *value, const char *file,
|
||||
unsigned int line ) {
|
||||
char fullname[ strlen ( testnet->dev.name ) + 1 /* "." or "/" */ +
|
||||
strlen ( name ) + 1 /* NUL */ ];
|
||||
struct settings *settings;
|
||||
struct setting setting;
|
||||
|
||||
/* Sanity check */
|
||||
okx ( testnet->netdev != NULL, file, line );
|
||||
settings = netdev_settings ( testnet->netdev );
|
||||
okx ( settings != NULL, file, line );
|
||||
okx ( strcmp ( settings->name, testnet->dev.name ) == 0, file, line );
|
||||
|
||||
/* Construct setting name */
|
||||
snprintf ( fullname, sizeof ( fullname ), "%s%c%s", testnet->dev.name,
|
||||
( strchr ( name, '/' ) ? '.' : '/' ), name );
|
||||
|
||||
/* Parse setting name */
|
||||
okx ( parse_setting_name ( fullname, autovivify_child_settings,
|
||||
&settings, &setting ) == 0, file, line );
|
||||
|
||||
/* Apply setting */
|
||||
okx ( storef_setting ( settings, &setting, value ) == 0, file, line );
|
||||
}
|
||||
|
||||
/**
|
||||
* Report a network device closing test result
|
||||
*
|
||||
* @v testnet Test network device
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
void testnet_close_okx ( struct testnet *testnet, const char *file,
|
||||
unsigned int line ) {
|
||||
|
||||
/* Sanity check */
|
||||
okx ( testnet->netdev != NULL, file, line );
|
||||
|
||||
/* Close device */
|
||||
netdev_close ( testnet->netdev );
|
||||
}
|
||||
|
||||
/**
|
||||
* Report a network device removal test result
|
||||
*
|
||||
* @v testnet Test network device
|
||||
* @v file Test code file
|
||||
* @v line Test code line
|
||||
*/
|
||||
void testnet_remove_okx ( struct testnet *testnet, const char *file,
|
||||
unsigned int line ) {
|
||||
|
||||
/* Sanity check */
|
||||
okx ( testnet->netdev != NULL, file, line );
|
||||
|
||||
/* Remove device */
|
||||
unregister_netdev ( testnet->netdev );
|
||||
netdev_nullify ( testnet->netdev );
|
||||
netdev_put ( testnet->netdev );
|
||||
testnet->netdev = NULL;
|
||||
}
|
||||
111
src/tests/netdev_test.h
Normal file
111
src/tests/netdev_test.h
Normal file
@@ -0,0 +1,111 @@
|
||||
#ifndef _NETDEV_TEST_H
|
||||
#define _NETDEV_TEST_H
|
||||
|
||||
/** @file
|
||||
*
|
||||
* Network device tests
|
||||
*
|
||||
*/
|
||||
|
||||
FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
|
||||
|
||||
#include <ipxe/device.h>
|
||||
#include <ipxe/netdevice.h>
|
||||
|
||||
/** A test network device setting */
|
||||
struct testnet_setting {
|
||||
/** Setting name (relative to network device's settings) */
|
||||
const char *name;
|
||||
/** Value */
|
||||
const char *value;
|
||||
};
|
||||
|
||||
/** A test network device */
|
||||
struct testnet {
|
||||
/** Network device */
|
||||
struct net_device *netdev;
|
||||
/** Dummy physical device */
|
||||
struct device dev;
|
||||
/** Initial settings */
|
||||
struct testnet_setting *testset;
|
||||
/** Number of initial settings */
|
||||
unsigned int count;
|
||||
};
|
||||
|
||||
/**
|
||||
* Declare a test network device
|
||||
*
|
||||
* @v NAME Network device name
|
||||
* @v ... Initial network device settings
|
||||
*/
|
||||
#define TESTNET( NAME, ... ) \
|
||||
static struct testnet_setting NAME ## _setting[] = { \
|
||||
__VA_ARGS__ \
|
||||
}; \
|
||||
static struct testnet NAME = { \
|
||||
.dev = { \
|
||||
.name = #NAME, \
|
||||
.driver_name = "testnet", \
|
||||
.siblings = \
|
||||
LIST_HEAD_INIT ( NAME.dev.siblings ), \
|
||||
.children = \
|
||||
LIST_HEAD_INIT ( NAME.dev.children ), \
|
||||
}, \
|
||||
.testset = NAME ## _setting, \
|
||||
.count = ( sizeof ( NAME ## _setting ) / \
|
||||
sizeof ( NAME ## _setting[0] ) ), \
|
||||
};
|
||||
|
||||
/**
|
||||
* Report a network device creation test result
|
||||
*
|
||||
* @v testnet Test network device
|
||||
*/
|
||||
#define testnet_ok( testnet ) testnet_okx ( testnet, __FILE__, __LINE__ )
|
||||
extern void testnet_okx ( struct testnet *testnet, const char *file,
|
||||
unsigned int line );
|
||||
|
||||
/**
|
||||
* Report a network device opening test result
|
||||
*
|
||||
* @v testnet Test network device
|
||||
*/
|
||||
#define testnet_open_ok( testnet ) \
|
||||
testnet_open_okx ( testnet, __FILE__, __LINE__ )
|
||||
extern void testnet_open_okx ( struct testnet *testnet, const char *file,
|
||||
unsigned int line );
|
||||
|
||||
/**
|
||||
* Report a network device setting test result
|
||||
*
|
||||
* @v testnet Test network device
|
||||
* @v name Setting name (relative to network device's settings)
|
||||
* @v value Setting value
|
||||
*/
|
||||
#define testnet_set_ok( testnet, name, value ) \
|
||||
testnet_set_okx ( testnet, name, value, __FILE__, __LINE__ )
|
||||
extern void testnet_set_okx ( struct testnet *testnet, const char *name,
|
||||
const char *value, const char *file,
|
||||
unsigned int line );
|
||||
|
||||
/**
|
||||
* Report a network device closing test result
|
||||
*
|
||||
* @v testnet Test network device
|
||||
*/
|
||||
#define testnet_close_ok( testnet ) \
|
||||
testnet_close_okx ( testnet, __FILE__, __LINE__ )
|
||||
extern void testnet_close_okx ( struct testnet *testnet, const char *file,
|
||||
unsigned int line );
|
||||
|
||||
/**
|
||||
* Report a network device removal test result
|
||||
*
|
||||
* @v testnet Test network device
|
||||
*/
|
||||
#define testnet_remove_ok( testnet ) \
|
||||
testnet_remove_okx ( testnet, __FILE__, __LINE__ )
|
||||
extern void testnet_remove_okx ( struct testnet *testnet, const char *file,
|
||||
unsigned int line );
|
||||
|
||||
#endif /* _NETDEV_TEST_H */
|
||||
Reference in New Issue
Block a user