diff options
| author | Charles Keepax <ckeepax@opensource.wolfsonmicro.com> | 2015-04-07 07:55:10 -0400 |
|---|---|---|
| committer | Mark Brown <broonie@kernel.org> | 2015-04-08 14:25:40 -0400 |
| commit | 1a60667fc81fdab3733a1d41480da3a5d0ccecea (patch) | |
| tree | d146cc70eb718062b3f7546a896c21bbaf9e4b1d | |
| parent | 5631f18763f1b0989cec7cbf8f3badcb74dfe469 (diff) | |
ASoC: wm8804: Enable runtime PM
Currently both the oscillator and the PLL are powered up in
set_bias_level. This can be problematic when using output clocks from
the wm8804 for other devices. The snd_soc_codec_set_pll API defines that
a clock should be available once the call returns, however, with all the
clocking controlled in set_bias_level this is not currently the case.
This patch enables pm_runtime for the wm8804, enabling both the
regulators and the oscillator when the chip resumes, and enabling the
PLL in the snd_soc_codec_set_pll call. Naturally the enabling the PLL
will also cause the chip to resume.
Signed-off-by: Charles Keepax <ckeepax@opensource.wolfsonmicro.com>
Signed-off-by: Mark Brown <broonie@kernel.org>
| -rw-r--r-- | sound/soc/codecs/wm8804-i2c.c | 1 | ||||
| -rw-r--r-- | sound/soc/codecs/wm8804-spi.c | 1 | ||||
| -rw-r--r-- | sound/soc/codecs/wm8804.c | 109 | ||||
| -rw-r--r-- | sound/soc/codecs/wm8804.h | 1 |
4 files changed, 62 insertions, 50 deletions
diff --git a/sound/soc/codecs/wm8804-i2c.c b/sound/soc/codecs/wm8804-i2c.c index 5bd4af2b4059..6596f5f3a0c3 100644 --- a/sound/soc/codecs/wm8804-i2c.c +++ b/sound/soc/codecs/wm8804-i2c.c | |||
| @@ -50,6 +50,7 @@ static struct i2c_driver wm8804_i2c_driver = { | |||
| 50 | .driver = { | 50 | .driver = { |
| 51 | .name = "wm8804", | 51 | .name = "wm8804", |
| 52 | .owner = THIS_MODULE, | 52 | .owner = THIS_MODULE, |
| 53 | .pm = &wm8804_pm, | ||
| 53 | .of_match_table = wm8804_of_match, | 54 | .of_match_table = wm8804_of_match, |
| 54 | }, | 55 | }, |
| 55 | .probe = wm8804_i2c_probe, | 56 | .probe = wm8804_i2c_probe, |
diff --git a/sound/soc/codecs/wm8804-spi.c b/sound/soc/codecs/wm8804-spi.c index 287e11e90794..407a3cf391e5 100644 --- a/sound/soc/codecs/wm8804-spi.c +++ b/sound/soc/codecs/wm8804-spi.c | |||
| @@ -43,6 +43,7 @@ static struct spi_driver wm8804_spi_driver = { | |||
| 43 | .driver = { | 43 | .driver = { |
| 44 | .name = "wm8804", | 44 | .name = "wm8804", |
| 45 | .owner = THIS_MODULE, | 45 | .owner = THIS_MODULE, |
| 46 | .pm = &wm8804_pm, | ||
| 46 | .of_match_table = wm8804_of_match, | 47 | .of_match_table = wm8804_of_match, |
| 47 | }, | 48 | }, |
| 48 | .probe = wm8804_spi_probe, | 49 | .probe = wm8804_spi_probe, |
diff --git a/sound/soc/codecs/wm8804.c b/sound/soc/codecs/wm8804.c index cff34be61f88..1e403f67cf16 100644 --- a/sound/soc/codecs/wm8804.c +++ b/sound/soc/codecs/wm8804.c | |||
| @@ -16,6 +16,7 @@ | |||
| 16 | #include <linux/gpio/consumer.h> | 16 | #include <linux/gpio/consumer.h> |
| 17 | #include <linux/delay.h> | 17 | #include <linux/delay.h> |
| 18 | #include <linux/pm.h> | 18 | #include <linux/pm.h> |
| 19 | #include <linux/pm_runtime.h> | ||
| 19 | #include <linux/of_device.h> | 20 | #include <linux/of_device.h> |
| 20 | #include <linux/regulator/consumer.h> | 21 | #include <linux/regulator/consumer.h> |
| 21 | #include <linux/slab.h> | 22 | #include <linux/slab.h> |
| @@ -59,6 +60,7 @@ static const struct reg_default wm8804_reg_defaults[] = { | |||
| 59 | }; | 60 | }; |
| 60 | 61 | ||
| 61 | struct wm8804_priv { | 62 | struct wm8804_priv { |
| 63 | struct device *dev; | ||
| 62 | struct regmap *regmap; | 64 | struct regmap *regmap; |
| 63 | struct regulator_bulk_data supplies[WM8804_NUM_SUPPLIES]; | 65 | struct regulator_bulk_data supplies[WM8804_NUM_SUPPLIES]; |
| 64 | struct notifier_block disable_nb[WM8804_NUM_SUPPLIES]; | 66 | struct notifier_block disable_nb[WM8804_NUM_SUPPLIES]; |
| @@ -403,19 +405,19 @@ static int wm8804_set_pll(struct snd_soc_dai *dai, int pll_id, | |||
| 403 | int source, unsigned int freq_in, | 405 | int source, unsigned int freq_in, |
| 404 | unsigned int freq_out) | 406 | unsigned int freq_out) |
| 405 | { | 407 | { |
| 406 | struct snd_soc_codec *codec; | 408 | struct snd_soc_codec *codec = dai->codec; |
| 409 | struct wm8804_priv *wm8804 = snd_soc_codec_get_drvdata(codec); | ||
| 410 | bool change; | ||
| 407 | 411 | ||
| 408 | codec = dai->codec; | ||
| 409 | if (!freq_in || !freq_out) { | 412 | if (!freq_in || !freq_out) { |
| 410 | /* disable the PLL */ | 413 | /* disable the PLL */ |
| 411 | snd_soc_update_bits(codec, WM8804_PWRDN, 0x1, 0x1); | 414 | regmap_update_bits_check(wm8804->regmap, WM8804_PWRDN, |
| 412 | return 0; | 415 | 0x1, 0x1, &change); |
| 416 | if (change) | ||
| 417 | pm_runtime_put(wm8804->dev); | ||
| 413 | } else { | 418 | } else { |
| 414 | int ret; | 419 | int ret; |
| 415 | struct pll_div pll_div; | 420 | struct pll_div pll_div; |
| 416 | struct wm8804_priv *wm8804; | ||
| 417 | |||
| 418 | wm8804 = snd_soc_codec_get_drvdata(codec); | ||
| 419 | 421 | ||
| 420 | ret = pll_factors(&pll_div, freq_out, freq_in, | 422 | ret = pll_factors(&pll_div, freq_out, freq_in, |
| 421 | wm8804->mclk_div); | 423 | wm8804->mclk_div); |
| @@ -423,7 +425,10 @@ static int wm8804_set_pll(struct snd_soc_dai *dai, int pll_id, | |||
| 423 | return ret; | 425 | return ret; |
| 424 | 426 | ||
| 425 | /* power down the PLL before reprogramming it */ | 427 | /* power down the PLL before reprogramming it */ |
| 426 | snd_soc_update_bits(codec, WM8804_PWRDN, 0x1, 0x1); | 428 | regmap_update_bits_check(wm8804->regmap, WM8804_PWRDN, |
| 429 | 0x1, 0x1, &change); | ||
| 430 | if (!change) | ||
| 431 | pm_runtime_get_sync(wm8804->dev); | ||
| 427 | 432 | ||
| 428 | /* set PLLN and PRESCALE */ | 433 | /* set PLLN and PRESCALE */ |
| 429 | snd_soc_update_bits(codec, WM8804_PLL4, 0xf | 0x10, | 434 | snd_soc_update_bits(codec, WM8804_PLL4, 0xf | 0x10, |
| @@ -501,47 +506,6 @@ static int wm8804_set_clkdiv(struct snd_soc_dai *dai, | |||
| 501 | return 0; | 506 | return 0; |
| 502 | } | 507 | } |
| 503 | 508 | ||
| 504 | static int wm8804_set_bias_level(struct snd_soc_codec *codec, | ||
| 505 | enum snd_soc_bias_level level) | ||
| 506 | { | ||
| 507 | int ret; | ||
| 508 | struct wm8804_priv *wm8804; | ||
| 509 | |||
| 510 | wm8804 = snd_soc_codec_get_drvdata(codec); | ||
| 511 | switch (level) { | ||
| 512 | case SND_SOC_BIAS_ON: | ||
| 513 | break; | ||
| 514 | case SND_SOC_BIAS_PREPARE: | ||
| 515 | /* power up the OSC and the PLL */ | ||
| 516 | snd_soc_update_bits(codec, WM8804_PWRDN, 0x9, 0); | ||
| 517 | break; | ||
| 518 | case SND_SOC_BIAS_STANDBY: | ||
| 519 | if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) { | ||
| 520 | ret = regulator_bulk_enable(ARRAY_SIZE(wm8804->supplies), | ||
| 521 | wm8804->supplies); | ||
| 522 | if (ret) { | ||
| 523 | dev_err(codec->dev, | ||
| 524 | "Failed to enable supplies: %d\n", | ||
| 525 | ret); | ||
| 526 | return ret; | ||
| 527 | } | ||
| 528 | regcache_sync(wm8804->regmap); | ||
| 529 | } | ||
| 530 | /* power down the OSC and the PLL */ | ||
| 531 | snd_soc_update_bits(codec, WM8804_PWRDN, 0x9, 0x9); | ||
| 532 | break; | ||
| 533 | case SND_SOC_BIAS_OFF: | ||
| 534 | /* power down the OSC and the PLL */ | ||
| 535 | snd_soc_update_bits(codec, WM8804_PWRDN, 0x9, 0x9); | ||
| 536 | regulator_bulk_disable(ARRAY_SIZE(wm8804->supplies), | ||
| 537 | wm8804->supplies); | ||
| 538 | break; | ||
| 539 | } | ||
| 540 | |||
| 541 | codec->dapm.bias_level = level; | ||
| 542 | return 0; | ||
| 543 | } | ||
| 544 | |||
| 545 | static const struct snd_soc_dai_ops wm8804_dai_ops = { | 509 | static const struct snd_soc_dai_ops wm8804_dai_ops = { |
| 546 | .hw_params = wm8804_hw_params, | 510 | .hw_params = wm8804_hw_params, |
| 547 | .set_fmt = wm8804_set_fmt, | 511 | .set_fmt = wm8804_set_fmt, |
| @@ -579,7 +543,6 @@ static struct snd_soc_dai_driver wm8804_dai = { | |||
| 579 | }; | 543 | }; |
| 580 | 544 | ||
| 581 | static const struct snd_soc_codec_driver soc_codec_dev_wm8804 = { | 545 | static const struct snd_soc_codec_driver soc_codec_dev_wm8804 = { |
| 582 | .set_bias_level = wm8804_set_bias_level, | ||
| 583 | .idle_bias_off = true, | 546 | .idle_bias_off = true, |
| 584 | 547 | ||
| 585 | .dapm_widgets = wm8804_dapm_widgets, | 548 | .dapm_widgets = wm8804_dapm_widgets, |
| @@ -613,6 +576,7 @@ int wm8804_probe(struct device *dev, struct regmap *regmap) | |||
| 613 | 576 | ||
| 614 | dev_set_drvdata(dev, wm8804); | 577 | dev_set_drvdata(dev, wm8804); |
| 615 | 578 | ||
| 579 | wm8804->dev = dev; | ||
| 616 | wm8804->regmap = regmap; | 580 | wm8804->regmap = regmap; |
| 617 | 581 | ||
| 618 | wm8804->reset = devm_gpiod_get_optional(dev, "wlf,reset", | 582 | wm8804->reset = devm_gpiod_get_optional(dev, "wlf,reset", |
| @@ -703,6 +667,10 @@ int wm8804_probe(struct device *dev, struct regmap *regmap) | |||
| 703 | goto err_reg_enable; | 667 | goto err_reg_enable; |
| 704 | } | 668 | } |
| 705 | 669 | ||
| 670 | pm_runtime_set_active(dev); | ||
| 671 | pm_runtime_enable(dev); | ||
| 672 | pm_runtime_idle(dev); | ||
| 673 | |||
| 706 | return 0; | 674 | return 0; |
| 707 | 675 | ||
| 708 | err_reg_enable: | 676 | err_reg_enable: |
| @@ -713,10 +681,51 @@ EXPORT_SYMBOL_GPL(wm8804_probe); | |||
| 713 | 681 | ||
| 714 | void wm8804_remove(struct device *dev) | 682 | void wm8804_remove(struct device *dev) |
| 715 | { | 683 | { |
| 684 | pm_runtime_disable(dev); | ||
| 716 | snd_soc_unregister_codec(dev); | 685 | snd_soc_unregister_codec(dev); |
| 717 | } | 686 | } |
| 718 | EXPORT_SYMBOL_GPL(wm8804_remove); | 687 | EXPORT_SYMBOL_GPL(wm8804_remove); |
| 719 | 688 | ||
| 689 | #if IS_ENABLED(CONFIG_PM) | ||
| 690 | static int wm8804_runtime_resume(struct device *dev) | ||
| 691 | { | ||
| 692 | struct wm8804_priv *wm8804 = dev_get_drvdata(dev); | ||
| 693 | int ret; | ||
| 694 | |||
| 695 | ret = regulator_bulk_enable(ARRAY_SIZE(wm8804->supplies), | ||
| 696 | wm8804->supplies); | ||
| 697 | if (ret) { | ||
| 698 | dev_err(wm8804->dev, "Failed to enable supplies: %d\n", ret); | ||
| 699 | return ret; | ||
| 700 | } | ||
| 701 | |||
| 702 | regcache_sync(wm8804->regmap); | ||
| 703 | |||
| 704 | /* Power up OSCCLK */ | ||
| 705 | regmap_update_bits(wm8804->regmap, WM8804_PWRDN, 0x8, 0x0); | ||
| 706 | |||
| 707 | return 0; | ||
| 708 | } | ||
| 709 | |||
| 710 | static int wm8804_runtime_suspend(struct device *dev) | ||
| 711 | { | ||
| 712 | struct wm8804_priv *wm8804 = dev_get_drvdata(dev); | ||
| 713 | |||
| 714 | /* Power down OSCCLK */ | ||
| 715 | regmap_update_bits(wm8804->regmap, WM8804_PWRDN, 0x8, 0x8); | ||
| 716 | |||
| 717 | regulator_bulk_disable(ARRAY_SIZE(wm8804->supplies), | ||
| 718 | wm8804->supplies); | ||
| 719 | |||
| 720 | return 0; | ||
| 721 | } | ||
| 722 | #endif | ||
| 723 | |||
| 724 | const struct dev_pm_ops wm8804_pm = { | ||
| 725 | SET_RUNTIME_PM_OPS(wm8804_runtime_suspend, wm8804_runtime_resume, NULL) | ||
| 726 | }; | ||
| 727 | EXPORT_SYMBOL_GPL(wm8804_pm); | ||
| 728 | |||
| 720 | MODULE_DESCRIPTION("ASoC WM8804 driver"); | 729 | MODULE_DESCRIPTION("ASoC WM8804 driver"); |
| 721 | MODULE_AUTHOR("Dimitris Papastamos <dp@opensource.wolfsonmicro.com>"); | 730 | MODULE_AUTHOR("Dimitris Papastamos <dp@opensource.wolfsonmicro.com>"); |
| 722 | MODULE_LICENSE("GPL"); | 731 | MODULE_LICENSE("GPL"); |
diff --git a/sound/soc/codecs/wm8804.h b/sound/soc/codecs/wm8804.h index a39a2563dc67..aa72fa66c932 100644 --- a/sound/soc/codecs/wm8804.h +++ b/sound/soc/codecs/wm8804.h | |||
| @@ -65,6 +65,7 @@ | |||
| 65 | #define WM8804_MCLKDIV_128FS 1 | 65 | #define WM8804_MCLKDIV_128FS 1 |
| 66 | 66 | ||
| 67 | extern const struct regmap_config wm8804_regmap_config; | 67 | extern const struct regmap_config wm8804_regmap_config; |
| 68 | extern const struct dev_pm_ops wm8804_pm; | ||
| 68 | 69 | ||
| 69 | int wm8804_probe(struct device *dev, struct regmap *regmap); | 70 | int wm8804_probe(struct device *dev, struct regmap *regmap); |
| 70 | void wm8804_remove(struct device *dev); | 71 | void wm8804_remove(struct device *dev); |
