-
-
Save fenrir-naru/3c4a312663cb9122c7ca to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/* | |
* SPI testing utility (using spidev driver) | |
* | |
* Copyright (c) 2007 MontaVista Software, Inc. | |
* Copyright (c) 2007 Anton Vorontsov <[email protected]> | |
* | |
* 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. | |
* | |
* Cross-compile with cross-gcc -I/path/to/cross-kernel/include | |
*/ | |
#include <stdint.h> | |
#include <unistd.h> | |
#include <stdio.h> | |
#include <stdlib.h> | |
#include <getopt.h> | |
#include <fcntl.h> | |
#include <sys/ioctl.h> | |
#include <linux/types.h> | |
#include <linux/spi/spidev.h> | |
#define ARRAY_SIZE(a) (sizeof(a) / sizeof((a)[0])) | |
static void pabort(const char *s) | |
{ | |
perror(s); | |
abort(); | |
} | |
static const char *device = "/dev/spidev0.0"; | |
static uint8_t mode; | |
static uint8_t bits = 8; | |
static uint32_t speed = 500000; | |
static uint16_t delay; | |
static uint8_t uart_no = 0; | |
static void transfer(int fd) | |
{ | |
int ret; | |
uint8_t tx[] = /*{ | |
0x48, 0x45, 0x4C, 0x4C, 0x4F, | |
0x20, | |
0x57, 0x4F, 0x52, 0x4C, 0x44, | |
0x0A | |
}*/ | |
{ | |
// UART0 0x19 => 0x00 // GPIO2 as LOW (default) | |
(0x80 | (uart_no << 5) | 0x18), 0x44, // UART0 0x18 => 0x44 // GPIO2 as open-drain and output | |
}; | |
uint8_t rx[ARRAY_SIZE(tx)] = {0, }; | |
struct spi_ioc_transfer tr = { | |
.tx_buf = (unsigned long)tx, | |
.rx_buf = (unsigned long)rx, | |
.len = ARRAY_SIZE(tx), | |
.delay_usecs = delay, | |
.speed_hz = speed, | |
.bits_per_word = bits, | |
}; | |
ret = ioctl(fd, SPI_IOC_MESSAGE(1), &tr); | |
if (ret < 1) | |
pabort("can't send spi message"); | |
/* | |
for (ret = 0; ret < ARRAY_SIZE(tx); ret++) { | |
if (!(ret % 6)) | |
puts(""); | |
printf("%.2X ", rx[ret]); | |
} | |
puts(""); | |
*/ | |
} | |
static void print_usage(const char *prog) | |
{ | |
printf("Usage: %s [-uDsbdlHOLC3]\n", prog); | |
puts(" -u --uart uart number\n" | |
" -D --device device to use (default /dev/spidev0.0)\n" | |
" -s --speed max speed (Hz)\n" | |
" -d --delay delay (usec)\n" | |
" -b --bpw bits per word \n" | |
" -l --loop loopback\n" | |
" -H --cpha clock phase\n" | |
" -O --cpol clock polarity\n" | |
" -L --lsb least significant bit first\n" | |
" -C --cs-high chip select active high\n" | |
" -3 --3wire SI/SO signals shared\n"); | |
exit(1); | |
} | |
static void parse_opts(int argc, char *argv[]) | |
{ | |
while (1) { | |
static const struct option lopts[] = { | |
{ "device", 1, 0, 'D' }, | |
{ "speed", 1, 0, 's' }, | |
{ "delay", 1, 0, 'd' }, | |
{ "bpw", 1, 0, 'b' }, | |
{ "loop", 0, 0, 'l' }, | |
{ "cpha", 0, 0, 'H' }, | |
{ "cpol", 0, 0, 'O' }, | |
{ "lsb", 0, 0, 'L' }, | |
{ "cs-high", 0, 0, 'C' }, | |
{ "3wire", 0, 0, '3' }, | |
{ "no-cs", 0, 0, 'N' }, | |
{ "ready", 0, 0, 'R' }, | |
{ "uart", 1, 0, 'u' }, | |
{ NULL, 0, 0, 0 }, | |
}; | |
int c; | |
c = getopt_long(argc, argv, "u:D:s:d:b:lHOLC3NR", lopts, NULL); | |
if (c == -1) | |
break; | |
switch (c) { | |
case 'u': | |
uart_no = atoi(optarg) & 0x03; | |
break; | |
case 'D': | |
device = optarg; | |
break; | |
case 's': | |
speed = atoi(optarg); | |
break; | |
case 'd': | |
delay = atoi(optarg); | |
break; | |
case 'b': | |
bits = atoi(optarg); | |
break; | |
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; | |
case 'N': | |
mode |= SPI_NO_CS; | |
break; | |
case 'R': | |
mode |= SPI_READY; | |
break; | |
default: | |
print_usage(argv[0]); | |
break; | |
} | |
} | |
} | |
int main(int argc, char *argv[]) | |
{ | |
int ret = 0; | |
int fd; | |
parse_opts(argc, argv); | |
fd = open(device, O_RDWR); | |
if (fd < 0) | |
pabort("can't open device"); | |
/* | |
* spi mode | |
*/ | |
ret = ioctl(fd, SPI_IOC_WR_MODE, &mode); | |
if (ret == -1) | |
pabort("can't set spi mode"); | |
ret = ioctl(fd, SPI_IOC_RD_MODE, &mode); | |
if (ret == -1) | |
pabort("can't get spi mode"); | |
/* | |
* 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"); | |
printf("spi mode: %d\n", mode); | |
printf("bits per word: %d\n", bits); | |
printf("max speed: %d Hz (%d KHz)\n", speed, speed/1000); | |
transfer(fd); | |
close(fd); | |
return ret; | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment