| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962 |
- // SPDX-License-Identifier: GPL-2.0+
- /*
- * (C) Copyright 2015
- * Texas Instruments Incorporated - http://www.ti.com/
- */
- #define LOG_CATEGORY UCLASS_REMOTEPROC
- #define pr_fmt(fmt) "%s: " fmt, __func__
- #include <common.h>
- #include <elf.h>
- #include <errno.h>
- #include <log.h>
- #include <malloc.h>
- #include <virtio_ring.h>
- #include <remoteproc.h>
- #include <asm/io.h>
- #include <dm/device-internal.h>
- #include <dm.h>
- #include <dm/uclass.h>
- #include <dm/uclass-internal.h>
- #include <linux/compat.h>
- DECLARE_GLOBAL_DATA_PTR;
- struct resource_table {
- u32 ver;
- u32 num;
- u32 reserved[2];
- u32 offset[0];
- } __packed;
- typedef int (*handle_resource_t) (struct udevice *, void *, int offset, int avail);
- static struct resource_table *rsc_table;
- /**
- * for_each_remoteproc_device() - iterate through the list of rproc devices
- * @fn: check function to call per match, if this function returns fail,
- * iteration is aborted with the resultant error value
- * @skip_dev: Device to skip calling the callback about.
- * @data: Data to pass to the callback function
- *
- * Return: 0 if none of the callback returned a non 0 result, else returns the
- * result from the callback function
- */
- static int for_each_remoteproc_device(int (*fn) (struct udevice *dev,
- struct dm_rproc_uclass_pdata *uc_pdata,
- const void *data),
- struct udevice *skip_dev,
- const void *data)
- {
- struct udevice *dev;
- struct dm_rproc_uclass_pdata *uc_pdata;
- int ret;
- for (ret = uclass_find_first_device(UCLASS_REMOTEPROC, &dev); dev;
- ret = uclass_find_next_device(&dev)) {
- if (ret || dev == skip_dev)
- continue;
- uc_pdata = dev_get_uclass_plat(dev);
- ret = fn(dev, uc_pdata, data);
- if (ret)
- return ret;
- }
- return 0;
- }
- /**
- * _rproc_name_is_unique() - iteration helper to check if rproc name is unique
- * @dev: device that we are checking name for
- * @uc_pdata: uclass platform data
- * @data: compare data (this is the name we want to ensure is unique)
- *
- * Return: 0 is there is no match(is unique); if there is a match(we dont
- * have a unique name), return -EINVAL.
- */
- static int _rproc_name_is_unique(struct udevice *dev,
- struct dm_rproc_uclass_pdata *uc_pdata,
- const void *data)
- {
- const char *check_name = data;
- /* devices not yet populated with data - so skip them */
- if (!uc_pdata->name || !check_name)
- return 0;
- /* Return 0 to search further if we dont match */
- if (strlen(uc_pdata->name) != strlen(check_name))
- return 0;
- if (!strcmp(uc_pdata->name, check_name))
- return -EINVAL;
- return 0;
- }
- /**
- * rproc_name_is_unique() - Check if the rproc name is unique
- * @check_dev: Device we are attempting to ensure is unique
- * @check_name: Name we are trying to ensure is unique.
- *
- * Return: true if we have a unique name, false if name is not unique.
- */
- static bool rproc_name_is_unique(struct udevice *check_dev,
- const char *check_name)
- {
- int ret;
- ret = for_each_remoteproc_device(_rproc_name_is_unique,
- check_dev, check_name);
- return ret ? false : true;
- }
- /**
- * rproc_pre_probe() - Pre probe accessor for the uclass
- * @dev: device for which we are preprobing
- *
- * Parses and fills up the uclass pdata for use as needed by core and
- * remote proc drivers.
- *
- * Return: 0 if all wernt ok, else appropriate error value.
- */
- static int rproc_pre_probe(struct udevice *dev)
- {
- struct dm_rproc_uclass_pdata *uc_pdata;
- const struct dm_rproc_ops *ops;
- uc_pdata = dev_get_uclass_plat(dev);
- /* See if we need to populate via fdt */
- if (!dev_get_plat(dev)) {
- #if CONFIG_IS_ENABLED(OF_CONTROL)
- bool tmp;
- debug("'%s': using fdt\n", dev->name);
- uc_pdata->name = dev_read_string(dev, "remoteproc-name");
- /* Default is internal memory mapped */
- uc_pdata->mem_type = RPROC_INTERNAL_MEMORY_MAPPED;
- tmp = dev_read_bool(dev, "remoteproc-internal-memory-mapped");
- if (tmp)
- uc_pdata->mem_type = RPROC_INTERNAL_MEMORY_MAPPED;
- #else
- /* Nothing much we can do about this, can we? */
- return -EINVAL;
- #endif
- } else {
- struct dm_rproc_uclass_pdata *pdata = dev_get_plat(dev);
- debug("'%s': using legacy data\n", dev->name);
- if (pdata->name)
- uc_pdata->name = pdata->name;
- uc_pdata->mem_type = pdata->mem_type;
- uc_pdata->driver_plat_data = pdata->driver_plat_data;
- }
- /* Else try using device Name */
- if (!uc_pdata->name)
- uc_pdata->name = dev->name;
- if (!uc_pdata->name) {
- debug("Unnamed device!");
- return -EINVAL;
- }
- if (!rproc_name_is_unique(dev, uc_pdata->name)) {
- debug("%s duplicate name '%s'\n", dev->name, uc_pdata->name);
- return -EINVAL;
- }
- ops = rproc_get_ops(dev);
- if (!ops) {
- debug("%s driver has no ops?\n", dev->name);
- return -EINVAL;
- }
- if (!ops->load || !ops->start) {
- debug("%s driver has missing mandatory ops?\n", dev->name);
- return -EINVAL;
- }
- return 0;
- }
- /**
- * rproc_post_probe() - post probe accessor for the uclass
- * @dev: deivce we finished probing
- *
- * initiate init function after the probe is completed. This allows
- * the remote processor drivers to split up the initializations between
- * probe and init as needed.
- *
- * Return: if the remote proc driver has a init routine, invokes it and
- * hands over the return value. overall, 0 if all went well, else appropriate
- * error value.
- */
- static int rproc_post_probe(struct udevice *dev)
- {
- const struct dm_rproc_ops *ops;
- ops = rproc_get_ops(dev);
- if (!ops) {
- debug("%s driver has no ops?\n", dev->name);
- return -EINVAL;
- }
- if (ops->init)
- return ops->init(dev);
- return 0;
- }
- /**
- * rproc_add_res() - After parsing the resource table add the mappings
- * @dev: device we finished probing
- * @mapping: rproc_mem_entry for the resource
- *
- * Return: if the remote proc driver has a add_res routine, invokes it and
- * hands over the return value. overall, 0 if all went well, else appropriate
- * error value.
- */
- static int rproc_add_res(struct udevice *dev, struct rproc_mem_entry *mapping)
- {
- const struct dm_rproc_ops *ops = rproc_get_ops(dev);
- if (!ops->add_res)
- return -ENOSYS;
- return ops->add_res(dev, mapping);
- }
- /**
- * rproc_alloc_mem() - After parsing the resource table allocat mem
- * @dev: device we finished probing
- * @len: rproc_mem_entry for the resource
- * @align: alignment for the resource
- *
- * Return: if the remote proc driver has a add_res routine, invokes it and
- * hands over the return value. overall, 0 if all went well, else appropriate
- * error value.
- */
- static void *rproc_alloc_mem(struct udevice *dev, unsigned long len,
- unsigned long align)
- {
- const struct dm_rproc_ops *ops;
- ops = rproc_get_ops(dev);
- if (!ops) {
- debug("%s driver has no ops?\n", dev->name);
- return NULL;
- }
- if (ops->alloc_mem)
- return ops->alloc_mem(dev, len, align);
- return NULL;
- }
- /**
- * rproc_config_pagetable() - Configure page table for remote processor
- * @dev: device we finished probing
- * @virt: Virtual address of the resource
- * @phys: Physical address the resource
- * @len: length the resource
- *
- * Return: if the remote proc driver has a add_res routine, invokes it and
- * hands over the return value. overall, 0 if all went well, else appropriate
- * error value.
- */
- static int rproc_config_pagetable(struct udevice *dev, unsigned int virt,
- unsigned int phys, unsigned int len)
- {
- const struct dm_rproc_ops *ops;
- ops = rproc_get_ops(dev);
- if (!ops) {
- debug("%s driver has no ops?\n", dev->name);
- return -EINVAL;
- }
- if (ops->config_pagetable)
- return ops->config_pagetable(dev, virt, phys, len);
- return 0;
- }
- UCLASS_DRIVER(rproc) = {
- .id = UCLASS_REMOTEPROC,
- .name = "remoteproc",
- .flags = DM_UC_FLAG_SEQ_ALIAS,
- .pre_probe = rproc_pre_probe,
- .post_probe = rproc_post_probe,
- .per_device_plat_auto = sizeof(struct dm_rproc_uclass_pdata),
- };
- /* Remoteproc subsystem access functions */
- /**
- * _rproc_probe_dev() - iteration helper to probe a rproc device
- * @dev: device to probe
- * @uc_pdata: uclass data allocated for the device
- * @data: unused
- *
- * Return: 0 if all ok, else appropriate error value.
- */
- static int _rproc_probe_dev(struct udevice *dev,
- struct dm_rproc_uclass_pdata *uc_pdata,
- const void *data)
- {
- int ret;
- ret = device_probe(dev);
- if (ret)
- debug("%s: Failed to initialize - %d\n", dev->name, ret);
- return ret;
- }
- /**
- * _rproc_dev_is_probed() - check if the device has been probed
- * @dev: device to check
- * @uc_pdata: unused
- * @data: unused
- *
- * Return: -EAGAIN if not probed else return 0
- */
- static int _rproc_dev_is_probed(struct udevice *dev,
- struct dm_rproc_uclass_pdata *uc_pdata,
- const void *data)
- {
- if (dev_get_flags(dev) & DM_FLAG_ACTIVATED)
- return 0;
- return -EAGAIN;
- }
- bool rproc_is_initialized(void)
- {
- int ret = for_each_remoteproc_device(_rproc_dev_is_probed, NULL, NULL);
- return ret ? false : true;
- }
- int rproc_init(void)
- {
- int ret;
- if (rproc_is_initialized()) {
- debug("Already initialized\n");
- return -EINVAL;
- }
- ret = for_each_remoteproc_device(_rproc_probe_dev, NULL, NULL);
- return ret;
- }
- int rproc_dev_init(int id)
- {
- struct udevice *dev = NULL;
- int ret;
- ret = uclass_get_device_by_seq(UCLASS_REMOTEPROC, id, &dev);
- if (ret) {
- debug("Unknown remote processor id '%d' requested(%d)\n",
- id, ret);
- return ret;
- }
- ret = device_probe(dev);
- if (ret)
- debug("%s: Failed to initialize - %d\n", dev->name, ret);
- return ret;
- }
- int rproc_load(int id, ulong addr, ulong size)
- {
- struct udevice *dev = NULL;
- struct dm_rproc_uclass_pdata *uc_pdata;
- const struct dm_rproc_ops *ops;
- int ret;
- ret = uclass_get_device_by_seq(UCLASS_REMOTEPROC, id, &dev);
- if (ret) {
- debug("Unknown remote processor id '%d' requested(%d)\n",
- id, ret);
- return ret;
- }
- uc_pdata = dev_get_uclass_plat(dev);
- ops = rproc_get_ops(dev);
- if (!ops) {
- debug("%s driver has no ops?\n", dev->name);
- return -EINVAL;
- }
- debug("Loading to '%s' from address 0x%08lX size of %lu bytes\n",
- uc_pdata->name, addr, size);
- if (ops->load)
- return ops->load(dev, addr, size);
- debug("%s: data corruption?? mandatory function is missing!\n",
- dev->name);
- return -EINVAL;
- };
- /*
- * Completely internal helper enums..
- * Keeping this isolated helps this code evolve independent of other
- * parts..
- */
- enum rproc_ops {
- RPROC_START,
- RPROC_STOP,
- RPROC_RESET,
- RPROC_PING,
- RPROC_RUNNING,
- };
- /**
- * _rproc_ops_wrapper() - wrapper for invoking remote proc driver callback
- * @id: id of the remote processor
- * @op: one of rproc_ops that indicate what operation to invoke
- *
- * Most of the checks and verification for remoteproc operations are more
- * or less same for almost all operations. This allows us to put a wrapper
- * and use the common checks to allow the driver to function appropriately.
- *
- * Return: 0 if all ok, else appropriate error value.
- */
- static int _rproc_ops_wrapper(int id, enum rproc_ops op)
- {
- struct udevice *dev = NULL;
- struct dm_rproc_uclass_pdata *uc_pdata;
- const struct dm_rproc_ops *ops;
- int (*fn)(struct udevice *dev);
- bool mandatory = false;
- char *op_str;
- int ret;
- ret = uclass_get_device_by_seq(UCLASS_REMOTEPROC, id, &dev);
- if (ret) {
- debug("Unknown remote processor id '%d' requested(%d)\n",
- id, ret);
- return ret;
- }
- uc_pdata = dev_get_uclass_plat(dev);
- ops = rproc_get_ops(dev);
- if (!ops) {
- debug("%s driver has no ops?\n", dev->name);
- return -EINVAL;
- }
- switch (op) {
- case RPROC_START:
- fn = ops->start;
- mandatory = true;
- op_str = "Starting";
- break;
- case RPROC_STOP:
- fn = ops->stop;
- op_str = "Stopping";
- break;
- case RPROC_RESET:
- fn = ops->reset;
- op_str = "Resetting";
- break;
- case RPROC_RUNNING:
- fn = ops->is_running;
- op_str = "Checking if running:";
- break;
- case RPROC_PING:
- fn = ops->ping;
- op_str = "Pinging";
- break;
- default:
- debug("what is '%d' operation??\n", op);
- return -EINVAL;
- }
- debug("%s %s...\n", op_str, uc_pdata->name);
- if (fn)
- return fn(dev);
- if (mandatory)
- debug("%s: data corruption?? mandatory function is missing!\n",
- dev->name);
- return -ENOSYS;
- }
- int rproc_start(int id)
- {
- return _rproc_ops_wrapper(id, RPROC_START);
- };
- int rproc_stop(int id)
- {
- return _rproc_ops_wrapper(id, RPROC_STOP);
- };
- int rproc_reset(int id)
- {
- return _rproc_ops_wrapper(id, RPROC_RESET);
- };
- int rproc_ping(int id)
- {
- return _rproc_ops_wrapper(id, RPROC_PING);
- };
- int rproc_is_running(int id)
- {
- return _rproc_ops_wrapper(id, RPROC_RUNNING);
- };
- static int handle_trace(struct udevice *dev, struct fw_rsc_trace *rsc,
- int offset, int avail)
- {
- if (sizeof(*rsc) > avail) {
- debug("trace rsc is truncated\n");
- return -EINVAL;
- }
- /*
- * make sure reserved bytes are zeroes
- */
- if (rsc->reserved) {
- debug("trace rsc has non zero reserved bytes\n");
- return -EINVAL;
- }
- debug("trace rsc: da 0x%x, len 0x%x\n", rsc->da, rsc->len);
- return 0;
- }
- static int handle_devmem(struct udevice *dev, struct fw_rsc_devmem *rsc,
- int offset, int avail)
- {
- struct rproc_mem_entry *mapping;
- if (sizeof(*rsc) > avail) {
- debug("devmem rsc is truncated\n");
- return -EINVAL;
- }
- /*
- * make sure reserved bytes are zeroes
- */
- if (rsc->reserved) {
- debug("devmem rsc has non zero reserved bytes\n");
- return -EINVAL;
- }
- debug("devmem rsc: pa 0x%x, da 0x%x, len 0x%x\n",
- rsc->pa, rsc->da, rsc->len);
- rproc_config_pagetable(dev, rsc->da, rsc->pa, rsc->len);
- mapping = kzalloc(sizeof(*mapping), GFP_KERNEL);
- if (!mapping)
- return -ENOMEM;
- /*
- * We'll need this info later when we'll want to unmap everything
- * (e.g. on shutdown).
- *
- * We can't trust the remote processor not to change the resource
- * table, so we must maintain this info independently.
- */
- mapping->dma = rsc->pa;
- mapping->da = rsc->da;
- mapping->len = rsc->len;
- rproc_add_res(dev, mapping);
- debug("mapped devmem pa 0x%x, da 0x%x, len 0x%x\n",
- rsc->pa, rsc->da, rsc->len);
- return 0;
- }
- static int handle_carveout(struct udevice *dev, struct fw_rsc_carveout *rsc,
- int offset, int avail)
- {
- struct rproc_mem_entry *mapping;
- if (sizeof(*rsc) > avail) {
- debug("carveout rsc is truncated\n");
- return -EINVAL;
- }
- /*
- * make sure reserved bytes are zeroes
- */
- if (rsc->reserved) {
- debug("carveout rsc has non zero reserved bytes\n");
- return -EINVAL;
- }
- debug("carveout rsc: da %x, pa %x, len %x, flags %x\n",
- rsc->da, rsc->pa, rsc->len, rsc->flags);
- rsc->pa = (uintptr_t)rproc_alloc_mem(dev, rsc->len, 8);
- if (!rsc->pa) {
- debug
- ("failed to allocate carveout rsc: da %x, pa %x, len %x, flags %x\n",
- rsc->da, rsc->pa, rsc->len, rsc->flags);
- return -ENOMEM;
- }
- rproc_config_pagetable(dev, rsc->da, rsc->pa, rsc->len);
- /*
- * Ok, this is non-standard.
- *
- * Sometimes we can't rely on the generic iommu-based DMA API
- * to dynamically allocate the device address and then set the IOMMU
- * tables accordingly, because some remote processors might
- * _require_ us to use hard coded device addresses that their
- * firmware was compiled with.
- *
- * In this case, we must use the IOMMU API directly and map
- * the memory to the device address as expected by the remote
- * processor.
- *
- * Obviously such remote processor devices should not be configured
- * to use the iommu-based DMA API: we expect 'dma' to contain the
- * physical address in this case.
- */
- mapping = kzalloc(sizeof(*mapping), GFP_KERNEL);
- if (!mapping)
- return -ENOMEM;
- /*
- * We'll need this info later when we'll want to unmap
- * everything (e.g. on shutdown).
- *
- * We can't trust the remote processor not to change the
- * resource table, so we must maintain this info independently.
- */
- mapping->dma = rsc->pa;
- mapping->da = rsc->da;
- mapping->len = rsc->len;
- rproc_add_res(dev, mapping);
- debug("carveout mapped 0x%x to 0x%x\n", rsc->da, rsc->pa);
- return 0;
- }
- #define RPROC_PAGE_SHIFT 12
- #define RPROC_PAGE_SIZE BIT(RPROC_PAGE_SHIFT)
- #define RPROC_PAGE_ALIGN(x) (((x) + (RPROC_PAGE_SIZE - 1)) & ~(RPROC_PAGE_SIZE - 1))
- static int alloc_vring(struct udevice *dev, struct fw_rsc_vdev *rsc, int i)
- {
- struct fw_rsc_vdev_vring *vring = &rsc->vring[i];
- int size;
- int order;
- void *pa;
- debug("vdev rsc: vring%d: da %x, qsz %d, align %d\n",
- i, vring->da, vring->num, vring->align);
- /*
- * verify queue size and vring alignment are sane
- */
- if (!vring->num || !vring->align) {
- debug("invalid qsz (%d) or alignment (%d)\n", vring->num,
- vring->align);
- return -EINVAL;
- }
- /*
- * actual size of vring (in bytes)
- */
- size = RPROC_PAGE_ALIGN(vring_size(vring->num, vring->align));
- order = vring->align >> RPROC_PAGE_SHIFT;
- pa = rproc_alloc_mem(dev, size, order);
- if (!pa) {
- debug("failed to allocate vring rsc\n");
- return -ENOMEM;
- }
- debug("alloc_mem(%#x, %d): %p\n", size, order, pa);
- vring->da = (uintptr_t)pa;
- return !pa;
- }
- static int handle_vdev(struct udevice *dev, struct fw_rsc_vdev *rsc,
- int offset, int avail)
- {
- int i, ret;
- void *pa;
- /*
- * make sure resource isn't truncated
- */
- if (sizeof(*rsc) + rsc->num_of_vrings * sizeof(struct fw_rsc_vdev_vring)
- + rsc->config_len > avail) {
- debug("vdev rsc is truncated\n");
- return -EINVAL;
- }
- /*
- * make sure reserved bytes are zeroes
- */
- if (rsc->reserved[0] || rsc->reserved[1]) {
- debug("vdev rsc has non zero reserved bytes\n");
- return -EINVAL;
- }
- debug("vdev rsc: id %d, dfeatures %x, cfg len %d, %d vrings\n",
- rsc->id, rsc->dfeatures, rsc->config_len, rsc->num_of_vrings);
- /*
- * we currently support only two vrings per rvdev
- */
- if (rsc->num_of_vrings > 2) {
- debug("too many vrings: %d\n", rsc->num_of_vrings);
- return -EINVAL;
- }
- /*
- * allocate the vrings
- */
- for (i = 0; i < rsc->num_of_vrings; i++) {
- ret = alloc_vring(dev, rsc, i);
- if (ret)
- goto alloc_error;
- }
- pa = rproc_alloc_mem(dev, RPMSG_TOTAL_BUF_SPACE, 6);
- if (!pa) {
- debug("failed to allocate vdev rsc\n");
- return -ENOMEM;
- }
- debug("vring buffer alloc_mem(%#x, 6): %p\n", RPMSG_TOTAL_BUF_SPACE,
- pa);
- return 0;
- alloc_error:
- return ret;
- }
- /*
- * A lookup table for resource handlers. The indices are defined in
- * enum fw_resource_type.
- */
- static handle_resource_t loading_handlers[RSC_LAST] = {
- [RSC_CARVEOUT] = (handle_resource_t)handle_carveout,
- [RSC_DEVMEM] = (handle_resource_t)handle_devmem,
- [RSC_TRACE] = (handle_resource_t)handle_trace,
- [RSC_VDEV] = (handle_resource_t)handle_vdev,
- };
- /*
- * handle firmware resource entries before booting the remote processor
- */
- static int handle_resources(struct udevice *dev, int len,
- handle_resource_t handlers[RSC_LAST])
- {
- handle_resource_t handler;
- int ret = 0, i;
- for (i = 0; i < rsc_table->num; i++) {
- int offset = rsc_table->offset[i];
- struct fw_rsc_hdr *hdr = (void *)rsc_table + offset;
- int avail = len - offset - sizeof(*hdr);
- void *rsc = (void *)hdr + sizeof(*hdr);
- /*
- * make sure table isn't truncated
- */
- if (avail < 0) {
- debug("rsc table is truncated\n");
- return -EINVAL;
- }
- debug("rsc: type %d\n", hdr->type);
- if (hdr->type >= RSC_LAST) {
- debug("unsupported resource %d\n", hdr->type);
- continue;
- }
- handler = handlers[hdr->type];
- if (!handler)
- continue;
- ret = handler(dev, rsc, offset + sizeof(*hdr), avail);
- if (ret)
- break;
- }
- return ret;
- }
- static int
- handle_intmem_to_l3_mapping(struct udevice *dev,
- struct rproc_intmem_to_l3_mapping *l3_mapping)
- {
- u32 i = 0;
- for (i = 0; i < l3_mapping->num_entries; i++) {
- struct l3_map *curr_map = &l3_mapping->mappings[i];
- struct rproc_mem_entry *mapping;
- mapping = kzalloc(sizeof(*mapping), GFP_KERNEL);
- if (!mapping)
- return -ENOMEM;
- mapping->dma = curr_map->l3_addr;
- mapping->da = curr_map->priv_addr;
- mapping->len = curr_map->len;
- rproc_add_res(dev, mapping);
- }
- return 0;
- }
- static Elf32_Shdr *rproc_find_table(unsigned int addr)
- {
- Elf32_Ehdr *ehdr; /* Elf header structure pointer */
- Elf32_Shdr *shdr; /* Section header structure pointer */
- Elf32_Shdr sectionheader;
- int i;
- u8 *elf_data;
- char *name_table;
- struct resource_table *ptable;
- ehdr = (Elf32_Ehdr *)(uintptr_t)addr;
- elf_data = (u8 *)ehdr;
- shdr = (Elf32_Shdr *)(elf_data + ehdr->e_shoff);
- memcpy(§ionheader, &shdr[ehdr->e_shstrndx], sizeof(sectionheader));
- name_table = (char *)(elf_data + sectionheader.sh_offset);
- for (i = 0; i < ehdr->e_shnum; i++, shdr++) {
- memcpy(§ionheader, shdr, sizeof(sectionheader));
- u32 size = sectionheader.sh_size;
- u32 offset = sectionheader.sh_offset;
- if (strcmp
- (name_table + sectionheader.sh_name, ".resource_table"))
- continue;
- ptable = (struct resource_table *)(elf_data + offset);
- /*
- * make sure table has at least the header
- */
- if (sizeof(struct resource_table) > size) {
- debug("header-less resource table\n");
- return NULL;
- }
- /*
- * we don't support any version beyond the first
- */
- if (ptable->ver != 1) {
- debug("unsupported fw ver: %d\n", ptable->ver);
- return NULL;
- }
- /*
- * make sure reserved bytes are zeroes
- */
- if (ptable->reserved[0] || ptable->reserved[1]) {
- debug("non zero reserved bytes\n");
- return NULL;
- }
- /*
- * make sure the offsets array isn't truncated
- */
- if (ptable->num * sizeof(ptable->offset[0]) +
- sizeof(struct resource_table) > size) {
- debug("resource table incomplete\n");
- return NULL;
- }
- return shdr;
- }
- return NULL;
- }
- struct resource_table *rproc_find_resource_table(struct udevice *dev,
- unsigned int addr,
- int *tablesz)
- {
- Elf32_Shdr *shdr;
- Elf32_Shdr sectionheader;
- struct resource_table *ptable;
- u8 *elf_data = (u8 *)(uintptr_t)addr;
- shdr = rproc_find_table(addr);
- if (!shdr) {
- debug("%s: failed to get resource section header\n", __func__);
- return NULL;
- }
- memcpy(§ionheader, shdr, sizeof(sectionheader));
- ptable = (struct resource_table *)(elf_data + sectionheader.sh_offset);
- if (tablesz)
- *tablesz = sectionheader.sh_size;
- return ptable;
- }
- unsigned long rproc_parse_resource_table(struct udevice *dev, struct rproc *cfg)
- {
- struct resource_table *ptable = NULL;
- int tablesz;
- int ret;
- unsigned long addr;
- addr = cfg->load_addr;
- ptable = rproc_find_resource_table(dev, addr, &tablesz);
- if (!ptable) {
- debug("%s : failed to find resource table\n", __func__);
- return 0;
- }
- debug("%s : found resource table\n", __func__);
- rsc_table = kzalloc(tablesz, GFP_KERNEL);
- if (!rsc_table) {
- debug("resource table alloc failed!\n");
- return 0;
- }
- /*
- * Copy the resource table into a local buffer before handling the
- * resource table.
- */
- memcpy(rsc_table, ptable, tablesz);
- if (cfg->intmem_to_l3_mapping)
- handle_intmem_to_l3_mapping(dev, cfg->intmem_to_l3_mapping);
- ret = handle_resources(dev, tablesz, loading_handlers);
- if (ret) {
- debug("handle_resources failed: %d\n", ret);
- return 0;
- }
- /*
- * Instead of trying to mimic the kernel flow of copying the
- * processed resource table into its post ELF load location in DDR
- * copying it into its original location.
- */
- memcpy(ptable, rsc_table, tablesz);
- free(rsc_table);
- rsc_table = NULL;
- return 1;
- }
|