HAL_ChibiOS: added checking on bouncebuffer allocation
fail operations if DMA bouncebuffer alloc fails
This commit is contained in:
parent
7c9a896f09
commit
6477180e87
@ -164,15 +164,23 @@ bool DeviceBus::adjust_timer(AP_HAL::Device::PeriodicHandle h, uint32_t period_u
|
|||||||
/*
|
/*
|
||||||
setup to use DMA-safe bouncebuffers for device transfers
|
setup to use DMA-safe bouncebuffers for device transfers
|
||||||
*/
|
*/
|
||||||
void DeviceBus::bouncebuffer_setup(const uint8_t *&buf_tx, uint16_t tx_len,
|
bool DeviceBus::bouncebuffer_setup(const uint8_t *&buf_tx, uint16_t tx_len,
|
||||||
uint8_t *&buf_rx, uint16_t rx_len)
|
uint8_t *&buf_rx, uint16_t rx_len)
|
||||||
{
|
{
|
||||||
if (buf_rx) {
|
if (buf_rx) {
|
||||||
bouncebuffer_setup_read(bounce_buffer_rx, &buf_rx, rx_len);
|
if (!bouncebuffer_setup_read(bounce_buffer_rx, &buf_rx, rx_len)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (buf_tx) {
|
if (buf_tx) {
|
||||||
bouncebuffer_setup_write(bounce_buffer_tx, &buf_tx, tx_len);
|
if (!bouncebuffer_setup_write(bounce_buffer_tx, &buf_tx, tx_len)) {
|
||||||
|
if (buf_rx) {
|
||||||
|
bouncebuffer_abort(bounce_buffer_rx);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -39,8 +39,8 @@ public:
|
|||||||
bool adjust_timer(AP_HAL::Device::PeriodicHandle h, uint32_t period_usec);
|
bool adjust_timer(AP_HAL::Device::PeriodicHandle h, uint32_t period_usec);
|
||||||
static void bus_thread(void *arg);
|
static void bus_thread(void *arg);
|
||||||
|
|
||||||
void bouncebuffer_setup(const uint8_t *&buf_tx, uint16_t tx_len,
|
bool bouncebuffer_setup(const uint8_t *&buf_tx, uint16_t tx_len,
|
||||||
uint8_t *&buf_rx, uint16_t rx_len);
|
uint8_t *&buf_rx, uint16_t rx_len) WARN_IF_UNUSED;
|
||||||
void bouncebuffer_finish(const uint8_t *buf_tx, uint8_t *buf_rx, uint16_t rx_len);
|
void bouncebuffer_finish(const uint8_t *buf_tx, uint8_t *buf_rx, uint16_t rx_len);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -258,7 +258,10 @@ bool I2CDevice::_transfer(const uint8_t *send, uint32_t send_len,
|
|||||||
{
|
{
|
||||||
i2cAcquireBus(I2CD[bus.busnum].i2c);
|
i2cAcquireBus(I2CD[bus.busnum].i2c);
|
||||||
|
|
||||||
bus.bouncebuffer_setup(send, send_len, recv, recv_len);
|
if (!bus.bouncebuffer_setup(send, send_len, recv, recv_len)) {
|
||||||
|
i2cReleaseBus(I2CD[bus.busnum].i2c);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
for(uint8_t i=0 ; i <= _retries; i++) {
|
for(uint8_t i=0 ; i <= _retries; i++) {
|
||||||
int ret;
|
int ret;
|
||||||
|
@ -185,7 +185,10 @@ bool SPIDevice::do_transfer(const uint8_t *send, uint8_t *recv, uint32_t len)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
bus.bouncebuffer_setup(send, len, recv, len);
|
if (!bus.bouncebuffer_setup(send, len, recv, len)) {
|
||||||
|
set_chip_select(old_cs_forced);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
osalSysLock();
|
osalSysLock();
|
||||||
hal.util->persistent_data.spi_count++;
|
hal.util->persistent_data.spi_count++;
|
||||||
if (send == nullptr) {
|
if (send == nullptr) {
|
||||||
|
@ -45,8 +45,9 @@ void bouncebuffer_init(struct bouncebuffer_t **bouncebuffer, uint32_t prealloc_b
|
|||||||
(*bouncebuffer)->is_sdcard = sdcard;
|
(*bouncebuffer)->is_sdcard = sdcard;
|
||||||
if (prealloc_bytes) {
|
if (prealloc_bytes) {
|
||||||
(*bouncebuffer)->dma_buf = sdcard?malloc_sdcard_dma(prealloc_bytes):malloc_dma(prealloc_bytes);
|
(*bouncebuffer)->dma_buf = sdcard?malloc_sdcard_dma(prealloc_bytes):malloc_dma(prealloc_bytes);
|
||||||
osalDbgAssert(((*bouncebuffer)->dma_buf != NULL), "bouncebuffer preallocate");
|
if ((*bouncebuffer)->dma_buf) {
|
||||||
(*bouncebuffer)->size = prealloc_bytes;
|
(*bouncebuffer)->size = prealloc_bytes;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -55,11 +56,11 @@ void bouncebuffer_init(struct bouncebuffer_t **bouncebuffer, uint32_t prealloc_b
|
|||||||
Note that *buf can be NULL, in which case we allocate DMA capable memory, but don't
|
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
|
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)
|
bool bouncebuffer_setup_read(struct bouncebuffer_t *bouncebuffer, uint8_t **buf, uint32_t size)
|
||||||
{
|
{
|
||||||
if (!bouncebuffer || IS_DMA_SAFE(*buf)) {
|
if (!bouncebuffer || IS_DMA_SAFE(*buf)) {
|
||||||
// nothing needs to be done
|
// nothing needs to be done
|
||||||
return;
|
return true;
|
||||||
}
|
}
|
||||||
osalDbgAssert((bouncebuffer->busy == false), "bouncebuffer read");
|
osalDbgAssert((bouncebuffer->busy == false), "bouncebuffer read");
|
||||||
bouncebuffer->orig_buf = *buf;
|
bouncebuffer->orig_buf = *buf;
|
||||||
@ -68,7 +69,10 @@ void bouncebuffer_setup_read(struct bouncebuffer_t *bouncebuffer, uint8_t **buf,
|
|||||||
free(bouncebuffer->dma_buf);
|
free(bouncebuffer->dma_buf);
|
||||||
}
|
}
|
||||||
bouncebuffer->dma_buf = bouncebuffer->is_sdcard?malloc_sdcard_dma(size):malloc_dma(size);
|
bouncebuffer->dma_buf = bouncebuffer->is_sdcard?malloc_sdcard_dma(size):malloc_dma(size);
|
||||||
osalDbgAssert((bouncebuffer->dma_buf != NULL), "bouncebuffer read allocate");
|
if (!bouncebuffer->dma_buf) {
|
||||||
|
bouncebuffer->size = 0;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
bouncebuffer->size = size;
|
bouncebuffer->size = size;
|
||||||
}
|
}
|
||||||
*buf = bouncebuffer->dma_buf;
|
*buf = bouncebuffer->dma_buf;
|
||||||
@ -77,6 +81,7 @@ void bouncebuffer_setup_read(struct bouncebuffer_t *bouncebuffer, uint8_t **buf,
|
|||||||
stm32_cacheBufferInvalidate(*buf, (size+31)&~31);
|
stm32_cacheBufferInvalidate(*buf, (size+31)&~31);
|
||||||
#endif
|
#endif
|
||||||
bouncebuffer->busy = true;
|
bouncebuffer->busy = true;
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -97,11 +102,11 @@ void bouncebuffer_finish_read(struct bouncebuffer_t *bouncebuffer, const uint8_t
|
|||||||
/*
|
/*
|
||||||
setup for reading from memory to a device, allocating a bouncebuffer if needed
|
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)
|
bool bouncebuffer_setup_write(struct bouncebuffer_t *bouncebuffer, const uint8_t **buf, uint32_t size)
|
||||||
{
|
{
|
||||||
if (!bouncebuffer || IS_DMA_SAFE(*buf)) {
|
if (!bouncebuffer || IS_DMA_SAFE(*buf)) {
|
||||||
// nothing needs to be done
|
// nothing needs to be done
|
||||||
return;
|
return true;
|
||||||
}
|
}
|
||||||
osalDbgAssert((bouncebuffer->busy == false), "bouncebuffer write");
|
osalDbgAssert((bouncebuffer->busy == false), "bouncebuffer write");
|
||||||
if (bouncebuffer->size < size) {
|
if (bouncebuffer->size < size) {
|
||||||
@ -109,7 +114,10 @@ void bouncebuffer_setup_write(struct bouncebuffer_t *bouncebuffer, const uint8_t
|
|||||||
free(bouncebuffer->dma_buf);
|
free(bouncebuffer->dma_buf);
|
||||||
}
|
}
|
||||||
bouncebuffer->dma_buf = bouncebuffer->is_sdcard?malloc_sdcard_dma(size):malloc_dma(size);
|
bouncebuffer->dma_buf = bouncebuffer->is_sdcard?malloc_sdcard_dma(size):malloc_dma(size);
|
||||||
osalDbgAssert((bouncebuffer->dma_buf != NULL), "bouncebuffer write allocate");
|
if (!bouncebuffer->dma_buf) {
|
||||||
|
bouncebuffer->size = 0;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
bouncebuffer->size = size;
|
bouncebuffer->size = size;
|
||||||
}
|
}
|
||||||
if (*buf) {
|
if (*buf) {
|
||||||
@ -121,6 +129,7 @@ void bouncebuffer_setup_write(struct bouncebuffer_t *bouncebuffer, const uint8_t
|
|||||||
stm32_cacheBufferFlush(*buf, (size+31)&~31);
|
stm32_cacheBufferFlush(*buf, (size+31)&~31);
|
||||||
#endif
|
#endif
|
||||||
bouncebuffer->busy = true;
|
bouncebuffer->busy = true;
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -134,3 +143,13 @@ void bouncebuffer_finish_write(struct bouncebuffer_t *bouncebuffer, const uint8_
|
|||||||
bouncebuffer->busy = false;
|
bouncebuffer->busy = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
abort an operation
|
||||||
|
*/
|
||||||
|
void bouncebuffer_abort(struct bouncebuffer_t *bouncebuffer)
|
||||||
|
{
|
||||||
|
if (bouncebuffer) {
|
||||||
|
bouncebuffer->busy = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -27,7 +27,7 @@ void bouncebuffer_init(struct bouncebuffer_t **bouncebuffer, uint32_t prealloc_b
|
|||||||
/*
|
/*
|
||||||
setup for reading from a device into memory, allocating a bouncebuffer if needed
|
setup for reading from a device into memory, allocating a bouncebuffer if needed
|
||||||
*/
|
*/
|
||||||
void bouncebuffer_setup_read(struct bouncebuffer_t *bouncebuffer, uint8_t **buf, uint32_t size);
|
bool bouncebuffer_setup_read(struct bouncebuffer_t *bouncebuffer, uint8_t **buf, uint32_t size);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
finish a read operation
|
finish a read operation
|
||||||
@ -37,13 +37,18 @@ void bouncebuffer_finish_read(struct bouncebuffer_t *bouncebuffer, const uint8_t
|
|||||||
/*
|
/*
|
||||||
setup for reading from memory to a device, allocating a bouncebuffer if needed
|
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);
|
bool bouncebuffer_setup_write(struct bouncebuffer_t *bouncebuffer, const uint8_t **buf, uint32_t size);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
finish a write operation
|
finish a write operation
|
||||||
*/
|
*/
|
||||||
void bouncebuffer_finish_write(struct bouncebuffer_t *bouncebuffer, const uint8_t *buf);
|
void bouncebuffer_finish_write(struct bouncebuffer_t *bouncebuffer, const uint8_t *buf);
|
||||||
|
|
||||||
|
/*
|
||||||
|
abort an operation
|
||||||
|
*/
|
||||||
|
void bouncebuffer_abort(struct bouncebuffer_t *bouncebuffer);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
Reference in New Issue
Block a user