mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
456 lines
14 KiB
C
456 lines
14 KiB
C
/*
|
|
* Copyright (C) 2022 Gunar Schorcht
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup cpu_esp8266
|
|
* @ingroup drivers_periph_spi
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Low-level SPI driver implementation for ESP8266
|
|
*
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <string.h>
|
|
|
|
#include "esp_common.h"
|
|
#include "log.h"
|
|
|
|
#include "cpu.h"
|
|
#include "mutex.h"
|
|
#include "periph/spi.h"
|
|
#include "macros/units.h"
|
|
|
|
#include "esp_attr.h"
|
|
#include "gpio_arch.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
#include "esp/iomux_regs.h"
|
|
#include "esp8266/spi_register.h"
|
|
#include "esp8266/spi_struct.h"
|
|
|
|
#define SPI_DOUTDIN (BIT(0))
|
|
|
|
#define SPI_BLOCK_SIZE 64 /* number of bytes per SPI transfer */
|
|
|
|
/** structure which describes all properties of one SPI bus */
|
|
struct _spi_bus_t {
|
|
spi_dev_t* regs; /* pointer to register data struct of the SPI device */
|
|
mutex_t lock; /* mutex for each possible SPI interface */
|
|
bool initialized; /* interface already initialized */
|
|
bool pins_initialized; /* pins interface initialized */
|
|
};
|
|
|
|
static struct _spi_bus_t _spi[] = {
|
|
#ifdef SPI0_CTRL
|
|
{
|
|
.initialized = false,
|
|
.pins_initialized = false,
|
|
.lock = MUTEX_INIT
|
|
},
|
|
#endif
|
|
};
|
|
|
|
/*
|
|
* GPIOs that were once initialized as SPI interface pins can not be used
|
|
* afterwards for anything else. Therefore, SPI interfaces are not initialized
|
|
* until they are used for the first time. The *spi_init* function is just a
|
|
* dummy for source code compatibility. The initialization of an SPI interface
|
|
* is performed by the *_spi_init_internal* function, which is called either by
|
|
* the *spi_init_cs* function or the *spi_acquire* function when the interface
|
|
* is used for the first time.
|
|
*/
|
|
void IRAM_ATTR spi_init(spi_t bus)
|
|
{
|
|
assert(bus < SPI_NUMOF_MAX);
|
|
assert(bus < SPI_NUMOF);
|
|
|
|
if (spi_config[bus].ctrl == HSPI) {
|
|
_spi[bus].regs = &SPI1;
|
|
}
|
|
else {
|
|
LOG_TAG_ERROR("spi", "invalid SPI interface controller "
|
|
"used for SPI_DEV(%d)\n", bus);
|
|
}
|
|
return;
|
|
}
|
|
|
|
/* Internal initialization function when the interface is used the first time */
|
|
static void IRAM_ATTR _spi_init_internal(spi_t bus)
|
|
{
|
|
assert(bus < SPI_NUMOF);
|
|
|
|
/* avoid multiple initializations */
|
|
if (_spi[bus].initialized) {
|
|
return;
|
|
}
|
|
_spi[bus].initialized = true;
|
|
|
|
DEBUG("%s bus=%u\n", __func__, bus);
|
|
|
|
/* initialize pins */
|
|
spi_init_pins(bus);
|
|
|
|
/* check whether pins could be initialized, otherwise return, CS is not
|
|
initialized in spi_init_pins */
|
|
if (gpio_get_pin_usage(spi_config[bus].sck) != _SPI &&
|
|
gpio_get_pin_usage(spi_config[bus].miso) != _SPI &&
|
|
gpio_get_pin_usage(spi_config[bus].mosi) != _SPI &&
|
|
gpio_get_pin_usage(spi_config[bus].cs) != _SPI) {
|
|
return;
|
|
}
|
|
|
|
/* bring the bus into a defined state */
|
|
_spi[bus].regs->user.val = SPI_USR_MOSI | SPI_CK_I_EDGE | SPI_DOUTDIN |
|
|
SPI_CS_SETUP | SPI_CS_HOLD;
|
|
|
|
/* set byte order to little endian for read and write operations */
|
|
_spi[bus].regs->user.wr_byte_order = 0;
|
|
_spi[bus].regs->user.rd_byte_order = 0;
|
|
|
|
/* set bit order to most significant first for read and write operations */
|
|
_spi[bus].regs->ctrl.wr_bit_order = 0;
|
|
_spi[bus].regs->ctrl.rd_bit_order = 0;
|
|
|
|
/* reset all DIO or QIO flags */
|
|
_spi[bus].regs->ctrl.fread_qio = 0;
|
|
_spi[bus].regs->ctrl.fread_dio = 0;
|
|
_spi[bus].regs->ctrl.fread_quad = 0;
|
|
_spi[bus].regs->ctrl.fread_dual = 0;
|
|
|
|
/* disable fast read mode and write protection */
|
|
_spi[bus].regs->ctrl.fastrd_mode = 0;
|
|
|
|
/* acquire and release to set default parameters */
|
|
spi_acquire(bus, GPIO_UNDEF, SPI_MODE_0, SPI_CLK_1MHZ);
|
|
spi_release(bus);
|
|
}
|
|
|
|
void spi_init_pins(spi_t bus)
|
|
{
|
|
assert(bus < SPI_NUMOF);
|
|
|
|
/* call initialization of the SPI interface if it is not initialized yet */
|
|
if (!_spi[bus].initialized) {
|
|
_spi_init_internal(bus);
|
|
}
|
|
|
|
/* avoid multiple pin initializations */
|
|
if (_spi[bus].pins_initialized) {
|
|
return;
|
|
}
|
|
_spi[bus].pins_initialized = true;
|
|
|
|
DEBUG("%s bus=%u\n", __func__, bus);
|
|
|
|
if (gpio_init(spi_config[bus].sck, GPIO_OUT) ||
|
|
gpio_init(spi_config[bus].mosi, GPIO_OUT) ||
|
|
gpio_init(spi_config[bus].miso, GPIO_IN)) {
|
|
LOG_TAG_ERROR("spi",
|
|
"SPI_DEV(%d) pins could not be initialized\n", bus);
|
|
return;
|
|
}
|
|
if (spi_init_cs(bus, spi_config[bus].cs) != SPI_OK) {
|
|
LOG_TAG_ERROR("spi",
|
|
"SPI_DEV(%d) CS signal could not be initialized\n",
|
|
bus);
|
|
return;
|
|
}
|
|
|
|
/* store the usage type in GPIO table */
|
|
gpio_set_pin_usage(spi_config[bus].sck, _SPI);
|
|
gpio_set_pin_usage(spi_config[bus].mosi, _SPI);
|
|
gpio_set_pin_usage(spi_config[bus].miso, _SPI);
|
|
|
|
/*
|
|
* CS is handled as normal GPIO output. Due to the small number of GPIOs
|
|
* we have, we do not initialize the default CS pin here. Either the app
|
|
* uses spi_init_cs to initialize the CS pin explicitly, or we initialize
|
|
* the default CS when spi_aquire is used first time.
|
|
*/
|
|
uint32_t iomux_func = IOMUX_FUNC(2);
|
|
|
|
IOMUX.PIN[_gpio_to_iomux[spi_config[bus].miso]] &= ~IOMUX_PIN_FUNC_MASK;
|
|
IOMUX.PIN[_gpio_to_iomux[spi_config[bus].mosi]] &= ~IOMUX_PIN_FUNC_MASK;
|
|
IOMUX.PIN[_gpio_to_iomux[spi_config[bus].sck]] &= ~IOMUX_PIN_FUNC_MASK;
|
|
|
|
IOMUX.PIN[_gpio_to_iomux[spi_config[bus].miso]] |= iomux_func;
|
|
IOMUX.PIN[_gpio_to_iomux[spi_config[bus].mosi]] |= iomux_func;
|
|
IOMUX.PIN[_gpio_to_iomux[spi_config[bus].sck]] |= iomux_func;
|
|
}
|
|
|
|
int spi_init_cs(spi_t bus, spi_cs_t cs)
|
|
{
|
|
DEBUG("%s bus=%u cs=%u\n", __func__, bus, cs);
|
|
|
|
assert(bus < SPI_NUMOF);
|
|
|
|
/* call initialization of the SPI interface if it is not initialized yet */
|
|
if (!_spi[bus].initialized) {
|
|
_spi_init_internal(bus);
|
|
}
|
|
|
|
/* return if pin is already initialized as SPI CS signal */
|
|
if (gpio_get_pin_usage(cs) == _SPI) {
|
|
return SPI_OK;
|
|
}
|
|
|
|
/* check whether CS pin is used otherwise */
|
|
if (gpio_get_pin_usage(cs) != _GPIO) {
|
|
return SPI_NOCS;
|
|
}
|
|
|
|
/* initialize the pin */
|
|
gpio_init(cs, GPIO_OUT);
|
|
gpio_set(cs);
|
|
|
|
/* pin cannot be used for anything else */
|
|
gpio_set_pin_usage(cs, _SPI);
|
|
|
|
return SPI_OK;
|
|
}
|
|
|
|
void IRAM_ATTR spi_acquire(spi_t bus, spi_cs_t cs, spi_mode_t mode, spi_clk_t clk)
|
|
{
|
|
DEBUG("%s bus=%u cs=%u mode=%u clk=%u\n", __func__, bus, cs, mode, clk);
|
|
|
|
assert(bus < SPI_NUMOF);
|
|
|
|
/* call initialization of the SPI interface if it is not initialized yet */
|
|
if (!_spi[bus].initialized) {
|
|
_spi_init_internal(bus);
|
|
}
|
|
|
|
/* if parameter cs is GPIO_UNDEF, the default CS pin is used */
|
|
cs = (cs == GPIO_UNDEF) ? spi_config[bus].cs : cs;
|
|
|
|
/* if the CS pin used is not yet initialized, we do it now */
|
|
if (gpio_get_pin_usage(cs) != _SPI && spi_init_cs(bus, cs) != SPI_OK) {
|
|
LOG_TAG_ERROR("spi",
|
|
"SPI_DEV(%d) CS signal could not be initialized\n",
|
|
bus);
|
|
assert(0);
|
|
}
|
|
|
|
/* lock the bus */
|
|
mutex_lock(&_spi[bus].lock);
|
|
|
|
/*
|
|
* set SPI mode
|
|
* see ESP32 Technical Reference, Table 27 and Section 7.4.1
|
|
*/
|
|
_spi[bus].regs->pin.ck_idle_edge = (mode == SPI_MODE_2 || mode == SPI_MODE_3);
|
|
_spi[bus].regs->user.ck_out_edge = (mode == SPI_MODE_1 || mode == SPI_MODE_2);
|
|
_spi[bus].regs->ctrl2.miso_delay_mode = (mode == SPI_MODE_0 || mode == SPI_MODE_3) ? 2 : 1;
|
|
_spi[bus].regs->ctrl2.miso_delay_num = 0;
|
|
_spi[bus].regs->ctrl2.mosi_delay_mode = 0;
|
|
_spi[bus].regs->ctrl2.mosi_delay_num = 0;
|
|
|
|
/* set SPI clock
|
|
* see ESP8266 Technical Reference Appendix 2 - SPI registers
|
|
* https://www.espressif.com/sites/default/files/documentation/esp8266-technical_reference_en.pdf
|
|
*/
|
|
|
|
uint32_t spi_clkdiv_pre;
|
|
uint32_t spi_clkcnt_N;
|
|
|
|
switch (clk) {
|
|
case SPI_CLK_10MHZ: spi_clkdiv_pre = 2; /* predivides 80 MHz to 40 MHz */
|
|
spi_clkcnt_N = 4; /* 4 cycles results into 10 MHz */
|
|
break;
|
|
case SPI_CLK_5MHZ: spi_clkdiv_pre = 2; /* predivides 80 MHz to 40 MHz */
|
|
spi_clkcnt_N = 8; /* 8 cycles results into 5 MHz */
|
|
break;
|
|
case SPI_CLK_1MHZ: spi_clkdiv_pre = 2; /* predivides 80 MHz to 40 MHz */
|
|
spi_clkcnt_N = 40; /* 40 cycles results into 1 MHz */
|
|
break;
|
|
case SPI_CLK_400KHZ: spi_clkdiv_pre = 20; /* predivides 80 MHz to 4 MHz */
|
|
spi_clkcnt_N = 10; /* 10 cycles results into 400 kHz */
|
|
break;
|
|
case SPI_CLK_100KHZ: spi_clkdiv_pre = 20; /* predivides 80 MHz to 4 MHz */
|
|
spi_clkcnt_N = 40; /* 20 cycles results into 100 kHz */
|
|
break;
|
|
default: spi_clkdiv_pre = 20; /* predivides 80 MHz to 4 MHz */
|
|
spi_clkcnt_N = 40; /* 20 cycles results into 100 kHz */
|
|
}
|
|
|
|
/* register values are set to deviders-1 */
|
|
spi_clkdiv_pre--;
|
|
spi_clkcnt_N--;
|
|
|
|
DEBUG("%s spi_clkdiv_prev=%u spi_clkcnt_N=%u\n",
|
|
__func__, spi_clkdiv_pre, spi_clkcnt_N);
|
|
|
|
IOMUX.CONF &= ~IOMUX_CONF_SPI1_CLOCK_EQU_SYS_CLOCK;
|
|
|
|
/* SPI clock is derived from APB clock by dividers */
|
|
_spi[bus].regs->clock.clk_equ_sysclk = 0;
|
|
|
|
/* set SPI clock dividers */
|
|
_spi[bus].regs->clock.clkdiv_pre = spi_clkdiv_pre;
|
|
_spi[bus].regs->clock.clkcnt_n = spi_clkcnt_N;
|
|
_spi[bus].regs->clock.clkcnt_h = (spi_clkcnt_N+1)/2-1;
|
|
_spi[bus].regs->clock.clkcnt_l = spi_clkcnt_N;
|
|
|
|
DEBUG("%s bus %d: SPI_CLOCK_REG=%08x\n",
|
|
__func__, bus, _spi[bus].regs->clock.val);
|
|
}
|
|
|
|
void IRAM_ATTR spi_release(spi_t bus)
|
|
{
|
|
DEBUG("%s bus=%u\n", __func__, bus);
|
|
|
|
assert(bus < SPI_NUMOF);
|
|
|
|
/* release the bus */
|
|
mutex_unlock(&_spi[bus].lock);
|
|
}
|
|
|
|
static const char* _spi_names[] = { "FSPI", "HSPI" };
|
|
|
|
void spi_print_config(void)
|
|
{
|
|
for (unsigned bus = 0; bus < SPI_NUMOF; bus++) {
|
|
printf("\tSPI_DEV(%u)\t%s ", bus, _spi_names[spi_config[bus].ctrl]);
|
|
printf("sck=%d ", spi_config[bus].sck);
|
|
printf("miso=%d ", spi_config[bus].miso);
|
|
printf("mosi=%d ", spi_config[bus].mosi);
|
|
printf("cs=%d\n", spi_config[bus].cs);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Following functions are from the hardware SPI driver of the esp-open-rtos
|
|
* project.
|
|
*
|
|
* Copyright (c) Ruslan V. Uss, 2016
|
|
* BSD Licensed as described in the file LICENSE
|
|
* https://github.com/SuperHouse/esp-open-rtos/blob/master/LICENSE
|
|
*/
|
|
|
|
inline static void IRAM_ATTR _set_size(uint8_t bus, uint8_t bytes)
|
|
{
|
|
uint32_t bits = ((uint32_t)bytes << 3) - 1;
|
|
|
|
_spi[bus].regs->user1.usr_mosi_bitlen = bits;
|
|
_spi[bus].regs->user1.usr_miso_bitlen = bits;
|
|
}
|
|
|
|
inline static void IRAM_ATTR _wait(uint8_t bus)
|
|
{
|
|
/* SPI_CMD_REG.SPI_USR is cleared when operation has been finished */
|
|
while (_spi[bus].regs->cmd.usr) {}
|
|
}
|
|
|
|
inline static void IRAM_ATTR _start(uint8_t bus)
|
|
{
|
|
/* set SPI_CMD_REG.SPI_USR to start an operation */
|
|
_spi[bus].regs->cmd.usr = 1;
|
|
}
|
|
|
|
inline static void IRAM_ATTR _store_data(uint8_t bus, const void *data, size_t len)
|
|
{
|
|
uint8_t words = len / 4;
|
|
uint8_t tail = len % 4;
|
|
|
|
memcpy((void *)_spi[bus].regs->data_buf, data, len - tail);
|
|
|
|
if (!tail) {
|
|
return;
|
|
}
|
|
|
|
uint32_t last = 0;
|
|
uint8_t *offs = (uint8_t *)data + len - tail;
|
|
for (uint8_t i = 0; i < tail; i++) {
|
|
last = last | (offs[i] << (i * 8));
|
|
}
|
|
_spi[bus].regs->data_buf[words] = last;
|
|
}
|
|
|
|
static const uint8_t spi_empty_out[SPI_BLOCK_SIZE] = { 0 };
|
|
|
|
static void IRAM_ATTR _spi_buf_transfer(uint8_t bus, const void *out, void *in, size_t len)
|
|
{
|
|
DEBUG("%s bus=%u out=%p in=%p len=%u\n", __func__, bus, out, in, len);
|
|
|
|
/* transfer one block data */
|
|
_wait(bus);
|
|
_set_size(bus, len);
|
|
_store_data(bus, out ? out : spi_empty_out, len);
|
|
_start(bus);
|
|
_wait(bus);
|
|
if (in) {
|
|
memcpy(in, (void *)_spi[bus].regs->data_buf, len);
|
|
}
|
|
}
|
|
|
|
void IRAM_ATTR spi_transfer_bytes(spi_t bus, spi_cs_t cs, bool cont,
|
|
const void *out, void *in, size_t len)
|
|
{
|
|
assert(bus < SPI_NUMOF);
|
|
|
|
DEBUG("%s bus=%u cs=%u cont=%d out=%p in=%p len=%u\n",
|
|
__func__, bus, cs, cont, out, in, len);
|
|
|
|
if (!len) {
|
|
return;
|
|
}
|
|
|
|
if (IS_ACTIVE(ENABLE_DEBUG)) {
|
|
if (out) {
|
|
DEBUG("out = ");
|
|
for (size_t i = 0; i < len; i++) {
|
|
DEBUG("%02x ", ((const uint8_t *)out)[i]);
|
|
}
|
|
DEBUG("\n");
|
|
}
|
|
}
|
|
|
|
if (cs != SPI_CS_UNDEF) {
|
|
gpio_clear(cs);
|
|
}
|
|
|
|
size_t blocks = len / SPI_BLOCK_SIZE;
|
|
uint8_t tail = len % SPI_BLOCK_SIZE;
|
|
|
|
DEBUG("%s bus=%u cs=%u blocks=%d tail=%d\n",
|
|
__func__, bus, cs, blocks, tail);
|
|
|
|
for (size_t i = 0; i < blocks; i++) {
|
|
_spi_buf_transfer(bus,
|
|
out ? (const uint8_t *)out + i * SPI_BLOCK_SIZE : NULL,
|
|
in ? (uint8_t *)in + i * SPI_BLOCK_SIZE : NULL, SPI_BLOCK_SIZE);
|
|
}
|
|
|
|
if (tail) {
|
|
_spi_buf_transfer(bus,
|
|
out ? (const uint8_t *)out + blocks * SPI_BLOCK_SIZE : 0,
|
|
in ? (uint8_t *)in + blocks * SPI_BLOCK_SIZE : NULL, tail);
|
|
}
|
|
|
|
if (!cont && (cs != SPI_CS_UNDEF)) {
|
|
gpio_set (cs);
|
|
}
|
|
|
|
if (IS_ACTIVE(ENABLE_DEBUG)) {
|
|
if (in) {
|
|
DEBUG("in = ");
|
|
for (size_t i = 0; i < len; i++) {
|
|
DEBUG("%02x ", ((const uint8_t *)in)[i]);
|
|
}
|
|
DEBUG("\n");
|
|
}
|
|
}
|
|
}
|