Commit d880e5ad authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'rproc-v4.9' of git://github.com/andersson/remoteproc

Pull remoteproc updates from Bjorn Andersson:
 "In addition to a slew of minor fixes and cleanups these patches
  refactor how we deal with remoteprocs that will be auto-booting
  themselves.

  That does clean up the remote resource handling but makes for
  additional work to clarify responsibilities and life cycles of
  resources. We also revise how module locking of remoteproc drivers
  work, so that they are locked as we hand out references to them to
  third parties, rather than only when booted by anyone.

  In addition to that we also introduce the Qualcomm Wireless Subsystem
  remoteproc driver"

* tag 'rproc-v4.9' of git://github.com/andersson/remoteproc: (26 commits)
  remoteproc: Refactor rproc module locking
  remoteproc: Split driver and consumer dereferencing
  remoteproc: Correct resource handling upon boot failure
  remoteproc: Drop unnecessary NULL check
  remoteproc: core: transform struct fw_rsc_vdev_vring reserved field in pa
  remoteproc: Modify FW_RSC_ADDR_ANY definition
  remoteproc: qcom: wcnss: Fix return value check in wcnss_probe()
  remoteproc: qcom: Introduce WCNSS peripheral image loader
  dt-binding: remoteproc: Introduce Qualcomm WCNSS loader binding
  remoteproc: Only update table_ptr if we have a loaded table
  remoteproc: Move handling of cached table to boot/shutdown
  remoteproc: Move vdev handling to boot/shutdown
  remoteproc: Calculate max_notifyid during load
  remoteproc: Introduce auto-boot flag
  remoteproc/omap: revise a minor error trace message
  remoteproc/omap: fix various code formatting issues
  remoteproc: print hex numbers with a leading 0x format
  remoteproc: align code with open parenthesis
  remoteproc: fix bare unsigned type usage
  remoteproc: use variable names for sizeof() operator
  ...
