Merge pull request #11483 from SteveCEvans/elrs_race
[betaflight.git] / src / main / drivers / bus_spi.c
blob05f91eb757fc99b3a7275b42f9969e5ab11bd93b
1 /*
2 * This file is part of Cleanflight and Betaflight.
4 * Cleanflight and Betaflight are free software. You can redistribute
5 * this software and/or modify this software under the terms of the
6 * GNU General Public License as published by the Free Software
7 * Foundation, either version 3 of the License, or (at your option)
8 * any later version.
10 * Cleanflight and Betaflight are distributed in the hope that they
11 * will be useful, but WITHOUT ANY WARRANTY; without even the implied
12 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13 * See the GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this software.
18 * If not, see <http://www.gnu.org/licenses/>.
21 #include <stdbool.h>
22 #include <stdint.h>
23 #include <string.h>
25 #include "platform.h"
27 #include "build/atomic.h"
29 #ifdef USE_SPI
31 #include "drivers/bus.h"
32 #include "drivers/bus_spi.h"
33 #include "drivers/bus_spi_impl.h"
34 #include "drivers/dma_reqmap.h"
35 #include "drivers/exti.h"
36 #include "drivers/io.h"
37 #include "drivers/motor.h"
38 #include "drivers/rcc.h"
39 #include "nvic.h"
40 #include "pg/bus_spi.h"
42 #define NUM_QUEUE_SEGS 5
44 static uint8_t spiRegisteredDeviceCount = 0;
46 spiDevice_t spiDevice[SPIDEV_COUNT];
47 busDevice_t spiBusDevice[SPIDEV_COUNT];
49 SPIDevice spiDeviceByInstance(SPI_TypeDef *instance)
51 #ifdef USE_SPI_DEVICE_1
52 if (instance == SPI1) {
53 return SPIDEV_1;
55 #endif
57 #ifdef USE_SPI_DEVICE_2
58 if (instance == SPI2) {
59 return SPIDEV_2;
61 #endif
63 #ifdef USE_SPI_DEVICE_3
64 if (instance == SPI3) {
65 return SPIDEV_3;
67 #endif
69 #ifdef USE_SPI_DEVICE_4
70 if (instance == SPI4) {
71 return SPIDEV_4;
73 #endif
75 #ifdef USE_SPI_DEVICE_5
76 if (instance == SPI5) {
77 return SPIDEV_5;
79 #endif
81 #ifdef USE_SPI_DEVICE_6
82 if (instance == SPI6) {
83 return SPIDEV_6;
85 #endif
87 return SPIINVALID;
90 SPI_TypeDef *spiInstanceByDevice(SPIDevice device)
92 if (device == SPIINVALID || device >= SPIDEV_COUNT) {
93 return NULL;
96 return spiDevice[device].dev;
99 bool spiInit(SPIDevice device)
101 switch (device) {
102 case SPIINVALID:
103 return false;
105 case SPIDEV_1:
106 #ifdef USE_SPI_DEVICE_1
107 spiInitDevice(device);
108 return true;
109 #else
110 break;
111 #endif
113 case SPIDEV_2:
114 #ifdef USE_SPI_DEVICE_2
115 spiInitDevice(device);
116 return true;
117 #else
118 break;
119 #endif
121 case SPIDEV_3:
122 #if defined(USE_SPI_DEVICE_3) && !defined(STM32F1)
123 spiInitDevice(device);
124 return true;
125 #else
126 break;
127 #endif
129 case SPIDEV_4:
130 #if defined(USE_SPI_DEVICE_4)
131 spiInitDevice(device);
132 return true;
133 #else
134 break;
135 #endif
137 case SPIDEV_5:
138 #if defined(USE_SPI_DEVICE_5)
139 spiInitDevice(device);
140 return true;
141 #else
142 break;
143 #endif
145 case SPIDEV_6:
146 #if defined(USE_SPI_DEVICE_6)
147 spiInitDevice(device);
148 return true;
149 #else
150 break;
151 #endif
153 return false;
156 // Return true if DMA engine is busy
157 bool spiIsBusy(const extDevice_t *dev)
159 return (dev->bus->curSegment != (busSegment_t *)BUS_SPI_FREE);
162 // Wait for DMA completion
163 void spiWait(const extDevice_t *dev)
165 // Wait for completion
166 while (dev->bus->curSegment != (busSegment_t *)BUS_SPI_FREE);
169 // Wait for bus to become free, then read/write block of data
170 void spiReadWriteBuf(const extDevice_t *dev, uint8_t *txData, uint8_t *rxData, int len)
172 // This routine blocks so no need to use static data
173 busSegment_t segments[] = {
174 {.u.buffers = {txData, rxData}, len, true, NULL},
175 {.u.link = {NULL, NULL}, 0, true, NULL},
178 spiSequence(dev, &segments[0]);
180 spiWait(dev);
183 // Read/Write a block of data, returning false if the bus is busy
184 bool spiReadWriteBufRB(const extDevice_t *dev, uint8_t *txData, uint8_t *rxData, int length)
186 // Ensure any prior DMA has completed before continuing
187 if (spiIsBusy(dev)) {
188 return false;
191 spiReadWriteBuf(dev, txData, rxData, length);
193 return true;
196 // Wait for bus to become free, then read/write a single byte
197 uint8_t spiReadWrite(const extDevice_t *dev, uint8_t data)
199 uint8_t retval;
201 // This routine blocks so no need to use static data
202 busSegment_t segments[] = {
203 {.u.buffers = {&data, &retval}, sizeof(data), true, NULL},
204 {.u.link = {NULL, NULL}, 0, true, NULL},
207 spiSequence(dev, &segments[0]);
209 spiWait(dev);
211 return retval;
214 // Wait for bus to become free, then read/write a single byte from a register
215 uint8_t spiReadWriteReg(const extDevice_t *dev, uint8_t reg, uint8_t data)
217 uint8_t retval;
219 // This routine blocks so no need to use static data
220 busSegment_t segments[] = {
221 {.u.buffers = {&reg, NULL}, sizeof(reg), false, NULL},
222 {.u.buffers = {&data, &retval}, sizeof(data), true, NULL},
223 {.u.link = {NULL, NULL}, 0, true, NULL},
226 spiSequence(dev, &segments[0]);
228 spiWait(dev);
230 return retval;
233 // Wait for bus to become free, then write a single byte
234 void spiWrite(const extDevice_t *dev, uint8_t data)
236 // This routine blocks so no need to use static data
237 busSegment_t segments[] = {
238 {.u.buffers = {&data, NULL}, sizeof(data), true, NULL},
239 {.u.link = {NULL, NULL}, 0, true, NULL},
242 spiSequence(dev, &segments[0]);
244 spiWait(dev);
247 // Write data to a register
248 void spiWriteReg(const extDevice_t *dev, uint8_t reg, uint8_t data)
250 // This routine blocks so no need to use static data
251 busSegment_t segments[] = {
252 {.u.buffers = {&reg, NULL}, sizeof(reg), false, NULL},
253 {.u.buffers = {&data, NULL}, sizeof(data), true, NULL},
254 {.u.link = {NULL, NULL}, 0, true, NULL},
257 spiSequence(dev, &segments[0]);
259 spiWait(dev);
262 // Write data to a register, returning false if the bus is busy
263 bool spiWriteRegRB(const extDevice_t *dev, uint8_t reg, uint8_t data)
265 // Ensure any prior DMA has completed before continuing
266 if (spiIsBusy(dev)) {
267 return false;
270 spiWriteReg(dev, reg, data);
272 return true;
275 // Read a block of data from a register
276 void spiReadRegBuf(const extDevice_t *dev, uint8_t reg, uint8_t *data, uint8_t length)
278 // This routine blocks so no need to use static data
279 busSegment_t segments[] = {
280 {.u.buffers = {&reg, NULL}, sizeof(reg), false, NULL},
281 {.u.buffers = {NULL, data}, length, true, NULL},
282 {.u.link = {NULL, NULL}, 0, true, NULL},
285 spiSequence(dev, &segments[0]);
287 spiWait(dev);
290 // Read a block of data from a register, returning false if the bus is busy
291 bool spiReadRegBufRB(const extDevice_t *dev, uint8_t reg, uint8_t *data, uint8_t length)
293 // Ensure any prior DMA has completed before continuing
294 if (spiIsBusy(dev)) {
295 return false;
298 spiReadRegBuf(dev, reg, data, length);
300 return true;
303 // Read a block of data where the register is ORed with 0x80, returning false if the bus is busy
304 bool spiReadRegMskBufRB(const extDevice_t *dev, uint8_t reg, uint8_t *data, uint8_t length)
306 return spiReadRegBufRB(dev, reg | 0x80, data, length);
309 // Wait for bus to become free, then write a block of data to a register
310 void spiWriteRegBuf(const extDevice_t *dev, uint8_t reg, uint8_t *data, uint32_t length)
312 // This routine blocks so no need to use static data
313 busSegment_t segments[] = {
314 {.u.buffers = {&reg, NULL}, sizeof(reg), false, NULL},
315 {.u.buffers = {data, NULL}, length, true, NULL},
316 {.u.link = {NULL, NULL}, 0, true, NULL},
319 spiSequence(dev, &segments[0]);
321 spiWait(dev);
324 // Wait for bus to become free, then read a byte from a register
325 uint8_t spiReadReg(const extDevice_t *dev, uint8_t reg)
327 uint8_t data;
328 // This routine blocks so no need to use static data
329 busSegment_t segments[] = {
330 {.u.buffers = {&reg, NULL}, sizeof(reg), false, NULL},
331 {.u.buffers = {NULL, &data}, sizeof(data), true, NULL},
332 {.u.link = {NULL, NULL}, 0, true, NULL},
335 spiSequence(dev, &segments[0]);
337 spiWait(dev);
339 return data;
342 // Wait for bus to become free, then read a byte of data where the register is ORed with 0x80
343 uint8_t spiReadRegMsk(const extDevice_t *dev, uint8_t reg)
345 return spiReadReg(dev, reg | 0x80);
348 uint16_t spiCalculateDivider(uint32_t freq)
350 #if defined(STM32F4) || defined(STM32G4) || defined(STM32F7)
351 uint32_t spiClk = SystemCoreClock / 2;
352 #elif defined(STM32H7)
353 uint32_t spiClk = 100000000;
354 #else
355 #error "Base SPI clock not defined for this architecture"
356 #endif
358 uint16_t divisor = 2;
360 spiClk >>= 1;
362 for (; (spiClk > freq) && (divisor < 256); divisor <<= 1, spiClk >>= 1);
364 return divisor;
367 uint32_t spiCalculateClock(uint16_t spiClkDivisor)
369 #if defined(STM32F4) || defined(STM32G4) || defined(STM32F7)
370 uint32_t spiClk = SystemCoreClock / 2;
371 #elif defined(STM32H7)
372 uint32_t spiClk = 100000000;
373 #else
374 #error "Base SPI clock not defined for this architecture"
375 #endif
377 return spiClk / spiClkDivisor;
380 // Interrupt handler for SPI receive DMA completion
381 static void spiIrqHandler(const extDevice_t *dev)
383 busDevice_t *bus = dev->bus;
384 busSegment_t *nextSegment;
386 if (bus->curSegment->callback) {
387 switch(bus->curSegment->callback(dev->callbackArg)) {
388 case BUS_BUSY:
389 // Repeat the last DMA segment
390 bus->curSegment--;
391 // Reinitialise the cached init values as segment is not progressing
392 spiInternalInitStream(dev, true);
393 break;
395 case BUS_ABORT:
396 bus->curSegment = (busSegment_t *)BUS_SPI_FREE;
397 return;
399 case BUS_READY:
400 default:
401 // Advance to the next DMA segment
402 break;
406 // Advance through the segment list
407 // OK to discard the volatile qualifier here
408 nextSegment = (busSegment_t *)bus->curSegment + 1;
410 if (nextSegment->len == 0) {
411 if (!bus->curSegment->negateCS) {
412 // Negate Chip Select if not done so already
413 IOHi(dev->busType_u.spi.csnPin);
416 // If a following transaction has been linked, start it
417 if (nextSegment->u.link.dev) {
418 const extDevice_t *nextDev = nextSegment->u.link.dev;
419 busSegment_t *nextSegments = (busSegment_t *)nextSegment->u.link.segments;
420 // The end of the segment list has been reached
421 bus->curSegment = nextSegments;
422 nextSegment->u.link.dev = NULL;
423 spiSequenceStart(nextDev);
424 } else {
425 // The end of the segment list has been reached, so mark transactions as complete
426 bus->curSegment = (busSegment_t *)BUS_SPI_FREE;
428 } else {
429 // Do as much processing as possible before asserting CS to avoid violating minimum high time
430 bool negateCS = bus->curSegment->negateCS;
432 bus->curSegment = nextSegment;
434 // After the completion of the first segment setup the init structure for the subsequent segment
435 if (bus->initSegment) {
436 spiInternalInitStream(dev, false);
437 bus->initSegment = false;
440 if (negateCS) {
441 // Assert Chip Select - it's costly so only do so if necessary
442 IOLo(dev->busType_u.spi.csnPin);
445 // Launch the next transfer
446 spiInternalStartDMA(dev);
448 // Prepare the init structures ready for the next segment to reduce inter-segment time
449 spiInternalInitStream(dev, true);
453 // Interrupt handler for SPI receive DMA completion
454 static void spiRxIrqHandler(dmaChannelDescriptor_t* descriptor)
456 const extDevice_t *dev = (const extDevice_t *)descriptor->userParam;
458 if (!dev) {
459 return;
462 busDevice_t *bus = dev->bus;
464 if (bus->curSegment->negateCS) {
465 // Negate Chip Select
466 IOHi(dev->busType_u.spi.csnPin);
469 spiInternalStopDMA(dev);
471 #ifdef __DCACHE_PRESENT
472 #ifdef STM32H7
473 if (bus->curSegment->u.buffers.rxData &&
474 ((bus->curSegment->u.buffers.rxData < &_dmaram_start__) || (bus->curSegment->u.buffers.rxData >= &_dmaram_end__))) {
475 #else
476 if (bus->curSegment->u.buffers.rxData) {
477 #endif
478 // Invalidate the D cache covering the area into which data has been read
479 SCB_InvalidateDCache_by_Addr(
480 (uint32_t *)((uint32_t)bus->curSegment->u.buffers.rxData & ~CACHE_LINE_MASK),
481 (((uint32_t)bus->curSegment->u.buffers.rxData & CACHE_LINE_MASK) +
482 bus->curSegment->len - 1 + CACHE_LINE_SIZE) & ~CACHE_LINE_MASK);
484 #endif // __DCACHE_PRESENT
486 spiIrqHandler(dev);
489 #if !defined(STM32G4) && !defined(STM32H7)
490 // Interrupt handler for SPI transmit DMA completion
491 static void spiTxIrqHandler(dmaChannelDescriptor_t* descriptor)
493 const extDevice_t *dev = (const extDevice_t *)descriptor->userParam;
495 if (!dev) {
496 return;
499 busDevice_t *bus = dev->bus;
501 spiInternalStopDMA(dev);
503 if (bus->curSegment->negateCS) {
504 // Negate Chip Select
505 IOHi(dev->busType_u.spi.csnPin);
508 spiIrqHandler(dev);
510 #endif
512 // Mark this bus as being SPI and record the first owner to use it
513 bool spiSetBusInstance(extDevice_t *dev, uint32_t device)
515 if ((device == 0) || (device > SPIDEV_COUNT)) {
516 return false;
519 dev->bus = &spiBusDevice[SPI_CFG_TO_DEV(device)];
521 // By default each device should use SPI DMA if the bus supports it
522 dev->useDMA = true;
524 if (dev->bus->busType == BUS_TYPE_SPI) {
525 // This bus has already been initialised
526 dev->bus->deviceCount++;
527 return true;
530 busDevice_t *bus = dev->bus;
532 bus->busType_u.spi.instance = spiInstanceByDevice(SPI_CFG_TO_DEV(device));
534 if (bus->busType_u.spi.instance == NULL) {
535 return false;
538 bus->busType = BUS_TYPE_SPI;
539 bus->useDMA = false;
540 bus->deviceCount = 1;
541 bus->initTx = &dev->initTx;
542 bus->initRx = &dev->initRx;
544 return true;
547 void spiInitBusDMA()
549 uint32_t device;
550 #if defined(STM32F4) && defined(USE_DSHOT_BITBANG)
551 /* Check https://www.st.com/resource/en/errata_sheet/dm00037591-stm32f405407xx-and-stm32f415417xx-device-limitations-stmicroelectronics.pdf
552 * section 2.1.10 which reports an errata that corruption may occurs on DMA2 if AHB peripherals (eg GPIO ports) are
553 * access concurrently with APB peripherals (eg SPI busses). Bitbang DSHOT uses DMA2 to write to GPIO ports. If this
554 * is enabled, then don't enable DMA on an SPI bus using DMA2
556 const bool dshotBitbangActive = isDshotBitbangActive(&motorConfig()->dev);
557 #endif
559 for (device = 0; device < SPIDEV_COUNT; device++) {
560 busDevice_t *bus = &spiBusDevice[device];
562 if (bus->busType != BUS_TYPE_SPI) {
563 // This bus is not in use
564 continue;
567 dmaIdentifier_e dmaTxIdentifier = DMA_NONE;
568 dmaIdentifier_e dmaRxIdentifier = DMA_NONE;
570 int8_t txDmaopt = spiPinConfig(device)->txDmaopt;
571 uint8_t txDmaoptMin = 0;
572 uint8_t txDmaoptMax = MAX_PERIPHERAL_DMA_OPTIONS - 1;
574 if (txDmaopt != -1) {
575 txDmaoptMin = txDmaopt;
576 txDmaoptMax = txDmaopt;
579 for (uint8_t opt = txDmaoptMin; opt <= txDmaoptMax; opt++) {
580 const dmaChannelSpec_t *dmaTxChannelSpec = dmaGetChannelSpecByPeripheral(DMA_PERIPH_SPI_MOSI, device, opt);
582 if (dmaTxChannelSpec) {
583 dmaTxIdentifier = dmaGetIdentifier(dmaTxChannelSpec->ref);
584 #if defined(STM32F4) && defined(USE_DSHOT_BITBANG)
585 if (dshotBitbangActive && (DMA_DEVICE_NO(dmaTxIdentifier) == 2)) {
586 dmaTxIdentifier = DMA_NONE;
587 break;
589 #endif
590 if (!dmaAllocate(dmaTxIdentifier, OWNER_SPI_MOSI, device + 1)) {
591 dmaTxIdentifier = DMA_NONE;
592 continue;
594 bus->dmaTx = dmaGetDescriptorByIdentifier(dmaTxIdentifier);
595 bus->dmaTx->stream = DMA_DEVICE_INDEX(dmaTxIdentifier);
596 bus->dmaTx->channel = dmaTxChannelSpec->channel;
598 dmaEnable(dmaTxIdentifier);
600 break;
604 int8_t rxDmaopt = spiPinConfig(device)->rxDmaopt;
605 uint8_t rxDmaoptMin = 0;
606 uint8_t rxDmaoptMax = MAX_PERIPHERAL_DMA_OPTIONS - 1;
608 if (rxDmaopt != -1) {
609 rxDmaoptMin = rxDmaopt;
610 rxDmaoptMax = rxDmaopt;
613 for (uint8_t opt = rxDmaoptMin; opt <= rxDmaoptMax; opt++) {
614 const dmaChannelSpec_t *dmaRxChannelSpec = dmaGetChannelSpecByPeripheral(DMA_PERIPH_SPI_MISO, device, opt);
616 if (dmaRxChannelSpec) {
617 dmaRxIdentifier = dmaGetIdentifier(dmaRxChannelSpec->ref);
618 #if defined(STM32F4) && defined(USE_DSHOT_BITBANG)
619 if (dshotBitbangActive && (DMA_DEVICE_NO(dmaRxIdentifier) == 2)) {
620 dmaRxIdentifier = DMA_NONE;
621 break;
623 #endif
624 if (!dmaAllocate(dmaRxIdentifier, OWNER_SPI_MISO, device + 1)) {
625 dmaRxIdentifier = DMA_NONE;
626 continue;
628 bus->dmaRx = dmaGetDescriptorByIdentifier(dmaRxIdentifier);
629 bus->dmaRx->stream = DMA_DEVICE_INDEX(dmaRxIdentifier);
630 bus->dmaRx->channel = dmaRxChannelSpec->channel;
632 dmaEnable(dmaRxIdentifier);
634 break;
638 if (dmaTxIdentifier && dmaRxIdentifier) {
639 // Ensure streams are disabled
640 spiInternalResetStream(bus->dmaRx);
641 spiInternalResetStream(bus->dmaTx);
643 spiInternalResetDescriptors(bus);
645 /* Note that this driver may be called both from the normal thread of execution, or from USB interrupt
646 * handlers, so the DMA completion interrupt must be at a higher priority
648 dmaSetHandler(dmaRxIdentifier, spiRxIrqHandler, NVIC_PRIO_SPI_DMA, 0);
650 bus->useDMA = true;
651 #if !defined(STM32G4) && !defined(STM32H7)
652 } else if (dmaTxIdentifier) {
653 // Transmit on DMA is adequate for OSD so worth having
654 bus->dmaTx = dmaGetDescriptorByIdentifier(dmaTxIdentifier);
655 bus->dmaRx = (dmaChannelDescriptor_t *)NULL;
657 // Ensure streams are disabled
658 spiInternalResetStream(bus->dmaTx);
660 spiInternalResetDescriptors(bus);
662 dmaSetHandler(dmaTxIdentifier, spiTxIrqHandler, NVIC_PRIO_SPI_DMA, 0);
664 bus->useDMA = true;
665 #endif
666 } else {
667 // Disassociate channels from bus
668 bus->dmaRx = (dmaChannelDescriptor_t *)NULL;
669 bus->dmaTx = (dmaChannelDescriptor_t *)NULL;
674 void spiSetClkDivisor(const extDevice_t *dev, uint16_t divisor)
676 ((extDevice_t *)dev)->busType_u.spi.speed = divisor;
679 // Set the clock phase/polarity to be used for accesses by the given device
680 void spiSetClkPhasePolarity(const extDevice_t *dev, bool leadingEdge)
682 ((extDevice_t *)dev)->busType_u.spi.leadingEdge = leadingEdge;
685 // Enable/disable DMA on a specific device. Enabled by default.
686 void spiDmaEnable(const extDevice_t *dev, bool enable)
688 ((extDevice_t *)dev)->useDMA = enable;
691 bool spiUseDMA(const extDevice_t *dev)
693 // Full DMA only requires both transmit and receive}
694 return dev->bus->useDMA && dev->bus->dmaRx && dev->useDMA;
697 bool spiUseMOSI_DMA(const extDevice_t *dev)
699 return dev->bus->useDMA && dev->useDMA;
702 void spiBusDeviceRegister(const extDevice_t *dev)
704 UNUSED(dev);
706 spiRegisteredDeviceCount++;
709 uint8_t spiGetRegisteredDeviceCount(void)
711 return spiRegisteredDeviceCount;
714 uint8_t spiGetExtDeviceCount(const extDevice_t *dev)
716 return dev->bus->deviceCount;
719 // DMA transfer setup and start
720 void spiSequence(const extDevice_t *dev, busSegment_t *segments)
722 busDevice_t *bus = dev->bus;
724 ATOMIC_BLOCK(NVIC_PRIO_MAX) {
725 if (spiIsBusy(dev)) {
726 busSegment_t *endSegment;
728 // Defer this transfer to be triggered upon completion of the current transfer
730 // Find the last segment of the new transfer
731 for (endSegment = segments; endSegment->len; endSegment++);
733 // Safe to discard the volatile qualifier as we're in an atomic block
734 busSegment_t *endCmpSegment = (busSegment_t *)bus->curSegment;
736 if (endCmpSegment) {
737 while (true) {
738 // Find the last segment of the current transfer
739 for (; endCmpSegment->len; endCmpSegment++);
741 if (endCmpSegment == endSegment) {
742 /* Attempt to use the new segment list twice in the same queue. Abort.
743 * Note that this can only happen with non-blocking transfers so drivers must take
744 * care to avoid this.
745 * */
746 return;
749 if (endCmpSegment->u.link.dev == NULL) {
750 // End of the segment list queue reached
751 break;
752 } else {
753 // Follow the link to the next queued segment list
754 endCmpSegment = (busSegment_t *)endCmpSegment->u.link.segments;
759 // Record the dev and segments parameters in the terminating segment entry
760 endCmpSegment->u.link.dev = dev;
761 endCmpSegment->u.link.segments = segments;
763 return;
764 } else {
765 // Claim the bus with this list of segments
766 bus->curSegment = segments;
770 spiSequenceStart(dev);
772 #endif