2014-06-11 00:12:51 +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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2014-10-25 15:37:04 +02:00
|
|
|
* @defgroup driver_periph_spi SPI
|
2014-06-11 00:12:51 +02:00
|
|
|
* @ingroup driver_periph
|
|
|
|
* @brief Low-level SPI peripheral driver
|
|
|
|
*
|
2014-06-20 17:52:55 +02:00
|
|
|
* The current design of this interface targets implementations that use the SPI in blocking mode.
|
|
|
|
*
|
|
|
|
* TODO: add means for asynchronous SPI usage
|
2014-06-11 00:12:51 +02:00
|
|
|
*
|
2014-12-04 10:03:15 +01:00
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @brief Low-level SPI peripheral driver interface definitions
|
|
|
|
*
|
2014-06-11 00:12:51 +02:00
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __SPI_H
|
|
|
|
#define __SPI_H
|
|
|
|
|
|
|
|
#include "periph_conf.h"
|
|
|
|
|
2014-10-13 15:49:17 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2014-07-24 21:38:17 +02:00
|
|
|
/* add guard for the case that no SPI device is defined */
|
|
|
|
#if SPI_NUMOF
|
2014-06-11 00:12:51 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Definition available SPI devices
|
|
|
|
*/
|
|
|
|
typedef enum {
|
|
|
|
#if SPI_0_EN
|
|
|
|
SPI_0 = 0, /**< SPI device 0 */
|
|
|
|
#endif
|
|
|
|
#if SPI_1_EN
|
|
|
|
SPI_1, /**< SPI device 1 */
|
|
|
|
#endif
|
|
|
|
#if SPI_2_EN
|
|
|
|
SPI_2, /**< SPI device 2 */
|
|
|
|
#endif
|
|
|
|
#if SPI_3_EN
|
|
|
|
SPI_3, /**< SPI device 3 */
|
|
|
|
#endif
|
|
|
|
} spi_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief The SPI mode is defined by the four possible combinations of clock polarity and
|
|
|
|
* clock phase.
|
|
|
|
*/
|
|
|
|
typedef enum {
|
2015-01-20 09:06:16 +01:00
|
|
|
/**
|
|
|
|
* The first data bit is sampled by the receiver on the first SCK edge. The
|
|
|
|
* first edge of SCK is rising. This is sometimes also referred to as SPI
|
|
|
|
* mode 0, or (CPOL=0, CPHA=0).
|
|
|
|
*/
|
|
|
|
SPI_CONF_FIRST_RISING = 0,
|
|
|
|
/**
|
|
|
|
* The first data bit is sampled by the receiver on the second SCK edge. The
|
|
|
|
* first edge of SCK is rising, i.e. the sampling edge is falling. This is
|
|
|
|
* sometimes also referred to as SPI mode 1, or (CPOL=0, CPHA=1).
|
|
|
|
*/
|
|
|
|
SPI_CONF_SECOND_RISING = 1,
|
|
|
|
/**
|
|
|
|
* The first data bit is sampled by the receiver on the first SCK edge. The
|
|
|
|
* first edge of SCK is falling. This is sometimes also referred to as SPI
|
|
|
|
* mode 2, or (CPOL=1, CPHA=0).
|
|
|
|
*/
|
|
|
|
SPI_CONF_FIRST_FALLING = 2,
|
|
|
|
/**
|
|
|
|
* The first data bit is sampled by the receiver on the second SCK edge. The
|
|
|
|
* first edge of SCK is falling, i.e. the sampling edge is rising. This is
|
|
|
|
* sometimes also referred to as SPI mode 3, or (CPOL=1, CPHA=1).
|
|
|
|
*/
|
|
|
|
SPI_CONF_SECOND_FALLING = 3
|
2014-06-11 00:12:51 +02:00
|
|
|
} spi_conf_t;
|
|
|
|
|
2014-07-24 14:48:57 +02:00
|
|
|
/**
|
|
|
|
* @brief Define a set of pre-defined SPI clock speeds.
|
|
|
|
*
|
|
|
|
* The actual speed of the bus can vary to some extend, as the combination of CPU clock and
|
|
|
|
* available prescale values on certain platforms may not make the exact values possible.
|
|
|
|
*/
|
|
|
|
typedef enum {
|
|
|
|
SPI_SPEED_100KHZ = 0, /**< drive the SPI bus with 100KHz */
|
|
|
|
SPI_SPEED_400KHZ, /**< drive the SPI bus with 400KHz */
|
|
|
|
SPI_SPEED_1MHZ, /**< drive the SPI bus with 1MHz */
|
|
|
|
SPI_SPEED_5MHZ, /**< drive the SPI bus with 5MHz */
|
|
|
|
SPI_SPEED_10MHZ /**< drive the SPI bus with 10MHz */
|
|
|
|
} spi_speed_t;
|
2014-06-11 00:12:51 +02:00
|
|
|
|
|
|
|
/**
|
2014-06-12 13:34:13 +02:00
|
|
|
* @brief Initialize the given SPI device to work in master mode
|
|
|
|
*
|
|
|
|
* In master mode the SPI device is configured to control the SPI bus. This means the device
|
|
|
|
* will start and end all communication on the bus and control the CLK line. For transferring
|
|
|
|
* data on the bus the below defined transfer functions should be used.
|
2014-06-11 00:12:51 +02:00
|
|
|
*
|
|
|
|
* @param[in] dev SPI device to initialize
|
|
|
|
* @param[in] conf Mode of clock phase and clock polarity
|
2014-07-24 14:48:57 +02:00
|
|
|
* @param[in] speed desired clock speed for driving the SPI bus
|
2014-06-11 00:12:51 +02:00
|
|
|
*
|
|
|
|
* @return 0 on success
|
2014-07-24 23:57:15 +02:00
|
|
|
* @return -1 on unavailable speed value
|
|
|
|
* @return -2 on other errors
|
2014-06-11 00:12:51 +02:00
|
|
|
*/
|
2014-07-24 14:48:57 +02:00
|
|
|
int spi_init_master(spi_t dev, spi_conf_t conf, spi_speed_t speed);
|
2014-06-12 13:34:13 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize the given SPI device to work in slave mode
|
|
|
|
*
|
|
|
|
* In slave mode the SPI device is purely reacting to the bus. Transaction will be started and
|
|
|
|
* ended by a connected SPI master. When a byte is received, the callback is called in interrupt
|
|
|
|
* context with this byte as argument. The return byte of the callback is transferred to the
|
|
|
|
* master in the next transmission cycle. This interface enables easy implementation of a register
|
|
|
|
* based access paradigm for the SPI slave.
|
|
|
|
*
|
|
|
|
* @param[in] dev The SPI device to initialize as SPI slave
|
|
|
|
* @param[in] conf Mode of clock phase and polarity
|
2014-07-24 23:57:15 +02:00
|
|
|
* @param[in] cb callback called every time a byte was received
|
2014-06-12 13:34:13 +02:00
|
|
|
*
|
|
|
|
* @return 0 on success
|
2014-07-24 23:57:15 +02:00
|
|
|
* @return -1 on error
|
2014-06-12 13:34:13 +02:00
|
|
|
*/
|
2014-07-24 23:57:15 +02:00
|
|
|
int spi_init_slave(spi_t dev, spi_conf_t conf, char (*cb)(char data));
|
2014-06-11 00:12:51 +02:00
|
|
|
|
2014-10-06 15:23:07 +02:00
|
|
|
/**
|
|
|
|
* @brief Configure SCK, MISO and MOSI pins for the given SPI device
|
|
|
|
*
|
|
|
|
* @param[in] dev SPI device to use
|
|
|
|
*
|
|
|
|
* @return 0 on success
|
|
|
|
* @return -1 on error
|
|
|
|
*/
|
|
|
|
int spi_conf_pins(spi_t dev);
|
|
|
|
|
2015-01-13 14:58:57 +01:00
|
|
|
/**
|
|
|
|
* @brief Get mutually exclusive access to the given SPI bus
|
|
|
|
*
|
|
|
|
* In case the SPI device is busy, this function will block until the bus is free again.
|
|
|
|
*
|
|
|
|
* @param[in] dev SPI device to access
|
|
|
|
*
|
|
|
|
* @return 0 on success
|
|
|
|
* @return -1 on error
|
|
|
|
*/
|
|
|
|
int spi_acquire(spi_t dev);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Release the given SPI device to be used by others
|
|
|
|
*
|
|
|
|
* @param[in] dev SPI device to release
|
|
|
|
*
|
|
|
|
* @return 0 on success
|
|
|
|
* @return -1 on error
|
|
|
|
*/
|
|
|
|
int spi_release(spi_t dev);
|
|
|
|
|
2014-06-11 00:12:51 +02:00
|
|
|
/**
|
|
|
|
* @brief Transfer one byte on the given SPI bus
|
|
|
|
*
|
|
|
|
* @param[in] dev SPI device to use
|
|
|
|
* @param[in] out Byte to send out, set NULL if only receiving
|
|
|
|
* @param[out] in Byte to read, set NULL if only sending
|
|
|
|
*
|
|
|
|
* @return Number of bytes that were transfered
|
|
|
|
* @return -1 on error
|
|
|
|
*/
|
|
|
|
int spi_transfer_byte(spi_t dev, char out, char *in);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Transfer a number bytes on the given SPI bus
|
|
|
|
*
|
|
|
|
* @param[in] dev SPI device to use
|
|
|
|
* @param[in] out Array of bytes to send, set NULL if only receiving
|
|
|
|
* @param[out] in Buffer to receive bytes to, set NULL if only sending
|
|
|
|
* @param[in] length Number of bytes to transfer
|
|
|
|
*
|
|
|
|
* @return Number of bytes that were transfered
|
|
|
|
* @return -1 on error
|
|
|
|
*/
|
2014-06-20 17:52:55 +02:00
|
|
|
int spi_transfer_bytes(spi_t dev, char *out, char *in, unsigned int length);
|
2014-06-11 00:12:51 +02:00
|
|
|
|
|
|
|
/**
|
2014-06-12 13:34:13 +02:00
|
|
|
* @brief Transfer one byte to/from a given register address
|
2014-06-11 00:12:51 +02:00
|
|
|
*
|
|
|
|
* This function is a shortcut function for easier handling of register based SPI devices. As
|
2014-06-12 13:34:13 +02:00
|
|
|
* many SPI devices use a register based addressing scheme, this function is a convenient short-
|
2014-06-11 00:12:51 +02:00
|
|
|
* cut for interfacing with such devices.
|
|
|
|
*
|
|
|
|
* @param[in] dev SPI device to use
|
|
|
|
* @param[in] reg Register address to transfer data to/from
|
|
|
|
* @param[in] out Byte to send, set NULL if only receiving data
|
|
|
|
* @param[out] in Byte to read, set NULL if only sending
|
|
|
|
*
|
|
|
|
* @return Number of bytes that were transfered
|
|
|
|
* @return -1 on error
|
|
|
|
*/
|
2014-07-24 22:55:17 +02:00
|
|
|
int spi_transfer_reg(spi_t dev, uint8_t reg, char out, char *in);
|
2014-06-11 00:12:51 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Transfer a number of bytes from/to a given register address
|
|
|
|
*
|
|
|
|
* This function is a shortcut function for easier handling of register based SPI devices. As
|
2014-06-12 13:34:13 +02:00
|
|
|
* many SPI devices use a register based addressing scheme, this function is a convenient short-
|
2014-06-11 00:12:51 +02:00
|
|
|
* cut for interfacing with such devices.
|
|
|
|
*
|
|
|
|
* @param[in] dev SPI device to use
|
|
|
|
* @param[in] reg Register address to transfer data to/from
|
|
|
|
* @param[in] out Byte array to send data from, set NULL if only receiving
|
|
|
|
* @param[out] in Byte buffer to read into, set NULL if only sending
|
|
|
|
* @param[in] length Number of bytes to transfer
|
|
|
|
*
|
|
|
|
* @return Number of bytes that were transfered
|
|
|
|
* @return -1 on error
|
|
|
|
*/
|
2014-06-20 17:52:55 +02:00
|
|
|
int spi_transfer_regs(spi_t dev, uint8_t reg, char *out, char *in, unsigned int length);
|
2014-06-11 00:12:51 +02:00
|
|
|
|
2014-07-24 23:57:15 +02:00
|
|
|
/**
|
|
|
|
* @brief Tell the SPI driver that a new transaction was started. Call only when SPI in slave mode!
|
|
|
|
*
|
|
|
|
* @param[in] dev SPI device that is active
|
|
|
|
* @param[in] reset_val The byte that is send to the master as first byte
|
|
|
|
*/
|
|
|
|
void spi_transmission_begin(spi_t dev, char reset_val);
|
|
|
|
|
2014-06-11 00:12:51 +02:00
|
|
|
/**
|
|
|
|
* @brief Power on the given SPI device
|
|
|
|
*
|
|
|
|
* @param[in] dev SPI device to power on
|
|
|
|
*/
|
2014-07-24 21:38:17 +02:00
|
|
|
void spi_poweron(spi_t dev);
|
2014-06-11 00:12:51 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Power off the given SPI device
|
|
|
|
*
|
|
|
|
* @param[in] dev SPI device to power off
|
|
|
|
*/
|
2014-07-24 21:38:17 +02:00
|
|
|
void spi_poweroff(spi_t dev);
|
|
|
|
|
|
|
|
#endif /* SPI_NUMOF */
|
2014-06-20 17:52:55 +02:00
|
|
|
|
2014-10-13 15:49:17 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-06-20 17:52:55 +02:00
|
|
|
#endif /* __SPI_H */
|
|
|
|
/** @} */
|