|
|
// Copyright 2020-2021 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 <stdio.h>
#include <string.h>
#include "esp_log.h"
#include "freertos/FreeRTOS.h"
#include "freertos/semphr.h"
#include "driver/spi_master.h"
#include "driver/spi_common.h"
#include "spi_bus.h"
typedef struct { spi_host_device_t host_id; /*!<spi device number */ bool is_init; spi_bus_config_t conf; /*!<spi bus active configuration */ } _spi_bus_t;
typedef struct { spi_device_handle_t handle; spi_bus_handle_t spi_bus; /*!<spi bus handle */ spi_device_interface_config_t conf; /*!<spi device active configuration */ SemaphoreHandle_t mutex; /* mutex to achive device thread-safe*/ } _spi_device_t;
static const char *TAG = "spi_bus"; static _spi_bus_t s_spi_bus[2]; #define ESP_SPI_MUTEX_TICKS_TO_WAIT 2
#define SPI_BUS_CHECK(a, str, ret) if(!(a)) { \
ESP_LOGE(TAG,"%s:%d (%s):%s", __FILE__, __LINE__, __FUNCTION__, str); \ return (ret); \ }
#define SPI_BUS_CHECK_GOTO(a, str, lable) if(!(a)) { \
ESP_LOGE(TAG,"%s:%d (%s):%s", __FILE__, __LINE__, __FUNCTION__, str); \ goto lable; \ }
#define SPI_DEVICE_MUTEX_TAKE(p_spi_dev, ret) if (!xSemaphoreTake((p_spi_dev)->mutex, ESP_SPI_MUTEX_TICKS_TO_WAIT)) { \
ESP_LOGE(TAG, "spi device(%d) take mutex timeout, max wait = %d ticks", (int32_t)((p_spi_dev)->handle), ESP_SPI_MUTEX_TICKS_TO_WAIT); \ return (ret); \ }
#define SPI_DEVICE_MUTEX_GIVE(p_spi_dev, ret) if (!xSemaphoreGive((p_spi_dev)->mutex)) { \
ESP_LOGE(TAG, "spi device(%d) give mutex failed", (int32_t)((p_spi_dev)->handle)); \ return (ret); \ }
spi_bus_handle_t spi_bus_create(spi_host_device_t host_id, const spi_config_t *bus_conf) { SPI_BUS_CHECK(SPI1_HOST < host_id && host_id <= SPI3_HOST, "Invalid spi host_id", NULL); uint8_t index = host_id - 1; //find related index
spi_bus_config_t buscfg = { .miso_io_num = bus_conf->miso_io_num, .mosi_io_num = bus_conf->mosi_io_num, .sclk_io_num = bus_conf->sclk_io_num, .quadwp_io_num = -1, .quadhd_io_num = -1, .max_transfer_sz = bus_conf->max_transfer_sz, }; #if (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(4, 3, 0))
esp_err_t ret = spi_bus_initialize(host_id, &buscfg, SPI_DMA_CH_AUTO); #else
int dma_chan = host_id; //set dma channel equals to host_id by default
esp_err_t ret = spi_bus_initialize(host_id, &buscfg, dma_chan); #endif
SPI_BUS_CHECK(ESP_OK == ret, "spi bus create failed", NULL); s_spi_bus[index].host_id = host_id; memcpy(&s_spi_bus[index].conf, &buscfg, sizeof(spi_bus_config_t)); s_spi_bus[index].is_init = true; ESP_LOGI(TAG, "SPI%d bus created", host_id + 1); return (spi_bus_handle_t)&s_spi_bus[index]; }
esp_err_t spi_bus_delete(spi_bus_handle_t *p_bus_handle) { SPI_BUS_CHECK((NULL != p_bus_handle) && (NULL != *p_bus_handle), "Handle error", ESP_ERR_INVALID_ARG); _spi_bus_t *spi_bus = (_spi_bus_t *)(*p_bus_handle);
if (!spi_bus->is_init) { ESP_LOGW(TAG, "spi_bus%d has been de-inited", spi_bus->host_id); return ESP_ERR_INVALID_STATE; }
esp_err_t ret = spi_bus_free(spi_bus->host_id); SPI_BUS_CHECK(ESP_OK == ret, "spi bus delete failed", ESP_FAIL); ESP_LOGI(TAG, "SPI%d bus delete", spi_bus->host_id + 1); memset(spi_bus, 0, sizeof(_spi_bus_t)); *p_bus_handle = NULL; return ESP_OK; }
spi_bus_device_handle_t spi_bus_device_create(spi_bus_device_handle_t bus_handle, const spi_device_config_t *device_conf) { SPI_BUS_CHECK(NULL != bus_handle, "Pointer error", NULL); _spi_bus_t *spi_bus = (_spi_bus_t *)bus_handle;
_spi_device_t *spi_dev = malloc(sizeof(_spi_device_t)); spi_device_interface_config_t devcfg = { .command_bits = 0, .address_bits = 0, .dummy_bits = 0, .clock_speed_hz = device_conf->clock_speed_hz, .duty_cycle_pos = 128, //50% duty cycle
.mode = device_conf->mode, .spics_io_num = device_conf->cs_io_num, .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 }; esp_err_t ret = spi_bus_add_device(spi_bus->host_id, &devcfg, &spi_dev->handle); SPI_BUS_CHECK_GOTO(ESP_OK == ret, "add spi device failed", cleanup_device); spi_dev->mutex = xSemaphoreCreateMutex(); SPI_BUS_CHECK_GOTO(NULL != spi_dev->mutex, "spi device create mutex failed", cleanup_device); spi_dev->spi_bus = bus_handle; memcpy(&spi_dev->conf, &devcfg, sizeof(spi_device_interface_config_t)); ESP_LOGI(TAG, "SPI%d bus device added, CS=%d Mode=%u Speed=%d", spi_bus->host_id + 1, device_conf->cs_io_num, device_conf->mode, device_conf->clock_speed_hz); return (spi_bus_device_handle_t)spi_dev;
cleanup_device: free(spi_dev); return NULL; }
esp_err_t spi_bus_device_delete(spi_bus_device_handle_t *p_dev_handle) { SPI_BUS_CHECK((NULL != p_dev_handle) && (NULL != *p_dev_handle), "Pointer error", ESP_ERR_INVALID_ARG); _spi_device_t *spi_dev = (_spi_device_t *)(*p_dev_handle); _spi_bus_t *spi_bus = (_spi_bus_t *)(spi_dev->spi_bus); SPI_DEVICE_MUTEX_TAKE(spi_dev, ESP_FAIL); esp_err_t ret = spi_bus_remove_device(spi_dev->handle); SPI_DEVICE_MUTEX_GIVE(spi_dev, ESP_FAIL); SPI_BUS_CHECK(ESP_OK == ret, "spi bus delete device failed", ret); vSemaphoreDelete(spi_dev->mutex); ESP_LOGI(TAG, "SPI%d device removed, CS=%d", spi_bus->host_id + 1, spi_dev->conf.spics_io_num); free(spi_dev); *p_dev_handle = NULL; return ESP_OK; }
/* this function should lable with inline*/ inline static esp_err_t _spi_device_polling_transmit(spi_bus_device_handle_t dev_handle, spi_transaction_t *trans) { SPI_BUS_CHECK(NULL != dev_handle, "Pointer error", ESP_ERR_INVALID_ARG); _spi_device_t *spi_dev = (_spi_device_t *)(dev_handle); esp_err_t ret; SPI_DEVICE_MUTEX_TAKE(spi_dev, ESP_FAIL); ret = spi_device_polling_transmit(spi_dev->handle, trans); SPI_DEVICE_MUTEX_GIVE(spi_dev, ESP_FAIL); return ret; }
esp_err_t spi_bus_transfer_byte(spi_bus_device_handle_t dev_handle, uint8_t data_out, uint8_t *data_in) { esp_err_t ret; spi_transaction_t trans = { .length = 8, .flags = SPI_TRANS_USE_RXDATA | SPI_TRANS_USE_TXDATA, .tx_data = { [0] = data_out } }; ret = _spi_device_polling_transmit(dev_handle, &trans); SPI_BUS_CHECK(ret == ESP_OK, "spi transfer byte failed", ret);
if (data_in) { *data_in = trans.rx_data[0]; }
return ESP_OK; }
esp_err_t spi_bus_transfer_bytes(spi_bus_device_handle_t dev_handle, const uint8_t *data_out, uint8_t *data_in, uint32_t data_len) { esp_err_t ret; #if 1
#define MIN(a,b) (((a)<(b))?(a):(b))
uint32_t remain = data_len; while (remain > 0) { uint32_t chunk_len = MIN(remain, 2048); spi_transaction_t trans = { .length = chunk_len * 8, .tx_buffer = NULL, .rx_buffer = NULL };
if (data_out) { trans.tx_buffer = data_out+(data_len-remain); }
if (data_in) { trans.rx_buffer = data_in+(data_len-remain); } ret = _spi_device_polling_transmit(dev_handle, &trans); SPI_BUS_CHECK(ret == ESP_OK, "spi transfer bytes failed", ret); remain -= chunk_len; } #else
spi_transaction_t trans = { .length = data_len * 8, .tx_buffer = NULL, .rx_buffer = NULL };
if (data_out) { trans.tx_buffer = data_out; }
if (data_in) { trans.rx_buffer = data_in; }
ret = _spi_device_polling_transmit(dev_handle, &trans); SPI_BUS_CHECK(ret == ESP_OK, "spi transfer bytes failed", ret);
#endif
return ESP_OK; }
/**************************************** Public Functions (Low level)*********************************************/
esp_err_t spi_bus_transmit_begin(spi_bus_device_handle_t dev_handle, spi_transaction_t *p_trans) { return _spi_device_polling_transmit(dev_handle, p_trans); }
esp_err_t spi_bus_transfer_reg16(spi_bus_device_handle_t dev_handle, uint16_t data_out, uint16_t *data_in) { esp_err_t ret; spi_transaction_t trans = { .length = 16, .flags = SPI_TRANS_USE_RXDATA | SPI_TRANS_USE_TXDATA, /* default MSB first */ .tx_data = { [0] = (data_out >> 8) & 0xff, [1] = data_out & 0xff, } }; ret = _spi_device_polling_transmit(dev_handle, &trans); SPI_BUS_CHECK(ret == ESP_OK, "spi transfer reg16 failed", ret);
if (data_in) { *data_in = (trans.rx_data[0] << 8) | (trans.rx_data[1]); }
return ESP_OK; }
esp_err_t spi_bus_transfer_reg32(spi_bus_device_handle_t dev_handle, uint32_t data_out, uint32_t *data_in) { esp_err_t ret; spi_transaction_t trans = { .length = 32, .flags = SPI_TRANS_USE_RXDATA | SPI_TRANS_USE_TXDATA, /* default MSB first */ .tx_data = { [0] = (data_out >> 24) & 0xff, [1] = (data_out >> 16) & 0xff, [2] = (data_out >> 8) & 0xff, [3] = data_out & 0xff } }; ret = _spi_device_polling_transmit(dev_handle, &trans); SPI_BUS_CHECK(ret == ESP_OK, "spi transfer reg32 failed", ret);
if (data_in) { *data_in = (trans.rx_data[0] << 24) | (trans.rx_data[1] << 16) | (trans.rx_data[2] << 8) | (trans.rx_data[3]); }
return ESP_OK; }
|