mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-19 23:28:32 -04:00
26d1371d25
this fixes an issue with DMA to SPI6 on STM32H7. On the H7 SRAM4 can be used for DMA and BDMA to all devices except for SDMMC which uses IDMA. by separating out DMA of sdcard data from other data we can arrange for DMA to all devices to work
135 lines
4.7 KiB
C
135 lines
4.7 KiB
C
/*
|
|
* This file is free software: you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by the
|
|
* Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This file is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
* See the GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
/*
|
|
bouncebuffer code for DMA safe memory operations
|
|
*/
|
|
#include "stm32_util.h"
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include "bouncebuffer.h"
|
|
|
|
#if defined(STM32H7)
|
|
// always use a bouncebuffer on H7, to ensure alignment and padding
|
|
#define IS_DMA_SAFE(addr) false
|
|
#elif defined(STM32F7)
|
|
// on F76x we only consider first half of DTCM memory as DMA safe, 2nd half is used as fast memory for EKF
|
|
// on F74x we only have 64k of DTCM
|
|
#define IS_DMA_SAFE(addr) ((((uint32_t)(addr)) & ((0xFFFFFFFF & ~(64*1024U-1)) | 1U)) == 0x20000000)
|
|
#else
|
|
// this checks an address is in main memory and 16 bit aligned
|
|
#define IS_DMA_SAFE(addr) ((((uint32_t)(addr)) & 0xF0000001) == 0x20000000)
|
|
#endif
|
|
|
|
/*
|
|
initialise a bouncebuffer
|
|
*/
|
|
void bouncebuffer_init(struct bouncebuffer_t **bouncebuffer, uint32_t prealloc_bytes, bool sdcard)
|
|
{
|
|
(*bouncebuffer) = calloc(1, sizeof(struct bouncebuffer_t));
|
|
osalDbgAssert(((*bouncebuffer) != NULL), "bouncebuffer init");
|
|
(*bouncebuffer)->is_sdcard = sdcard;
|
|
if (prealloc_bytes) {
|
|
(*bouncebuffer)->dma_buf = sdcard?malloc_sdcard_dma(prealloc_bytes):malloc_dma(prealloc_bytes);
|
|
osalDbgAssert(((*bouncebuffer)->dma_buf != NULL), "bouncebuffer preallocate");
|
|
(*bouncebuffer)->size = prealloc_bytes;
|
|
}
|
|
}
|
|
|
|
/*
|
|
setup for reading from a device into memory, allocating a bouncebuffer if needed
|
|
Note that *buf can be NULL, in which case we allocate DMA capable memory, but don't
|
|
copy to it in bouncebuffer_finish_read(). This avoids DMA failures in dummyrx in the SPI LLD
|
|
*/
|
|
void bouncebuffer_setup_read(struct bouncebuffer_t *bouncebuffer, uint8_t **buf, uint32_t size)
|
|
{
|
|
if (!bouncebuffer || IS_DMA_SAFE(*buf)) {
|
|
// nothing needs to be done
|
|
return;
|
|
}
|
|
osalDbgAssert((bouncebuffer->busy == false), "bouncebuffer read");
|
|
bouncebuffer->orig_buf = *buf;
|
|
if (bouncebuffer->size < size) {
|
|
if (bouncebuffer->size > 0) {
|
|
free(bouncebuffer->dma_buf);
|
|
}
|
|
bouncebuffer->dma_buf = bouncebuffer->is_sdcard?malloc_sdcard_dma(size):malloc_dma(size);
|
|
osalDbgAssert((bouncebuffer->dma_buf != NULL), "bouncebuffer read allocate");
|
|
bouncebuffer->size = size;
|
|
}
|
|
*buf = bouncebuffer->dma_buf;
|
|
#if defined(STM32H7)
|
|
osalDbgAssert((((uint32_t)*buf)&31) == 0, "bouncebuffer read align");
|
|
cacheBufferInvalidate(*buf, (size+31)&~31);
|
|
#endif
|
|
bouncebuffer->busy = true;
|
|
}
|
|
|
|
/*
|
|
finish a read operation
|
|
*/
|
|
void bouncebuffer_finish_read(struct bouncebuffer_t *bouncebuffer, const uint8_t *buf, uint32_t size)
|
|
{
|
|
if (bouncebuffer && buf == bouncebuffer->dma_buf) {
|
|
osalDbgAssert((bouncebuffer->busy == true), "bouncebuffer finish_read");
|
|
if (bouncebuffer->orig_buf) {
|
|
memcpy(bouncebuffer->orig_buf, buf, size);
|
|
}
|
|
bouncebuffer->busy = false;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
setup for reading from memory to a device, allocating a bouncebuffer if needed
|
|
*/
|
|
void bouncebuffer_setup_write(struct bouncebuffer_t *bouncebuffer, const uint8_t **buf, uint32_t size)
|
|
{
|
|
if (!bouncebuffer || IS_DMA_SAFE(*buf)) {
|
|
// nothing needs to be done
|
|
return;
|
|
}
|
|
osalDbgAssert((bouncebuffer->busy == false), "bouncebuffer write");
|
|
if (bouncebuffer->size < size) {
|
|
if (bouncebuffer->size > 0) {
|
|
free(bouncebuffer->dma_buf);
|
|
}
|
|
bouncebuffer->dma_buf = bouncebuffer->is_sdcard?malloc_sdcard_dma(size):malloc_dma(size);
|
|
osalDbgAssert((bouncebuffer->dma_buf != NULL), "bouncebuffer write allocate");
|
|
bouncebuffer->size = size;
|
|
}
|
|
if (*buf) {
|
|
memcpy(bouncebuffer->dma_buf, *buf, size);
|
|
}
|
|
*buf = bouncebuffer->dma_buf;
|
|
#if defined(STM32H7)
|
|
osalDbgAssert((((uint32_t)*buf)&31) == 0, "bouncebuffer write align");
|
|
cacheBufferFlush(*buf, (size+31)&~31);
|
|
#endif
|
|
bouncebuffer->busy = true;
|
|
}
|
|
|
|
|
|
/*
|
|
finish a write operation
|
|
*/
|
|
void bouncebuffer_finish_write(struct bouncebuffer_t *bouncebuffer, const uint8_t *buf)
|
|
{
|
|
if (bouncebuffer && buf == bouncebuffer->dma_buf) {
|
|
osalDbgAssert((bouncebuffer->busy == true), "bouncebuffer finish_wite");
|
|
bouncebuffer->busy = false;
|
|
}
|
|
}
|