/* * 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. */ /** * @ingroup cpu_efm32 * @ingroup drivers_periph_spi * @{ * * @file * @brief Low-level SPI driver implementation * * @author Ryan Kurte * @author Bas Stottelaar * @author Christian Amsüss * @} */ #include #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); } #define GET_PIN(x) (x & 0xf) #define GET_PORT(x) (x >> 4) #define USART_NUM(ref) ((ref == USART0) ? 0 : -1) void spi_acquire(spi_t bus, spi_cs_t cs, spi_mode_t mode, spi_clk_t clk) { (void)cs; assert((unsigned)bus < SPI_NUMOF); mutex_lock(&spi_lock[bus]); /* power on spi bus */ #if defined(_SILICON_LABS_32B_SERIES_0) || defined(_SILICON_LABS_32B_SERIES_1) CMU_ClockEnable(cmuClock_HFPER, true); #endif CMU_ClockEnable(spi_config[bus].cmu, true); USART_InitSync_TypeDef init = USART_INITSYNC_DEFAULT; init.baudrate = (uint32_t)clk; init.clockMode = (USART_ClockMode_TypeDef)mode; init.msbf = true; USART_InitSync(spi_config[bus].dev, &init); /* configure pin functions */ #if defined(_SILICON_LABS_32B_SERIES_0) spi_config[bus].dev->ROUTE = (spi_config[bus].loc | USART_ROUTE_RXPEN | USART_ROUTE_TXPEN | USART_ROUTE_CLKPEN); #elif defined(_SILICON_LABS_32B_SERIES_1) spi_config[bus].dev->ROUTELOC0 = spi_config[bus].loc; spi_config[bus].dev->ROUTEPEN = (USART_ROUTEPEN_RXPEN | USART_ROUTEPEN_TXPEN | USART_ROUTEPEN_CLKPEN); #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); #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); } }