1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/drivers/soft_spi/soft_spi.c
2019-08-06 19:43:54 +02:00

212 lines
5.7 KiB
C

/*
* 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.
*/
/**
* @ingroup drivers_soft_spi
* @{
*
* @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"
#define ENABLE_DEBUG (0)
#include "debug.h"
/**
* @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;
if (ARRAY_SIZE(soft_spi_config) < soft_spi_num) {
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*/
assert(soft_spi_config[bus].mosi_pin != soft_spi_config[bus].miso_pin);
assert(soft_spi_config[bus].mosi_pin != soft_spi_config[bus].clk_pin);
assert(soft_spi_config[bus].miso_pin != soft_spi_config[bus].clk_pin);
/* mandatory pins */
assert((GPIO_UNDEF != soft_spi_config[bus].mosi_pin) || (GPIO_UNDEF != soft_spi_config[bus].miso_pin));
assert(GPIO_UNDEF != soft_spi_config[bus].clk_pin);
/* initialize clock pin */
gpio_init(soft_spi_config[bus].clk_pin, GPIO_OUT);
/* initialize optional pins */
if (GPIO_UNDEF != soft_spi_config[bus].mosi_pin) {
gpio_init(soft_spi_config[bus].mosi_pin, GPIO_OUT);
gpio_clear(soft_spi_config[bus].mosi_pin);
}
if (GPIO_UNDEF != soft_spi_config[bus].miso_pin) {
gpio_init(soft_spi_config[bus].mosi_pin, GPIO_IN);
}
}
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;
}
if ((cs != GPIO_UNDEF) && (cs != SOFT_SPI_CS_UNDEF)) {
DEBUG("Soft SPI set user CS line\n");
gpio_init(cs, GPIO_OUT);
gpio_set(cs);
}
return SOFT_SPI_OK;
}
int soft_spi_acquire(soft_spi_t bus, soft_spi_cs_t cs, soft_spi_mode_t mode, soft_spi_clk_t clk)
{
(void) cs;
assert(soft_spi_bus_is_valid(bus));
/* lock bus */
mutex_lock(&locks[bus]);
if ((mode != SOFT_SPI_MODE_0) && (mode != SOFT_SPI_MODE_1) &&
(mode != SOFT_SPI_MODE_2) && (mode != SOFT_SPI_MODE_3)) {
return SOFT_SPI_NOMODE;
}
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;
return SOFT_SPI_OK;
}
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)
{
int8_t bit = 0, i = 0;
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);
}
bit = (out & (1 << 7)) >> 7;
gpio_write(soft_spi_config[bus].mosi_pin, bit);
for (i = 6; i >= 0; i--) {
xtimer_nanosleep(soft_spi_config[bus].soft_spi_clk);
gpio_toggle(soft_spi_config[bus].clk_pin);
xtimer_nanosleep(soft_spi_config[bus].soft_spi_clk);
gpio_toggle(soft_spi_config[bus].clk_pin);
bit = (out & (1 << i)) >> i;
gpio_write(soft_spi_config[bus].mosi_pin, bit);
}
xtimer_nanosleep(soft_spi_config[bus].soft_spi_clk);
gpio_toggle(soft_spi_config[bus].clk_pin);
if (SOFT_SPI_MODE_0 == soft_spi_config[bus].soft_spi_mode ||
SOFT_SPI_MODE_2 == soft_spi_config[bus].soft_spi_mode) {
/* CPHASE = 1 */
xtimer_nanosleep(soft_spi_config[bus].soft_spi_clk);
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 */
if ((cs != GPIO_UNDEF) && (cs != SOFT_SPI_CS_UNDEF)) {
gpio_clear((gpio_t)cs);
}
retval = _transfer_one_byte(bus, out);
if (!cont) {
if ((cs != GPIO_UNDEF) && (cs != SOFT_SPI_CS_UNDEF)) {
gpio_set((gpio_t)cs);
}
}
return retval;
}
void soft_spi_transfer_bytes(soft_spi_t bus, soft_spi_cs_t cs, bool cont,
const void *out, void *in, size_t len)
{
DEBUG("Soft SPI soft_spi_transfer_bytes\n");
assert(soft_spi_bus_is_valid(bus));
uint8_t tmp = 0;
for (size_t i = 0; i < len-1; i++) {
tmp = (NULL != out) ? ((uint8_t *)out)[i] : 0;
uint8_t retval = soft_spi_transfer_byte(bus, cs, true, tmp);
if (NULL != in) {
((uint8_t *)in)[0] = retval;
}
}
tmp = (NULL != out) ? ((uint8_t *)out)[len-1] : 0;
soft_spi_transfer_byte(bus, cs, cont, tmp);
}