2019-05-28 09:57:07 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2007-07-31 00:38:44 -07:00
/*
* SPI testing utility ( using spidev driver )
*
* Copyright ( c ) 2007 MontaVista Software , Inc .
* Copyright ( c ) 2007 Anton Vorontsov < avorontsov @ ru . mvista . com >
*
* Cross - compile with cross - gcc - I / path / to / cross - kernel / include
*/
# include <stdint.h>
# include <unistd.h>
# include <stdio.h>
# include <stdlib.h>
2015-03-10 16:12:30 -04:00
# include <string.h>
2020-02-13 12:16:07 +08:00
# include <errno.h>
2007-07-31 00:38:44 -07:00
# include <getopt.h>
# include <fcntl.h>
2017-03-21 16:48:46 +01:00
# include <time.h>
2007-07-31 00:38:44 -07:00
# include <sys/ioctl.h>
2016-08-12 16:04:33 +03:00
# include <linux/ioctl.h>
2015-11-18 14:30:39 -08:00
# include <sys/stat.h>
2007-07-31 00:38:44 -07:00
# include <linux/types.h>
# include <linux/spi/spidev.h>
# define ARRAY_SIZE(a) (sizeof(a) / sizeof((a)[0]))
static void pabort ( const char * s )
{
2020-02-13 12:16:07 +08:00
if ( errno ! = 0 )
perror ( s ) ;
else
printf ( " %s \n " , s ) ;
2007-07-31 00:38:44 -07:00
abort ( ) ;
}
2007-10-16 01:27:47 -07:00
static const char * device = " /dev/spidev1.1 " ;
2014-02-25 11:40:18 +01:00
static uint32_t mode ;
2007-07-31 00:38:44 -07:00
static uint8_t bits = 8 ;
2015-11-18 14:30:39 -08:00
static char * input_file ;
2015-11-18 14:30:40 -08:00
static char * output_file ;
2007-07-31 00:38:44 -07:00
static uint32_t speed = 500000 ;
static uint16_t delay ;
2015-03-10 16:12:31 -04:00
static int verbose ;
2017-03-21 16:48:46 +01:00
static int transfer_size ;
static int iterations ;
static int interval = 5 ; /* interval in seconds for showing transfer rate */
2007-07-31 00:38:44 -07:00
2020-06-11 20:56:51 +08:00
static uint8_t default_tx [ ] = {
2015-03-10 16:12:32 -04:00
0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF ,
0x40 , 0x00 , 0x00 , 0x00 , 0x00 , 0x95 ,
0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF ,
0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF ,
0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF ,
0xF0 , 0x0D ,
} ;
2020-06-11 20:56:51 +08:00
static uint8_t default_rx [ ARRAY_SIZE ( default_tx ) ] = { 0 , } ;
static char * input_tx ;
2015-03-10 16:12:32 -04:00
2015-11-18 14:30:42 -08:00
static void hex_dump ( const void * src , size_t length , size_t line_size ,
char * prefix )
2015-03-10 16:12:30 -04:00
{
int i = 0 ;
const unsigned char * address = src ;
const unsigned char * line = address ;
unsigned char c ;
printf ( " %s | " , prefix ) ;
while ( length - - > 0 ) {
printf ( " %02X " , * address + + ) ;
if ( ! ( + + i % line_size ) | | ( length = = 0 & & i % line_size ) ) {
if ( length = = 0 ) {
while ( i + + % line_size )
printf ( " __ " ) ;
}
2018-09-03 19:33:23 +02:00
printf ( " | " ) ;
2015-03-10 16:12:30 -04:00
while ( line < address ) {
c = * line + + ;
2018-09-03 19:33:23 +02:00
printf ( " %c " , ( c < 32 | | c > 126 ) ? ' . ' : c ) ;
2015-03-10 16:12:30 -04:00
}
2018-09-03 19:33:23 +02:00
printf ( " | \n " ) ;
2015-03-10 16:12:30 -04:00
if ( length > 0 )
printf ( " %s | " , prefix ) ;
}
}
}
2015-03-10 16:12:32 -04:00
/*
* Unescape - process hexadecimal escape character
* converts shell input " \x23 " - > 0x23
*/
2015-04-16 12:49:29 -07:00
static int unescape ( char * _dst , char * _src , size_t len )
2015-03-10 16:12:32 -04:00
{
int ret = 0 ;
2015-11-18 14:30:41 -08:00
int match ;
2015-03-10 16:12:32 -04:00
char * src = _src ;
char * dst = _dst ;
unsigned int ch ;
while ( * src ) {
if ( * src = = ' \\ ' & & * ( src + 1 ) = = ' x ' ) {
2015-11-18 14:30:41 -08:00
match = sscanf ( src + 2 , " %2x " , & ch ) ;
if ( ! match )
pabort ( " malformed input string " ) ;
2015-03-10 16:12:32 -04:00
src + = 4 ;
* dst + + = ( unsigned char ) ch ;
} else {
* dst + + = * src + + ;
}
ret + + ;
}
return ret ;
}
static void transfer ( int fd , uint8_t const * tx , uint8_t const * rx , size_t len )
2007-07-31 00:38:44 -07:00
{
int ret ;
2015-11-18 14:30:40 -08:00
int out_fd ;
2007-07-31 00:38:44 -07:00
struct spi_ioc_transfer tr = {
. tx_buf = ( unsigned long ) tx ,
. rx_buf = ( unsigned long ) rx ,
2015-03-10 16:12:32 -04:00
. len = len ,
2007-07-31 00:38:44 -07:00
. delay_usecs = delay ,
. speed_hz = speed ,
. bits_per_word = bits ,
} ;
2020-04-16 12:18:35 +02:00
if ( mode & SPI_TX_OCTAL )
tr . tx_nbits = 8 ;
else if ( mode & SPI_TX_QUAD )
2014-02-25 11:40:18 +01:00
tr . tx_nbits = 4 ;
else if ( mode & SPI_TX_DUAL )
tr . tx_nbits = 2 ;
2020-04-16 12:18:35 +02:00
if ( mode & SPI_RX_OCTAL )
tr . rx_nbits = 8 ;
else if ( mode & SPI_RX_QUAD )
2014-02-25 11:40:18 +01:00
tr . rx_nbits = 4 ;
else if ( mode & SPI_RX_DUAL )
tr . rx_nbits = 2 ;
if ( ! ( mode & SPI_LOOP ) ) {
2020-04-16 12:18:35 +02:00
if ( mode & ( SPI_TX_OCTAL | SPI_TX_QUAD | SPI_TX_DUAL ) )
2014-02-25 11:40:18 +01:00
tr . rx_buf = 0 ;
2020-04-16 12:18:35 +02:00
else if ( mode & ( SPI_RX_OCTAL | SPI_RX_QUAD | SPI_RX_DUAL ) )
2014-02-25 11:40:18 +01:00
tr . tx_buf = 0 ;
}
2007-07-31 00:38:44 -07:00
ret = ioctl ( fd , SPI_IOC_MESSAGE ( 1 ) , & tr ) ;
2010-04-29 15:02:28 -07:00
if ( ret < 1 )
2007-07-31 00:38:44 -07:00
pabort ( " can't send spi message " ) ;
2015-03-10 16:12:31 -04:00
if ( verbose )
2015-03-10 16:12:32 -04:00
hex_dump ( tx , len , 32 , " TX " ) ;
2015-11-18 14:30:40 -08:00
if ( output_file ) {
out_fd = open ( output_file , O_WRONLY | O_CREAT | O_TRUNC , 0666 ) ;
if ( out_fd < 0 )
pabort ( " could not open output file " ) ;
ret = write ( out_fd , rx , len ) ;
if ( ret ! = len )
2015-12-04 10:59:14 -02:00
pabort ( " not all bytes written to output file " ) ;
2015-11-18 14:30:40 -08:00
close ( out_fd ) ;
}
2017-03-21 16:48:46 +01:00
if ( verbose )
2015-11-18 14:30:40 -08:00
hex_dump ( rx , len , 32 , " RX " ) ;
2007-07-31 00:38:44 -07:00
}
2009-09-22 16:43:42 -07:00
static void print_usage ( const char * prog )
2007-07-31 00:38:44 -07:00
{
2023-05-30 16:16:41 +02:00
printf ( " Usage: %s [-2348CDFHILMNORSZbdilopsv] \n " , prog ) ;
2023-05-30 16:16:40 +02:00
puts ( " general device settings: \n "
" -D --device device to use (default /dev/spidev1.1) \n "
" -s --speed max speed (Hz) \n "
" -d --delay delay (usec) \n "
" -l --loop loopback \n "
" spi mode: \n "
" -H --cpha clock phase \n "
" -O --cpol clock polarity \n "
2023-05-30 16:16:41 +02:00
" -F --rx-cpha-flip flip CPHA on Rx only xfer \n "
2023-05-30 16:16:40 +02:00
" number of wires for transmission: \n "
" -2 --dual dual transfer \n "
" -4 --quad quad transfer \n "
" -8 --octal octal transfer \n "
" -3 --3wire SI/SO signals shared \n "
2023-05-30 16:16:41 +02:00
" -Z --3wire-hiz high impedance turnaround \n "
2023-05-30 16:16:40 +02:00
" data: \n "
" -i --input input data from a file (e.g. \" test.bin \" ) \n "
" -o --output output data to a file (e.g. \" results.bin \" ) \n "
" -p Send data (e.g. \" 1234 \\ xde \\ xad \" ) \n "
" -S --size transfer size \n "
" -I --iter iterations \n "
" additional parameters: \n "
" -b --bpw bits per word \n "
" -L --lsb least significant bit first \n "
" -C --cs-high chip select active high \n "
" -N --no-cs no chip select \n "
" -R --ready slave pulls low to pause \n "
2023-05-30 16:16:41 +02:00
" -M --mosi-idle-low leave mosi line low when idle \n "
2023-05-30 16:16:40 +02:00
" misc: \n "
" -v --verbose Verbose (show tx buffer) \n " ) ;
2007-07-31 00:38:44 -07:00
exit ( 1 ) ;
}
2009-09-22 16:43:42 -07:00
static void parse_opts ( int argc , char * argv [ ] )
2007-07-31 00:38:44 -07:00
{
while ( 1 ) {
2007-10-16 01:27:47 -07:00
static const struct option lopts [ ] = {
2023-05-30 16:16:40 +02:00
{ " device " , 1 , 0 , ' D ' } ,
{ " speed " , 1 , 0 , ' s ' } ,
{ " delay " , 1 , 0 , ' d ' } ,
{ " loop " , 0 , 0 , ' l ' } ,
{ " cpha " , 0 , 0 , ' H ' } ,
{ " cpol " , 0 , 0 , ' O ' } ,
2023-05-30 16:16:41 +02:00
{ " rx-cpha-flip " , 0 , 0 , ' F ' } ,
2023-05-30 16:16:40 +02:00
{ " dual " , 0 , 0 , ' 2 ' } ,
{ " quad " , 0 , 0 , ' 4 ' } ,
{ " octal " , 0 , 0 , ' 8 ' } ,
{ " 3wire " , 0 , 0 , ' 3 ' } ,
2023-05-30 16:16:41 +02:00
{ " 3wire-hiz " , 0 , 0 , ' Z ' } ,
2023-05-30 16:16:40 +02:00
{ " input " , 1 , 0 , ' i ' } ,
{ " output " , 1 , 0 , ' o ' } ,
{ " size " , 1 , 0 , ' S ' } ,
{ " iter " , 1 , 0 , ' I ' } ,
{ " bpw " , 1 , 0 , ' b ' } ,
{ " lsb " , 0 , 0 , ' L ' } ,
{ " cs-high " , 0 , 0 , ' C ' } ,
{ " no-cs " , 0 , 0 , ' N ' } ,
{ " ready " , 0 , 0 , ' R ' } ,
2023-05-30 16:16:41 +02:00
{ " mosi-idle-low " , 0 , 0 , ' M ' } ,
2023-05-30 16:16:40 +02:00
{ " verbose " , 0 , 0 , ' v ' } ,
2007-07-31 00:38:44 -07:00
{ NULL , 0 , 0 , 0 } ,
} ;
int c ;
2023-05-30 16:16:41 +02:00
c = getopt_long ( argc , argv , " D:s:d:b:i:o:lHOLC3ZFMNR248p:vS:I: " ,
2015-11-18 14:30:39 -08:00
lopts , NULL ) ;
2007-07-31 00:38:44 -07:00
if ( c = = - 1 )
break ;
switch ( c ) {
case ' D ' :
device = optarg ;
break ;
case ' s ' :
speed = atoi ( optarg ) ;
break ;
case ' d ' :
delay = atoi ( optarg ) ;
break ;
case ' b ' :
bits = atoi ( optarg ) ;
break ;
2015-11-18 14:30:39 -08:00
case ' i ' :
input_file = optarg ;
break ;
2015-11-18 14:30:40 -08:00
case ' o ' :
output_file = optarg ;
break ;
2007-07-31 00:38:44 -07:00
case ' l ' :
mode | = SPI_LOOP ;
break ;
case ' H ' :
mode | = SPI_CPHA ;
break ;
case ' O ' :
mode | = SPI_CPOL ;
break ;
case ' L ' :
mode | = SPI_LSB_FIRST ;
break ;
case ' C ' :
mode | = SPI_CS_HIGH ;
break ;
case ' 3 ' :
mode | = SPI_3WIRE ;
break ;
2023-05-30 16:16:41 +02:00
case ' Z ' :
mode | = SPI_3WIRE_HIZ ;
break ;
case ' F ' :
mode | = SPI_RX_CPHA_FLIP ;
break ;
case ' M ' :
mode | = SPI_MOSI_IDLE_LOW ;
break ;
2009-06-30 11:41:26 -07:00
case ' N ' :
mode | = SPI_NO_CS ;
break ;
2015-03-10 16:12:31 -04:00
case ' v ' :
verbose = 1 ;
break ;
2009-06-30 11:41:26 -07:00
case ' R ' :
mode | = SPI_READY ;
break ;
2015-03-10 16:12:32 -04:00
case ' p ' :
input_tx = optarg ;
break ;
2014-02-25 11:40:18 +01:00
case ' 2 ' :
mode | = SPI_TX_DUAL ;
break ;
case ' 4 ' :
mode | = SPI_TX_QUAD ;
break ;
2020-04-16 12:18:35 +02:00
case ' 8 ' :
mode | = SPI_TX_OCTAL ;
break ;
2017-03-21 16:48:46 +01:00
case ' S ' :
transfer_size = atoi ( optarg ) ;
break ;
case ' I ' :
iterations = atoi ( optarg ) ;
break ;
2007-07-31 00:38:44 -07:00
default :
print_usage ( argv [ 0 ] ) ;
}
}
2014-02-25 11:40:18 +01:00
if ( mode & SPI_LOOP ) {
if ( mode & SPI_TX_DUAL )
mode | = SPI_RX_DUAL ;
if ( mode & SPI_TX_QUAD )
mode | = SPI_RX_QUAD ;
2020-04-16 12:18:35 +02:00
if ( mode & SPI_TX_OCTAL )
mode | = SPI_RX_OCTAL ;
2014-02-25 11:40:18 +01:00
}
2007-07-31 00:38:44 -07:00
}
2015-11-18 14:30:38 -08:00
static void transfer_escaped_string ( int fd , char * str )
{
2016-09-09 09:02:51 +02:00
size_t size = strlen ( str ) ;
2015-11-18 14:30:38 -08:00
uint8_t * tx ;
uint8_t * rx ;
tx = malloc ( size ) ;
if ( ! tx )
pabort ( " can't allocate tx buffer " ) ;
rx = malloc ( size ) ;
if ( ! rx )
pabort ( " can't allocate rx buffer " ) ;
size = unescape ( ( char * ) tx , str , size ) ;
transfer ( fd , tx , rx , size ) ;
free ( rx ) ;
free ( tx ) ;
}
2015-11-18 14:30:39 -08:00
static void transfer_file ( int fd , char * filename )
{
ssize_t bytes ;
struct stat sb ;
int tx_fd ;
uint8_t * tx ;
uint8_t * rx ;
if ( stat ( filename , & sb ) = = - 1 )
pabort ( " can't stat input file " ) ;
tx_fd = open ( filename , O_RDONLY ) ;
2016-11-04 11:30:03 +01:00
if ( tx_fd < 0 )
2015-11-18 14:30:39 -08:00
pabort ( " can't open input file " ) ;
tx = malloc ( sb . st_size ) ;
if ( ! tx )
pabort ( " can't allocate tx buffer " ) ;
rx = malloc ( sb . st_size ) ;
if ( ! rx )
pabort ( " can't allocate rx buffer " ) ;
bytes = read ( tx_fd , tx , sb . st_size ) ;
if ( bytes ! = sb . st_size )
pabort ( " failed to read input file " ) ;
transfer ( fd , tx , rx , sb . st_size ) ;
free ( rx ) ;
free ( tx ) ;
close ( tx_fd ) ;
}
2017-03-21 16:48:46 +01:00
static uint64_t _read_count ;
static uint64_t _write_count ;
static void show_transfer_rate ( void )
{
static uint64_t prev_read_count , prev_write_count ;
double rx_rate , tx_rate ;
rx_rate = ( ( _read_count - prev_read_count ) * 8 ) / ( interval * 1000.0 ) ;
tx_rate = ( ( _write_count - prev_write_count ) * 8 ) / ( interval * 1000.0 ) ;
printf ( " rate: tx %.1fkbps, rx %.1fkbps \n " , rx_rate , tx_rate ) ;
prev_read_count = _read_count ;
prev_write_count = _write_count ;
}
static void transfer_buf ( int fd , int len )
{
uint8_t * tx ;
uint8_t * rx ;
int i ;
tx = malloc ( len ) ;
if ( ! tx )
pabort ( " can't allocate tx buffer " ) ;
for ( i = 0 ; i < len ; i + + )
tx [ i ] = random ( ) ;
rx = malloc ( len ) ;
if ( ! rx )
pabort ( " can't allocate rx buffer " ) ;
transfer ( fd , tx , rx , len ) ;
_write_count + = len ;
_read_count + = len ;
if ( mode & SPI_LOOP ) {
if ( memcmp ( tx , rx , len ) ) {
fprintf ( stderr , " transfer error ! \n " ) ;
hex_dump ( tx , len , 32 , " TX " ) ;
hex_dump ( rx , len , 32 , " RX " ) ;
exit ( 1 ) ;
}
}
free ( rx ) ;
free ( tx ) ;
}
2007-07-31 00:38:44 -07:00
int main ( int argc , char * argv [ ] )
{
int ret = 0 ;
int fd ;
2022-06-13 00:39:41 -05:00
uint32_t request ;
2007-07-31 00:38:44 -07:00
parse_opts ( argc , argv ) ;
2020-02-13 12:16:06 +08:00
if ( input_tx & & input_file )
pabort ( " only one of -p and --input may be selected " ) ;
2007-07-31 00:38:44 -07:00
fd = open ( device , O_RDWR ) ;
if ( fd < 0 )
pabort ( " can't open device " ) ;
/*
* spi mode
*/
2022-06-13 00:39:41 -05:00
/* WR is make a request to assign 'mode' */
request = mode ;
2014-02-25 11:40:18 +01:00
ret = ioctl ( fd , SPI_IOC_WR_MODE32 , & mode ) ;
2007-07-31 00:38:44 -07:00
if ( ret = = - 1 )
pabort ( " can't set spi mode " ) ;
2022-06-13 00:39:41 -05:00
/* RD is read what mode the device actually is in */
2014-02-25 11:40:18 +01:00
ret = ioctl ( fd , SPI_IOC_RD_MODE32 , & mode ) ;
2007-07-31 00:38:44 -07:00
if ( ret = = - 1 )
pabort ( " can't get spi mode " ) ;
2022-06-13 00:39:41 -05:00
/* Drivers can reject some mode bits without returning an error.
* Read the current value to identify what mode it is in , and if it
* differs from the requested mode , warn the user .
*/
if ( request ! = mode )
printf ( " WARNING device does not support requested mode 0x%x \n " ,
request ) ;
2007-07-31 00:38:44 -07:00
/*
* bits per word
*/
ret = ioctl ( fd , SPI_IOC_WR_BITS_PER_WORD , & bits ) ;
if ( ret = = - 1 )
pabort ( " can't set bits per word " ) ;
ret = ioctl ( fd , SPI_IOC_RD_BITS_PER_WORD , & bits ) ;
if ( ret = = - 1 )
pabort ( " can't get bits per word " ) ;
/*
* max speed hz
*/
ret = ioctl ( fd , SPI_IOC_WR_MAX_SPEED_HZ , & speed ) ;
if ( ret = = - 1 )
pabort ( " can't set max speed hz " ) ;
ret = ioctl ( fd , SPI_IOC_RD_MAX_SPEED_HZ , & speed ) ;
if ( ret = = - 1 )
pabort ( " can't get max speed hz " ) ;
2014-02-25 11:40:18 +01:00
printf ( " spi mode: 0x%x \n " , mode ) ;
2020-06-08 12:00:49 +02:00
printf ( " bits per word: %u \n " , bits ) ;
printf ( " max speed: %u Hz (%u kHz) \n " , speed , speed / 1000 ) ;
2007-07-31 00:38:44 -07:00
2015-11-18 14:30:38 -08:00
if ( input_tx )
transfer_escaped_string ( fd , input_tx ) ;
2015-11-18 14:30:39 -08:00
else if ( input_file )
transfer_file ( fd , input_file ) ;
2017-03-21 16:48:46 +01:00
else if ( transfer_size ) {
struct timespec last_stat ;
clock_gettime ( CLOCK_MONOTONIC , & last_stat ) ;
while ( iterations - - > 0 ) {
struct timespec current ;
transfer_buf ( fd , transfer_size ) ;
clock_gettime ( CLOCK_MONOTONIC , & current ) ;
if ( current . tv_sec - last_stat . tv_sec > interval ) {
show_transfer_rate ( ) ;
last_stat = current ;
}
}
printf ( " total: tx %.1fKB, rx %.1fKB \n " ,
_write_count / 1024.0 , _read_count / 1024.0 ) ;
} else
2015-03-10 16:12:32 -04:00
transfer ( fd , default_tx , default_rx , sizeof ( default_tx ) ) ;
2007-07-31 00:38:44 -07:00
close ( fd ) ;
return ret ;
}