Merge tag 'remoteproc-4.2' of git://git.kernel.org/pub/scm/linux/kernel/git/ohad...
authorLinus Torvalds <torvalds@linux-foundation.org>
Fri, 3 Jul 2015 21:57:50 +0000 (14:57 -0700)
committerLinus Torvalds <torvalds@linux-foundation.org>
Fri, 3 Jul 2015 21:57:50 +0000 (14:57 -0700)
Pull remoteproc updates from Ohad Ben-Cohen:

 - remoteproc fixes/cleanups from Suman Anna

 - new remoteproc TI Wakeup M3 driver from Dave Gerlach

 - remoteproc core support for TI's Wakeup M3 driver from both Dave and Suman

 - tiny remoteproc build fix from myself

* tag 'remoteproc-4.2' of git://git.kernel.org/pub/scm/linux/kernel/git/ohad/remoteproc:
  remoteproc: fix !CONFIG_OF build breakage
  remoteproc/wkup_m3: add a remoteproc driver for TI Wakeup M3
  Documentation: dt: add bindings for TI Wakeup M3 processor
  remoteproc: add a rproc ops for performing address translation
  remoteproc: introduce rproc_get_by_phandle API
  remoteproc: fix various checkpatch warnings
  remoteproc/davinci: fix quoted split string checkpatch warning
  remoteproc/ste: add blank lines after declarations

Documentation/devicetree/bindings/remoteproc/wkup_m3_rproc.txt [new file with mode: 0644]
Documentation/remoteproc.txt
drivers/remoteproc/Kconfig
drivers/remoteproc/Makefile
drivers/remoteproc/da8xx_remoteproc.c
drivers/remoteproc/remoteproc_core.c
drivers/remoteproc/remoteproc_internal.h
drivers/remoteproc/ste_modem_rproc.c
drivers/remoteproc/wkup_m3_rproc.c [new file with mode: 0644]
include/linux/platform_data/wkup_m3.h [new file with mode: 0644]
include/linux/remoteproc.h

diff --git a/Documentation/devicetree/bindings/remoteproc/wkup_m3_rproc.txt b/Documentation/devicetree/bindings/remoteproc/wkup_m3_rproc.txt
new file mode 100644 (file)
index 0000000..3a70073
--- /dev/null
@@ -0,0 +1,52 @@
+TI Wakeup M3 Remoteproc Driver
+==============================
+
+The TI AM33xx and AM43xx family of devices use a small Cortex M3 co-processor
+(commonly referred to as Wakeup M3 or CM3) to help with various low power tasks
+that cannot be controlled from the MPU. This CM3 processor requires a firmware
+binary to accomplish this. The wkup_m3 remoteproc driver handles the loading of
+the firmware and booting of the CM3.
+
+Wkup M3 Device Node:
+====================
+A wkup_m3 device node is used to represent the Wakeup M3 processor instance
+within the SoC. It is added as a child node of the parent interconnect bus
+(l4_wkup) through which it is accessible to the MPU.
+
+Required properties:
+--------------------
+- compatible:          Should be one of,
+                               "ti,am3352-wkup-m3" for AM33xx SoCs
+                               "ti,am4372-wkup-m3" for AM43xx SoCs
+- reg:                 Should contain the address ranges for the two internal
+                       memory regions, UMEM and DMEM. The parent node should
+                       provide an appropriate ranges property for properly
+                       translating these into bus addresses.
+- reg-names:           Contains the corresponding names for the two memory
+                       regions. These should be named "umem" & "dmem".
+- ti,hwmods:           Name of the hwmod associated with the wkupm3 device.
+- ti,pm-firmware:      Name of firmware file to be used for loading and
+                       booting the wkup_m3 remote processor.
+
+Example:
+--------
+/* AM33xx */
+ocp {
+        l4_wkup: l4_wkup@44c00000 {
+               compatible = "am335-l4-wkup", "simple-bus";
+               ranges = <0 0x44c00000 0x400000>;
+               #address-cells = <1>;
+               #size-cells = <1>;
+
+               wkup_m3: wkup_m3@100000 {
+                       compatible = "ti,am3352-wkup-m3";
+                       reg = <0x100000 0x4000>,
+                             <0x180000 0x2000>;
+                       reg-names = "umem", "dmem";
+                       ti,hwmods = "wkup_m3";
+                       ti,pm-firmware = "am335x-pm-firmware.elf";
+               };
+       };
+
+       ...
+};
index e6469fdcf89a0344b017f90c821019750d8e9039..ef0219fa4bb4cf5beb9078293a92b3ccbcbe0d48 100644 (file)
@@ -51,6 +51,12 @@ cost.
         rproc_shutdown() returns, and users can still use it with a subsequent
         rproc_boot(), if needed.
 
