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