mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-19 16:03:31 +08:00
867 lines
26 KiB
C
867 lines
26 KiB
C
/**
|
|
* \file
|
|
*
|
|
* \brief Serial Peripheral Interface Driver for SAMB11
|
|
*
|
|
* Copyright (C) 2015-2016 Atmel Corporation. All rights reserved.
|
|
*
|
|
* \asf_license_start
|
|
*
|
|
* \page License
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
*
|
|
* 3. The name of Atmel may not be used to endorse or promote products derived
|
|
* from this software without specific prior written permission.
|
|
*
|
|
* 4. This software may only be redistributed and used in connection with an
|
|
* Atmel microcontroller product.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR IMPLIED
|
|
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
|
* EXPRESSLY AND SPECIFICALLY DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR
|
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* \asf_license_stop
|
|
*
|
|
*/
|
|
/*
|
|
* Support and FAQ: visit <a href="http://www.atmel.com/design-support/">Atmel Support</a>
|
|
*/
|
|
#include "spi.h"
|
|
|
|
/**
|
|
* \brief Determines if the SPI module is currently synchronizing to the bus.
|
|
*
|
|
* This function will check if the underlying hardware peripheral module is
|
|
* currently synchronizing across multiple clock domains to the hardware bus.
|
|
* This function can be used to delay further operations on the module until it
|
|
* is ready.
|
|
*
|
|
* \param[in] module SPI hardware module
|
|
*
|
|
* \return Synchronization status of the underlying hardware module
|
|
* \retval true Module synchronization is ongoing
|
|
* \retval false Module synchronization is not ongoing
|
|
*
|
|
*/
|
|
static bool _spi_is_active(Spi *const spi_module)
|
|
{
|
|
Assert(spi_module);
|
|
|
|
return spi_module->SPI_BUS_STATUS.bit.SPI_ACTIVE;
|
|
}
|
|
|
|
/**
|
|
* \internal Enable SPI clock.
|
|
*
|
|
* This function will enable SPI clock.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
*/
|
|
static void _spi_clock_enable(struct spi_module *const module)
|
|
{
|
|
Assert(module);
|
|
|
|
Spi *const spi_module = (module->hw);
|
|
|
|
if (spi_module == (void *)SPI0) {
|
|
system_clock_peripheral_enable(PERIPHERAL_SPI0_SCK_CLK);
|
|
system_clock_peripheral_enable(PERIPHERAL_SPI0_SCK_PHASE);
|
|
system_clock_peripheral_enable(PERIPHERAL_SPI0_IF);
|
|
system_clock_peripheral_enable(PERIPHERAL_SPI0_CORE);
|
|
} else if (spi_module == (void *)SPI1) {
|
|
system_clock_peripheral_enable(PERIPHERAL_SPI1_SCK_CLK);
|
|
system_clock_peripheral_enable(PERIPHERAL_SPI1_SCK_PHASE);
|
|
system_clock_peripheral_enable(PERIPHERAL_SPI1_IF);
|
|
system_clock_peripheral_enable(PERIPHERAL_SPI1_CORE);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* \internal Disable SPI clock.
|
|
*
|
|
* This function will disable SPI clock.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
*/
|
|
static void _spi_clock_disable(struct spi_module *const module)
|
|
{
|
|
Assert(module);
|
|
|
|
Spi *const spi_module = (module->hw);
|
|
|
|
if (spi_module == (void *)SPI0) {
|
|
system_clock_peripheral_disable(PERIPHERAL_SPI0_SCK_CLK);
|
|
system_clock_peripheral_disable(PERIPHERAL_SPI0_SCK_PHASE);
|
|
system_clock_peripheral_disable(PERIPHERAL_SPI0_IF);
|
|
system_clock_peripheral_disable(PERIPHERAL_SPI0_CORE);
|
|
} else if (spi_module == (void *)SPI1) {
|
|
system_clock_peripheral_disable(PERIPHERAL_SPI1_SCK_CLK);
|
|
system_clock_peripheral_disable(PERIPHERAL_SPI1_SCK_PHASE);
|
|
system_clock_peripheral_disable(PERIPHERAL_SPI1_IF);
|
|
system_clock_peripheral_disable(PERIPHERAL_SPI1_CORE);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* \internal Writes an SPI configuration to the hardware module.
|
|
*
|
|
* This function will write out a given configuration to the hardware module.
|
|
* Can only be done when the module is disabled.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
* \param[in] config Pointer to the configuration struct
|
|
*
|
|
* \return The status of the configuration
|
|
* \retval STATUS_ERR_INVALID_ARG If invalid argument(s) were provided
|
|
* \retval STATUS_OK If the configuration was written
|
|
*/
|
|
static enum status_code _spi_set_config(
|
|
struct spi_module *const module,
|
|
const struct spi_config *const config)
|
|
{
|
|
Assert(module);
|
|
Assert(config);
|
|
|
|
Spi *const spi_module = (module->hw);
|
|
|
|
module->mode = config->mode;
|
|
|
|
#if CONF_SPI_MASTER_ENABLE == true
|
|
/* Find baud value and write it */
|
|
if (config->mode == SPI_MODE_MASTER) {
|
|
spi_module->SPI_CLK_DIVIDER.reg = config->clock_divider;
|
|
}
|
|
#endif
|
|
|
|
/* Set data order */
|
|
if (config->data_order == SPI_DATA_ORDER_LSB) {
|
|
spi_module->SPI_CONFIGURATION.bit.LSB_FIRST_ENABLE = 0x1;
|
|
} else {
|
|
spi_module->SPI_CONFIGURATION.bit.LSB_FIRST_ENABLE = 0x0;
|
|
}
|
|
|
|
/* Set clock polarity and clock phase */
|
|
switch(config->transfer_mode)
|
|
{
|
|
case SPI_TRANSFER_MODE_0:
|
|
spi_module->SPI_CONFIGURATION.bit.SCK_PHASE = 0x0;
|
|
spi_module->SPI_CONFIGURATION.bit.SCK_POLARITY = 0x0;
|
|
break;
|
|
case SPI_TRANSFER_MODE_1:
|
|
spi_module->SPI_CONFIGURATION.bit.SCK_PHASE = 0x1;
|
|
spi_module->SPI_CONFIGURATION.bit.SCK_POLARITY = 0x0;
|
|
break;
|
|
case SPI_TRANSFER_MODE_2:
|
|
spi_module->SPI_CONFIGURATION.bit.SCK_PHASE = 0x0;
|
|
spi_module->SPI_CONFIGURATION.bit.SCK_POLARITY = 0x1;
|
|
break;
|
|
case SPI_TRANSFER_MODE_3:
|
|
spi_module->SPI_CONFIGURATION.bit.SCK_PHASE = 0x1;
|
|
spi_module->SPI_CONFIGURATION.bit.SCK_POLARITY = 0x1;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return STATUS_OK;
|
|
}
|
|
|
|
/**
|
|
* \brief Checks if the SPI in master mode has shifted out last data, or if the
|
|
* master has ended the transfer in slave mode.
|
|
*
|
|
* This function will check if the SPI master module has shifted out last data,
|
|
* or if the slave select pin has been drawn high by the master for the SPI
|
|
* slave module.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
*
|
|
* \return Indication of whether any writes are ongoing
|
|
* \retval true If the SPI master module has shifted out data, or slave select
|
|
* has been drawn high for SPI slave
|
|
* \retval false If the SPI master module has not shifted out data
|
|
*/
|
|
static inline bool _spi_is_write_complete(
|
|
Spi *const spi_module)
|
|
{
|
|
Assert(spi_module);
|
|
|
|
/* Check interrupt flag */
|
|
return (spi_module->TRANSMIT_STATUS.bit.TX_FIFO_EMPTY);
|
|
}
|
|
|
|
|
|
/**
|
|
* \brief Checks if the SPI module is ready to write data
|
|
*
|
|
* This function will check if the SPI module is ready to write data.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
*
|
|
* \return Indication of whether the module is ready to read data or not
|
|
* \retval true If the SPI module is ready to write data
|
|
* \retval false If the SPI module is not ready to write data
|
|
*/
|
|
static inline bool _spi_is_ready_to_write(
|
|
Spi *const spi_module)
|
|
{
|
|
Assert(spi_module);
|
|
|
|
/* Check interrupt flag */
|
|
return (spi_module->TRANSMIT_STATUS.bit.TX_FIFO_NOT_FULL);
|
|
}
|
|
|
|
/**
|
|
* \brief Checks if the SPI module is ready to read data
|
|
*
|
|
* This function will check if the SPI module is ready to read data.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
*
|
|
* \return Indication of whether the module is ready to read data or not
|
|
* \retval true If the SPI module is ready to read data
|
|
* \retval false If the SPI module is not ready to read data
|
|
*/
|
|
static inline bool _spi_is_ready_to_read(
|
|
Spi *const spi_module)
|
|
{
|
|
Assert(spi_module);
|
|
|
|
/* Check interrupt flag */
|
|
return (spi_module->RECEIVE_STATUS.bit.RX_FIFO_NOT_EMPTY);
|
|
}
|
|
|
|
/**
|
|
* \brief Initializes an SPI peripheral slave device configuration structure to default values
|
|
*
|
|
* This function will initialize a given SPI slave device configuration
|
|
* structure to a set of known default values. This function should be called
|
|
* on any new instance of the configuration structures before being modified by
|
|
* the user application.
|
|
*
|
|
* The default configuration is as follows:
|
|
* \li Slave Select on GPIO pin 12
|
|
* \li Addressing not enabled
|
|
*
|
|
* \param[out] config Configuration structure to initialize to default values
|
|
*/
|
|
void spi_slave_inst_get_config_defaults(
|
|
struct spi_slave_inst_config *const config)
|
|
{
|
|
Assert(config);
|
|
|
|
config->ss_pin = PIN_LP_GPIO_12;
|
|
config->address_enabled = false;
|
|
config->address = 0;
|
|
}
|
|
|
|
/**
|
|
* \brief Initializes an SPI configuration structure to default values
|
|
*
|
|
* This function will initialize a given SPI configuration structure to a set
|
|
* of known default values. This function should be called on any new
|
|
* instance of the configuration structures before being modified by the
|
|
* user application.
|
|
*
|
|
* The default configuration is as follows:
|
|
* \li Master mode enabled
|
|
* \li MSB of the data is transmitted first
|
|
* \li Transfer mode 0
|
|
* \li MUX Setting 0
|
|
* \li Character size 8 bit
|
|
* \li Not enabled in sleep mode
|
|
* \li Receiver enabled
|
|
* \li Baudrate 50000
|
|
* \li Default pinmux settings for all pads
|
|
* \li Clock source 0 (26MHz)
|
|
* \li Clock divider (Formula: baud_rate = ((clock input freq/clock_divider+1)/2))
|
|
* (For Example: if clock source is CLOCK_INPUT_0 then
|
|
* ((26000000/(129+1))/2) = 100000 bps)
|
|
*
|
|
* \param[in,out] config Configuration structure to initialize to default values
|
|
*/
|
|
void spi_get_config_defaults(
|
|
struct spi_config *const config)
|
|
{
|
|
Assert(config);
|
|
|
|
config->mode = SPI_MODE_MASTER;
|
|
config->data_order = SPI_DATA_ORDER_MSB;
|
|
config->transfer_mode = SPI_TRANSFER_MODE_0;
|
|
config->clock_source = SPI_CLK_INPUT_0;
|
|
config->clock_divider = 129;
|
|
|
|
config->pin_number_pad[0] = PIN_LP_GPIO_10;
|
|
config->pin_number_pad[1] = PIN_LP_GPIO_11;
|
|
config->pin_number_pad[2] = PIN_LP_GPIO_12;
|
|
config->pin_number_pad[3] = PIN_LP_GPIO_13;
|
|
|
|
config->pinmux_sel_pad[0] = MUX_LP_GPIO_10_SPI0_SCK;
|
|
config->pinmux_sel_pad[1] = MUX_LP_GPIO_11_SPI0_MOSI;
|
|
config->pinmux_sel_pad[2] = MUX_LP_GPIO_12_SPI0_SSN;
|
|
config->pinmux_sel_pad[3] = MUX_LP_GPIO_13_SPI0_MISO;
|
|
};
|
|
|
|
/**
|
|
* \brief Attaches an SPI peripheral slave
|
|
*
|
|
* This function will initialize the software SPI peripheral slave, based on
|
|
* the values of the config struct. The slave can then be selected and
|
|
* optionally addressed by the \ref spi_select_slave function.
|
|
*
|
|
* \param[out] slave Pointer to the software slave instance struct
|
|
* \param[in] config Pointer to the config struct
|
|
*
|
|
*/
|
|
void spi_attach_slave(
|
|
struct spi_slave_inst *const slave,
|
|
struct spi_slave_inst_config *const config)
|
|
{
|
|
Assert(slave);
|
|
Assert(config);
|
|
|
|
slave->ss_pin = config->ss_pin;
|
|
slave->address_enabled = config->address_enabled;
|
|
slave->address = config->address;
|
|
|
|
struct gpio_config config_gpio;
|
|
gpio_get_config_defaults(&config_gpio);
|
|
config_gpio.direction = GPIO_PIN_DIR_OUTPUT;
|
|
gpio_pin_set_config(slave->ss_pin, &config_gpio);
|
|
|
|
gpio_pin_set_output_level(slave->ss_pin, true);
|
|
}
|
|
|
|
/**
|
|
* \brief Resets the SPI module
|
|
*
|
|
* This function will reset the SPI module to its power on default values and
|
|
* disable it.
|
|
*
|
|
* \param[in,out] module Pointer to the software instance struct
|
|
*/
|
|
void spi_reset(struct spi_module *const module)
|
|
{
|
|
/* Sanity check arguments */
|
|
Spi *const spi_module = (module->hw);
|
|
|
|
/* Disable the module */
|
|
spi_disable(module);
|
|
|
|
/* Software reset the module */
|
|
if(spi_module == (void *)SPI0) {
|
|
system_peripheral_reset(PERIPHERAL_SPI0_CORE);
|
|
system_peripheral_reset(PERIPHERAL_SPI0_IF);
|
|
} else if (spi_module == (void *)SPI1) {
|
|
system_peripheral_reset(PERIPHERAL_SPI1_CORE);
|
|
system_peripheral_reset(PERIPHERAL_SPI1_IF);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* \brief Initializes the SPI module
|
|
*
|
|
* This function will initialize the SPI module, based on the values
|
|
* of the config struct.
|
|
*
|
|
* \param[out] module Pointer to the software instance struct
|
|
* \param[in] hw Pointer to hardware instance
|
|
* \param[in] config Pointer to the config struct
|
|
*
|
|
* \return Status of the initialization
|
|
* \retval STATUS_OK Module initiated correctly
|
|
* \retval STATUS_ERR_DENIED If module is enabled
|
|
* \retval STATUS_BUSY If module is busy resetting
|
|
* \retval STATUS_ERR_INVALID_ARG If invalid argument(s) were provided
|
|
*/
|
|
enum status_code spi_init(
|
|
struct spi_module *const module,
|
|
Spi *const hw,
|
|
const struct spi_config *const config)
|
|
{
|
|
/* Sanity check arguments */
|
|
Assert(module);
|
|
Assert(hw);
|
|
Assert(config);
|
|
|
|
uint8_t idx;
|
|
|
|
/* Initialize device instance */
|
|
module->hw = hw;
|
|
|
|
Spi *const spi_module = (module->hw);
|
|
|
|
/* Check if module is enabled. */
|
|
if (spi_module->SPI_MODULE_ENABLE.reg & SPI_MODULE_ENABLE_MASK) {
|
|
spi_module->SPI_MODULE_ENABLE.reg = (0x0ul << SPI_MODULE_ENABLE_ENABLE_Pos);
|
|
}
|
|
|
|
spi_reset(module);
|
|
_spi_clock_enable(module);
|
|
|
|
#if SPI_CALLBACK_MODE == true
|
|
if (module->hw == SPI0) {
|
|
_spi_instances[0] = module;
|
|
system_register_isr(RAM_ISR_TABLE_SPIRX0_INDEX, (uint32_t)spi_rx0_isr_handler);
|
|
system_register_isr(RAM_ISR_TABLE_SPITX0_INDEX, (uint32_t)spi_tx0_isr_handler);
|
|
} else if (module->hw == SPI1) {
|
|
_spi_instances[1] = module;
|
|
system_register_isr(RAM_ISR_TABLE_SPIRX1_INDEX, (uint32_t)spi_rx1_isr_handler);
|
|
system_register_isr(RAM_ISR_TABLE_SPITX1_INDEX, (uint32_t)spi_tx1_isr_handler);
|
|
}
|
|
#endif
|
|
|
|
//Program the pinmux.
|
|
struct gpio_config config_gpio;
|
|
gpio_get_config_defaults(&config_gpio);
|
|
|
|
/* Set the pinmux for this spi module. */
|
|
for(idx = 0; idx < 4; idx++) {
|
|
if (config->pin_number_pad[idx] != PINMUX_UNUSED) {
|
|
if (config->mode == SPI_MODE_MASTER) {
|
|
config_gpio.direction = GPIO_PIN_DIR_OUTPUT;
|
|
} else if (config->mode == SPI_MODE_SLAVE) {
|
|
config_gpio.direction = GPIO_PIN_DIR_INPUT;
|
|
}
|
|
gpio_pin_set_config(config->pin_number_pad[idx], &config_gpio);
|
|
gpio_pinmux_cofiguration(config->pin_number_pad[idx], \
|
|
(uint16_t)(config->pinmux_sel_pad[idx]));
|
|
}
|
|
}
|
|
|
|
/* Set up the input clock for the module */
|
|
spi_module->CLOCK_SOURCE_SELECT.reg = config->clock_source;
|
|
|
|
# if CONF_SPI_MASTER_ENABLE == true
|
|
if (config->mode == SPI_MODE_MASTER) {
|
|
/* Set the mode in SPI master mode */
|
|
spi_module->SPI_MASTER_MODE.reg = SPI_MODE_MASTER;
|
|
}
|
|
# endif
|
|
|
|
# if CONF_SPI_SLAVE_ENABLE == true
|
|
if (config->mode == SPI_MODE_SLAVE) {
|
|
/* Set the mode in SPI slave mode */
|
|
spi_module->SPI_MASTER_MODE.reg = SPI_MODE_SLAVE;
|
|
}
|
|
# endif
|
|
|
|
#if SPI_CALLBACK_MODE == true
|
|
/* Temporary variables */
|
|
uint8_t i;
|
|
|
|
/* Initialize parameters */
|
|
for (i = 0; i < SPI_CALLBACK_N; i++) {
|
|
module->callback[i] = NULL;
|
|
}
|
|
module->tx_buffer_ptr = NULL;
|
|
module->rx_buffer_ptr = NULL;
|
|
module->remaining_tx_buffer_length = 0x0000;
|
|
module->remaining_rx_buffer_length = 0x0000;
|
|
module->registered_callback = 0x00;
|
|
module->enabled_callback = 0x00;
|
|
module->status = STATUS_OK;
|
|
module->dir = SPI_DIRECTION_IDLE;
|
|
module->locked = 0;
|
|
#endif
|
|
|
|
/* Write configuration to module and return status code */
|
|
return _spi_set_config(module, config);
|
|
}
|
|
|
|
/**
|
|
* \name Enable/Disable
|
|
* @{
|
|
*/
|
|
|
|
/**
|
|
* \brief Enables the SPI module
|
|
*
|
|
* This function will enable the SPI module.
|
|
*
|
|
* \param[in,out] module Pointer to the software instance struct
|
|
*/
|
|
void spi_enable(struct spi_module *const module)
|
|
{
|
|
Spi *const spi_module = (module->hw);
|
|
|
|
#if SPI_CALLBACK_MODE == true
|
|
if(spi_module == SPI0) {
|
|
NVIC_EnableIRQ(SPI0_RX_IRQn);
|
|
NVIC_EnableIRQ(SPI0_TX_IRQn);
|
|
} else if(spi_module == SPI1) {
|
|
NVIC_EnableIRQ(SPI1_RX_IRQn);
|
|
NVIC_EnableIRQ(SPI1_TX_IRQn);
|
|
}
|
|
#endif
|
|
|
|
/* Enable SPI */
|
|
spi_module->SPI_MODULE_ENABLE.reg = SPI_MODULE_ENABLE_ENABLE;
|
|
}
|
|
|
|
/**
|
|
* \brief Disables the SPI module
|
|
*
|
|
* This function will disable the SPI module.
|
|
*
|
|
* \param[in,out] module Pointer to the software instance struct
|
|
*/
|
|
void spi_disable(struct spi_module *const module)
|
|
{
|
|
Spi *const spi_module = (module->hw);
|
|
|
|
# if SPI_CALLBACK_MODE == true
|
|
if(spi_module == SPI0) {
|
|
NVIC_DisableIRQ(SPI0_RX_IRQn);
|
|
NVIC_DisableIRQ(SPI0_TX_IRQn);
|
|
} else if(spi_module == SPI1) {
|
|
NVIC_DisableIRQ(SPI1_RX_IRQn);
|
|
NVIC_DisableIRQ(SPI1_TX_IRQn);
|
|
}
|
|
# endif
|
|
|
|
/* Disable SPI */
|
|
spi_module->SPI_MODULE_ENABLE.reg = (0x0ul << SPI_MODULE_ENABLE_ENABLE_Pos);
|
|
_spi_clock_disable(module);
|
|
}
|
|
|
|
/**
|
|
* \brief Attempt to get lock on driver instance
|
|
*
|
|
* This function checks the instance's lock, which indicates whether or not it
|
|
* is currently in use, and sets the lock if it was not already set.
|
|
*
|
|
* The purpose of this is to enable exclusive access to driver instances, so
|
|
* that, e.g., transactions by different services will not interfere with each
|
|
* other.
|
|
*
|
|
* \param[in,out] module Pointer to the driver instance to lock
|
|
*
|
|
* \retval STATUS_OK If the module was locked
|
|
* \retval STATUS_BUSY If the module was already locked
|
|
*/
|
|
enum status_code spi_lock(struct spi_module *const module)
|
|
{
|
|
enum status_code status;
|
|
|
|
if (module->locked) {
|
|
status = STATUS_BUSY;
|
|
} else {
|
|
module->locked = true;
|
|
status = STATUS_OK;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
/**
|
|
* \brief Unlock driver instance
|
|
*
|
|
* This function clears the instance lock, indicating that it is available for
|
|
* use.
|
|
*
|
|
* \param[in,out] module Pointer to the driver instance to lock.
|
|
*
|
|
* \retval STATUS_OK If the module was locked
|
|
* \retval STATUS_BUSY If the module was already locked
|
|
*/
|
|
void spi_unlock(struct spi_module *const module)
|
|
{
|
|
module->locked = false;
|
|
}
|
|
|
|
/**
|
|
* \brief Transfers a single SPI character
|
|
*
|
|
* This function will send a single SPI character via SPI and ignore any data
|
|
* shifted in by the connected device. To both send and receive data, use the
|
|
* \ref spi_transceive_wait function or use the \ref spi_read function after
|
|
* writing a character.
|
|
*
|
|
* Note that this function does not handle the SS (Slave Select)
|
|
* pin(s) in master mode; this must be handled from the user application.
|
|
*
|
|
* \note In slave mode, the data will not be transferred before a master
|
|
* initiates a transaction.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
* \param[in] tx_data Data to transmit
|
|
*
|
|
* \return Status of the procedure
|
|
* \retval STATUS_OK If the data was written
|
|
* \retval STATUS_BUSY If the last write was not completed
|
|
*/
|
|
enum status_code spi_write(struct spi_module *module, uint8_t tx_data)
|
|
{
|
|
/* Sanity check arguments */
|
|
Assert(module);
|
|
Assert(module->hw);
|
|
|
|
Spi *const spi_module = (module->hw);
|
|
|
|
/* Check if the data register has been copied to the shift register */
|
|
if (!_spi_is_ready_to_write(spi_module)) {
|
|
/* Data register has not been copied to the shift register, return */
|
|
return STATUS_BUSY;
|
|
}
|
|
|
|
/* Write the character to the DATA register */
|
|
spi_module->TRANSMIT_DATA.reg = tx_data & SPI_TRANSMIT_DATA_MASK;
|
|
|
|
return STATUS_OK;
|
|
}
|
|
|
|
/**
|
|
* \brief Reads last received SPI character
|
|
*
|
|
* This function will return the last SPI character shifted into the receive
|
|
* register by the \ref spi_write function
|
|
*
|
|
* \note Receiver must be enabled in the configuration
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
* \param[out] rx_data Pointer to store the received data
|
|
*
|
|
* \returns Status of the read operation.
|
|
* \retval STATUS_OK If data was read
|
|
* \retval STATUS_ERR_IO If no data is available
|
|
* \retval STATUS_ERR_OVERFLOW If the data is overflown
|
|
*/
|
|
enum status_code spi_read(
|
|
struct spi_module *const module,
|
|
uint8_t *rx_data)
|
|
{
|
|
/* Sanity check arguments */
|
|
Assert(module);
|
|
Assert(module->hw);
|
|
|
|
Spi *const spi_module = (module->hw);
|
|
|
|
/* Check if data is ready to be read */
|
|
if (!_spi_is_ready_to_read(spi_module)) {
|
|
/* No data has been received, return */
|
|
return STATUS_ERR_IO;
|
|
}
|
|
|
|
/* Return value */
|
|
enum status_code retval = STATUS_OK;
|
|
|
|
/* Check if data is overflown */
|
|
if (spi_module->RECEIVE_STATUS.bit.FIFO_OVERRUN) {
|
|
retval = STATUS_ERR_OVERFLOW;
|
|
}
|
|
|
|
/* Read the character from the DATA register */
|
|
*rx_data = ((uint8_t)spi_module->RECEIVE_DATA.reg & SPI_RECEIVE_DATA_MASK);
|
|
|
|
return retval;
|
|
}
|
|
|
|
/**
|
|
* \brief Transceive requested amount of data to and from the SPI.
|
|
*
|
|
* This function will return after sending and receiving requested amount of data
|
|
*
|
|
* \note Receiver must be enabled in the configuration
|
|
* \note The \ref spi_select_slave function should be called before calling
|
|
* this function.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
* \param[in] tx_data Pointer containing the data to be transmitted
|
|
* \param[in] length Length of data to be read
|
|
* \param[out] rx_data Pointer to store the received data
|
|
*
|
|
* \returns Status of the read operation.
|
|
* \retval STATUS_OK If data was read
|
|
* \retval STATUS_ERR_IO If no data is available
|
|
* \retval STATUS_ERR_OVERFLOW If the data is overflown
|
|
*/
|
|
enum status_code spi_transceive_buffer_wait(
|
|
struct spi_module *const module,
|
|
uint8_t *tx_data,
|
|
uint8_t *rx_data,
|
|
uint16_t length)
|
|
{
|
|
Spi *spi_module = module->hw;
|
|
uint8_t dummy = 0;
|
|
uint8_t skip_mosi = 0;
|
|
uint8_t skip_miso = 0;
|
|
uint8_t status;
|
|
uint16_t transfer_len = 0;
|
|
|
|
if(spi_module == 0) {
|
|
return STATUS_ERR_NOT_INITIALIZED;
|
|
}
|
|
if(!tx_data) {
|
|
tx_data = &dummy;
|
|
*tx_data = module->tx_dummy_byte;
|
|
skip_mosi = 1;
|
|
} else if(!rx_data) {
|
|
rx_data = &dummy;
|
|
skip_miso = 1;
|
|
} else if(length == 0) {
|
|
return STATUS_ERR_INVALID_ARG;
|
|
}
|
|
|
|
/* Check for Idle */
|
|
do {
|
|
status = _spi_is_active(spi_module);
|
|
}while(status);
|
|
|
|
/* Clear all status registers */
|
|
spi_module->RECEIVE_STATUS.reg;
|
|
spi_module->TRANSMIT_STATUS.reg;
|
|
|
|
/* Start transfer */
|
|
while(transfer_len < length) {
|
|
/* Write data to MOSI */
|
|
while(!_spi_is_ready_to_write(spi_module));
|
|
spi_module->TRANSMIT_DATA.reg = *tx_data;
|
|
/* Read data shifted from MISO */
|
|
while(!_spi_is_ready_to_read(spi_module));
|
|
*rx_data = spi_module->RECEIVE_DATA.reg;
|
|
transfer_len++;
|
|
if (!skip_mosi) {
|
|
tx_data++;
|
|
}
|
|
if (!skip_miso) {
|
|
rx_data++;
|
|
}
|
|
}
|
|
/* check TXFIFO is empty */
|
|
do {
|
|
status = _spi_is_write_complete(spi_module);
|
|
}while(!status);
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
/**
|
|
* \brief Transceive single byte of data to and from the SPI.
|
|
*
|
|
* This function will return after single byte of data transceived.
|
|
*
|
|
* \note Receiver must be enabled in the configuration
|
|
* \note The \ref spi_select_slave function should be called before calling
|
|
* this function.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
* \param[in] tx_data Pointer containing the data to be transmitted
|
|
* \param[out] rx_data Pointer to store the received data
|
|
*
|
|
* \returns Status of the read operation.
|
|
* \retval STATUS_OK If data was read
|
|
* \retval STATUS_ERR_IO If no data is available
|
|
* \retval STATUS_ERR_OVERFLOW If the data is overflown
|
|
*/
|
|
enum status_code spi_transceive_wait(
|
|
struct spi_module *const module,
|
|
uint8_t *tx_data,
|
|
uint8_t *rx_data)
|
|
{
|
|
return spi_transceive_buffer_wait(module, tx_data, rx_data, 1);
|
|
}
|
|
|
|
/**
|
|
* \brief Reads requested amount of data from the SPI.
|
|
*
|
|
* This function will return after reading requested amount of data
|
|
*
|
|
* \note Receiver must be enabled in the configuration
|
|
* \note The \ref spi_select_slave function should be called before calling
|
|
* this function.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
* \param[in] length Length of data to be read
|
|
* \param[in] dummy Dummy byte to be sent on bus when reading data
|
|
* \param[out] rx_data Pointer to store the received data
|
|
*
|
|
* \returns Status of the read operation.
|
|
* \retval STATUS_OK If data was read
|
|
* \retval STATUS_ERR_IO If no data is available
|
|
* \retval STATUS_ERR_OVERFLOW If the data is overflown
|
|
*/
|
|
enum status_code spi_read_buffer_wait(
|
|
struct spi_module *const module,
|
|
uint8_t *rx_data,
|
|
uint16_t length,
|
|
uint8_t dummy)
|
|
{
|
|
module->tx_dummy_byte = dummy;
|
|
return spi_transceive_buffer_wait(module, NULL, rx_data, length);
|
|
}
|
|
|
|
/**
|
|
* \brief Writes requested amount of data to the SPI.
|
|
*
|
|
* This function will return after writing requested amount of data
|
|
*
|
|
* \note The \ref spi_select_slave function should be called before calling
|
|
* this function.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
* \param[in] length length of data to be read
|
|
* \param[out] tx_data Pointer to buffer to be transmitted
|
|
*
|
|
* \returns Status of the read operation.
|
|
* \retval STATUS_OK If data was read
|
|
* \retval STATUS_ERR_IO If no data is available
|
|
* \retval STATUS_ERR_OVERFLOW If the data is overflown
|
|
*/
|
|
enum status_code spi_write_buffer_wait(
|
|
struct spi_module *const module,
|
|
uint8_t *tx_data,
|
|
uint16_t length)
|
|
{
|
|
return spi_transceive_buffer_wait(module, tx_data, NULL, length);
|
|
}
|
|
|
|
/**
|
|
* \brief Asserting/Deasserting the slave select for the corresponding slave.
|
|
*
|
|
* This function will assert or deassert the SS of the requested slave device.
|
|
*
|
|
* \param[in] module Pointer to the software instance struct
|
|
* \param[in] slave Pointer containing slave instance
|
|
* \param[in] select Bool to select the salve or deselect
|
|
*
|
|
* \returns Status of the slave select operation.
|
|
*
|
|
* \retval STATUS_OK If SS pin is a valid one and selected/deselected
|
|
* \retval STATUS_ERR_INVALID_ARG Invalid SS pin
|
|
*/
|
|
enum status_code spi_select_slave(
|
|
struct spi_module *const module,
|
|
struct spi_slave_inst *const slave,
|
|
bool select)
|
|
{
|
|
uint8_t gpio_num = slave->ss_pin;
|
|
if(select) {
|
|
/* ASSERT Slave select pin */
|
|
gpio_pin_set_output_level(gpio_num, false);
|
|
} else {
|
|
/* DEASSERT Slave select pin */
|
|
gpio_pin_set_output_level(gpio_num, true);
|
|
}
|
|
|
|
return STATUS_OK;
|
|
}
|