Commit 8fda5c1f authored by Dmitry Osipenko's avatar Dmitry Osipenko Committed by MyungJoo Ham

PM / devfreq: tegra: Clean up driver's probe / remove

Reset hardware, disable ACTMON clock, release OPP's and handle all
possible error cases correctly, maintaining the correct tear down
order. Also use devm_platform_ioremap_resource() which is now available
in the kernel.
Reviewed-by: default avatarChanwoo Choi <cw00.choi@samsung.com>
Signed-off-by: default avatarDmitry Osipenko <digetx@gmail.com>
Acked-by: default avatarThierry Reding <treding@nvidia.com>
Signed-off-by: default avatarMyungJoo Ham <myungjoo.ham@samsung.com>
parent 7514dd05
...@@ -598,7 +598,6 @@ static int tegra_devfreq_probe(struct platform_device *pdev) ...@@ -598,7 +598,6 @@ static int tegra_devfreq_probe(struct platform_device *pdev)
{ {
struct tegra_devfreq *tegra; struct tegra_devfreq *tegra;
struct tegra_devfreq_device *dev; struct tegra_devfreq_device *dev;
struct resource *res;
unsigned int i; unsigned int i;
unsigned long rate; unsigned long rate;
int err; int err;
...@@ -607,9 +606,7 @@ static int tegra_devfreq_probe(struct platform_device *pdev) ...@@ -607,9 +606,7 @@ static int tegra_devfreq_probe(struct platform_device *pdev)
if (!tegra) if (!tegra)
return -ENOMEM; return -ENOMEM;
res = platform_get_resource(pdev, IORESOURCE_MEM, 0); tegra->regs = devm_platform_ioremap_resource(pdev, 0);
tegra->regs = devm_ioremap_resource(&pdev->dev, res);
if (IS_ERR(tegra->regs)) if (IS_ERR(tegra->regs))
return PTR_ERR(tegra->regs); return PTR_ERR(tegra->regs);
...@@ -631,11 +628,10 @@ static int tegra_devfreq_probe(struct platform_device *pdev) ...@@ -631,11 +628,10 @@ static int tegra_devfreq_probe(struct platform_device *pdev)
return PTR_ERR(tegra->emc_clock); return PTR_ERR(tegra->emc_clock);
} }
tegra->rate_change_nb.notifier_call = tegra_actmon_rate_notify_cb; tegra->irq = platform_get_irq(pdev, 0);
err = clk_notifier_register(tegra->emc_clock, &tegra->rate_change_nb); if (tegra->irq < 0) {
if (err) { err = tegra->irq;
dev_err(&pdev->dev, dev_err(&pdev->dev, "Failed to get IRQ: %d\n", err);
"Failed to register rate change notifier\n");
return err; return err;
} }
...@@ -666,54 +662,69 @@ static int tegra_devfreq_probe(struct platform_device *pdev) ...@@ -666,54 +662,69 @@ static int tegra_devfreq_probe(struct platform_device *pdev)
for (rate = 0; rate <= tegra->max_freq * KHZ; rate++) { for (rate = 0; rate <= tegra->max_freq * KHZ; rate++) {
rate = clk_round_rate(tegra->emc_clock, rate); rate = clk_round_rate(tegra->emc_clock, rate);
dev_pm_opp_add(&pdev->dev, rate, 0);
}
tegra->irq = platform_get_irq(pdev, 0); err = dev_pm_opp_add(&pdev->dev, rate, 0);
if (tegra->irq < 0) { if (err) {
err = tegra->irq; dev_err(&pdev->dev, "Failed to add OPP: %d\n", err);
dev_err(&pdev->dev, "Failed to get IRQ: %d\n", err); goto remove_opps;
return err; }
} }
platform_set_drvdata(pdev, tegra); platform_set_drvdata(pdev, tegra);
err = devm_request_threaded_irq(&pdev->dev, tegra->irq, NULL, tegra->rate_change_nb.notifier_call = tegra_actmon_rate_notify_cb;
actmon_thread_isr, IRQF_ONESHOT, err = clk_notifier_register(tegra->emc_clock, &tegra->rate_change_nb);
"tegra-devfreq", tegra);
if (err) { if (err) {
dev_err(&pdev->dev, "Interrupt request failed\n"); dev_err(&pdev->dev,
return err; "Failed to register rate change notifier\n");
goto remove_opps;
} }
tegra_devfreq_profile.initial_freq = clk_get_rate(tegra->emc_clock); tegra_devfreq_profile.initial_freq = clk_get_rate(tegra->emc_clock);
tegra->devfreq = devm_devfreq_add_device(&pdev->dev, tegra->devfreq = devfreq_add_device(&pdev->dev,
&tegra_devfreq_profile, &tegra_devfreq_profile,
"tegra_actmon", "tegra_actmon",
NULL); NULL);
if (IS_ERR(tegra->devfreq)) {
err = PTR_ERR(tegra->devfreq);
goto unreg_notifier;
}
err = devm_request_threaded_irq(&pdev->dev, tegra->irq, NULL,
actmon_thread_isr, IRQF_ONESHOT,
"tegra-devfreq", tegra);
if (err) {
dev_err(&pdev->dev, "Interrupt request failed: %d\n", err);
goto remove_devfreq;
}
return 0; return 0;
remove_devfreq:
devfreq_remove_device(tegra->devfreq);
unreg_notifier:
clk_notifier_unregister(tegra->emc_clock, &tegra->rate_change_nb);
remove_opps:
dev_pm_opp_remove_all_dynamic(&pdev->dev);
reset_control_reset(tegra->reset);
clk_disable_unprepare(tegra->clock);
return err;
} }
static int tegra_devfreq_remove(struct platform_device *pdev) static int tegra_devfreq_remove(struct platform_device *pdev)
{ {
struct tegra_devfreq *tegra = platform_get_drvdata(pdev); struct tegra_devfreq *tegra = platform_get_drvdata(pdev);
int irq = platform_get_irq(pdev, 0);
u32 val;
unsigned int i;
for (i = 0; i < ARRAY_SIZE(actmon_device_configs); i++) {
val = device_readl(&tegra->devices[i], ACTMON_DEV_CTRL);
val &= ~ACTMON_DEV_CTRL_ENB;
device_writel(&tegra->devices[i], val, ACTMON_DEV_CTRL);
}
actmon_write_barrier(tegra);
devm_free_irq(&pdev->dev, irq, tegra); devfreq_remove_device(tegra->devfreq);
clk_notifier_unregister(tegra->emc_clock, &tegra->rate_change_nb); clk_notifier_unregister(tegra->emc_clock, &tegra->rate_change_nb);
dev_pm_opp_remove_all_dynamic(&pdev->dev);
reset_control_reset(tegra->reset);
clk_disable_unprepare(tegra->clock); clk_disable_unprepare(tegra->clock);
return 0; return 0;
......
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