diff options
author | Kuninori Morimoto <morimoto.kuninori@renesas.com> | 2009-11-30 06:24:48 -0500 |
---|---|---|
committer | Mark Brown <broonie@opensource.wolfsonmicro.com> | 2009-11-30 07:56:44 -0500 |
commit | 785d1c45ce11820d5838eb6399caa0ac98c836cf (patch) | |
tree | 48a6ef11e402d484804dcb1a95b2bb4b6b3836c7 | |
parent | d53bd80cb32d917e224b19925bb8f500941a3659 (diff) |
ASoC: sh: fsi: Add runtime PM support
This patch add support runtime PM.
Driver callbacks for Runtime PM are empty because
the device registers are always re-initialized after
pm_runtime_get_sync(). The Runtime PM functions replaces the
clock framework module stop bit handling in this driver.
Signed-off-by: Kuninori Morimoto <morimoto.kuninori@renesas.com>
Acked-by: Paul Mundt <lethal@linux-sh.org>
Acked-by: Liam Girdwood <lrg@slimlogic.co.uk>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
-rw-r--r-- | sound/soc/sh/fsi.c | 39 |
1 files changed, 25 insertions, 14 deletions
diff --git a/sound/soc/sh/fsi.c b/sound/soc/sh/fsi.c index e1a3d1a2b4c8..9c49c11c43ce 100644 --- a/sound/soc/sh/fsi.c +++ b/sound/soc/sh/fsi.c | |||
@@ -17,7 +17,7 @@ | |||
17 | #include <linux/platform_device.h> | 17 | #include <linux/platform_device.h> |
18 | #include <linux/delay.h> | 18 | #include <linux/delay.h> |
19 | #include <linux/list.h> | 19 | #include <linux/list.h> |
20 | #include <linux/clk.h> | 20 | #include <linux/pm_runtime.h> |
21 | #include <linux/io.h> | 21 | #include <linux/io.h> |
22 | #include <sound/core.h> | 22 | #include <sound/core.h> |
23 | #include <sound/pcm.h> | 23 | #include <sound/pcm.h> |
@@ -105,7 +105,6 @@ struct fsi_priv { | |||
105 | struct fsi_master { | 105 | struct fsi_master { |
106 | void __iomem *base; | 106 | void __iomem *base; |
107 | int irq; | 107 | int irq; |
108 | struct clk *clk; | ||
109 | struct fsi_priv fsia; | 108 | struct fsi_priv fsia; |
110 | struct fsi_priv fsib; | 109 | struct fsi_priv fsib; |
111 | struct sh_fsi_platform_info *info; | 110 | struct sh_fsi_platform_info *info; |
@@ -559,7 +558,7 @@ static int fsi_dai_startup(struct snd_pcm_substream *substream, | |||
559 | int is_master; | 558 | int is_master; |
560 | int ret = 0; | 559 | int ret = 0; |
561 | 560 | ||
562 | clk_enable(master->clk); | 561 | pm_runtime_get_sync(dai->dev); |
563 | 562 | ||
564 | /* CKG1 */ | 563 | /* CKG1 */ |
565 | data = is_play ? (1 << 0) : (1 << 4); | 564 | data = is_play ? (1 << 0) : (1 << 4); |
@@ -674,7 +673,7 @@ static void fsi_dai_shutdown(struct snd_pcm_substream *substream, | |||
674 | fsi_irq_disable(fsi, is_play); | 673 | fsi_irq_disable(fsi, is_play); |
675 | fsi_clk_ctrl(fsi, 0); | 674 | fsi_clk_ctrl(fsi, 0); |
676 | 675 | ||
677 | clk_disable(master->clk); | 676 | pm_runtime_put_sync(dai->dev); |
678 | } | 677 | } |
679 | 678 | ||
680 | static int fsi_dai_trigger(struct snd_pcm_substream *substream, int cmd, | 679 | static int fsi_dai_trigger(struct snd_pcm_substream *substream, int cmd, |
@@ -872,7 +871,6 @@ EXPORT_SYMBOL_GPL(fsi_soc_platform); | |||
872 | static int fsi_probe(struct platform_device *pdev) | 871 | static int fsi_probe(struct platform_device *pdev) |
873 | { | 872 | { |
874 | struct resource *res; | 873 | struct resource *res; |
875 | char clk_name[8]; | ||
876 | unsigned int irq; | 874 | unsigned int irq; |
877 | int ret; | 875 | int ret; |
878 | 876 | ||
@@ -903,14 +901,8 @@ static int fsi_probe(struct platform_device *pdev) | |||
903 | master->fsia.base = master->base; | 901 | master->fsia.base = master->base; |
904 | master->fsib.base = master->base + 0x40; | 902 | master->fsib.base = master->base + 0x40; |
905 | 903 | ||
906 | /* FSI is based on SPU mstp */ | 904 | pm_runtime_enable(&pdev->dev); |
907 | snprintf(clk_name, sizeof(clk_name), "spu%d", pdev->id); | 905 | pm_runtime_resume(&pdev->dev); |
908 | master->clk = clk_get(NULL, clk_name); | ||
909 | if (IS_ERR(master->clk)) { | ||
910 | dev_err(&pdev->dev, "cannot get %s mstp\n", clk_name); | ||
911 | ret = -EIO; | ||
912 | goto exit_iounmap; | ||
913 | } | ||
914 | 906 | ||
915 | fsi_soc_dai[0].dev = &pdev->dev; | 907 | fsi_soc_dai[0].dev = &pdev->dev; |
916 | fsi_soc_dai[1].dev = &pdev->dev; | 908 | fsi_soc_dai[1].dev = &pdev->dev; |
@@ -935,6 +927,7 @@ exit_free_irq: | |||
935 | free_irq(irq, master); | 927 | free_irq(irq, master); |
936 | exit_iounmap: | 928 | exit_iounmap: |
937 | iounmap(master->base); | 929 | iounmap(master->base); |
930 | pm_runtime_disable(&pdev->dev); | ||
938 | exit_kfree: | 931 | exit_kfree: |
939 | kfree(master); | 932 | kfree(master); |
940 | master = NULL; | 933 | master = NULL; |
@@ -947,7 +940,7 @@ static int fsi_remove(struct platform_device *pdev) | |||
947 | snd_soc_unregister_dais(fsi_soc_dai, ARRAY_SIZE(fsi_soc_dai)); | 940 | snd_soc_unregister_dais(fsi_soc_dai, ARRAY_SIZE(fsi_soc_dai)); |
948 | snd_soc_unregister_platform(&fsi_soc_platform); | 941 | snd_soc_unregister_platform(&fsi_soc_platform); |
949 | 942 | ||
950 | clk_put(master->clk); | 943 | pm_runtime_disable(&pdev->dev); |
951 | 944 | ||
952 | free_irq(master->irq, master); | 945 | free_irq(master->irq, master); |
953 | 946 | ||
@@ -957,9 +950,27 @@ static int fsi_remove(struct platform_device *pdev) | |||
957 | return 0; | 950 | return 0; |
958 | } | 951 | } |
959 | 952 | ||
953 | static int fsi_runtime_nop(struct device *dev) | ||
954 | { | ||
955 | /* Runtime PM callback shared between ->runtime_suspend() | ||
956 | * and ->runtime_resume(). Simply returns success. | ||
957 | * | ||
958 | * This driver re-initializes all registers after | ||
959 | * pm_runtime_get_sync() anyway so there is no need | ||
960 | * to save and restore registers here. | ||
961 | */ | ||
962 | return 0; | ||
963 | } | ||
964 | |||
965 | static struct dev_pm_ops fsi_pm_ops = { | ||
966 | .runtime_suspend = fsi_runtime_nop, | ||
967 | .runtime_resume = fsi_runtime_nop, | ||
968 | }; | ||
969 | |||
960 | static struct platform_driver fsi_driver = { | 970 | static struct platform_driver fsi_driver = { |
961 | .driver = { | 971 | .driver = { |
962 | .name = "sh_fsi", | 972 | .name = "sh_fsi", |
973 | .pm = &fsi_pm_ops, | ||
963 | }, | 974 | }, |
964 | .probe = fsi_probe, | 975 | .probe = fsi_probe, |
965 | .remove = fsi_remove, | 976 | .remove = fsi_remove, |