|
@@ -0,0 +1,871 @@
|
|
|
+/*
|
|
|
+ * Remoteproc Framework
|
|
|
+ *
|
|
|
+ * Copyright(c) 2018 Xilinx Ltd.
|
|
|
+ * Copyright(c) 2011 Texas Instruments, Inc.
|
|
|
+ * Copyright(c) 2011 Google, Inc.
|
|
|
+ * All rights reserved.
|
|
|
+ *
|
|
|
+ * SPDX-License-Identifier: BSD-3-Clause
|
|
|
+ */
|
|
|
+
|
|
|
+#ifndef REMOTEPROC_H
|
|
|
+#define REMOTEPROC_H
|
|
|
+
|
|
|
+#include <metal/io.h>
|
|
|
+#include <metal/mutex.h>
|
|
|
+#include <openamp/compiler.h>
|
|
|
+
|
|
|
+#if defined __cplusplus
|
|
|
+extern "C" {
|
|
|
+#endif
|
|
|
+
|
|
|
+#define RSC_NOTIFY_ID_ANY 0xFFFFFFFFUL
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct resource_table - firmware resource table header
|
|
|
+ * @ver: version number
|
|
|
+ * @num: number of resource entries
|
|
|
+ * @reserved: reserved (must be zero)
|
|
|
+ * @offset: array of offsets pointing at the various resource entries
|
|
|
+ *
|
|
|
+ * A resource table is essentially a list of system resources required
|
|
|
+ * by the remote remote_proc. It may also include configuration entries.
|
|
|
+ * If needed, the remote remote_proc firmware should contain this table
|
|
|
+ * as a dedicated ".resource_table" ELF section.
|
|
|
+ *
|
|
|
+ * Some resources entries are mere announcements, where the host is informed
|
|
|
+ * of specific remoteproc configuration. Other entries require the host to
|
|
|
+ * do something (e.g. allocate a system resource). Sometimes a negotiation
|
|
|
+ * is expected, where the firmware requests a resource, and once allocated,
|
|
|
+ * the host should provide back its details (e.g. address of an allocated
|
|
|
+ * memory region).
|
|
|
+ *
|
|
|
+ * The header of the resource table, as expressed by this structure,
|
|
|
+ * contains a version number (should we need to change this format in the
|
|
|
+ * future), the number of available resource entries, and their offsets
|
|
|
+ * in the table.
|
|
|
+ *
|
|
|
+ * Immediately following this header are the resource entries themselves,
|
|
|
+ * each of which begins with a resource entry header (as described below).
|
|
|
+ */
|
|
|
+OPENAMP_PACKED_BEGIN
|
|
|
+struct resource_table {
|
|
|
+ uint32_t ver;
|
|
|
+ uint32_t num;
|
|
|
+ uint32_t reserved[2];
|
|
|
+ uint32_t offset[0];
|
|
|
+} OPENAMP_PACKED_END;
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct fw_rsc_hdr - firmware resource entry header
|
|
|
+ * @type: resource type
|
|
|
+ * @data: resource data
|
|
|
+ *
|
|
|
+ * Every resource entry begins with a 'struct fw_rsc_hdr' header providing
|
|
|
+ * its @type. The content of the entry itself will immediately follow
|
|
|
+ * this header, and it should be parsed according to the resource type.
|
|
|
+ */
|
|
|
+OPENAMP_PACKED_BEGIN
|
|
|
+struct fw_rsc_hdr {
|
|
|
+ uint32_t type;
|
|
|
+ uint8_t data[0];
|
|
|
+} OPENAMP_PACKED_END;
|
|
|
+
|
|
|
+/**
|
|
|
+ * enum fw_resource_type - types of resource entries
|
|
|
+ *
|
|
|
+ * @RSC_CARVEOUT: request for allocation of a physically contiguous
|
|
|
+ * memory region.
|
|
|
+ * @RSC_DEVMEM: request to iommu_map a memory-based peripheral.
|
|
|
+ * @RSC_TRACE: announces the availability of a trace buffer into which
|
|
|
+ * the remote remote_proc will be writing logs.
|
|
|
+ * @RSC_VDEV: declare support for a virtio device, and serve as its
|
|
|
+ * virtio header.
|
|
|
+ * @RSC_VENDOR_START: start of the vendor specific resource types range
|
|
|
+ * @RSC_VENDOR_END : end of the vendor specific resource types range
|
|
|
+ * @RSC_LAST: just keep this one at the end
|
|
|
+ *
|
|
|
+ * For more details regarding a specific resource type, please see its
|
|
|
+ * dedicated structure below.
|
|
|
+ *
|
|
|
+ * Please note that these values are used as indices to the rproc_handle_rsc
|
|
|
+ * lookup table, so please keep them sane. Moreover, @RSC_LAST is used to
|
|
|
+ * check the validity of an index before the lookup table is accessed, so
|
|
|
+ * please update it as needed.
|
|
|
+ */
|
|
|
+enum fw_resource_type {
|
|
|
+ RSC_CARVEOUT = 0,
|
|
|
+ RSC_DEVMEM = 1,
|
|
|
+ RSC_TRACE = 2,
|
|
|
+ RSC_VDEV = 3,
|
|
|
+ RSC_RPROC_MEM = 4,
|
|
|
+ RSC_FW_CHKSUM = 5,
|
|
|
+ RSC_LAST = 6,
|
|
|
+ RSC_VENDOR_START = 128,
|
|
|
+ RSC_VENDOR_END = 512,
|
|
|
+};
|
|
|
+
|
|
|
+#define FW_RSC_ADDR_ANY (0xFFFFFFFFFFFFFFFF)
|
|
|
+#define FW_RSC_U32_ADDR_ANY (0xFFFFFFFF)
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct fw_rsc_carveout - physically contiguous memory request
|
|
|
+ * @da: device address
|
|
|
+ * @pa: physical address
|
|
|
+ * @len: length (in bytes)
|
|
|
+ * @flags: iommu protection flags
|
|
|
+ * @reserved: reserved (must be zero)
|
|
|
+ * @name: human-readable name of the requested memory region
|
|
|
+ *
|
|
|
+ * This resource entry requests the host to allocate a physically contiguous
|
|
|
+ * memory region.
|
|
|
+ *
|
|
|
+ * These request entries should precede other firmware resource entries,
|
|
|
+ * as other entries might request placing other data objects inside
|
|
|
+ * these memory regions (e.g. data/code segments, trace resource entries, ...).
|
|
|
+ *
|
|
|
+ * Allocating memory this way helps utilizing the reserved physical memory
|
|
|
+ * (e.g. CMA) more efficiently, and also minimizes the number of TLB entries
|
|
|
+ * needed to map it (in case @rproc is using an IOMMU). Reducing the TLB
|
|
|
+ * pressure is important; it may have a substantial impact on performance.
|
|
|
+ *
|
|
|
+ * If the firmware is compiled with static addresses, then @da should specify
|
|
|
+ * the expected device address of this memory region. If @da is set to
|
|
|
+ * FW_RSC_ADDR_ANY, then the host will dynamically allocate it, and then
|
|
|
+ * overwrite @da with the dynamically allocated address.
|
|
|
+ *
|
|
|
+ * We will always use @da to negotiate the device addresses, even if it
|
|
|
+ * isn't using an iommu. In that case, though, it will obviously contain
|
|
|
+ * physical addresses.
|
|
|
+ *
|
|
|
+ * Some remote remote_procs needs to know the allocated physical address
|
|
|
+ * even if they do use an iommu. This is needed, e.g., if they control
|
|
|
+ * hardware accelerators which access the physical memory directly (this
|
|
|
+ * is the case with OMAP4 for instance). In that case, the host will
|
|
|
+ * overwrite @pa with the dynamically allocated physical address.
|
|
|
+ * Generally we don't want to expose physical addresses if we don't have to
|
|
|
+ * (remote remote_procs are generally _not_ trusted), so we might want to
|
|
|
+ * change this to happen _only_ when explicitly required by the hardware.
|
|
|
+ *
|
|
|
+ * @flags is used to provide IOMMU protection flags, and @name should
|
|
|
+ * (optionally) contain a human readable name of this carveout region
|
|
|
+ * (mainly for debugging purposes).
|
|
|
+ */
|
|
|
+OPENAMP_PACKED_BEGIN
|
|
|
+struct fw_rsc_carveout {
|
|
|
+ uint32_t type;
|
|
|
+ uint32_t da;
|
|
|
+ uint32_t pa;
|
|
|
+ uint32_t len;
|
|
|
+ uint32_t flags;
|
|
|
+ uint32_t reserved;
|
|
|
+ uint8_t name[32];
|
|
|
+} OPENAMP_PACKED_END;
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct fw_rsc_devmem - iommu mapping request
|
|
|
+ * @da: device address
|
|
|
+ * @pa: physical address
|
|
|
+ * @len: length (in bytes)
|
|
|
+ * @flags: iommu protection flags
|
|
|
+ * @reserved: reserved (must be zero)
|
|
|
+ * @name: human-readable name of the requested region to be mapped
|
|
|
+ *
|
|
|
+ * This resource entry requests the host to iommu map a physically contiguous
|
|
|
+ * memory region. This is needed in case the remote remote_proc requires
|
|
|
+ * access to certain memory-based peripherals; _never_ use it to access
|
|
|
+ * regular memory.
|
|
|
+ *
|
|
|
+ * This is obviously only needed if the remote remote_proc is accessing memory
|
|
|
+ * via an iommu.
|
|
|
+ *
|
|
|
+ * @da should specify the required device address, @pa should specify
|
|
|
+ * the physical address we want to map, @len should specify the size of
|
|
|
+ * the mapping and @flags is the IOMMU protection flags. As always, @name may
|
|
|
+ * (optionally) contain a human readable name of this mapping (mainly for
|
|
|
+ * debugging purposes).
|
|
|
+ *
|
|
|
+ * Note: at this point we just "trust" those devmem entries to contain valid
|
|
|
+ * physical addresses, but this isn't safe and will be changed: eventually we
|
|
|
+ * want remoteproc implementations to provide us ranges of physical addresses
|
|
|
+ * the firmware is allowed to request, and not allow firmwares to request
|
|
|
+ * access to physical addresses that are outside those ranges.
|
|
|
+ */
|
|
|
+OPENAMP_PACKED_BEGIN
|
|
|
+struct fw_rsc_devmem {
|
|
|
+ uint32_t type;
|
|
|
+ uint32_t da;
|
|
|
+ uint32_t pa;
|
|
|
+ uint32_t len;
|
|
|
+ uint32_t flags;
|
|
|
+ uint32_t reserved;
|
|
|
+ uint8_t name[32];
|
|
|
+} OPENAMP_PACKED_END;
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct fw_rsc_trace - trace buffer declaration
|
|
|
+ * @da: device address
|
|
|
+ * @len: length (in bytes)
|
|
|
+ * @reserved: reserved (must be zero)
|
|
|
+ * @name: human-readable name of the trace buffer
|
|
|
+ *
|
|
|
+ * This resource entry provides the host information about a trace buffer
|
|
|
+ * into which the remote remote_proc will write log messages.
|
|
|
+ *
|
|
|
+ * @da specifies the device address of the buffer, @len specifies
|
|
|
+ * its size, and @name may contain a human readable name of the trace buffer.
|
|
|
+ *
|
|
|
+ * After booting the remote remote_proc, the trace buffers are exposed to the
|
|
|
+ * user via debugfs entries (called trace0, trace1, etc..).
|
|
|
+ */
|
|
|
+OPENAMP_PACKED_BEGIN
|
|
|
+struct fw_rsc_trace {
|
|
|
+ uint32_t type;
|
|
|
+ uint32_t da;
|
|
|
+ uint32_t len;
|
|
|
+ uint32_t reserved;
|
|
|
+ uint8_t name[32];
|
|
|
+} OPENAMP_PACKED_END;
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct fw_rsc_vdev_vring - vring descriptor entry
|
|
|
+ * @da: device address
|
|
|
+ * @align: the alignment between the consumer and producer parts of the vring
|
|
|
+ * @num: num of buffers supported by this vring (must be power of two)
|
|
|
+ * @notifyid is a unique rproc-wide notify index for this vring. This notify
|
|
|
+ * index is used when kicking a remote remote_proc, to let it know that this
|
|
|
+ * vring is triggered.
|
|
|
+ * @reserved: reserved (must be zero)
|
|
|
+ *
|
|
|
+ * This descriptor is not a resource entry by itself; it is part of the
|
|
|
+ * vdev resource type (see below).
|
|
|
+ *
|
|
|
+ * Note that @da should either contain the device address where
|
|
|
+ * the remote remote_proc is expecting the vring, or indicate that
|
|
|
+ * dynamically allocation of the vring's device address is supported.
|
|
|
+ */
|
|
|
+OPENAMP_PACKED_BEGIN
|
|
|
+struct fw_rsc_vdev_vring {
|
|
|
+ uint32_t da;
|
|
|
+ uint32_t align;
|
|
|
+ uint32_t num;
|
|
|
+ uint32_t notifyid;
|
|
|
+ uint32_t reserved;
|
|
|
+} OPENAMP_PACKED_END;
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct fw_rsc_vdev - virtio device header
|
|
|
+ * @id: virtio device id (as in virtio_ids.h)
|
|
|
+ * @notifyid is a unique rproc-wide notify index for this vdev. This notify
|
|
|
+ * index is used when kicking a remote remote_proc, to let it know that the
|
|
|
+ * status/features of this vdev have changes.
|
|
|
+ * @dfeatures specifies the virtio device features supported by the firmware
|
|
|
+ * @gfeatures is a place holder used by the host to write back the
|
|
|
+ * negotiated features that are supported by both sides.
|
|
|
+ * @config_len is the size of the virtio config space of this vdev. The config
|
|
|
+ * space lies in the resource table immediate after this vdev header.
|
|
|
+ * @status is a place holder where the host will indicate its virtio progress.
|
|
|
+ * @num_of_vrings indicates how many vrings are described in this vdev header
|
|
|
+ * @reserved: reserved (must be zero)
|
|
|
+ * @vring is an array of @num_of_vrings entries of 'struct fw_rsc_vdev_vring'.
|
|
|
+ *
|
|
|
+ * This resource is a virtio device header: it provides information about
|
|
|
+ * the vdev, and is then used by the host and its peer remote remote_procs
|
|
|
+ * to negotiate and share certain virtio properties.
|
|
|
+ *
|
|
|
+ * By providing this resource entry, the firmware essentially asks remoteproc
|
|
|
+ * to statically allocate a vdev upon registration of the rproc (dynamic vdev
|
|
|
+ * allocation is not yet supported).
|
|
|
+ *
|
|
|
+ * Note: unlike virtualization systems, the term 'host' here means
|
|
|
+ * the Linux side which is running remoteproc to control the remote
|
|
|
+ * remote_procs. We use the name 'gfeatures' to comply with virtio's terms,
|
|
|
+ * though there isn't really any virtualized guest OS here: it's the host
|
|
|
+ * which is responsible for negotiating the final features.
|
|
|
+ * Yeah, it's a bit confusing.
|
|
|
+ *
|
|
|
+ * Note: immediately following this structure is the virtio config space for
|
|
|
+ * this vdev (which is specific to the vdev; for more info, read the virtio
|
|
|
+ * spec). the size of the config space is specified by @config_len.
|
|
|
+ */
|
|
|
+OPENAMP_PACKED_BEGIN
|
|
|
+struct fw_rsc_vdev {
|
|
|
+ uint32_t type;
|
|
|
+ uint32_t id;
|
|
|
+ uint32_t notifyid;
|
|
|
+ uint32_t dfeatures;
|
|
|
+ uint32_t gfeatures;
|
|
|
+ uint32_t config_len;
|
|
|
+ uint8_t status;
|
|
|
+ uint8_t num_of_vrings;
|
|
|
+ uint8_t reserved[2];
|
|
|
+ struct fw_rsc_vdev_vring vring[0];
|
|
|
+} OPENAMP_PACKED_END;
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct fw_rsc_vendor - remote processor vendor specific resource
|
|
|
+ * @len: length of the resource
|
|
|
+ *
|
|
|
+ * This resource entry tells the host the vendor specific resource
|
|
|
+ * required by the remote.
|
|
|
+ *
|
|
|
+ * These request entries should precede other shared resource entries
|
|
|
+ * such as vdevs, vrings.
|
|
|
+ */
|
|
|
+OPENAMP_PACKED_BEGIN
|
|
|
+struct fw_rsc_vendor {
|
|
|
+ uint32_t type;
|
|
|
+ uint32_t len;
|
|
|
+} OPENAMP_PACKED_END;
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct fw_rsc_rproc_mem - remote processor memory
|
|
|
+ * @da: device address
|
|
|
+ * @pa: physical address
|
|
|
+ * @len: length (in bytes)
|
|
|
+ * @reserved: reserved (must be zero)
|
|
|
+ *
|
|
|
+ * This resource entry tells the host to the remote processor
|
|
|
+ * memory that the host can be used as shared memory.
|
|
|
+ *
|
|
|
+ * These request entries should precede other shared resource entries
|
|
|
+ * such as vdevs, vrings.
|
|
|
+ */
|
|
|
+OPENAMP_PACKED_BEGIN
|
|
|
+struct fw_rsc_rproc_mem {
|
|
|
+ uint32_t type;
|
|
|
+ uint32_t da;
|
|
|
+ uint32_t pa;
|
|
|
+ uint32_t len;
|
|
|
+ uint32_t reserved;
|
|
|
+} OPENAMP_PACKED_END;
|
|
|
+
|
|
|
+/*
|
|
|
+ * struct fw_rsc_fw_chksum - firmware checksum
|
|
|
+ * @algo: algorithm to generate the cheksum
|
|
|
+ * @chksum: checksum of the firmware loadable sections.
|
|
|
+ *
|
|
|
+ * This resource entry provides checksum for the firmware loadable sections.
|
|
|
+ * It is used to check if the remote already runs with the expected firmware to
|
|
|
+ * decide if it needs to start the remote if the remote is already running.
|
|
|
+ */
|
|
|
+OPENAMP_PACKED_BEGIN
|
|
|
+struct fw_rsc_fw_chksum {
|
|
|
+ uint32_t type;
|
|
|
+ uint8_t algo[16];
|
|
|
+ uint8_t chksum[64];
|
|
|
+} OPENAMP_PACKED_END;
|
|
|
+
|
|
|
+struct loader_ops;
|
|
|
+struct image_store_ops;
|
|
|
+struct remoteproc_ops;
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct remoteproc_mem
|
|
|
+ *
|
|
|
+ * This structure presents the memory used by the remote processor
|
|
|
+ *
|
|
|
+ * @da: device memory
|
|
|
+ * @pa: physical memory
|
|
|
+ * @size: size of the memory
|
|
|
+ * @io: pointer to the I/O region
|
|
|
+ * @node: list node
|
|
|
+ */
|
|
|
+struct remoteproc_mem {
|
|
|
+ metal_phys_addr_t da;
|
|
|
+ metal_phys_addr_t pa;
|
|
|
+ size_t size;
|
|
|
+ char name[32];
|
|
|
+ struct metal_io_region *io;
|
|
|
+ struct metal_list node;
|
|
|
+};
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct remoteproc
|
|
|
+ *
|
|
|
+ * This structure is maintained by the remoteproc to represent the remote
|
|
|
+ * processor instance. This structure acts as a prime parameter to use
|
|
|
+ * the remoteproc APIs.
|
|
|
+ *
|
|
|
+ * @bootadd: boot address
|
|
|
+ * @loader: executable loader
|
|
|
+ * @lock: mutext lock
|
|
|
+ * @ops: remoteproc operations
|
|
|
+ * @rsc_table: pointer to resource table
|
|
|
+ * @rsc_len: length of resource table
|
|
|
+ * @rsc_io: metal I/O region of resource table
|
|
|
+ * @mems: remoteproc memories
|
|
|
+ * @vdevs: remoteproc virtio devices
|
|
|
+ * @bitmap: bitmap for notify IDs for remoteproc subdevices
|
|
|
+ * @state: remote processor state
|
|
|
+ * @priv: private data
|
|
|
+ */
|
|
|
+struct remoteproc {
|
|
|
+ metal_mutex_t lock;
|
|
|
+ void *rsc_table;
|
|
|
+ size_t rsc_len;
|
|
|
+ struct metal_io_region *rsc_io;
|
|
|
+ struct metal_list mems;
|
|
|
+ struct metal_list vdevs;
|
|
|
+ unsigned long bitmap;
|
|
|
+ struct remoteproc_ops *ops;
|
|
|
+ metal_phys_addr_t bootaddr;
|
|
|
+ struct loader_ops *loader;
|
|
|
+ unsigned int state;
|
|
|
+ void *priv;
|
|
|
+};
|
|
|
+
|
|
|
+/**
|
|
|
+ * struct remoteproc_ops
|
|
|
+ *
|
|
|
+ * remoteproc operations needs to be implemented by each remoteproc driver
|
|
|
+ *
|
|
|
+ * @init: initialize the remoteproc instance
|
|
|
+ * @remove: remove the remoteproc instance
|
|
|
+ * @mmap: memory mapped the mempory with physical address or destination
|
|
|
+ * address as input.
|
|
|
+ * @handle_rsc: handle the vendor specific resource
|
|
|
+ * @config: configure the remoteproc to make it ready to load and run
|
|
|
+ * executable
|
|
|
+ * @start: kick the remoteproc to run application
|
|
|
+ * @stop: stop the remoteproc from running application, the resource such as
|
|
|
+ * memory may not be off.
|
|
|
+ * @shutdown: shutdown the remoteproc and release its resources.
|
|
|
+ * @notify: notify the remote
|
|
|
+ */
|
|
|
+struct remoteproc_ops {
|
|
|
+ struct remoteproc *(*init)(struct remoteproc *rproc,
|
|
|
+ struct remoteproc_ops *ops, void *arg);
|
|
|
+ void (*remove)(struct remoteproc *rproc);
|
|
|
+ void *(*mmap)(struct remoteproc *rproc,
|
|
|
+ metal_phys_addr_t *pa, metal_phys_addr_t *da,
|
|
|
+ size_t size, unsigned int attribute,
|
|
|
+ struct metal_io_region **io);
|
|
|
+ int (*handle_rsc)(struct remoteproc *rproc, void *rsc, size_t len);
|
|
|
+ int (*config)(struct remoteproc *rproc, void *data);
|
|
|
+ int (*start)(struct remoteproc *rproc);
|
|
|
+ int (*stop)(struct remoteproc *rproc);
|
|
|
+ int (*shutdown)(struct remoteproc *rproc);
|
|
|
+ int (*notify)(struct remoteproc *rproc, uint32_t id);
|
|
|
+};
|
|
|
+
|
|
|
+/* Remoteproc error codes */
|
|
|
+#define RPROC_EBASE 0
|
|
|
+#define RPROC_ENOMEM (RPROC_EBASE + 1)
|
|
|
+#define RPROC_EINVAL (RPROC_EBASE + 2)
|
|
|
+#define RPROC_ENODEV (RPROC_EBASE + 3)
|
|
|
+#define RPROC_EAGAIN (RPROC_EBASE + 4)
|
|
|
+#define RPROC_ERR_RSC_TAB_TRUNC (RPROC_EBASE + 5)
|
|
|
+#define RPROC_ERR_RSC_TAB_VER (RPROC_EBASE + 6)
|
|
|
+#define RPROC_ERR_RSC_TAB_RSVD (RPROC_EBASE + 7)
|
|
|
+#define RPROC_ERR_RSC_TAB_VDEV_NRINGS (RPROC_EBASE + 9)
|
|
|
+#define RPROC_ERR_RSC_TAB_NP (RPROC_EBASE + 10)
|
|
|
+#define RPROC_ERR_RSC_TAB_NS (RPROC_EBASE + 11)
|
|
|
+#define RPROC_ERR_LOADER_STATE (RPROC_EBASE + 12)
|
|
|
+#define RPROC_EMAX (RPROC_EBASE + 16)
|
|
|
+#define RPROC_EPTR (void *)(-1)
|
|
|
+#define RPROC_EOF (void *)(-1)
|
|
|
+
|
|
|
+static inline long RPROC_PTR_ERR(const void *ptr)
|
|
|
+{
|
|
|
+ return (long)ptr;
|
|
|
+}
|
|
|
+
|
|
|
+static inline int RPROC_IS_ERR(const void *ptr)
|
|
|
+{
|
|
|
+ if ((unsigned long)ptr >= (unsigned long)(-RPROC_EMAX))
|
|
|
+ return 1;
|
|
|
+ else
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static inline void *RPROC_ERR_PTR(long error)
|
|
|
+{
|
|
|
+ return (void *)error;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * enum rproc_state - remote processor states
|
|
|
+ * @RPROC_OFFLINE: remote is offline
|
|
|
+ * @RPROC_READY: remote is ready to start
|
|
|
+ * @RPROC_RUNNING: remote is up and running
|
|
|
+ * @RPROC_SUSPENDED: remote is suspended
|
|
|
+ * @RPROC_ERROR: remote has error; need to recover
|
|
|
+ * @RPROC_STOPPED: remote is stopped
|
|
|
+ * @RPROC_LAST: just keep this one at the end
|
|
|
+ */
|
|
|
+enum remoteproc_state {
|
|
|
+ RPROC_OFFLINE = 0,
|
|
|
+ RPROC_CONFIGURED = 1,
|
|
|
+ RPROC_READY = 2,
|
|
|
+ RPROC_RUNNING = 3,
|
|
|
+ RPROC_SUSPENDED = 4,
|
|
|
+ RPROC_ERROR = 5,
|
|
|
+ RPROC_STOPPED = 6,
|
|
|
+ RPROC_LAST = 7,
|
|
|
+};
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_init
|
|
|
+ *
|
|
|
+ * Initializes remoteproc resource.
|
|
|
+ *
|
|
|
+ * @rproc - pointer to remoteproc instance
|
|
|
+ * @ops - pointer to remoteproc operations
|
|
|
+ * @priv - pointer to private data
|
|
|
+ *
|
|
|
+ * @returns created remoteproc pointer
|
|
|
+ */
|
|
|
+struct remoteproc *remoteproc_init(struct remoteproc *rproc,
|
|
|
+ struct remoteproc_ops *ops, void *priv);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_remove
|
|
|
+ *
|
|
|
+ * Remove remoteproc resource
|
|
|
+ *
|
|
|
+ * @rproc - pointer to remoteproc instance
|
|
|
+ *
|
|
|
+ * returns 0 for success, negative value for failure
|
|
|
+ */
|
|
|
+int remoteproc_remove(struct remoteproc *rproc);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_init_mem
|
|
|
+ *
|
|
|
+ * Initialize remoteproc memory
|
|
|
+ *
|
|
|
+ * @mem - pointer to remoteproc memory
|
|
|
+ * @char - memory name
|
|
|
+ * @pa - physcial address
|
|
|
+ * @da - device address
|
|
|
+ * @size - memory size
|
|
|
+ * @io - pointer to the I/O region
|
|
|
+ */
|
|
|
+static inline void
|
|
|
+remoteproc_init_mem(struct remoteproc_mem *mem, const char *name,
|
|
|
+ metal_phys_addr_t pa, metal_phys_addr_t da,
|
|
|
+ size_t size, struct metal_io_region *io)
|
|
|
+{
|
|
|
+ if (!mem)
|
|
|
+ return;
|
|
|
+ if (name)
|
|
|
+ strncpy(mem->name, name, sizeof(mem->name));
|
|
|
+ else
|
|
|
+ mem->name[0] = 0;
|
|
|
+ mem->pa = pa;
|
|
|
+ mem->da = da;
|
|
|
+ mem->io = io;
|
|
|
+ mem->size = size;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_add_mem
|
|
|
+ *
|
|
|
+ * Add remoteproc memory
|
|
|
+ *
|
|
|
+ * @rproc - pointer to remoteproc
|
|
|
+ * @mem - pointer to remoteproc memory
|
|
|
+ */
|
|
|
+static inline void
|
|
|
+remoteproc_add_mem(struct remoteproc *rproc, struct remoteproc_mem *mem)
|
|
|
+{
|
|
|
+ if (!rproc || !mem)
|
|
|
+ return;
|
|
|
+ metal_list_add_tail(&rproc->mems, &mem->node);
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_get_io_with_name
|
|
|
+ *
|
|
|
+ * get remoteproc memory I/O region with name
|
|
|
+ *
|
|
|
+ * @rproc - pointer to the remote processor
|
|
|
+ * @name - name of the shared memory
|
|
|
+ * @io - pointer to the pointer of the I/O region
|
|
|
+ *
|
|
|
+ * returns metal I/O region pointer, NULL for failure
|
|
|
+ */
|
|
|
+struct metal_io_region *
|
|
|
+remoteproc_get_io_with_name(struct remoteproc *rproc,
|
|
|
+ const char *name);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_get_io_with_pa
|
|
|
+ *
|
|
|
+ * get remoteproc memory I/O region with physical address
|
|
|
+ *
|
|
|
+ * @rproc - pointer to the remote processor
|
|
|
+ * @pa - physical address
|
|
|
+ *
|
|
|
+ * returns metal I/O region pointer, NULL for failure
|
|
|
+ */
|
|
|
+struct metal_io_region *
|
|
|
+remoteproc_get_io_with_pa(struct remoteproc *rproc,
|
|
|
+ metal_phys_addr_t pa);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_get_io_with_da
|
|
|
+ *
|
|
|
+ * get remoteproc memory I/O region with device address
|
|
|
+ *
|
|
|
+ * @rproc - pointer to the remote processor
|
|
|
+ * @da - device address
|
|
|
+ * @offset - I/O region offset of the device address
|
|
|
+ *
|
|
|
+ * returns metal I/O region pointer, NULL for failure
|
|
|
+ */
|
|
|
+struct metal_io_region *
|
|
|
+remoteproc_get_io_with_da(struct remoteproc *rproc,
|
|
|
+ metal_phys_addr_t da,
|
|
|
+ unsigned long *offset);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_get_io_with_va
|
|
|
+ *
|
|
|
+ * get remoteproc memory I/O region with virtual address
|
|
|
+ *
|
|
|
+ * @rproc - pointer to the remote processor
|
|
|
+ * @va - virtual address
|
|
|
+ *
|
|
|
+ * returns metal I/O region pointer, NULL for failure
|
|
|
+ */
|
|
|
+struct metal_io_region *
|
|
|
+remoteproc_get_io_with_va(struct remoteproc *rproc,
|
|
|
+ void *va);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_mmap
|
|
|
+ *
|
|
|
+ * remoteproc mmap memory
|
|
|
+ *
|
|
|
+ * @rproc - pointer to the remote processor
|
|
|
+ * @pa - physical address pointer
|
|
|
+ * @da - device address pointer
|
|
|
+ * @size - size of the memory
|
|
|
+ * @attribute - memory attribute
|
|
|
+ * @io - pointer to the I/O region
|
|
|
+ *
|
|
|
+ * returns pointer to the memory
|
|
|
+ */
|
|
|
+void *remoteproc_mmap(struct remoteproc *rproc,
|
|
|
+ metal_phys_addr_t *pa, metal_phys_addr_t *da,
|
|
|
+ size_t size, unsigned int attribute,
|
|
|
+ struct metal_io_region **io);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_parse_rsc_table
|
|
|
+ *
|
|
|
+ * Parse resource table of remoteproc
|
|
|
+ *
|
|
|
+ * @rproc - pointer to remoteproc instance
|
|
|
+ * @rsc_table - pointer to resource table
|
|
|
+ * @rsc_size - resource table size
|
|
|
+ *
|
|
|
+ * returns 0 for success and negative value for errors
|
|
|
+ */
|
|
|
+int remoteproc_parse_rsc_table(struct remoteproc *rproc,
|
|
|
+ struct resource_table *rsc_table,
|
|
|
+ size_t rsc_size);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_set_rsc_table
|
|
|
+ *
|
|
|
+ * Parse and set resource table of remoteproc
|
|
|
+ *
|
|
|
+ * @rproc - pointer to remoteproc instance
|
|
|
+ * @rsc_table - pointer to resource table
|
|
|
+ * @rsc_size - resource table size
|
|
|
+ *
|
|
|
+ * returns 0 for success and negative value for errors
|
|
|
+ */
|
|
|
+int remoteproc_set_rsc_table(struct remoteproc *rproc,
|
|
|
+ struct resource_table *rsc_table,
|
|
|
+ size_t rsc_size);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_config
|
|
|
+ *
|
|
|
+ * This function configures the remote processor to get it
|
|
|
+ * ready to load and run executable.
|
|
|
+ *
|
|
|
+ * @rproc - pointer to remoteproc instance to start
|
|
|
+ * @data - configuration data
|
|
|
+ *
|
|
|
+ * returns 0 for success and negative value for errors
|
|
|
+ */
|
|
|
+int remoteproc_config(struct remoteproc *rproc, void *data);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_start
|
|
|
+ *
|
|
|
+ * This function starts the remote processor.
|
|
|
+ * It assumes the firmware is already loaded,
|
|
|
+ *
|
|
|
+ * @rproc - pointer to remoteproc instance to start
|
|
|
+ *
|
|
|
+ * returns 0 for success and negative value for errors
|
|
|
+ */
|
|
|
+int remoteproc_start(struct remoteproc *rproc);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_stop
|
|
|
+ *
|
|
|
+ * This function stops the remote processor but it
|
|
|
+ * will not release its resource.
|
|
|
+ *
|
|
|
+ * @rproc - pointer to remoteproc instance
|
|
|
+ *
|
|
|
+ * returns 0 for success and negative value for errors
|
|
|
+ */
|
|
|
+int remoteproc_stop(struct remoteproc *rproc);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_shutdown
|
|
|
+ *
|
|
|
+ * This function shutdown the remote processor and
|
|
|
+ * release its resources.
|
|
|
+ *
|
|
|
+ * @rproc - pointer to remoteproc instance
|
|
|
+ *
|
|
|
+ * returns 0 for success and negative value for errors
|
|
|
+ */
|
|
|
+int remoteproc_shutdown(struct remoteproc *rproc);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_load
|
|
|
+ *
|
|
|
+ * load executable, it expects the user application defines how to
|
|
|
+ * open the executable file and how to get data from the executable file
|
|
|
+ * and how to load data to the target memory.
|
|
|
+ *
|
|
|
+ * @rproc: pointer to the remoteproc instance
|
|
|
+ * @path: optional path to the image file
|
|
|
+ * @store: pointer to user defined image store argument
|
|
|
+ * @store_ops: pointer to image store operations
|
|
|
+ * @image_info: pointer to memory which stores image information used
|
|
|
+ * by remoteproc loader
|
|
|
+ *
|
|
|
+ * return 0 for success and negative value for failure
|
|
|
+ */
|
|
|
+int remoteproc_load(struct remoteproc *rproc, const char *path,
|
|
|
+ void *store, struct image_store_ops *store_ops,
|
|
|
+ void **img_info);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_load_noblock
|
|
|
+ *
|
|
|
+ * load executable, it expects the caller has loaded image data to local
|
|
|
+ * memory and passed to the this function. If the function needs more
|
|
|
+ * image data it will return the next expected image data offset and
|
|
|
+ * the next expected image data length. If the function requires the
|
|
|
+ * caller to download image data to the target memory, it will also
|
|
|
+ * return the target physical address besides the offset and length.
|
|
|
+ * This function can be used to load firmware in stream mode. In this
|
|
|
+ * mode, you cannot do seek to the executable file. If the executable
|
|
|
+ * is ELF, it cannot get the resource table section before it loads
|
|
|
+ * the full ELF file. Furthermore, application usually don't store
|
|
|
+ * the data which is loaded to local memory in streaming mode, and
|
|
|
+ * thus, in this mode, it will load the binrary to the target memory
|
|
|
+ * before it gets the resource table. And thus, when calling this funciton
|
|
|
+ * don't put the target exectuable memory in the resource table, as
|
|
|
+ * this function will parse the resource table after it loads the binary
|
|
|
+ * to target memory.
|
|
|
+ *
|
|
|
+ * @rproc: pointer to the remoteproc instance
|
|
|
+ * @img_data: pointer to image data for remoteproc loader to parse
|
|
|
+ * @offset: image data offset to the beginning of the image file
|
|
|
+ * @len: image data length
|
|
|
+ * @image_info: pointer to memory which stores image information used
|
|
|
+ * by remoteproc loader
|
|
|
+ * @pa: pointer to the target memory physical address. If the next expected
|
|
|
+ * data doesn't need to load to the target memory, the function will
|
|
|
+ * set it to ANY.
|
|
|
+ * @io: pointer to the target memory physical address. If the next expected
|
|
|
+ * data doesn't need to load to the target memory, the function will
|
|
|
+ * set it to ANY.
|
|
|
+ * @noffset: pointer to the next image data offset to the beginning of
|
|
|
+ * the image file needs to load to local or to the target
|
|
|
+ * memory.
|
|
|
+ * @nlen: pointer to the next image data length needs to load to local
|
|
|
+ * or to the target memory.
|
|
|
+ * @nmlen: pointer to the memory size. It is only used when the next
|
|
|
+ * expected data is going to be loaded to the target memory. E.g.
|
|
|
+ * in ELF, it is possible that loadable segment in memory is
|
|
|
+ * larger that the segment data in the ELF file. In this case,
|
|
|
+ * application will need to pad the rest of the memory with
|
|
|
+ * padding.
|
|
|
+ * @padding: pointer to the padding value. It is only used when the next
|
|
|
+ * expected data is going to be loaded to the target memory.
|
|
|
+ * and the target memory size is larger than the segment data in
|
|
|
+ * the executable file.
|
|
|
+ *
|
|
|
+ * return 0 for success and negative value for failure
|
|
|
+ */
|
|
|
+int remoteproc_load_noblock(struct remoteproc *rproc,
|
|
|
+ const void *img_data, size_t offset, size_t len,
|
|
|
+ void **img_info,
|
|
|
+ metal_phys_addr_t *pa, struct metal_io_region **io,
|
|
|
+ size_t *noffset, size_t *nlen,
|
|
|
+ size_t *nmlen, unsigned char *padding);
|
|
|
+
|
|
|
+/**
|
|
|
+ * remoteproc_allocate_id
|
|
|
+ *
|
|
|
+ * allocate notifyid for resource
|
|
|
+ *
|
|
|
+ * @rproc - pointer to the remoteproc instance
|
|
|
+ * @start - start of the id range
|
|
|
+ * @end - end of the id range
|
|
|
+ *
|
|
|
+ * return allocated notify id
|
|
|
+ */
|
|
|
+unsigned int remoteproc_allocate_id(struct remoteproc *rproc,
|
|
|
+ unsigned int start,
|
|
|
+ unsigned int end);
|
|
|
+
|
|
|
+/* remoteproc_create_virtio
|
|
|
+ *
|
|
|
+ * create virtio device, it returns pointer to the created virtio device.
|
|
|
+ *
|
|
|
+ * @rproc: pointer to the remoteproc instance
|
|
|
+ * @vdev_id: virtio device ID
|
|
|
+ * @role: virtio device role
|
|
|
+ * @rst_cb: virtio device reset callback
|
|
|
+ *
|
|
|
+ * return pointer to the created virtio device, NULL for failure.
|
|
|
+ */
|
|
|
+struct virtio_device *
|
|
|
+remoteproc_create_virtio(struct remoteproc *rproc,
|
|
|
+ int vdev_id, unsigned int role,
|
|
|
+ void (*rst_cb)(struct virtio_device *vdev));
|
|
|
+
|
|
|
+/* remoteproc_remove_virtio
|
|
|
+ *
|
|
|
+ * Remove virtio device
|
|
|
+ *
|
|
|
+ * @rproc: pointer to the remoteproc instance
|
|
|
+ * @vdev: pointer to the virtio device
|
|
|
+ *
|
|
|
+ */
|
|
|
+void remoteproc_remove_virtio(struct remoteproc *rproc,
|
|
|
+ struct virtio_device *vdev);
|
|
|
+
|
|
|
+/* remoteproc_get_notification
|
|
|
+ *
|
|
|
+ * remoteproc is got notified, it will check its subdevices
|
|
|
+ * for the notification
|
|
|
+ *
|
|
|
+ * @rproc - pointer to the remoteproc instance
|
|
|
+ * @notifyid - notificatin id
|
|
|
+ *
|
|
|
+ * return 0 for succeed, negative value for failure
|
|
|
+ */
|
|
|
+int remoteproc_get_notification(struct remoteproc *rproc,
|
|
|
+ uint32_t notifyid);
|
|
|
+#if defined __cplusplus
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+#endif /* REMOTEPROC_H_ */
|