2014-08-07 16:51:52 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License v2.1. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup tests
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
2014-08-14 18:32:33 +02:00
|
|
|
* @brief Application for testing low-level SPI driver implementations
|
|
|
|
*
|
|
|
|
* This implementation covers both, master and slave configurations.
|
2014-08-07 16:51:52 +02:00
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
2014-08-14 18:32:33 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
2014-08-07 16:51:52 +02:00
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
#include "xtimer.h"
|
2014-08-07 16:51:52 +02:00
|
|
|
#include "shell.h"
|
|
|
|
#include "periph/spi.h"
|
2020-05-15 11:23:30 +02:00
|
|
|
#include "schedstatistics.h"
|
|
|
|
#include "thread.h"
|
2014-08-14 18:32:33 +02:00
|
|
|
|
2020-05-19 17:07:11 +02:00
|
|
|
/**
|
|
|
|
* @brief Default port number used for the CS pin if unassigned
|
|
|
|
*/
|
|
|
|
#ifndef DEFAULT_SPI_CS_PORT
|
|
|
|
#define DEFAULT_SPI_CS_PORT 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Default port number used for the CS pin if unassigned
|
|
|
|
*/
|
|
|
|
#ifndef DEFAULT_SPI_CS_PIN
|
|
|
|
#define DEFAULT_SPI_CS_PIN 0
|
|
|
|
#endif
|
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
/**
|
|
|
|
* @brief Some parameters used for benchmarking
|
|
|
|
*/
|
|
|
|
#define BENCH_REDOS (1000)
|
|
|
|
#define BENCH_SMALL (2)
|
|
|
|
#define BENCH_LARGE (100)
|
|
|
|
#define BENCH_PAYLOAD ('b')
|
|
|
|
#define BENCH_REGADDR (0x23)
|
|
|
|
|
|
|
|
#define BUF_SIZE (512U)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Benchmark buffers
|
|
|
|
*/
|
|
|
|
static uint8_t bench_wbuf[BENCH_LARGE];
|
|
|
|
static uint8_t bench_rbuf[BENCH_LARGE];
|
|
|
|
|
2020-05-15 11:23:30 +02:00
|
|
|
extern void sched_statistics_cb(kernel_pid_t active_thread, kernel_pid_t next_thread);
|
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
/**
|
|
|
|
* @brief Generic buffer used for receiving
|
|
|
|
*/
|
|
|
|
static uint8_t buf[BUF_SIZE];
|
|
|
|
|
|
|
|
static struct {
|
|
|
|
spi_t dev;
|
|
|
|
spi_mode_t mode;
|
|
|
|
spi_clk_t clk;
|
|
|
|
spi_cs_t cs;
|
|
|
|
} spiconf;
|
|
|
|
|
2020-05-15 11:23:30 +02:00
|
|
|
/*
|
|
|
|
* @brief Trigger an update of the scheduler runtime statistics.
|
|
|
|
*
|
|
|
|
* Increases the number of context switches by one as a side effect
|
|
|
|
*/
|
|
|
|
static void _sched_statistics_trigger(void)
|
|
|
|
{
|
2020-08-23 21:28:33 +02:00
|
|
|
sched_statistics_cb(thread_getpid(), thread_getpid());
|
2020-05-15 11:23:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static xtimer_ticks32_t _sched_ticks(void)
|
|
|
|
{
|
|
|
|
_sched_statistics_trigger();
|
|
|
|
xtimer_ticks32_t runtime_ticks = {
|
2020-08-23 21:28:33 +02:00
|
|
|
.ticks32 = sched_pidlist[thread_getpid()].runtime_ticks
|
2020-05-15 11:23:30 +02:00
|
|
|
};
|
|
|
|
return runtime_ticks;
|
|
|
|
}
|
|
|
|
|
2020-05-20 17:48:15 +02:00
|
|
|
static uint32_t _xtimer_diff_usec(xtimer_ticks32_t stop, xtimer_ticks32_t start)
|
|
|
|
{
|
|
|
|
return xtimer_usec_from_ticks(xtimer_diff(stop, start));
|
|
|
|
}
|
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
void print_bytes(char* title, uint8_t* data, size_t len)
|
2014-08-14 18:32:33 +02:00
|
|
|
{
|
2016-11-04 18:30:33 +01:00
|
|
|
printf("%4s\n", title);
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
|
|
printf(" %2i ", (int)i);
|
2014-08-14 18:32:33 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
printf("\n ");
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
|
|
printf(" 0x%02x", (int)data[i]);
|
2014-08-14 18:32:33 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
printf("\n ");
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
|
|
if (data[i] < ' ' || data[i] > '~') {
|
2014-08-14 18:32:33 +02:00
|
|
|
printf(" ?? ");
|
|
|
|
}
|
|
|
|
else {
|
2016-11-04 18:30:33 +01:00
|
|
|
printf(" %c ", (char)data[i]);
|
2014-08-14 18:32:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("\n\n");
|
|
|
|
}
|
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
int cmd_init(int argc, char **argv)
|
2014-08-07 16:51:52 +02:00
|
|
|
{
|
2016-11-04 18:30:33 +01:00
|
|
|
int dev, mode, clk, port, pin, tmp;
|
2014-08-15 13:57:37 +02:00
|
|
|
|
2020-05-19 17:07:11 +02:00
|
|
|
if (argc < 4) {
|
|
|
|
printf("usage: %s <dev> <mode> <clk> [cs port] [cs pin]\n", argv[0]);
|
2016-11-04 18:30:33 +01:00
|
|
|
puts("\tdev:");
|
|
|
|
for (int i = 0; i < (int)SPI_NUMOF; i++) {
|
|
|
|
printf("\t\t%i: SPI_DEV(%i)\n", i, i);
|
|
|
|
}
|
|
|
|
puts("\tmode:");
|
|
|
|
puts("\t\t0: POL:0, PHASE:0 - on first rising edge");
|
|
|
|
puts("\t\t1: POL:0, PHASE:1 - on second rising edge");
|
|
|
|
puts("\t\t2: POL:1, PHASE:0 - on first falling edge");
|
|
|
|
puts("\t\t3: POL:1, PHASE:1 - on second falling edge");
|
|
|
|
puts("\tclk:");
|
|
|
|
puts("\t\t0: 100 KHz");
|
|
|
|
puts("\t\t1: 400 KHz");
|
|
|
|
puts("\t\t2: 1 MHz");
|
|
|
|
puts("\t\t3: 5 MHz");
|
|
|
|
puts("\t\t4: 10 MHz");
|
|
|
|
puts("\tcs port:");
|
|
|
|
puts("\t\tPort of the CS pin, set to -1 for hardware chip select");
|
|
|
|
puts("\tcs pin:");
|
|
|
|
puts("\t\tPin used for chip select. If hardware chip select is enabled,\n"
|
|
|
|
"\t\tthis value specifies the internal HWCS line");
|
|
|
|
return 1;
|
|
|
|
}
|
2014-08-07 16:51:52 +02:00
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
/* parse the given SPI device */
|
|
|
|
dev = atoi(argv[1]);
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
if (dev < 0 || dev >= (int)SPI_NUMOF) {
|
2016-11-04 18:30:33 +01:00
|
|
|
puts("error: invalid SPI device specified");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
spiconf.dev = SPI_DEV(dev);
|
2014-08-07 16:51:52 +02:00
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
/* parse the SPI mode */
|
|
|
|
mode = atoi(argv[2]);
|
|
|
|
switch (mode) {
|
|
|
|
case 0: spiconf.mode = SPI_MODE_0; break;
|
|
|
|
case 1: spiconf.mode = SPI_MODE_1; break;
|
|
|
|
case 2: spiconf.mode = SPI_MODE_2; break;
|
|
|
|
case 3: spiconf.mode = SPI_MODE_3; break;
|
|
|
|
default:
|
|
|
|
puts("error: invalid SPI mode specified");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* parse the targeted clock speed */
|
|
|
|
clk = atoi(argv[3]);
|
|
|
|
switch (clk) {
|
|
|
|
case 0: spiconf.clk = SPI_CLK_100KHZ; break;
|
|
|
|
case 1: spiconf.clk = SPI_CLK_400KHZ; break;
|
|
|
|
case 2: spiconf.clk = SPI_CLK_1MHZ; break;
|
|
|
|
case 3: spiconf.clk = SPI_CLK_5MHZ; break;
|
|
|
|
case 4: spiconf.clk = SPI_CLK_10MHZ; break;
|
|
|
|
default:
|
|
|
|
puts("error: invalid bus speed specified");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* parse chip select port and pin */
|
2020-05-19 17:07:11 +02:00
|
|
|
if (argc > 5) {
|
|
|
|
pin = atoi(argv[5]);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
pin = DEFAULT_SPI_CS_PIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (argc > 4) {
|
|
|
|
port = atoi(argv[4]);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
port = DEFAULT_SPI_CS_PORT;
|
|
|
|
}
|
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
if (pin < 0 || port < -1) {
|
|
|
|
puts("error: invalid CS port/pin combination specified");
|
|
|
|
}
|
|
|
|
if (port == -1) { /* hardware chip select line */
|
|
|
|
spiconf.cs = SPI_HWCS(pin);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
spiconf.cs = (spi_cs_t)GPIO_PIN(port, pin);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* test setup */
|
|
|
|
tmp = spi_init_cs(spiconf.dev, spiconf.cs);
|
|
|
|
if (tmp != SPI_OK) {
|
|
|
|
puts("error: unable to initialize the given chip select line");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
tmp = spi_acquire(spiconf.dev, spiconf.cs, spiconf.mode, spiconf.clk);
|
|
|
|
if (tmp == SPI_NOMODE) {
|
|
|
|
puts("error: given SPI mode is not supported");
|
2015-03-20 08:51:45 +01:00
|
|
|
return 1;
|
2014-08-14 18:32:33 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
else if (tmp == SPI_NOCLK) {
|
|
|
|
puts("error: targeted clock speed is not supported");
|
2015-03-20 08:51:45 +01:00
|
|
|
return 1;
|
2014-08-14 18:32:33 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
else if (tmp != SPI_OK) {
|
|
|
|
puts("error: unable to acquire bus with given parameters");
|
2015-03-20 08:51:45 +01:00
|
|
|
return 1;
|
2014-08-14 18:32:33 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
spi_release(spiconf.dev);
|
|
|
|
|
|
|
|
printf("SPI_DEV(%i) initialized: mode: %i, clk: %i, cs_port: %i, cs_pin: %i\n",
|
|
|
|
dev, mode, clk, port, pin);
|
|
|
|
|
2015-03-20 08:51:45 +01:00
|
|
|
return 0;
|
2014-08-07 16:51:52 +02:00
|
|
|
}
|
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
int cmd_transfer(int argc, char **argv)
|
2014-08-07 16:51:52 +02:00
|
|
|
{
|
2016-11-04 18:30:33 +01:00
|
|
|
size_t len;
|
|
|
|
|
|
|
|
if (argc < 2) {
|
|
|
|
printf("usage: %s <data>\n", argv[0]);
|
2015-03-20 08:51:45 +01:00
|
|
|
return 1;
|
2014-08-14 18:32:33 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
if (spiconf.dev == SPI_UNDEF) {
|
|
|
|
puts("error: SPI is not initialized, please initialize bus first");
|
2015-03-20 08:51:45 +01:00
|
|
|
return 1;
|
2014-08-14 18:32:33 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* get bus access */
|
|
|
|
if (spi_acquire(spiconf.dev, spiconf.cs,
|
|
|
|
spiconf.mode, spiconf.clk) != SPI_OK) {
|
|
|
|
puts("error: unable to acquire the SPI bus");
|
2015-03-20 08:51:45 +01:00
|
|
|
return 1;
|
2014-08-14 18:32:33 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* transfer data */
|
|
|
|
len = strlen(argv[1]);
|
|
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
spi_transfer_bytes(spiconf.dev, spiconf.cs, false, argv[1], buf, len);
|
|
|
|
|
|
|
|
/* release the bus */
|
|
|
|
spi_release(spiconf.dev);
|
|
|
|
|
|
|
|
/* print results */
|
|
|
|
print_bytes("Sent bytes", (uint8_t *)argv[1], len);
|
|
|
|
print_bytes("Received bytes", buf, len);
|
|
|
|
|
2015-03-20 08:51:45 +01:00
|
|
|
return 0;
|
2014-08-07 16:51:52 +02:00
|
|
|
}
|
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
int cmd_bench(int argc, char **argv)
|
2014-08-07 16:51:52 +02:00
|
|
|
{
|
2017-06-03 21:52:47 +02:00
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
uint32_t start, stop;
|
2020-05-15 11:23:30 +02:00
|
|
|
xtimer_ticks32_t sched_start, sched_stop;
|
|
|
|
uint32_t sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
uint32_t sum = 0;
|
2020-05-15 11:23:30 +02:00
|
|
|
uint32_t sched_sum = 0;
|
2016-11-04 18:30:33 +01:00
|
|
|
uint8_t in;
|
|
|
|
uint8_t out = (uint8_t)BENCH_PAYLOAD;
|
2014-08-07 16:51:52 +02:00
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
if (spiconf.dev == SPI_UNDEF) {
|
|
|
|
puts("error: SPI is not initialized, please initialize bus first");
|
2015-03-20 08:51:45 +01:00
|
|
|
return 1;
|
2014-08-14 18:32:33 +02:00
|
|
|
}
|
2014-08-07 16:51:52 +02:00
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
/* prepare buffer */
|
|
|
|
memset(bench_wbuf, BENCH_PAYLOAD, BENCH_LARGE);
|
|
|
|
|
|
|
|
/* get access to the bus */
|
|
|
|
if (spi_acquire(spiconf.dev, spiconf.cs,
|
|
|
|
spiconf.mode, spiconf.clk) != SPI_OK) {
|
|
|
|
puts("error: unable to acquire the SPI bus");
|
|
|
|
return 1;
|
2014-08-07 16:51:52 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
|
2020-05-15 11:23:30 +02:00
|
|
|
puts("### Running some benchmarks, all values in [us] ###");
|
|
|
|
puts("### Test\t\t\t\tTransfer time\tuser time\n");
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 1 - write 1000 times 1 byte */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
in = spi_transfer_byte(spiconf.dev, spiconf.cs, false, out);
|
|
|
|
(void)in;
|
2014-08-07 16:51:52 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf(" 1 - write %i times %i byte:", BENCH_REDOS, 1);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t\t\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2014-08-07 16:51:52 +02:00
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
/* 2 - write 1000 times 2 byte */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_bytes(spiconf.dev, spiconf.cs, false,
|
|
|
|
bench_wbuf, NULL, BENCH_SMALL);
|
|
|
|
}
|
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf(" 2 - write %i times %i byte:", BENCH_REDOS, BENCH_SMALL);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t\t\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2014-08-14 18:32:33 +02:00
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
/* 3 - write 1000 times 100 byte */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_bytes(spiconf.dev, spiconf.cs, false,
|
|
|
|
bench_wbuf, NULL, BENCH_LARGE);
|
2014-08-07 16:51:52 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf(" 3 - write %i times %i byte:", BENCH_REDOS, BENCH_LARGE);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 4 - write 1000 times 1 byte to register */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
in = spi_transfer_reg(spiconf.dev, spiconf.cs, BENCH_REGADDR, out);
|
|
|
|
(void)in;
|
2014-08-07 16:51:52 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf(" 4 - write %i times %i byte to register:", BENCH_REDOS, 1);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2014-08-14 18:32:33 +02:00
|
|
|
|
2016-11-04 18:30:33 +01:00
|
|
|
/* 5 - write 1000 times 2 byte to register */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_regs(spiconf.dev, spiconf.cs, BENCH_REGADDR,
|
|
|
|
bench_wbuf, NULL, BENCH_SMALL);
|
2014-10-17 10:09:25 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf(" 5 - write %i times %i byte to register:", BENCH_REDOS, BENCH_SMALL);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 6 - write 1000 times 100 byte to register */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_regs(spiconf.dev, spiconf.cs, BENCH_REGADDR,
|
|
|
|
bench_wbuf, NULL, BENCH_LARGE);
|
2014-10-17 10:09:25 +02:00
|
|
|
}
|
2016-11-04 18:30:33 +01:00
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf(" 6 - write %i times %i byte to register:", BENCH_REDOS, BENCH_LARGE);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 7 - read 1000 times 2 byte */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_bytes(spiconf.dev, spiconf.cs, false,
|
|
|
|
NULL, bench_rbuf, BENCH_SMALL);
|
|
|
|
}
|
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf(" 7 - read %i times %i byte:", BENCH_REDOS, BENCH_SMALL);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t\t\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 8 - read 1000 times 100 byte */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_bytes(spiconf.dev, spiconf.cs, false,
|
|
|
|
NULL, bench_rbuf, BENCH_LARGE);
|
|
|
|
}
|
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf(" 8 - read %i times %i byte:", BENCH_REDOS, BENCH_LARGE);
|
2020-05-19 11:47:56 +02:00
|
|
|
printf("\t\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 9 - read 1000 times 2 byte from register */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_regs(spiconf.dev, spiconf.cs, BENCH_REGADDR,
|
|
|
|
NULL, bench_rbuf, BENCH_SMALL);
|
|
|
|
}
|
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf(" 9 - read %i times %i byte from register:", BENCH_REDOS, BENCH_SMALL);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 10 - read 1000 times 100 byte from register */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_regs(spiconf.dev, spiconf.cs, BENCH_REGADDR,
|
|
|
|
NULL, bench_rbuf, BENCH_LARGE);
|
|
|
|
}
|
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf("10 - read %i times %i byte from register:", BENCH_REDOS, BENCH_LARGE);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 11 - transfer 1000 times 2 byte */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_bytes(spiconf.dev, spiconf.cs, false,
|
|
|
|
bench_wbuf, bench_rbuf, BENCH_SMALL);
|
|
|
|
}
|
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf("11 - transfer %i times %i byte:", BENCH_REDOS, BENCH_SMALL);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 12 - transfer 1000 times 100 byte */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_bytes(spiconf.dev, spiconf.cs, false,
|
|
|
|
bench_wbuf, bench_rbuf, BENCH_LARGE);
|
|
|
|
}
|
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf("12 - transfer %i times %i byte:", BENCH_REDOS, BENCH_LARGE);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 13 - transfer 1000 times 2 byte from/to register */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_regs(spiconf.dev, spiconf.cs, BENCH_REGADDR,
|
|
|
|
bench_wbuf, bench_rbuf, BENCH_SMALL);
|
|
|
|
}
|
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf("13 - transfer %i times %i byte to register:", BENCH_REDOS, BENCH_SMALL);
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("\t%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
/* 14 - transfer 1000 times 100 byte from/to register */
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_start = _sched_ticks();
|
2016-11-04 18:30:33 +01:00
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_transfer_regs(spiconf.dev, spiconf.cs, BENCH_REGADDR,
|
|
|
|
bench_wbuf, bench_rbuf, BENCH_LARGE);
|
|
|
|
}
|
|
|
|
stop = xtimer_now_usec();
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_stop = _sched_ticks();
|
2020-05-20 17:48:15 +02:00
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
2016-11-04 18:30:33 +01:00
|
|
|
printf("14 - transfer %i times %i byte to register:", BENCH_REDOS, BENCH_LARGE);
|
2020-05-19 11:47:56 +02:00
|
|
|
printf("%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
2016-11-04 18:30:33 +01:00
|
|
|
sum += (stop - start);
|
2020-05-15 11:23:30 +02:00
|
|
|
sched_sum += sched_diff_us;
|
|
|
|
|
2020-05-20 17:54:40 +02:00
|
|
|
/* 15 - release & acquire the bus 1000 times */
|
|
|
|
sched_start = _sched_ticks();
|
|
|
|
start = xtimer_now_usec();
|
|
|
|
for (int i = 0; i < BENCH_REDOS; i++) {
|
|
|
|
spi_release(spiconf.dev);
|
|
|
|
if (spi_acquire(spiconf.dev, spiconf.cs, spiconf.mode, spiconf.clk) != SPI_OK) {
|
|
|
|
puts("ERROR - spi_acquire() failed.");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
stop = xtimer_now_usec();
|
|
|
|
sched_stop = _sched_ticks();
|
|
|
|
sched_diff_us = _xtimer_diff_usec(sched_stop, sched_start);
|
|
|
|
printf("15 - acquire/release %i times:\t\t", BENCH_REDOS);
|
|
|
|
printf("%"PRIu32"\t%"PRIu32"\n", (stop - start), sched_diff_us);
|
|
|
|
sum += (stop - start);
|
|
|
|
sched_sum += sched_diff_us;
|
|
|
|
|
2020-05-15 11:23:30 +02:00
|
|
|
xtimer_sleep(1);
|
2016-11-04 18:30:33 +01:00
|
|
|
|
2020-05-19 11:06:06 +02:00
|
|
|
printf("-- - SUM:\t\t\t\t\t%"PRIu32"\t%"PRIu32"\n", sum, sched_sum);
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
spi_release(spiconf.dev);
|
|
|
|
puts("\n### All runs complete ###");
|
|
|
|
|
2015-03-20 08:51:45 +01:00
|
|
|
return 0;
|
2014-10-17 10:09:25 +02:00
|
|
|
}
|
|
|
|
|
2020-06-11 16:21:41 +02:00
|
|
|
#ifdef MODULE_PERIPH_SPI_RECONFIGURE
|
|
|
|
int cmd_spi_gpio(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int dev = -1;
|
|
|
|
|
|
|
|
/* parse the given SPI device */
|
|
|
|
if (argc > 1) {
|
|
|
|
dev = atoi(argv[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dev < 0 || dev >= (int)SPI_NUMOF) {
|
|
|
|
puts("error: invalid SPI device specified");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
gpio_t miso_pin = spi_pin_miso(dev);
|
|
|
|
gpio_t mosi_pin = spi_pin_mosi(dev);
|
|
|
|
|
|
|
|
printf("Command: spi_deinit_pins(%i)\n", dev);
|
|
|
|
spi_deinit_pins(dev);
|
|
|
|
|
|
|
|
gpio_init(miso_pin, GPIO_OUT);
|
|
|
|
gpio_init(mosi_pin, GPIO_OUT);
|
|
|
|
|
|
|
|
xtimer_sleep(1);
|
|
|
|
|
|
|
|
printf("Command: gpio_set()\n");
|
|
|
|
gpio_set(miso_pin);
|
|
|
|
gpio_set(mosi_pin);
|
|
|
|
|
|
|
|
xtimer_sleep(1);
|
|
|
|
|
|
|
|
printf("Command: gpio_clear()\n");
|
|
|
|
gpio_clear(miso_pin);
|
|
|
|
gpio_clear(mosi_pin);
|
|
|
|
|
|
|
|
xtimer_sleep(1);
|
|
|
|
|
|
|
|
printf("Command: spi_init_pins(%i)\n", dev);
|
|
|
|
spi_init_pins(dev);
|
|
|
|
|
|
|
|
printf("Success: spi_%i re-init\n", dev);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-08-14 18:32:33 +02:00
|
|
|
static const shell_command_t shell_commands[] = {
|
2016-11-04 18:30:33 +01:00
|
|
|
{ "init", "Setup a particular SPI configuration", cmd_init },
|
|
|
|
{ "send", "Transfer string to slave", cmd_transfer },
|
|
|
|
{ "bench", "Runs some benchmarks", cmd_bench },
|
2020-06-11 16:21:41 +02:00
|
|
|
#ifdef MODULE_PERIPH_SPI_RECONFIGURE
|
|
|
|
{ "spi_gpio", "Re-configures MISO & MOSI pins to GPIO mode and back.", cmd_spi_gpio },
|
|
|
|
#endif
|
2014-08-14 18:32:33 +02:00
|
|
|
{ NULL, NULL, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
2021-02-25 12:38:00 +01:00
|
|
|
puts("Manual SPI peripheral driver test (see README.md)");
|
2016-11-04 18:30:33 +01:00
|
|
|
|
|
|
|
printf("There are %i SPI devices configured for your platform.\n",
|
|
|
|
(int)SPI_NUMOF);
|
|
|
|
|
|
|
|
/* reset local SPI configuration */
|
|
|
|
spiconf.dev = SPI_UNDEF;
|
2014-08-07 16:51:52 +02:00
|
|
|
|
|
|
|
/* run the shell */
|
2015-07-14 18:42:46 +02:00
|
|
|
char line_buf[SHELL_DEFAULT_BUFSIZE];
|
|
|
|
shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
|
2014-08-07 16:51:52 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|