/* * 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. */ /** * @defgroup driver_soft_spi Soft SPI * @ingroup drivers * @brief Software implemented Serial Peripheral Interface bus * This module provides a software implemented Serial Peripheral Interface bus. * It is intended to be used in situation where hardware spi is not available. * The signatures of the functions are similar to the functions declared in spi.h * The clock speed is approximated by using xtimer_nanosleep. * Currently only the use of MOSI in master mode is implemented. Therefore receiving * data from a slave is currently not possible. * @{ * * @file * @brief Software SPI port descriptor definition * * @author Markus Blechschmidt */ #ifndef SOFT_SPI_H #define SOFT_SPI_H #include "periph/gpio.h" #include "periph/spi.h" #include "mutex.h" #ifdef __cplusplus extern "C" { #endif /** * @brief Default SPI device access macro */ #ifndef SOFT_SPI_DEV #define SOFT_SPI_DEV(x) (x) #endif /** * @brief Define global value for undefined SPI device */ #ifndef SOFT_SPI_UNDEF #define SOFT_SPI_UNDEF (UINT_MAX) #endif /** * @brief Define value for unused CS line */ #ifndef SOFT_SPI_CS_UNDEF #define SOFT_SPI_CS_UNDEF (GPIO_UNDEF) #endif /** * @brief Default type for SPI devices */ typedef unsigned int soft_spi_t; /** * @brief Chip select pin type overlaps with gpio_t so it can be casted to * this */ typedef gpio_t soft_spi_cs_t; /** * @brief Status codes used by the SPI driver interface */ enum { SOFT_SPI_OK = 0, /**< everything went as planned */ SOFT_SPI_NODEV = -1, /**< invalid SPI bus specified */ SOFT_SPI_NOCS = -2, /**< invalid chip select line specified */ SOFT_SPI_NOMODE = -3, /**< selected mode is not supported */ SOFT_SPI_NOCLK = -4 /**< selected clock value is not supported */ }; /** * @brief Available SPI modes, defining the configuration of clock polarity * and clock phase * * RIOT is using the mode numbers as commonly defined by most vendors * (https://en.wikipedia.org/wiki/Serial_Peripheral_Interface_Bus#Mode_numbers): * * - MODE_0: CPOL=0, CPHA=0 - The first data bit is sampled by the receiver on * the first SCK rising SCK edge (this mode is used most often). * - MODE_1: CPOL=0, CPHA=1 - The first data bit is sampled by the receiver on * the second rising SCK edge. * - MODE_2: CPOL=1, CPHA=0 - The first data bit is sampled by the receiver on * the first falling SCK edge. * - MODE_3: CPOL=1, CPHA=1 - The first data bit is sampled by the receiver on * the second falling SCK edge. */ typedef enum { SOFT_SPI_MODE_0 = 0, /**< CPOL=0, CPHA=0 */ SOFT_SPI_MODE_1, /**< CPOL=0, CPHA=1 */ SOFT_SPI_MODE_2, /**< CPOL=1, CPHA=0 */ SOFT_SPI_MODE_3 /**< CPOL=1, CPHA=1 */ } soft_spi_mode_t; /** * @brief Available SPI clock speeds * * The actual speed of the bus varies between CPUs and depends on the speed * of the processing. The values of the enum entries represent the approximate * delay between two clock edges. */ typedef enum { SOFT_SPI_CLK_100KHZ = 5000, /**< drive the SPI bus with less than 100kHz */ SOFT_SPI_CLK_400KHZ = 1250, /**< drive the SPI bus with less than 400kHz */ SOFT_SPI_CLK_DEFAULT = 0, /**< drive the SPI bus with maximum speed possible */ } soft_spi_clk_t; /** * @brief Software SPI port descriptor */ typedef struct { gpio_t miso_pin; /**< MOSI pin */ gpio_t mosi_pin; /**< MOSI pin */ gpio_t clk_pin; /**< CLK pin */ soft_spi_mode_t soft_spi_mode; /**< data and clock polarity */ soft_spi_clk_t soft_spi_clk; /**< clock speed */ } soft_spi_conf_t; /** * @brief Basic initialization of the given SPI bus * * This function does the basic initialization including pin configuration for * MISO, MOSI, and CLK pins. * * Errors (e.g. invalid @p bus parameter) are not signaled through a return * value, but should be signaled using the assert() function internally. * * @note This function MUST not be called more than once per bus! * * @param[in] bus SPI device to initialize */ void soft_spi_init(soft_spi_t bus); /** * @brief Initialize the used SPI bus pins, i.e. MISO, MOSI, and CLK * * After calling soft_spi_init, the pins must be initialized. In normal cases, * this function will not be used. * * The pins used are configured in the board's periph_conf.h. * * @param[in] bus SPI device the pins are configure for */ void soft_spi_init_pins(soft_spi_t bus); /** * @brief Initialize the given chip select pin * * The chip select must be any generic GPIO pin (e.g. GPIO_PIN(x,y)). It must be * called once before the use of the chip select pin in transaction. * * @param[in] bus SPI device that is used with the given CS line * @param[in] cs chip select pin to initialize * * @return SOFT_SPI_OK on success * @return SOFT_SPI_NODEV on invalid device * @return SOFT_SPI_NOCS on invalid CS pin/line */ int soft_spi_init_cs(soft_spi_t bus, soft_spi_cs_t cs); /** * @brief Start a new SPI transaction * * Starting a new SPI transaction will get exclusive access to the SPI bus * and configure it according to the given values. If another SPI transaction * is active when this function is called, this function will block until the * other transaction is complete (soft_spi_relase was called). * * @note This function expects the @p bus and the @p cs parameters to be * valid (they are checked in soft_spi_init and soft_spi_init_cs before) * * @param[in] bus SPI device to access * @param[in] cs chip select pin/line to use * @param[in] mode mode to use for the new transaction * @param[in] clk bus clock speed to use for the transaction * * @return SOFT_SPI_OK on success * @return SOFT_SPI_NOMODE if given mode is not supported * @return SOFT_SPI_NOCLK if given clock speed is not supported */ int soft_spi_acquire(soft_spi_t bus, soft_spi_cs_t cs, soft_spi_mode_t mode, soft_spi_clk_t clk); /** * @brief Finish an ongoing SPI transaction by releasing the given SPI bus * * After release, the given SPI bus should be fully powered down until acquired * again. * * @param[in] bus SPI device to release */ void soft_spi_release(soft_spi_t bus); /** * @brief Transfer one byte on the given SPI bus * Currently only the use of MOSI in master mode is implemented. Therefore receiving * data from a slave is currently not possible. * * @param[in] bus SPI device to use * @param[in] cs chip select pin/line to use * @param[in] cont if true, keep device selected after transfer * @param[in] out byte to send out, set NULL if only receiving * * @return the received byte */ uint8_t soft_spi_transfer_byte(soft_spi_t bus, soft_spi_cs_t cs, bool cont, uint8_t out); /** * @brief Transfer a number bytes using the given SPI bus * * @param[in] bus SPI device to use * @param[in] cs chip select pin/line to use * @param[in] cont if true, keep device selected after transfer * @param[in] out buffer to send data from, set NULL if only receiving * @param[out] in buffer to read into, set NULL if only sending * @param[in] len number of bytes to transfer */ void soft_spi_transfer_bytes(soft_spi_t bus, soft_spi_cs_t cs, bool cont, const void *out, void *in, size_t len); /** * @brief Transfer one byte to/from a given register address * * This function is a shortcut function for easier handling of SPI devices that * implement a register based access scheme. * * @param[in] bus SPI device to use * @param[in] cs chip select pin/line to use * @param[in] reg register address to transfer data to/from * @param[in] out byte to send, set NULL if only receiving data * * @return value that was read from the given register address */ uint8_t soft_spi_transfer_reg(soft_spi_t bus, soft_spi_cs_t cs, uint8_t reg, uint8_t out); /** * @brief Transfer a number of bytes to/from a given register address * * This function is a shortcut function for easier handling of SPI devices that * implement a register based access scheme. * * @param[in] bus SPI device to use * @param[in] cs chip select pin/line to use * @param[in] reg register address to transfer data to/from * @param[in] out buffer to send data from, set NULL if only receiving * @param[out] in buffer to read into, set NULL if only sending * @param[in] len number of bytes to transfer */ void soft_spi_transfer_regs(soft_spi_t bus, soft_spi_cs_t cs, uint8_t reg, const void *out, void *in, size_t len); #ifdef __cplusplus } #endif #endif /* SOFT_SPI_H */ /** @} */