+  struct rproc *rproc_get_by_phandle(phandle phandle)
+    - Find an rproc handle using a device tree phandle. Returns the rproc
+      handle on success, and NULL on failure. This function increments
+      the remote processor's refcount, so always use rproc_put() to
+      decrement it back once rproc isn't needed anymore.
+
 3. Typical usage
 
 #include <linux/remoteproc.h>
index 5e343bab9458e430f18f54c1be713a0a7686b593..28c711f0ac6bd5c3d77c32018c64abad77c4bdfa 100644 (file)
@@ -41,6 +41,19 @@ config STE_MODEM_RPROC
          This can be either built-in or a loadable module.
          If unsure say N.
 
+config WKUP_M3_RPROC
+       tristate "AMx3xx Wakeup M3 remoteproc support"
+       depends on SOC_AM33XX || SOC_AM43XX
+       select REMOTEPROC
+       help
+         Say y here to support Wakeup M3 remote processor on TI AM33xx
+         and AM43xx family of SoCs.
+
+         Required for Suspend-to-RAM on AM33xx and AM43xx SoCs. Also needed
+         for deep CPUIdle states on AM33xx SoCs. Allows for loading of the
+         firmware onto these remote processors.
+         If unsure say N.
+
 config DA8XX_REMOTEPROC
        tristate "DA8xx/OMAP-L13x remoteproc support"
        depends on ARCH_DAVINCI_DA8XX
index ac2ff75686d20da708d77728aca94f23cbdf1764..81b04d1e2e5888b8ebdaca78a551045e3aa0db40 100644 (file)
@@ -9,4 +9,5 @@ remoteproc-y                            += remoteproc_virtio.o
 remoteproc-y                           += remoteproc_elf_loader.o
 obj-$(CONFIG_OMAP_REMOTEPROC)          += omap_remoteproc.o
 obj-$(CONFIG_STE_MODEM_RPROC)          += ste_modem_rproc.o
+obj-$(CONFIG_WKUP_M3_RPROC)            += wkup_m3_rproc.o
 obj-$(CONFIG_DA8XX_REMOTEPROC)         += da8xx_remoteproc.o
