mirror of
https://bitbucket.org/librepilot/librepilot.git
synced 2024-12-12 20:08:48 +01:00
43b31efb76
Allocate per-instance data for drivers from the heap rather than as static variables from the .data segment. This converts > 800 bytes of RAM from being always consumed as static data into being allocated from the heap only when a particular feature is enabled in the hwsettings object. A minimal config (no receivers, flexi port disabled, main port disabled) leaves 2448 bytes of free heap. That's our new baseline. Approximate RAM (heap) costs of enabling various features: + 632 Serial Telemetry (includes 400 bytes of Rx/Tx buffers) + 108 PWM Rcvr + 152 PPM Rcvr + 112 Spektrum Rcvr + 24 S.Bus (Should be closer to 68 since driver is still using static memory) There are still some drivers that pre-allocate all of their memory as static data. It'll take some work to convert those over to dynamically allocating their instance data.
99 lines
2.0 KiB
C
99 lines
2.0 KiB
C
/* Project Includes */
|
|
#include "pios.h"
|
|
|
|
#if defined(PIOS_INCLUDE_RCVR)
|
|
|
|
#include <pios_rcvr_priv.h>
|
|
|
|
enum pios_rcvr_dev_magic {
|
|
PIOS_RCVR_DEV_MAGIC = 0x99aabbcc,
|
|
};
|
|
|
|
struct pios_rcvr_dev {
|
|
enum pios_rcvr_dev_magic magic;
|
|
uint32_t lower_id;
|
|
const struct pios_rcvr_driver * driver;
|
|
};
|
|
|
|
static bool PIOS_RCVR_validate(struct pios_rcvr_dev * rcvr_dev)
|
|
{
|
|
return (rcvr_dev->magic == PIOS_RCVR_DEV_MAGIC);
|
|
}
|
|
|
|
#if defined(PIOS_INCLUDE_FREERTOS)
|
|
static struct pios_rcvr_dev * PIOS_RCVR_alloc(void)
|
|
{
|
|
struct pios_rcvr_dev * rcvr_dev;
|
|
|
|
rcvr_dev = (struct pios_rcvr_dev *)pvPortMalloc(sizeof(*rcvr_dev));
|
|
if (!rcvr_dev) return (NULL);
|
|
|
|
rcvr_dev->magic = PIOS_RCVR_DEV_MAGIC;
|
|
return(rcvr_dev);
|
|
}
|
|
#else
|
|
static struct pios_rcvr_dev pios_rcvr_devs[PIOS_RCVR_MAX_DEVS];
|
|
static uint8_t pios_rcvr_num_devs;
|
|
static struct pios_rcvr_dev * PIOS_RCVR_alloc(void)
|
|
{
|
|
struct pios_rcvr_dev * rcvr_dev;
|
|
|
|
if (pios_rcvr_num_devs >= PIOS_RCVR_MAX_DEVS) {
|
|
return (NULL);
|
|
}
|
|
|
|
rcvr_dev = &pios_rcvr_devs[pios_rcvr_num_devs++];
|
|
rcvr_dev->magic = PIOS_RCVR_DEV_MAGIC;
|
|
|
|
return (rcvr_dev);
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* Initialises RCVR layer
|
|
* \param[out] handle
|
|
* \param[in] driver
|
|
* \param[in] id
|
|
* \return < 0 if initialisation failed
|
|
*/
|
|
int32_t PIOS_RCVR_Init(uint32_t * rcvr_id, const struct pios_rcvr_driver * driver, uint32_t lower_id)
|
|
{
|
|
PIOS_DEBUG_Assert(rcvr_id);
|
|
PIOS_DEBUG_Assert(driver);
|
|
|
|
struct pios_rcvr_dev * rcvr_dev;
|
|
|
|
rcvr_dev = (struct pios_rcvr_dev *) PIOS_RCVR_alloc();
|
|
if (!rcvr_dev) goto out_fail;
|
|
|
|
rcvr_dev->driver = driver;
|
|
rcvr_dev->lower_id = lower_id;
|
|
|
|
*rcvr_id = (uint32_t)rcvr_dev;
|
|
return(0);
|
|
|
|
out_fail:
|
|
return(-1);
|
|
}
|
|
|
|
int32_t PIOS_RCVR_Read(uint32_t rcvr_id, uint8_t channel)
|
|
{
|
|
struct pios_rcvr_dev * rcvr_dev = (struct pios_rcvr_dev *)rcvr_id;
|
|
|
|
if (!PIOS_RCVR_validate(rcvr_dev)) {
|
|
/* Undefined RCVR port for this board (see pios_board.c) */
|
|
PIOS_Assert(0);
|
|
}
|
|
|
|
PIOS_DEBUG_Assert(rcvr_dev->driver->read);
|
|
|
|
return rcvr_dev->driver->read(rcvr_dev->lower_id, channel);
|
|
}
|
|
|
|
#endif
|
|
|
|
/**
|
|
* @}
|
|
* @}
|
|
*/
|