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_component_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_component *component = dai->component;
 163        struct wm8711_priv *wm8711 =  snd_soc_component_get_drvdata(component);
 164        u16 iface = snd_soc_component_read32(component, 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_component_write(component, 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_component_write(component, 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_component *component = dai->component;
 191
 192        /* set active */
 193        snd_soc_component_write(component, 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_component *component = dai->component;
 202
 203        /* deactivate */
 204        if (!snd_soc_component_is_active(component)) {
 205                udelay(50);
 206                snd_soc_component_write(component, WM8711_ACTIVE, 0x0);
 207        }
 208}
 209
 210static int wm8711_mute(struct snd_soc_dai *dai, int mute)
 211{
 212        struct snd_soc_component *component = dai->component;
 213        u16 mute_reg = snd_soc_component_read32(component, WM8711_APDIGI) & 0xfff7;
 214
 215        if (mute)
 216                snd_soc_component_write(component, WM8711_APDIGI, mute_reg | 0x8);
 217        else
 218                snd_soc_component_write(component, 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_component *component = codec_dai->component;
 227        struct wm8711_priv *wm8711 =  snd_soc_component_get_drvdata(component);
 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_component *component = codec_dai->component;
 245        u16 iface = snd_soc_component_read32(component, 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_component_write(component, WM8711_IFACE, iface);
 297        return 0;
 298}
 299
 300static int wm8711_set_bias_level(struct snd_soc_component *component,
 301        enum snd_soc_bias_level level)
 302{
 303        struct wm8711_priv *wm8711 = snd_soc_component_get_drvdata(component);
 304        u16 reg = snd_soc_component_read32(component, WM8711_PWR) & 0xff7f;
 305
 306        switch (level) {
 307        case SND_SOC_BIAS_ON:
 308                snd_soc_component_write(component, WM8711_PWR, reg);
 309                break;
 310        case SND_SOC_BIAS_PREPARE:
 311                break;
 312        case SND_SOC_BIAS_STANDBY:
 313                if (snd_soc_component_get_bias_level(component) == SND_SOC_BIAS_OFF)
 314                        regcache_sync(wm8711->regmap);
 315
 316                snd_soc_component_write(component, WM8711_PWR, reg | 0x0040);
 317                break;
 318        case SND_SOC_BIAS_OFF:
 319                snd_soc_component_write(component, WM8711_ACTIVE, 0x0);
 320                snd_soc_component_write(component, 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_component *component)
 353{
 354        int ret;
 355
 356        ret = wm8711_reset(component);
 357        if (ret < 0) {
 358                dev_err(component->dev, "Failed to issue reset\n");
 359                return ret;
 360        }
 361
 362        /* Latch the update bits */
 363        snd_soc_component_update_bits(component, WM8711_LOUT1V, 0x0100, 0x0100);
 364        snd_soc_component_update_bits(component, WM8711_ROUT1V, 0x0100, 0x0100);
 365
 366        return ret;
 367
 368}
 369
 370static const struct snd_soc_component_driver soc_component_dev_wm8711 = {
 371        .probe                  = wm8711_probe,
 372        .set_bias_level         = wm8711_set_bias_level,
 373        .controls               = wm8711_snd_controls,
 374        .num_controls           = ARRAY_SIZE(wm8711_snd_controls),
 375        .dapm_widgets           = wm8711_dapm_widgets,
 376        .num_dapm_widgets       = ARRAY_SIZE(wm8711_dapm_widgets),
 377        .dapm_routes            = wm8711_intercon,
 378        .num_dapm_routes        = ARRAY_SIZE(wm8711_intercon),
 379        .suspend_bias_off       = 1,
 380        .idle_bias_on           = 1,
 381        .use_pmdown_time        = 1,
 382        .endianness             = 1,
 383        .non_legacy_dai_naming  = 1,
 384};
 385
 386static const struct of_device_id wm8711_of_match[] = {
 387        { .compatible = "wlf,wm8711", },
 388        { }
 389};
 390MODULE_DEVICE_TABLE(of, wm8711_of_match);
 391
 392static const struct regmap_config wm8711_regmap = {
 393        .reg_bits = 7,
 394        .val_bits = 9,
 395        .max_register = WM8711_RESET,
 396
 397        .reg_defaults = wm8711_reg_defaults,
 398        .num_reg_defaults = ARRAY_SIZE(wm8711_reg_defaults),
 399        .cache_type = REGCACHE_RBTREE,
 400
 401        .volatile_reg = wm8711_volatile,
 402};
 403
 404#if defined(CONFIG_SPI_MASTER)
 405static int wm8711_spi_probe(struct spi_device *spi)
 406{
 407        struct wm8711_priv *wm8711;
 408        int ret;
 409
 410        wm8711 = devm_kzalloc(&spi->dev, sizeof(struct wm8711_priv),
 411                              GFP_KERNEL);
 412        if (wm8711 == NULL)
 413                return -ENOMEM;
 414
 415        wm8711->regmap = devm_regmap_init_spi(spi, &wm8711_regmap);
 416        if (IS_ERR(wm8711->regmap))
 417                return PTR_ERR(wm8711->regmap);
 418
 419        spi_set_drvdata(spi, wm8711);
 420
 421        ret = devm_snd_soc_register_component(&spi->dev,
 422                        &soc_component_dev_wm8711, &wm8711_dai, 1);
 423
 424        return ret;
 425}
 426
 427static struct spi_driver wm8711_spi_driver = {
 428        .driver = {
 429                .name   = "wm8711",
 430                .of_match_table = wm8711_of_match,
 431        },
 432        .probe          = wm8711_spi_probe,
 433};
 434#endif /* CONFIG_SPI_MASTER */
 435
 436#if IS_ENABLED(CONFIG_I2C)
 437static int wm8711_i2c_probe(struct i2c_client *client,
 438                            const struct i2c_device_id *id)
 439{
 440        struct wm8711_priv *wm8711;
 441        int ret;
 442
 443        wm8711 = devm_kzalloc(&client->dev, sizeof(struct wm8711_priv),
 444                              GFP_KERNEL);
 445        if (wm8711 == NULL)
 446                return -ENOMEM;
 447
 448        wm8711->regmap = devm_regmap_init_i2c(client, &wm8711_regmap);
 449        if (IS_ERR(wm8711->regmap))
 450                return PTR_ERR(wm8711->regmap);
 451
 452        i2c_set_clientdata(client, wm8711);
 453
 454        ret = devm_snd_soc_register_component(&client->dev,
 455                        &soc_component_dev_wm8711, &wm8711_dai, 1);
 456
 457        return ret;
 458}
 459
 460static const struct i2c_device_id wm8711_i2c_id[] = {
 461        { "wm8711", 0 },
 462        { }
 463};
 464MODULE_DEVICE_TABLE(i2c, wm8711_i2c_id);
 465
 466static struct i2c_driver wm8711_i2c_driver = {
 467        .driver = {
 468                .name = "wm8711",
 469                .of_match_table = wm8711_of_match,
 470        },
 471        .probe =    wm8711_i2c_probe,
 472        .id_table = wm8711_i2c_id,
 473};
 474#endif
 475
 476static int __init wm8711_modinit(void)
 477{
 478        int ret;
 479#if IS_ENABLED(CONFIG_I2C)
 480        ret = i2c_add_driver(&wm8711_i2c_driver);
 481        if (ret != 0) {
 482                printk(KERN_ERR "Failed to register WM8711 I2C driver: %d\n",
 483                       ret);
 484        }
 485#endif
 486#if defined(CONFIG_SPI_MASTER)
 487        ret = spi_register_driver(&wm8711_spi_driver);
 488        if (ret != 0) {
 489                printk(KERN_ERR "Failed to register WM8711 SPI driver: %d\n",
 490                       ret);
 491        }
 492#endif
 493        return 0;
 494}
 495module_init(wm8711_modinit);
 496
 497static void __exit wm8711_exit(void)
 498{
 499#if IS_ENABLED(CONFIG_I2C)
 500        i2c_del_driver(&wm8711_i2c_driver);
 501#endif
 502#if defined(CONFIG_SPI_MASTER)
 503        spi_unregister_driver(&wm8711_spi_driver);
 504#endif
 505}
 506module_exit(wm8711_exit);
 507
 508MODULE_DESCRIPTION("ASoC WM8711 driver");
 509MODULE_AUTHOR("Mike Arthur");
 510MODULE_LICENSE("GPL");
 511