index f8d6a0661c14cd51e9a8bbc1bf6fe0bee182b484..009e56f67de239779e3710bffc1b430257441ea3 100644 (file)
@@ -26,8 +26,7 @@
 static char *da8xx_fw_name;
 module_param(da8xx_fw_name, charp, S_IRUGO);
 MODULE_PARM_DESC(da8xx_fw_name,
-                "\n\t\tName of DSP firmware file in /lib/firmware"
-                " (if not specified defaults to 'rproc-dsp-fw')");
+                "Name of DSP firmware file in /lib/firmware (if not specified defaults to 'rproc-dsp-fw')");
 
 /*
  * OMAP-L138 Technical References:
index 11cdb119e4f3bbdee5d73293dd71cec450f2ae65..8b3130f22b42b334ff0d22718e46484012fb2c2b 100644 (file)
@@ -44,6 +44,9 @@
 
 #include "remoteproc_internal.h"
 
+static DEFINE_MUTEX(rproc_list_mutex);
+static LIST_HEAD(rproc_list);
+
 typedef int (*rproc_handle_resources_t)(struct rproc *rproc,
                                struct resource_table *table, int len);
 typedef int (*rproc_handle_resource_t)(struct rproc *rproc,
@@ -132,32 +135,48 @@ static void rproc_disable_iommu(struct rproc *rproc)
 
        iommu_detach_device(domain, dev);
        iommu_domain_free(domain);
-
-       return;
 }
 
-/*
+/**
+ * rproc_da_to_va() - lookup the kernel virtual address for a remoteproc address
+ * @rproc: handle of a remote processor
+ * @da: remoteproc device address to translate
+ * @len: length of the memory region @da is pointing to
+ *
  * Some remote processors will ask us to allocate them physically contiguous
  * memory regions (which we call "carveouts"), and map them to specific
- * device addresses (which are hardcoded in the firmware).
+ * device addresses (which are hardcoded in the firmware). They may also have
+ * dedicated memory regions internal to the processors, and use them either
+ * exclusively or alongside carveouts.
  *
  * They may then ask us to copy objects into specific device addresses (e.g.
  * code/data sections) or expose us certain symbols in other device address
  * (e.g. their trace buffer).
  *
- * This function is an internal helper with which we can go over the allocated
- * carveouts and translate specific device address to kernel virtual addresses
- * so we can access the referenced memory.
+ * This function is a helper function with which we can go over the allocated
+ * carveouts and translate specific device addresses to kernel virtual addresses
+ * so we can access the referenced memory. This function also allows to perform
+ * translations on the internal remoteproc memory regions through a platform
+ * implementation specific da_to_va ops, if present.
+ *
+ * The function returns a valid kernel address on success or NULL on failure.
  *
  * Note: phys_to_virt(iommu_iova_to_phys(rproc->domain, da)) will work too,
  * but only on kernel direct mapped RAM memory. Instead, we're just using
- * here the output of the DMA API, which should be more correct.
+ * here the output of the DMA API for the carveouts, which should be more
+ * correct.
  */
 void *rproc_da_to_va(struct rproc *rproc, u64 da, int len)
 {
        struct rproc_mem_entry *carveout;
        void *ptr = NULL;
 
+       if (rproc->ops->da_to_va) {
+               ptr = rproc->ops->da_to_va(rproc, da, len);
+               if (ptr)
+                       goto out;
+       }
+
        list_for_each_entry(carveout, &rproc->carveouts, node) {
                int offset = da - carveout->da;
 
@@ -174,6 +193,7 @@ void *rproc_da_to_va(struct rproc *rproc, u64 da, int len)
                break;
        }
 
+out:
        return ptr;
 }
 EXPORT_SYMBOL(rproc_da_to_va);
@@ -411,10 +431,8 @@ static int rproc_handle_trace(struct rproc *rproc, struct fw_rsc_trace *rsc,
        }
 
        trace = kzalloc(sizeof(*trace), GFP_KERNEL);
-       if (!trace) {
-               dev_err(dev, "kzalloc trace failed\n");
+       if (!trace)
                return -ENOMEM;
-       }
 
        /* set the trace buffer dma properties */
        trace->len = rsc->len;
@@ -489,10 +507,8 @@ static int rproc_handle_devmem(struct rproc *rproc, struct fw_rsc_devmem *rsc,
        }
 
        mapping = kzalloc(sizeof(*mapping), GFP_KERNEL);
