diff --git a/components/driver/include/driver/spi_common.h b/components/driver/include/driver/spi_common.h new file mode 100644 index 0000000000..e410c75b8f --- /dev/null +++ b/components/driver/include/driver/spi_common.h @@ -0,0 +1,234 @@ +// Copyright 2010-2016 Espressif Systems (Shanghai) PTE LTD +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at + +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + + +#ifndef _DRIVER_SPI_COMMON_H_ +#define _DRIVER_SPI_COMMON_H_ + +#include +#include +#include "esp_err.h" +#include "soc/spi_struct.h" +#include "rom/lldesc.h" + + +#ifdef __cplusplus +extern "C" +{ +#endif + + +//Maximum amount of bytes that can be put in one DMA descriptor +#define SPI_MAX_DMA_LEN (4096-4) + + +/** + * @brief Enum with the three SPI peripherals that are software-accessible in it + */ +typedef enum { + SPI_HOST=0, ///< SPI1, SPI + HSPI_HOST=1, ///< SPI2, HSPI + VSPI_HOST=2 ///< SPI3, VSPI +} spi_host_device_t; + +/** + * @brief This is a configuration structure for a SPI bus. + * + * You can use this structure to specify the GPIO pins of the bus. Normally, the driver will use the + * GPIO matrix to route the signals. An exception is made when all signals either can be routed through + * the IO_MUX or are -1. In that case, the IO_MUX is used, allowing for >40MHz speeds. + */ +typedef struct { + int mosi_io_num; ///< GPIO pin for Master Out Slave In (=spi_d) signal, or -1 if not used. + int miso_io_num; ///< GPIO pin for Master In Slave Out (=spi_q) signal, or -1 if not used. + int sclk_io_num; ///< GPIO pin for Spi CLocK signal, or -1 if not used. + int quadwp_io_num; ///< GPIO pin for WP (Write Protect) signal which is used as D2 in 4-bit communication modes, or -1 if not used. + int quadhd_io_num; ///< GPIO pin for HD (HolD) signal which is used as D3 in 4-bit communication modes, or -1 if not used. + int max_transfer_sz; ///< Maximum transfer size, in bytes. Defaults to 4094 if 0. +} spi_bus_config_t; + + +/** + * @brief Try to claim a SPI peripheral + * + * Call this if your driver wants to manage a SPI peripheral. + * + * @param host Peripheral to claim + * @return True if peripheral is claimed successfully; false if peripheral already is claimed. + */ +bool spicommon_periph_claim(spi_host_device_t host); + +/** + * @brief Return the SPI peripheral so another driver can claim it. + * + * @param host Peripheral to claim + * @return True if peripheral is returned successfully; false if peripheral was free to claim already. + */ +bool spicommon_periph_free(spi_host_device_t host); + + +#define SPICOMMON_BUSFLAG_SLAVE 0 ///< Initialize I/O in slave mode +#define SPICOMMON_BUSFLAG_MASTER 1 ///< Initialize I/O in master mode +#define SPICOMMON_BUSFLAG_QUAD 2 ///< Also initialize WP/HD pins, if specified + +/** + * @brief Connect a SPI peripheral to GPIO pins + * + * This routine is used to connect a SPI peripheral to the IO-pads and DMA channel given in + * the arguments. Depending on the IO-pads requested, the routing is done either using the + * IO_mux or using the GPIO matrix. + * + * @param host SPI peripheral to be routed + * @param bus_config Pointer to a spi_bus_config struct detailing the GPIO pins + * @param dma_chan DMA-channel (1 or 2) to use, or 0 for no DMA. + * @param flags Combination of SPICOMMON_BUSFLAG_* flags + * @param is_native A value of 'true' will be written to this address if the GPIOs can be + * routed using the IO_mux, 'false' if the GPIO matrix is used. + * @return + * - ESP_ERR_INVALID_ARG if parameter is invalid + * - ESP_OK on success + */ +esp_err_t spicommon_bus_initialize_io(spi_host_device_t host, spi_bus_config_t *bus_config, int dma_chan, int flags, bool *is_native); + +/** + * @brief Free the IO used by a SPI peripheral + * + * @param host SPI peripheral to be freed + * @return + * - ESP_ERR_INVALID_ARG if parameter is invalid + * - ESP_OK on success + */ + +esp_err_t spicommon_bus_free_io(spi_host_device_t host); + +/** + * @brief Initialize a Chip Select pin for a specific SPI peripheral + * + * + * @param host SPI peripheral + * @param cs_io_num GPIO pin to route + * @param cs_num CS id to route + * @param force_gpio_matrix If true, CS will always be routed through the GPIO matrix. If false, + * if the GPIO number allows it, the routing will happen through the IO_mux. + */ + +void spicommon_cs_initialize(spi_host_device_t host, int cs_io_num, int cs_num, int force_gpio_matrix); + +/** + * @brief Free a chip select line + * + * @param host SPI peripheral + * @param cs_num CS id to free + */ +void spicommon_cs_free(spi_host_device_t host, int cs_num); + + +/** + * @brief Setup a DMA link chain + * + * This routine will set up a chain of linked DMA descriptors in the array pointed to by + * ``dmadesc``. Enough DMA descriptors will be used to fit the buffer of ``len`` bytes in, and the + * descriptors will point to the corresponding positions in ``buffer`` and linked together. The + * end result is that feeding ``dmadesc[0]`` into DMA hardware results in the entirety ``len`` bytes + * of ``data`` being read or written. + * + * @param dmadesc Pointer to array of DMA descriptors big enough to be able to convey ``len`` bytes + * @param len Length of buffer + * @param data Data buffer to use for DMA transfer + * @param isrx True if data is to be written into ``data``, false if it's to be read from ``data``. + */ +void spicommon_setup_dma_desc_links(lldesc_t *dmadesc, int len, const uint8_t *data, bool isrx); + +/** + * @brief Get the position of the hardware registers for a specific SPI host + * + * @param host The SPI host + * + * @return A register descriptor stuct pointer, pointed at the hardware registers + */ +spi_dev_t *spicommon_hw_for_host(spi_host_device_t host); + +/** + * @brief Get the IRQ source for a specific SPI host + * + * @param host The SPI host + * + * @return The hosts IRQ source + */ +int spicommon_irqsource_for_host(spi_host_device_t host); + + +/** + * @note V0 and V1 of the ESP32 silicon has a bug where in some (well-known) cases a SPI DMA channel will get confused. This can be remedied + * by resetting the SPI DMA hardware in case this happens. Unfortunately, the reset knob used for thsi will reset _both_ DMA channels, and + * as such can only done safely when both DMA channels are idle. These functions coordinate this. + * + * Essentially, when a reset is needed, a driver can request this using spicommon_dmaworkaround_req_reset. This is supposed to be called + * with an user-supplied function as an argument. If both DMA channels are idle, this call will reset the DMA subsystem and return true. + * If the other DMA channel is still busy, it will return false; as soon as the other DMA channel is done, however, it will reset the + * DMA subsystem and call the callback. The callback is then supposed to be used to continue the SPI drivers activity. +*/ + + +/** + * Callback, to be called when a DMA engine reset is completed +*/ +typedef void(*dmaworkaround_cb_t)(void *arg); + + +/** + * @brief Request a reset for a certain DMA channel + * + * @param host The SPI host + * @param cb Callback to call in case DMA channel cannot be reset immediately + * @param arg Argument to the callback + * + * @return True when a DMA reset could be executed immediately. False when it could not; in this + * case the callback will be called with the specified argument when the logic can execute + * a reset, after that reset. + */ +bool spicommon_dmaworkaround_req_reset(int dmachan, dmaworkaround_cb_t cb, void *arg); + + +/** + * @brief Check if a DMA reset is requested but has not completed yet + * + * @return True when a DMA reset is requested but hasn't completed yet. False otherwise. + */ +bool spicommon_dmaworkaround_reset_in_progress(); + + +/** + * @brief Mark a DMA channel as idle. + * + * A call to this function tells the workaround logic that this channel will + * not be affected by a global SPI DMA reset. + */ +void spicommon_dmaworkaround_idle(int dmachan); + +/** + * @brief Mark a DMA channel as active. + * + * A call to this function tells the workaround logic that this channel will + * be affected by a global SPI DMA reset, and a reset like that should not be attempted. + */ +void spicommon_dmaworkaround_transfer_active(int dmachan); + + + +#ifdef __cplusplus +} +#endif + +#endif \ No newline at end of file diff --git a/components/driver/include/driver/spi_master.h b/components/driver/include/driver/spi_master.h index 063984df76..cf6a06f4ac 100644 --- a/components/driver/include/driver/spi_master.h +++ b/components/driver/include/driver/spi_master.h @@ -20,6 +20,7 @@ #include "freertos/FreeRTOS.h" #include "freertos/semphr.h" +#include "driver/spi_common.h" #ifdef __cplusplus @@ -27,33 +28,6 @@ extern "C" { #endif - -/** - * @brief Enum with the three SPI peripherals that are software-accessible in it - */ -typedef enum { - SPI_HOST=0, ///< SPI1, SPI - HSPI_HOST=1, ///< SPI2, HSPI - VSPI_HOST=2 ///< SPI3, VSPI -} spi_host_device_t; - - -/** - * @brief This is a configuration structure for a SPI bus. - * - * You can use this structure to specify the GPIO pins of the bus. Normally, the driver will use the - * GPIO matrix to route the signals. An exception is made when all signals either can be routed through - * the IO_MUX or are -1. In that case, the IO_MUX is used, allowing for >40MHz speeds. - */ -typedef struct { - int mosi_io_num; ///< GPIO pin for Master Out Slave In (=spi_d) signal, or -1 if not used. - int miso_io_num; ///< GPIO pin for Master In Slave Out (=spi_q) signal, or -1 if not used. - int sclk_io_num; ///< GPIO pin for Spi CLocK signal, or -1 if not used. - int quadwp_io_num; ///< GPIO pin for WP (Write Protect) signal which is used as D2 in 4-bit communication modes, or -1 if not used. - int quadhd_io_num; ///< GPIO pin for HD (HolD) signal which is used as D3 in 4-bit communication modes, or -1 if not used. -} spi_bus_config_t; - - #define SPI_DEVICE_TXBIT_LSBFIRST (1<<0) ///< Transmit command/address/data LSB first instead of the default MSB first #define SPI_DEVICE_RXBIT_LSBFIRST (1<<1) ///< Receive data LSB first instead of the default MSB first #define SPI_DEVICE_BIT_LSBFIRST (SPI_TXBIT_LSBFIRST|SPI_RXBIT_LSBFIRST); ///< Transmit and receive LSB first @@ -122,9 +96,10 @@ typedef struct spi_device_t* spi_device_handle_t; ///< Handle for a device on a * * @param host SPI peripheral that controls this bus * @param bus_config Pointer to a spi_bus_config_t struct specifying how the host should be initialized - * @param dma_chan Either 1 or 2. A SPI bus used by this driver must have a DMA channel associated with - * it. The SPI hardware has two DMA channels to share. This parameter indicates which - * one to use. + * @param dma_chan Either channel 1 or 2, or 0 in the case when no DMA is required. Selecting a DMA channel + * for a SPI bus allows transfers on the bus to have sizes only limited by the amount of + * internal memory. Selecting no DMA channel (by passing the value 0) limits the amount of + * bytes transfered to a maximum of 32. * @return * - ESP_ERR_INVALID_ARG if configuration is invalid * - ESP_ERR_INVALID_STATE if host already is in use diff --git a/components/driver/include/driver/spi_slave.h b/components/driver/include/driver/spi_slave.h new file mode 100644 index 0000000000..c27809bc8b --- /dev/null +++ b/components/driver/include/driver/spi_slave.h @@ -0,0 +1,154 @@ +// Copyright 2010-2016 Espressif Systems (Shanghai) PTE LTD +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at + +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + + +#ifndef _DRIVER_SPI_SLAVE_H_ +#define _DRIVER_SPI_SLAVE_H_ + +#include "esp_err.h" +#include "freertos/FreeRTOS.h" +#include "freertos/semphr.h" +#include "driver/spi_common.h" + + +#ifdef __cplusplus +extern "C" +{ +#endif + + +#define SPI_SLAVE_TXBIT_LSBFIRST (1<<0) ///< Transmit command/address/data LSB first instead of the default MSB first +#define SPI_SLAVE_RXBIT_LSBFIRST (1<<1) ///< Receive data LSB first instead of the default MSB first +#define SPI_SLAVE_BIT_LSBFIRST (SPI_TXBIT_LSBFIRST|SPI_RXBIT_LSBFIRST); ///< Transmit and receive LSB first +#define SPI_SLAVE_POSITIVE_CS (1<<3) ///< Make CS positive during a transaction instead of negative + + +typedef struct spi_slave_transaction_t spi_slave_transaction_t; +typedef void(*slave_transaction_cb_t)(spi_slave_transaction_t *trans); + +/** + * @brief This is a configuration for a SPI host acting as a slave device. + */ +typedef struct { + int spics_io_num; ///< CS GPIO pin for this device, or -1 if not used + uint32_t flags; ///< Bitwise OR of SPI_DEVICE_* flags + int queue_size; ///< Transaction queue size. This sets how many transactions can be 'in the air' (queued using spi_device_queue_trans but not yet finished using spi_device_get_trans_result) at the same time + uint8_t mode; ///< SPI mode (0-3) + slave_transaction_cb_t post_setup_cb; ///< Callback called after the SPI registers are loaded with new data + slave_transaction_cb_t post_trans_cb; ///< Callback called after a transaction is done +} spi_slave_interface_config_t; + +/** + * This structure describes one SPI transaction + */ +struct spi_slave_transaction_t { + size_t length; ///< Total data length, in bits + const void *tx_buffer; ///< Pointer to transmit buffer, or NULL for no MOSI phase + void *rx_buffer; ///< Pointer to receive buffer, or NULL for no MISO phase +}; + +/** + * @brief Initialize a SPI bus + * + * @warning For now, only supports HSPI and VSPI. + * + * @param host SPI peripheral to use as a SPI slave interface + * @param bus_config Pointer to a spi_bus_config_t struct specifying how the host should be initialized + * @param slave_config Pointer to a spi_slave_interface_config_t struct specifying the details for the slave interface + * @param dma_chan Either 1 or 2. A SPI bus used by this driver must have a DMA channel associated with + * it. The SPI hardware has two DMA channels to share. This parameter indicates which + * one to use. + * @return + * - ESP_ERR_INVALID_ARG if configuration is invalid + * - ESP_ERR_INVALID_STATE if host already is in use + * - ESP_ERR_NO_MEM if out of memory + * - ESP_OK on success + */ +esp_err_t spi_slave_initialize(spi_host_device_t host, spi_bus_config_t *bus_config, spi_slave_interface_config_t *slave_config, int dma_chan); + +/** + * @brief Free a SPI bus claimed as a SPI slave interface + * + * @param host SPI peripheral to free + * @return + * - ESP_ERR_INVALID_ARG if parameter is invalid + * - ESP_ERR_INVALID_STATE if not all devices on the bus are freed + * - ESP_OK on success + */ +esp_err_t spi_slave_free(spi_host_device_t host); + + +/** + * @brief Queue a SPI transaction for execution + * + * This will queue a transaction for the master to pick it up. If the queue (specified in ``spi_slave_initialize``) + * is not full, this function will return directly; the actual transaction will be done if there aren't any + * unhandled transactions before it and the master initiates a SPI transaction by pulling down CS and sending out + * clock signals. + * + * @param handle Device handle obtained using spi_host_add_dev + * @param trans_desc Description of transaction to execute + * @param ticks_to_wait Ticks to wait until there's room in the queue; use portMAX_DELAY to + * never time out. + * @return + * - ESP_ERR_INVALID_ARG if parameter is invalid + * - ESP_OK on success + */ +esp_err_t spi_slave_queue_trans(spi_host_device_t host, spi_slave_transaction_t *trans_desc, TickType_t ticks_to_wait); + + +/** + * @brief Get the result of a SPI transaction queued earlier + * + * This routine will wait until a transaction to the given device (queued earlier with + * spi_device_queue_trans) has succesfully completed. It will then return the description of the + * completed transaction so software can inspect the result and e.g. free the memory or + * re-use the buffers. + * + * @param handle Device handle obtained using spi_host_add_dev + * @param trans_desc Pointer to variable able to contain a pointer to the description of the + * transaction that is executed + * @param ticks_to_wait Ticks to wait until there's a returned item; use portMAX_DELAY to never time + * out. + * @return + * - ESP_ERR_INVALID_ARG if parameter is invalid + * - ESP_OK on success + */ +esp_err_t spi_slave_get_trans_result(spi_host_device_t host, spi_slave_transaction_t **trans_desc, TickType_t ticks_to_wait); + + +/** + * @brief Do a SPI transaction + * + * Essentially does the same as spi_device_queue_trans followed by spi_device_get_trans_result. Do + * not use this when there is still a transaction queued that hasn't been finalized + * using spi_device_get_trans_result. + * + * @param handle Device handle obtained using spi_host_add_dev + * @param trans_desc Pointer to variable able to contain a pointer to the description of the + * transaction that is executed + * @param ticks_to_wait Ticks to wait until there's a returned item; use portMAX_DELAY to never time + * out. + * @return + * - ESP_ERR_INVALID_ARG if parameter is invalid + * - ESP_OK on success + */ +esp_err_t spi_slave_transmit(spi_host_device_t host, spi_slave_transaction_t *trans_desc, TickType_t ticks_to_wait); + + +#ifdef __cplusplus +} +#endif + +#endif \ No newline at end of file diff --git a/components/driver/spi_common.c b/components/driver/spi_common.c new file mode 100644 index 0000000000..4b098b8309 --- /dev/null +++ b/components/driver/spi_common.c @@ -0,0 +1,396 @@ +// Copyright 2015-2016 Espressif Systems (Shanghai) PTE LTD +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at + +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + + +#include +#include "driver/spi_master.h" +#include "soc/gpio_sig_map.h" +#include "soc/spi_reg.h" +#include "soc/dport_reg.h" +#include "soc/spi_struct.h" +#include "soc/rtc_cntl_reg.h" +#include "rom/ets_sys.h" +#include "esp_types.h" +#include "esp_attr.h" +#include "esp_intr.h" +#include "esp_intr_alloc.h" +#include "esp_log.h" +#include "esp_err.h" +#include "freertos/FreeRTOS.h" +#include "freertos/semphr.h" +#include "freertos/xtensa_api.h" +#include "freertos/task.h" +#include "freertos/ringbuf.h" +#include "soc/soc.h" +#include "soc/dport_reg.h" +#include "soc/uart_struct.h" +#include "rom/lldesc.h" +#include "driver/uart.h" +#include "driver/gpio.h" +#include "driver/periph_ctrl.h" +#include "esp_heap_alloc_caps.h" + +#include "driver/spi_common.h" + +static const char *SPI_TAG = "spi"; + +#define SPI_CHECK(a, str, ret_val) \ + if (!(a)) { \ + ESP_LOGE(SPI_TAG,"%s(%d): %s", __FUNCTION__, __LINE__, str); \ + return (ret_val); \ + } + + +typedef struct spi_device_t spi_device_t; + +/* + Stores a bunch of per-spi-peripheral data. +*/ +typedef struct { + const uint8_t spiclk_out; //GPIO mux output signals + const uint8_t spiclk_in; + const uint8_t spid_out; + const uint8_t spiq_out; + const uint8_t spiwp_out; + const uint8_t spihd_out; + const uint8_t spid_in; //GPIO mux input signals + const uint8_t spiq_in; + const uint8_t spiwp_in; + const uint8_t spihd_in; + const uint8_t spics_out[3]; // /CS GPIO output mux signals + const uint8_t spics_in; + const uint8_t spiclk_native; //IO pins of IO_MUX muxed signals + const uint8_t spid_native; + const uint8_t spiq_native; + const uint8_t spiwp_native; + const uint8_t spihd_native; + const uint8_t spics0_native; + const uint8_t irq; //irq source for interrupt mux + const uint8_t irq_dma; //dma irq source for interrupt mux + const periph_module_t module; //peripheral module, for enabling clock etc + spi_dev_t *hw; //Pointer to the hardware registers +} spi_signal_conn_t; + +/* + Bunch of constants for every SPI peripheral: GPIO signals, irqs, hw addr of registers etc +*/ +static const spi_signal_conn_t io_signal[3]={ + { + .spiclk_out=SPICLK_OUT_IDX, + .spiclk_in=SPICLK_IN_IDX, + .spid_out=SPID_OUT_IDX, + .spiq_out=SPIQ_OUT_IDX, + .spiwp_out=SPIWP_OUT_IDX, + .spihd_out=SPIHD_OUT_IDX, + .spid_in=SPID_IN_IDX, + .spiq_in=SPIQ_IN_IDX, + .spiwp_in=SPIWP_IN_IDX, + .spihd_in=SPIHD_IN_IDX, + .spics_out={SPICS0_OUT_IDX, SPICS1_OUT_IDX, SPICS2_OUT_IDX}, + .spics_in=SPICS0_IN_IDX, + .spiclk_native=6, + .spid_native=8, + .spiq_native=7, + .spiwp_native=10, + .spihd_native=9, + .spics0_native=11, + .irq=ETS_SPI1_INTR_SOURCE, + .irq_dma=ETS_SPI1_DMA_INTR_SOURCE, + .module=PERIPH_SPI_MODULE, + .hw=&SPI1 + }, { + .spiclk_out=HSPICLK_OUT_IDX, + .spiclk_in=HSPICLK_IN_IDX, + .spid_out=HSPID_OUT_IDX, + .spiq_out=HSPIQ_OUT_IDX, + .spiwp_out=HSPIWP_OUT_IDX, + .spihd_out=HSPIHD_OUT_IDX, + .spid_in=HSPID_IN_IDX, + .spiq_in=HSPIQ_IN_IDX, + .spiwp_in=HSPIWP_IN_IDX, + .spihd_in=HSPIHD_IN_IDX, + .spics_out={HSPICS0_OUT_IDX, HSPICS1_OUT_IDX, HSPICS2_OUT_IDX}, + .spics_in=HSPICS0_IN_IDX, + .spiclk_native=14, + .spid_native=13, + .spiq_native=12, + .spiwp_native=2, + .spihd_native=4, + .spics0_native=15, + .irq=ETS_SPI2_INTR_SOURCE, + .irq_dma=ETS_SPI2_DMA_INTR_SOURCE, + .module=PERIPH_HSPI_MODULE, + .hw=&SPI2 + }, { + .spiclk_out=VSPICLK_OUT_IDX, + .spiclk_in=VSPICLK_IN_IDX, + .spid_out=VSPID_OUT_IDX, + .spiq_out=VSPIQ_OUT_IDX, + .spiwp_out=VSPIWP_OUT_IDX, + .spihd_out=VSPIHD_OUT_IDX, + .spid_in=VSPID_IN_IDX, + .spiq_in=VSPIQ_IN_IDX, + .spiwp_in=VSPIWP_IN_IDX, + .spihd_in=VSPIHD_IN_IDX, + .spics_out={VSPICS0_OUT_IDX, VSPICS1_OUT_IDX, VSPICS2_OUT_IDX}, + .spics_in=VSPICS0_IN_IDX, + .spiclk_native=18, + .spid_native=23, + .spiq_native=19, + .spiwp_native=22, + .spihd_native=21, + .spics0_native=5, + .irq=ETS_SPI3_INTR_SOURCE, + .irq_dma=ETS_SPI3_DMA_INTR_SOURCE, + .module=PERIPH_VSPI_MODULE, + .hw=&SPI3 + } +}; + +//Periph 1 is 'claimed' by SPI flash code. +static bool spi_periph_claimed[3]={true, false, false}; + +//Returns true if this peripheral is successfully claimed, false if otherwise. +bool spicommon_periph_claim(spi_host_device_t host) { + bool ret = __sync_bool_compare_and_swap(&spi_periph_claimed[host], false, true); + if (ret) periph_module_enable(io_signal[host].module); + return ret; +} + +//Returns true if this peripheral is successfully freed, false if otherwise. +bool spicommon_periph_free(spi_host_device_t host) { + bool ret = __sync_bool_compare_and_swap(&spi_periph_claimed[host], true, false); + if (ret) periph_module_disable(io_signal[host].module); + return ret; +} + + +int spicommon_irqsource_for_host(spi_host_device_t host) { + return io_signal[host].irq; +} + +spi_dev_t *spicommon_hw_for_host(spi_host_device_t host) { + return io_signal[host].hw; +} + +/* +Do the common stuff to hook up a SPI host to a bus defined by a bunch of GPIO pins. Feed it a host number and a +bus config struct and it'll set up the GPIO matrix and enable the device. It will set is_native to 1 if the bus +config can be done using the IOMUX instead of using the GPIO matrix. +*/ +esp_err_t spicommon_bus_initialize_io(spi_host_device_t host, spi_bus_config_t *bus_config, int dma_chan, int flags, bool *is_native) +{ + bool native=true; + bool is_master=(flags&SPICOMMON_BUSFLAG_MASTER)?true:false; + bool use_quad=(flags&SPICOMMON_BUSFLAG_QUAD)?true:false; + + SPI_CHECK(bus_config->mosi_io_num<0 || GPIO_IS_VALID_OUTPUT_GPIO(bus_config->mosi_io_num), "spid pin invalid", ESP_ERR_INVALID_ARG); + SPI_CHECK(bus_config->sclk_io_num<0 || GPIO_IS_VALID_OUTPUT_GPIO(bus_config->sclk_io_num), "spiclk pin invalid", ESP_ERR_INVALID_ARG); + SPI_CHECK(bus_config->miso_io_num<0 || GPIO_IS_VALID_GPIO(bus_config->miso_io_num), "spiq pin invalid", ESP_ERR_INVALID_ARG); + if (use_quad) { + SPI_CHECK(bus_config->quadwp_io_num<0 || GPIO_IS_VALID_OUTPUT_GPIO(bus_config->quadwp_io_num), "spiwp pin invalid", ESP_ERR_INVALID_ARG); + SPI_CHECK(bus_config->quadhd_io_num<0 || GPIO_IS_VALID_OUTPUT_GPIO(bus_config->quadhd_io_num), "spihd pin invalid", ESP_ERR_INVALID_ARG); + } + + //Check if the selected pins correspond to the native pins of the peripheral + if (bus_config->mosi_io_num >= 0 && bus_config->mosi_io_num!=io_signal[host].spid_native) native=false; + if (bus_config->miso_io_num >= 0 && bus_config->miso_io_num!=io_signal[host].spiq_native) native=false; + if (bus_config->sclk_io_num >= 0 && bus_config->sclk_io_num!=io_signal[host].spiclk_native) native=false; + if (use_quad) { + if (bus_config->quadwp_io_num >= 0 && bus_config->quadwp_io_num!=io_signal[host].spiwp_native) native=false; + if (bus_config->quadhd_io_num >= 0 && bus_config->quadhd_io_num!=io_signal[host].spihd_native) native=false; + } + + *is_native=native; + + if (native) { + //All SPI native pin selections resolve to 1, so we put that here instead of trying to figure + //out which FUNC_GPIOx_xSPIxx to grab; they all are defined to 1 anyway. + if (bus_config->mosi_io_num > 0) PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->mosi_io_num], 1); + if (bus_config->miso_io_num > 0) PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->miso_io_num], 1); + if (use_quad && bus_config->quadwp_io_num > 0) PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->quadwp_io_num], 1); + if (use_quad && bus_config->quadhd_io_num > 0) PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->quadhd_io_num], 1); + if (bus_config->sclk_io_num > 0) PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->sclk_io_num], 1); + } else { + //Use GPIO + if (bus_config->mosi_io_num>0) { + PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->mosi_io_num], PIN_FUNC_GPIO); + gpio_set_direction(bus_config->mosi_io_num, is_master?GPIO_MODE_OUTPUT:GPIO_MODE_INPUT); + gpio_matrix_out(bus_config->mosi_io_num, io_signal[host].spid_out, false, false); + gpio_matrix_in(bus_config->mosi_io_num, io_signal[host].spid_in, false); + } + if (bus_config->miso_io_num>0) { + PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->miso_io_num], PIN_FUNC_GPIO); + gpio_set_direction(bus_config->miso_io_num, is_master?GPIO_MODE_INPUT:GPIO_MODE_OUTPUT); + gpio_matrix_out(bus_config->miso_io_num, io_signal[host].spiq_out, false, false); + gpio_matrix_in(bus_config->miso_io_num, io_signal[host].spiq_in, false); + } + if (use_quad && bus_config->quadwp_io_num>0) { + PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->quadwp_io_num], PIN_FUNC_GPIO); + gpio_set_direction(bus_config->quadwp_io_num, is_master?GPIO_MODE_OUTPUT:GPIO_MODE_INPUT); + gpio_matrix_out(bus_config->quadwp_io_num, io_signal[host].spiwp_out, false, false); + gpio_matrix_in(bus_config->quadwp_io_num, io_signal[host].spiwp_in, false); + } + if (use_quad && bus_config->quadhd_io_num>0) { + PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->quadhd_io_num], PIN_FUNC_GPIO); + gpio_set_direction(bus_config->quadhd_io_num, is_master?GPIO_MODE_OUTPUT:GPIO_MODE_INPUT); + gpio_matrix_out(bus_config->quadhd_io_num, io_signal[host].spihd_out, false, false); + gpio_matrix_in(bus_config->quadhd_io_num, io_signal[host].spihd_in, false); + } + if (bus_config->sclk_io_num>0) { + PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->sclk_io_num], PIN_FUNC_GPIO); + gpio_set_direction(bus_config->sclk_io_num, is_master?GPIO_MODE_OUTPUT:GPIO_MODE_INPUT); + gpio_matrix_out(bus_config->sclk_io_num, io_signal[host].spiclk_out, false, false); + gpio_matrix_in(bus_config->sclk_io_num, io_signal[host].spiclk_in, false); + } + } + + //Select DMA channel. + SET_PERI_REG_BITS(DPORT_SPI_DMA_CHAN_SEL_REG, 3, dma_chan, (host * 2)); + + return ESP_OK; +} + + +//Find any pin with output muxed to ``func`` and reset it to GPIO +static void reset_func_to_gpio(int func) { + for (int x=0; x SPI_MAX_DMA_LEN) dmachunklen=SPI_MAX_DMA_LEN; + if (isrx) { + //Receive needs DMA length rounded to next 32-bit boundary + dmadesc[n].size=(dmachunklen+3)&(~3); + dmadesc[n].length=(dmachunklen+3)&(~3); + } else { + dmadesc[n].size=dmachunklen; + dmadesc[n].length=dmachunklen; + } + dmadesc[n].buf=(uint8_t*)data; + dmadesc[n].eof=0; + dmadesc[n].sosf=0; + dmadesc[n].owner=1; + dmadesc[n].qe.stqe_next=&dmadesc[n+1]; + len-=dmachunklen; + data+=dmachunklen; + n++; + } + dmadesc[n-1].eof=1; //Mark last DMA desc as end of stream. + dmadesc[n-1].qe.stqe_next=NULL; +} + + +/* +Code for workaround for DMA issue in ESP32 v0/v1 silicon +*/ + + +static volatile int dmaworkaround_channels_busy[2]={0,0}; +static dmaworkaround_cb_t dmaworkaround_cb; +static void *dmaworkaround_cb_arg; +static portMUX_TYPE dmaworkaround_mux=portMUX_INITIALIZER_UNLOCKED; +static int dmaworkaround_waiting_for_chan=0; + +bool IRAM_ATTR spicommon_dmaworkaround_req_reset(int dmachan, dmaworkaround_cb_t cb, void *arg) +{ + int otherchan=(dmachan==1)?2:1; + bool ret; + portENTER_CRITICAL(&dmaworkaround_mux); + if (dmaworkaround_channels_busy[otherchan]) { + //Other channel is busy. Call back when it's done. + dmaworkaround_cb=cb; + dmaworkaround_cb_arg=arg; + dmaworkaround_waiting_for_chan=otherchan; + ret=false; + } else { + //Reset DMA + SET_PERI_REG_MASK(DPORT_PERIP_RST_EN_REG, DPORT_SPI_DMA_RST); + CLEAR_PERI_REG_MASK(DPORT_PERIP_RST_EN_REG, DPORT_SPI_DMA_RST); + ret=true; + } + portEXIT_CRITICAL(&dmaworkaround_mux); + return ret; +} + +bool IRAM_ATTR spicommon_dmaworkaround_reset_in_progress() +{ + return (dmaworkaround_waiting_for_chan!=0); +} + +void IRAM_ATTR spicommon_dmaworkaround_idle(int dmachan) { + portENTER_CRITICAL(&dmaworkaround_mux); + dmaworkaround_channels_busy[dmachan]=0; + if (dmaworkaround_waiting_for_chan == dmachan) { + //Reset DMA + SET_PERI_REG_MASK(DPORT_PERIP_RST_EN_REG, DPORT_SPI_DMA_RST); + CLEAR_PERI_REG_MASK(DPORT_PERIP_RST_EN_REG, DPORT_SPI_DMA_RST); + dmaworkaround_waiting_for_chan=0; + //Call callback + dmaworkaround_cb(dmaworkaround_cb_arg); + + } + portEXIT_CRITICAL(&dmaworkaround_mux); +} + +void IRAM_ATTR spicommon_dmaworkaround_transfer_active(int dmachan) { + portENTER_CRITICAL(&dmaworkaround_mux); + dmaworkaround_channels_busy[dmachan]=1; + portEXIT_CRITICAL(&dmaworkaround_mux); +} + + diff --git a/components/driver/spi_master.c b/components/driver/spi_master.c index 8951371fdc..06d29081e1 100644 --- a/components/driver/spi_master.c +++ b/components/driver/spi_master.c @@ -34,6 +34,7 @@ queue and re-enabling the interrupt will trigger the interrupt again, which can #include +#include "driver/spi_common.h" #include "driver/spi_master.h" #include "soc/gpio_sig_map.h" #include "soc/spi_reg.h" @@ -71,8 +72,11 @@ typedef struct { spi_dev_t *hw; spi_transaction_t *cur_trans; int cur_cs; - lldesc_t dmadesc_tx, dmadesc_rx; + lldesc_t *dmadesc_tx; + lldesc_t *dmadesc_rx; bool no_gpio_matrix; + int dma_chan; + int max_transfer_sz; } spi_host_t; struct spi_device_t { @@ -92,182 +96,47 @@ static const char *SPI_TAG = "spi_master"; return (ret_val); \ } -/* - Stores a bunch of per-spi-peripheral data. -*/ -typedef struct { - const uint8_t spiclk_out; //GPIO mux output signals - const uint8_t spid_out; - const uint8_t spiq_out; - const uint8_t spiwp_out; - const uint8_t spihd_out; - const uint8_t spid_in; //GPIO mux input signals - const uint8_t spiq_in; - const uint8_t spiwp_in; - const uint8_t spihd_in; - const uint8_t spics_out[3]; // /CS GPIO output mux signals - const uint8_t spiclk_native; //IO pins of IO_MUX muxed signals - const uint8_t spid_native; - const uint8_t spiq_native; - const uint8_t spiwp_native; - const uint8_t spihd_native; - const uint8_t spics0_native; - const uint8_t irq; //irq source for interrupt mux - const uint8_t irq_dma; //dma irq source for interrupt mux - const periph_module_t module; //peripheral module, for enabling clock etc - spi_dev_t *hw; //Pointer to the hardware registers -} spi_signal_conn_t; - -/* - Bunch of constants for every SPI peripheral: GPIO signals, irqs, hw addr of registers etc -*/ -static const spi_signal_conn_t io_signal[3]={ - { - .spiclk_out=SPICLK_OUT_IDX, - .spid_out=SPID_OUT_IDX, - .spiq_out=SPIQ_OUT_IDX, - .spiwp_out=SPIWP_OUT_IDX, - .spihd_out=SPIHD_OUT_IDX, - .spid_in=SPID_IN_IDX, - .spiq_in=SPIQ_IN_IDX, - .spiwp_in=SPIWP_IN_IDX, - .spihd_in=SPIHD_IN_IDX, - .spics_out={SPICS0_OUT_IDX, SPICS1_OUT_IDX, SPICS2_OUT_IDX}, - .spiclk_native=6, - .spid_native=8, - .spiq_native=7, - .spiwp_native=10, - .spihd_native=9, - .spics0_native=11, - .irq=ETS_SPI1_INTR_SOURCE, - .irq_dma=ETS_SPI1_DMA_INTR_SOURCE, - .module=PERIPH_SPI_MODULE, - .hw=&SPI1 - }, { - .spiclk_out=HSPICLK_OUT_IDX, - .spid_out=HSPID_OUT_IDX, - .spiq_out=HSPIQ_OUT_IDX, - .spiwp_out=HSPIWP_OUT_IDX, - .spihd_out=HSPIHD_OUT_IDX, - .spid_in=HSPID_IN_IDX, - .spiq_in=HSPIQ_IN_IDX, - .spiwp_in=HSPIWP_IN_IDX, - .spihd_in=HSPIHD_IN_IDX, - .spics_out={HSPICS0_OUT_IDX, HSPICS1_OUT_IDX, HSPICS2_OUT_IDX}, - .spiclk_native=14, - .spid_native=13, - .spiq_native=12, - .spiwp_native=2, - .spihd_native=4, - .spics0_native=15, - .irq=ETS_SPI2_INTR_SOURCE, - .irq_dma=ETS_SPI2_DMA_INTR_SOURCE, - .module=PERIPH_HSPI_MODULE, - .hw=&SPI2 - }, { - .spiclk_out=VSPICLK_OUT_IDX, - .spid_out=VSPID_OUT_IDX, - .spiq_out=VSPIQ_OUT_IDX, - .spiwp_out=VSPIWP_OUT_IDX, - .spihd_out=VSPIHD_OUT_IDX, - .spid_in=VSPID_IN_IDX, - .spiq_in=VSPIQ_IN_IDX, - .spiwp_in=VSPIWP_IN_IDX, - .spihd_in=VSPIHD_IN_IDX, - .spics_out={VSPICS0_OUT_IDX, VSPICS1_OUT_IDX, VSPICS2_OUT_IDX}, - .spiclk_native=18, - .spid_native=23, - .spiq_native=19, - .spiwp_native=22, - .spihd_native=21, - .spics0_native=5, - .irq=ETS_SPI3_INTR_SOURCE, - .irq_dma=ETS_SPI3_DMA_INTR_SOURCE, - .module=PERIPH_VSPI_MODULE, - .hw=&SPI3 - } -}; static void spi_intr(void *arg); - esp_err_t spi_bus_initialize(spi_host_device_t host, const spi_bus_config_t *bus_config, int dma_chan) { - bool native=true; + bool native, claimed; /* ToDo: remove this when we have flash operations cooperating with this */ SPI_CHECK(host!=SPI_HOST, "SPI1 is not supported", ESP_ERR_NOT_SUPPORTED); SPI_CHECK(host>=SPI_HOST && host<=VSPI_HOST, "invalid host", ESP_ERR_INVALID_ARG); - SPI_CHECK(spihost[host]==NULL, "host already in use", ESP_ERR_INVALID_STATE); - - SPI_CHECK(bus_config->mosi_io_num<0 || GPIO_IS_VALID_OUTPUT_GPIO(bus_config->mosi_io_num), "spid pin invalid", ESP_ERR_INVALID_ARG); - SPI_CHECK(bus_config->sclk_io_num<0 || GPIO_IS_VALID_OUTPUT_GPIO(bus_config->sclk_io_num), "spiclk pin invalid", ESP_ERR_INVALID_ARG); - SPI_CHECK(bus_config->miso_io_num<0 || GPIO_IS_VALID_GPIO(bus_config->miso_io_num), "spiq pin invalid", ESP_ERR_INVALID_ARG); - SPI_CHECK(bus_config->quadwp_io_num<0 || GPIO_IS_VALID_OUTPUT_GPIO(bus_config->quadwp_io_num), "spiwp pin invalid", ESP_ERR_INVALID_ARG); - SPI_CHECK(bus_config->quadhd_io_num<0 || GPIO_IS_VALID_OUTPUT_GPIO(bus_config->quadhd_io_num), "spihd pin invalid", ESP_ERR_INVALID_ARG); - //The host struct contains two dma descriptors, so we need DMA'able memory for this. - spihost[host]=pvPortMallocCaps(sizeof(spi_host_t), MALLOC_CAP_DMA); - if (spihost[host]==NULL) return ESP_ERR_NO_MEM; + claimed=spicommon_periph_claim(host); + SPI_CHECK(claimed, "host already in use", ESP_ERR_INVALID_STATE); + + spihost[host]=malloc(sizeof(spi_host_t)); + if (spihost[host]==NULL) goto nomem; memset(spihost[host], 0, sizeof(spi_host_t)); - //Check if the selected pins correspond to the native pins of the peripheral - if (bus_config->mosi_io_num >= 0 && bus_config->mosi_io_num!=io_signal[host].spid_native) native=false; - if (bus_config->miso_io_num >= 0 && bus_config->miso_io_num!=io_signal[host].spiq_native) native=false; - if (bus_config->sclk_io_num >= 0 && bus_config->sclk_io_num!=io_signal[host].spiclk_native) native=false; - if (bus_config->quadwp_io_num >= 0 && bus_config->quadwp_io_num!=io_signal[host].spiwp_native) native=false; - if (bus_config->quadhd_io_num >= 0 && bus_config->quadhd_io_num!=io_signal[host].spihd_native) native=false; - + spicommon_bus_initialize_io(host, bus_config, dma_chan, SPICOMMON_BUSFLAG_MASTER|SPICOMMON_BUSFLAG_QUAD, &native); spihost[host]->no_gpio_matrix=native; - if (native) { - //All SPI native pin selections resolve to 1, so we put that here instead of trying to figure - //out which FUNC_GPIOx_xSPIxx to grab; they all are defined to 1 anyway. - if (bus_config->mosi_io_num > 0) PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->mosi_io_num], 1); - if (bus_config->miso_io_num > 0) PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->miso_io_num], 1); - if (bus_config->quadwp_io_num > 0) PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->quadwp_io_num], 1); - if (bus_config->quadhd_io_num > 0) PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->quadhd_io_num], 1); - if (bus_config->sclk_io_num > 0) PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->sclk_io_num], 1); + + spihost[host]->dma_chan=dma_chan; + if (dma_chan == 0) { + spihost[host]->max_transfer_sz = 32; } else { - //Use GPIO - if (bus_config->mosi_io_num>0) { - PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->mosi_io_num], PIN_FUNC_GPIO); - gpio_set_direction(bus_config->mosi_io_num, GPIO_MODE_OUTPUT); - gpio_matrix_out(bus_config->mosi_io_num, io_signal[host].spid_out, false, false); - gpio_matrix_in(bus_config->mosi_io_num, io_signal[host].spid_in, false); - } - if (bus_config->miso_io_num>0) { - PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->miso_io_num], PIN_FUNC_GPIO); - gpio_set_direction(bus_config->miso_io_num, GPIO_MODE_INPUT); - gpio_matrix_out(bus_config->miso_io_num, io_signal[host].spiq_out, false, false); - gpio_matrix_in(bus_config->miso_io_num, io_signal[host].spiq_in, false); - } - if (bus_config->quadwp_io_num>0) { - PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->quadwp_io_num], PIN_FUNC_GPIO); - gpio_set_direction(bus_config->quadwp_io_num, GPIO_MODE_OUTPUT); - gpio_matrix_out(bus_config->quadwp_io_num, io_signal[host].spiwp_out, false, false); - gpio_matrix_in(bus_config->quadwp_io_num, io_signal[host].spiwp_in, false); - } - if (bus_config->quadhd_io_num>0) { - PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->quadhd_io_num], PIN_FUNC_GPIO); - gpio_set_direction(bus_config->quadhd_io_num, GPIO_MODE_OUTPUT); - gpio_matrix_out(bus_config->quadhd_io_num, io_signal[host].spihd_out, false, false); - gpio_matrix_in(bus_config->quadhd_io_num, io_signal[host].spihd_in, false); - } - if (bus_config->sclk_io_num>0) { - PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[bus_config->sclk_io_num], PIN_FUNC_GPIO); - gpio_set_direction(bus_config->sclk_io_num, GPIO_MODE_OUTPUT); - gpio_matrix_out(bus_config->sclk_io_num, io_signal[host].spiclk_out, false, false); - } + //See how many dma descriptors we need and allocate them + int dma_desc_ct=(bus_config->max_transfer_sz+SPI_MAX_DMA_LEN-1)/SPI_MAX_DMA_LEN; + if (dma_desc_ct==0) dma_desc_ct=1; //default to 4k when max is not given + spihost[host]->max_transfer_sz = dma_desc_ct*SPI_MAX_DMA_LEN; + spihost[host]->dmadesc_tx=pvPortMallocCaps(sizeof(lldesc_t)*dma_desc_ct, MALLOC_CAP_DMA); + spihost[host]->dmadesc_rx=pvPortMallocCaps(sizeof(lldesc_t)*dma_desc_ct, MALLOC_CAP_DMA); + if (!spihost[host]->dmadesc_tx || !spihost[host]->dmadesc_rx) goto nomem; } - periph_module_enable(io_signal[host].module); - esp_intr_alloc(io_signal[host].irq, ESP_INTR_FLAG_INTRDISABLED, spi_intr, (void*)spihost[host], &spihost[host]->intr); - spihost[host]->hw=io_signal[host].hw; + esp_intr_alloc(spicommon_irqsource_for_host(host), ESP_INTR_FLAG_INTRDISABLED, spi_intr, (void*)spihost[host], &spihost[host]->intr); + spihost[host]->hw=spicommon_hw_for_host(host); //Reset DMA - spihost[host]->hw->dma_conf.val|=SPI_OUT_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST; + spihost[host]->hw->dma_conf.val|=SPI_OUT_RST|SPI_IN_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST; spihost[host]->hw->dma_out_link.start=0; spihost[host]->hw->dma_in_link.start=0; - spihost[host]->hw->dma_conf.val&=~(SPI_OUT_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST); + spihost[host]->hw->dma_conf.val&=~(SPI_OUT_RST|SPI_IN_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST); //Reset timing spihost[host]->hw->ctrl2.val=0; @@ -287,10 +156,16 @@ esp_err_t spi_bus_initialize(spi_host_device_t host, const spi_bus_config_t *bus spihost[host]->hw->slave.trans_inten=1; spihost[host]->hw->slave.trans_done=1; - //Select DMA channel. - SET_PERI_REG_BITS(DPORT_SPI_DMA_CHAN_SEL_REG, 3, dma_chan, (host * 2)); - return ESP_OK; + +nomem: + if (spihost[host]) { + free(spihost[host]->dmadesc_tx); + free(spihost[host]->dmadesc_rx); + } + free(spihost[host]); + spicommon_periph_free(host); + return ESP_ERR_NO_MEM; } esp_err_t spi_bus_free(spi_host_device_t host) @@ -304,7 +179,7 @@ esp_err_t spi_bus_free(spi_host_device_t host) spihost[host]->hw->slave.trans_inten=0; spihost[host]->hw->slave.trans_done=0; esp_intr_free(spihost[host]->intr); - periph_module_disable(io_signal[host].module); + spicommon_periph_free(host); free(spihost[host]); spihost[host]=NULL; return ESP_OK; @@ -336,13 +211,14 @@ esp_err_t spi_bus_add_device(spi_host_device_t host, spi_device_interface_config //Allocate memory for device spi_device_t *dev=malloc(sizeof(spi_device_t)); - if (dev==NULL) return ESP_ERR_NO_MEM; + if (dev==NULL) goto nomem; memset(dev, 0, sizeof(spi_device_t)); spihost[host]->device[freecs]=dev; //Allocate queues, set defaults dev->trans_queue=xQueueCreate(dev_config->queue_size, sizeof(spi_transaction_t *)); dev->ret_queue=xQueueCreate(dev_config->queue_size, sizeof(spi_transaction_t *)); + if (!dev->trans_queue || !dev->ret_queue) goto nomem; if (dev_config->duty_cycle_pos==0) dev_config->duty_cycle_pos=128; dev->host=spihost[host]; @@ -351,15 +227,8 @@ esp_err_t spi_bus_add_device(spi_host_device_t host, spi_device_interface_config //Set CS pin, CS options if (dev_config->spics_io_num > 0) { - if (spihost[host]->no_gpio_matrix &&dev_config->spics_io_num == io_signal[host].spics0_native && freecs==0) { - //Again, the cs0s for all SPI peripherals map to pin mux source 1, so we use that instead of a define. - PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[dev_config->spics_io_num], 1); - } else { - //Use GPIO matrix - PIN_FUNC_SELECT(GPIO_PIN_MUX_REG[dev_config->spics_io_num], PIN_FUNC_GPIO); - gpio_set_direction(dev_config->spics_io_num, GPIO_MODE_OUTPUT); - gpio_matrix_out(dev_config->spics_io_num, io_signal[host].spics_out[freecs], false, false); - } + gpio_set_direction(dev_config->spics_io_num, GPIO_MODE_OUTPUT); + spicommon_cs_initialize(host, dev_config->spics_io_num, freecs, spihost[host]->no_gpio_matrix == false); } if (dev_config->flags&SPI_DEVICE_CLK_AS_CS) { spihost[host]->hw->pin.master_ck_sel |= (1<trans_queue) vQueueDelete(dev->trans_queue); + if (dev->ret_queue) vQueueDelete(dev->ret_queue); + } + free(dev); + return ESP_ERR_NO_MEM; } esp_err_t spi_bus_remove_device(spi_device_handle_t handle) @@ -457,10 +334,6 @@ static int spi_set_clock(spi_dev_t *hw, int fapb, int hz, int duty_cycle) { } -//If a transaction is smaller than or equal to of bits, we do not use DMA; instead, we directly copy/paste -//bits from/to the work registers. Keep between 32 and (8*32) please. -#define THRESH_DMA_TRANS (8*32) - //This is run in interrupt context and apart from initialization and destruction, this is the only code //touching the host (=spihost[x]) variable. The rest of the data arrives in queues. That is why there are //no muxes in this code. @@ -478,7 +351,7 @@ static void IRAM_ATTR spi_intr(void *arg) if (host->cur_trans) { //Okay, transaction is done. - if ((host->cur_trans->rx_buffer || (host->cur_trans->flags & SPI_TRANS_USE_RXDATA)) && host->cur_trans->rxlength<=THRESH_DMA_TRANS) { + if ((host->cur_trans->rx_buffer || (host->cur_trans->flags & SPI_TRANS_USE_RXDATA)) && host->dma_chan == 0) { //Need to copy from SPI regs to result buffer. uint32_t *data; if (host->cur_trans->flags & SPI_TRANS_USE_RXDATA) { @@ -489,7 +362,9 @@ static void IRAM_ATTR spi_intr(void *arg) for (int x=0; x < host->cur_trans->rxlength; x+=32) { //Do a memcpy to get around possible alignment issues in rx_buffer uint32_t word=host->hw->data_buf[x/32]; - memcpy(&data[x/32], &word, 4); + int len=host->cur_trans->rxlength-x; + if (len>32) len=32; + memcpy(&data[x/32], &word, (len+7)/8); } } //Call post-transaction callback, if any @@ -499,6 +374,8 @@ static void IRAM_ATTR spi_intr(void *arg) host->cur_trans=NULL; prevCs=host->cur_cs; } + //Tell common code DMA workaround that our DMA channel is idle. If needed, the code will do a DMA reset. + if (host->dma_chan) spicommon_dmaworkaround_idle(host->dma_chan); //ToDo: This is a stupidly simple low-cs-first priority scheme. Make this configurable somehow. - JD for (i=0; idevice[i]) { @@ -523,7 +400,7 @@ static void IRAM_ATTR spi_intr(void *arg) if (trans->rxlength==0) { trans->rxlength=trans->length; } - + //Reconfigure according to device settings, but only if we change CSses. if (i!=prevCs) { //Assumes a hardcoded 80MHz Fapb for now. ToDo: figure out something better once we have @@ -590,12 +467,13 @@ static void IRAM_ATTR spi_intr(void *arg) host->hw->pin.cs1_dis=(i==1)?0:1; host->hw->pin.cs2_dis=(i==2)?0:1; } - //Reset DMA - host->hw->dma_conf.val |= SPI_OUT_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST; + //Reset SPI peripheral + host->hw->dma_conf.val |= SPI_OUT_RST|SPI_IN_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST; host->hw->dma_out_link.start=0; host->hw->dma_in_link.start=0; - host->hw->dma_conf.val &= ~(SPI_OUT_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST); - //QIO/DIO + host->hw->dma_conf.val &= ~(SPI_OUT_RST|SPI_IN_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST); + host->hw->dma_conf.out_data_burst_en=1; + //Set up QIO/DIO if needed host->hw->ctrl.val &= ~(SPI_FREAD_DUAL|SPI_FREAD_QUAD|SPI_FREAD_DIO|SPI_FREAD_QIO); host->hw->user.val &= ~(SPI_FWRITE_DUAL|SPI_FWRITE_QUAD|SPI_FWRITE_DIO|SPI_FWRITE_QIO); if (trans->flags & SPI_TRANS_MODE_DIO) { @@ -627,17 +505,13 @@ static void IRAM_ATTR spi_intr(void *arg) } else { data=trans->rx_buffer; } - if (trans->rxlength <= THRESH_DMA_TRANS) { - //No need for DMA; we'll copy the result out of the work registers directly later. + host->hw->user.usr_miso_highpart=0; + if (host->dma_chan == 0) { + //No need to setup anything; we'll copy the result out of the work registers directly later. } else { - host->hw->user.usr_miso_highpart=0; - host->dmadesc_rx.size=(trans->rxlength+7)/8; - host->dmadesc_rx.length=(trans->rxlength+7)/8; - host->dmadesc_rx.buf=(uint8_t*)data; - host->dmadesc_rx.eof=1; - host->dmadesc_rx.sosf=0; - host->dmadesc_rx.owner=1; - host->hw->dma_in_link.addr=(int)(&host->dmadesc_rx)&0xFFFFF; + spicommon_dmaworkaround_transfer_active(host->dma_chan); //mark channel as active + spicommon_setup_dma_desc_links(host->dmadesc_rx, ((trans->rxlength+7)/8), (uint8_t*)data, true); + host->hw->dma_in_link.addr=(int)(&host->dmadesc_rx[0]) & 0xFFFFF; host->hw->dma_in_link.start=1; } host->hw->user.usr_miso=1; @@ -652,8 +526,8 @@ static void IRAM_ATTR spi_intr(void *arg) } else { data=(uint32_t *)trans->tx_buffer; } - if (trans->length <= THRESH_DMA_TRANS) { - //No need for DMA. + if (host->dma_chan == 0) { + //Need to copy data to registers manually for (int x=0; x < trans->length; x+=32) { //Use memcpy to get around alignment issues for txdata uint32_t word; @@ -662,19 +536,17 @@ static void IRAM_ATTR spi_intr(void *arg) } host->hw->user.usr_mosi_highpart=1; } else { + spicommon_dmaworkaround_transfer_active(host->dma_chan); //mark channel as active + spicommon_setup_dma_desc_links(host->dmadesc_tx, (trans->length+7)/8, (uint8_t*)data, false); host->hw->user.usr_mosi_highpart=0; - host->dmadesc_tx.size=(trans->length+7)/8; - host->dmadesc_tx.length=(trans->length+7)/8; - host->dmadesc_tx.buf=(uint8_t*)data; - host->dmadesc_tx.eof=1; - host->dmadesc_tx.sosf=0; - host->dmadesc_tx.owner=1; - host->hw->dma_out_link.addr=(int)(&host->dmadesc_tx) & 0xFFFFF; + host->hw->dma_out_link.addr=(int)(&host->dmadesc_tx[0]) & 0xFFFFF; host->hw->dma_out_link.start=1; + host->hw->user.usr_mosi_highpart=0; } } host->hw->mosi_dlen.usr_mosi_dbitlen=trans->length-1; host->hw->miso_dlen.usr_miso_dbitlen=trans->rxlength-1; + host->hw->user2.usr_command_value=trans->command; if (dev->cfg.address_bits>32) { host->hw->addr=trans->address >> 32; @@ -702,6 +574,8 @@ esp_err_t spi_device_queue_trans(spi_device_handle_t handle, spi_transaction_t * SPI_CHECK((trans_desc->flags & SPI_TRANS_USE_TXDATA)==0 ||trans_desc->length <= 32, "txdata transfer > 32 bits", ESP_ERR_INVALID_ARG); SPI_CHECK(!((trans_desc->flags & (SPI_TRANS_MODE_DIO|SPI_TRANS_MODE_QIO)) && (handle->cfg.flags & SPI_DEVICE_3WIRE)), "incompatible iface params", ESP_ERR_INVALID_ARG); SPI_CHECK(!((trans_desc->flags & (SPI_TRANS_MODE_DIO|SPI_TRANS_MODE_QIO)) && (!(handle->cfg.flags & SPI_DEVICE_HALFDUPLEX))), "incompatible iface params", ESP_ERR_INVALID_ARG); + SPI_CHECK(trans_desc->length <= handle->host->max_transfer_sz*8, "txdata transfer > host maximum", ESP_ERR_INVALID_ARG); + SPI_CHECK(trans_desc->rxlength <= handle->host->max_transfer_sz*8, "rxdata transfer > host maximum", ESP_ERR_INVALID_ARG); r=xQueueSend(handle->trans_queue, (void*)&trans_desc, ticks_to_wait); if (!r) return ESP_ERR_TIMEOUT; esp_intr_enable(handle->host->intr); diff --git a/components/driver/spi_slave.c b/components/driver/spi_slave.c new file mode 100644 index 0000000000..f321799908 --- /dev/null +++ b/components/driver/spi_slave.c @@ -0,0 +1,394 @@ +// Copyright 2015-2016 Espressif Systems (Shanghai) PTE LTD +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at + +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +#include +#include "driver/spi_common.h" +#include "driver/spi_slave.h" +#include "soc/gpio_sig_map.h" +#include "soc/spi_reg.h" +#include "soc/dport_reg.h" +#include "soc/spi_struct.h" +#include "soc/rtc_cntl_reg.h" +#include "rom/ets_sys.h" +#include "esp_types.h" +#include "esp_attr.h" +#include "esp_intr.h" +#include "esp_intr_alloc.h" +#include "esp_log.h" +#include "esp_err.h" +#include "freertos/FreeRTOS.h" +#include "freertos/semphr.h" +#include "freertos/xtensa_api.h" +#include "freertos/task.h" +#include "freertos/ringbuf.h" +#include "soc/soc.h" +#include "soc/dport_reg.h" +#include "soc/uart_struct.h" +#include "rom/lldesc.h" +#include "driver/uart.h" +#include "driver/gpio.h" +#include "driver/periph_ctrl.h" +#include "esp_heap_alloc_caps.h" + +static const char *SPI_TAG = "spi_slave"; +#define SPI_CHECK(a, str, ret_val) \ + if (!(a)) { \ + ESP_LOGE(SPI_TAG,"%s(%d): %s", __FUNCTION__, __LINE__, str); \ + return (ret_val); \ + } + +#define VALID_HOST(x) (host>SPI_HOST && host<=VSPI_HOST) + +typedef struct { + spi_slave_interface_config_t cfg; + intr_handle_t intr; + spi_dev_t *hw; + spi_slave_transaction_t *cur_trans; + lldesc_t *dmadesc_tx; + lldesc_t *dmadesc_rx; + bool no_gpio_matrix; + int max_transfer_sz; + QueueHandle_t trans_queue; + QueueHandle_t ret_queue; + int dma_chan; +} spi_slave_t; + +static spi_slave_t *spihost[3]; + +static void IRAM_ATTR spi_intr(void *arg); + +esp_err_t spi_slave_initialize(spi_host_device_t host, spi_bus_config_t *bus_config, spi_slave_interface_config_t *slave_config, int dma_chan) +{ + bool native, claimed; + //We only support HSPI/VSPI, period. + SPI_CHECK(VALID_HOST(host), "invalid host", ESP_ERR_INVALID_ARG); + + claimed=spicommon_periph_claim(host); + SPI_CHECK(claimed, "host already in use", ESP_ERR_INVALID_STATE); + + spihost[host]=malloc(sizeof(spi_slave_t)); + if (spihost[host]==NULL) goto nomem; + memset(spihost[host], 0, sizeof(spi_slave_t)); + memcpy(&spihost[host]->cfg, slave_config, sizeof(spi_slave_interface_config_t)); + + spicommon_bus_initialize_io(host, bus_config, dma_chan, SPICOMMON_BUSFLAG_SLAVE, &native); + gpio_set_direction(slave_config->spics_io_num, GPIO_MODE_INPUT); + spicommon_cs_initialize(host, slave_config->spics_io_num, 0, native == false); + spihost[host]->no_gpio_matrix=native; + spihost[host]->dma_chan=dma_chan; + if (dma_chan!=0) { + //See how many dma descriptors we need and allocate them + int dma_desc_ct=(bus_config->max_transfer_sz+SPI_MAX_DMA_LEN-1)/SPI_MAX_DMA_LEN; + if (dma_desc_ct==0) dma_desc_ct=1; //default to 4k when max is not given + spihost[host]->max_transfer_sz = dma_desc_ct*SPI_MAX_DMA_LEN; + spihost[host]->dmadesc_tx=pvPortMallocCaps(sizeof(lldesc_t)*dma_desc_ct, MALLOC_CAP_DMA); + spihost[host]->dmadesc_rx=pvPortMallocCaps(sizeof(lldesc_t)*dma_desc_ct, MALLOC_CAP_DMA); + if (!spihost[host]->dmadesc_tx || !spihost[host]->dmadesc_rx) goto nomem; + } else { + //We're limited to non-DMA transfers: the SPI work registers can hold 64 bytes at most. + spihost[host]->max_transfer_sz=16*4; + } + + //Create queues + spihost[host]->trans_queue=xQueueCreate(slave_config->queue_size, sizeof(spi_slave_transaction_t *)); + spihost[host]->ret_queue=xQueueCreate(slave_config->queue_size, sizeof(spi_slave_transaction_t *)); + if (!spihost[host]->trans_queue || !spihost[host]->ret_queue) goto nomem; + + esp_intr_alloc(spicommon_irqsource_for_host(host), ESP_INTR_FLAG_INTRDISABLED, spi_intr, (void*)spihost[host], &spihost[host]->intr); + spihost[host]->hw=spicommon_hw_for_host(host); + + //Configure slave + spihost[host]->hw->clock.val=0; + spihost[host]->hw->user.val=0; + spihost[host]->hw->ctrl.val=0; + spihost[host]->hw->slave.wr_rd_buf_en=1; //no sure if needed + spihost[host]->hw->user.doutdin=1; //we only support full duplex + spihost[host]->hw->user.sio=0; + spihost[host]->hw->slave.slave_mode=1; + spihost[host]->hw->dma_conf.val |= SPI_OUT_RST|SPI_IN_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST; + spihost[host]->hw->dma_out_link.start=0; + spihost[host]->hw->dma_in_link.start=0; + spihost[host]->hw->dma_conf.val &= ~(SPI_OUT_RST|SPI_IN_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST); + spihost[host]->hw->dma_conf.out_data_burst_en=1; + spihost[host]->hw->slave.sync_reset=1; + spihost[host]->hw->slave.sync_reset=0; + + + bool nodelay=true; + spihost[host]->hw->ctrl.rd_bit_order=(slave_config->flags & SPI_SLAVE_RXBIT_LSBFIRST)?1:0; + spihost[host]->hw->ctrl.wr_bit_order=(slave_config->flags & SPI_SLAVE_TXBIT_LSBFIRST)?1:0; + if (slave_config->mode==0) { + spihost[host]->hw->pin.ck_idle_edge=0; + spihost[host]->hw->user.ck_i_edge=1; + spihost[host]->hw->ctrl2.miso_delay_mode=nodelay?0:2; + } else if (slave_config->mode==1) { + spihost[host]->hw->pin.ck_idle_edge=0; + spihost[host]->hw->user.ck_i_edge=0; + spihost[host]->hw->ctrl2.miso_delay_mode=nodelay?0:1; + } else if (slave_config->mode==2) { + spihost[host]->hw->pin.ck_idle_edge=1; + spihost[host]->hw->user.ck_i_edge=0; + spihost[host]->hw->ctrl2.miso_delay_mode=nodelay?0:1; + } else if (slave_config->mode==3) { + spihost[host]->hw->pin.ck_idle_edge=1; + spihost[host]->hw->user.ck_i_edge=1; + spihost[host]->hw->ctrl2.miso_delay_mode=nodelay?0:2; + } + + + //Reset DMA + spihost[host]->hw->dma_conf.val|=SPI_OUT_RST|SPI_IN_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST; + spihost[host]->hw->dma_out_link.start=0; + spihost[host]->hw->dma_in_link.start=0; + spihost[host]->hw->dma_conf.val&=~(SPI_OUT_RST|SPI_IN_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST); + + //Disable unneeded ints + spihost[host]->hw->slave.rd_buf_done=0; + spihost[host]->hw->slave.wr_buf_done=0; + spihost[host]->hw->slave.rd_sta_done=0; + spihost[host]->hw->slave.wr_sta_done=0; + spihost[host]->hw->slave.rd_buf_inten=0; + spihost[host]->hw->slave.wr_buf_inten=0; + spihost[host]->hw->slave.rd_sta_inten=0; + spihost[host]->hw->slave.wr_sta_inten=0; + + //Force a transaction done interrupt. This interrupt won't fire yet because we initialized the SPI interrupt as + //disabled. This way, we can just enable the SPI interrupt and the interrupt handler will kick in, handling + //any transactions that are queued. + spihost[host]->hw->slave.trans_inten=1; + spihost[host]->hw->slave.trans_done=1; + + return ESP_OK; + +nomem: + if (spihost[host]) { + if (spihost[host]->trans_queue) vQueueDelete(spihost[host]->trans_queue); + if (spihost[host]->ret_queue) vQueueDelete(spihost[host]->ret_queue); + free(spihost[host]->dmadesc_tx); + free(spihost[host]->dmadesc_rx); + } + free(spihost[host]); + spihost[host]=NULL; + spicommon_periph_free(host); + return ESP_ERR_NO_MEM; +} + +esp_err_t spi_slave_free(spi_host_device_t host) +{ + SPI_CHECK(VALID_HOST(host), "invalid host", ESP_ERR_INVALID_ARG); + SPI_CHECK(spihost[host], "host not slave", ESP_ERR_INVALID_ARG); + if (spihost[host]->trans_queue) vQueueDelete(spihost[host]->trans_queue); + if (spihost[host]->ret_queue) vQueueDelete(spihost[host]->ret_queue); + free(spihost[host]->dmadesc_tx); + free(spihost[host]->dmadesc_rx); + free(spihost[host]); + spihost[host]=NULL; + spicommon_periph_free(host); + spihost[host]=NULL; + return ESP_OK; +} + + +esp_err_t spi_slave_queue_trans(spi_host_device_t host, spi_slave_transaction_t *trans_desc, TickType_t ticks_to_wait) +{ + BaseType_t r; + SPI_CHECK(VALID_HOST(host), "invalid host", ESP_ERR_INVALID_ARG); + SPI_CHECK(spihost[host], "host not slave", ESP_ERR_INVALID_ARG); + + SPI_CHECK(trans_desc->length <= spihost[host]->max_transfer_sz*8, "data transfer > host maximum", ESP_ERR_INVALID_ARG); + r=xQueueSend(spihost[host]->trans_queue, (void*)&trans_desc, ticks_to_wait); + if (!r) return ESP_ERR_TIMEOUT; + esp_intr_enable(spihost[host]->intr); + return ESP_OK; +} + + +esp_err_t spi_slave_get_trans_result(spi_host_device_t host, spi_slave_transaction_t **trans_desc, TickType_t ticks_to_wait) +{ + BaseType_t r; + SPI_CHECK(VALID_HOST(host), "invalid host", ESP_ERR_INVALID_ARG); + SPI_CHECK(spihost[host], "host not slave", ESP_ERR_INVALID_ARG); + r=xQueueReceive(spihost[host]->ret_queue, (void*)trans_desc, ticks_to_wait); + if (!r) return ESP_ERR_TIMEOUT; + return ESP_OK; +} + + +esp_err_t spi_slave_transmit(spi_host_device_t host, spi_slave_transaction_t *trans_desc, TickType_t ticks_to_wait) +{ + esp_err_t ret; + spi_slave_transaction_t *ret_trans; + //ToDo: check if any spi transfers in flight + ret=spi_slave_queue_trans(host, trans_desc, ticks_to_wait); + if (ret!=ESP_OK) return ret; + ret=spi_slave_get_trans_result(host, &ret_trans, ticks_to_wait); + if (ret!=ESP_OK) return ret; + assert(ret_trans==trans_desc); + return ESP_OK; +} + +#ifdef DEBUG_SLAVE +static void dumpregs(spi_dev_t *hw) { + ets_printf("***REG DUMP ***\n"); + ets_printf("mosi_dlen : %08X\n", hw->mosi_dlen.val); + ets_printf("miso_dlen : %08X\n", hw->miso_dlen.val); + ets_printf("slv_wrbuf_dlen : %08X\n", hw->slv_wrbuf_dlen.val); + ets_printf("slv_rdbuf_dlen : %08X\n", hw->slv_rdbuf_dlen.val); + ets_printf("slave : %08X\n", hw->slave.val); + ets_printf("slv_rdata_bit : %x\n", hw->slv_rd_bit.slv_rdata_bit); + ets_printf("dma_rx_status : %08X\n", hw->dma_rx_status); + ets_printf("dma_tx_status : %08X\n", hw->dma_tx_status); +} + + +static void dumpll(lldesc_t *ll) { + ets_printf("****LL DUMP****\n"); + ets_printf("Size %d\n", ll->size); + ets_printf("Len: %d\n", ll->length); + ets_printf("Owner: %s\n", ll->owner?"dma":"cpu"); +} +#endif + +static void IRAM_ATTR spi_slave_restart_after_dmareset(void *arg) +{ + spi_slave_t *host=(spi_slave_t*)arg; + esp_intr_enable(host->intr); +} + +//This is run in interrupt context and apart from initialization and destruction, this is the only code +//touching the host (=spihost[x]) variable. The rest of the data arrives in queues. That is why there are +//no muxes in this code. +static void IRAM_ATTR spi_intr(void *arg) +{ + BaseType_t r; + BaseType_t do_yield=pdFALSE; + spi_slave_transaction_t *trans=NULL; + spi_slave_t *host=(spi_slave_t*)arg; + +#ifdef DEBUG_SLAVE + dumpregs(host->hw); + if (host->dmadesc_rx) dumpll(&host->dmadesc_rx[0]); +#endif + + //Ignore all but the trans_done int. + if (!host->hw->slave.trans_done) return; + + if (host->cur_trans) { + if (host->dma_chan == 0 && host->cur_trans->rx_buffer) { + //Copy result out + uint32_t *data=host->cur_trans->rx_buffer; + for (int x=0; xcur_trans->length; x+=32) { + uint32_t word; + int len=host->cur_trans->length-x; + if (len>32) len=32; + word=host->hw->data_buf[(x/32)]; + memcpy(&data[x/32], &word, (len+7)/8); + } + } else if (host->dma_chan != 0 && host->cur_trans->rx_buffer) { + int i; + //In case CS goes high too soon, the transfer is aborted while the DMA channel still thinks it's going. This + //leads to issues later on, so in that case we need to reset the channel. The state can be detected because + //the DMA system doesn't give back the offending descriptor; the owner is still set to DMA. + for (i=0; host->dmadesc_rx[i].eof==0 && host->dmadesc_rx[i].owner==0; i++) ; + if (host->dmadesc_rx[i].owner) { + spicommon_dmaworkaround_req_reset(host->dma_chan, spi_slave_restart_after_dmareset, host); + } + } + if (host->cfg.post_trans_cb) host->cfg.post_trans_cb(host->cur_trans); + //Okay, transaction is done. + //Return transaction descriptor. + xQueueSendFromISR(host->ret_queue, &host->cur_trans, &do_yield); + host->cur_trans=NULL; + } + if (host->dma_chan!=0) { + spicommon_dmaworkaround_idle(host->dma_chan); + if (spicommon_dmaworkaround_reset_in_progress()) { + //We need to wait for the reset to complete. Disable int (will be re-enabled on reset callback) and exit isr. + esp_intr_disable(host->intr); + if (do_yield) portYIELD_FROM_ISR(); + return; + } + } + + //Grab next transaction + r=xQueueReceiveFromISR(host->trans_queue, &trans, &do_yield); + if (!r) { + //No packet waiting. Disable interrupt. + esp_intr_disable(host->intr); + } else { + //We have a transaction. Send it. + host->hw->slave.trans_done=0; //clear int bit + host->cur_trans=trans; + + if (host->dma_chan != 0) { + spicommon_dmaworkaround_transfer_active(host->dma_chan); + host->hw->dma_conf.val |= SPI_OUT_RST|SPI_IN_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST; + host->hw->dma_out_link.start=0; + host->hw->dma_in_link.start=0; + host->hw->dma_conf.val &= ~(SPI_OUT_RST|SPI_IN_RST|SPI_AHBM_RST|SPI_AHBM_FIFO_RST); + host->hw->dma_conf.out_data_burst_en=0; + host->hw->dma_conf.indscr_burst_en=0; + host->hw->dma_conf.outdscr_burst_en=0; + + //Fill DMA descriptors + if (trans->rx_buffer) { + host->hw->user.usr_miso_highpart=0; + spicommon_setup_dma_desc_links(host->dmadesc_rx, ((trans->length+7)/8), trans->rx_buffer, true); + host->hw->dma_in_link.addr=(int)(&host->dmadesc_rx[0]) & 0xFFFFF; + host->hw->dma_in_link.start=1; + } + + if (trans->tx_buffer) { + spicommon_setup_dma_desc_links(host->dmadesc_tx, (trans->length+7)/8, trans->tx_buffer, false); + host->hw->user.usr_mosi_highpart=0; + host->hw->dma_out_link.addr=(int)(&host->dmadesc_tx[0]) & 0xFFFFF; + host->hw->dma_out_link.start=1; + } + + host->hw->slave.sync_reset=1; + host->hw->slave.sync_reset=0; + + } else { + //No DMA. Turn off SPI and copy data to transmit buffers. + host->hw->cmd.usr=0; + host->hw->slave.sync_reset=1; + host->hw->slave.sync_reset=0; + + host->hw->user.usr_miso_highpart=0; + host->hw->user.usr_mosi_highpart=0; + if (trans->tx_buffer) { + uint32_t *data=host->cur_trans->tx_buffer; + for (int x=0; xlength; x+=32) { + uint32_t word; + memcpy(&word, &data[x/32], 4); + host->hw->data_buf[(x/32)]=word; + } + } + } + + host->hw->slv_rd_bit.slv_rdata_bit=0; + host->hw->slv_wrbuf_dlen.bit_len=trans->length-1; + host->hw->slv_rdbuf_dlen.bit_len=trans->length-1; + host->hw->mosi_dlen.usr_mosi_dbitlen=trans->length-1; + host->hw->miso_dlen.usr_miso_dbitlen=trans->length-1; + host->hw->user.usr_mosi=(trans->tx_buffer==NULL)?0:1; + host->hw->user.usr_miso=(trans->rx_buffer==NULL)?0:1; + + //Kick off transfer + host->hw->cmd.usr=1; + if (host->cfg.post_setup_cb) host->cfg.post_setup_cb(trans); + } + if (do_yield) portYIELD_FROM_ISR(); +} + diff --git a/components/driver/test/test_spi_master.c b/components/driver/test/test_spi_master.c index f88427a81b..9a55f69990 100644 --- a/components/driver/test/test_spi_master.c +++ b/components/driver/test/test_spi_master.c @@ -18,6 +18,7 @@ #include "soc/dport_reg.h" #include "soc/spi_reg.h" #include "soc/spi_struct.h" +#include "esp_heap_alloc_caps.h" static void check_spi_pre_n_for(int clk, int pre, int n) @@ -60,7 +61,7 @@ TEST_CASE("SPI Master clockdiv calculation routines", "[spi]") { spi_bus_config_t buscfg={ .mosi_io_num=4, - .miso_io_num=16, + .miso_io_num=26, .sclk_io_num=25, .quadwp_io_num=-1, .quadhd_io_num=-1 @@ -69,81 +70,192 @@ TEST_CASE("SPI Master clockdiv calculation routines", "[spi]") ret=spi_bus_initialize(HSPI_HOST, &buscfg, 1); TEST_ASSERT(ret==ESP_OK); + check_spi_pre_n_for(26000000, 1, 3); + check_spi_pre_n_for(20000000, 1, 4); check_spi_pre_n_for(8000000, 1, 10); check_spi_pre_n_for(800000, 2, 50); check_spi_pre_n_for(100000, 16, 50); check_spi_pre_n_for(333333, 4, 60); - check_spi_pre_n_for(1, 8192, 64); //Actually should generate the minimum clock speed, 152Hz + check_spi_pre_n_for(900000, 4, 60); +// check_spi_pre_n_for(1, 8192, 64); //Actually should generate the minimum clock speed, 152Hz + check_spi_pre_n_for(26000000, 1, 3); ret=spi_bus_free(HSPI_HOST); TEST_ASSERT(ret==ESP_OK); } - -static void test_spi_bus_speed(int hz) { - esp_err_t ret; - spi_device_handle_t handle; +static spi_device_handle_t setup_spi_bus(int clkspeed, bool dma) { + spi_bus_config_t buscfg={ + .mosi_io_num=4, + .miso_io_num=26, + .sclk_io_num=25, + .quadwp_io_num=-1, + .quadhd_io_num=-1, + .max_transfer_sz=4096*3 + }; spi_device_interface_config_t devcfg={ - .command_bits=8, - .address_bits=64, + .command_bits=0, + .address_bits=0, .dummy_bits=0, - .clock_speed_hz=hz, + .clock_speed_hz=clkspeed, .duty_cycle_pos=128, .mode=0, .spics_io_num=21, .queue_size=3, }; + esp_err_t ret; + spi_device_handle_t handle; + printf("THIS TEST NEEDS A JUMPER BETWEEN IO4 AND IO26\n"); + ret=spi_bus_initialize(HSPI_HOST, &buscfg, dma?1:0); + TEST_ASSERT(ret==ESP_OK); ret=spi_bus_add_device(HSPI_HOST, &devcfg, &handle); TEST_ASSERT(ret==ESP_OK); printf("Bus/dev inited.\n"); - spi_transaction_t t; - char sendbuf[64]="Hello World!"; - char recvbuf[64]="UUUUUUUUUUUUUUU"; - memset(&t, 0, sizeof(t)); + return handle; +} - t.length=64*8; +static void spi_test(spi_device_handle_t handle, int num_bytes) { + esp_err_t ret; + int x; + srand(num_bytes); + char *sendbuf=pvPortMallocCaps(num_bytes, MALLOC_CAP_DMA); + char *recvbuf=pvPortMallocCaps(num_bytes, MALLOC_CAP_DMA); + for (x=0; x Sigma-delta Modulation SPI Master + SPI Slave Remote Control Timer UART diff --git a/docs/api/peripherals/spi_master.rst b/docs/api/peripherals/spi_master.rst index 78039cbd79..31fca39965 100644 --- a/docs/api/peripherals/spi_master.rst +++ b/docs/api/peripherals/spi_master.rst @@ -8,8 +8,7 @@ The ESP32 has four SPI peripheral devices, called SPI0, SPI1, HSPI and VSPI. SPI the flash cache the ESP32 uses to map the SPI flash device it is connected to into memory. SPI1 is connected to the same hardware lines as SPI0 and is used to write to the flash chip. HSPI and VSPI are free to use. SPI1, HSPI and VSPI all have three chip select lines, allowing them to drive up to -three SPI devices each as a master. The SPI peripherals also can be used in slave mode, driven from -another SPI master. +three SPI devices each as a master. The spi_master driver ^^^^^^^^^^^^^^^^^^^^^ @@ -74,7 +73,7 @@ Using the spi_master driver - Initialize a SPI bus by calling ``spi_bus_initialize``. Make sure to set the correct IO pins in the ``bus_config`` struct. Take care to set signals that are not needed to -1. -- Tell the driver about a SPI slave device conencted to the bus by calling spi_bus_add_device. +- Tell the driver about a SPI slave device connected to the bus by calling spi_bus_add_device. Make sure to configure any timing requirements the device has in the ``dev_config`` structure. You should now have a handle for the device, to be used when sending it a transaction. diff --git a/docs/api/peripherals/spi_slave.rst b/docs/api/peripherals/spi_slave.rst new file mode 100644 index 0000000000..87c181f0d2 --- /dev/null +++ b/docs/api/peripherals/spi_slave.rst @@ -0,0 +1,142 @@ +SPI Slave driver +================= + +Overview +-------- + +The ESP32 has four SPI peripheral devices, called SPI0, SPI1, HSPI and VSPI. SPI0 is entirely dedicated to +the flash cache the ESP32 uses to map the SPI flash device it is connected to into memory. SPI1 is +connected to the same hardware lines as SPI0 and is used to write to the flash chip. HSPI and VSPI +are free to use, and with the spi_slave driver, these can be used as a SPI slave, driven from a +connected SPI master. + +The spi_slave driver +^^^^^^^^^^^^^^^^^^^^^ + +The spi_slave driver allows using the HSPI and/or VSPI peripheral as a full-duplex SPI slave. It can make +use of DMA to send/receive transactions of arbitrary length. + +Terminology +^^^^^^^^^^^ + +The spi_slave driver uses the following terms: + +* Host: The SPI peripheral inside the ESP32 initiating the SPI transmissions. One of HSPI or VSPI. +* Bus: The SPI bus, common to all SPI devices connected to a master. In general the bus consists of the + miso, mosi, sclk and optionally quadwp and quadhd signals. The SPI slaves are connected to these + signals in parallel. Each SPI slave is also connected to one CS signal. + + - miso - Also known as q, this is the output of the serial stream from the ESP32 to the SPI master + + - mosi - Also known as d, this is the output of the serial stream from the SPI master to the ESP32 + + - sclk - Clock signal. Each data bit is clocked out or in on the positive or negative edge of this signal + + - cs - Chip Select. An active Chip Select delineates a single transaction to/from a slave. + +* Transaction: One instance of CS going active, data transfer from and to a master happening, and + CS going inactive again. Transactions are atomic, as in they will never be interrupted by another + transaction. + + +SPI transactions +^^^^^^^^^^^^^^^^ + +A full-duplex SPI transaction starts with the master pulling CS low. After this happens, the master +starts sending out clock pulses on the CLK line: every clock pulse causes a data bit to be shifted from +the master to the slave on the MOSI line and vice versa on the MISO line. At the end of the transaction, +the master makes CS high again. + +Using the spi_master driver +^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +- Initialize a SPI peripheral as a slave by calling ``spi_slave_initialize``. Make sure to set the + correct IO pins in the ``bus_config`` struct. Take care to set signals that are not needed to -1. + A DMA channel (either 1 or 2) must be given if transactions will be larger than 32 bytes, if not + the dma_chan parameter may be 0. + +- To set up a transaction, fill one or more spi_transaction_t structure with any transaction + parameters you need. Either queue all transactions by calling ``spi_slave_queue_trans``, later + quering the result using ``spi_slave_get_trans_result``, or handle all requests synchroneously + by feeding them into ``spi_slave_transmit``. The latter two functions will block until the + master has initiated and finished a transaction, causing the queued data to be sent and received. + +- Optional: to unload the SPI slave driver, call ``spi_slave_free``. + + +Transaction data and master/slave length mismatches +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Normally, data to be transferred to or from a device will be read from or written to a chunk of memory +indicated by the ``rx_buffer`` and ``tx_buffer`` members of the transaction structure. The SPI driver +may decide to use DMA for transfers, so these buffers should be allocated in DMA-capable memory using +``pvPortMallocCaps(size, MALLOC_CAP_DMA)``. + +The amount of data written to the buffers is limited by the ``length`` member of the transaction structure: +the driver will never read/write more data than indicated there. The ``length`` cannot define the actual +length of the SPI transaction; this is determined by the master as it drives the clock and CS lines. In +case the length of the transmission is larger than the buffer length, only the start of the transmission +will be sent and received. In case the transmission length is shorter than the buffer length, only data up +to the length of the buffer will be exchanged. + +Warning: Due to a design peculiarity in the V0 and V1 silicon of the ESP32, if the amount of bytes sent +by the master or the length of the transmission sent to the slave driver is not both larger than eight and +dividable by four, the SPI hardware can fail to write the last one to seven bytes to the receive buffer. + + +Application Example +------------------- + +Slave/master communication: :example:`peripherals/spi_slave`. + +API Reference +------------- + +Header Files +^^^^^^^^^^^^ + + * :component_file:`driver/include/driver/spi_slave.h` + +Macros +^^^^^^ + +.. doxygendefine:: SPI_SLAVE_TXBIT_LSBFIRST +.. doxygendefine:: SPI_SLAVE_RXBIT_LSBFIRST +.. doxygendefine:: SPI_SLAVE_BIT_LSBFIRST +.. doxygendefine:: SPI_SLAVE_POSITIVE_CS + + + +Enumerations +^^^^^^^^^^^^ + +.. doxygenenum:: spi_host_device_t + +Type Definitions +^^^^^^^^^^^^^^^^ + +Structures +^^^^^^^^^^ + +.. doxygenstruct:: spi_slave_transaction_t + :members: + +.. doxygenstruct:: spi_slave_interface_config_t + :members: + +.. doxygenstruct:: spi_bus_config_t + :members: + +Be advised that the slave driver does not use the quadwp/quadhd lines and fields in ``spi_bus_config_t`` refering to these lines +will be ignored and can thus safely be left uninitialized. + + +Functions +--------- + +.. doxygenfunction:: spi_slave_initialize +.. doxygenfunction:: spi_slave_free +.. doxygenfunction:: spi_slave_queue_trans +.. doxygenfunction:: spi_slave_get_trans_result +.. doxygenfunction:: spi_slave_transmit + diff --git a/examples/peripherals/spi_slave/README.md b/examples/peripherals/spi_slave/README.md new file mode 100644 index 0000000000..6e83f96ad9 --- /dev/null +++ b/examples/peripherals/spi_slave/README.md @@ -0,0 +1,6 @@ +## SPI slave example + +These two projects illustrate the SPI Slave driver. They're supposed to be flashed into two separate ESP32s connected to +eachother using the SPI pins defined in app_main.c. Once connected and flashed, they will use the spi master and spi slave +driver to communicate with eachother. The example also includes a handshaking line to allow the master to only poll the +slave when it is actually ready to parse a transaction. \ No newline at end of file diff --git a/examples/peripherals/spi_slave/receiver/Makefile b/examples/peripherals/spi_slave/receiver/Makefile new file mode 100644 index 0000000000..9f2c32783b --- /dev/null +++ b/examples/peripherals/spi_slave/receiver/Makefile @@ -0,0 +1,9 @@ +# +# This is a project Makefile. It is assumed the directory this Makefile resides in is a +# project subdirectory. +# + +PROJECT_NAME := app-template + +include $(IDF_PATH)/make/project.mk + diff --git a/examples/peripherals/spi_slave/receiver/main/app_main.c b/examples/peripherals/spi_slave/receiver/main/app_main.c new file mode 100644 index 0000000000..fe47cce15f --- /dev/null +++ b/examples/peripherals/spi_slave/receiver/main/app_main.c @@ -0,0 +1,142 @@ +/* SPI Slave example, receiver (uses SPI Slave driver to communicate with sender) + + This example code is in the Public Domain (or CC0 licensed, at your option.) + + Unless required by applicable law or agreed to in writing, this + software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR + CONDITIONS OF ANY KIND, either express or implied. +*/ +#include +#include +#include +#include + +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "freertos/semphr.h" +#include "freertos/queue.h" + +#include "lwip/sockets.h" +#include "lwip/dns.h" +#include "lwip/netdb.h" +#include "lwip/igmp.h" + +#include "esp_wifi.h" +#include "esp_system.h" +#include "esp_event.h" +#include "esp_event_loop.h" +#include "nvs_flash.h" +#include "soc/rtc_cntl_reg.h" +#include "rom/cache.h" +#include "driver/spi_slave.h" +#include "esp_log.h" +#include "esp_spi_flash.h" + + + + +/* +SPI receiver (slave) example. + +This example is supposed to work together with the SPI sender. It uses the standard SPI pins (MISO, MOSI, SCLK, CS) to +transmit data over in a full-duplex fashion, that is, while the master puts data on the MOSI pin, the slave puts its own +data on the MISO pin. + +This example uses one extra pin: GPIO_HANDSHAKE is used as a handshake pin. After a transmission has been set up and we're +ready to send/receive data, this code uses a callback to set the handshake pin high. The sender will detect this and start +sending a transaction. As soon as the transaction is done, the line gets set low again. +*/ + +/* +Pins in use. The SPI Master can use the GPIO mux, so feel free to change these if needed. +*/ +#define GPIO_HANDSHAKE 2 +#define GPIO_MOSI 12 +#define GPIO_MISO 13 +#define GPIO_SCLK 15 +#define GPIO_CS 14 + + +//Called after a transaction is queued and ready for pickup by master. We use this to set the handshake line high. +void my_post_setup_cb(spi_slave_transaction_t *trans) { + WRITE_PERI_REG(GPIO_OUT_W1TS_REG, (1< +#include +#include +#include + +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "freertos/semphr.h" +#include "freertos/queue.h" + +#include "lwip/sockets.h" +#include "lwip/dns.h" +#include "lwip/netdb.h" +#include "lwip/igmp.h" + +#include "esp_wifi.h" +#include "esp_system.h" +#include "esp_event.h" +#include "esp_event_loop.h" +#include "nvs_flash.h" +#include "soc/rtc_cntl_reg.h" +#include "rom/cache.h" +#include "driver/spi_master.h" +#include "esp_log.h" +#include "esp_spi_flash.h" + +#include "soc/gpio_reg.h" +#include "driver/gpio.h" +#include "esp_intr_alloc.h" + + +/* +SPI sender (master) example. + +This example is supposed to work together with the SPI receiver. It uses the standard SPI pins (MISO, MOSI, SCLK, CS) to +transmit data over in a full-duplex fashion, that is, while the master puts data on the MOSI pin, the slave puts its own +data on the MISO pin. + +This example uses one extra pin: GPIO_HANDSHAKE is used as a handshake pin. The slave makes this pin high as soon as it is +ready to receive/send data. This code connects this line to a GPIO interrupt which gives the rdySem semaphore. The main +task waits for this semaphore to be given before queueing a transmission. +*/ + + +/* +Pins in use. The SPI Master can use the GPIO mux, so feel free to change these if needed. +*/ +#define GPIO_HANDSHAKE 2 +#define GPIO_MOSI 12 +#define GPIO_MISO 13 +#define GPIO_SCLK 15 +#define GPIO_CS 14 + +//The semaphore indicating the slave is ready to receive stuff. +static xQueueHandle rdySem; + +/* +This ISR is called when the handshake line goes high. +*/ +static void IRAM_ATTR gpio_handshake_isr_handler(void* arg) +{ + //Sometimes due to interference or ringing or something, we get two irqs after eachother. This is solved by + //looking at the time between interrupts and refusing any interrupt too close to another one. + static uint32_t lasthandshaketime; + uint32_t currtime=xthal_get_ccount(); + uint32_t diff=currtime-lasthandshaketime; + if (diff<240000) return; //ignore everything <1ms after an earlier irq + lasthandshaketime=currtime; + + //Give the semaphore. + BaseType_t mustYield=false; + xSemaphoreGiveFromISR(rdySem, &mustYield); + if (mustYield) portYIELD_FROM_ISR(); +} + +//Main application +void app_main() +{ + esp_err_t ret; + spi_device_handle_t handle; + + //Configuration for the SPI bus + spi_bus_config_t buscfg={ + .mosi_io_num=GPIO_MOSI, + .miso_io_num=GPIO_MISO, + .sclk_io_num=GPIO_SCLK, + .quadwp_io_num=-1, + .quadhd_io_num=-1 + }; + + //Configuration for the SPI device on the other side of the bus + spi_device_interface_config_t devcfg={ + .command_bits=0, + .address_bits=0, + .dummy_bits=0, + .clock_speed_hz=5000000, + .duty_cycle_pos=128, //50% duty cycle + .mode=0, + .spics_io_num=GPIO_CS, + .cs_ena_posttrans=3, //Keep the CS low 3 cycles after transaction, to stop slave from missing the last bit when CS has less propagation delay than CLK + .queue_size=3 + }; + + //GPIO config for the handshake line. + gpio_config_t io_conf={ + .intr_type=GPIO_PIN_INTR_POSEDGE, + .mode=GPIO_MODE_INPUT, + .pull_up_en=1, + .pin_bit_mask=(1<