parents 4c1fad64 fbb6aacb
Qualcomm WCNSS Peripheral Image Loader
This document defines the binding for a component that loads and boots firmware
on the Qualcomm WCNSS core.
- compatible:
Usage: required
Value type: <string>
Definition: must be one of:
"qcom,riva-pil",
"qcom,pronto-v1-pil",
"qcom,pronto-v2-pil"
- reg:
Usage: required
Value type: <prop-encoded-array>
Definition: must specify the base address and size of the CCU, DXE and
PMU register blocks
- reg-names:
Usage: required
Value type: <stringlist>
Definition: must be "ccu", "dxe", "pmu"
- interrupts-extended:
Usage: required
Value type: <prop-encoded-array>
Definition: must list the watchdog and fatal IRQs and may specify the
ready, handover and stop-ack IRQs
- interrupt-names:
Usage: required
Value type: <stringlist>
Definition: should be "wdog", "fatal", optionally followed by "ready",
"handover", "stop-ack"
- vddmx-supply:
- vddcx-supply:
- vddpx-supply:
Usage: required
Value type: <phandle>
Definition: reference to the regulators to be held on behalf of the
booting of the WCNSS core
- qcom,smem-states:
Usage: optional
Value type: <prop-encoded-array>
Definition: reference to the SMEM state used to indicate to WCNSS that
it should shut down
- qcom,smem-state-names:
Usage: optional
Value type: <stringlist>
Definition: should be "stop"
- memory-region:
Usage: required
Value type: <prop-encoded-array>
Definition: reference to reserved-memory node for the remote processor
see ../reserved-memory/reserved-memory.txt
= SUBNODES
A single subnode of the WCNSS PIL describes the attached rf module and its
resource dependencies.
- compatible:
Usage: required
Value type: <string>
Definition: must be one of:
"qcom,wcn3620",
"qcom,wcn3660",
"qcom,wcn3680"
- clocks:
Usage: required
Value type: <prop-encoded-array>
Definition: should specify the xo clock and optionally the rf clock
- clock-names:
Usage: required
Value type: <stringlist>
Definition: should be "xo", optionally followed by "rf"
- vddxo-supply:
- vddrfa-supply:
- vddpa-supply:
- vdddig-supply:
Usage: required
Value type: <phandle>
Definition: reference to the regulators to be held on behalf of the
booting of the WCNSS core
= EXAMPLE
The following example describes the resources needed to boot control the WCNSS,
with attached WCN3680, as it is commonly found on MSM8974 boards.
pronto@fb204000 {
compatible = "qcom,pronto-v2-pil";
reg = <0xfb204000 0x2000>, <0xfb202000 0x1000>, <0xfb21b000 0x3000>;
reg-names = "ccu", "dxe", "pmu";
interrupts-extended = <&intc 0 149 1>,
<&wcnss_smp2p_slave 0 0>,
<&wcnss_smp2p_slave 1 0>,
<&wcnss_smp2p_slave 2 0>,
<&wcnss_smp2p_slave 3 0>;
interrupt-names = "wdog", "fatal", "ready", "handover", "stop-ack";
vddmx-supply = <&pm8841_s1>;
vddcx-supply = <&pm8841_s2>;
vddpx-supply = <&pm8941_s3>;
qcom,smem-states = <&wcnss_smp2p_out 0>;
qcom,smem-state-names = "stop";
memory-region = <&wcnss_region>;
pinctrl-names = "default";
pinctrl-0 = <&wcnss_pin_a>;
iris {
compatible = "qcom,wcn3680";
clocks = <&rpmcc RPM_CXO_CLK_SRC>, <&rpmcc RPM_CXO_A2>;
clock-names = "xo", "rf";
vddxo-supply = <&pm8941_l6>;
vddrfa-supply = <&pm8941_l11>;
vddpa-supply = <&pm8941_l19>;
vdddig-supply = <&pm8941_s3>;
};
};
...@@ -101,9 +101,9 @@ int dummy_rproc_example(struct rproc *my_rproc) ...@@ -101,9 +101,9 @@ int dummy_rproc_example(struct rproc *my_rproc)
On success, the new rproc is returned, and on failure, NULL. On success, the new rproc is returned, and on failure, NULL.
Note: _never_ directly deallocate @rproc, even if it was not registered Note: _never_ directly deallocate @rproc, even if it was not registered
yet. Instead, when you need to unroll rproc_alloc(), use rproc_put(). yet. Instead, when you need to unroll rproc_alloc(), use rproc_free().
void rproc_put(struct rproc *rproc) void rproc_free(struct rproc *rproc)
- Free an rproc handle that was allocated by rproc_alloc. - Free an rproc handle that was allocated by rproc_alloc.
This function essentially unrolls rproc_alloc(), by decrementing the This function essentially unrolls rproc_alloc(), by decrementing the
rproc's refcount. It doesn't directly free rproc; that would happen rproc's refcount. It doesn't directly free rproc; that would happen
...@@ -131,7 +131,7 @@ int dummy_rproc_example(struct rproc *my_rproc) ...@@ -131,7 +131,7 @@ int dummy_rproc_example(struct rproc *my_rproc)
has completed successfully. has completed successfully.
After rproc_del() returns, @rproc is still valid, and its After rproc_del() returns, @rproc is still valid, and its
last refcount should be decremented by calling rproc_put(). last refcount should be decremented by calling rproc_free().
Returns 0 on success and -EINVAL if @rproc isn't valid. Returns 0 on success and -EINVAL if @rproc isn't valid.
......
...@@ -91,6 +91,22 @@ config QCOM_Q6V5_PIL ...@@ -91,6 +91,22 @@ config QCOM_Q6V5_PIL
Say y here to support the Qualcomm Peripherial Image Loader for the Say y here to support the Qualcomm Peripherial Image Loader for the
Hexagon V5 based remote processors. Hexagon V5 based remote processors.
config QCOM_WCNSS_IRIS
tristate
depends on OF && ARCH_QCOM
config QCOM_WCNSS_PIL
tristate "Qualcomm WCNSS Peripheral Image Loader"
depends on OF && ARCH_QCOM
depends on QCOM_SMEM
select QCOM_MDT_LOADER
select QCOM_SCM
select QCOM_WCNSS_IRIS
select REMOTEPROC
help
Say y here to support the Peripheral Image Loader for the Qualcomm
Wireless Connectivity Subsystem.
config ST_REMOTEPROC config ST_REMOTEPROC
tristate "ST remoteproc support" tristate "ST remoteproc support"
depends on ARCH_STI depends on ARCH_STI
......
...@@ -13,4 +13,6 @@ obj-$(CONFIG_WKUP_M3_RPROC) += wkup_m3_rproc.o ...@@ -13,4 +13,6 @@ obj-$(CONFIG_WKUP_M3_RPROC) += wkup_m3_rproc.o
obj-$(CONFIG_DA8XX_REMOTEPROC) += da8xx_remoteproc.o obj-$(CONFIG_DA8XX_REMOTEPROC) += da8xx_remoteproc.o
obj-$(CONFIG_QCOM_MDT_LOADER) += qcom_mdt_loader.o obj-$(CONFIG_QCOM_MDT_LOADER) += qcom_mdt_loader.o
obj-$(CONFIG_QCOM_Q6V5_PIL) += qcom_q6v5_pil.o obj-$(CONFIG_QCOM_Q6V5_PIL) += qcom_q6v5_pil.o
obj-$(CONFIG_QCOM_WCNSS_IRIS) += qcom_wcnss_iris.o
obj-$(CONFIG_QCOM_WCNSS_PIL) += qcom_wcnss.o
obj-$(CONFIG_ST_REMOTEPROC) += st_remoteproc.o obj-$(CONFIG_ST_REMOTEPROC) += st_remoteproc.o
...@@ -147,7 +147,7 @@ static void da8xx_rproc_kick(struct rproc *rproc, int vqid) ...@@ -147,7 +147,7 @@ static void da8xx_rproc_kick(struct rproc *rproc, int vqid)
{ {
struct da8xx_rproc *drproc = (struct da8xx_rproc *)rproc->priv; struct da8xx_rproc *drproc = (struct da8xx_rproc *)rproc->priv;
/* Interupt remote proc */ /* Interrupt remote proc */
writel(SYSCFG_CHIPSIG2, drproc->chipsig); writel(SYSCFG_CHIPSIG2, drproc->chipsig);
} }
...@@ -261,7 +261,7 @@ static int da8xx_rproc_probe(struct platform_device *pdev) ...@@ -261,7 +261,7 @@ static int da8xx_rproc_probe(struct platform_device *pdev)
return 0; return 0;
free_rproc: free_rproc:
rproc_put(rproc); rproc_free(rproc);
return ret; return ret;
} }
...@@ -290,7 +290,7 @@ static int da8xx_rproc_remove(struct platform_device *pdev) ...@@ -290,7 +290,7 @@ static int da8xx_rproc_remove(struct platform_device *pdev)
disable_irq(drproc->irq); disable_irq(drproc->irq);
rproc_del(rproc); rproc_del(rproc);
rproc_put(rproc); rproc_free(rproc);
return 0; return 0;
} }
......
...@@ -96,7 +96,8 @@ static void omap_rproc_kick(struct rproc *rproc, int vqid) ...@@ -96,7 +96,8 @@ static void omap_rproc_kick(struct rproc *rproc, int vqid)
/* send the index of the triggered virtqueue in the mailbox payload */ /* send the index of the triggered virtqueue in the mailbox payload */
ret = mbox_send_message(oproc->mbox, (void *)vqid); ret = mbox_send_message(oproc->mbox, (void *)vqid);
if (ret < 0) if (ret < 0)
dev_err(dev, "omap_mbox_msg_send failed: %d\n", ret); dev_err(dev, "failed to send mailbox message, status = %d\n",
ret);
} }
/* /*
...@@ -196,7 +197,7 @@ static int omap_rproc_probe(struct platform_device *pdev) ...@@ -196,7 +197,7 @@ static int omap_rproc_probe(struct platform_device *pdev)
} }
rproc = rproc_alloc(&pdev->dev, pdata->name, &omap_rproc_ops, rproc = rproc_alloc(&pdev->dev, pdata->name, &omap_rproc_ops,
pdata->firmware, sizeof(*oproc)); pdata->firmware, sizeof(*oproc));
if (!rproc) if (!rproc)
return -ENOMEM; return -ENOMEM;
...@@ -214,7 +215,7 @@ static int omap_rproc_probe(struct platform_device *pdev) ...@@ -214,7 +215,7 @@ static int omap_rproc_probe(struct platform_device *pdev)
return 0; return 0;
free_rproc: free_rproc:
rproc_put(rproc); rproc_free(rproc);
return ret; return ret;
} }
...@@ -223,7 +224,7 @@ static int omap_rproc_remove(struct platform_device *pdev) ...@@ -223,7 +224,7 @@ static int omap_rproc_remove(struct platform_device *pdev)
struct rproc *rproc = platform_get_drvdata(pdev); struct rproc *rproc = platform_get_drvdata(pdev);
rproc_del(rproc); rproc_del(rproc);
rproc_put(rproc); rproc_free(rproc);
return 0; return 0;
} }
......
...@@ -863,8 +863,10 @@ static int q6v5_probe(struct platform_device *pdev) ...@@ -863,8 +863,10 @@ static int q6v5_probe(struct platform_device *pdev)
goto free_rproc; goto free_rproc;
qproc->state = qcom_smem_state_get(&pdev->dev, "stop", &qproc->stop_bit); qproc->state = qcom_smem_state_get(&pdev->dev, "stop", &qproc->stop_bit);
if (IS_ERR(qproc->state)) if (IS_ERR(qproc->state)) {
ret = PTR_ERR(qproc->state);
goto free_rproc; goto free_rproc;
}
ret = rproc_add(rproc); ret = rproc_add(rproc);
if (ret) if (ret)
...@@ -873,7 +875,7 @@ static int q6v5_probe(struct platform_device *pdev) ...@@ -873,7 +875,7 @@ static int q6v5_probe(struct platform_device *pdev)
return 0; return 0;
free_rproc: free_rproc:
rproc_put(rproc); rproc_free(rproc);
return ret; return ret;
} }
...@@ -883,7 +885,7 @@ static int q6v5_remove(struct platform_device *pdev) ...@@ -883,7 +885,7 @@ static int q6v5_remove(struct platform_device *pdev)
struct q6v5 *qproc = platform_get_drvdata(pdev); struct q6v5 *qproc = platform_get_drvdata(pdev);
rproc_del(qproc->rproc); rproc_del(qproc->rproc);
rproc_put(qproc->rproc); rproc_free(qproc->rproc);
return 0; return 0;
} }
......
This diff is collapsed.
#ifndef __QCOM_WNCSS_H__
#define __QCOM_WNCSS_H__
struct qcom_iris;
struct qcom_wcnss;
struct wcnss_vreg_info {
const char * const name;
int min_voltage;
int max_voltage;
int load_uA;
bool super_turbo;
};
int qcom_iris_enable(struct qcom_iris *iris);
void qcom_iris_disable(struct qcom_iris *iris);
void qcom_wcnss_assign_iris(struct qcom_wcnss *wcnss, struct qcom_iris *iris, bool use_48mhz_xo);
#endif
/*
* Qualcomm Wireless Connectivity Subsystem Iris driver
*
* Copyright (C) 2016 Linaro Ltd
* Copyright (C) 2014 Sony Mobile Communications AB
* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
*
* 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/clk.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/of_device.h>
#include <linux/platform_device.h>
#include <linux/regulator/consumer.h>
#include "qcom_wcnss.h"
struct qcom_iris {
struct device *dev;
struct clk *xo_clk;
struct regulator_bulk_data *vregs;
size_t num_vregs;
};
struct iris_data {
const struct wcnss_vreg_info *vregs;
size_t num_vregs;
bool use_48mhz_xo;
};
static const struct iris_data wcn3620_data = {
.vregs = (struct wcnss_vreg_info[]) {
{ "vddxo", 1800000, 1800000, 10000 },
{ "vddrfa", 1300000, 1300000, 100000 },
{ "vddpa", 3300000, 3300000, 515000 },
{ "vdddig", 1800000, 1800000, 10000 },
},
.num_vregs = 4,
.use_48mhz_xo = false,
};
static const struct iris_data wcn3660_data = {
.vregs = (struct wcnss_vreg_info[]) {
{ "vddxo", 1800000, 1800000, 10000 },
{ "vddrfa", 1300000, 1300000, 100000 },
{ "vddpa", 2900000, 3000000, 515000 },
{ "vdddig", 1200000, 1225000, 10000 },
},
.num_vregs = 4,
.use_48mhz_xo = true,
};
static const struct iris_data wcn3680_data = {
.vregs = (struct wcnss_vreg_info[]) {
{ "vddxo", 1800000, 1800000, 10000 },
{ "vddrfa", 1300000, 1300000, 100000 },
{ "vddpa", 3300000, 3300000, 515000 },
{ "vdddig", 1800000, 1800000, 10000 },
},
.num_vregs = 4,
.use_48mhz_xo = true,
};
int qcom_iris_enable(struct qcom_iris *iris)
{
int ret;
ret = regulator_bulk_enable(iris->num_vregs, iris->vregs);
if (ret)
return ret;
ret = clk_prepare_enable(iris->xo_clk);
if (ret) {
dev_err(iris->dev, "failed to enable xo clk\n");
goto disable_regulators;
}
return 0;
disable_regulators:
regulator_bulk_disable(iris->num_vregs, iris->vregs);
return ret;
}
EXPORT_SYMBOL_GPL(qcom_iris_enable);
void qcom_iris_disable(struct qcom_iris *iris)
{
clk_disable_unprepare(iris->xo_clk);
regulator_bulk_disable(iris->num_vregs, iris->vregs);
}
EXPORT_SYMBOL_GPL(qcom_iris_disable);
static int qcom_iris_probe(struct platform_device *pdev)
{
const struct iris_data *data;
struct qcom_wcnss *wcnss;
struct qcom_iris *iris;
int ret;
int i;
iris = devm_kzalloc(&pdev->dev, sizeof(struct qcom_iris), GFP_KERNEL);
if (!iris)
return -ENOMEM;
data = of_device_get_match_data(&pdev->dev);
wcnss = dev_get_drvdata(pdev->dev.parent);
iris->xo_clk = devm_clk_get(&pdev->dev, "xo");
if (IS_ERR(iris->xo_clk)) {
if (PTR_ERR(iris->xo_clk) != -EPROBE_DEFER)
dev_err(&pdev->dev, "failed to acquire xo clk\n");
return PTR_ERR(iris->xo_clk);
}
iris->num_vregs = data->num_vregs;
iris->vregs = devm_kcalloc(&pdev->dev,
iris->num_vregs,
sizeof(struct regulator_bulk_data),
GFP_KERNEL);
if (!iris->vregs)
return -ENOMEM;
for (i = 0; i < iris->num_vregs; i++)
iris->vregs[i].supply = data->vregs[i].name;
ret = devm_regulator_bulk_get(&pdev->dev, iris->num_vregs, iris->vregs);
if (ret) {
dev_err(&pdev->dev, "failed to get regulators\n");
return ret;
}
for (i = 0; i < iris->num_vregs; i++) {
if (data->vregs[i].max_voltage)
regulator_set_voltage(iris->vregs[i].consumer,
data->vregs[i].min_voltage,
data->vregs[i].max_voltage);
if (data->vregs[i].load_uA)
regulator_set_load(iris->vregs[i].consumer,
data->vregs[i].load_uA);
}
qcom_wcnss_assign_iris(wcnss, iris, data->use_48mhz_xo);
return 0;
}
static int qcom_iris_remove(struct platform_device *pdev)
{
struct qcom_wcnss *wcnss = dev_get_drvdata(pdev->dev.parent);
qcom_wcnss_assign_iris(wcnss, NULL, false);
return 0;
}
static const struct of_device_id iris_of_match[] = {
{ .compatible = "qcom,wcn3620", .data = &wcn3620_data },
{ .compatible = "qcom,wcn3660", .data = &wcn3660_data },
{ .compatible = "qcom,wcn3680", .data = &wcn3680_data },
{}
};
static struct platform_driver wcnss_driver = {
.probe = qcom_iris_probe,
.remove = qcom_iris_remove,
.driver = {
.name = "qcom-iris",
.of_match_table = iris_of_match,
},
};
module_platform_driver(wcnss_driver);
MODULE_DESCRIPTION("Qualcomm Wireless Subsystem Iris driver");
MODULE_LICENSE("GPL v2");
This diff is collapsed.
...@@ -45,7 +45,7 @@ static struct dentry *rproc_dbg; ...@@ -45,7 +45,7 @@ static struct dentry *rproc_dbg;
* as it provides very early tracing with little to no dependencies at all. * as it provides very early tracing with little to no dependencies at all.
*/ */
static ssize_t rproc_trace_read(struct file *filp, char __user *userbuf, static ssize_t rproc_trace_read(struct file *filp, char __user *userbuf,
size_t count, loff_t *ppos) size_t count, loff_t *ppos)
{ {
struct rproc_mem_entry *trace = filp->private_data; struct rproc_mem_entry *trace = filp->private_data;
int len = strnlen(trace->va, trace->len); int len = strnlen(trace->va, trace->len);
...@@ -73,7 +73,7 @@ static const char * const rproc_state_string[] = { ...@@ -73,7 +73,7 @@ static const char * const rproc_state_string[] = {
/* expose the state of the remote processor via debugfs */ /* expose the state of the remote processor via debugfs */
static ssize_t rproc_state_read(struct file *filp, char __user *userbuf, static ssize_t rproc_state_read(struct file *filp, char __user *userbuf,
size_t count, loff_t *ppos) size_t count, loff_t *ppos)
{ {
struct rproc *rproc = filp->private_data; struct rproc *rproc = filp->private_data;
unsigned int state; unsigned int state;
...@@ -83,7 +83,7 @@ static ssize_t rproc_state_read(struct file *filp, char __user *userbuf, ...@@ -83,7 +83,7 @@ static ssize_t rproc_state_read(struct file *filp, char __user *userbuf,
state = rproc->state > RPROC_LAST ? RPROC_LAST : rproc->state; state = rproc->state > RPROC_LAST ? RPROC_LAST : rproc->state;
i = scnprintf(buf, 30, "%.28s (%d)\n", rproc_state_string[state], i = scnprintf(buf, 30, "%.28s (%d)\n", rproc_state_string[state],
rproc->state); rproc->state);
return simple_read_from_buffer(userbuf, count, ppos, buf, i); return simple_read_from_buffer(userbuf, count, ppos, buf, i);
} }
...@@ -130,7 +130,7 @@ static const struct file_operations rproc_state_ops = { ...@@ -130,7 +130,7 @@ static const struct file_operations rproc_state_ops = {
/* expose the name of the remote processor via debugfs */ /* expose the name of the remote processor via debugfs */
static ssize_t rproc_name_read(struct file *filp, char __user *userbuf, static ssize_t rproc_name_read(struct file *filp, char __user *userbuf,
size_t count, loff_t *ppos) size_t count, loff_t *ppos)
{ {
struct rproc *rproc = filp->private_data; struct rproc *rproc = filp->private_data;
/* need room for the name, a newline and a terminating null */ /* need room for the name, a newline and a terminating null */
...@@ -230,12 +230,12 @@ void rproc_remove_trace_file(struct dentry *tfile) ...@@ -230,12 +230,12 @@ void rproc_remove_trace_file(struct dentry *tfile)
} }
struct dentry *rproc_create_trace_file(const char *name, struct rproc *rproc, struct dentry *rproc_create_trace_file(const char *name, struct rproc *rproc,
struct rproc_mem_entry *trace) struct rproc_mem_entry *trace)
{ {
struct dentry *tfile; struct dentry *tfile;
tfile = debugfs_create_file(name, 0400, rproc->dbg_dir, tfile = debugfs_create_file(name, 0400, rproc->dbg_dir, trace,
trace, &trace_rproc_ops); &trace_rproc_ops);
if (!tfile) { if (!tfile) {
dev_err(&rproc->dev, "failed to create debugfs trace entry\n"); dev_err(&rproc->dev, "failed to create debugfs trace entry\n");
return NULL; return NULL;
...@@ -264,11 +264,11 @@ void rproc_create_debug_dir(struct rproc *rproc) ...@@ -264,11 +264,11 @@ void rproc_create_debug_dir(struct rproc *rproc)
return; return;
debugfs_create_file("name", 0400, rproc->dbg_dir, debugfs_create_file("name", 0400, rproc->dbg_dir,
rproc, &rproc_name_ops); rproc, &rproc_name_ops);
debugfs_create_file("state", 0400, rproc->dbg_dir, debugfs_create_file("state", 0400, rproc->dbg_dir,
rproc, &rproc_state_ops); rproc, &rproc_state_ops);
debugfs_create_file("recovery", 0400, rproc->dbg_dir, debugfs_create_file("recovery", 0400, rproc->dbg_dir,
rproc, &rproc_recovery_ops); rproc, &rproc_recovery_ops);
} }
void __init rproc_init_debugfs(void) void __init rproc_init_debugfs(void)
......
...@@ -166,18 +166,18 @@ rproc_elf_load_segments(struct rproc *rproc, const struct firmware *fw) ...@@ -166,18 +166,18 @@ rproc_elf_load_segments(struct rproc *rproc, const struct firmware *fw)
continue; continue;
dev_dbg(dev, "phdr: type %d da 0x%x memsz 0x%x filesz 0x%x\n", dev_dbg(dev, "phdr: type %d da 0x%x memsz 0x%x filesz 0x%x\n",
phdr->p_type, da, memsz, filesz); phdr->p_type, da, memsz, filesz);
if (filesz > memsz) { if (filesz > memsz) {
dev_err(dev, "bad phdr filesz 0x%x memsz 0x%x\n", dev_err(dev, "bad phdr filesz 0x%x memsz 0x%x\n",
filesz, memsz); filesz, memsz);
ret = -EINVAL; ret = -EINVAL;
break; break;
} }
if (offset + filesz > fw->size) { if (offset + filesz > fw->size) {
dev_err(dev, "truncated fw: need 0x%x avail 0x%zx\n", dev_err(dev, "truncated fw: need 0x%x avail 0x%zx\n",
offset + filesz, fw->size); offset + filesz, fw->size);
ret = -EINVAL; ret = -EINVAL;
break; break;
} }
......
...@@ -36,10 +36,10 @@ struct rproc; ...@@ -36,10 +36,10 @@ struct rproc;
*/ */
struct rproc_fw_ops { 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, const struct firmware *fw,
int *tablesz); int *tablesz);
struct resource_table *(*find_loaded_rsc_table)(struct rproc *rproc, struct resource_table *(*find_loaded_rsc_table)(
const struct firmware *fw); struct rproc *rproc, const struct firmware *fw);
int (*load)(struct rproc *rproc, const struct firmware *fw); int (*load)(struct rproc *rproc, const struct firmware *fw);
int (*sanity_check)(struct rproc *rproc, const struct firmware *fw); int (*sanity_check)(struct rproc *rproc, const struct firmware *fw);
u32 (*get_boot_addr)(struct rproc *rproc, const struct firmware *fw); u32 (*get_boot_addr)(struct rproc *rproc, const struct firmware *fw);
...@@ -57,7 +57,7 @@ void rproc_remove_virtio_dev(struct rproc_vdev *rvdev); ...@@ -57,7 +57,7 @@ void rproc_remove_virtio_dev(struct rproc_vdev *rvdev);
/* from remoteproc_debugfs.c */ /* from remoteproc_debugfs.c */
void rproc_remove_trace_file(struct dentry *tfile); void rproc_remove_trace_file(struct dentry *tfile);
struct dentry *rproc_create_trace_file(const char *name, struct rproc *rproc, struct dentry *rproc_create_trace_file(const char *name, struct rproc *rproc,
struct rproc_mem_entry *trace); struct rproc_mem_entry *trace);
void rproc_delete_debug_dir(struct rproc *rproc); void rproc_delete_debug_dir(struct rproc *rproc);
void rproc_create_debug_dir(struct rproc *rproc); void rproc_create_debug_dir(struct rproc *rproc);
void rproc_init_debugfs(void); void rproc_init_debugfs(void);
...@@ -98,7 +98,8 @@ int rproc_load_segments(struct rproc *rproc, const struct firmware *fw) ...@@ -98,7 +98,8 @@ int rproc_load_segments(struct rproc *rproc, const struct firmware *fw)
static inline static inline
struct resource_table *rproc_find_rsc_table(struct rproc *rproc, struct resource_table *rproc_find_rsc_table(struct rproc *rproc,
const struct firmware *fw, int *tablesz) const struct firmware *fw,
int *tablesz)
{ {
if (rproc->fw_ops->find_rsc_table) if (rproc->fw_ops->find_rsc_table)
return rproc->fw_ops->find_rsc_table(rproc, fw, tablesz); return rproc->fw_ops->find_rsc_table(rproc, fw, tablesz);
...@@ -108,7 +109,7 @@ struct resource_table *rproc_find_rsc_table(struct rproc *rproc, ...@@ -108,7 +109,7 @@ struct resource_table *rproc_find_rsc_table(struct rproc *rproc,
static inline static inline
struct resource_table *rproc_find_loaded_rsc_table(struct rproc *rproc, struct resource_table *rproc_find_loaded_rsc_table(struct rproc *rproc,
const struct firmware *fw) const struct firmware *fw)
{ {
if (rproc->fw_ops->find_loaded_rsc_table) if (rproc->fw_ops->find_loaded_rsc_table)
return rproc->fw_ops->find_loaded_rsc_table(rproc, fw); return rproc->fw_ops->find_loaded_rsc_table(rproc, fw);
......
...@@ -69,7 +69,7 @@ irqreturn_t rproc_vq_interrupt(struct rproc *rproc, int notifyid) ...@@ -69,7 +69,7 @@ irqreturn_t rproc_vq_interrupt(struct rproc *rproc, int notifyid)
EXPORT_SYMBOL(rproc_vq_interrupt); EXPORT_SYMBOL(rproc_vq_interrupt);
static struct virtqueue *rp_find_vq(struct virtio_device *vdev, static struct virtqueue *rp_find_vq(struct virtio_device *vdev,
unsigned id, unsigned int id,
void (*callback)(struct virtqueue *vq), void (*callback)(struct virtqueue *vq),
const char *name) const char *name)
{ {
...@@ -101,14 +101,14 @@ static struct virtqueue *rp_find_vq(struct virtio_device *vdev, ...@@ -101,14 +101,14 @@ static struct virtqueue *rp_find_vq(struct virtio_device *vdev,
memset(addr, 0, size); memset(addr, 0, size);
dev_dbg(dev, "vring%d: va %p qsz %d notifyid %d\n", dev_dbg(dev, "vring%d: va %p qsz %d notifyid %d\n",
id, addr, len, rvring->notifyid); id, addr, len, rvring->notifyid);
/* /*
* Create the new vq, and tell virtio we're not interested in * Create the new vq, and tell virtio we're not interested in
* the 'weak' smp barriers, since we're talking with a real device. * the 'weak' smp barriers, since we're talking with a real device.
*/ */
vq = vring_new_virtqueue(id, len, rvring->align, vdev, false, addr, vq = vring_new_virtqueue(id, len, rvring->align, vdev, false, addr,
rproc_virtio_notify, callback, name); rproc_virtio_notify, callback, name);
if (!vq) { if (!vq) {
dev_err(dev, "vring_new_virtqueue %s failed\n", name); dev_err(dev, "vring_new_virtqueue %s failed\n", name);
rproc_free_vring(rvring); rproc_free_vring(rvring);
...@@ -136,20 +136,14 @@ static void __rproc_virtio_del_vqs(struct virtio_device *vdev) ...@@ -136,20 +136,14 @@ static void __rproc_virtio_del_vqs(struct virtio_device *vdev)
static void rproc_virtio_del_vqs(struct virtio_device *vdev) static void rproc_virtio_del_vqs(struct virtio_device *vdev)
{ {
struct rproc *rproc = vdev_to_rproc(vdev);
/* power down the remote processor before deleting vqs */
rproc_shutdown(rproc);
__rproc_virtio_del_vqs(vdev); __rproc_virtio_del_vqs(vdev);
} }
static int rproc_virtio_find_vqs(struct virtio_device *vdev, unsigned nvqs, static int rproc_virtio_find_vqs(struct virtio_device *vdev, unsigned int nvqs,
struct virtqueue *vqs[], struct virtqueue *vqs[],
vq_callback_t *callbacks[], vq_callback_t *callbacks[],
const char * const names[]) const char * const names[])
{ {
struct rproc *rproc = vdev_to_rproc(vdev);
int i, ret; int i, ret;
for (i = 0; i < nvqs; ++i) { for (i = 0; i < nvqs; ++i) {
...@@ -160,13 +154,6 @@ static int rproc_virtio_find_vqs(struct virtio_device *vdev, unsigned nvqs, ...@@ -160,13 +154,6 @@ static int rproc_virtio_find_vqs(struct virtio_device *vdev, unsigned nvqs,
} }
} }
/* now that the vqs are all set, boot the remote processor */
ret = rproc_boot_nowait(rproc);
if (ret) {
dev_err(&rproc->dev, "rproc_boot() failed %d\n", ret);
goto error;
}
return 0; return 0;
error: error:
...@@ -239,8 +226,8 @@ static int rproc_virtio_finalize_features(struct virtio_device *vdev) ...@@ -239,8 +226,8 @@ static int rproc_virtio_finalize_features(struct virtio_device *vdev)
return 0; return 0;
} }
static void rproc_virtio_get(struct virtio_device *vdev, unsigned offset, static void rproc_virtio_get(struct virtio_device *vdev, unsigned int offset,
void *buf, unsigned len) void *buf, unsigned int len)
{ {
struct rproc_vdev *rvdev = vdev_to_rvdev(vdev); struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
struct fw_rsc_vdev *rsc; struct fw_rsc_vdev *rsc;
...@@ -257,8 +244,8 @@ static void rproc_virtio_get(struct virtio_device *vdev, unsigned offset, ...@@ -257,8 +244,8 @@ static void rproc_virtio_get(struct virtio_device *vdev, unsigned offset,
memcpy(buf, cfg + offset, len); memcpy(buf, cfg + offset, len);
} }
static void rproc_virtio_set(struct virtio_device *vdev, unsigned offset, static void rproc_virtio_set(struct virtio_device *vdev, unsigned int offset,
const void *buf, unsigned len) const void *buf, unsigned int len)
{ {
struct rproc_vdev *rvdev = vdev_to_rvdev(vdev); struct rproc_vdev *rvdev = vdev_to_rvdev(vdev);
struct fw_rsc_vdev *rsc; struct fw_rsc_vdev *rsc;
......
...@@ -262,7 +262,7 @@ static int st_rproc_probe(struct platform_device *pdev) ...@@ -262,7 +262,7 @@ static int st_rproc_probe(struct platform_device *pdev)
return 0; return 0;
free_rproc: free_rproc:
rproc_put(rproc); rproc_free(rproc);
return ret; return ret;
} }
...@@ -277,7 +277,7 @@ static int st_rproc_remove(struct platform_device *pdev) ...@@ -277,7 +277,7 @@ static int st_rproc_remove(struct platform_device *pdev)
of_reserved_mem_device_release(&pdev->dev); of_reserved_mem_device_release(&pdev->dev);
rproc_put(rproc); rproc_free(rproc);
return 0; return 0;
} }
......
...@@ -257,7 +257,7 @@ static int sproc_drv_remove(struct platform_device *pdev) ...@@ -257,7 +257,7 @@ static int sproc_drv_remove(struct platform_device *pdev)
rproc_del(sproc->rproc); rproc_del(sproc->rproc);
dma_free_coherent(sproc->rproc->dev.parent, SPROC_FW_SIZE, dma_free_coherent(sproc->rproc->dev.parent, SPROC_FW_SIZE,
sproc->fw_addr, sproc->fw_dma_addr); sproc->fw_addr, sproc->fw_dma_addr);
rproc_put(sproc->rproc); rproc_free(sproc->rproc);
mdev->drv_data = NULL; mdev->drv_data = NULL;
...@@ -325,7 +325,7 @@ static int sproc_probe(struct platform_device *pdev) ...@@ -325,7 +325,7 @@ static int sproc_probe(struct platform_device *pdev)
free_rproc: free_rproc:
/* Reset device data upon error */ /* Reset device data upon error */
mdev->drv_data = NULL; mdev->drv_data = NULL;
rproc_put(rproc); rproc_free(rproc);
return err; return err;
} }
......
...@@ -167,6 +167,8 @@ static int wkup_m3_rproc_probe(struct platform_device *pdev) ...@@ -167,6 +167,8 @@ static int wkup_m3_rproc_probe(struct platform_device *pdev)
goto err; goto err;
} }
rproc->auto_boot = false;
wkupm3 = rproc->priv; wkupm3 = rproc->priv;
wkupm3->rproc = rproc; wkupm3->rproc = rproc;
wkupm3->pdev = pdev; wkupm3->pdev = pdev;
...@@ -206,7 +208,7 @@ static int wkup_m3_rproc_probe(struct platform_device *pdev) ...@@ -206,7 +208,7 @@ static int wkup_m3_rproc_probe(struct platform_device *pdev)
return 0; return 0;
err_put_rproc: err_put_rproc:
rproc_put(rproc); rproc_free(rproc);
err: err:
pm_runtime_put_noidle(dev); pm_runtime_put_noidle(dev);
pm_runtime_disable(dev); pm_runtime_disable(dev);
...@@ -218,7 +220,7 @@ static int wkup_m3_rproc_remove(struct platform_device *pdev) ...@@ -218,7 +220,7 @@ static int wkup_m3_rproc_remove(struct platform_device *pdev)
struct rproc *rproc = platform_get_drvdata(pdev); struct rproc *rproc = platform_get_drvdata(pdev);
rproc_del(rproc); rproc_del(rproc);
rproc_put(rproc); rproc_free(rproc);
pm_runtime_put_sync(&pdev->dev); pm_runtime_put_sync(&pdev->dev);
pm_runtime_disable(&pdev->dev); pm_runtime_disable(&pdev->dev);
......
...@@ -39,9 +39,9 @@ struct omap_rproc_pdata { ...@@ -39,9 +39,9 @@ struct omap_rproc_pdata {
const char *firmware; const char *firmware;
const char *mbox_name; const char *mbox_name;
const struct rproc_ops *ops; const struct rproc_ops *ops;
int (*device_enable) (struct platform_device *pdev); int (*device_enable)(struct platform_device *pdev);
int (*device_shutdown) (struct platform_device *pdev); int (*device_shutdown)(struct platform_device *pdev);
void(*set_bootaddr)(u32); void (*set_bootaddr)(u32);
}; };
#if defined(CONFIG_OMAP_REMOTEPROC) || defined(CONFIG_OMAP_REMOTEPROC_MODULE) #if defined(CONFIG_OMAP_REMOTEPROC) || defined(CONFIG_OMAP_REMOTEPROC_MODULE)
......
...@@ -118,7 +118,7 @@ enum fw_resource_type { ...@@ -118,7 +118,7 @@ enum fw_resource_type {
RSC_LAST = 4, RSC_LAST = 4,
}; };
#define FW_RSC_ADDR_ANY (0xFFFFFFFFFFFFFFFF) #define FW_RSC_ADDR_ANY (-1)
/** /**
* struct fw_rsc_carveout - physically contiguous memory request * struct fw_rsc_carveout - physically contiguous memory request
...@@ -241,7 +241,7 @@ struct fw_rsc_trace { ...@@ -241,7 +241,7 @@ struct fw_rsc_trace {
* @notifyid is a unique rproc-wide notify index for this vring. This notify * @notifyid is a unique rproc-wide notify index for this vring. This notify
* index is used when kicking a remote processor, to let it know that this * index is used when kicking a remote processor, to let it know that this
* vring is triggered. * vring is triggered.
* @reserved: reserved (must be zero) * @pa: physical address
* *
* This descriptor is not a resource entry by itself; it is part of the * This descriptor is not a resource entry by itself; it is part of the
* vdev resource type (see below). * vdev resource type (see below).
...@@ -255,7 +255,7 @@ struct fw_rsc_vdev_vring { ...@@ -255,7 +255,7 @@ struct fw_rsc_vdev_vring {
u32 align; u32 align;
u32 num; u32 num;
u32 notifyid; u32 notifyid;
u32 reserved; u32 pa;
} __packed; } __packed;
/** /**
...@@ -409,7 +409,6 @@ enum rproc_crash_type { ...@@ -409,7 +409,6 @@ enum rproc_crash_type {
* @max_notifyid: largest allocated notify id. * @max_notifyid: largest allocated notify id.
* @table_ptr: pointer to the resource table in effect * @table_ptr: pointer to the resource table in effect
* @cached_table: copy of the resource table * @cached_table: copy of the resource table
* @table_csum: checksum of the resource table
* @has_iommu: flag to indicate if remote processor is behind an MMU * @has_iommu: flag to indicate if remote processor is behind an MMU
*/ */
struct rproc { struct rproc {
...@@ -435,14 +434,14 @@ struct rproc { ...@@ -435,14 +434,14 @@ struct rproc {
struct idr notifyids; struct idr notifyids;
int index; int index;
struct work_struct crash_handler; struct work_struct crash_handler;
unsigned crash_cnt; unsigned int crash_cnt;
struct completion crash_comp; struct completion crash_comp;
bool recovery_disabled; bool recovery_disabled;
int max_notifyid; int max_notifyid;
struct resource_table *table_ptr; struct resource_table *table_ptr;
struct resource_table *cached_table; struct resource_table *cached_table;
u32 table_csum;
bool has_iommu; bool has_iommu;
bool auto_boot;
}; };
/* we currently support only two vrings per rvdev */ /* we currently support only two vrings per rvdev */
...@@ -489,11 +488,12 @@ struct rproc_vdev { ...@@ -489,11 +488,12 @@ struct rproc_vdev {
struct rproc *rproc_get_by_phandle(phandle phandle); struct rproc *rproc_get_by_phandle(phandle phandle);
struct rproc *rproc_alloc(struct device *dev, const char *name, struct rproc *rproc_alloc(struct device *dev, const char *name,
const struct rproc_ops *ops, const struct rproc_ops *ops,
const char *firmware, int len); const char *firmware, int len);
void rproc_put(struct rproc *rproc); void rproc_put(struct rproc *rproc);
int rproc_add(struct rproc *rproc); int rproc_add(struct rproc *rproc);
int rproc_del(struct rproc *rproc); int rproc_del(struct rproc *rproc);
void rproc_free(struct rproc *rproc);
int rproc_boot(struct rproc *rproc); int rproc_boot(struct rproc *rproc);
void rproc_shutdown(struct rproc *rproc); void rproc_shutdown(struct rproc *rproc);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment