2019-07-31 14:58:55 -04:00
|
|
|
/*
|
|
|
|
* This file is part of the MicroPython project, http://micropython.org/
|
|
|
|
*
|
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
|
|
|
* Copyright (c) 2016 Scott Shawcroft
|
2019-09-27 17:59:55 -04:00
|
|
|
* Copyright (c) 2019 Lucian Copeland for Adafruit Industries
|
2019-07-31 14:58:55 -04:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
2019-09-26 11:35:08 -04:00
|
|
|
#include <stdbool.h>
|
2019-07-31 14:58:55 -04:00
|
|
|
|
|
|
|
#include "shared-bindings/busio/SPI.h"
|
|
|
|
#include "py/mperrno.h"
|
|
|
|
#include "py/runtime.h"
|
2019-09-26 11:35:08 -04:00
|
|
|
#include "stm32f4xx_hal.h"
|
2019-07-31 14:58:55 -04:00
|
|
|
|
2019-09-26 11:35:08 -04:00
|
|
|
#include "shared-bindings/microcontroller/__init__.h"
|
2019-07-31 14:58:55 -04:00
|
|
|
#include "boards/board.h"
|
2019-09-27 17:59:55 -04:00
|
|
|
#include "supervisor/shared/translate.h"
|
2019-07-31 14:58:55 -04:00
|
|
|
#include "common-hal/microcontroller/Pin.h"
|
|
|
|
|
2019-09-26 11:35:08 -04:00
|
|
|
STATIC bool reserved_spi[6];
|
|
|
|
|
|
|
|
void spi_reset(void) {
|
|
|
|
#ifdef SPI1
|
|
|
|
reserved_spi[0] = false;
|
|
|
|
__HAL_RCC_SPI1_CLK_DISABLE();
|
|
|
|
#endif
|
|
|
|
#ifdef SPI2
|
|
|
|
reserved_spi[1] = false;
|
|
|
|
__HAL_RCC_SPI2_CLK_DISABLE();
|
|
|
|
#endif
|
|
|
|
#ifdef SPI3
|
|
|
|
reserved_spi[2] = false;
|
|
|
|
__HAL_RCC_SPI3_CLK_DISABLE();
|
|
|
|
#endif
|
|
|
|
#ifdef SPI4
|
|
|
|
reserved_spi[3] = false;
|
|
|
|
__HAL_RCC_SPI4_CLK_DISABLE();
|
|
|
|
#endif
|
|
|
|
#ifdef SPI5
|
|
|
|
reserved_spi[4] = false;
|
|
|
|
__HAL_RCC_SPI5_CLK_DISABLE();
|
|
|
|
#endif
|
|
|
|
#ifdef SPI6
|
|
|
|
reserved_spi[5] = false;
|
|
|
|
__HAL_RCC_SPI6_CLK_DISABLE();
|
|
|
|
#endif
|
|
|
|
}
|
2019-09-25 11:37:27 -04:00
|
|
|
|
2019-07-31 14:58:55 -04:00
|
|
|
void common_hal_busio_spi_construct(busio_spi_obj_t *self,
|
2019-09-27 11:00:09 -04:00
|
|
|
const mcu_pin_obj_t * sck, const mcu_pin_obj_t * mosi,
|
2019-07-31 14:58:55 -04:00
|
|
|
const mcu_pin_obj_t * miso) {
|
2019-09-26 11:35:08 -04:00
|
|
|
|
2019-09-27 11:00:09 -04:00
|
|
|
//match pins to SPI objects
|
2019-09-26 11:35:08 -04:00
|
|
|
SPI_TypeDef * SPIx;
|
|
|
|
|
|
|
|
uint8_t sck_len = sizeof(mcu_spi_sck_list)/sizeof(*mcu_spi_sck_list);
|
|
|
|
uint8_t mosi_len = sizeof(mcu_spi_mosi_list)/sizeof(*mcu_spi_mosi_list);
|
|
|
|
uint8_t miso_len = sizeof(mcu_spi_miso_list)/sizeof(*mcu_spi_miso_list);
|
|
|
|
|
|
|
|
//sck
|
|
|
|
for(uint i=0; i<sck_len;i++) {
|
2019-09-27 11:00:09 -04:00
|
|
|
if (mcu_spi_sck_list[i].pin == sck) {
|
2019-09-26 11:35:08 -04:00
|
|
|
//mosi
|
|
|
|
for(uint j=0; j<mosi_len;j++) {
|
|
|
|
if (mcu_spi_mosi_list[j].pin == mosi) {
|
|
|
|
//miso
|
|
|
|
for(uint k=0; k<miso_len;k++) {
|
|
|
|
if ((mcu_spi_miso_list[k].pin == miso)
|
|
|
|
&& (mcu_spi_sck_list[i].spi_index == mcu_spi_mosi_list[j].spi_index)
|
|
|
|
&& (mcu_spi_sck_list[i].spi_index == mcu_spi_miso_list[k].spi_index)) {
|
|
|
|
//everything needs the same index
|
|
|
|
self->sck = &mcu_spi_sck_list[j];
|
|
|
|
self->mosi = &mcu_spi_mosi_list[j];
|
|
|
|
self->miso = &mcu_spi_miso_list[k];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//handle typedef selection, errors
|
2019-09-27 11:00:09 -04:00
|
|
|
if(self->sck!=NULL && self->mosi!=NULL && self->miso!=NULL ) {
|
|
|
|
SPIx = mcu_spi_banks[self->sck->spi_index-1];
|
2019-09-26 11:35:08 -04:00
|
|
|
} else {
|
|
|
|
mp_raise_RuntimeError(translate("Invalid SPI pin selection"));
|
|
|
|
}
|
|
|
|
|
|
|
|
if(reserved_spi[self->sck->spi_index-1]) {
|
|
|
|
mp_raise_RuntimeError(translate("Hardware busy, try alternative pins"));
|
|
|
|
}
|
|
|
|
|
|
|
|
//Start GPIO for each pin
|
|
|
|
GPIO_InitTypeDef GPIO_InitStruct = {0};
|
|
|
|
GPIO_InitStruct.Pin = pin_mask(sck->number);
|
|
|
|
GPIO_InitStruct.Mode = GPIO_MODE_AF_OD;
|
|
|
|
GPIO_InitStruct.Pull = GPIO_PULLUP;
|
|
|
|
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
|
|
|
|
GPIO_InitStruct.Alternate = self->sck->altfn_index;
|
|
|
|
HAL_GPIO_Init(pin_port(sck->port), &GPIO_InitStruct);
|
|
|
|
|
|
|
|
GPIO_InitStruct.Pin = pin_mask(mosi->number);
|
|
|
|
GPIO_InitStruct.Mode = GPIO_MODE_AF_OD;
|
|
|
|
GPIO_InitStruct.Pull = GPIO_PULLUP;
|
|
|
|
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
|
|
|
|
GPIO_InitStruct.Alternate = self->mosi->altfn_index;
|
|
|
|
HAL_GPIO_Init(pin_port(mosi->port), &GPIO_InitStruct);
|
|
|
|
|
|
|
|
GPIO_InitStruct.Pin = pin_mask(miso->number);
|
|
|
|
GPIO_InitStruct.Mode = GPIO_MODE_AF_OD;
|
|
|
|
GPIO_InitStruct.Pull = GPIO_PULLUP;
|
|
|
|
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
|
|
|
|
GPIO_InitStruct.Alternate = self->miso->altfn_index;
|
|
|
|
HAL_GPIO_Init(pin_port(miso->port), &GPIO_InitStruct);
|
|
|
|
|
|
|
|
//Keep separate so above hack can be cleanly replaced
|
|
|
|
#ifdef SPI1
|
|
|
|
if(SPIx==SPI1) {
|
|
|
|
reserved_spi[0] = true;
|
|
|
|
__HAL_RCC_SPI1_CLK_ENABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SPI2
|
|
|
|
if(SPIx==SPI2) {
|
|
|
|
reserved_spi[1] = true;
|
|
|
|
__HAL_RCC_SPI2_CLK_ENABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SPI3
|
|
|
|
if(SPIx==SPI3) {
|
|
|
|
reserved_spi[2] = true;
|
|
|
|
__HAL_RCC_SPI3_CLK_ENABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SPI4
|
|
|
|
if(SPIx==SPI4) {
|
|
|
|
reserved_spi[3] = true;
|
|
|
|
__HAL_RCC_SPI4_CLK_ENABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SPI5
|
|
|
|
if(SPIx==SPI5) {
|
|
|
|
reserved_spi[4] = true;
|
|
|
|
__HAL_RCC_SPI5_CLK_ENABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SPI6
|
|
|
|
if(SPIx==SPI6) {
|
|
|
|
reserved_spi[5] = true;
|
|
|
|
__HAL_RCC_SPI6_CLK_ENABLE();
|
|
|
|
}
|
|
|
|
#endif
|
2019-09-25 11:37:27 -04:00
|
|
|
|
2019-09-26 11:35:08 -04:00
|
|
|
self->handle.Instance = SPIx;
|
|
|
|
self->handle.Init.Mode = SPI_MODE_MASTER;
|
|
|
|
self->handle.Init.Direction = SPI_DIRECTION_2LINES;
|
|
|
|
self->handle.Init.DataSize = SPI_DATASIZE_8BIT;
|
|
|
|
self->handle.Init.CLKPolarity = SPI_POLARITY_LOW;
|
|
|
|
self->handle.Init.CLKPhase = SPI_PHASE_1EDGE;
|
|
|
|
self->handle.Init.NSS = SPI_NSS_SOFT;
|
|
|
|
self->handle.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_2;
|
|
|
|
self->handle.Init.FirstBit = SPI_FIRSTBIT_MSB;
|
|
|
|
self->handle.Init.TIMode = SPI_TIMODE_DISABLE;
|
|
|
|
self->handle.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE;
|
|
|
|
self->handle.Init.CRCPolynomial = 10;
|
|
|
|
if (HAL_SPI_Init(&self->handle) != HAL_OK)
|
2019-09-25 11:37:27 -04:00
|
|
|
{
|
2019-09-26 11:35:08 -04:00
|
|
|
mp_raise_RuntimeError(translate("SPI Init Error"));
|
2019-09-25 11:37:27 -04:00
|
|
|
}
|
2019-09-26 11:35:08 -04:00
|
|
|
|
|
|
|
claim_pin(sck);
|
|
|
|
claim_pin(mosi);
|
|
|
|
claim_pin(miso);
|
2019-07-31 14:58:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void common_hal_busio_spi_never_reset(busio_spi_obj_t *self) {
|
|
|
|
}
|
|
|
|
|
|
|
|
bool common_hal_busio_spi_deinited(busio_spi_obj_t *self) {
|
2019-09-26 11:35:08 -04:00
|
|
|
return self->sck->pin == mp_const_none;
|
2019-07-31 14:58:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void common_hal_busio_spi_deinit(busio_spi_obj_t *self) {
|
2019-09-26 11:35:08 -04:00
|
|
|
#ifdef SPI1
|
|
|
|
if(self->handle.Instance==SPI1) {
|
|
|
|
reserved_spi[0] = false;
|
|
|
|
__HAL_RCC_SPI1_CLK_DISABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SPI2
|
|
|
|
if(self->handle.Instance==SPI2) {
|
|
|
|
reserved_spi[1] = false;
|
|
|
|
__HAL_RCC_SPI2_CLK_DISABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SPI3
|
|
|
|
if(self->handle.Instance==SPI3) {
|
|
|
|
reserved_spi[2] = false;
|
|
|
|
__HAL_RCC_SPI3_CLK_DISABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SPI4
|
|
|
|
if(self->handle.Instance==SPI4) {
|
|
|
|
reserved_spi[3] = false;
|
|
|
|
__HAL_RCC_SPI4_CLK_DISABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SPI5
|
|
|
|
if(self->handle.Instance==SPI5) {
|
|
|
|
reserved_spi[4] = false;
|
|
|
|
__HAL_RCC_SPI5_CLK_DISABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SPI6
|
|
|
|
if(self->handle.Instance==SPI6) {
|
|
|
|
reserved_spi[5] = false;
|
|
|
|
__HAL_RCC_SPI6_CLK_DISABLE();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
reset_pin_number(self->sck->pin->port,self->sck->pin->number);
|
|
|
|
reset_pin_number(self->mosi->pin->port,self->mosi->pin->number);
|
|
|
|
reset_pin_number(self->miso->pin->port,self->miso->pin->number);
|
|
|
|
self->sck = mp_const_none;
|
|
|
|
self->mosi = mp_const_none;
|
|
|
|
self->miso = mp_const_none;
|
2019-07-31 14:58:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool common_hal_busio_spi_configure(busio_spi_obj_t *self,
|
2019-09-26 11:35:08 -04:00
|
|
|
uint32_t baudrate, uint8_t polarity, uint8_t phase, uint8_t bits) {
|
|
|
|
return true;
|
2019-07-31 14:58:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool common_hal_busio_spi_try_lock(busio_spi_obj_t *self) {
|
|
|
|
bool grabbed_lock = false;
|
2019-09-26 11:35:08 -04:00
|
|
|
|
|
|
|
//Critical section code that may be required at some point.
|
|
|
|
// uint32_t store_primask = __get_PRIMASK();
|
|
|
|
// __disable_irq();
|
|
|
|
// __DMB();
|
|
|
|
|
|
|
|
if (!self->has_lock) {
|
|
|
|
grabbed_lock = true;
|
|
|
|
self->has_lock = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// __DMB();
|
|
|
|
// __set_PRIMASK(store_primask);
|
|
|
|
|
2019-07-31 14:58:55 -04:00
|
|
|
return grabbed_lock;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool common_hal_busio_spi_has_lock(busio_spi_obj_t *self) {
|
|
|
|
return self->has_lock;
|
|
|
|
}
|
|
|
|
|
|
|
|
void common_hal_busio_spi_unlock(busio_spi_obj_t *self) {
|
|
|
|
self->has_lock = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool common_hal_busio_spi_write(busio_spi_obj_t *self,
|
|
|
|
const uint8_t *data, size_t len) {
|
2019-09-27 17:59:55 -04:00
|
|
|
HAL_StatusTypeDef result = HAL_SPI_Transmit (&self->handle, (uint8_t *)data, (uint16_t)len, 2);
|
|
|
|
return result == HAL_OK ? 0 : 1;
|
2019-07-31 14:58:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool common_hal_busio_spi_read(busio_spi_obj_t *self,
|
|
|
|
uint8_t *data, size_t len, uint8_t write_value) {
|
2019-09-27 17:59:55 -04:00
|
|
|
HAL_StatusTypeDef result = HAL_SPI_Receive (&self->handle, data, (uint16_t)len, 2);
|
|
|
|
return result == HAL_OK ? 0 : 1;
|
2019-07-31 14:58:55 -04:00
|
|
|
}
|
|
|
|
|
2019-09-27 17:59:55 -04:00
|
|
|
bool common_hal_busio_spi_transfer(busio_spi_obj_t *self,
|
|
|
|
uint8_t *data_out, uint8_t *data_in, size_t len) {
|
|
|
|
HAL_StatusTypeDef result = HAL_SPI_TransmitReceive (&self->handle,
|
|
|
|
data_out, data_in, (uint16_t)len,2);
|
|
|
|
return result == HAL_OK ? 0 : 1;
|
2019-07-31 14:58:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t common_hal_busio_spi_get_frequency(busio_spi_obj_t* self) {
|
2019-08-15 15:12:50 -04:00
|
|
|
return 0;
|
2019-07-31 14:58:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t common_hal_busio_spi_get_phase(busio_spi_obj_t* self) {
|
2019-08-15 15:12:50 -04:00
|
|
|
return 0;
|
2019-07-31 14:58:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t common_hal_busio_spi_get_polarity(busio_spi_obj_t* self) {
|
2019-08-15 15:12:50 -04:00
|
|
|
return 0;
|
2019-08-15 15:55:33 -04:00
|
|
|
}
|