2019-05-30 02:57:55 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2017-03-27 12:45:16 +03:00
/**
* Userspace PCI Endpoint Test Module
*
* Copyright ( C ) 2017 Texas Instruments
* Author : Kishon Vijay Abraham I < kishon @ ti . com >
*/
# include <errno.h>
# include <fcntl.h>
# include <stdbool.h>
# include <stdio.h>
# include <stdlib.h>
# include <sys/ioctl.h>
# include <unistd.h>
# include <linux/pcitest.h>
# define BILLION 1E9
static char * result [ ] = { " NOT OKAY " , " OKAY " } ;
2018-07-19 11:32:21 +03:00
static char * irq [ ] = { " LEGACY " , " MSI " , " MSI-X " } ;
2017-03-27 12:45:16 +03:00
struct pci_test {
char * device ;
char barnum ;
bool legacyirq ;
unsigned int msinum ;
2018-07-19 11:32:21 +03:00
unsigned int msixnum ;
int irqtype ;
bool set_irqtype ;
bool get_irqtype ;
2020-03-17 13:01:56 +03:00
bool clear_irq ;
2017-03-27 12:45:16 +03:00
bool read ;
bool write ;
bool copy ;
unsigned long size ;
2020-03-16 14:24:23 +03:00
bool use_dma ;
2017-03-27 12:45:16 +03:00
} ;
2019-05-24 00:18:00 +03:00
static int run_test ( struct pci_test * test )
2017-03-27 12:45:16 +03:00
{
2021-07-14 16:23:31 +03:00
struct pci_endpoint_test_xfer_param param = { } ;
2019-05-24 00:18:00 +03:00
int ret = - EINVAL ;
2017-03-27 12:45:16 +03:00
int fd ;
fd = open ( test - > device , O_RDWR ) ;
if ( fd < 0 ) {
perror ( " can't open PCI Endpoint Test device " ) ;
2019-05-24 00:18:00 +03:00
return - ENODEV ;
2017-03-27 12:45:16 +03:00
}
if ( test - > barnum > = 0 & & test - > barnum < = 5 ) {
ret = ioctl ( fd , PCITEST_BAR , test - > barnum ) ;
fprintf ( stdout , " BAR%d: \t \t " , test - > barnum ) ;
if ( ret < 0 )
fprintf ( stdout , " TEST FAILED \n " ) ;
else
fprintf ( stdout , " %s \n " , result [ ret ] ) ;
}
2018-07-19 11:32:21 +03:00
if ( test - > set_irqtype ) {
ret = ioctl ( fd , PCITEST_SET_IRQTYPE , test - > irqtype ) ;
fprintf ( stdout , " SET IRQ TYPE TO %s: \t \t " , irq [ test - > irqtype ] ) ;
if ( ret < 0 )
fprintf ( stdout , " FAILED \n " ) ;
else
fprintf ( stdout , " %s \n " , result [ ret ] ) ;
}
if ( test - > get_irqtype ) {
ret = ioctl ( fd , PCITEST_GET_IRQTYPE ) ;
fprintf ( stdout , " GET IRQ TYPE: \t \t " ) ;
if ( ret < 0 )
fprintf ( stdout , " FAILED \n " ) ;
else
fprintf ( stdout , " %s \n " , irq [ ret ] ) ;
}
2020-03-17 13:01:56 +03:00
if ( test - > clear_irq ) {
ret = ioctl ( fd , PCITEST_CLEAR_IRQ ) ;
fprintf ( stdout , " CLEAR IRQ: \t \t " ) ;
if ( ret < 0 )
fprintf ( stdout , " FAILED \n " ) ;
else
fprintf ( stdout , " %s \n " , result [ ret ] ) ;
}
2017-03-27 12:45:16 +03:00
if ( test - > legacyirq ) {
ret = ioctl ( fd , PCITEST_LEGACY_IRQ , 0 ) ;
fprintf ( stdout , " LEGACY IRQ: \t " ) ;
if ( ret < 0 )
fprintf ( stdout , " TEST FAILED \n " ) ;
else
fprintf ( stdout , " %s \n " , result [ ret ] ) ;
}
if ( test - > msinum > 0 & & test - > msinum < = 32 ) {
ret = ioctl ( fd , PCITEST_MSI , test - > msinum ) ;
fprintf ( stdout , " MSI%d: \t \t " , test - > msinum ) ;
if ( ret < 0 )
fprintf ( stdout , " TEST FAILED \n " ) ;
else
fprintf ( stdout , " %s \n " , result [ ret ] ) ;
}
2018-07-19 11:32:21 +03:00
if ( test - > msixnum > 0 & & test - > msixnum < = 2048 ) {
ret = ioctl ( fd , PCITEST_MSIX , test - > msixnum ) ;
fprintf ( stdout , " MSI-X%d: \t \t " , test - > msixnum ) ;
if ( ret < 0 )
fprintf ( stdout , " TEST FAILED \n " ) ;
else
fprintf ( stdout , " %s \n " , result [ ret ] ) ;
}
2017-03-27 12:45:16 +03:00
if ( test - > write ) {
2020-03-16 14:24:23 +03:00
param . size = test - > size ;
if ( test - > use_dma )
param . flags = PCITEST_FLAGS_USE_DMA ;
ret = ioctl ( fd , PCITEST_WRITE , & param ) ;
2017-03-27 12:45:16 +03:00
fprintf ( stdout , " WRITE (%7ld bytes): \t \t " , test - > size ) ;
if ( ret < 0 )
fprintf ( stdout , " TEST FAILED \n " ) ;
else
fprintf ( stdout , " %s \n " , result [ ret ] ) ;
}
if ( test - > read ) {
2020-03-16 14:24:23 +03:00
param . size = test - > size ;
if ( test - > use_dma )
param . flags = PCITEST_FLAGS_USE_DMA ;
ret = ioctl ( fd , PCITEST_READ , & param ) ;
2017-03-27 12:45:16 +03:00
fprintf ( stdout , " READ (%7ld bytes): \t \t " , test - > size ) ;
if ( ret < 0 )
fprintf ( stdout , " TEST FAILED \n " ) ;
else
fprintf ( stdout , " %s \n " , result [ ret ] ) ;
}
if ( test - > copy ) {
2020-03-16 14:24:23 +03:00
param . size = test - > size ;
if ( test - > use_dma )
param . flags = PCITEST_FLAGS_USE_DMA ;
ret = ioctl ( fd , PCITEST_COPY , & param ) ;
2017-03-27 12:45:16 +03:00
fprintf ( stdout , " COPY (%7ld bytes): \t \t " , test - > size ) ;
if ( ret < 0 )
fprintf ( stdout , " TEST FAILED \n " ) ;
else
fprintf ( stdout , " %s \n " , result [ ret ] ) ;
}
fflush ( stdout ) ;
2019-10-26 04:35:55 +03:00
close ( fd ) ;
2018-09-20 18:02:53 +03:00
return ( ret < 0 ) ? ret : 1 - ret ; /* return 0 if test succeeded */
2017-03-27 12:45:16 +03:00
}
int main ( int argc , char * * argv )
{
int c ;
struct pci_test * test ;
test = calloc ( 1 , sizeof ( * test ) ) ;
if ( ! test ) {
perror ( " Fail to allocate memory for pci_test \n " ) ;
return - ENOMEM ;
}
/* since '0' is a valid BAR number, initialize it to -1 */
test - > barnum = - 1 ;
/* set default size as 100KB */
test - > size = 0x19000 ;
/* set default endpoint device */
test - > device = " /dev/pci-endpoint-test.0 " ;
2020-03-17 13:01:56 +03:00
while ( ( c = getopt ( argc , argv , " D:b:m:x:i:deIlhrwcs: " ) ) ! = EOF )
2017-03-27 12:45:16 +03:00
switch ( c ) {
case ' D ' :
test - > device = optarg ;
continue ;
case ' b ' :
test - > barnum = atoi ( optarg ) ;
if ( test - > barnum < 0 | | test - > barnum > 5 )
goto usage ;
continue ;
case ' l ' :
test - > legacyirq = true ;
continue ;
case ' m ' :
test - > msinum = atoi ( optarg ) ;
if ( test - > msinum < 1 | | test - > msinum > 32 )
goto usage ;
continue ;
2018-07-19 11:32:21 +03:00
case ' x ' :
test - > msixnum = atoi ( optarg ) ;
if ( test - > msixnum < 1 | | test - > msixnum > 2048 )
goto usage ;
continue ;
case ' i ' :
test - > irqtype = atoi ( optarg ) ;
if ( test - > irqtype < 0 | | test - > irqtype > 2 )
goto usage ;
test - > set_irqtype = true ;
continue ;
case ' I ' :
test - > get_irqtype = true ;
continue ;
2017-03-27 12:45:16 +03:00
case ' r ' :
test - > read = true ;
continue ;
case ' w ' :
test - > write = true ;
continue ;
case ' c ' :
test - > copy = true ;
continue ;
2020-03-17 13:01:56 +03:00
case ' e ' :
test - > clear_irq = true ;
continue ;
2017-03-27 12:45:16 +03:00
case ' s ' :
test - > size = strtoul ( optarg , NULL , 0 ) ;
continue ;
2020-03-16 14:24:23 +03:00
case ' d ' :
test - > use_dma = true ;
continue ;
2017-03-27 12:45:16 +03:00
case ' h ' :
default :
usage :
fprintf ( stderr ,
" usage: %s [options] \n "
" Options: \n "
" \t -D <dev> PCI endpoint test device {default: /dev/pci-endpoint-test.0} \n "
" \t -b <bar num> BAR test (bar number between 0..5) \n "
" \t -m <msi num> MSI test (msi number between 1..32) \n "
2018-07-19 11:32:21 +03:00
" \t -x <msix num> \t MSI-X test (msix number between 1..2048) \n "
" \t -i <irq type> \t Set IRQ type (0 - Legacy, 1 - MSI, 2 - MSI-X) \n "
2020-03-17 13:01:56 +03:00
" \t -e Clear IRQ \n "
2018-07-19 11:32:21 +03:00
" \t -I Get current IRQ type configured \n "
2020-03-16 14:24:23 +03:00
" \t -d Use DMA \n "
2017-08-18 17:58:10 +03:00
" \t -l Legacy IRQ test \n "
2017-03-27 12:45:16 +03:00
" \t -r Read buffer test \n "
" \t -w Write buffer test \n "
" \t -c Copy buffer test \n "
2019-05-24 00:18:01 +03:00
" \t -s <size> Size of buffer {default: 100KB} \n "
2019-04-04 14:06:08 +03:00
" \t -h Print this help message \n " ,
2017-03-27 12:45:16 +03:00
argv [ 0 ] ) ;
return - EINVAL ;
}
2018-09-20 18:02:53 +03:00
return run_test ( test ) ;
2017-03-27 12:45:16 +03:00
}