grlib: Customizable allocation in ambapp_scan()

Make the memory allocations in ambapp_scan() customizable via the new struct
ambapp_context parameter which generalizes the memory copy handler.
This commit is contained in:
Sebastian Huber
2021-06-10 13:25:26 +02:00
parent f89a527336
commit c4c8d3fd50
2 changed files with 42 additions and 26 deletions

View File

@@ -15,6 +15,15 @@
#ifndef __AMBAPP_H__
#define __AMBAPP_H__
#include <stddef.h>
/* Include VENDOR and DEVICE definitions */
#include "ambapp_ids.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @defgroup amba AMBA
*
@@ -25,13 +34,6 @@
* @{
*/
/* Include VENDOR and DEVICE definitions */
#include "ambapp_ids.h"
#ifdef __cplusplus
extern "C" {
#endif
/* Max supported AHB buses */
#define AHB_BUS_MAX 6
@@ -181,6 +183,11 @@ typedef void *(*ambapp_memcpy_t)(
struct ambapp_bus *abus /* Optional AMBA Bus pointer */
);
struct ambapp_context {
ambapp_memcpy_t copy_from_device;
void *(*alloc)(size_t);
};
/* Scan a AMBA Plug & Play bus and create all device structures describing the
* the devices. The devices will form a tree, where every node describes one
* interface. The resulting tree is placed in the location pointed to by root.
@@ -188,16 +195,16 @@ typedef void *(*ambapp_memcpy_t)(
* Since it the tree is located in RAM it is easier to work with AMBA buses
* that is located over PCI and SpaceWire etc.
*
* \param abus Resulting device node tree root is stored here.
* \param ioarea The IO-AREA where Plug & Play information can be found.
* \param parent Used internally when recursing down a bridge. Set to NULL.
* \param ctx The scan context. May be NULL.
* \param mmaps Is used to perform address translation if needed.
* \param root Resulting device node tree root is stored here.
*
*/
extern int ambapp_scan(
struct ambapp_bus *abus,
unsigned int ioarea,
ambapp_memcpy_t memfunc,
const struct ambapp_context *ctx,
struct ambapp_mmap *mmaps
);

View File

@@ -23,7 +23,8 @@
#define AMBA_APB_SLAVES 16
/* Allocate one AMBA device */
static struct ambapp_dev *ambapp_alloc_dev_struct(int dev_type)
static struct ambapp_dev *
ambapp_alloc_dev_struct(const struct ambapp_context *ctx, int dev_type)
{
struct ambapp_dev *dev;
size_t size = sizeof(*dev);
@@ -32,9 +33,11 @@ static struct ambapp_dev *ambapp_alloc_dev_struct(int dev_type)
size += sizeof(struct ambapp_apb_info);
else
size += sizeof(struct ambapp_ahb_info); /* AHB */
dev = grlib_calloc(1, size);
if (dev != NULL)
dev = (*ctx->alloc)(size);
if (dev != NULL) {
dev = memset(dev, 0, size);
dev->dev_type = dev_type;
}
return dev;
}
@@ -145,7 +148,7 @@ static int ambapp_add_ahbbus(
static int ambapp_scan2(
struct ambapp_bus *abus,
unsigned int ioarea,
ambapp_memcpy_t memfunc,
const struct ambapp_context *ctx,
struct ambapp_dev *parent,
struct ambapp_dev **root
)
@@ -176,12 +179,12 @@ static int ambapp_scan2(
/* AHB MASTERS */
ahb = (struct ambapp_pnp_ahb *) (ioarea | AMBA_CONF_AREA);
for (i = 0; i < maxloops; i++, ahb++) {
memfunc(&ahb_buf, ahb, sizeof(struct ambapp_pnp_ahb), abus);
(*ctx->copy_from_device)(&ahb_buf, ahb, sizeof(struct ambapp_pnp_ahb), abus);
if (ahb_buf.id == 0)
continue;
/* An AHB device present here */
dev = ambapp_alloc_dev_struct(DEV_AHB_MST);
dev = ambapp_alloc_dev_struct(ctx, DEV_AHB_MST);
if (!dev)
return -1;
@@ -200,12 +203,12 @@ static int ambapp_scan2(
ahb = (struct ambapp_pnp_ahb *)
(ioarea | AMBA_CONF_AREA | AMBA_AHB_SLAVE_CONF_AREA);
for (i = 0; i < maxloops; i++, ahb++) {
memfunc(&ahb_buf, ahb, sizeof(struct ambapp_pnp_ahb), abus);
(*ctx->copy_from_device)(&ahb_buf, ahb, sizeof(struct ambapp_pnp_ahb), abus);
if (ahb_buf.id == 0)
continue;
/* An AHB device present here */
dev = ambapp_alloc_dev_struct(DEV_AHB_SLV);
dev = ambapp_alloc_dev_struct(ctx, DEV_AHB_SLV);
if (!dev)
return -1;
@@ -235,7 +238,7 @@ static int ambapp_scan2(
bridge_adr = ambapp_addr_from(abus->mmaps,
ahb_info->custom[1]);
/* Scan next bus if not already scanned */
if (ambapp_scan2(abus, bridge_adr, memfunc, dev,
if (ambapp_scan2(abus, bridge_adr, ctx, dev,
&dev->children))
return -1;
}
@@ -251,11 +254,11 @@ static int ambapp_scan2(
apb = (struct ambapp_pnp_apb *)
(apbbase | AMBA_CONF_AREA);
for (j=0; j<AMBA_APB_SLAVES; j++, apb++) {
memfunc(&apb_buf, apb, sizeof(*apb), abus);
(*ctx->copy_from_device)(&apb_buf, apb, sizeof(*apb), abus);
if (apb_buf.id == 0)
continue;
apbdev = ambapp_alloc_dev_struct(DEV_APB_SLV);
apbdev = ambapp_alloc_dev_struct(ctx, DEV_APB_SLV);
if (!apbdev)
return -1;
@@ -279,22 +282,28 @@ static int ambapp_scan2(
return 0;
}
static const struct ambapp_context default_ctx = {
.copy_from_device = (ambapp_memcpy_t)memcpy,
.alloc = rtems_malloc
};
/* Build AMBA Plug & Play device graph */
int ambapp_scan(
struct ambapp_bus *abus,
unsigned int ioarea,
ambapp_memcpy_t memfunc,
const struct ambapp_context *ctx,
struct ambapp_mmap *mmaps
)
{
memset(abus, 0, sizeof(*abus));
abus->mmaps = mmaps;
/* Default to memcpy() */
if (!memfunc)
memfunc = (ambapp_memcpy_t)memcpy;
if (ctx == NULL) {
ctx = &default_ctx;
}
return ambapp_scan2(abus, ioarea, memfunc, NULL, &abus->root);
return ambapp_scan2(abus, ioarea, ctx, NULL, &abus->root);
}
/* Match search options againt device */