linux/sound/soc/codecs/wm8711.c
<<
>>
Prefs
   1/*
   2 * wm8711.c  --  WM8711 ALSA SoC Audio driver
   3 *
   4 * Copyright 2006 Wolfson Microelectronics
   5 *
   6 * Author: Mike Arthur <Mike.Arthur@wolfsonmicro.com>
   7 *
   8 * Based on wm8731.c by Richard Purdie
   9 *
  10 * This program is free software; you can redistribute it and/or modify
  11 * it under the terms of the GNU General Public License version 2 as
  12 * published by the Free Software Foundation.
  13 */
  14
  15#include <linux/module.h>
  16#include <linux/moduleparam.h>
  17#include <linux/init.h>
  18#include <linux/delay.h>
  19#include <linux/pm.h>
  20#include <linux/i2c.h>
  21#include <linux/regmap.h>
  22#include <linux/spi/spi.h>
  23#include <linux/slab.h>
  24#include <linux/of_device.h>
  25#include <sound/core.h>
  26#include <sound/pcm.h>
  27#include <sound/pcm_params.h>
  28#include <sound/soc.h>
  29#include <sound/tlv.h>
  30#include <sound/initval.h>
  31
  32#include "wm8711.h"
  33
  34/* codec private data */
  35struct wm8711_priv {
  36        struct regmap *regmap;
  37        unsigned int sysclk;
  38};
  39
  40/*
  41 * wm8711 register cache
  42 * We can't read the WM8711 register space when we are
  43 * using 2 wire for device control, so we cache them instead.
  44 * There is no point in caching the reset register
  45 */
  46static const struct reg_default wm8711_reg_defaults[] = {
  47        { 0, 0x0079 }, { 1, 0x0079 }, { 2, 0x000a }, { 3, 0x0008 },
  48        { 4, 0x009f }, { 5, 0x000a }, { 6, 0x0000 }, { 7, 0x0000 },
  49};
  50
  51static bool wm8711_volatile(struct device *dev, unsigned int reg)
  52{
  53        switch (reg) {
  54        case WM8711_RESET:
  55                return true;
  56        default:
  57                return false;
  58        }
  59}
  60
  61#define wm8711_reset(c) snd_soc_write(c, WM8711_RESET, 0)
  62
  63static const DECLARE_TLV_DB_SCALE(out_tlv, -12100, 100, 1);
  64
  65static const struct snd_kcontrol_new wm8711_snd_controls[] = {
  66
  67SOC_DOUBLE_R_TLV("Master Playback Volume", WM8711_LOUT1V, WM8711_ROUT1V,
  68                 0, 127, 0, out_tlv),
  69SOC_DOUBLE_R("Master Playback ZC Switch", WM8711_LOUT1V, WM8711_ROUT1V,
  70        7, 1, 0),
  71
  72};
  73
  74/* Output Mixer */
  75static const struct snd_kcontrol_new wm8711_output_mixer_controls[] = {
  76SOC_DAPM_SINGLE("Line Bypass Switch", WM8711_APANA, 3, 1, 0),
  77SOC_DAPM_SINGLE("HiFi Playback Switch", WM8711_APANA, 4, 1, 0),
  78};
  79
  80static const struct snd_soc_dapm_widget wm8711_dapm_widgets[] = {
  81SND_SOC_DAPM_MIXER("Output Mixer", WM8711_PWR, 4, 1,
  82        &wm8711_output_mixer_controls[0],
  83        ARRAY_SIZE(wm8711_output_mixer_controls)),
  84SND_SOC_DAPM_DAC("DAC", "HiFi Playback", WM8711_PWR, 3, 1),
  85SND_SOC_DAPM_OUTPUT("LOUT"),
  86SND_SOC_DAPM_OUTPUT("LHPOUT"),
  87SND_SOC_DAPM_OUTPUT("ROUT"),
  88SND_SOC_DAPM_OUTPUT("RHPOUT"),
  89};
  90
  91static const struct snd_soc_dapm_route wm8711_intercon[] = {
  92        /* output mixer */
  93        {"Output Mixer", "Line Bypass Switch", "Line Input"},
  94        {"Output Mixer", "HiFi Playback Switch", "DAC"},
  95
  96        /* outputs */
  97        {"RHPOUT", NULL, "Output Mixer"},
  98        {"ROUT", NULL, "Output Mixer"},
  99        {"LHPOUT", NULL, "Output Mixer"},
 100        {"LOUT", NULL, "Output Mixer"},
 101};
 102
 103struct _coeff_div {
 104        u32 mclk;
 105        u32 rate;
 106        u16 fs;
 107        u8 sr:4;
 108        u8 bosr:1;
 109        u8 usb:1;
 110};
 111
 112/* codec mclk clock divider coefficients */
 113static const struct _coeff_div coeff_div[] = {
 114        /* 48k */
 115        {12288000, 48000, 256, 0x0, 0x0, 0x0},
 116        {18432000, 48000, 384, 0x0, 0x1, 0x0},
 117        {12000000, 48000, 250, 0x0, 0x0, 0x1},
 118
 119        /* 32k */
 120        {12288000, 32000, 384, 0x6, 0x0, 0x0},
 121        {18432000, 32000, 576, 0x6, 0x1, 0x0},
 122        {12000000, 32000, 375, 0x6, 0x0, 0x1},
 123
 124        /* 8k */
 125        {12288000, 8000, 1536, 0x3, 0x0, 0x0},
 126        {18432000, 8000, 2304, 0x3, 0x1, 0x0},
 127        {11289600, 8000, 1408, 0xb, 0x0, 0x0},
 128        {16934400, 8000, 2112, 0xb, 0x1, 0x0},
 129        {12000000, 8000, 1500, 0x3, 0x0, 0x1},
 130
 131        /* 96k */
 132        {12288000, 96000, 128, 0x7, 0x0, 0x0},
 133        {18432000, 96000, 192, 0x7, 0x1, 0x0},
 134        {12000000, 96000, 125, 0x7, 0x0, 0x1},
 135
 136        /* 44.1k */
 137        {11289600, 44100, 256, 0x8, 0x0, 0x0},
 138        {16934400, 44100, 384, 0x8, 0x1, 0x0},
 139        {12000000, 44100, 272, 0x8, 0x1, 0x1},
 140
 141        /* 88.2k */
 142        {11289600, 88200, 128, 0xf, 0x0, 0x0},
 143        {16934400, 88200, 192, 0xf, 0x1, 0x0},
 144        {12000000, 88200, 136, 0xf, 0x1, 0x1},
 145};
 146
 147static inline int get_coeff(int mclk, int rate)
 148{
 149        int i;
 150
 151        for (i = 0; i < ARRAY_SIZE(coeff_div); i++) {
 152                if (coeff_div[i].rate == rate && coeff_div[i].mclk == mclk)
 153                        return i;
 154        }
 155        return 0;
 156}
 157
 158static int wm8711_hw_params(struct snd_pcm_substream *substream,
 159        struct snd_pcm_hw_params *params,
 160        struct snd_soc_dai *dai)
 161{
 162        struct snd_soc_codec *codec = dai->codec;
 163        struct wm8711_priv *wm8711 =  snd_soc_codec_get_drvdata(codec);
 164        u16 iface = snd_soc_read(codec, WM8711_IFACE) & 0xfff3;
 165        int i = get_coeff(wm8711->sysclk, params_rate(params));
 166        u16 srate = (coeff_div[i].sr << 2) |
 167                (coeff_div[i].bosr << 1) | coeff_div[i].usb;
 168
 169        snd_soc_write(codec, WM8711_SRATE, srate);
 170
 171        /* bit size */
 172        switch (params_width(params)) {
 173        case 16:
 174                break;
 175        case 20:
 176                iface |= 0x0004;
 177                break;
 178        case 24:
 179                iface |= 0x0008;
 180                break;
 181        }
 182
 183        snd_soc_write(codec, WM8711_IFACE, iface);
 184        return 0;
 185}
 186
 187static int wm8711_pcm_prepare(struct snd_pcm_substream *substream,
 188                              struct snd_soc_dai *dai)
 189{
 190        struct snd_soc_codec *codec = dai->codec;
 191
 192        /* set active */
 193        snd_soc_write(codec, WM8711_ACTIVE, 0x0001);
 194
 195        return 0;
 196}
 197
 198static void wm8711_shutdown(struct snd_pcm_substream *substream,
 199                            struct snd_soc_dai *dai)
 200{
 201        struct snd_soc_codec *codec = dai->codec;
 202
 203        /* deactivate */
 204        if (!snd_soc_codec_is_active(codec)) {
 205                udelay(50);
 206                snd_soc_write(codec, WM8711_ACTIVE, 0x0);
 207        }
 208}
 209
 210static int wm8711_mute(struct snd_soc_dai *dai, int mute)
 211{
 212        struct snd_soc_codec *codec = dai->codec;
 213        u16 mute_reg = snd_soc_read(codec, WM8711_APDIGI) & 0xfff7;
 214
 215        if (mute)
 216                snd_soc_write(codec, WM8711_APDIGI, mute_reg | 0x8);
 217        else
 218                snd_soc_write(codec, WM8711_APDIGI, mute_reg);
 219
 220        return 0;
 221}
 222
 223static int wm8711_set_dai_sysclk(struct snd_soc_dai *codec_dai,
 224                int clk_id, unsigned int freq, int dir)
 225{
 226        struct snd_soc_codec *codec = codec_dai->codec;
 227        struct wm8711_priv *wm8711 =  snd_soc_codec_get_drvdata(codec);
 228
 229        switch (freq) {
 230        case 11289600:
 231        case 12000000:
 232        case 12288000:
 233        case 16934400:
 234        case 18432000:
 235                wm8711->sysclk = freq;
 236                return 0;
 237        }
 238        return -EINVAL;
 239}
 240
 241static int wm8711_set_dai_fmt(struct snd_soc_dai *codec_dai,
 242                unsigned int fmt)
 243{
 244        struct snd_soc_codec *codec = codec_dai->codec;
 245        u16 iface = snd_soc_read(codec, WM8711_IFACE) & 0x000c;
 246
 247        /* set master/slave audio interface */
 248        switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
 249        case SND_SOC_DAIFMT_CBM_CFM:
 250                iface |= 0x0040;
 251                break;
 252        case SND_SOC_DAIFMT_CBS_CFS:
 253                break;
 254        default:
 255                return -EINVAL;
 256        }
 257
 258        /* interface format */
 259        switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
 260        case SND_SOC_DAIFMT_I2S:
 261                iface |= 0x0002;
 262                break;
 263        case SND_SOC_DAIFMT_RIGHT_J:
 264                break;
 265        case SND_SOC_DAIFMT_LEFT_J:
 266                iface |= 0x0001;
 267                break;
 268        case SND_SOC_DAIFMT_DSP_A:
 269                iface |= 0x0003;
 270                break;
 271        case SND_SOC_DAIFMT_DSP_B:
 272                iface |= 0x0013;
 273                break;
 274        default:
 275                return -EINVAL;
 276        }
 277
 278        /* clock inversion */
 279        switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
 280        case SND_SOC_DAIFMT_NB_NF:
 281                break;
 282        case SND_SOC_DAIFMT_IB_IF:
 283                iface |= 0x0090;
 284                break;
 285        case SND_SOC_DAIFMT_IB_NF:
 286                iface |= 0x0080;
 287                break;
 288        case SND_SOC_DAIFMT_NB_IF:
 289                iface |= 0x0010;
 290                break;
 291        default:
 292                return -EINVAL;
 293        }
 294
 295        /* set iface */
 296        snd_soc_write(codec, WM8711_IFACE, iface);
 297        return 0;
 298}
 299
 300static int wm8711_set_bias_level(struct snd_soc_codec *codec,
 301        enum snd_soc_bias_level level)
 302{
 303        struct wm8711_priv *wm8711 = snd_soc_codec_get_drvdata(codec);
 304        u16 reg = snd_soc_read(codec, WM8711_PWR) & 0xff7f;
 305
 306        switch (level) {
 307        case SND_SOC_BIAS_ON:
 308                snd_soc_write(codec, WM8711_PWR, reg);
 309                break;
 310        case SND_SOC_BIAS_PREPARE:
 311                break;
 312        case SND_SOC_BIAS_STANDBY:
 313                if (snd_soc_codec_get_bias_level(codec) == SND_SOC_BIAS_OFF)
 314                        regcache_sync(wm8711->regmap);
 315
 316                snd_soc_write(codec, WM8711_PWR, reg | 0x0040);
 317                break;
 318        case SND_SOC_BIAS_OFF:
 319                snd_soc_write(codec, WM8711_ACTIVE, 0x0);
 320                snd_soc_write(codec, WM8711_PWR, 0xffff);
 321                break;
 322        }
 323        return 0;
 324}
 325
 326#define WM8711_RATES SNDRV_PCM_RATE_8000_96000
 327
 328#define WM8711_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S20_3LE |\
 329        SNDRV_PCM_FMTBIT_S24_LE)
 330
 331static const struct snd_soc_dai_ops wm8711_ops = {
 332        .prepare = wm8711_pcm_prepare,
 333        .hw_params = wm8711_hw_params,
 334        .shutdown = wm8711_shutdown,
 335        .digital_mute = wm8711_mute,
 336        .set_sysclk = wm8711_set_dai_sysclk,
 337        .set_fmt = wm8711_set_dai_fmt,
 338};
 339
 340static struct snd_soc_dai_driver wm8711_dai = {
 341        .name = "wm8711-hifi",
 342        .playback = {
 343                .stream_name = "Playback",
 344                .channels_min = 1,
 345                .channels_max = 2,
 346                .rates = WM8711_RATES,
 347                .formats = WM8711_FORMATS,
 348        },
 349        .ops = &wm8711_ops,
 350};
 351
 352static int wm8711_probe(struct snd_soc_codec *codec)
 353{
 354        int ret;
 355
 356        ret = wm8711_reset(codec);
 357        if (ret < 0) {
 358                dev_err(codec->dev, "Failed to issue reset\n");
 359                return ret;
 360        }
 361
 362        /* Latch the update bits */
 363        snd_soc_update_bits(codec, WM8711_LOUT1V, 0x0100, 0x0100);
 364        snd_soc_update_bits(codec, WM8711_ROUT1V, 0x0100, 0x0100);
 365
 366        return ret;
 367
 368}
 369
 370static struct snd_soc_codec_driver soc_codec_dev_wm8711 = {
 371        .probe =        wm8711_probe,
 372        .set_bias_level = wm8711_set_bias_level,
 373        .suspend_bias_off = true,
 374
 375        .controls = wm8711_snd_controls,
 376        .num_controls = ARRAY_SIZE(wm8711_snd_controls),
 377        .dapm_widgets = wm8711_dapm_widgets,
 378        .num_dapm_widgets = ARRAY_SIZE(wm8711_dapm_widgets),
 379        .dapm_routes = wm8711_intercon,
 380        .num_dapm_routes = ARRAY_SIZE(wm8711_intercon),
 381};
 382
 383static const struct of_device_id wm8711_of_match[] = {
 384        { .compatible = "wlf,wm8711", },
 385        { }
 386};
 387MODULE_DEVICE_TABLE(of, wm8711_of_match);
 388
 389static const struct regmap_config wm8711_regmap = {
 390        .reg_bits = 7,
 391        .val_bits = 9,
 392        .max_register = WM8711_RESET,
 393
 394        .reg_defaults = wm8711_reg_defaults,
 395        .num_reg_defaults = ARRAY_SIZE(wm8711_reg_defaults),
 396        .cache_type = REGCACHE_RBTREE,
 397
 398        .volatile_reg = wm8711_volatile,
 399};
 400
 401#if defined(CONFIG_SPI_MASTER)
 402static int wm8711_spi_probe(struct spi_device *spi)
 403{
 404        struct wm8711_priv *wm8711;
 405        int ret;
 406
 407        wm8711 = devm_kzalloc(&spi->dev, sizeof(struct wm8711_priv),
 408                              GFP_KERNEL);
 409        if (wm8711 == NULL)
 410                return -ENOMEM;
 411
 412        wm8711->regmap = devm_regmap_init_spi(spi, &wm8711_regmap);
 413        if (IS_ERR(wm8711->regmap))
 414                return PTR_ERR(wm8711->regmap);
 415
 416        spi_set_drvdata(spi, wm8711);
 417
 418        ret = snd_soc_register_codec(&spi->dev,
 419                        &soc_codec_dev_wm8711, &wm8711_dai, 1);
 420
 421        return ret;
 422}
 423
 424static int wm8711_spi_remove(struct spi_device *spi)
 425{
 426        snd_soc_unregister_codec(&spi->dev);
 427
 428        return 0;
 429}
 430
 431static struct spi_driver wm8711_spi_driver = {
 432        .driver = {
 433                .name   = "wm8711",
 434                .of_match_table = wm8711_of_match,
 435        },
 436        .probe          = wm8711_spi_probe,
 437        .remove         = wm8711_spi_remove,
 438};
 439#endif /* CONFIG_SPI_MASTER */
 440
 441#if IS_ENABLED(CONFIG_I2C)
 442static int wm8711_i2c_probe(struct i2c_client *client,
 443                            const struct i2c_device_id *id)
 444{
 445        struct wm8711_priv *wm8711;
 446        int ret;
 447
 448        wm8711 = devm_kzalloc(&client->dev, sizeof(struct wm8711_priv),
 449                              GFP_KERNEL);
 450        if (wm8711 == NULL)
 451                return -ENOMEM;
 452
 453        wm8711->regmap = devm_regmap_init_i2c(client, &wm8711_regmap);
 454        if (IS_ERR(wm8711->regmap))
 455                return PTR_ERR(wm8711->regmap);
 456
 457        i2c_set_clientdata(client, wm8711);
 458
 459        ret =  snd_soc_register_codec(&client->dev,
 460                        &soc_codec_dev_wm8711, &wm8711_dai, 1);
 461
 462        return ret;
 463}
 464
 465static int wm8711_i2c_remove(struct i2c_client *client)
 466{
 467        snd_soc_unregister_codec(&client->dev);
 468        return 0;
 469}
 470
 471static const struct i2c_device_id wm8711_i2c_id[] = {
 472        { "wm8711", 0 },
 473        { }
 474};
 475MODULE_DEVICE_TABLE(i2c, wm8711_i2c_id);
 476
 477static struct i2c_driver wm8711_i2c_driver = {
 478        .driver = {
 479                .name = "wm8711",
 480                .of_match_table = wm8711_of_match,
 481        },
 482        .probe =    wm8711_i2c_probe,
 483        .remove =   wm8711_i2c_remove,
 484        .id_table = wm8711_i2c_id,
 485};
 486#endif
 487
 488static int __init wm8711_modinit(void)
 489{
 490        int ret;
 491#if IS_ENABLED(CONFIG_I2C)
 492        ret = i2c_add_driver(&wm8711_i2c_driver);
 493        if (ret != 0) {
 494                printk(KERN_ERR "Failed to register WM8711 I2C driver: %d\n",
 495                       ret);
 496        }
 497#endif
 498#if defined(CONFIG_SPI_MASTER)
 499        ret = spi_register_driver(&wm8711_spi_driver);
 500        if (ret != 0) {
 501                printk(KERN_ERR "Failed to register WM8711 SPI driver: %d\n",
 502                       ret);
 503        }
 504#endif
 505        return 0;
 506}
 507module_init(wm8711_modinit);
 508
 509static void __exit wm8711_exit(void)
 510{
 511#if IS_ENABLED(CONFIG_I2C)
 512        i2c_del_driver(&wm8711_i2c_driver);
 513#endif
 514#if defined(CONFIG_SPI_MASTER)
 515        spi_unregister_driver(&wm8711_spi_driver);
 516#endif
 517}
 518module_exit(wm8711_exit);
 519
 520MODULE_DESCRIPTION("ASoC WM8711 driver");
 521MODULE_AUTHOR("Mike Arthur");
 522MODULE_LICENSE("GPL");
 523