2017-05-10 23:06:43 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 Hamburg University of Applied Sciences
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2018-01-22 09:04:10 +01:00
|
|
|
* @ingroup drivers_soft_spi
|
2017-05-10 23:06:43 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Software SPI implementation
|
|
|
|
*
|
|
|
|
* @author Markus Blechschmidt <Markus.Blechschmidt@haw-hamburg.de>
|
|
|
|
* @author Peter Kietzmann <peter.kietzmann@haw-hamburg.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include "mutex.h"
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
#include "xtimer.h"
|
|
|
|
|
|
|
|
#include "soft_spi.h"
|
|
|
|
#include "soft_spi_params.h"
|
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2017-05-10 23:06:43 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
2021-03-08 07:23:29 +01:00
|
|
|
#define READ_PADDING_BYTE (0x00)
|
2017-05-10 23:06:43 +02:00
|
|
|
/**
|
|
|
|
* @brief Allocate one lock per SPI device
|
|
|
|
*/
|
|
|
|
static mutex_t locks[sizeof soft_spi_config];
|
|
|
|
|
|
|
|
static inline bool soft_spi_bus_is_valid(soft_spi_t bus)
|
|
|
|
{
|
|
|
|
unsigned int soft_spi_num = (unsigned int) bus;
|
|
|
|
|
2019-07-18 15:17:31 +02:00
|
|
|
if (ARRAY_SIZE(soft_spi_config) < soft_spi_num) {
|
2017-05-10 23:06:43 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void soft_spi_init(soft_spi_t bus)
|
|
|
|
{
|
|
|
|
DEBUG("Soft SPI init\n");
|
|
|
|
|
|
|
|
assert(soft_spi_bus_is_valid(bus));
|
|
|
|
|
|
|
|
/* initialize device lock */
|
|
|
|
mutex_init(&locks[bus]);
|
|
|
|
soft_spi_init_pins(bus);
|
|
|
|
}
|
|
|
|
|
|
|
|
void soft_spi_init_pins(soft_spi_t bus)
|
|
|
|
{
|
|
|
|
DEBUG("Soft SPI soft_spi_init_pins\n");
|
|
|
|
|
|
|
|
assert(soft_spi_bus_is_valid(bus));
|
|
|
|
|
|
|
|
/* check that miso is not mosi is not clk*/
|
2020-01-17 12:45:13 +01:00
|
|
|
assert(!gpio_is_equal(soft_spi_config[bus].mosi_pin, soft_spi_config[bus].miso_pin));
|
|
|
|
assert(!gpio_is_equal(soft_spi_config[bus].mosi_pin, soft_spi_config[bus].clk_pin));
|
|
|
|
assert(!gpio_is_equal(soft_spi_config[bus].miso_pin, soft_spi_config[bus].clk_pin));
|
2017-05-10 23:06:43 +02:00
|
|
|
/* mandatory pins */
|
2020-01-17 12:45:13 +01:00
|
|
|
assert(gpio_is_valid(soft_spi_config[bus].mosi_pin) ||
|
|
|
|
gpio_is_valid(soft_spi_config[bus].miso_pin));
|
|
|
|
assert(gpio_is_valid(soft_spi_config[bus].clk_pin));
|
2017-05-10 23:06:43 +02:00
|
|
|
|
|
|
|
/* initialize clock pin */
|
|
|
|
gpio_init(soft_spi_config[bus].clk_pin, GPIO_OUT);
|
|
|
|
/* initialize optional pins */
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(soft_spi_config[bus].mosi_pin)) {
|
2017-05-10 23:06:43 +02:00
|
|
|
gpio_init(soft_spi_config[bus].mosi_pin, GPIO_OUT);
|
|
|
|
gpio_clear(soft_spi_config[bus].mosi_pin);
|
|
|
|
}
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(soft_spi_config[bus].miso_pin)) {
|
2021-03-08 07:23:29 +01:00
|
|
|
gpio_init(soft_spi_config[bus].miso_pin, GPIO_IN);
|
2017-05-10 23:06:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int soft_spi_init_cs(soft_spi_t bus, soft_spi_cs_t cs)
|
|
|
|
{
|
|
|
|
DEBUG("Soft SPI init CS\n");
|
|
|
|
if (!soft_spi_bus_is_valid(bus)) {
|
|
|
|
DEBUG("Soft SPI bus not valid\n");
|
|
|
|
return SOFT_SPI_NODEV;
|
|
|
|
}
|
|
|
|
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(cs) && !gpio_is_equal(cs, SOFT_SPI_CS_UNDEF)) {
|
2017-05-10 23:06:43 +02:00
|
|
|
DEBUG("Soft SPI set user CS line\n");
|
|
|
|
gpio_init(cs, GPIO_OUT);
|
|
|
|
gpio_set(cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
return SOFT_SPI_OK;
|
|
|
|
}
|
|
|
|
|
2021-02-01 14:23:28 +01:00
|
|
|
static inline int soft_spi_mode_is_valid(soft_spi_mode_t mode)
|
2017-05-10 23:06:43 +02:00
|
|
|
{
|
2021-02-01 14:23:28 +01:00
|
|
|
if ((mode != SOFT_SPI_MODE_0) && (mode != SOFT_SPI_MODE_1) &&
|
|
|
|
(mode != SOFT_SPI_MODE_2) && (mode != SOFT_SPI_MODE_3)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void soft_spi_acquire(soft_spi_t bus, soft_spi_cs_t cs, soft_spi_mode_t mode, soft_spi_clk_t clk)
|
|
|
|
{
|
|
|
|
(void)cs;
|
2017-05-10 23:06:43 +02:00
|
|
|
assert(soft_spi_bus_is_valid(bus));
|
2021-02-01 14:23:28 +01:00
|
|
|
assert(soft_spi_mode_is_valid(mode));
|
2017-05-10 23:06:43 +02:00
|
|
|
|
|
|
|
/* lock bus */
|
|
|
|
mutex_lock(&locks[bus]);
|
|
|
|
|
|
|
|
soft_spi_config[bus].soft_spi_mode = mode;
|
|
|
|
switch (mode) {
|
|
|
|
case SOFT_SPI_MODE_0:
|
|
|
|
case SOFT_SPI_MODE_1:
|
|
|
|
/* CPOL=0 */
|
|
|
|
gpio_clear(soft_spi_config[bus].clk_pin);
|
|
|
|
break;
|
|
|
|
case SOFT_SPI_MODE_2:
|
|
|
|
case SOFT_SPI_MODE_3:
|
|
|
|
/* CPOL=1 */
|
|
|
|
gpio_set(soft_spi_config[bus].clk_pin);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
soft_spi_config[bus].soft_spi_clk = clk;
|
|
|
|
}
|
|
|
|
|
|
|
|
void soft_spi_release(soft_spi_t bus)
|
|
|
|
{
|
|
|
|
assert(soft_spi_bus_is_valid(bus));
|
|
|
|
mutex_unlock(&locks[bus]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint8_t _transfer_one_byte(soft_spi_t bus, uint8_t out)
|
|
|
|
{
|
2021-03-08 07:23:29 +01:00
|
|
|
uint8_t i = 8;
|
2017-05-10 23:06:43 +02:00
|
|
|
if (SOFT_SPI_MODE_1 == soft_spi_config[bus].soft_spi_mode ||
|
|
|
|
SOFT_SPI_MODE_3 == soft_spi_config[bus].soft_spi_mode) {
|
|
|
|
/* CPHA = 1*/
|
|
|
|
gpio_toggle(soft_spi_config[bus].clk_pin);
|
|
|
|
}
|
|
|
|
|
2021-03-08 07:23:29 +01:00
|
|
|
do {
|
|
|
|
uint8_t bit = out >> 7;
|
|
|
|
gpio_write(soft_spi_config[bus].mosi_pin, bit);
|
|
|
|
|
2021-11-30 09:32:54 +01:00
|
|
|
xtimer_usleep(soft_spi_config[bus].soft_spi_clk);
|
2017-05-10 23:06:43 +02:00
|
|
|
gpio_toggle(soft_spi_config[bus].clk_pin);
|
2021-03-08 07:23:29 +01:00
|
|
|
|
|
|
|
out <<= 1; /*shift transfer register*/
|
|
|
|
|
|
|
|
bit = gpio_read(soft_spi_config[bus].miso_pin);
|
|
|
|
out = bit ? (out | 0x01) : (out & 0xfe); /*set or delete bit 0*/
|
|
|
|
|
2021-11-30 09:32:54 +01:00
|
|
|
xtimer_usleep(soft_spi_config[bus].soft_spi_clk);
|
2021-03-08 07:23:29 +01:00
|
|
|
--i;
|
|
|
|
if (i > 0) {
|
|
|
|
gpio_toggle(soft_spi_config[bus].clk_pin);
|
|
|
|
}
|
|
|
|
} while (i > 0);
|
2017-05-10 23:06:43 +02:00
|
|
|
|
|
|
|
if (SOFT_SPI_MODE_0 == soft_spi_config[bus].soft_spi_mode ||
|
|
|
|
SOFT_SPI_MODE_2 == soft_spi_config[bus].soft_spi_mode) {
|
2021-03-08 07:23:29 +01:00
|
|
|
/* CPHA = 0 */
|
2021-11-30 09:32:54 +01:00
|
|
|
xtimer_usleep(soft_spi_config[bus].soft_spi_clk);
|
2017-05-10 23:06:43 +02:00
|
|
|
gpio_toggle(soft_spi_config[bus].clk_pin);
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t soft_spi_transfer_byte(soft_spi_t bus, soft_spi_cs_t cs, bool cont, uint8_t out)
|
|
|
|
{
|
|
|
|
DEBUG("Soft SPI soft_spi_transfer_bytes\n");
|
|
|
|
assert(soft_spi_bus_is_valid(bus));
|
|
|
|
|
|
|
|
uint8_t retval = 0;
|
|
|
|
|
|
|
|
/* activate the given chip select line */
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(cs) && !gpio_is_equal(cs, SOFT_SPI_CS_UNDEF)) {
|
2017-05-10 23:06:43 +02:00
|
|
|
gpio_clear((gpio_t)cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
retval = _transfer_one_byte(bus, out);
|
|
|
|
|
|
|
|
if (!cont) {
|
2020-01-17 12:45:13 +01:00
|
|
|
if (gpio_is_valid(cs) && !gpio_is_equal(cs, SOFT_SPI_CS_UNDEF)) {
|
2017-05-10 23:06:43 +02:00
|
|
|
gpio_set((gpio_t)cs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
void soft_spi_transfer_bytes(soft_spi_t bus, soft_spi_cs_t cs, bool cont,
|
2021-03-08 07:23:29 +01:00
|
|
|
const void *out, void *in, size_t len)
|
2017-05-10 23:06:43 +02:00
|
|
|
{
|
|
|
|
DEBUG("Soft SPI soft_spi_transfer_bytes\n");
|
|
|
|
assert(soft_spi_bus_is_valid(bus));
|
2021-03-08 07:23:29 +01:00
|
|
|
/* make sure at least one input or one output buffer is given */
|
|
|
|
assert(out || in);
|
2017-05-10 23:06:43 +02:00
|
|
|
|
2021-03-08 07:23:29 +01:00
|
|
|
const uint8_t *outbuf = out;
|
|
|
|
uint8_t *inbuf = in;
|
2017-05-10 23:06:43 +02:00
|
|
|
|
2021-03-08 07:23:29 +01:00
|
|
|
/* activate the given chip select line */
|
|
|
|
if ((cs != GPIO_UNDEF) && (cs != SOFT_SPI_CS_UNDEF)) {
|
|
|
|
gpio_clear((gpio_t)cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!inbuf) {
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
|
|
_transfer_one_byte(bus, outbuf[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!outbuf) {
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
|
|
inbuf[i] = _transfer_one_byte(bus, READ_PADDING_BYTE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
|
|
inbuf[i] = _transfer_one_byte(bus, outbuf[i]);
|
2017-05-10 23:06:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-08 07:23:29 +01:00
|
|
|
if (!cont) {
|
|
|
|
if ((cs != GPIO_UNDEF) && (cs != SOFT_SPI_CS_UNDEF)) {
|
|
|
|
gpio_set((gpio_t)cs);
|
|
|
|
}
|
|
|
|
}
|
2017-05-10 23:06:43 +02:00
|
|
|
}
|