linux/drivers/gpu/drm/bridge/synopsys/dw-hdmi-ahb-audio.c
<<
>>
Prefs
   1/*
   2 * DesignWare HDMI audio driver
   3 *
   4 * This program is free software; you can redistribute it and/or modify
   5 * it under the terms of the GNU General Public License version 2 as
   6 * published by the Free Software Foundation.
   7 *
   8 * Written and tested against the Designware HDMI Tx found in iMX6.
   9 */
  10#include <linux/io.h>
  11#include <linux/interrupt.h>
  12#include <linux/module.h>
  13#include <linux/platform_device.h>
  14#include <drm/bridge/dw_hdmi.h>
  15#include <drm/drm_edid.h>
  16
  17#include <sound/asoundef.h>
  18#include <sound/core.h>
  19#include <sound/initval.h>
  20#include <sound/pcm.h>
  21#include <sound/pcm_drm_eld.h>
  22#include <sound/pcm_iec958.h>
  23
  24#include "dw-hdmi-audio.h"
  25
  26#define DRIVER_NAME "dw-hdmi-ahb-audio"
  27
  28/* Provide some bits rather than bit offsets */
  29enum {
  30        HDMI_AHB_DMA_CONF0_SW_FIFO_RST = BIT(7),
  31        HDMI_AHB_DMA_CONF0_EN_HLOCK = BIT(3),
  32        HDMI_AHB_DMA_START_START = BIT(0),
  33        HDMI_AHB_DMA_STOP_STOP = BIT(0),
  34        HDMI_IH_MUTE_AHBDMAAUD_STAT0_ERROR = BIT(5),
  35        HDMI_IH_MUTE_AHBDMAAUD_STAT0_LOST = BIT(4),
  36        HDMI_IH_MUTE_AHBDMAAUD_STAT0_RETRY = BIT(3),
  37        HDMI_IH_MUTE_AHBDMAAUD_STAT0_DONE = BIT(2),
  38        HDMI_IH_MUTE_AHBDMAAUD_STAT0_BUFFFULL = BIT(1),
  39        HDMI_IH_MUTE_AHBDMAAUD_STAT0_BUFFEMPTY = BIT(0),
  40        HDMI_IH_MUTE_AHBDMAAUD_STAT0_ALL =
  41                HDMI_IH_MUTE_AHBDMAAUD_STAT0_ERROR |
  42                HDMI_IH_MUTE_AHBDMAAUD_STAT0_LOST |
  43                HDMI_IH_MUTE_AHBDMAAUD_STAT0_RETRY |
  44                HDMI_IH_MUTE_AHBDMAAUD_STAT0_DONE |
  45                HDMI_IH_MUTE_AHBDMAAUD_STAT0_BUFFFULL |
  46                HDMI_IH_MUTE_AHBDMAAUD_STAT0_BUFFEMPTY,
  47        HDMI_IH_AHBDMAAUD_STAT0_ERROR = BIT(5),
  48        HDMI_IH_AHBDMAAUD_STAT0_LOST = BIT(4),
  49        HDMI_IH_AHBDMAAUD_STAT0_RETRY = BIT(3),
  50        HDMI_IH_AHBDMAAUD_STAT0_DONE = BIT(2),
  51        HDMI_IH_AHBDMAAUD_STAT0_BUFFFULL = BIT(1),
  52        HDMI_IH_AHBDMAAUD_STAT0_BUFFEMPTY = BIT(0),
  53        HDMI_IH_AHBDMAAUD_STAT0_ALL =
  54                HDMI_IH_AHBDMAAUD_STAT0_ERROR |
  55                HDMI_IH_AHBDMAAUD_STAT0_LOST |
  56                HDMI_IH_AHBDMAAUD_STAT0_RETRY |
  57                HDMI_IH_AHBDMAAUD_STAT0_DONE |
  58                HDMI_IH_AHBDMAAUD_STAT0_BUFFFULL |
  59                HDMI_IH_AHBDMAAUD_STAT0_BUFFEMPTY,
  60        HDMI_AHB_DMA_CONF0_INCR16 = 2 << 1,
  61        HDMI_AHB_DMA_CONF0_INCR8 = 1 << 1,
  62        HDMI_AHB_DMA_CONF0_INCR4 = 0,
  63        HDMI_AHB_DMA_CONF0_BURST_MODE = BIT(0),
  64        HDMI_AHB_DMA_MASK_DONE = BIT(7),
  65
  66        HDMI_REVISION_ID = 0x0001,
  67        HDMI_IH_AHBDMAAUD_STAT0 = 0x0109,
  68        HDMI_IH_MUTE_AHBDMAAUD_STAT0 = 0x0189,
  69        HDMI_FC_AUDICONF2 = 0x1027,
  70        HDMI_FC_AUDSCONF = 0x1063,
  71        HDMI_FC_AUDSCONF_LAYOUT1 = 1 << 0,
  72        HDMI_FC_AUDSCONF_LAYOUT0 = 0 << 0,
  73        HDMI_AHB_DMA_CONF0 = 0x3600,
  74        HDMI_AHB_DMA_START = 0x3601,
  75        HDMI_AHB_DMA_STOP = 0x3602,
  76        HDMI_AHB_DMA_THRSLD = 0x3603,
  77        HDMI_AHB_DMA_STRADDR0 = 0x3604,
  78        HDMI_AHB_DMA_STPADDR0 = 0x3608,
  79        HDMI_AHB_DMA_MASK = 0x3614,
  80        HDMI_AHB_DMA_POL = 0x3615,
  81        HDMI_AHB_DMA_CONF1 = 0x3616,
  82        HDMI_AHB_DMA_BUFFPOL = 0x361a,
  83};
  84
  85struct dw_hdmi_channel_conf {
  86        u8 conf1;
  87        u8 ca;
  88};
  89
  90/*
  91 * The default mapping of ALSA channels to HDMI channels and speaker
  92 * allocation bits.  Note that we can't do channel remapping here -
  93 * channels must be in the same order.
  94 *
  95 * Mappings for alsa-lib pcm/surround*.conf files:
  96 *
  97 *              Front   Sur4.0  Sur4.1  Sur5.0  Sur5.1  Sur7.1
  98 * Channels     2       4       6       6       6       8
  99 *
 100 * Our mapping from ALSA channel to CEA686D speaker name and HDMI channel:
 101 *
 102 *                              Number of ALSA channels
 103 * ALSA Channel 2       3       4       5       6       7       8
 104 * 0            FL:0    =       =       =       =       =       =
 105 * 1            FR:1    =       =       =       =       =       =
 106 * 2                    FC:3    RL:4    LFE:2   =       =       =
 107 * 3                            RR:5    RL:4    FC:3    =       =
 108 * 4                                    RR:5    RL:4    =       =
 109 * 5                                            RR:5    =       =
 110 * 6                                                    RC:6    =
 111 * 7                                                    RLC/FRC RLC/FRC
 112 */
 113static struct dw_hdmi_channel_conf default_hdmi_channel_config[7] = {
 114        { 0x03, 0x00 }, /* FL,FR */
 115        { 0x0b, 0x02 }, /* FL,FR,FC */
 116        { 0x33, 0x08 }, /* FL,FR,RL,RR */
 117        { 0x37, 0x09 }, /* FL,FR,LFE,RL,RR */
 118        { 0x3f, 0x0b }, /* FL,FR,LFE,FC,RL,RR */
 119        { 0x7f, 0x0f }, /* FL,FR,LFE,FC,RL,RR,RC */
 120        { 0xff, 0x13 }, /* FL,FR,LFE,FC,RL,RR,[FR]RC,[FR]LC */
 121};
 122
 123struct snd_dw_hdmi {
 124        struct snd_card *card;
 125        struct snd_pcm *pcm;
 126        spinlock_t lock;
 127        struct dw_hdmi_audio_data data;
 128        struct snd_pcm_substream *substream;
 129        void (*reformat)(struct snd_dw_hdmi *, size_t, size_t);
 130        void *buf_src;
 131        void *buf_dst;
 132        dma_addr_t buf_addr;
 133        unsigned buf_offset;
 134        unsigned buf_period;
 135        unsigned buf_size;
 136        unsigned channels;
 137        u8 revision;
 138        u8 iec_offset;
 139        u8 cs[192][8];
 140};
 141
 142static void dw_hdmi_writel(u32 val, void __iomem *ptr)
 143{
 144        writeb_relaxed(val, ptr);
 145        writeb_relaxed(val >> 8, ptr + 1);
 146        writeb_relaxed(val >> 16, ptr + 2);
 147        writeb_relaxed(val >> 24, ptr + 3);
 148}
 149
 150/*
 151 * Convert to hardware format: The userspace buffer contains IEC958 samples,
 152 * with the PCUV bits in bits 31..28 and audio samples in bits 27..4.  We
 153 * need these to be in bits 27..24, with the IEC B bit in bit 28, and audio
 154 * samples in 23..0.
 155 *
 156 * Default preamble in bits 3..0: 8 = block start, 4 = even 2 = odd
 157 *
 158 * Ideally, we could do with having the data properly formatted in userspace.
 159 */
 160static void dw_hdmi_reformat_iec958(struct snd_dw_hdmi *dw,
 161        size_t offset, size_t bytes)
 162{
 163        u32 *src = dw->buf_src + offset;
 164        u32 *dst = dw->buf_dst + offset;
 165        u32 *end = dw->buf_src + offset + bytes;
 166
 167        do {
 168                u32 b, sample = *src++;
 169
 170                b = (sample & 8) << (28 - 3);
 171
 172                sample >>= 4;
 173
 174                *dst++ = sample | b;
 175        } while (src < end);
 176}
 177
 178static u32 parity(u32 sample)
 179{
 180        sample ^= sample >> 16;
 181        sample ^= sample >> 8;
 182        sample ^= sample >> 4;
 183        sample ^= sample >> 2;
 184        sample ^= sample >> 1;
 185        return (sample & 1) << 27;
 186}
 187
 188static void dw_hdmi_reformat_s24(struct snd_dw_hdmi *dw,
 189        size_t offset, size_t bytes)
 190{
 191        u32 *src = dw->buf_src + offset;
 192        u32 *dst = dw->buf_dst + offset;
 193        u32 *end = dw->buf_src + offset + bytes;
 194
 195        do {
 196                unsigned i;
 197                u8 *cs;
 198
 199                cs = dw->cs[dw->iec_offset++];
 200                if (dw->iec_offset >= 192)
 201                        dw->iec_offset = 0;
 202
 203                i = dw->channels;
 204                do {
 205                        u32 sample = *src++;
 206
 207                        sample &= ~0xff000000;
 208                        sample |= *cs++ << 24;
 209                        sample |= parity(sample & ~0xf8000000);
 210
 211                        *dst++ = sample;
 212                } while (--i);
 213        } while (src < end);
 214}
 215
 216static void dw_hdmi_create_cs(struct snd_dw_hdmi *dw,
 217        struct snd_pcm_runtime *runtime)
 218{
 219        u8 cs[4];
 220        unsigned ch, i, j;
 221
 222        snd_pcm_create_iec958_consumer(runtime, cs, sizeof(cs));
 223
 224        memset(dw->cs, 0, sizeof(dw->cs));
 225
 226        for (ch = 0; ch < 8; ch++) {
 227                cs[2] &= ~IEC958_AES2_CON_CHANNEL;
 228                cs[2] |= (ch + 1) << 4;
 229
 230                for (i = 0; i < ARRAY_SIZE(cs); i++) {
 231                        unsigned c = cs[i];
 232
 233                        for (j = 0; j < 8; j++, c >>= 1)
 234                                dw->cs[i * 8 + j][ch] = (c & 1) << 2;
 235                }
 236        }
 237        dw->cs[0][0] |= BIT(4);
 238}
 239
 240static void dw_hdmi_start_dma(struct snd_dw_hdmi *dw)
 241{
 242        void __iomem *base = dw->data.base;
 243        unsigned offset = dw->buf_offset;
 244        unsigned period = dw->buf_period;
 245        u32 start, stop;
 246
 247        dw->reformat(dw, offset, period);
 248
 249        /* Clear all irqs before enabling irqs and starting DMA */
 250        writeb_relaxed(HDMI_IH_AHBDMAAUD_STAT0_ALL,
 251                       base + HDMI_IH_AHBDMAAUD_STAT0);
 252
 253        start = dw->buf_addr + offset;
 254        stop = start + period - 1;
 255
 256        /* Setup the hardware start/stop addresses */
 257        dw_hdmi_writel(start, base + HDMI_AHB_DMA_STRADDR0);
 258        dw_hdmi_writel(stop, base + HDMI_AHB_DMA_STPADDR0);
 259
 260        writeb_relaxed((u8)~HDMI_AHB_DMA_MASK_DONE, base + HDMI_AHB_DMA_MASK);
 261        writeb(HDMI_AHB_DMA_START_START, base + HDMI_AHB_DMA_START);
 262
 263        offset += period;
 264        if (offset >= dw->buf_size)
 265                offset = 0;
 266        dw->buf_offset = offset;
 267}
 268
 269static void dw_hdmi_stop_dma(struct snd_dw_hdmi *dw)
 270{
 271        /* Disable interrupts before disabling DMA */
 272        writeb_relaxed(~0, dw->data.base + HDMI_AHB_DMA_MASK);
 273        writeb_relaxed(HDMI_AHB_DMA_STOP_STOP, dw->data.base + HDMI_AHB_DMA_STOP);
 274}
 275
 276static irqreturn_t snd_dw_hdmi_irq(int irq, void *data)
 277{
 278        struct snd_dw_hdmi *dw = data;
 279        struct snd_pcm_substream *substream;
 280        unsigned stat;
 281
 282        stat = readb_relaxed(dw->data.base + HDMI_IH_AHBDMAAUD_STAT0);
 283        if (!stat)
 284                return IRQ_NONE;
 285
 286        writeb_relaxed(stat, dw->data.base + HDMI_IH_AHBDMAAUD_STAT0);
 287
 288        substream = dw->substream;
 289        if (stat & HDMI_IH_AHBDMAAUD_STAT0_DONE && substream) {
 290                snd_pcm_period_elapsed(substream);
 291
 292                spin_lock(&dw->lock);
 293                if (dw->substream)
 294                        dw_hdmi_start_dma(dw);
 295                spin_unlock(&dw->lock);
 296        }
 297
 298        return IRQ_HANDLED;
 299}
 300
 301static struct snd_pcm_hardware dw_hdmi_hw = {
 302        .info = SNDRV_PCM_INFO_INTERLEAVED |
 303                SNDRV_PCM_INFO_BLOCK_TRANSFER |
 304                SNDRV_PCM_INFO_MMAP |
 305                SNDRV_PCM_INFO_MMAP_VALID,
 306        .formats = SNDRV_PCM_FMTBIT_IEC958_SUBFRAME_LE |
 307                   SNDRV_PCM_FMTBIT_S24_LE,
 308        .rates = SNDRV_PCM_RATE_32000 |
 309                 SNDRV_PCM_RATE_44100 |
 310                 SNDRV_PCM_RATE_48000 |
 311                 SNDRV_PCM_RATE_88200 |
 312                 SNDRV_PCM_RATE_96000 |
 313                 SNDRV_PCM_RATE_176400 |
 314                 SNDRV_PCM_RATE_192000,
 315        .channels_min = 2,
 316        .channels_max = 8,
 317        .buffer_bytes_max = 1024 * 1024,
 318        .period_bytes_min = 256,
 319        .period_bytes_max = 8192,       /* ERR004323: must limit to 8k */
 320        .periods_min = 2,
 321        .periods_max = 16,
 322        .fifo_size = 0,
 323};
 324
 325static int dw_hdmi_open(struct snd_pcm_substream *substream)
 326{
 327        struct snd_pcm_runtime *runtime = substream->runtime;
 328        struct snd_dw_hdmi *dw = substream->private_data;
 329        void __iomem *base = dw->data.base;
 330        int ret;
 331
 332        runtime->hw = dw_hdmi_hw;
 333
 334        ret = snd_pcm_hw_constraint_eld(runtime, dw->data.eld);
 335        if (ret < 0)
 336                return ret;
 337
 338        ret = snd_pcm_limit_hw_rates(runtime);
 339        if (ret < 0)
 340                return ret;
 341
 342        ret = snd_pcm_hw_constraint_integer(runtime,
 343                                            SNDRV_PCM_HW_PARAM_PERIODS);
 344        if (ret < 0)
 345                return ret;
 346
 347        /* Limit the buffer size to the size of the preallocated buffer */
 348        ret = snd_pcm_hw_constraint_minmax(runtime,
 349                                           SNDRV_PCM_HW_PARAM_BUFFER_SIZE,
 350                                           0, substream->dma_buffer.bytes);
 351        if (ret < 0)
 352                return ret;
 353
 354        /* Clear FIFO */
 355        writeb_relaxed(HDMI_AHB_DMA_CONF0_SW_FIFO_RST,
 356                       base + HDMI_AHB_DMA_CONF0);
 357
 358        /* Configure interrupt polarities */
 359        writeb_relaxed(~0, base + HDMI_AHB_DMA_POL);
 360        writeb_relaxed(~0, base + HDMI_AHB_DMA_BUFFPOL);
 361
 362        /* Keep interrupts masked, and clear any pending */
 363        writeb_relaxed(~0, base + HDMI_AHB_DMA_MASK);
 364        writeb_relaxed(~0, base + HDMI_IH_AHBDMAAUD_STAT0);
 365
 366        ret = request_irq(dw->data.irq, snd_dw_hdmi_irq, IRQF_SHARED,
 367                          "dw-hdmi-audio", dw);
 368        if (ret)
 369                return ret;
 370
 371        /* Un-mute done interrupt */
 372        writeb_relaxed(HDMI_IH_MUTE_AHBDMAAUD_STAT0_ALL &
 373                       ~HDMI_IH_MUTE_AHBDMAAUD_STAT0_DONE,
 374                       base + HDMI_IH_MUTE_AHBDMAAUD_STAT0);
 375
 376        return 0;
 377}
 378
 379static int dw_hdmi_close(struct snd_pcm_substream *substream)
 380{
 381        struct snd_dw_hdmi *dw = substream->private_data;
 382
 383        /* Mute all interrupts */
 384        writeb_relaxed(HDMI_IH_MUTE_AHBDMAAUD_STAT0_ALL,
 385                       dw->data.base + HDMI_IH_MUTE_AHBDMAAUD_STAT0);
 386
 387        free_irq(dw->data.irq, dw);
 388
 389        return 0;
 390}
 391
 392static int dw_hdmi_hw_free(struct snd_pcm_substream *substream)
 393{
 394        return snd_pcm_lib_free_vmalloc_buffer(substream);
 395}
 396
 397static int dw_hdmi_hw_params(struct snd_pcm_substream *substream,
 398        struct snd_pcm_hw_params *params)
 399{
 400        /* Allocate the PCM runtime buffer, which is exposed to userspace. */
 401        return snd_pcm_lib_alloc_vmalloc_buffer(substream,
 402                                                params_buffer_bytes(params));
 403}
 404
 405static int dw_hdmi_prepare(struct snd_pcm_substream *substream)
 406{
 407        struct snd_pcm_runtime *runtime = substream->runtime;
 408        struct snd_dw_hdmi *dw = substream->private_data;
 409        u8 threshold, conf0, conf1, layout, ca;
 410
 411        /* Setup as per 3.0.5 FSL 4.1.0 BSP */
 412        switch (dw->revision) {
 413        case 0x0a:
 414                conf0 = HDMI_AHB_DMA_CONF0_BURST_MODE |
 415                        HDMI_AHB_DMA_CONF0_INCR4;
 416                if (runtime->channels == 2)
 417                        threshold = 126;
 418                else
 419                        threshold = 124;
 420                break;
 421        case 0x1a:
 422                conf0 = HDMI_AHB_DMA_CONF0_BURST_MODE |
 423                        HDMI_AHB_DMA_CONF0_INCR8;
 424                threshold = 128;
 425                break;
 426        default:
 427                /* NOTREACHED */
 428                return -EINVAL;
 429        }
 430
 431        dw_hdmi_set_sample_rate(dw->data.hdmi, runtime->rate);
 432
 433        /* Minimum number of bytes in the fifo. */
 434        runtime->hw.fifo_size = threshold * 32;
 435
 436        conf0 |= HDMI_AHB_DMA_CONF0_EN_HLOCK;
 437        conf1 = default_hdmi_channel_config[runtime->channels - 2].conf1;
 438        ca = default_hdmi_channel_config[runtime->channels - 2].ca;
 439
 440        /*
 441         * For >2 channel PCM audio, we need to select layout 1
 442         * and set an appropriate channel map.
 443         */
 444        if (runtime->channels > 2)
 445                layout = HDMI_FC_AUDSCONF_LAYOUT1;
 446        else
 447                layout = HDMI_FC_AUDSCONF_LAYOUT0;
 448
 449        writeb_relaxed(threshold, dw->data.base + HDMI_AHB_DMA_THRSLD);
 450        writeb_relaxed(conf0, dw->data.base + HDMI_AHB_DMA_CONF0);
 451        writeb_relaxed(conf1, dw->data.base + HDMI_AHB_DMA_CONF1);
 452        writeb_relaxed(layout, dw->data.base + HDMI_FC_AUDSCONF);
 453        writeb_relaxed(ca, dw->data.base + HDMI_FC_AUDICONF2);
 454
 455        switch (runtime->format) {
 456        case SNDRV_PCM_FORMAT_IEC958_SUBFRAME_LE:
 457                dw->reformat = dw_hdmi_reformat_iec958;
 458                break;
 459        case SNDRV_PCM_FORMAT_S24_LE:
 460                dw_hdmi_create_cs(dw, runtime);
 461                dw->reformat = dw_hdmi_reformat_s24;
 462                break;
 463        }
 464        dw->iec_offset = 0;
 465        dw->channels = runtime->channels;
 466        dw->buf_src  = runtime->dma_area;
 467        dw->buf_dst  = substream->dma_buffer.area;
 468        dw->buf_addr = substream->dma_buffer.addr;
 469        dw->buf_period = snd_pcm_lib_period_bytes(substream);
 470        dw->buf_size = snd_pcm_lib_buffer_bytes(substream);
 471
 472        return 0;
 473}
 474
 475static int dw_hdmi_trigger(struct snd_pcm_substream *substream, int cmd)
 476{
 477        struct snd_dw_hdmi *dw = substream->private_data;
 478        unsigned long flags;
 479        int ret = 0;
 480
 481        switch (cmd) {
 482        case SNDRV_PCM_TRIGGER_START:
 483                spin_lock_irqsave(&dw->lock, flags);
 484                dw->buf_offset = 0;
 485                dw->substream = substream;
 486                dw_hdmi_start_dma(dw);
 487                dw_hdmi_audio_enable(dw->data.hdmi);
 488                spin_unlock_irqrestore(&dw->lock, flags);
 489                substream->runtime->delay = substream->runtime->period_size;
 490                break;
 491
 492        case SNDRV_PCM_TRIGGER_STOP:
 493                spin_lock_irqsave(&dw->lock, flags);
 494                dw->substream = NULL;
 495                dw_hdmi_stop_dma(dw);
 496                dw_hdmi_audio_disable(dw->data.hdmi);
 497                spin_unlock_irqrestore(&dw->lock, flags);
 498                break;
 499
 500        default:
 501                ret = -EINVAL;
 502                break;
 503        }
 504
 505        return ret;
 506}
 507
 508static snd_pcm_uframes_t dw_hdmi_pointer(struct snd_pcm_substream *substream)
 509{
 510        struct snd_pcm_runtime *runtime = substream->runtime;
 511        struct snd_dw_hdmi *dw = substream->private_data;
 512
 513        /*
 514         * We are unable to report the exact hardware position as
 515         * reading the 32-bit DMA position using 8-bit reads is racy.
 516         */
 517        return bytes_to_frames(runtime, dw->buf_offset);
 518}
 519
 520static const struct snd_pcm_ops snd_dw_hdmi_ops = {
 521        .open = dw_hdmi_open,
 522        .close = dw_hdmi_close,
 523        .ioctl = snd_pcm_lib_ioctl,
 524        .hw_params = dw_hdmi_hw_params,
 525        .hw_free = dw_hdmi_hw_free,
 526        .prepare = dw_hdmi_prepare,
 527        .trigger = dw_hdmi_trigger,
 528        .pointer = dw_hdmi_pointer,
 529        .page = snd_pcm_lib_get_vmalloc_page,
 530};
 531
 532static int snd_dw_hdmi_probe(struct platform_device *pdev)
 533{
 534        const struct dw_hdmi_audio_data *data = pdev->dev.platform_data;
 535        struct device *dev = pdev->dev.parent;
 536        struct snd_dw_hdmi *dw;
 537        struct snd_card *card;
 538        struct snd_pcm *pcm;
 539        unsigned revision;
 540        int ret;
 541
 542        writeb_relaxed(HDMI_IH_MUTE_AHBDMAAUD_STAT0_ALL,
 543                       data->base + HDMI_IH_MUTE_AHBDMAAUD_STAT0);
 544        revision = readb_relaxed(data->base + HDMI_REVISION_ID);
 545        if (revision != 0x0a && revision != 0x1a) {
 546                dev_err(dev, "dw-hdmi-audio: unknown revision 0x%02x\n",
 547                        revision);
 548                return -ENXIO;
 549        }
 550
 551        ret = snd_card_new(dev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1,
 552                              THIS_MODULE, sizeof(struct snd_dw_hdmi), &card);
 553        if (ret < 0)
 554                return ret;
 555
 556        strlcpy(card->driver, DRIVER_NAME, sizeof(card->driver));
 557        strlcpy(card->shortname, "DW-HDMI", sizeof(card->shortname));
 558        snprintf(card->longname, sizeof(card->longname),
 559                 "%s rev 0x%02x, irq %d", card->shortname, revision,
 560                 data->irq);
 561
 562        dw = card->private_data;
 563        dw->card = card;
 564        dw->data = *data;
 565        dw->revision = revision;
 566
 567        spin_lock_init(&dw->lock);
 568
 569        ret = snd_pcm_new(card, "DW HDMI", 0, 1, 0, &pcm);
 570        if (ret < 0)
 571                goto err;
 572
 573        dw->pcm = pcm;
 574        pcm->private_data = dw;
 575        strlcpy(pcm->name, DRIVER_NAME, sizeof(pcm->name));
 576        snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &snd_dw_hdmi_ops);
 577
 578        /*
 579         * To support 8-channel 96kHz audio reliably, we need 512k
 580         * to satisfy alsa with our restricted period (ERR004323).
 581         */
 582        snd_pcm_lib_preallocate_pages_for_all(pcm, SNDRV_DMA_TYPE_DEV,
 583                        dev, 128 * 1024, 1024 * 1024);
 584
 585        ret = snd_card_register(card);
 586        if (ret < 0)
 587                goto err;
 588
 589        platform_set_drvdata(pdev, dw);
 590
 591        return 0;
 592
 593err:
 594        snd_card_free(card);
 595        return ret;
 596}
 597
 598static int snd_dw_hdmi_remove(struct platform_device *pdev)
 599{
 600        struct snd_dw_hdmi *dw = platform_get_drvdata(pdev);
 601
 602        snd_card_free(dw->card);
 603
 604        return 0;
 605}
 606
 607#if defined(CONFIG_PM_SLEEP) && defined(IS_NOT_BROKEN)
 608/*
 609 * This code is fine, but requires implementation in the dw_hdmi_trigger()
 610 * method which is currently missing as I have no way to test this.
 611 */
 612static int snd_dw_hdmi_suspend(struct device *dev)
 613{
 614        struct snd_dw_hdmi *dw = dev_get_drvdata(dev);
 615
 616        snd_power_change_state(dw->card, SNDRV_CTL_POWER_D3cold);
 617        snd_pcm_suspend_all(dw->pcm);
 618
 619        return 0;
 620}
 621
 622static int snd_dw_hdmi_resume(struct device *dev)
 623{
 624        struct snd_dw_hdmi *dw = dev_get_drvdata(dev);
 625
 626        snd_power_change_state(dw->card, SNDRV_CTL_POWER_D0);
 627
 628        return 0;
 629}
 630
 631static SIMPLE_DEV_PM_OPS(snd_dw_hdmi_pm, snd_dw_hdmi_suspend,
 632                         snd_dw_hdmi_resume);
 633#define PM_OPS &snd_dw_hdmi_pm
 634#else
 635#define PM_OPS NULL
 636#endif
 637
 638static struct platform_driver snd_dw_hdmi_driver = {
 639        .probe  = snd_dw_hdmi_probe,
 640        .remove = snd_dw_hdmi_remove,
 641        .driver = {
 642                .name = DRIVER_NAME,
 643                .pm = PM_OPS,
 644        },
 645};
 646
 647module_platform_driver(snd_dw_hdmi_driver);
 648
 649MODULE_AUTHOR("Russell King <rmk+kernel@arm.linux.org.uk>");
 650MODULE_DESCRIPTION("Synopsis Designware HDMI AHB ALSA interface");
 651MODULE_LICENSE("GPL v2");
 652MODULE_ALIAS("platform:" DRIVER_NAME);
 653