2017-11-13 16:08:08 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014-2016 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2017-11-29 19:50:50 +01:00
|
|
|
* @ingroup cpu_efm32
|
2017-11-13 16:08:08 +01:00
|
|
|
* @ingroup drivers_periph_spi
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Low-level SPI driver implementation
|
|
|
|
*
|
|
|
|
* @author Ryan Kurte <ryankurte@gmail.com>
|
|
|
|
* @author Bas Stottelaar <basstottelaar@gmail.com>
|
|
|
|
* @author Christian Amsüss <c@amsuess.com>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2020-10-21 15:56:42 +02:00
|
|
|
#include <assert.h>
|
|
|
|
|
2017-11-13 16:08:08 +01:00
|
|
|
#include "cpu.h"
|
|
|
|
#include "sched.h"
|
|
|
|
#include "thread.h"
|
|
|
|
#include "mutex.h"
|
|
|
|
|
|
|
|
#include "periph_conf.h"
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
#include "periph/spi.h"
|
|
|
|
|
|
|
|
#include "em_device.h"
|
|
|
|
#include "em_cmu.h"
|
|
|
|
#include "em_usart.h"
|
|
|
|
|
|
|
|
static mutex_t spi_lock[SPI_NUMOF];
|
|
|
|
|
|
|
|
void spi_init(spi_t bus)
|
|
|
|
{
|
|
|
|
assert(bus < SPI_NUMOF);
|
|
|
|
|
|
|
|
/* initialize lock */
|
|
|
|
mutex_init(&spi_lock[bus]);
|
|
|
|
|
|
|
|
/* initialize pins */
|
|
|
|
spi_init_pins(bus);
|
|
|
|
}
|
|
|
|
|
|
|
|
void spi_init_pins(spi_t bus)
|
|
|
|
{
|
|
|
|
/* configure the pins */
|
|
|
|
gpio_init(spi_config[bus].clk_pin, GPIO_OUT);
|
|
|
|
gpio_init(spi_config[bus].mosi_pin, GPIO_OUT);
|
|
|
|
gpio_init(spi_config[bus].miso_pin, GPIO_IN_PD);
|
|
|
|
}
|
|
|
|
|
2022-10-18 12:00:52 +02:00
|
|
|
#define GET_PIN(x) (x & 0xf)
|
|
|
|
#define GET_PORT(x) (x >> 4)
|
|
|
|
#define USART_NUM(ref) ((ref == USART0) ? 0 : -1)
|
|
|
|
|
2021-02-01 14:05:16 +01:00
|
|
|
void spi_acquire(spi_t bus, spi_cs_t cs, spi_mode_t mode, spi_clk_t clk)
|
2017-11-13 16:08:08 +01:00
|
|
|
{
|
|
|
|
(void)cs;
|
2021-02-01 14:05:16 +01:00
|
|
|
assert((unsigned)bus < SPI_NUMOF);
|
2017-11-13 16:08:08 +01:00
|
|
|
|
|
|
|
mutex_lock(&spi_lock[bus]);
|
|
|
|
|
|
|
|
/* power on spi bus */
|
2022-10-18 12:00:52 +02:00
|
|
|
#if defined(_SILICON_LABS_32B_SERIES_0) || defined(_SILICON_LABS_32B_SERIES_1)
|
2017-11-13 16:08:08 +01:00
|
|
|
CMU_ClockEnable(cmuClock_HFPER, true);
|
2022-10-18 12:00:52 +02:00
|
|
|
#endif
|
2017-11-13 16:08:08 +01:00
|
|
|
CMU_ClockEnable(spi_config[bus].cmu, true);
|
|
|
|
|
|
|
|
USART_InitSync_TypeDef init = USART_INITSYNC_DEFAULT;
|
|
|
|
|
2021-02-01 14:05:16 +01:00
|
|
|
init.baudrate = (uint32_t)clk;
|
|
|
|
init.clockMode = (USART_ClockMode_TypeDef)mode;
|
2017-11-13 16:08:08 +01:00
|
|
|
init.msbf = true;
|
|
|
|
|
|
|
|
USART_InitSync(spi_config[bus].dev, &init);
|
|
|
|
|
|
|
|
/* configure pin functions */
|
2020-06-26 08:58:27 +02:00
|
|
|
#if defined(_SILICON_LABS_32B_SERIES_0)
|
2017-11-13 16:08:08 +01:00
|
|
|
spi_config[bus].dev->ROUTE = (spi_config[bus].loc |
|
|
|
|
USART_ROUTE_RXPEN |
|
|
|
|
USART_ROUTE_TXPEN |
|
|
|
|
USART_ROUTE_CLKPEN);
|
2020-06-26 08:58:27 +02:00
|
|
|
#elif defined(_SILICON_LABS_32B_SERIES_1)
|
2017-11-13 16:08:08 +01:00
|
|
|
spi_config[bus].dev->ROUTELOC0 = spi_config[bus].loc;
|
|
|
|
spi_config[bus].dev->ROUTEPEN = (USART_ROUTEPEN_RXPEN |
|
|
|
|
USART_ROUTEPEN_TXPEN |
|
|
|
|
USART_ROUTEPEN_CLKPEN);
|
2022-10-18 12:00:52 +02:00
|
|
|
#elif defined(_SILICON_LABS_32B_SERIES_2)
|
|
|
|
GPIO->USARTROUTE[USART_NUM(spi_config[bus].dev)].TXROUTE =
|
|
|
|
(GET_PORT(spi_config[bus].mosi_pin) << _GPIO_USART_TXROUTE_PORT_SHIFT) |
|
|
|
|
(GET_PIN(spi_config[bus].mosi_pin) << _GPIO_USART_TXROUTE_PIN_SHIFT);
|
|
|
|
GPIO->USARTROUTE[USART_NUM(spi_config[bus].dev)].RXROUTE =
|
|
|
|
(GET_PORT(spi_config[bus].miso_pin) << _GPIO_USART_RXROUTE_PORT_SHIFT) |
|
|
|
|
(GET_PIN(spi_config[bus].miso_pin) << _GPIO_USART_RXROUTE_PIN_SHIFT);
|
|
|
|
GPIO->USARTROUTE[USART_NUM(spi_config[bus].dev)].CLKROUTE =
|
|
|
|
(GET_PORT(spi_config[bus].clk_pin) << _GPIO_USART_CLKROUTE_PORT_SHIFT) |
|
|
|
|
(GET_PIN(spi_config[bus].clk_pin) << _GPIO_USART_CLKROUTE_PIN_SHIFT);
|
|
|
|
GPIO->USARTROUTE[USART_NUM(spi_config[bus].dev)].ROUTEEN =
|
|
|
|
(GPIO_USART_ROUTEEN_RXPEN | GPIO_USART_ROUTEEN_TXPEN | GPIO_USART_ROUTEEN_CLKPEN);
|
2017-11-13 16:08:08 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void spi_release(spi_t bus)
|
|
|
|
{
|
|
|
|
/* power off spi bus */
|
|
|
|
CMU_ClockEnable(spi_config[bus].cmu, false);
|
|
|
|
|
|
|
|
mutex_unlock(&spi_lock[bus]);
|
|
|
|
}
|
|
|
|
|
|
|
|
void spi_transfer_bytes(spi_t bus, spi_cs_t cs, bool cont,
|
|
|
|
const void *out, void *in, size_t len)
|
|
|
|
{
|
|
|
|
uint8_t *out_buf = (uint8_t *)out;
|
|
|
|
uint8_t *in_buf = (uint8_t *)in;
|
|
|
|
|
|
|
|
if (cs != SPI_CS_UNDEF) {
|
|
|
|
gpio_clear((gpio_t)cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
|
|
uint8_t ret = USART_SpiTransfer(spi_config[bus].dev,
|
|
|
|
out != NULL ? out_buf[i] : 0);
|
|
|
|
|
|
|
|
if (in != NULL) {
|
|
|
|
in_buf[i] = ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((!cont) && (cs != SPI_CS_UNDEF)) {
|
|
|
|
gpio_set((gpio_t)cs);
|
|
|
|
}
|
|
|
|
}
|