aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc')
-rw-r--r--sound/soc/tegra/harmony.c11
-rw-r--r--sound/soc/tegra/tegra_asoc_utils.c91
-rw-r--r--sound/soc/tegra/tegra_asoc_utils.h20
3 files changed, 68 insertions, 54 deletions
diff --git a/sound/soc/tegra/harmony.c b/sound/soc/tegra/harmony.c
index 76793a93c133..d1faa63af034 100644
--- a/sound/soc/tegra/harmony.c
+++ b/sound/soc/tegra/harmony.c
@@ -50,6 +50,7 @@
50#define DRV_NAME "tegra-snd-harmony" 50#define DRV_NAME "tegra-snd-harmony"
51 51
52struct tegra_harmony { 52struct tegra_harmony {
53 struct tegra_asoc_utils_data util_data;
53 struct harmony_audio_platform_data *pdata; 54 struct harmony_audio_platform_data *pdata;
54 int gpio_spkr_en_requested; 55 int gpio_spkr_en_requested;
55}; 56};
@@ -62,6 +63,7 @@ static int harmony_asoc_hw_params(struct snd_pcm_substream *substream,
62 struct snd_soc_dai *cpu_dai = rtd->cpu_dai; 63 struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
63 struct snd_soc_codec *codec = rtd->codec; 64 struct snd_soc_codec *codec = rtd->codec;
64 struct snd_soc_card *card = codec->card; 65 struct snd_soc_card *card = codec->card;
66 struct tegra_harmony *harmony = snd_soc_card_get_drvdata(card);
65 int srate, mclk, mclk_change; 67 int srate, mclk, mclk_change;
66 int err; 68 int err;
67 69
@@ -80,7 +82,8 @@ static int harmony_asoc_hw_params(struct snd_pcm_substream *substream,
80 while (mclk < 6000000) 82 while (mclk < 6000000)
81 mclk *= 2; 83 mclk *= 2;
82 84
83 err = tegra_asoc_utils_set_rate(srate, mclk, &mclk_change); 85 err = tegra_asoc_utils_set_rate(&harmony->util_data, srate, mclk,
86 &mclk_change);
84 if (err < 0) { 87 if (err < 0) {
85 dev_err(card->dev, "Can't configure clocks\n"); 88 dev_err(card->dev, "Can't configure clocks\n");
86 return err; 89 return err;
@@ -226,7 +229,7 @@ static __devinit int tegra_snd_harmony_probe(struct platform_device *pdev)
226 229
227 harmony->pdata = pdata; 230 harmony->pdata = pdata;
228 231
229 ret = tegra_asoc_utils_init(); 232 ret = tegra_asoc_utils_init(&harmony->util_data, &pdev->dev);
230 if (ret) 233 if (ret)
231 goto err_free_harmony; 234 goto err_free_harmony;
232 235
@@ -247,7 +250,7 @@ err_clear_drvdata:
247 snd_soc_card_set_drvdata(card, NULL); 250 snd_soc_card_set_drvdata(card, NULL);
248 platform_set_drvdata(pdev, NULL); 251 platform_set_drvdata(pdev, NULL);
249 card->dev = NULL; 252 card->dev = NULL;
250 tegra_asoc_utils_fini(); 253 tegra_asoc_utils_fini(&harmony->util_data);
251err_free_harmony: 254err_free_harmony:
252 kfree(harmony); 255 kfree(harmony);
253 return ret; 256 return ret;
@@ -265,7 +268,7 @@ static int __devexit tegra_snd_harmony_remove(struct platform_device *pdev)
265 platform_set_drvdata(pdev, NULL); 268 platform_set_drvdata(pdev, NULL);
266 card->dev = NULL; 269 card->dev = NULL;
267 270
268 tegra_asoc_utils_fini(); 271 tegra_asoc_utils_fini(&harmony->util_data);
269 272
270 if (harmony->gpio_spkr_en_requested) 273 if (harmony->gpio_spkr_en_requested)
271 gpio_free(pdata->gpio_spkr_en); 274 gpio_free(pdata->gpio_spkr_en);
diff --git a/sound/soc/tegra/tegra_asoc_utils.c b/sound/soc/tegra/tegra_asoc_utils.c
index cfe2ea890dc0..cb4fc13c7d22 100644
--- a/sound/soc/tegra/tegra_asoc_utils.c
+++ b/sound/soc/tegra/tegra_asoc_utils.c
@@ -21,20 +21,14 @@
21 */ 21 */
22 22
23#include <linux/clk.h> 23#include <linux/clk.h>
24#include <linux/device.h>
24#include <linux/err.h> 25#include <linux/err.h>
25#include <linux/kernel.h> 26#include <linux/kernel.h>
26 27
27#include "tegra_asoc_utils.h" 28#include "tegra_asoc_utils.h"
28 29
29#define PREFIX "ASoC Tegra: " 30int tegra_asoc_utils_set_rate(struct tegra_asoc_utils_data *data, int srate,
30 31 int mclk, int *mclk_change)
31static struct clk *clk_pll_a;
32static struct clk *clk_pll_a_out0;
33static struct clk *clk_cdev1;
34
35static int set_baseclock, set_mclk;
36
37int tegra_asoc_utils_set_rate(int srate, int mclk, int *mclk_change)
38{ 32{
39 int new_baseclock; 33 int new_baseclock;
40 int err; 34 int err;
@@ -58,95 +52,98 @@ int tegra_asoc_utils_set_rate(int srate, int mclk, int *mclk_change)
58 return -EINVAL; 52 return -EINVAL;
59 } 53 }
60 54
61 *mclk_change = ((new_baseclock != set_baseclock) || 55 *mclk_change = ((new_baseclock != data->set_baseclock) ||
62 (mclk != set_mclk)); 56 (mclk != data->set_mclk));
63 if (!*mclk_change) 57 if (!*mclk_change)
64 return 0; 58 return 0;
65 59
66 set_baseclock = 0; 60 data->set_baseclock = 0;
67 set_mclk = 0; 61 data->set_mclk = 0;
68 62
69 clk_disable(clk_cdev1); 63 clk_disable(data->clk_cdev1);
70 clk_disable(clk_pll_a_out0); 64 clk_disable(data->clk_pll_a_out0);
71 clk_disable(clk_pll_a); 65 clk_disable(data->clk_pll_a);
72 66
73 err = clk_set_rate(clk_pll_a, new_baseclock); 67 err = clk_set_rate(data->clk_pll_a, new_baseclock);
74 if (err) { 68 if (err) {
75 pr_err(PREFIX "Can't set pll_a rate: %d\n", err); 69 dev_err(data->dev, "Can't set pll_a rate: %d\n", err);
76 return err; 70 return err;
77 } 71 }
78 72
79 err = clk_set_rate(clk_pll_a_out0, mclk); 73 err = clk_set_rate(data->clk_pll_a_out0, mclk);
80 if (err) { 74 if (err) {
81 pr_err(PREFIX "Can't set pll_a_out0 rate: %d\n", err); 75 dev_err(data->dev, "Can't set pll_a_out0 rate: %d\n", err);
82 return err; 76 return err;
83 } 77 }
84 78
85 /* Don't set cdev1 rate; its locked to pll_a_out0 */ 79 /* Don't set cdev1 rate; its locked to pll_a_out0 */
86 80
87 err = clk_enable(clk_pll_a); 81 err = clk_enable(data->clk_pll_a);
88 if (err) { 82 if (err) {
89 pr_err(PREFIX "Can't enable pll_a: %d\n", err); 83 dev_err(data->dev, "Can't enable pll_a: %d\n", err);
90 return err; 84 return err;
91 } 85 }
92 86
93 err = clk_enable(clk_pll_a_out0); 87 err = clk_enable(data->clk_pll_a_out0);
94 if (err) { 88 if (err) {
95 pr_err(PREFIX "Can't enable pll_a_out0: %d\n", err); 89 dev_err(data->dev, "Can't enable pll_a_out0: %d\n", err);
96 return err; 90 return err;
97 } 91 }
98 92
99 err = clk_enable(clk_cdev1); 93 err = clk_enable(data->clk_cdev1);
100 if (err) { 94 if (err) {
101 pr_err(PREFIX "Can't enable cdev1: %d\n", err); 95 dev_err(data->dev, "Can't enable cdev1: %d\n", err);
102 return err; 96 return err;
103 } 97 }
104 98
105 set_baseclock = new_baseclock; 99 data->set_baseclock = new_baseclock;
106 set_mclk = mclk; 100 data->set_mclk = mclk;
107 101
108 return 0; 102 return 0;
109} 103}
110 104
111int tegra_asoc_utils_init(void) 105int tegra_asoc_utils_init(struct tegra_asoc_utils_data *data,
106 struct device *dev)
112{ 107{
113 int ret; 108 int ret;
114 109
115 clk_pll_a = clk_get_sys(NULL, "pll_a"); 110 data->dev = dev;
116 if (IS_ERR(clk_pll_a)) { 111
117 pr_err(PREFIX "Can't retrieve clk pll_a\n"); 112 data->clk_pll_a = clk_get_sys(NULL, "pll_a");
118 ret = PTR_ERR(clk_pll_a); 113 if (IS_ERR(data->clk_pll_a)) {
114 dev_err(data->dev, "Can't retrieve clk pll_a\n");
115 ret = PTR_ERR(data->clk_pll_a);
119 goto err; 116 goto err;
120 } 117 }
121 118
122 clk_pll_a_out0 = clk_get_sys(NULL, "pll_a_out0"); 119 data->clk_pll_a_out0 = clk_get_sys(NULL, "pll_a_out0");
123 if (IS_ERR(clk_pll_a_out0)) { 120 if (IS_ERR(data->clk_pll_a_out0)) {
124 pr_err(PREFIX "Can't retrieve clk pll_a_out0\n"); 121 dev_err(data->dev, "Can't retrieve clk pll_a_out0\n");
125 ret = PTR_ERR(clk_pll_a_out0); 122 ret = PTR_ERR(data->clk_pll_a_out0);
126 goto err_put_pll_a; 123 goto err_put_pll_a;
127 } 124 }
128 125
129 clk_cdev1 = clk_get_sys(NULL, "cdev1"); 126 data->clk_cdev1 = clk_get_sys(NULL, "cdev1");
130 if (IS_ERR(clk_cdev1)) { 127 if (IS_ERR(data->clk_cdev1)) {
131 pr_err(PREFIX "Can't retrieve clk cdev1\n"); 128 dev_err(data->dev, "Can't retrieve clk cdev1\n");
132 ret = PTR_ERR(clk_cdev1); 129 ret = PTR_ERR(data->clk_cdev1);
133 goto err_put_pll_a_out0; 130 goto err_put_pll_a_out0;
134 } 131 }
135 132
136 return 0; 133 return 0;
137 134
138err_put_pll_a_out0: 135err_put_pll_a_out0:
139 clk_put(clk_pll_a_out0); 136 clk_put(data->clk_pll_a_out0);
140err_put_pll_a: 137err_put_pll_a:
141 clk_put(clk_pll_a); 138 clk_put(data->clk_pll_a);
142err: 139err:
143 return ret; 140 return ret;
144} 141}
145 142
146void tegra_asoc_utils_fini(void) 143void tegra_asoc_utils_fini(struct tegra_asoc_utils_data *data)
147{ 144{
148 clk_put(clk_cdev1); 145 clk_put(data->clk_cdev1);
149 clk_put(clk_pll_a_out0); 146 clk_put(data->clk_pll_a_out0);
150 clk_put(clk_pll_a); 147 clk_put(data->clk_pll_a);
151} 148}
152 149
diff --git a/sound/soc/tegra/tegra_asoc_utils.h b/sound/soc/tegra/tegra_asoc_utils.h
index 855f8f6e44ca..bbba7afdfc2c 100644
--- a/sound/soc/tegra/tegra_asoc_utils.h
+++ b/sound/soc/tegra/tegra_asoc_utils.h
@@ -23,9 +23,23 @@
23#ifndef __TEGRA_ASOC_UTILS_H__ 23#ifndef __TEGRA_ASOC_UTILS_H__
24#define __TEGRA_ASOC_UTILS_H_ 24#define __TEGRA_ASOC_UTILS_H_
25 25
26int tegra_asoc_utils_set_rate(int srate, int mclk_rate, int *mclk_change); 26struct clk;
27int tegra_asoc_utils_init(void); 27struct device;
28void tegra_asoc_utils_fini(void); 28
29struct tegra_asoc_utils_data {
30 struct device *dev;
31 struct clk *clk_pll_a;
32 struct clk *clk_pll_a_out0;
33 struct clk *clk_cdev1;
34 int set_baseclock;
35 int set_mclk;
36};
37
38int tegra_asoc_utils_set_rate(struct tegra_asoc_utils_data *data, int srate,
39 int mclk, int *mclk_change);
40int tegra_asoc_utils_init(struct tegra_asoc_utils_data *data,
41 struct device *dev);
42void tegra_asoc_utils_fini(struct tegra_asoc_utils_data *data);
29 43
30#endif 44#endif
31 45