mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
1420 lines
45 KiB
C
1420 lines
45 KiB
C
/*
|
|
* SPDX-FileCopyrightText: 2015-2023 Espressif Systems (Shanghai) CO LTD
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/*******************************************************************************
|
|
* NOTICE
|
|
* The hal is not public api, don't use in application code.
|
|
* See readme.md in hal/include/hal/readme.md
|
|
******************************************************************************/
|
|
|
|
// The LL layer for ESP32-S2 SPI register operations
|
|
|
|
#pragma once
|
|
|
|
#include <stdlib.h> //for abs()
|
|
#include <string.h>
|
|
#include "esp_types.h"
|
|
#include "esp_attr.h"
|
|
#include "soc/spi_periph.h"
|
|
#include "soc/spi_struct.h"
|
|
#include "soc/lldesc.h"
|
|
#include "hal/assert.h"
|
|
#include "hal/misc.h"
|
|
#include "hal/spi_types.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/// Registers to reset during initialization. Don't use in app.
|
|
#define SPI_LL_DMA_FIFO_RST_MASK (SPI_AHBM_RST | SPI_AHBM_FIFO_RST)
|
|
/// Interrupt not used. Don't use in app.
|
|
#define SPI_LL_UNUSED_INT_MASK (SPI_INT_TRANS_DONE_EN | SPI_INT_WR_DMA_DONE_EN | SPI_INT_RD_DMA_DONE_EN | SPI_INT_WR_BUF_DONE_EN | SPI_INT_RD_BUF_DONE_EN)
|
|
/// These 2 masks together will set SPI transaction to one line mode
|
|
#define SPI_LL_ONE_LINE_CTRL_MASK (SPI_FREAD_OCT | SPI_FREAD_QUAD | SPI_FREAD_DUAL | SPI_FCMD_OCT | \
|
|
SPI_FCMD_QUAD | SPI_FCMD_DUAL | SPI_FADDR_OCT | SPI_FADDR_QUAD | SPI_FADDR_DUAL)
|
|
#define SPI_LL_ONE_LINE_USER_MASK (SPI_FWRITE_OCT | SPI_FWRITE_QUAD | SPI_FWRITE_DUAL)
|
|
/// Swap the bit order to its correct place to send
|
|
#define HAL_SPI_SWAP_DATA_TX(data, len) HAL_SWAP32((uint32_t)(data) << (32 - len))
|
|
#define SPI_LL_GET_HW(ID) ((ID)==0? ({abort();NULL;}):((ID)==1? &GPSPI2 : &GPSPI3))
|
|
|
|
#define SPI_LL_DMA_MAX_BIT_LEN (1 << 23) //reg len: 23 bits
|
|
#define SPI_LL_CPU_MAX_BIT_LEN (18 * 32) //Fifo len: 18 words
|
|
#define SPI_LL_MOSI_FREE_LEVEL 1 //Default level after bus initialized
|
|
|
|
/**
|
|
* The data structure holding calculated clock configuration. Since the
|
|
* calculation needs long time, it should be calculated during initialization and
|
|
* stored somewhere to be quickly used.
|
|
*/
|
|
typedef uint32_t spi_ll_clock_val_t;
|
|
|
|
//On ESP32-S2 and earlier chips, DMA registers are part of SPI registers. So set the registers of SPI peripheral to control DMA.
|
|
typedef spi_dev_t spi_dma_dev_t;
|
|
|
|
/// Type definition of all supported interrupts
|
|
typedef enum {
|
|
SPI_LL_INTR_TRANS_DONE = BIT(0), ///< A transaction has done
|
|
SPI_LL_INTR_IN_SUC_EOF = BIT(1), ///< DMA in_suc_eof triggered
|
|
SPI_LL_INTR_OUT_EOF = BIT(2), ///< DMA out_eof triggered
|
|
SPI_LL_INTR_OUT_TOTAL_EOF = BIT(3), ///< DMA out_total_eof triggered
|
|
SPI_LL_INTR_IN_FULL = BIT(4), ///< DMA in_full error happened
|
|
SPI_LL_INTR_OUT_EMPTY = BIT(5), ///< DMA out_empty error happened
|
|
SPI_LL_INTR_RDBUF = BIT(6), ///< Has received RDBUF command. Only available in slave HD.
|
|
SPI_LL_INTR_WRBUF = BIT(7), ///< Has received WRBUF command. Only available in slave HD.
|
|
SPI_LL_INTR_RDDMA = BIT(8), ///< Has received RDDMA command. Only available in slave HD.
|
|
SPI_LL_INTR_WRDMA = BIT(9), ///< Has received WRDMA command. Only available in slave HD.
|
|
SPI_LL_INTR_CMD7 = BIT(10), ///< Has received CMD7 command. Only available in slave HD.
|
|
SPI_LL_INTR_CMD8 = BIT(11), ///< Has received CMD8 command. Only available in slave HD.
|
|
SPI_LL_INTR_CMD9 = BIT(12), ///< Has received CMD9 command. Only available in slave HD.
|
|
SPI_LL_INTR_CMDA = BIT(13), ///< Has received CMDA command. Only available in slave HD.
|
|
SPI_LL_INTR_SEG_DONE = BIT(14),
|
|
} spi_ll_intr_t;
|
|
FLAG_ATTR(spi_ll_intr_t)
|
|
|
|
///< Flags for conditions under which the transaction length should be recorded
|
|
typedef enum {
|
|
SPI_LL_TRANS_LEN_COND_WRBUF = BIT(0), ///< WRBUF length will be recorded
|
|
SPI_LL_TRANS_LEN_COND_RDBUF = BIT(1), ///< RDBUF length will be recorded
|
|
SPI_LL_TRANS_LEN_COND_WRDMA = BIT(2), ///< WRDMA length will be recorded
|
|
SPI_LL_TRANS_LEN_COND_RDDMA = BIT(3), ///< RDDMA length will be recorded
|
|
} spi_ll_trans_len_cond_t;
|
|
FLAG_ATTR(spi_ll_trans_len_cond_t)
|
|
|
|
// SPI base command in esp32s2
|
|
typedef enum {
|
|
/* Slave HD Only */
|
|
SPI_LL_BASE_CMD_HD_WRBUF = 0x01,
|
|
SPI_LL_BASE_CMD_HD_RDBUF = 0x02,
|
|
SPI_LL_BASE_CMD_HD_WRDMA = 0x03,
|
|
SPI_LL_BASE_CMD_HD_RDDMA = 0x04,
|
|
SPI_LL_BASE_CMD_HD_SEG_END = 0x05,
|
|
SPI_LL_BASE_CMD_HD_EN_QPI = 0x06,
|
|
SPI_LL_BASE_CMD_HD_WR_END = 0x07,
|
|
SPI_LL_BASE_CMD_HD_INT0 = 0x08,
|
|
SPI_LL_BASE_CMD_HD_INT1 = 0x09,
|
|
SPI_LL_BASE_CMD_HD_INT2 = 0x0A,
|
|
} spi_ll_base_command_t;
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* Control
|
|
*----------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* Select SPI peripheral clock source (master).
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param clk_source clock source to select, see valid sources in type `spi_clock_source_t`
|
|
*/
|
|
static inline void spi_ll_set_clk_source(spi_dev_t *hw, spi_clock_source_t clk_source)
|
|
{
|
|
//empty, keep this for compatibility
|
|
}
|
|
|
|
/**
|
|
* Initialize SPI peripheral (master).
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_master_init(spi_dev_t *hw)
|
|
{
|
|
//Reset timing
|
|
hw->ctrl2.val = 0;
|
|
|
|
//use all 64 bytes of the buffer
|
|
hw->user.usr_miso_highpart = 0;
|
|
hw->user.usr_mosi_highpart = 0;
|
|
|
|
//Disable unneeded ints
|
|
hw->slave.val &= ~SPI_LL_UNUSED_INT_MASK;
|
|
|
|
//disable a feature may cause transaction to be too long
|
|
hw->user.usr_prep_hold = 0;
|
|
}
|
|
|
|
/**
|
|
* Initialize SPI peripheral (slave).
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_slave_init(spi_dev_t *hw)
|
|
{
|
|
//Configure slave
|
|
hw->clock.val = 0;
|
|
hw->user.val = 0;
|
|
hw->ctrl.val = 0;
|
|
hw->user.doutdin = 1; //we only support full duplex
|
|
hw->user.sio = 0;
|
|
hw->slave.slave_mode = 1;
|
|
hw->slave.soft_reset = 1;
|
|
hw->slave.soft_reset = 0;
|
|
//use all 64 bytes of the buffer
|
|
hw->user.usr_miso_highpart = 0;
|
|
hw->user.usr_mosi_highpart = 0;
|
|
|
|
// Configure DMA In-Link to not be terminated when transaction bit counter exceeds
|
|
hw->dma_conf.rx_eof_en = 0;
|
|
|
|
//Disable unneeded ints
|
|
hw->slave.val &= ~SPI_LL_UNUSED_INT_MASK;
|
|
}
|
|
|
|
static inline void spi_ll_slave_hd_init(spi_dev_t *hw)
|
|
{
|
|
hw->clock.val = 0;
|
|
hw->user.val = 0;
|
|
hw->ctrl.val = 0;
|
|
hw->user.sio = 0;
|
|
|
|
hw->slave.soft_reset = 1;
|
|
hw->slave.soft_reset = 0;
|
|
|
|
hw->user.doutdin = 0; //we only support half duplex
|
|
hw->slave.slave_mode = 1;
|
|
}
|
|
|
|
/**
|
|
* Determine and unify the default level of mosi line when bus free
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_set_mosi_free_level(spi_dev_t *hw, bool level)
|
|
{
|
|
hw->ctrl.d_pol = level; //set default level for MOSI only on IDLE state
|
|
}
|
|
|
|
/**
|
|
* Apply the register configurations and wait until it's done
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_apply_config(spi_dev_t *hw)
|
|
{
|
|
// S2 don't need this option
|
|
}
|
|
|
|
/**
|
|
* Check whether user-defined transaction is done.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*
|
|
* @return true if transaction is done, otherwise false.
|
|
*/
|
|
static inline bool spi_ll_usr_is_done(spi_dev_t *hw)
|
|
{
|
|
return hw->slave.trans_done;
|
|
}
|
|
|
|
/**
|
|
* Trigger start of user-defined transaction.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_user_start(spi_dev_t *hw)
|
|
{
|
|
hw->cmd.usr = 1;
|
|
}
|
|
|
|
/**
|
|
* Get current running command bit-mask. (Preview)
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*
|
|
* @return Bitmask of running command, see ``SPI_CMD_REG``. 0 if no in-flight command.
|
|
*/
|
|
static inline uint32_t spi_ll_get_running_cmd(spi_dev_t *hw)
|
|
{
|
|
return hw->cmd.val;
|
|
}
|
|
|
|
/**
|
|
* Reset SPI CPU TX FIFO
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_cpu_tx_fifo_reset(spi_dev_t *hw)
|
|
{
|
|
//This is not used in esp32s2
|
|
}
|
|
|
|
/**
|
|
* Reset SPI CPU RX FIFO
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_cpu_rx_fifo_reset(spi_dev_t *hw)
|
|
{
|
|
//This is not used in esp32s2
|
|
}
|
|
|
|
/**
|
|
* Reset SPI DMA TX FIFO
|
|
*
|
|
* On ESP32S2, this function is not seperated
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_dma_tx_fifo_reset(spi_dev_t *hw)
|
|
{
|
|
hw->dma_conf.val |= SPI_LL_DMA_FIFO_RST_MASK;
|
|
hw->dma_conf.val &= ~SPI_LL_DMA_FIFO_RST_MASK;
|
|
}
|
|
|
|
/**
|
|
* Reset SPI DMA RX FIFO
|
|
*
|
|
* On ESP32S2, this function is not seperated
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_dma_rx_fifo_reset(spi_dev_t *hw)
|
|
{
|
|
hw->dma_conf.val |= SPI_LL_DMA_FIFO_RST_MASK;
|
|
hw->dma_conf.val &= ~SPI_LL_DMA_FIFO_RST_MASK;
|
|
}
|
|
|
|
/**
|
|
* Clear in fifo full error
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_infifo_full_clr(spi_dev_t *hw)
|
|
{
|
|
hw->dma_conf.infifo_full_clr = 1;
|
|
hw->dma_conf.infifo_full_clr = 0;
|
|
}
|
|
|
|
/**
|
|
* Clear out fifo empty error
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_outfifo_empty_clr(spi_dev_t *hw)
|
|
{
|
|
hw->dma_conf.outfifo_empty_clr = 1;
|
|
hw->dma_conf.outfifo_empty_clr = 0;
|
|
}
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* SPI configuration for DMA
|
|
*----------------------------------------------------------------------------*/
|
|
/**
|
|
* Enable/Disable RX DMA (Peripherals->DMA->RAM)
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param enable 1: enable; 2: disable
|
|
*/
|
|
static inline void spi_ll_dma_rx_enable(spi_dev_t *hw, bool enable)
|
|
{
|
|
//This is not used in esp32s2
|
|
}
|
|
|
|
/**
|
|
* Enable/Disable TX DMA (RAM->DMA->Peripherals)
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param enable 1: enable; 2: disable
|
|
*/
|
|
static inline void spi_ll_dma_tx_enable(spi_dev_t *hw, bool enable)
|
|
{
|
|
//This is not used in esp32s2
|
|
}
|
|
|
|
/**
|
|
* Configuration of RX DMA EOF interrupt generation way
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param enable 1: spi_dma_inlink_eof is set when the number of dma pushed data bytes is equal to the value of spi_slv/mst_dma_rd_bytelen[19:0] in spi dma transition. 0: spi_dma_inlink_eof is set by spi_trans_done in non-seg-trans or spi_dma_seg_trans_done in seg-trans.
|
|
*/
|
|
static inline void spi_ll_dma_set_rx_eof_generation(spi_dev_t *hw, bool enable)
|
|
{
|
|
hw->dma_conf.rx_eof_en = enable;
|
|
}
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* Buffer
|
|
*----------------------------------------------------------------------------*/
|
|
/**
|
|
* Write to SPI buffer.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param buffer_to_send Data address to copy to the buffer.
|
|
* @param bitlen Length to copy, in bits.
|
|
*/
|
|
static inline void spi_ll_write_buffer(spi_dev_t *hw, const uint8_t *buffer_to_send, size_t bitlen)
|
|
{
|
|
for (size_t x = 0; x < bitlen; x += 32) {
|
|
//Use memcpy to get around alignment issues for txdata
|
|
uint32_t word;
|
|
memcpy(&word, &buffer_to_send[x / 8], 4);
|
|
hw->data_buf[(x / 32)] = word;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Read from SPI buffer.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param buffer_to_rcv Address to copy buffer data to.
|
|
* @param bitlen Length to copy, in bits.
|
|
*/
|
|
static inline void spi_ll_read_buffer(spi_dev_t *hw, uint8_t *buffer_to_rcv, size_t bitlen)
|
|
{
|
|
for (size_t x = 0; x < bitlen; x += 32) {
|
|
//Do a memcpy to get around possible alignment issues in rx_buffer
|
|
uint32_t word = hw->data_buf[x / 32];
|
|
int len = bitlen - x;
|
|
if (len > 32) {
|
|
len = 32;
|
|
}
|
|
memcpy(&buffer_to_rcv[x / 8], &word, (len + 7) / 8);
|
|
}
|
|
}
|
|
|
|
static inline void spi_ll_read_buffer_byte(spi_dev_t *hw, int byte_addr, uint8_t *out_data, int len)
|
|
{
|
|
while (len > 0) {
|
|
uint32_t word = hw->data_buf[byte_addr / 4];
|
|
int offset = byte_addr % 4;
|
|
|
|
int copy_len = 4 - offset;
|
|
if (copy_len > len) {
|
|
copy_len = len;
|
|
}
|
|
|
|
memcpy(out_data, ((uint8_t *)&word) + offset, copy_len);
|
|
byte_addr += copy_len;
|
|
out_data += copy_len;
|
|
len -= copy_len;
|
|
}
|
|
}
|
|
|
|
static inline void spi_ll_write_buffer_byte(spi_dev_t *hw, int byte_addr, uint8_t *data, int len)
|
|
{
|
|
HAL_ASSERT(byte_addr + len <= 72);
|
|
HAL_ASSERT(len > 0);
|
|
HAL_ASSERT(byte_addr >= 0);
|
|
|
|
while (len > 0) {
|
|
uint32_t word;
|
|
int offset = byte_addr % 4;
|
|
|
|
int copy_len = 4 - offset;
|
|
if (copy_len > len) {
|
|
copy_len = len;
|
|
}
|
|
|
|
//read-modify-write
|
|
if (copy_len != 4) {
|
|
word = hw->data_buf[byte_addr / 4];
|
|
}
|
|
|
|
memcpy(((uint8_t *)&word) + offset, data, copy_len);
|
|
hw->data_buf[byte_addr / 4] = word;
|
|
data += copy_len;
|
|
byte_addr += copy_len;
|
|
len -= copy_len;
|
|
}
|
|
}
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* Configs: mode
|
|
*----------------------------------------------------------------------------*/
|
|
/**
|
|
* Enable/disable the postive-cs feature.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param cs One of the CS (0-2) to enable/disable the feature.
|
|
* @param pos_cs true to enable the feature, otherwise disable (default).
|
|
*/
|
|
static inline void spi_ll_master_set_pos_cs(spi_dev_t *hw, int cs, uint32_t pos_cs)
|
|
{
|
|
if (pos_cs) {
|
|
hw->misc.master_cs_pol |= (1 << cs);
|
|
} else {
|
|
hw->misc.master_cs_pol &= ~(1 << cs);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Enable/disable the LSBFIRST feature for TX data.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param lsbfirst true if LSB of TX data to be sent first, otherwise MSB is sent first (default).
|
|
*/
|
|
static inline void spi_ll_set_tx_lsbfirst(spi_dev_t *hw, bool lsbfirst)
|
|
{
|
|
hw->ctrl.wr_bit_order = lsbfirst;
|
|
}
|
|
|
|
/**
|
|
* Enable/disable the LSBFIRST feature for RX data.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param lsbfirst true if first bit received as LSB, otherwise as MSB (default).
|
|
*/
|
|
static inline void spi_ll_set_rx_lsbfirst(spi_dev_t *hw, bool lsbfirst)
|
|
{
|
|
hw->ctrl.rd_bit_order = lsbfirst;
|
|
}
|
|
|
|
/**
|
|
* Set SPI mode for the peripheral as master.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param mode SPI mode to work at, 0-3.
|
|
*/
|
|
static inline void spi_ll_master_set_mode(spi_dev_t *hw, uint8_t mode)
|
|
{
|
|
//Configure polarity
|
|
if (mode == 0) {
|
|
hw->misc.ck_idle_edge = 0;
|
|
hw->user.ck_out_edge = 0;
|
|
} else if (mode == 1) {
|
|
hw->misc.ck_idle_edge = 0;
|
|
hw->user.ck_out_edge = 1;
|
|
} else if (mode == 2) {
|
|
hw->misc.ck_idle_edge = 1;
|
|
hw->user.ck_out_edge = 1;
|
|
} else if (mode == 3) {
|
|
hw->misc.ck_idle_edge = 1;
|
|
hw->user.ck_out_edge = 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Set SPI mode for the peripheral as slave.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param mode SPI mode to work at, 0-3.
|
|
*/
|
|
static inline void spi_ll_slave_set_mode(spi_dev_t *hw, const int mode, bool dma_used)
|
|
{
|
|
if (mode == 0) {
|
|
hw->misc.ck_idle_edge = 0;
|
|
hw->user.rsck_i_edge = 0;
|
|
hw->user.tsck_i_edge = 0;
|
|
hw->ctrl1.clk_mode_13 = 0;
|
|
} else if (mode == 1) {
|
|
hw->misc.ck_idle_edge = 0;
|
|
hw->user.rsck_i_edge = 1;
|
|
hw->user.tsck_i_edge = 1;
|
|
hw->ctrl1.clk_mode_13 = 1;
|
|
} else if (mode == 2) {
|
|
hw->misc.ck_idle_edge = 1;
|
|
hw->user.rsck_i_edge = 1;
|
|
hw->user.tsck_i_edge = 1;
|
|
hw->ctrl1.clk_mode_13 = 0;
|
|
} else if (mode == 3) {
|
|
hw->misc.ck_idle_edge = 1;
|
|
hw->user.rsck_i_edge = 0;
|
|
hw->user.tsck_i_edge = 0;
|
|
hw->ctrl1.clk_mode_13 = 1;
|
|
}
|
|
hw->ctrl1.rsck_data_out = 0;
|
|
}
|
|
|
|
/**
|
|
* Set SPI to work in full duplex or half duplex mode.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param half_duplex true to work in half duplex mode, otherwise in full duplex mode.
|
|
*/
|
|
static inline void spi_ll_set_half_duplex(spi_dev_t *hw, bool half_duplex)
|
|
{
|
|
hw->user.doutdin = !half_duplex;
|
|
}
|
|
|
|
/**
|
|
* Set SPI to work in SIO mode or not.
|
|
*
|
|
* SIO is a mode which MOSI and MISO share a line. The device MUST work in half-duplexmode.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param sio_mode true to work in SIO mode, otherwise false.
|
|
*/
|
|
static inline void spi_ll_set_sio_mode(spi_dev_t *hw, int sio_mode)
|
|
{
|
|
hw->user.sio = sio_mode;
|
|
}
|
|
|
|
/**
|
|
* Configure the SPI transaction line mode for the master to use.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param line_mode SPI transaction line mode to use, see ``spi_line_mode_t``.
|
|
*/
|
|
static inline void spi_ll_master_set_line_mode(spi_dev_t *hw, spi_line_mode_t line_mode)
|
|
{
|
|
hw->ctrl.val &= ~SPI_LL_ONE_LINE_CTRL_MASK;
|
|
hw->user.val &= ~SPI_LL_ONE_LINE_USER_MASK;
|
|
hw->ctrl.fcmd_dual = (line_mode.cmd_lines == 2);
|
|
hw->ctrl.fcmd_quad = (line_mode.cmd_lines == 4);
|
|
hw->ctrl.fcmd_oct = (line_mode.cmd_lines == 8);
|
|
hw->ctrl.faddr_dual = (line_mode.addr_lines == 2);
|
|
hw->ctrl.faddr_quad = (line_mode.addr_lines == 4);
|
|
hw->ctrl.faddr_oct = (line_mode.addr_lines == 8);
|
|
hw->ctrl.fread_dual = (line_mode.data_lines == 2);
|
|
hw->user.fwrite_dual = (line_mode.data_lines == 2);
|
|
hw->ctrl.fread_quad = (line_mode.data_lines == 4);
|
|
hw->user.fwrite_quad = (line_mode.data_lines == 4);
|
|
hw->ctrl.fread_oct = (line_mode.data_lines == 8);
|
|
hw->user.fwrite_oct = (line_mode.data_lines == 8);
|
|
}
|
|
|
|
static inline void spi_ll_slave_set_seg_mode(spi_dev_t *hw, bool seg_trans)
|
|
{
|
|
hw->dma_conf.dma_seg_trans_en = seg_trans;
|
|
}
|
|
|
|
/**
|
|
* Select one of the CS to use in current transaction.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param cs_id The cs to use, 0-2, otherwise none of them is used.
|
|
*/
|
|
static inline void spi_ll_master_select_cs(spi_dev_t *hw, int cs_id)
|
|
{
|
|
hw->misc.cs0_dis = (cs_id == 0) ? 0 : 1;
|
|
hw->misc.cs1_dis = (cs_id == 1) ? 0 : 1;
|
|
hw->misc.cs2_dis = (cs_id == 2) ? 0 : 1;
|
|
hw->misc.cs3_dis = (cs_id == 3) ? 0 : 1;
|
|
hw->misc.cs4_dis = (cs_id == 4) ? 0 : 1;
|
|
hw->misc.cs5_dis = (cs_id == 5) ? 0 : 1;
|
|
}
|
|
|
|
/**
|
|
* Keep Chip Select activated after the current transaction.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param keep_active if 0 don't keep CS activated, else keep CS activated
|
|
*/
|
|
static inline void spi_ll_master_keep_cs(spi_dev_t *hw, int keep_active)
|
|
{
|
|
hw->misc.cs_keep_active = (keep_active != 0) ? 1 : 0;
|
|
}
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* Configs: parameters
|
|
*----------------------------------------------------------------------------*/
|
|
/**
|
|
* Set the clock for master by stored value.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param val stored clock configuration calculated before (by ``spi_ll_cal_clock``).
|
|
*/
|
|
static inline void spi_ll_master_set_clock_by_reg(spi_dev_t *hw, const spi_ll_clock_val_t *val)
|
|
{
|
|
hw->clock.val = *(uint32_t *)val;
|
|
}
|
|
|
|
/**
|
|
* Get the frequency of given dividers. Don't use in app.
|
|
*
|
|
* @param fapb APB clock of the system.
|
|
* @param pre Pre devider.
|
|
* @param n main divider.
|
|
*
|
|
* @return Frequency of given dividers.
|
|
*/
|
|
static inline int spi_ll_freq_for_pre_n(int fapb, int pre, int n)
|
|
{
|
|
return (fapb / (pre * n));
|
|
}
|
|
|
|
/**
|
|
* Calculate the nearest frequency avaliable for master.
|
|
*
|
|
* @param fapb APB clock of the system.
|
|
* @param hz Frequncy desired.
|
|
* @param duty_cycle Duty cycle desired.
|
|
* @param out_reg Output address to store the calculated clock configurations for the return frequency.
|
|
*
|
|
* @return Actual (nearest) frequency.
|
|
*/
|
|
static inline int spi_ll_master_cal_clock(int fapb, int hz, int duty_cycle, spi_ll_clock_val_t *out_reg)
|
|
{
|
|
typeof(GPSPI2.clock) reg;
|
|
int eff_clk;
|
|
|
|
//In hw, n, h and l are 1-64, pre is 1-8K. Value written to register is one lower than used value.
|
|
if (hz > ((fapb / 4) * 3)) {
|
|
//Using Fapb directly will give us the best result here.
|
|
reg.clkcnt_l = 0;
|
|
reg.clkcnt_h = 0;
|
|
reg.clkcnt_n = 0;
|
|
reg.clkdiv_pre = 0;
|
|
reg.clk_equ_sysclk = 1;
|
|
eff_clk = fapb;
|
|
} else {
|
|
//For best duty cycle resolution, we want n to be as close to 32 as possible, but
|
|
//we also need a pre/n combo that gets us as close as possible to the intended freq.
|
|
//To do this, we bruteforce n and calculate the best pre to go along with that.
|
|
//If there's a choice between pre/n combos that give the same result, use the one
|
|
//with the higher n.
|
|
int pre, n, h, l;
|
|
int bestn = -1;
|
|
int bestpre = -1;
|
|
int besterr = 0;
|
|
int errval;
|
|
for (n = 2; n <= 64; n++) { //Start at 2: we need to be able to set h/l so we have at least one high and one low pulse.
|
|
//Effectively, this does pre=round((fapb/n)/hz).
|
|
pre = ((fapb / n) + (hz / 2)) / hz;
|
|
if (pre <= 0) {
|
|
pre = 1;
|
|
}
|
|
if (pre > 8192) {
|
|
pre = 8192;
|
|
}
|
|
errval = abs(spi_ll_freq_for_pre_n(fapb, pre, n) - hz);
|
|
if (bestn == -1 || errval <= besterr) {
|
|
besterr = errval;
|
|
bestn = n;
|
|
bestpre = pre;
|
|
}
|
|
}
|
|
|
|
n = bestn;
|
|
pre = bestpre;
|
|
l = n;
|
|
//This effectively does round((duty_cycle*n)/256)
|
|
h = (duty_cycle * n + 127) / 256;
|
|
if (h <= 0) {
|
|
h = 1;
|
|
}
|
|
|
|
reg.clk_equ_sysclk = 0;
|
|
reg.clkcnt_n = n - 1;
|
|
reg.clkdiv_pre = pre - 1;
|
|
reg.clkcnt_h = h - 1;
|
|
reg.clkcnt_l = l - 1;
|
|
eff_clk = spi_ll_freq_for_pre_n(fapb, pre, n);
|
|
}
|
|
if (out_reg != NULL) {
|
|
*(uint32_t *)out_reg = reg.val;
|
|
}
|
|
return eff_clk;
|
|
}
|
|
|
|
/**
|
|
* Calculate and set clock for SPI master according to desired parameters.
|
|
*
|
|
* This takes long, suggest to calculate the configuration during
|
|
* initialization by ``spi_ll_master_cal_clock`` and store the result, then
|
|
* configure the clock by stored value when used by
|
|
* ``spi_ll_msater_set_clock_by_reg``.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param fapb APB clock of the system.
|
|
* @param hz Frequncy desired.
|
|
* @param duty_cycle Duty cycle desired.
|
|
*
|
|
* @return Actual frequency that is used.
|
|
*/
|
|
static inline int spi_ll_master_set_clock(spi_dev_t *hw, int fapb, int hz, int duty_cycle)
|
|
{
|
|
spi_ll_clock_val_t reg_val;
|
|
int freq = spi_ll_master_cal_clock(fapb, hz, duty_cycle, ®_val);
|
|
spi_ll_master_set_clock_by_reg(hw, ®_val);
|
|
return freq;
|
|
}
|
|
|
|
/**
|
|
* Set the mosi delay after the output edge to the signal. (Preview)
|
|
*
|
|
* The delay mode/num is a Espressif conception, may change in the new chips.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param delay_mode Delay mode, see TRM.
|
|
* @param delay_num APB clocks to delay.
|
|
*/
|
|
static inline void spi_ll_set_mosi_delay(spi_dev_t *hw, int delay_mode, int delay_num)
|
|
{
|
|
//TODO: this doesn't make sense
|
|
hw->dout_num.dout0_num = 0;
|
|
hw->dout_num.dout1_num = 0;
|
|
}
|
|
|
|
/**
|
|
* Set the miso delay applied to the input signal before the internal peripheral. (Preview)
|
|
*
|
|
* The delay mode/num is a Espressif conception, may change in the new chips.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param delay_mode Delay mode, see TRM.
|
|
* @param delay_num APB clocks to delay.
|
|
*/
|
|
static inline void spi_ll_set_miso_delay(spi_dev_t *hw, int delay_mode, int delay_num)
|
|
{
|
|
//TODO: this doesn't make sense
|
|
hw->din_num.din0_num = 1;
|
|
hw->din_num.din1_num = 1;
|
|
}
|
|
|
|
/**
|
|
* Set dummy clocks to output before RX phase (master), or clocks to skip
|
|
* before the data phase and after the address phase (slave).
|
|
*
|
|
* Note this phase is also used to compensate RX timing in half duplex mode.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param dummy_n Dummy cycles used. 0 to disable the dummy phase.
|
|
*/
|
|
static inline void spi_ll_set_dummy(spi_dev_t *hw, int dummy_n)
|
|
{
|
|
hw->user.usr_dummy = dummy_n ? 1 : 0;
|
|
HAL_FORCE_MODIFY_U32_REG_FIELD(hw->user1, usr_dummy_cyclelen, dummy_n - 1);
|
|
}
|
|
|
|
/**
|
|
* Set the delay of SPI clocks before the CS inactive edge after the last SPI clock.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param hold Delay of SPI clocks after the last clock, 0 to disable the hold phase.
|
|
*/
|
|
static inline void spi_ll_master_set_cs_hold(spi_dev_t *hw, int hold)
|
|
{
|
|
hw->ctrl2.cs_hold_time = hold;
|
|
hw->user.cs_hold = hold ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Set the delay of SPI clocks before the first SPI clock after the CS active edge.
|
|
*
|
|
* Note ESP32 doesn't support to use this feature when command/address phases
|
|
* are used in full duplex mode.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param setup Delay of SPI clocks after the CS active edge, 0 to disable the setup phase.
|
|
*/
|
|
static inline void spi_ll_master_set_cs_setup(spi_dev_t *hw, uint8_t setup)
|
|
{
|
|
hw->ctrl2.cs_setup_time = setup - 1;
|
|
hw->user.cs_setup = setup ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Enable/disable the segment transfer feature for the slave.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param en true to enable, false to disable.
|
|
*/
|
|
static inline void spi_ll_slave_set_seg_en(spi_dev_t *hw, bool en)
|
|
{
|
|
hw->dma_conf.dma_seg_trans_en = en;
|
|
}
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* Configs: data
|
|
*----------------------------------------------------------------------------*/
|
|
/**
|
|
* Set the input length (master).
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param bitlen input length, in bits.
|
|
*/
|
|
static inline void spi_ll_set_miso_bitlen(spi_dev_t *hw, size_t bitlen)
|
|
{
|
|
hw->miso_dlen.usr_miso_bit_len = bitlen - 1;
|
|
}
|
|
|
|
/**
|
|
* Set the output length (master).
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param bitlen output length, in bits.
|
|
*/
|
|
static inline void spi_ll_set_mosi_bitlen(spi_dev_t *hw, size_t bitlen)
|
|
{
|
|
hw->mosi_dlen.usr_mosi_bit_len = bitlen - 1;
|
|
}
|
|
|
|
/**
|
|
* Set the maximum input length (slave).
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param bitlen input length, in bits.
|
|
*/
|
|
static inline void spi_ll_slave_set_rx_bitlen(spi_dev_t *hw, size_t bitlen)
|
|
{
|
|
//This is not used in esp32s2
|
|
}
|
|
|
|
/**
|
|
* Set the maximum output length (slave).
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param bitlen output length, in bits.
|
|
*/
|
|
static inline void spi_ll_slave_set_tx_bitlen(spi_dev_t *hw, size_t bitlen)
|
|
{
|
|
//This is not used in esp32s2
|
|
}
|
|
|
|
/**
|
|
* Set the length of command phase.
|
|
*
|
|
* When in 4-bit mode, the SPI cycles of the phase will be shorter. E.g. 16-bit
|
|
* command phases takes 4 cycles in 4-bit mode.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param bitlen Length of command phase, in bits. 0 to disable the command phase.
|
|
*/
|
|
static inline void spi_ll_set_command_bitlen(spi_dev_t *hw, int bitlen)
|
|
{
|
|
hw->user2.usr_command_bitlen = bitlen - 1;
|
|
hw->user.usr_command = bitlen ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Set the length of address phase.
|
|
*
|
|
* When in 4-bit mode, the SPI cycles of the phase will be shorter. E.g. 16-bit
|
|
* address phases takes 4 cycles in 4-bit mode.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param bitlen Length of address phase, in bits. 0 to disable the address phase.
|
|
*/
|
|
static inline void spi_ll_set_addr_bitlen(spi_dev_t *hw, int bitlen)
|
|
{
|
|
hw->user1.usr_addr_bitlen = bitlen - 1;
|
|
hw->user.usr_addr = bitlen ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Set the address value in an intuitive way.
|
|
*
|
|
* The length and lsbfirst is required to shift and swap the address to the right place.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param address Address to set
|
|
* @param addrlen Length of the address phase
|
|
* @param lsbfirst whether the LSB first feature is enabled.
|
|
*/
|
|
static inline void spi_ll_set_address(spi_dev_t *hw, uint64_t addr, int addrlen, uint32_t lsbfirst)
|
|
{
|
|
if (lsbfirst) {
|
|
/* The output address start from the LSB of the highest byte, i.e.
|
|
* addr[24] -> addr[31]
|
|
* ...
|
|
* addr[0] -> addr[7]
|
|
* So swap the byte order to let the LSB sent first.
|
|
*/
|
|
addr = HAL_SWAP32(addr);
|
|
//otherwise only addr register is sent
|
|
hw->addr = addr;
|
|
} else {
|
|
// shift the address to MSB of addr register.
|
|
// output address will be sent from MSB to LSB of addr register
|
|
hw->addr = addr << (32 - addrlen);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Set the command value in an intuitive way.
|
|
*
|
|
* The length and lsbfirst is required to shift and swap the command to the right place.
|
|
*
|
|
* @param hw Beginning command of the peripheral registers.
|
|
* @param command Command to set
|
|
* @param addrlen Length of the command phase
|
|
* @param lsbfirst whether the LSB first feature is enabled.
|
|
*/
|
|
static inline void spi_ll_set_command(spi_dev_t *hw, uint16_t cmd, int cmdlen, bool lsbfirst)
|
|
{
|
|
if (lsbfirst) {
|
|
// The output command start from bit0 to bit 15, kept as is.
|
|
HAL_FORCE_MODIFY_U32_REG_FIELD(hw->user2, usr_command_value, cmd);
|
|
} else {
|
|
/* Output command will be sent from bit 7 to 0 of command_value, and
|
|
* then bit 15 to 8 of the same register field. Shift and swap to send
|
|
* more straightly.
|
|
*/
|
|
HAL_FORCE_MODIFY_U32_REG_FIELD(hw->user2, usr_command_value, HAL_SPI_SWAP_DATA_TX(cmd, cmdlen));
|
|
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Enable/disable the RX data phase.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param enable true if RX phase exist, otherwise false.
|
|
*/
|
|
static inline void spi_ll_enable_miso(spi_dev_t *hw, int enable)
|
|
{
|
|
hw->user.usr_miso = enable;
|
|
}
|
|
|
|
/**
|
|
* Enable/disable the TX data phase.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
* @param enable true if TX phase exist, otherwise false.
|
|
*/
|
|
static inline void spi_ll_enable_mosi(spi_dev_t *hw, int enable)
|
|
{
|
|
hw->user.usr_mosi = enable;
|
|
}
|
|
|
|
/**
|
|
* Reset the slave peripheral before next transaction.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_slave_reset(spi_dev_t *hw)
|
|
{
|
|
hw->slave.soft_reset = 1;
|
|
hw->slave.soft_reset = 0;
|
|
}
|
|
|
|
/**
|
|
* Get the received bit length of the slave.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*
|
|
* @return Received bits of the slave.
|
|
*/
|
|
static inline uint32_t spi_ll_slave_get_rcv_bitlen(spi_dev_t *hw)
|
|
{
|
|
return hw->slv_rd_byte.data_bytelen * 8;
|
|
}
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* Interrupts
|
|
*----------------------------------------------------------------------------*/
|
|
//helper macros to generate code for each interrupts
|
|
#define FOR_EACH_ITEM(op, list) do { list(op) } while(0)
|
|
#define INTR_LIST(item) \
|
|
item(SPI_LL_INTR_TRANS_DONE, slave.int_trans_done_en, slave.trans_done, slave.trans_done=0) \
|
|
item(SPI_LL_INTR_RDBUF, slave.int_rd_buf_done_en, slv_rdbuf_dlen.rd_buf_done, slv_rdbuf_dlen.rd_buf_done=0) \
|
|
item(SPI_LL_INTR_WRBUF, slave.int_wr_buf_done_en, slv_wrbuf_dlen.wr_buf_done, slv_wrbuf_dlen.wr_buf_done=0) \
|
|
item(SPI_LL_INTR_RDDMA, slave.int_rd_dma_done_en, slv_rd_byte.rd_dma_done, slv_rd_byte.rd_dma_done=0) \
|
|
item(SPI_LL_INTR_WRDMA, slave.int_wr_dma_done_en, slave1.wr_dma_done, slave1.wr_dma_done=0) \
|
|
item(SPI_LL_INTR_IN_SUC_EOF, dma_int_ena.in_suc_eof, dma_int_raw.in_suc_eof, dma_int_clr.in_suc_eof=1) \
|
|
item(SPI_LL_INTR_OUT_EOF, dma_int_ena.out_eof, dma_int_raw.out_eof, dma_int_clr.out_eof=1) \
|
|
item(SPI_LL_INTR_OUT_TOTAL_EOF, dma_int_ena.out_total_eof, dma_int_raw.out_total_eof, dma_int_clr.out_total_eof=1) \
|
|
item(SPI_LL_INTR_SEG_DONE, slave.int_dma_seg_trans_en, hold.dma_seg_trans_done, hold.dma_seg_trans_done=0) \
|
|
item(SPI_LL_INTR_IN_FULL, dma_int_ena.infifo_full_err, dma_int_raw.infifo_full_err, dma_int_clr.infifo_full_err=1) \
|
|
item(SPI_LL_INTR_OUT_EMPTY, dma_int_ena.outfifo_empty_err, dma_int_raw.outfifo_empty_err, dma_int_clr.outfifo_empty_err=1) \
|
|
item(SPI_LL_INTR_CMD7, dma_int_ena.cmd7, dma_int_raw.cmd7, dma_int_clr.cmd7=1) \
|
|
item(SPI_LL_INTR_CMD8, dma_int_ena.cmd8, dma_int_raw.cmd8, dma_int_clr.cmd8=1) \
|
|
item(SPI_LL_INTR_CMD9, dma_int_ena.cmd9, dma_int_raw.cmd9, dma_int_clr.cmd9=1) \
|
|
item(SPI_LL_INTR_CMDA, dma_int_ena.cmda, dma_int_raw.cmda, dma_int_clr.cmda=1)
|
|
|
|
|
|
__attribute__((always_inline))
|
|
static inline void spi_ll_enable_intr(spi_dev_t *hw, spi_ll_intr_t intr_mask)
|
|
{
|
|
#define ENA_INTR(intr_bit, en_reg, ...) if (intr_mask & (intr_bit)) hw->en_reg = 1;
|
|
FOR_EACH_ITEM(ENA_INTR, INTR_LIST);
|
|
#undef ENA_INTR
|
|
}
|
|
|
|
__attribute__((always_inline))
|
|
static inline void spi_ll_disable_intr(spi_dev_t *hw, spi_ll_intr_t intr_mask)
|
|
{
|
|
#define DIS_INTR(intr_bit, en_reg, ...) if (intr_mask & (intr_bit)) hw->en_reg = 0;
|
|
FOR_EACH_ITEM(DIS_INTR, INTR_LIST);
|
|
#undef DIS_INTR
|
|
}
|
|
|
|
static inline void spi_ll_set_intr(spi_dev_t *hw, spi_ll_intr_t intr_mask)
|
|
{
|
|
#define SET_INTR(intr_bit, _, st_reg, ...) if (intr_mask & (intr_bit)) hw->st_reg = 1;
|
|
FOR_EACH_ITEM(SET_INTR, INTR_LIST);
|
|
#undef SET_INTR
|
|
}
|
|
|
|
__attribute__((always_inline))
|
|
static inline void spi_ll_clear_intr(spi_dev_t *hw, spi_ll_intr_t intr_mask)
|
|
{
|
|
#define CLR_INTR(intr_bit, _, __, clr_reg) if (intr_mask & (intr_bit)) hw->clr_reg;
|
|
FOR_EACH_ITEM(CLR_INTR, INTR_LIST);
|
|
#undef CLR_INTR
|
|
}
|
|
|
|
__attribute__((always_inline))
|
|
static inline bool spi_ll_get_intr(spi_dev_t *hw, spi_ll_intr_t intr_mask)
|
|
{
|
|
#define GET_INTR(intr_bit, _, st_reg, ...) if (intr_mask & (intr_bit) && hw->st_reg) return true;
|
|
FOR_EACH_ITEM(GET_INTR, INTR_LIST);
|
|
return false;
|
|
#undef GET_INTR
|
|
}
|
|
|
|
#undef FOR_EACH_ITEM
|
|
#undef INTR_LIST
|
|
|
|
/**
|
|
* Disable the trans_done interrupt.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_disable_int(spi_dev_t *hw)
|
|
{
|
|
hw->slave.int_trans_done_en = 0;
|
|
}
|
|
|
|
/**
|
|
* Clear the trans_done interrupt.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_clear_int_stat(spi_dev_t *hw)
|
|
{
|
|
hw->slave.trans_done = 0;
|
|
}
|
|
|
|
/**
|
|
* Set the trans_done interrupt.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_set_int_stat(spi_dev_t *hw)
|
|
{
|
|
hw->slave.trans_done = 1;
|
|
}
|
|
|
|
/**
|
|
* Enable the trans_done interrupt.
|
|
*
|
|
* @param hw Beginning address of the peripheral registers.
|
|
*/
|
|
static inline void spi_ll_enable_int(spi_dev_t *hw)
|
|
{
|
|
hw->slave.int_trans_done_en = 1;
|
|
}
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* Slave HD
|
|
*----------------------------------------------------------------------------*/
|
|
static inline void spi_ll_slave_hd_set_len_cond(spi_dev_t *hw, spi_ll_trans_len_cond_t cond_mask)
|
|
{
|
|
hw->slv_rd_byte.rdbuf_bytelen_en = (cond_mask & SPI_LL_TRANS_LEN_COND_RDBUF) ? 1 : 0;
|
|
hw->slv_rd_byte.wrbuf_bytelen_en = (cond_mask & SPI_LL_TRANS_LEN_COND_WRBUF) ? 1 : 0;
|
|
hw->slv_rd_byte.rddma_bytelen_en = (cond_mask & SPI_LL_TRANS_LEN_COND_RDDMA) ? 1 : 0;
|
|
hw->slv_rd_byte.wrdma_bytelen_en = (cond_mask & SPI_LL_TRANS_LEN_COND_WRDMA) ? 1 : 0;
|
|
}
|
|
|
|
static inline int spi_ll_slave_get_rx_byte_len(spi_dev_t *hw)
|
|
{
|
|
return hw->slv_rd_byte.data_bytelen;
|
|
}
|
|
|
|
static inline uint32_t spi_ll_slave_hd_get_last_addr(spi_dev_t *hw)
|
|
{
|
|
return HAL_FORCE_READ_U32_REG_FIELD(hw->slave1, last_addr);
|
|
}
|
|
|
|
/*------------------------------------------------------------------------------
|
|
* DMA:
|
|
* RX DMA (Peripherals->DMA->RAM)
|
|
* TX DMA (RAM->DMA->Peripherals)
|
|
*----------------------------------------------------------------------------*/
|
|
//---------------------------------------------------RX-------------------------------------------------//
|
|
/**
|
|
* Reset RX DMA which stores the data received from a peripheral into RAM.
|
|
*
|
|
* @param dma_in Beginning address of the DMA peripheral registers which stores the data received from a peripheral into RAM.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
*/
|
|
static inline void spi_dma_ll_rx_reset(spi_dma_dev_t *dma_in, uint32_t channel)
|
|
{
|
|
dma_in->dma_conf.in_rst = 1;
|
|
dma_in->dma_conf.in_rst = 0;
|
|
}
|
|
|
|
/**
|
|
* Start RX DMA.
|
|
*
|
|
* @param dma_in Beginning address of the DMA peripheral registers which stores the data received from a peripheral into RAM.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
* @param addr Address of the beginning DMA descriptor.
|
|
*/
|
|
static inline void spi_dma_ll_rx_start(spi_dma_dev_t *dma_in, uint32_t channel, lldesc_t *addr)
|
|
{
|
|
dma_in->dma_in_link.addr = (int) addr & 0xFFFFF;
|
|
dma_in->dma_in_link.start = 1;
|
|
}
|
|
|
|
/**
|
|
* Stop RX DMA.
|
|
*
|
|
* @param dma_in Beginning address of the DMA peripheral registers which stores the data received from a peripheral into RAM.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
*/
|
|
static inline void spi_dma_ll_rx_stop(spi_dma_dev_t *dma_in, uint32_t channel)
|
|
{
|
|
dma_in->dma_in_link.stop = 1;
|
|
}
|
|
|
|
/**
|
|
* Enable DMA RX channel burst for data
|
|
*
|
|
* @param dma_in Beginning address of the DMA peripheral registers which stores the data received from a peripheral into RAM.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
* @param enable True to enable, false to disable
|
|
*/
|
|
static inline void spi_dma_ll_rx_enable_burst_data(spi_dma_dev_t *dma_in, uint32_t channel, bool enable)
|
|
{
|
|
//This is not supported in esp32s2
|
|
}
|
|
|
|
/**
|
|
* Enable DMA RX channel burst for descriptor
|
|
*
|
|
* @param dma_in Beginning address of the DMA peripheral registers which stores the data received from a peripheral into RAM.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
* @param enable True to enable, false to disable
|
|
*/
|
|
static inline void spi_dma_ll_rx_enable_burst_desc(spi_dma_dev_t *dma_in, uint32_t channel, bool enable)
|
|
{
|
|
dma_in->dma_conf.indscr_burst_en = enable;
|
|
}
|
|
|
|
/**
|
|
* Get the last inlink descriptor address when DMA produces in_suc_eof interrupt
|
|
*
|
|
* @param dma_in Beginning address of the DMA peripheral registers which stores the data received from a peripheral into RAM.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
* @return The address
|
|
*/
|
|
__attribute__((always_inline))
|
|
static inline uint32_t spi_dma_ll_get_in_suc_eof_desc_addr(spi_dma_dev_t *dma_in, uint32_t channel)
|
|
{
|
|
return dma_in->dma_in_suc_eof_des_addr;
|
|
}
|
|
|
|
//---------------------------------------------------TX-------------------------------------------------//
|
|
/**
|
|
* Reset TX DMA which transmits the data from RAM to a peripheral.
|
|
*
|
|
* @param dma_out Beginning address of the DMA peripheral registers which transmits the data from RAM to a peripheral.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
*/
|
|
static inline void spi_dma_ll_tx_reset(spi_dma_dev_t *dma_out, uint32_t channel)
|
|
{
|
|
//Reset TX DMA peripheral
|
|
dma_out->dma_conf.out_rst = 1;
|
|
dma_out->dma_conf.out_rst = 0;
|
|
}
|
|
|
|
/**
|
|
* Start TX DMA.
|
|
*
|
|
* @param dma_out Beginning address of the DMA peripheral registers which transmits the data from RAM to a peripheral.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
* @param addr Address of the beginning DMA descriptor.
|
|
*/
|
|
static inline void spi_dma_ll_tx_start(spi_dma_dev_t *dma_out, uint32_t channel, lldesc_t *addr)
|
|
{
|
|
dma_out->dma_out_link.addr = (int) addr & 0xFFFFF;
|
|
dma_out->dma_out_link.start = 1;
|
|
}
|
|
|
|
/**
|
|
* Stop TX DMA.
|
|
*
|
|
* @param dma_out Beginning address of the DMA peripheral registers which transmits the data from RAM to a peripheral.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
*/
|
|
static inline void spi_dma_ll_tx_stop(spi_dma_dev_t *dma_out, uint32_t channel)
|
|
{
|
|
dma_out->dma_out_link.stop = 1;
|
|
}
|
|
|
|
/**
|
|
* Enable DMA TX channel burst for data
|
|
*
|
|
* @param dma_out Beginning address of the DMA peripheral registers which transmits the data from RAM to a peripheral.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
* @param enable True to enable, false to disable
|
|
*/
|
|
static inline void spi_dma_ll_tx_enable_burst_data(spi_dma_dev_t *dma_out, uint32_t channel, bool enable)
|
|
{
|
|
dma_out->dma_conf.out_data_burst_en = enable;
|
|
}
|
|
|
|
/**
|
|
* Enable DMA TX channel burst for descriptor
|
|
*
|
|
* @param dma_out Beginning address of the DMA peripheral registers which transmits the data from RAM to a peripheral.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
* @param enable True to enable, false to disable
|
|
*/
|
|
static inline void spi_dma_ll_tx_enable_burst_desc(spi_dma_dev_t *dma_out, uint32_t channel, bool enable)
|
|
{
|
|
dma_out->dma_conf.outdscr_burst_en = enable;
|
|
}
|
|
|
|
/**
|
|
* Configuration of OUT EOF flag generation way
|
|
*
|
|
* @param dma_out Beginning address of the DMA peripheral registers which transmits the data from RAM to a peripheral.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
* @param enable 1: when dma pop all data from fifo 0:when ahb push all data to fifo.
|
|
*/
|
|
static inline void spi_dma_ll_set_out_eof_generation(spi_dma_dev_t *dma_out, uint32_t channel, bool enable)
|
|
{
|
|
dma_out->dma_conf.out_eof_mode = enable;
|
|
}
|
|
|
|
/**
|
|
* Enable automatic outlink-writeback
|
|
*
|
|
* @param dma_out Beginning address of the DMA peripheral registers which transmits the data from RAM to a peripheral.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
* @param enable True to enable, false to disable
|
|
*/
|
|
static inline void spi_dma_ll_enable_out_auto_wrback(spi_dma_dev_t *dma_out, uint32_t channel, bool enable)
|
|
{
|
|
dma_out->dma_conf.out_auto_wrback = enable;
|
|
}
|
|
|
|
/**
|
|
* Get the last outlink descriptor address when DMA produces out_eof intrrupt
|
|
*
|
|
* @param dma_out Beginning address of the DMA peripheral registers which transmits the data from RAM to a peripheral.
|
|
* @param channel DMA channel, for chip version compatibility, not used.
|
|
* @return The address
|
|
*/
|
|
__attribute__((always_inline))
|
|
static inline uint32_t spi_dma_ll_get_out_eof_desc_addr(spi_dma_dev_t *dma_out, uint32_t channel)
|
|
{
|
|
return dma_out->dma_out_eof_des_addr;
|
|
}
|
|
|
|
static inline void spi_dma_ll_rx_restart(spi_dma_dev_t *dma_in, uint32_t channel)
|
|
{
|
|
dma_in->dma_in_link.restart = 1;
|
|
}
|
|
|
|
static inline void spi_dma_ll_tx_restart(spi_dma_dev_t *dma_out, uint32_t channel)
|
|
{
|
|
dma_out->dma_out_link.restart = 1;
|
|
}
|
|
|
|
static inline void spi_dma_ll_rx_disable(spi_dma_dev_t *dma_in)
|
|
{
|
|
dma_in->dma_in_link.dma_rx_ena = 0;
|
|
}
|
|
|
|
static inline void spi_dma_ll_tx_disable(spi_dma_dev_t *dma_out)
|
|
{
|
|
dma_out->dma_out_link.dma_tx_ena = 0;
|
|
}
|
|
|
|
static inline bool spi_ll_tx_get_empty_err(spi_dev_t *hw)
|
|
{
|
|
return hw->dma_int_raw.outfifo_empty_err;
|
|
}
|
|
|
|
#undef SPI_LL_RST_MASK
|
|
#undef SPI_LL_UNUSED_INT_MASK
|
|
|
|
/**
|
|
* Get the base spi command in esp32s2
|
|
*
|
|
* @param cmd_t Command value
|
|
*/
|
|
static inline uint8_t spi_ll_get_slave_hd_base_command(spi_command_t cmd_t)
|
|
{
|
|
uint8_t cmd_base = 0x00;
|
|
switch (cmd_t)
|
|
{
|
|
case SPI_CMD_HD_WRBUF:
|
|
cmd_base = SPI_LL_BASE_CMD_HD_WRBUF;
|
|
break;
|
|
case SPI_CMD_HD_RDBUF:
|
|
cmd_base = SPI_LL_BASE_CMD_HD_RDBUF;
|
|
break;
|
|
case SPI_CMD_HD_WRDMA:
|
|
cmd_base = SPI_LL_BASE_CMD_HD_WRDMA;
|
|
break;
|
|
case SPI_CMD_HD_RDDMA:
|
|
cmd_base = SPI_LL_BASE_CMD_HD_RDDMA;
|
|
break;
|
|
case SPI_CMD_HD_SEG_END:
|
|
cmd_base = SPI_LL_BASE_CMD_HD_SEG_END;
|
|
break;
|
|
case SPI_CMD_HD_EN_QPI:
|
|
cmd_base = SPI_LL_BASE_CMD_HD_EN_QPI;
|
|
break;
|
|
case SPI_CMD_HD_WR_END:
|
|
cmd_base = SPI_LL_BASE_CMD_HD_WR_END;
|
|
break;
|
|
case SPI_CMD_HD_INT0:
|
|
cmd_base = SPI_LL_BASE_CMD_HD_INT0;
|
|
break;
|
|
case SPI_CMD_HD_INT1:
|
|
cmd_base = SPI_LL_BASE_CMD_HD_INT1;
|
|
break;
|
|
case SPI_CMD_HD_INT2:
|
|
cmd_base = SPI_LL_BASE_CMD_HD_INT2;
|
|
break;
|
|
default:
|
|
HAL_ASSERT(cmd_base);
|
|
}
|
|
return cmd_base;
|
|
}
|
|
|
|
/**
|
|
* Get the spi communication command
|
|
*
|
|
* @param cmd_t Base command value
|
|
* @param line_mode Line mode of SPI transaction phases: CMD, ADDR, DOUT/DIN.
|
|
*/
|
|
static inline uint16_t spi_ll_get_slave_hd_command(spi_command_t cmd_t, spi_line_mode_t line_mode)
|
|
{
|
|
uint8_t cmd_base = spi_ll_get_slave_hd_base_command(cmd_t);
|
|
uint8_t cmd_mod = 0x00; //CMD:1-bit, ADDR:1-bit, DATA:1-bit
|
|
|
|
if (line_mode.data_lines == 2) {
|
|
if (line_mode.addr_lines == 2) {
|
|
cmd_mod = 0x50; //CMD:1-bit, ADDR:2-bit, DATA:2-bit
|
|
} else {
|
|
cmd_mod = 0x10; //CMD:1-bit, ADDR:1-bit, DATA:2-bit
|
|
}
|
|
} else if (line_mode.data_lines == 4) {
|
|
if (line_mode.addr_lines == 4) {
|
|
cmd_mod = 0xA0; //CMD:1-bit, ADDR:4-bit, DATA:4-bit
|
|
} else {
|
|
cmd_mod = 0x20; //CMD:1-bit, ADDR:1-bit, DATA:4-bit
|
|
}
|
|
}
|
|
if (cmd_base == SPI_LL_BASE_CMD_HD_SEG_END || cmd_base == SPI_LL_BASE_CMD_HD_EN_QPI) {
|
|
cmd_mod = 0x00;
|
|
}
|
|
|
|
return cmd_base | cmd_mod;
|
|
}
|
|
|
|
/**
|
|
* Get the dummy bits
|
|
*
|
|
* @param line_mode Line mode of SPI transaction phases: CMD, ADDR, DOUT/DIN.
|
|
*/
|
|
static inline int spi_ll_get_slave_hd_dummy_bits(spi_line_mode_t line_mode)
|
|
{
|
|
uint8_t dummy_bits = 0;
|
|
|
|
if (line_mode.data_lines == 2) {
|
|
dummy_bits = 4;
|
|
} else if (line_mode.data_lines == 4) {
|
|
dummy_bits = 4;
|
|
} else {
|
|
dummy_bits = 8;
|
|
}
|
|
|
|
HAL_ASSERT(dummy_bits);
|
|
return dummy_bits;
|
|
}
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|