mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-12 02:48:28 -04:00
a48f0ef90d
this changes the heuristics for UART TX DMA allocation to greatly reduce the chances of DMA contention causing long delays on other devices This fixes issues with FETTec driver output and gimbal status messages as reported by Amilcar and OlliW. The problem is particularly bad when no GPS is connected to GPS1 on fmuv3 and derived boards (such as CubeBlack) key changes: - remember the contention_counter across begin() calls, as the GPS calls begin with new baudrates regularly - added a is_shared() API to Shared_DMA, allowing the UART driver to avoid TX DMA on shared streams when at low baudrates.
118 lines
3.6 KiB
C++
118 lines
3.6 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/>.
|
|
*
|
|
* Code by Andrew Tridgell and Siddharth Bharat Purohit
|
|
*/
|
|
#pragma once
|
|
|
|
#include "AP_HAL_ChibiOS.h"
|
|
|
|
#define SHARED_DMA_MAX_STREAM_ID (8*2)
|
|
|
|
// DMA stream ID for stream_id2 when only one is needed
|
|
#define SHARED_DMA_NONE 255
|
|
|
|
#ifndef HAL_NO_SHARED_DMA
|
|
|
|
class ChibiOS::Shared_DMA
|
|
{
|
|
public:
|
|
FUNCTOR_TYPEDEF(dma_allocate_fn_t, void, Shared_DMA *);
|
|
FUNCTOR_TYPEDEF(dma_deallocate_fn_t, void, Shared_DMA *);
|
|
|
|
// the use of two stream IDs is for support of peripherals that
|
|
// need both a RX and TX DMA channel
|
|
Shared_DMA(uint8_t stream_id1, uint8_t stream_id2,
|
|
dma_allocate_fn_t allocate,
|
|
dma_allocate_fn_t deallocate);
|
|
|
|
// initialise the stream locks
|
|
static void init(void);
|
|
|
|
// blocking lock call
|
|
void lock(void);
|
|
|
|
// non-blocking lock call
|
|
bool lock_nonblock(void);
|
|
|
|
// unlock call. The DMA channel will not be immediately
|
|
// deallocated. Instead it will be deallocated if another driver
|
|
// needs it
|
|
void unlock(bool success = true);
|
|
|
|
//should be called inside the destructor of Shared DMA participants
|
|
void unregister(void);
|
|
|
|
// return true if this DMA channel is being actively contended for
|
|
// by multiple drivers
|
|
bool has_contention(void) const { return contention; }
|
|
|
|
// is this DMA channel locked?
|
|
bool is_locked(void) const { return have_lock; }
|
|
|
|
// lock all shared DMA channels. Used on reboot
|
|
static void lock_all(void);
|
|
|
|
// display dma contention statistics as text buffer for @SYS/dma.txt
|
|
static void dma_info(ExpandingString &str);
|
|
|
|
// return true if a stream ID is shared between two peripherals
|
|
static bool is_shared(uint8_t stream_id);
|
|
|
|
private:
|
|
dma_allocate_fn_t allocate;
|
|
dma_allocate_fn_t deallocate;
|
|
uint8_t stream_id1;
|
|
uint8_t stream_id2;
|
|
bool have_lock;
|
|
|
|
// we set the contention flag if two drivers are fighting over a DMA channel.
|
|
// the UART driver uses this to change its max transmit size to reduce latency
|
|
bool contention;
|
|
|
|
// core of lock call, after semaphores gained
|
|
void lock_core(void);
|
|
|
|
// lock one stream
|
|
static bool lock_stream(uint8_t stream_id);
|
|
|
|
// unlock one stream
|
|
void unlock_stream(uint8_t stream_id, bool success);
|
|
|
|
// lock one stream, non-blocking
|
|
bool lock_stream_nonblocking(uint8_t stream_id);
|
|
|
|
static struct dma_lock {
|
|
// semaphore to ensure only one peripheral uses a DMA channel at a time
|
|
#if CH_CFG_USE_MUTEXES == TRUE
|
|
mutex_t mutex;
|
|
#endif // CH_CFG_USE_MUTEXES
|
|
|
|
// a de-allocation function that is called to release an existing user
|
|
dma_deallocate_fn_t deallocate;
|
|
|
|
// point to object that holds the allocation, if allocated
|
|
Shared_DMA *obj;
|
|
} locks[SHARED_DMA_MAX_STREAM_ID+1];
|
|
|
|
// contention statistics
|
|
static volatile struct dma_stats {
|
|
uint32_t contended_locks;
|
|
uint32_t uncontended_locks;
|
|
uint32_t transactions;
|
|
} *_contention_stats;
|
|
};
|
|
|
|
#endif // HAL_NO_SHARED_DMA
|