-       if (!mapping) {
-               dev_err(dev, "kzalloc mapping failed\n");
+       if (!mapping)
                return -ENOMEM;
-       }
 
        ret = iommu_map(rproc->domain, rsc->da, rsc->pa, rsc->len, rsc->flags);
        if (ret) {
@@ -565,10 +581,8 @@ static int rproc_handle_carveout(struct rproc *rproc,
                        rsc->da, rsc->pa, rsc->len, rsc->flags);
 
        carveout = kzalloc(sizeof(*carveout), GFP_KERNEL);
-       if (!carveout) {
-               dev_err(dev, "kzalloc carveout failed\n");
+       if (!carveout)
                return -ENOMEM;
-       }
 
        va = dma_alloc_coherent(dev->parent, rsc->len, &dma, GFP_KERNEL);
        if (!va) {
@@ -768,7 +782,8 @@ static void rproc_resource_cleanup(struct rproc *rproc)
 
        /* clean up carveout allocations */
        list_for_each_entry_safe(entry, tmp, &rproc->carveouts, node) {
-               dma_free_coherent(dev->parent, entry->len, entry->va, entry->dma);
+               dma_free_coherent(dev->parent, entry->len, entry->va,
+                                 entry->dma);
                list_del(&entry->node);
                kfree(entry);
        }
@@ -808,9 +823,8 @@ static int rproc_fw_boot(struct rproc *rproc, const struct firmware *fw)
 
        /* look for the resource table */
        table = rproc_find_rsc_table(rproc, fw, &tablesz);
-       if (!table) {
+       if (!table)
                goto clean_up;
-       }
 
        /* Verify that resource table in loaded fw is unchanged */
        if (rproc->table_csum != crc32(0, table, tablesz)) {
@@ -911,7 +925,8 @@ static void rproc_fw_config_virtio(const struct firmware *fw, void *context)
 
        /* count the number of notify-ids */
        rproc->max_notifyid = -1;
-       ret = rproc_handle_resources(rproc, tablesz, rproc_count_vrings_handler);
+       ret = rproc_handle_resources(rproc, tablesz,
+                                    rproc_count_vrings_handler);
        if (ret)
                goto out;
 
@@ -1151,6 +1166,50 @@ out:
 }
 EXPORT_SYMBOL(rproc_shutdown);
 
+/**
+ * rproc_get_by_phandle() - find a remote processor by phandle
+ * @phandle: phandle to the rproc
+ *
+ * Finds an rproc handle using the remote processor's phandle, and then
+ * return a handle to the rproc.
+ *
+ * This function increments the remote processor's refcount, so always
+ * use rproc_put() to decrement it back once rproc isn't needed anymore.
+ *
+ * Returns the rproc handle on success, and NULL on failure.
+ */
+#ifdef CONFIG_OF
+struct rproc *rproc_get_by_phandle(phandle phandle)
+{
+       struct rproc *rproc = NULL, *r;
+       struct device_node *np;
+
+       np = of_find_node_by_phandle(phandle);
+       if (!np)
+               return NULL;
+
+       mutex_lock(&rproc_list_mutex);
+       list_for_each_entry(r, &rproc_list, node) {
+               if (r->dev.parent && r->dev.parent->of_node == np) {
+                       rproc = r;
+                       get_device(&rproc->dev);
+                       break;
+               }
+       }
+       mutex_unlock(&rproc_list_mutex);
+
+       of_node_put(np);
+
+       return rproc;
+}
+#else
+struct rproc *rproc_get_by_phandle(phandle phandle)
+{
+       return NULL;
+}
+#endif
+EXPORT_SYMBOL(rproc_get_by_phandle);
+
 /**
  * rproc_add() - register a remote processor
  * @rproc: the remote processor handle to register
@@ -1180,6 +1239,11 @@ int rproc_add(struct rproc *rproc)
        if (ret < 0)
                return ret;
 
+       /* expose to rproc_get_by_phandle users */
+       mutex_lock(&rproc_list_mutex);
+       list_add(&rproc->node, &rproc_list);
+       mutex_unlock(&rproc_list_mutex);
+
        dev_info(dev, "%s is available\n", rproc->name);
 
        dev_info(dev, "Note: remoteproc is still under development and considered experimental.\n");
@@ -1268,10 +1332,8 @@ struct rproc *rproc_alloc(struct device *dev, const char *name,
                name_len = strlen(name) + strlen(template) - 2 + 1;
 
        rproc = kzalloc(sizeof(struct rproc) + len + name_len, GFP_KERNEL);
-       if (!rproc) {
-               dev_err(dev, "%s: kzalloc failed\n", __func__);
+       if (!rproc)
                return NULL;
-       }
 
        if (!firmware) {
                p = (char *)rproc + sizeof(struct rproc) + len;
@@ -1369,6 +1431,11 @@ int rproc_del(struct rproc *rproc)
        /* Free the copy of the resource table */
        kfree(rproc->cached_table);
 
+       /* the rproc is downref'ed as soon as it's removed from the klist */
+       mutex_lock(&rproc_list_mutex);
+       list_del(&rproc->node);
+       mutex_unlock(&rproc_list_mutex);
+
        device_del(&rproc->dev);
 
        return 0;
index 70701a50ddfa6276105f383b33c81d88e885db33..8041b95cb05863c80ec6717bea03fab29c0ec27a 100644 (file)
@@ -35,7 +35,7 @@ struct rproc;
  * @get_boot_addr:     get boot address to entry point specified in firmware
  */
 struct rproc_fw_ops {
-       struct resource_table *(*find_rsc_table) (struct rproc *rproc,
+       struct resource_table *(*find_rsc_table)(struct rproc *rproc,
                                                const struct firmware *fw,
                                                int *tablesz);
        struct resource_table *(*find_loaded_rsc_table)(struct rproc *rproc,
index dd193f35a1ff22d3c36406edc44b3e106ea8d6e3..53dc17bdd54e0c3ef94f03a345e2d31e5a5bdfe6 100644 (file)
@@ -67,8 +67,7 @@ static int sproc_load_segments(struct rproc *rproc, const struct firmware *fw)
 static const struct ste_toc_entry *sproc_find_rsc_entry(const void *data)
 {
        int i;
-       const struct ste_toc *toc;
-       toc = data;
+       const struct ste_toc *toc = data;
 
        /* Search the table for the resource table */
        for (i = 0; i < SPROC_MAX_TOC_ENTRIES &&
@@ -230,6 +229,7 @@ static int sproc_start(struct rproc *rproc)
 static int sproc_stop(struct rproc *rproc)
 {
        struct sproc *sproc = rproc->priv;
+
        sproc_dbg(sproc, "stop ste-modem\n");
 
        return sproc->mdev->ops.power(sproc->mdev, false);
diff --git a/drivers/remoteproc/wkup_m3_rproc.c b/drivers/remoteproc/wkup_m3_rproc.c
new file mode 100644 (file)
index 0000000..edf8181
--- /dev/null
@@ -0,0 +1,257 @@
+/*
+ * TI AMx3 Wakeup M3 Remote Processor driver
+ *
+ * Copyright (C) 2014-2015 Texas Instruments, Inc.
+ *
+ * Dave Gerlach <d-gerlach@ti.com>
+ * Suman Anna <s-anna@ti.com>
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * version 2 as published by the Free Software Foundation.
+ *
+ * This program 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.
+ */
+
+#include <linux/err.h>
+#include <linux/interrupt.h>
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/of_device.h>
+#include <linux/of_address.h>
+#include <linux/platform_device.h>
+#include <linux/pm_runtime.h>
+#include <linux/remoteproc.h>
+
+#include <linux/platform_data/wkup_m3.h>
+
+#include "remoteproc_internal.h"
+
+#define WKUPM3_MEM_MAX 2
+
+/**
+ * struct wkup_m3_mem - WkupM3 internal memory structure
+ * @cpu_addr: MPU virtual address of the memory region
+ * @bus_addr: Bus address used to access the memory region
+ * @dev_addr: Device address from Wakeup M3 view
+ * @size: Size of the memory region
+ */
+struct wkup_m3_mem {
+       void __iomem *cpu_addr;
+       phys_addr_t bus_addr;
+       u32 dev_addr;
+       size_t size;
+};
+
+/**
+ * struct wkup_m3_rproc - WkupM3 remote processor state
+ * @rproc: rproc handle
+ * @pdev: pointer to platform device
+ * @mem: WkupM3 memory information
+ */
+struct wkup_m3_rproc {
+       struct rproc *rproc;
+       struct platform_device *pdev;
+       struct wkup_m3_mem mem[WKUPM3_MEM_MAX];
+};
+
+static int wkup_m3_rproc_start(struct rproc *rproc)
+{
+       struct wkup_m3_rproc *wkupm3 = rproc->priv;
+       struct platform_device *pdev = wkupm3->pdev;
+       struct device *dev = &pdev->dev;
+       struct wkup_m3_platform_data *pdata = dev_get_platdata(dev);
+
+       if (pdata->deassert_reset(pdev, pdata->reset_name)) {
+               dev_err(dev, "Unable to reset wkup_m3!\n");
+               return -ENODEV;
+       }
+
+       return 0;
+}
+
+static int wkup_m3_rproc_stop(struct rproc *rproc)
+{
+       struct wkup_m3_rproc *wkupm3 = rproc->priv;
+       struct platform_device *pdev = wkupm3->pdev;
+       struct device *dev = &pdev->dev;
+       struct wkup_m3_platform_data *pdata = dev_get_platdata(dev);
+
+       if (pdata->assert_reset(pdev, pdata->reset_name)) {
+               dev_err(dev, "Unable to assert reset of wkup_m3!\n");
+               return -ENODEV;
+       }
+
+       return 0;
+}
+
+static void *wkup_m3_rproc_da_to_va(struct rproc *rproc, u64 da, int len)
+{
+       struct wkup_m3_rproc *wkupm3 = rproc->priv;
+       void *va = NULL;
+       int i;
+       u32 offset;
+
+       if (len <= 0)
+               return NULL;
+
+       for (i = 0; i < WKUPM3_MEM_MAX; i++) {
+               if (da >= wkupm3->mem[i].dev_addr && da + len <=
+                   wkupm3->mem[i].dev_addr +  wkupm3->mem[i].size) {
+                       offset = da -  wkupm3->mem[i].dev_addr;
+                       /* __force to make sparse happy with type conversion */
+                       va = (__force void *)(wkupm3->mem[i].cpu_addr + offset);
+                       break;
+               }
+       }
+
+       return va;
+}
+
+static struct rproc_ops wkup_m3_rproc_ops = {
+       .start          = wkup_m3_rproc_start,
+       .stop           = wkup_m3_rproc_stop,
+       .da_to_va       = wkup_m3_rproc_da_to_va,
+};
+
+static const struct of_device_id wkup_m3_rproc_of_match[] = {
+       { .compatible = "ti,am3352-wkup-m3", },
+       { .compatible = "ti,am4372-wkup-m3", },
+       {},
+};
+
+static int wkup_m3_rproc_probe(struct platform_device *pdev)
+{
+       struct device *dev = &pdev->dev;
+       struct wkup_m3_platform_data *pdata = dev->platform_data;
+       /* umem always needs to be processed first */
+       const char *mem_names[WKUPM3_MEM_MAX] = { "umem", "dmem" };
+       struct wkup_m3_rproc *wkupm3;
+       const char *fw_name;
+       struct rproc *rproc;
+       struct resource *res;
+       const __be32 *addrp;
+       u32 l4_offset = 0;
+       u64 size;
+       int ret;
+       int i;
+
+       if (!(pdata && pdata->deassert_reset && pdata->assert_reset &&
+             pdata->reset_name)) {
+               dev_err(dev, "Platform data missing!\n");
+               return -ENODEV;
+       }
+
+       ret = of_property_read_string(dev->of_node, "ti,pm-firmware",
+                                     &fw_name);
+       if (ret) {
+               dev_err(dev, "No firmware filename given\n");
+               return -ENODEV;
+       }
+
+       pm_runtime_enable(&pdev->dev);
+       ret = pm_runtime_get_sync(&pdev->dev);
+       if (ret < 0) {
+               dev_err(&pdev->dev, "pm_runtime_get_sync() failed\n");
+               goto err;
+       }
+
+       rproc = rproc_alloc(dev, "wkup_m3", &wkup_m3_rproc_ops,
+                           fw_name, sizeof(*wkupm3));
+       if (!rproc) {
+               ret = -ENOMEM;
+               goto err;
+       }
+
+       wkupm3 = rproc->priv;
+       wkupm3->rproc = rproc;
+       wkupm3->pdev = pdev;
+
+       for (i = 0; i < ARRAY_SIZE(mem_names); i++) {
+               res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
+                                                  mem_names[i]);
+               wkupm3->mem[i].cpu_addr = devm_ioremap_resource(dev, res);
+               if (IS_ERR(wkupm3->mem[i].cpu_addr)) {
+                       dev_err(&pdev->dev, "devm_ioremap_resource failed for resource %d\n",
+                               i);
+                       ret = PTR_ERR(wkupm3->mem[i].cpu_addr);
+                       goto err;
+               }
+               wkupm3->mem[i].bus_addr = res->start;
+               wkupm3->mem[i].size = resource_size(res);
+               addrp = of_get_address(dev->of_node, i, &size, NULL);
+               /*
+                * The wkupm3 has umem at address 0 in its view, so the device
+                * addresses for each memory region is computed as a relative
+                * offset of the bus address for umem, and therefore needs to be
+                * processed first.
+                */
+               if (!strcmp(mem_names[i], "umem"))
+                       l4_offset = be32_to_cpu(*addrp);
+               wkupm3->mem[i].dev_addr = be32_to_cpu(*addrp) - l4_offset;
+       }
+
+       dev_set_drvdata(dev, rproc);
+
+       ret = rproc_add(rproc);
+       if (ret) {
+               dev_err(dev, "rproc_add failed\n");
+               goto err_put_rproc;
+       }
+
+       return 0;
+
+err_put_rproc:
+       rproc_put(rproc);
+err:
+       pm_runtime_put_noidle(dev);
+       pm_runtime_disable(dev);
+       return ret;
+}
+
+static int wkup_m3_rproc_remove(struct platform_device *pdev)
+{
+       struct rproc *rproc = platform_get_drvdata(pdev);
+
+       rproc_del(rproc);
+       rproc_put(rproc);
+       pm_runtime_put_sync(&pdev->dev);
+       pm_runtime_disable(&pdev->dev);
+
+       return 0;
+}
+
+#ifdef CONFIG_PM
+static int wkup_m3_rpm_suspend(struct device *dev)
+{
+       return -EBUSY;
+}
+
+static int wkup_m3_rpm_resume(struct device *dev)
+{
+       return 0;
+}
+#endif
+
+static const struct dev_pm_ops wkup_m3_rproc_pm_ops = {
+       SET_RUNTIME_PM_OPS(wkup_m3_rpm_suspend, wkup_m3_rpm_resume, NULL)
+};
+
+static struct platform_driver wkup_m3_rproc_driver = {
+       .probe = wkup_m3_rproc_probe,
+       .remove = wkup_m3_rproc_remove,
+       .driver = {
+               .name = "wkup_m3_rproc",
+               .of_match_table = wkup_m3_rproc_of_match,
+               .pm = &wkup_m3_rproc_pm_ops,
+       },
+};
+
+module_platform_driver(wkup_m3_rproc_driver);
+
+MODULE_LICENSE("GPL v2");
+MODULE_DESCRIPTION("TI Wakeup M3 remote processor control driver");
+MODULE_AUTHOR("Dave Gerlach <d-gerlach@ti.com>");
diff --git a/include/linux/platform_data/wkup_m3.h b/include/linux/platform_data/wkup_m3.h
new file mode 100644 (file)
index 0000000..3f1d77e
--- /dev/null
@@ -0,0 +1,30 @@
+/*
+ * TI Wakeup M3 remote processor platform data
+ *
+ * Copyright (C) 2014-2015 Texas Instruments, Inc.
+ *
+ * Dave Gerlach <d-gerlach@ti.com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ *
+ * This program 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.
+ */
+
+#ifndef _LINUX_PLATFORM_DATA_WKUP_M3_H
+#define _LINUX_PLATFORM_DATA_WKUP_M3_H
+
+struct platform_device;
+
+struct wkup_m3_platform_data {
+       const char *reset_name;
+
+       int (*assert_reset)(struct platform_device *pdev, const char *name);
+       int (*deassert_reset)(struct platform_device *pdev, const char *name);
+};
+
+#endif /* _LINUX_PLATFORM_DATA_WKUP_M3_H */
index 78b8a9b9d40a2ad70fe11986e8e6ffb3b04fe6c4..9c4e1384f63602f3eba04b46e22e323fa4a5c90f 100644 (file)
 #define REMOTEPROC_H
 
 #include <linux/types.h>
-#include <linux/klist.h>
 #include <linux/mutex.h>
 #include <linux/virtio.h>
 #include <linux/completion.h>
 #include <linux/idr.h>
+#include <linux/of.h>
 
 /**
  * struct resource_table - firmware resource table header
@@ -330,11 +330,13 @@ struct rproc;
  * @start:     power on the device and boot it
  * @stop:      power off the device
  * @kick:      kick a virtqueue (virtqueue id given as a parameter)
+ * @da_to_va:  optional platform hook to perform address translations
  */
 struct rproc_ops {
        int (*start)(struct rproc *rproc);
        int (*stop)(struct rproc *rproc);
        void (*kick)(struct rproc *rproc, int vqid);
+       void * (*da_to_va)(struct rproc *rproc, u64 da, int len);
 };
 
 /**
@@ -375,7 +377,7 @@ enum rproc_crash_type {
 
 /**
  * struct rproc - represents a physical remote processor device
- * @node: klist node of this rproc object
+ * @node: list node of this rproc object
  * @domain: iommu domain
  * @name: human readable name of the rproc
  * @firmware: name of firmware file to be loaded
@@ -407,7 +409,7 @@ enum rproc_crash_type {
  * @has_iommu: flag to indicate if remote processor is behind an MMU
  */
 struct rproc {
-       struct klist_node node;
+       struct list_head node;
        struct iommu_domain *domain;
        const char *name;
        const char *firmware;
@@ -481,6 +483,7 @@ struct rproc_vdev {
        u32 rsc_offset;
 };
 
+struct rproc *rproc_get_by_phandle(phandle phandle);
 struct rproc *rproc_alloc(struct device *dev, const char *name,
                                const struct rproc_ops *ops,
                                const char *firmware, int len);