mtd: rawnand: tegra: Add runtime PM and OPP support
The NAND on Tegra belongs to the core power domain and we're going to enable GENPD support for the core domain. Now NAND must be resumed using runtime PM API in order to initialize the NAND power state. Add runtime PM and OPP support to the NAND driver. Reviewed-by: Ulf Hansson <ulf.hansson@linaro.org> Acked-by: Miquel Raynal <miquel.raynal@bootlin.com> Signed-off-by: Dmitry Osipenko <digetx@gmail.com> Signed-off-by: Thierry Reding <treding@nvidia.com>
This commit is contained in:
parent
d618978dd4
commit
6902dc2fd5
@ -17,8 +17,11 @@
|
|||||||
#include <linux/mtd/rawnand.h>
|
#include <linux/mtd/rawnand.h>
|
||||||
#include <linux/of.h>
|
#include <linux/of.h>
|
||||||
#include <linux/platform_device.h>
|
#include <linux/platform_device.h>
|
||||||
|
#include <linux/pm_runtime.h>
|
||||||
#include <linux/reset.h>
|
#include <linux/reset.h>
|
||||||
|
|
||||||
|
#include <soc/tegra/common.h>
|
||||||
|
|
||||||
#define COMMAND 0x00
|
#define COMMAND 0x00
|
||||||
#define COMMAND_GO BIT(31)
|
#define COMMAND_GO BIT(31)
|
||||||
#define COMMAND_CLE BIT(30)
|
#define COMMAND_CLE BIT(30)
|
||||||
@ -1151,6 +1154,7 @@ static int tegra_nand_probe(struct platform_device *pdev)
|
|||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
ctrl->dev = &pdev->dev;
|
ctrl->dev = &pdev->dev;
|
||||||
|
platform_set_drvdata(pdev, ctrl);
|
||||||
nand_controller_init(&ctrl->controller);
|
nand_controller_init(&ctrl->controller);
|
||||||
ctrl->controller.ops = &tegra_nand_controller_ops;
|
ctrl->controller.ops = &tegra_nand_controller_ops;
|
||||||
|
|
||||||
@ -1166,14 +1170,23 @@ static int tegra_nand_probe(struct platform_device *pdev)
|
|||||||
if (IS_ERR(ctrl->clk))
|
if (IS_ERR(ctrl->clk))
|
||||||
return PTR_ERR(ctrl->clk);
|
return PTR_ERR(ctrl->clk);
|
||||||
|
|
||||||
err = clk_prepare_enable(ctrl->clk);
|
err = devm_tegra_core_dev_init_opp_table_common(&pdev->dev);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This driver doesn't support active power management yet,
|
||||||
|
* so we will simply keep device resumed.
|
||||||
|
*/
|
||||||
|
pm_runtime_enable(&pdev->dev);
|
||||||
|
err = pm_runtime_resume_and_get(&pdev->dev);
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
err = reset_control_reset(rst);
|
err = reset_control_reset(rst);
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_err(ctrl->dev, "Failed to reset HW: %d\n", err);
|
dev_err(ctrl->dev, "Failed to reset HW: %d\n", err);
|
||||||
goto err_disable_clk;
|
goto err_put_pm;
|
||||||
}
|
}
|
||||||
|
|
||||||
writel_relaxed(HWSTATUS_CMD_DEFAULT, ctrl->regs + HWSTATUS_CMD);
|
writel_relaxed(HWSTATUS_CMD_DEFAULT, ctrl->regs + HWSTATUS_CMD);
|
||||||
@ -1188,21 +1201,20 @@ static int tegra_nand_probe(struct platform_device *pdev)
|
|||||||
dev_name(&pdev->dev), ctrl);
|
dev_name(&pdev->dev), ctrl);
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_err(ctrl->dev, "Failed to get IRQ: %d\n", err);
|
dev_err(ctrl->dev, "Failed to get IRQ: %d\n", err);
|
||||||
goto err_disable_clk;
|
goto err_put_pm;
|
||||||
}
|
}
|
||||||
|
|
||||||
writel_relaxed(DMA_MST_CTRL_IS_DONE, ctrl->regs + DMA_MST_CTRL);
|
writel_relaxed(DMA_MST_CTRL_IS_DONE, ctrl->regs + DMA_MST_CTRL);
|
||||||
|
|
||||||
err = tegra_nand_chips_init(ctrl->dev, ctrl);
|
err = tegra_nand_chips_init(ctrl->dev, ctrl);
|
||||||
if (err)
|
if (err)
|
||||||
goto err_disable_clk;
|
goto err_put_pm;
|
||||||
|
|
||||||
platform_set_drvdata(pdev, ctrl);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_disable_clk:
|
err_put_pm:
|
||||||
clk_disable_unprepare(ctrl->clk);
|
pm_runtime_put_sync_suspend(ctrl->dev);
|
||||||
|
pm_runtime_force_suspend(ctrl->dev);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1219,11 +1231,40 @@ static int tegra_nand_remove(struct platform_device *pdev)
|
|||||||
|
|
||||||
nand_cleanup(chip);
|
nand_cleanup(chip);
|
||||||
|
|
||||||
|
pm_runtime_put_sync_suspend(ctrl->dev);
|
||||||
|
pm_runtime_force_suspend(ctrl->dev);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int __maybe_unused tegra_nand_runtime_resume(struct device *dev)
|
||||||
|
{
|
||||||
|
struct tegra_nand_controller *ctrl = dev_get_drvdata(dev);
|
||||||
|
int err;
|
||||||
|
|
||||||
|
err = clk_prepare_enable(ctrl->clk);
|
||||||
|
if (err) {
|
||||||
|
dev_err(dev, "Failed to enable clock: %d\n", err);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int __maybe_unused tegra_nand_runtime_suspend(struct device *dev)
|
||||||
|
{
|
||||||
|
struct tegra_nand_controller *ctrl = dev_get_drvdata(dev);
|
||||||
|
|
||||||
clk_disable_unprepare(ctrl->clk);
|
clk_disable_unprepare(ctrl->clk);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct dev_pm_ops tegra_nand_pm = {
|
||||||
|
SET_RUNTIME_PM_OPS(tegra_nand_runtime_suspend, tegra_nand_runtime_resume,
|
||||||
|
NULL)
|
||||||
|
};
|
||||||
|
|
||||||
static const struct of_device_id tegra_nand_of_match[] = {
|
static const struct of_device_id tegra_nand_of_match[] = {
|
||||||
{ .compatible = "nvidia,tegra20-nand" },
|
{ .compatible = "nvidia,tegra20-nand" },
|
||||||
{ /* sentinel */ }
|
{ /* sentinel */ }
|
||||||
@ -1234,6 +1275,7 @@ static struct platform_driver tegra_nand_driver = {
|
|||||||
.driver = {
|
.driver = {
|
||||||
.name = "tegra-nand",
|
.name = "tegra-nand",
|
||||||
.of_match_table = tegra_nand_of_match,
|
.of_match_table = tegra_nand_of_match,
|
||||||
|
.pm = &tegra_nand_pm,
|
||||||
},
|
},
|
||||||
.probe = tegra_nand_probe,
|
.probe = tegra_nand_probe,
|
||||||
.remove = tegra_nand_remove,
|
.remove = tegra_nand_remove,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user