1d2912cb1SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
235dc8aabSLaurent Pinchart /*
335dc8aabSLaurent Pinchart  * DesignWare HDMI audio driver
435dc8aabSLaurent Pinchart  *
535dc8aabSLaurent Pinchart  * Written and tested against the Designware HDMI Tx found in iMX6.
635dc8aabSLaurent Pinchart  */
735dc8aabSLaurent Pinchart #include <linux/io.h>
835dc8aabSLaurent Pinchart #include <linux/interrupt.h>
935dc8aabSLaurent Pinchart #include <linux/module.h>
1035dc8aabSLaurent Pinchart #include <linux/platform_device.h>
1135dc8aabSLaurent Pinchart #include <drm/bridge/dw_hdmi.h>
1235dc8aabSLaurent Pinchart #include <drm/drm_edid.h>
1335dc8aabSLaurent Pinchart 
1435dc8aabSLaurent Pinchart #include <sound/asoundef.h>
1535dc8aabSLaurent Pinchart #include <sound/core.h>
1635dc8aabSLaurent Pinchart #include <sound/initval.h>
1735dc8aabSLaurent Pinchart #include <sound/pcm.h>
1835dc8aabSLaurent Pinchart #include <sound/pcm_drm_eld.h>
1935dc8aabSLaurent Pinchart #include <sound/pcm_iec958.h>
2035dc8aabSLaurent Pinchart 
2135dc8aabSLaurent Pinchart #include "dw-hdmi-audio.h"
2235dc8aabSLaurent Pinchart 
2335dc8aabSLaurent Pinchart #define DRIVER_NAME "dw-hdmi-ahb-audio"
2435dc8aabSLaurent Pinchart 
2535dc8aabSLaurent Pinchart /* Provide some bits rather than bit offsets */
2635dc8aabSLaurent Pinchart enum {
2735dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_CONF0_SW_FIFO_RST = BIT(7),
2835dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_CONF0_EN_HLOCK = BIT(3),
2935dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_START_START = BIT(0),
3035dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_STOP_STOP = BIT(0),
3135dc8aabSLaurent Pinchart 	HDMI_IH_MUTE_AHBDMAAUD_STAT0_ERROR = BIT(5),
3235dc8aabSLaurent Pinchart 	HDMI_IH_MUTE_AHBDMAAUD_STAT0_LOST = BIT(4),
3335dc8aabSLaurent Pinchart 	HDMI_IH_MUTE_AHBDMAAUD_STAT0_RETRY = BIT(3),
3435dc8aabSLaurent Pinchart 	HDMI_IH_MUTE_AHBDMAAUD_STAT0_DONE = BIT(2),
3535dc8aabSLaurent Pinchart 	HDMI_IH_MUTE_AHBDMAAUD_STAT0_BUFFFULL = BIT(1),
3635dc8aabSLaurent Pinchart 	HDMI_IH_MUTE_AHBDMAAUD_STAT0_BUFFEMPTY = BIT(0),
3735dc8aabSLaurent Pinchart 	HDMI_IH_MUTE_AHBDMAAUD_STAT0_ALL =
3835dc8aabSLaurent Pinchart 		HDMI_IH_MUTE_AHBDMAAUD_STAT0_ERROR |
3935dc8aabSLaurent Pinchart 		HDMI_IH_MUTE_AHBDMAAUD_STAT0_LOST |
4035dc8aabSLaurent Pinchart 		HDMI_IH_MUTE_AHBDMAAUD_STAT0_RETRY |
4135dc8aabSLaurent Pinchart 		HDMI_IH_MUTE_AHBDMAAUD_STAT0_DONE |
4235dc8aabSLaurent Pinchart 		HDMI_IH_MUTE_AHBDMAAUD_STAT0_BUFFFULL |
4335dc8aabSLaurent Pinchart 		HDMI_IH_MUTE_AHBDMAAUD_STAT0_BUFFEMPTY,
4435dc8aabSLaurent Pinchart 	HDMI_IH_AHBDMAAUD_STAT0_ERROR = BIT(5),
4535dc8aabSLaurent Pinchart 	HDMI_IH_AHBDMAAUD_STAT0_LOST = BIT(4),
4635dc8aabSLaurent Pinchart 	HDMI_IH_AHBDMAAUD_STAT0_RETRY = BIT(3),
4735dc8aabSLaurent Pinchart 	HDMI_IH_AHBDMAAUD_STAT0_DONE = BIT(2),
4835dc8aabSLaurent Pinchart 	HDMI_IH_AHBDMAAUD_STAT0_BUFFFULL = BIT(1),
4935dc8aabSLaurent Pinchart 	HDMI_IH_AHBDMAAUD_STAT0_BUFFEMPTY = BIT(0),
5035dc8aabSLaurent Pinchart 	HDMI_IH_AHBDMAAUD_STAT0_ALL =
5135dc8aabSLaurent Pinchart 		HDMI_IH_AHBDMAAUD_STAT0_ERROR |
5235dc8aabSLaurent Pinchart 		HDMI_IH_AHBDMAAUD_STAT0_LOST |
5335dc8aabSLaurent Pinchart 		HDMI_IH_AHBDMAAUD_STAT0_RETRY |
5435dc8aabSLaurent Pinchart 		HDMI_IH_AHBDMAAUD_STAT0_DONE |
5535dc8aabSLaurent Pinchart 		HDMI_IH_AHBDMAAUD_STAT0_BUFFFULL |
5635dc8aabSLaurent Pinchart 		HDMI_IH_AHBDMAAUD_STAT0_BUFFEMPTY,
5735dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_CONF0_INCR16 = 2 << 1,
5835dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_CONF0_INCR8 = 1 << 1,
5935dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_CONF0_INCR4 = 0,
6035dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_CONF0_BURST_MODE = BIT(0),
6135dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_MASK_DONE = BIT(7),
6235dc8aabSLaurent Pinchart 
6335dc8aabSLaurent Pinchart 	HDMI_REVISION_ID = 0x0001,
6435dc8aabSLaurent Pinchart 	HDMI_IH_AHBDMAAUD_STAT0 = 0x0109,
6535dc8aabSLaurent Pinchart 	HDMI_IH_MUTE_AHBDMAAUD_STAT0 = 0x0189,
6635dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_CONF0 = 0x3600,
6735dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_START = 0x3601,
6835dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_STOP = 0x3602,
6935dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_THRSLD = 0x3603,
7035dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_STRADDR0 = 0x3604,
7135dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_STPADDR0 = 0x3608,
7235dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_MASK = 0x3614,
7335dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_POL = 0x3615,
7435dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_CONF1 = 0x3616,
7535dc8aabSLaurent Pinchart 	HDMI_AHB_DMA_BUFFPOL = 0x361a,
7635dc8aabSLaurent Pinchart };
7735dc8aabSLaurent Pinchart 
7835dc8aabSLaurent Pinchart struct dw_hdmi_channel_conf {
7935dc8aabSLaurent Pinchart 	u8 conf1;
8035dc8aabSLaurent Pinchart 	u8 ca;
8135dc8aabSLaurent Pinchart };
8235dc8aabSLaurent Pinchart 
8335dc8aabSLaurent Pinchart /*
8435dc8aabSLaurent Pinchart  * The default mapping of ALSA channels to HDMI channels and speaker
8535dc8aabSLaurent Pinchart  * allocation bits.  Note that we can't do channel remapping here -
8635dc8aabSLaurent Pinchart  * channels must be in the same order.
8735dc8aabSLaurent Pinchart  *
8835dc8aabSLaurent Pinchart  * Mappings for alsa-lib pcm/surround*.conf files:
8935dc8aabSLaurent Pinchart  *
9035dc8aabSLaurent Pinchart  *		Front	Sur4.0	Sur4.1	Sur5.0	Sur5.1	Sur7.1
9135dc8aabSLaurent Pinchart  * Channels	2	4	6	6	6	8
9235dc8aabSLaurent Pinchart  *
9335dc8aabSLaurent Pinchart  * Our mapping from ALSA channel to CEA686D speaker name and HDMI channel:
9435dc8aabSLaurent Pinchart  *
9535dc8aabSLaurent Pinchart  *				Number of ALSA channels
9635dc8aabSLaurent Pinchart  * ALSA Channel	2	3	4	5	6	7	8
9735dc8aabSLaurent Pinchart  * 0		FL:0	=	=	=	=	=	=
9835dc8aabSLaurent Pinchart  * 1		FR:1	=	=	=	=	=	=
9935dc8aabSLaurent Pinchart  * 2			FC:3	RL:4	LFE:2	=	=	=
10035dc8aabSLaurent Pinchart  * 3				RR:5	RL:4	FC:3	=	=
10135dc8aabSLaurent Pinchart  * 4					RR:5	RL:4	=	=
10235dc8aabSLaurent Pinchart  * 5						RR:5	=	=
10335dc8aabSLaurent Pinchart  * 6							RC:6	=
10435dc8aabSLaurent Pinchart  * 7							RLC/FRC	RLC/FRC
10535dc8aabSLaurent Pinchart  */
10635dc8aabSLaurent Pinchart static struct dw_hdmi_channel_conf default_hdmi_channel_config[7] = {
10735dc8aabSLaurent Pinchart 	{ 0x03, 0x00 },	/* FL,FR */
10835dc8aabSLaurent Pinchart 	{ 0x0b, 0x02 },	/* FL,FR,FC */
10935dc8aabSLaurent Pinchart 	{ 0x33, 0x08 },	/* FL,FR,RL,RR */
11035dc8aabSLaurent Pinchart 	{ 0x37, 0x09 },	/* FL,FR,LFE,RL,RR */
11135dc8aabSLaurent Pinchart 	{ 0x3f, 0x0b },	/* FL,FR,LFE,FC,RL,RR */
11235dc8aabSLaurent Pinchart 	{ 0x7f, 0x0f },	/* FL,FR,LFE,FC,RL,RR,RC */
11335dc8aabSLaurent Pinchart 	{ 0xff, 0x13 },	/* FL,FR,LFE,FC,RL,RR,[FR]RC,[FR]LC */
11435dc8aabSLaurent Pinchart };
11535dc8aabSLaurent Pinchart 
11635dc8aabSLaurent Pinchart struct snd_dw_hdmi {
11735dc8aabSLaurent Pinchart 	struct snd_card *card;
11835dc8aabSLaurent Pinchart 	struct snd_pcm *pcm;
11935dc8aabSLaurent Pinchart 	spinlock_t lock;
12035dc8aabSLaurent Pinchart 	struct dw_hdmi_audio_data data;
12135dc8aabSLaurent Pinchart 	struct snd_pcm_substream *substream;
12235dc8aabSLaurent Pinchart 	void (*reformat)(struct snd_dw_hdmi *, size_t, size_t);
12335dc8aabSLaurent Pinchart 	void *buf_src;
12435dc8aabSLaurent Pinchart 	void *buf_dst;
12535dc8aabSLaurent Pinchart 	dma_addr_t buf_addr;
12635dc8aabSLaurent Pinchart 	unsigned buf_offset;
12735dc8aabSLaurent Pinchart 	unsigned buf_period;
12835dc8aabSLaurent Pinchart 	unsigned buf_size;
12935dc8aabSLaurent Pinchart 	unsigned channels;
13035dc8aabSLaurent Pinchart 	u8 revision;
13135dc8aabSLaurent Pinchart 	u8 iec_offset;
13235dc8aabSLaurent Pinchart 	u8 cs[192][8];
13335dc8aabSLaurent Pinchart };
13435dc8aabSLaurent Pinchart 
dw_hdmi_writel(u32 val,void __iomem * ptr)13535dc8aabSLaurent Pinchart static void dw_hdmi_writel(u32 val, void __iomem *ptr)
13635dc8aabSLaurent Pinchart {
13735dc8aabSLaurent Pinchart 	writeb_relaxed(val, ptr);
13835dc8aabSLaurent Pinchart 	writeb_relaxed(val >> 8, ptr + 1);
13935dc8aabSLaurent Pinchart 	writeb_relaxed(val >> 16, ptr + 2);
14035dc8aabSLaurent Pinchart 	writeb_relaxed(val >> 24, ptr + 3);
14135dc8aabSLaurent Pinchart }
14235dc8aabSLaurent Pinchart 
14335dc8aabSLaurent Pinchart /*
14435dc8aabSLaurent Pinchart  * Convert to hardware format: The userspace buffer contains IEC958 samples,
14535dc8aabSLaurent Pinchart  * with the PCUV bits in bits 31..28 and audio samples in bits 27..4.  We
14635dc8aabSLaurent Pinchart  * need these to be in bits 27..24, with the IEC B bit in bit 28, and audio
14735dc8aabSLaurent Pinchart  * samples in 23..0.
14835dc8aabSLaurent Pinchart  *
14935dc8aabSLaurent Pinchart  * Default preamble in bits 3..0: 8 = block start, 4 = even 2 = odd
15035dc8aabSLaurent Pinchart  *
15135dc8aabSLaurent Pinchart  * Ideally, we could do with having the data properly formatted in userspace.
15235dc8aabSLaurent Pinchart  */
dw_hdmi_reformat_iec958(struct snd_dw_hdmi * dw,size_t offset,size_t bytes)15335dc8aabSLaurent Pinchart static void dw_hdmi_reformat_iec958(struct snd_dw_hdmi *dw,
15435dc8aabSLaurent Pinchart 	size_t offset, size_t bytes)
15535dc8aabSLaurent Pinchart {
15635dc8aabSLaurent Pinchart 	u32 *src = dw->buf_src + offset;
15735dc8aabSLaurent Pinchart 	u32 *dst = dw->buf_dst + offset;
15835dc8aabSLaurent Pinchart 	u32 *end = dw->buf_src + offset + bytes;
15935dc8aabSLaurent Pinchart 
16035dc8aabSLaurent Pinchart 	do {
16135dc8aabSLaurent Pinchart 		u32 b, sample = *src++;
16235dc8aabSLaurent Pinchart 
16335dc8aabSLaurent Pinchart 		b = (sample & 8) << (28 - 3);
16435dc8aabSLaurent Pinchart 
16535dc8aabSLaurent Pinchart 		sample >>= 4;
16635dc8aabSLaurent Pinchart 
16735dc8aabSLaurent Pinchart 		*dst++ = sample | b;
16835dc8aabSLaurent Pinchart 	} while (src < end);
16935dc8aabSLaurent Pinchart }
17035dc8aabSLaurent Pinchart 
parity(u32 sample)17135dc8aabSLaurent Pinchart static u32 parity(u32 sample)
17235dc8aabSLaurent Pinchart {
17335dc8aabSLaurent Pinchart 	sample ^= sample >> 16;
17435dc8aabSLaurent Pinchart 	sample ^= sample >> 8;
17535dc8aabSLaurent Pinchart 	sample ^= sample >> 4;
17635dc8aabSLaurent Pinchart 	sample ^= sample >> 2;
17735dc8aabSLaurent Pinchart 	sample ^= sample >> 1;
17835dc8aabSLaurent Pinchart 	return (sample & 1) << 27;
17935dc8aabSLaurent Pinchart }
18035dc8aabSLaurent Pinchart 
dw_hdmi_reformat_s24(struct snd_dw_hdmi * dw,size_t offset,size_t bytes)18135dc8aabSLaurent Pinchart static void dw_hdmi_reformat_s24(struct snd_dw_hdmi *dw,
18235dc8aabSLaurent Pinchart 	size_t offset, size_t bytes)
18335dc8aabSLaurent Pinchart {
18435dc8aabSLaurent Pinchart 	u32 *src = dw->buf_src + offset;
18535dc8aabSLaurent Pinchart 	u32 *dst = dw->buf_dst + offset;
18635dc8aabSLaurent Pinchart 	u32 *end = dw->buf_src + offset + bytes;
18735dc8aabSLaurent Pinchart 
18835dc8aabSLaurent Pinchart 	do {
18935dc8aabSLaurent Pinchart 		unsigned i;
19035dc8aabSLaurent Pinchart 		u8 *cs;
19135dc8aabSLaurent Pinchart 
19235dc8aabSLaurent Pinchart 		cs = dw->cs[dw->iec_offset++];
19335dc8aabSLaurent Pinchart 		if (dw->iec_offset >= 192)
19435dc8aabSLaurent Pinchart 			dw->iec_offset = 0;
19535dc8aabSLaurent Pinchart 
19635dc8aabSLaurent Pinchart 		i = dw->channels;
19735dc8aabSLaurent Pinchart 		do {
19835dc8aabSLaurent Pinchart 			u32 sample = *src++;
19935dc8aabSLaurent Pinchart 
20035dc8aabSLaurent Pinchart 			sample &= ~0xff000000;
20135dc8aabSLaurent Pinchart 			sample |= *cs++ << 24;
20235dc8aabSLaurent Pinchart 			sample |= parity(sample & ~0xf8000000);
20335dc8aabSLaurent Pinchart 
20435dc8aabSLaurent Pinchart 			*dst++ = sample;
20535dc8aabSLaurent Pinchart 		} while (--i);
20635dc8aabSLaurent Pinchart 	} while (src < end);
20735dc8aabSLaurent Pinchart }
20835dc8aabSLaurent Pinchart 
dw_hdmi_create_cs(struct snd_dw_hdmi * dw,struct snd_pcm_runtime * runtime)20935dc8aabSLaurent Pinchart static void dw_hdmi_create_cs(struct snd_dw_hdmi *dw,
21035dc8aabSLaurent Pinchart 	struct snd_pcm_runtime *runtime)
21135dc8aabSLaurent Pinchart {
21235dc8aabSLaurent Pinchart 	u8 cs[4];
21335dc8aabSLaurent Pinchart 	unsigned ch, i, j;
21435dc8aabSLaurent Pinchart 
21535dc8aabSLaurent Pinchart 	snd_pcm_create_iec958_consumer(runtime, cs, sizeof(cs));
21635dc8aabSLaurent Pinchart 
21735dc8aabSLaurent Pinchart 	memset(dw->cs, 0, sizeof(dw->cs));
21835dc8aabSLaurent Pinchart 
21935dc8aabSLaurent Pinchart 	for (ch = 0; ch < 8; ch++) {
22035dc8aabSLaurent Pinchart 		cs[2] &= ~IEC958_AES2_CON_CHANNEL;
22135dc8aabSLaurent Pinchart 		cs[2] |= (ch + 1) << 4;
22235dc8aabSLaurent Pinchart 
22335dc8aabSLaurent Pinchart 		for (i = 0; i < ARRAY_SIZE(cs); i++) {
22435dc8aabSLaurent Pinchart 			unsigned c = cs[i];
22535dc8aabSLaurent Pinchart 
22635dc8aabSLaurent Pinchart 			for (j = 0; j < 8; j++, c >>= 1)
22735dc8aabSLaurent Pinchart 				dw->cs[i * 8 + j][ch] = (c & 1) << 2;
22835dc8aabSLaurent Pinchart 		}
22935dc8aabSLaurent Pinchart 	}
23035dc8aabSLaurent Pinchart 	dw->cs[0][0] |= BIT(4);
23135dc8aabSLaurent Pinchart }
23235dc8aabSLaurent Pinchart 
dw_hdmi_start_dma(struct snd_dw_hdmi * dw)23335dc8aabSLaurent Pinchart static void dw_hdmi_start_dma(struct snd_dw_hdmi *dw)
23435dc8aabSLaurent Pinchart {
23535dc8aabSLaurent Pinchart 	void __iomem *base = dw->data.base;
23635dc8aabSLaurent Pinchart 	unsigned offset = dw->buf_offset;
23735dc8aabSLaurent Pinchart 	unsigned period = dw->buf_period;
23835dc8aabSLaurent Pinchart 	u32 start, stop;
23935dc8aabSLaurent Pinchart 
24035dc8aabSLaurent Pinchart 	dw->reformat(dw, offset, period);
24135dc8aabSLaurent Pinchart 
24235dc8aabSLaurent Pinchart 	/* Clear all irqs before enabling irqs and starting DMA */
24335dc8aabSLaurent Pinchart 	writeb_relaxed(HDMI_IH_AHBDMAAUD_STAT0_ALL,
24435dc8aabSLaurent Pinchart 		       base + HDMI_IH_AHBDMAAUD_STAT0);
24535dc8aabSLaurent Pinchart 
24635dc8aabSLaurent Pinchart 	start = dw->buf_addr + offset;
24735dc8aabSLaurent Pinchart 	stop = start + period - 1;
24835dc8aabSLaurent Pinchart 
24935dc8aabSLaurent Pinchart 	/* Setup the hardware start/stop addresses */
25035dc8aabSLaurent Pinchart 	dw_hdmi_writel(start, base + HDMI_AHB_DMA_STRADDR0);
25135dc8aabSLaurent Pinchart 	dw_hdmi_writel(stop, base + HDMI_AHB_DMA_STPADDR0);
25235dc8aabSLaurent Pinchart 
25335dc8aabSLaurent Pinchart 	writeb_relaxed((u8)~HDMI_AHB_DMA_MASK_DONE, base + HDMI_AHB_DMA_MASK);
25435dc8aabSLaurent Pinchart 	writeb(HDMI_AHB_DMA_START_START, base + HDMI_AHB_DMA_START);
25535dc8aabSLaurent Pinchart 
25635dc8aabSLaurent Pinchart 	offset += period;
25735dc8aabSLaurent Pinchart 	if (offset >= dw->buf_size)
25835dc8aabSLaurent Pinchart 		offset = 0;
25935dc8aabSLaurent Pinchart 	dw->buf_offset = offset;
26035dc8aabSLaurent Pinchart }
26135dc8aabSLaurent Pinchart 
dw_hdmi_stop_dma(struct snd_dw_hdmi * dw)26235dc8aabSLaurent Pinchart static void dw_hdmi_stop_dma(struct snd_dw_hdmi *dw)
26335dc8aabSLaurent Pinchart {
26435dc8aabSLaurent Pinchart 	/* Disable interrupts before disabling DMA */
26535dc8aabSLaurent Pinchart 	writeb_relaxed(~0, dw->data.base + HDMI_AHB_DMA_MASK);
26635dc8aabSLaurent Pinchart 	writeb_relaxed(HDMI_AHB_DMA_STOP_STOP, dw->data.base + HDMI_AHB_DMA_STOP);
26735dc8aabSLaurent Pinchart }
26835dc8aabSLaurent Pinchart 
snd_dw_hdmi_irq(int irq,void * data)26935dc8aabSLaurent Pinchart static irqreturn_t snd_dw_hdmi_irq(int irq, void *data)
27035dc8aabSLaurent Pinchart {
27135dc8aabSLaurent Pinchart 	struct snd_dw_hdmi *dw = data;
27235dc8aabSLaurent Pinchart 	struct snd_pcm_substream *substream;
27335dc8aabSLaurent Pinchart 	unsigned stat;
27435dc8aabSLaurent Pinchart 
27535dc8aabSLaurent Pinchart 	stat = readb_relaxed(dw->data.base + HDMI_IH_AHBDMAAUD_STAT0);
27635dc8aabSLaurent Pinchart 	if (!stat)
27735dc8aabSLaurent Pinchart 		return IRQ_NONE;
27835dc8aabSLaurent Pinchart 
27935dc8aabSLaurent Pinchart 	writeb_relaxed(stat, dw->data.base + HDMI_IH_AHBDMAAUD_STAT0);
28035dc8aabSLaurent Pinchart 
28135dc8aabSLaurent Pinchart 	substream = dw->substream;
28235dc8aabSLaurent Pinchart 	if (stat & HDMI_IH_AHBDMAAUD_STAT0_DONE && substream) {
28335dc8aabSLaurent Pinchart 		snd_pcm_period_elapsed(substream);
28435dc8aabSLaurent Pinchart 
28535dc8aabSLaurent Pinchart 		spin_lock(&dw->lock);
28635dc8aabSLaurent Pinchart 		if (dw->substream)
28735dc8aabSLaurent Pinchart 			dw_hdmi_start_dma(dw);
28835dc8aabSLaurent Pinchart 		spin_unlock(&dw->lock);
28935dc8aabSLaurent Pinchart 	}
29035dc8aabSLaurent Pinchart 
29135dc8aabSLaurent Pinchart 	return IRQ_HANDLED;
29235dc8aabSLaurent Pinchart }
29335dc8aabSLaurent Pinchart 
294d969ebe9SJulia Lawall static const struct snd_pcm_hardware dw_hdmi_hw = {
29535dc8aabSLaurent Pinchart 	.info = SNDRV_PCM_INFO_INTERLEAVED |
29635dc8aabSLaurent Pinchart 		SNDRV_PCM_INFO_BLOCK_TRANSFER |
29735dc8aabSLaurent Pinchart 		SNDRV_PCM_INFO_MMAP |
29835dc8aabSLaurent Pinchart 		SNDRV_PCM_INFO_MMAP_VALID,
29935dc8aabSLaurent Pinchart 	.formats = SNDRV_PCM_FMTBIT_IEC958_SUBFRAME_LE |
30035dc8aabSLaurent Pinchart 		   SNDRV_PCM_FMTBIT_S24_LE,
30135dc8aabSLaurent Pinchart 	.rates = SNDRV_PCM_RATE_32000 |
30235dc8aabSLaurent Pinchart 		 SNDRV_PCM_RATE_44100 |
30335dc8aabSLaurent Pinchart 		 SNDRV_PCM_RATE_48000 |
30435dc8aabSLaurent Pinchart 		 SNDRV_PCM_RATE_88200 |
30535dc8aabSLaurent Pinchart 		 SNDRV_PCM_RATE_96000 |
30635dc8aabSLaurent Pinchart 		 SNDRV_PCM_RATE_176400 |
30735dc8aabSLaurent Pinchart 		 SNDRV_PCM_RATE_192000,
30835dc8aabSLaurent Pinchart 	.channels_min = 2,
30935dc8aabSLaurent Pinchart 	.channels_max = 8,
31035dc8aabSLaurent Pinchart 	.buffer_bytes_max = 1024 * 1024,
31135dc8aabSLaurent Pinchart 	.period_bytes_min = 256,
31235dc8aabSLaurent Pinchart 	.period_bytes_max = 8192,	/* ERR004323: must limit to 8k */
31335dc8aabSLaurent Pinchart 	.periods_min = 2,
31435dc8aabSLaurent Pinchart 	.periods_max = 16,
31535dc8aabSLaurent Pinchart 	.fifo_size = 0,
31635dc8aabSLaurent Pinchart };
31735dc8aabSLaurent Pinchart 
dw_hdmi_open(struct snd_pcm_substream * substream)31835dc8aabSLaurent Pinchart static int dw_hdmi_open(struct snd_pcm_substream *substream)
31935dc8aabSLaurent Pinchart {
32035dc8aabSLaurent Pinchart 	struct snd_pcm_runtime *runtime = substream->runtime;
32135dc8aabSLaurent Pinchart 	struct snd_dw_hdmi *dw = substream->private_data;
32235dc8aabSLaurent Pinchart 	void __iomem *base = dw->data.base;
3233f2532d6SNeil Armstrong 	u8 *eld;
32435dc8aabSLaurent Pinchart 	int ret;
32535dc8aabSLaurent Pinchart 
32635dc8aabSLaurent Pinchart 	runtime->hw = dw_hdmi_hw;
32735dc8aabSLaurent Pinchart 
3283f2532d6SNeil Armstrong 	eld = dw->data.get_eld(dw->data.hdmi);
3293f2532d6SNeil Armstrong 	if (eld) {
3303f2532d6SNeil Armstrong 		ret = snd_pcm_hw_constraint_eld(runtime, eld);
33135dc8aabSLaurent Pinchart 		if (ret < 0)
33235dc8aabSLaurent Pinchart 			return ret;
3333f2532d6SNeil Armstrong 	}
33435dc8aabSLaurent Pinchart 
33535dc8aabSLaurent Pinchart 	ret = snd_pcm_limit_hw_rates(runtime);
33635dc8aabSLaurent Pinchart 	if (ret < 0)
33735dc8aabSLaurent Pinchart 		return ret;
33835dc8aabSLaurent Pinchart 
33935dc8aabSLaurent Pinchart 	ret = snd_pcm_hw_constraint_integer(runtime,
34035dc8aabSLaurent Pinchart 					    SNDRV_PCM_HW_PARAM_PERIODS);
34135dc8aabSLaurent Pinchart 	if (ret < 0)
34235dc8aabSLaurent Pinchart 		return ret;
34335dc8aabSLaurent Pinchart 
34435dc8aabSLaurent Pinchart 	/* Limit the buffer size to the size of the preallocated buffer */
34535dc8aabSLaurent Pinchart 	ret = snd_pcm_hw_constraint_minmax(runtime,
34635dc8aabSLaurent Pinchart 					   SNDRV_PCM_HW_PARAM_BUFFER_SIZE,
34735dc8aabSLaurent Pinchart 					   0, substream->dma_buffer.bytes);
34835dc8aabSLaurent Pinchart 	if (ret < 0)
34935dc8aabSLaurent Pinchart 		return ret;
35035dc8aabSLaurent Pinchart 
35135dc8aabSLaurent Pinchart 	/* Clear FIFO */
35235dc8aabSLaurent Pinchart 	writeb_relaxed(HDMI_AHB_DMA_CONF0_SW_FIFO_RST,
35335dc8aabSLaurent Pinchart 		       base + HDMI_AHB_DMA_CONF0);
35435dc8aabSLaurent Pinchart 
35535dc8aabSLaurent Pinchart 	/* Configure interrupt polarities */
35635dc8aabSLaurent Pinchart 	writeb_relaxed(~0, base + HDMI_AHB_DMA_POL);
35735dc8aabSLaurent Pinchart 	writeb_relaxed(~0, base + HDMI_AHB_DMA_BUFFPOL);
35835dc8aabSLaurent Pinchart 
35935dc8aabSLaurent Pinchart 	/* Keep interrupts masked, and clear any pending */
36035dc8aabSLaurent Pinchart 	writeb_relaxed(~0, base + HDMI_AHB_DMA_MASK);
36135dc8aabSLaurent Pinchart 	writeb_relaxed(~0, base + HDMI_IH_AHBDMAAUD_STAT0);
36235dc8aabSLaurent Pinchart 
36335dc8aabSLaurent Pinchart 	ret = request_irq(dw->data.irq, snd_dw_hdmi_irq, IRQF_SHARED,
36435dc8aabSLaurent Pinchart 			  "dw-hdmi-audio", dw);
36535dc8aabSLaurent Pinchart 	if (ret)
36635dc8aabSLaurent Pinchart 		return ret;
36735dc8aabSLaurent Pinchart 
36835dc8aabSLaurent Pinchart 	/* Un-mute done interrupt */
36935dc8aabSLaurent Pinchart 	writeb_relaxed(HDMI_IH_MUTE_AHBDMAAUD_STAT0_ALL &
37035dc8aabSLaurent Pinchart 		       ~HDMI_IH_MUTE_AHBDMAAUD_STAT0_DONE,
37135dc8aabSLaurent Pinchart 		       base + HDMI_IH_MUTE_AHBDMAAUD_STAT0);
37235dc8aabSLaurent Pinchart 
37335dc8aabSLaurent Pinchart 	return 0;
37435dc8aabSLaurent Pinchart }
37535dc8aabSLaurent Pinchart 
dw_hdmi_close(struct snd_pcm_substream * substream)37635dc8aabSLaurent Pinchart static int dw_hdmi_close(struct snd_pcm_substream *substream)
37735dc8aabSLaurent Pinchart {
37835dc8aabSLaurent Pinchart 	struct snd_dw_hdmi *dw = substream->private_data;
37935dc8aabSLaurent Pinchart 
38035dc8aabSLaurent Pinchart 	/* Mute all interrupts */
38135dc8aabSLaurent Pinchart 	writeb_relaxed(HDMI_IH_MUTE_AHBDMAAUD_STAT0_ALL,
38235dc8aabSLaurent Pinchart 		       dw->data.base + HDMI_IH_MUTE_AHBDMAAUD_STAT0);
38335dc8aabSLaurent Pinchart 
38435dc8aabSLaurent Pinchart 	free_irq(dw->data.irq, dw);
38535dc8aabSLaurent Pinchart 
38635dc8aabSLaurent Pinchart 	return 0;
38735dc8aabSLaurent Pinchart }
38835dc8aabSLaurent Pinchart 
dw_hdmi_hw_free(struct snd_pcm_substream * substream)38935dc8aabSLaurent Pinchart static int dw_hdmi_hw_free(struct snd_pcm_substream *substream)
39035dc8aabSLaurent Pinchart {
39135dc8aabSLaurent Pinchart 	return snd_pcm_lib_free_vmalloc_buffer(substream);
39235dc8aabSLaurent Pinchart }
39335dc8aabSLaurent Pinchart 
dw_hdmi_hw_params(struct snd_pcm_substream * substream,struct snd_pcm_hw_params * params)39435dc8aabSLaurent Pinchart static int dw_hdmi_hw_params(struct snd_pcm_substream *substream,
39535dc8aabSLaurent Pinchart 	struct snd_pcm_hw_params *params)
39635dc8aabSLaurent Pinchart {
39735dc8aabSLaurent Pinchart 	/* Allocate the PCM runtime buffer, which is exposed to userspace. */
39835dc8aabSLaurent Pinchart 	return snd_pcm_lib_alloc_vmalloc_buffer(substream,
39935dc8aabSLaurent Pinchart 						params_buffer_bytes(params));
40035dc8aabSLaurent Pinchart }
40135dc8aabSLaurent Pinchart 
dw_hdmi_prepare(struct snd_pcm_substream * substream)40235dc8aabSLaurent Pinchart static int dw_hdmi_prepare(struct snd_pcm_substream *substream)
40335dc8aabSLaurent Pinchart {
40435dc8aabSLaurent Pinchart 	struct snd_pcm_runtime *runtime = substream->runtime;
40535dc8aabSLaurent Pinchart 	struct snd_dw_hdmi *dw = substream->private_data;
4062a2a3d2fSJerome Brunet 	u8 threshold, conf0, conf1, ca;
40735dc8aabSLaurent Pinchart 
40835dc8aabSLaurent Pinchart 	/* Setup as per 3.0.5 FSL 4.1.0 BSP */
40935dc8aabSLaurent Pinchart 	switch (dw->revision) {
41035dc8aabSLaurent Pinchart 	case 0x0a:
41135dc8aabSLaurent Pinchart 		conf0 = HDMI_AHB_DMA_CONF0_BURST_MODE |
41235dc8aabSLaurent Pinchart 			HDMI_AHB_DMA_CONF0_INCR4;
41335dc8aabSLaurent Pinchart 		if (runtime->channels == 2)
41435dc8aabSLaurent Pinchart 			threshold = 126;
41535dc8aabSLaurent Pinchart 		else
41635dc8aabSLaurent Pinchart 			threshold = 124;
41735dc8aabSLaurent Pinchart 		break;
41835dc8aabSLaurent Pinchart 	case 0x1a:
41935dc8aabSLaurent Pinchart 		conf0 = HDMI_AHB_DMA_CONF0_BURST_MODE |
42035dc8aabSLaurent Pinchart 			HDMI_AHB_DMA_CONF0_INCR8;
42135dc8aabSLaurent Pinchart 		threshold = 128;
42235dc8aabSLaurent Pinchart 		break;
42335dc8aabSLaurent Pinchart 	default:
42435dc8aabSLaurent Pinchart 		/* NOTREACHED */
42535dc8aabSLaurent Pinchart 		return -EINVAL;
42635dc8aabSLaurent Pinchart 	}
42735dc8aabSLaurent Pinchart 
42835dc8aabSLaurent Pinchart 	dw_hdmi_set_sample_rate(dw->data.hdmi, runtime->rate);
42935dc8aabSLaurent Pinchart 
43035dc8aabSLaurent Pinchart 	/* Minimum number of bytes in the fifo. */
43135dc8aabSLaurent Pinchart 	runtime->hw.fifo_size = threshold * 32;
43235dc8aabSLaurent Pinchart 
43335dc8aabSLaurent Pinchart 	conf0 |= HDMI_AHB_DMA_CONF0_EN_HLOCK;
43435dc8aabSLaurent Pinchart 	conf1 = default_hdmi_channel_config[runtime->channels - 2].conf1;
43535dc8aabSLaurent Pinchart 	ca = default_hdmi_channel_config[runtime->channels - 2].ca;
43635dc8aabSLaurent Pinchart 
43735dc8aabSLaurent Pinchart 	writeb_relaxed(threshold, dw->data.base + HDMI_AHB_DMA_THRSLD);
43835dc8aabSLaurent Pinchart 	writeb_relaxed(conf0, dw->data.base + HDMI_AHB_DMA_CONF0);
43935dc8aabSLaurent Pinchart 	writeb_relaxed(conf1, dw->data.base + HDMI_AHB_DMA_CONF1);
4402a2a3d2fSJerome Brunet 
4412a2a3d2fSJerome Brunet 	dw_hdmi_set_channel_count(dw->data.hdmi, runtime->channels);
4422a2a3d2fSJerome Brunet 	dw_hdmi_set_channel_allocation(dw->data.hdmi, ca);
44335dc8aabSLaurent Pinchart 
44435dc8aabSLaurent Pinchart 	switch (runtime->format) {
44535dc8aabSLaurent Pinchart 	case SNDRV_PCM_FORMAT_IEC958_SUBFRAME_LE:
44635dc8aabSLaurent Pinchart 		dw->reformat = dw_hdmi_reformat_iec958;
44735dc8aabSLaurent Pinchart 		break;
44835dc8aabSLaurent Pinchart 	case SNDRV_PCM_FORMAT_S24_LE:
44935dc8aabSLaurent Pinchart 		dw_hdmi_create_cs(dw, runtime);
45035dc8aabSLaurent Pinchart 		dw->reformat = dw_hdmi_reformat_s24;
45135dc8aabSLaurent Pinchart 		break;
45235dc8aabSLaurent Pinchart 	}
45335dc8aabSLaurent Pinchart 	dw->iec_offset = 0;
45435dc8aabSLaurent Pinchart 	dw->channels = runtime->channels;
45535dc8aabSLaurent Pinchart 	dw->buf_src  = runtime->dma_area;
45635dc8aabSLaurent Pinchart 	dw->buf_dst  = substream->dma_buffer.area;
45735dc8aabSLaurent Pinchart 	dw->buf_addr = substream->dma_buffer.addr;
45835dc8aabSLaurent Pinchart 	dw->buf_period = snd_pcm_lib_period_bytes(substream);
45935dc8aabSLaurent Pinchart 	dw->buf_size = snd_pcm_lib_buffer_bytes(substream);
46035dc8aabSLaurent Pinchart 
46135dc8aabSLaurent Pinchart 	return 0;
46235dc8aabSLaurent Pinchart }
46335dc8aabSLaurent Pinchart 
dw_hdmi_trigger(struct snd_pcm_substream * substream,int cmd)46435dc8aabSLaurent Pinchart static int dw_hdmi_trigger(struct snd_pcm_substream *substream, int cmd)
46535dc8aabSLaurent Pinchart {
46635dc8aabSLaurent Pinchart 	struct snd_dw_hdmi *dw = substream->private_data;
46735dc8aabSLaurent Pinchart 	unsigned long flags;
46835dc8aabSLaurent Pinchart 	int ret = 0;
46935dc8aabSLaurent Pinchart 
47035dc8aabSLaurent Pinchart 	switch (cmd) {
47135dc8aabSLaurent Pinchart 	case SNDRV_PCM_TRIGGER_START:
47235dc8aabSLaurent Pinchart 		spin_lock_irqsave(&dw->lock, flags);
47335dc8aabSLaurent Pinchart 		dw->buf_offset = 0;
47435dc8aabSLaurent Pinchart 		dw->substream = substream;
47535dc8aabSLaurent Pinchart 		dw_hdmi_start_dma(dw);
47635dc8aabSLaurent Pinchart 		dw_hdmi_audio_enable(dw->data.hdmi);
47735dc8aabSLaurent Pinchart 		spin_unlock_irqrestore(&dw->lock, flags);
47835dc8aabSLaurent Pinchart 		substream->runtime->delay = substream->runtime->period_size;
47935dc8aabSLaurent Pinchart 		break;
48035dc8aabSLaurent Pinchart 
48135dc8aabSLaurent Pinchart 	case SNDRV_PCM_TRIGGER_STOP:
48235dc8aabSLaurent Pinchart 		spin_lock_irqsave(&dw->lock, flags);
48335dc8aabSLaurent Pinchart 		dw->substream = NULL;
48435dc8aabSLaurent Pinchart 		dw_hdmi_stop_dma(dw);
48535dc8aabSLaurent Pinchart 		dw_hdmi_audio_disable(dw->data.hdmi);
48635dc8aabSLaurent Pinchart 		spin_unlock_irqrestore(&dw->lock, flags);
48735dc8aabSLaurent Pinchart 		break;
48835dc8aabSLaurent Pinchart 
48935dc8aabSLaurent Pinchart 	default:
49035dc8aabSLaurent Pinchart 		ret = -EINVAL;
49135dc8aabSLaurent Pinchart 		break;
49235dc8aabSLaurent Pinchart 	}
49335dc8aabSLaurent Pinchart 
49435dc8aabSLaurent Pinchart 	return ret;
49535dc8aabSLaurent Pinchart }
49635dc8aabSLaurent Pinchart 
dw_hdmi_pointer(struct snd_pcm_substream * substream)49735dc8aabSLaurent Pinchart static snd_pcm_uframes_t dw_hdmi_pointer(struct snd_pcm_substream *substream)
49835dc8aabSLaurent Pinchart {
49935dc8aabSLaurent Pinchart 	struct snd_pcm_runtime *runtime = substream->runtime;
50035dc8aabSLaurent Pinchart 	struct snd_dw_hdmi *dw = substream->private_data;
50135dc8aabSLaurent Pinchart 
50235dc8aabSLaurent Pinchart 	/*
50335dc8aabSLaurent Pinchart 	 * We are unable to report the exact hardware position as
50435dc8aabSLaurent Pinchart 	 * reading the 32-bit DMA position using 8-bit reads is racy.
50535dc8aabSLaurent Pinchart 	 */
50635dc8aabSLaurent Pinchart 	return bytes_to_frames(runtime, dw->buf_offset);
50735dc8aabSLaurent Pinchart }
50835dc8aabSLaurent Pinchart 
509b27cc7b5SArvind Yadav static const struct snd_pcm_ops snd_dw_hdmi_ops = {
51035dc8aabSLaurent Pinchart 	.open = dw_hdmi_open,
51135dc8aabSLaurent Pinchart 	.close = dw_hdmi_close,
51235dc8aabSLaurent Pinchart 	.ioctl = snd_pcm_lib_ioctl,
51335dc8aabSLaurent Pinchart 	.hw_params = dw_hdmi_hw_params,
51435dc8aabSLaurent Pinchart 	.hw_free = dw_hdmi_hw_free,
51535dc8aabSLaurent Pinchart 	.prepare = dw_hdmi_prepare,
51635dc8aabSLaurent Pinchart 	.trigger = dw_hdmi_trigger,
51735dc8aabSLaurent Pinchart 	.pointer = dw_hdmi_pointer,
51835dc8aabSLaurent Pinchart 	.page = snd_pcm_lib_get_vmalloc_page,
51935dc8aabSLaurent Pinchart };
52035dc8aabSLaurent Pinchart 
snd_dw_hdmi_probe(struct platform_device * pdev)52135dc8aabSLaurent Pinchart static int snd_dw_hdmi_probe(struct platform_device *pdev)
52235dc8aabSLaurent Pinchart {
52335dc8aabSLaurent Pinchart 	const struct dw_hdmi_audio_data *data = pdev->dev.platform_data;
52435dc8aabSLaurent Pinchart 	struct device *dev = pdev->dev.parent;
52535dc8aabSLaurent Pinchart 	struct snd_dw_hdmi *dw;
52635dc8aabSLaurent Pinchart 	struct snd_card *card;
52735dc8aabSLaurent Pinchart 	struct snd_pcm *pcm;
52835dc8aabSLaurent Pinchart 	unsigned revision;
52935dc8aabSLaurent Pinchart 	int ret;
53035dc8aabSLaurent Pinchart 
53135dc8aabSLaurent Pinchart 	writeb_relaxed(HDMI_IH_MUTE_AHBDMAAUD_STAT0_ALL,
53235dc8aabSLaurent Pinchart 		       data->base + HDMI_IH_MUTE_AHBDMAAUD_STAT0);
53335dc8aabSLaurent Pinchart 	revision = readb_relaxed(data->base + HDMI_REVISION_ID);
53435dc8aabSLaurent Pinchart 	if (revision != 0x0a && revision != 0x1a) {
53535dc8aabSLaurent Pinchart 		dev_err(dev, "dw-hdmi-audio: unknown revision 0x%02x\n",
53635dc8aabSLaurent Pinchart 			revision);
53735dc8aabSLaurent Pinchart 		return -ENXIO;
53835dc8aabSLaurent Pinchart 	}
53935dc8aabSLaurent Pinchart 
54035dc8aabSLaurent Pinchart 	ret = snd_card_new(dev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1,
54135dc8aabSLaurent Pinchart 			      THIS_MODULE, sizeof(struct snd_dw_hdmi), &card);
54235dc8aabSLaurent Pinchart 	if (ret < 0)
54335dc8aabSLaurent Pinchart 		return ret;
54435dc8aabSLaurent Pinchart 
545110ae8a2SMinghao Chi 	strscpy(card->driver, DRIVER_NAME, sizeof(card->driver));
546110ae8a2SMinghao Chi 	strscpy(card->shortname, "DW-HDMI", sizeof(card->shortname));
54735dc8aabSLaurent Pinchart 	snprintf(card->longname, sizeof(card->longname),
54835dc8aabSLaurent Pinchart 		 "%s rev 0x%02x, irq %d", card->shortname, revision,
54935dc8aabSLaurent Pinchart 		 data->irq);
55035dc8aabSLaurent Pinchart 
55135dc8aabSLaurent Pinchart 	dw = card->private_data;
55235dc8aabSLaurent Pinchart 	dw->card = card;
55335dc8aabSLaurent Pinchart 	dw->data = *data;
55435dc8aabSLaurent Pinchart 	dw->revision = revision;
55535dc8aabSLaurent Pinchart 
55635dc8aabSLaurent Pinchart 	spin_lock_init(&dw->lock);
55735dc8aabSLaurent Pinchart 
55835dc8aabSLaurent Pinchart 	ret = snd_pcm_new(card, "DW HDMI", 0, 1, 0, &pcm);
55935dc8aabSLaurent Pinchart 	if (ret < 0)
56035dc8aabSLaurent Pinchart 		goto err;
56135dc8aabSLaurent Pinchart 
56235dc8aabSLaurent Pinchart 	dw->pcm = pcm;
56335dc8aabSLaurent Pinchart 	pcm->private_data = dw;
564110ae8a2SMinghao Chi 	strscpy(pcm->name, DRIVER_NAME, sizeof(pcm->name));
56535dc8aabSLaurent Pinchart 	snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &snd_dw_hdmi_ops);
56635dc8aabSLaurent Pinchart 
56735dc8aabSLaurent Pinchart 	/*
56835dc8aabSLaurent Pinchart 	 * To support 8-channel 96kHz audio reliably, we need 512k
56935dc8aabSLaurent Pinchart 	 * to satisfy alsa with our restricted period (ERR004323).
57035dc8aabSLaurent Pinchart 	 */
57135dc8aabSLaurent Pinchart 	snd_pcm_lib_preallocate_pages_for_all(pcm, SNDRV_DMA_TYPE_DEV,
57235dc8aabSLaurent Pinchart 			dev, 128 * 1024, 1024 * 1024);
57335dc8aabSLaurent Pinchart 
57435dc8aabSLaurent Pinchart 	ret = snd_card_register(card);
57535dc8aabSLaurent Pinchart 	if (ret < 0)
57635dc8aabSLaurent Pinchart 		goto err;
57735dc8aabSLaurent Pinchart 
57835dc8aabSLaurent Pinchart 	platform_set_drvdata(pdev, dw);
57935dc8aabSLaurent Pinchart 
58035dc8aabSLaurent Pinchart 	return 0;
58135dc8aabSLaurent Pinchart 
58235dc8aabSLaurent Pinchart err:
58335dc8aabSLaurent Pinchart 	snd_card_free(card);
58435dc8aabSLaurent Pinchart 	return ret;
58535dc8aabSLaurent Pinchart }
58635dc8aabSLaurent Pinchart 
snd_dw_hdmi_remove(struct platform_device * pdev)587*e85f4368SUwe Kleine-König static void snd_dw_hdmi_remove(struct platform_device *pdev)
58835dc8aabSLaurent Pinchart {
58935dc8aabSLaurent Pinchart 	struct snd_dw_hdmi *dw = platform_get_drvdata(pdev);
59035dc8aabSLaurent Pinchart 
59135dc8aabSLaurent Pinchart 	snd_card_free(dw->card);
59235dc8aabSLaurent Pinchart }
59335dc8aabSLaurent Pinchart 
59435dc8aabSLaurent Pinchart #if defined(CONFIG_PM_SLEEP) && defined(IS_NOT_BROKEN)
59535dc8aabSLaurent Pinchart /*
59635dc8aabSLaurent Pinchart  * This code is fine, but requires implementation in the dw_hdmi_trigger()
59735dc8aabSLaurent Pinchart  * method which is currently missing as I have no way to test this.
59835dc8aabSLaurent Pinchart  */
snd_dw_hdmi_suspend(struct device * dev)59935dc8aabSLaurent Pinchart static int snd_dw_hdmi_suspend(struct device *dev)
60035dc8aabSLaurent Pinchart {
60135dc8aabSLaurent Pinchart 	struct snd_dw_hdmi *dw = dev_get_drvdata(dev);
60235dc8aabSLaurent Pinchart 
60335dc8aabSLaurent Pinchart 	snd_power_change_state(dw->card, SNDRV_CTL_POWER_D3cold);
60435dc8aabSLaurent Pinchart 
60535dc8aabSLaurent Pinchart 	return 0;
60635dc8aabSLaurent Pinchart }
60735dc8aabSLaurent Pinchart 
snd_dw_hdmi_resume(struct device * dev)60835dc8aabSLaurent Pinchart static int snd_dw_hdmi_resume(struct device *dev)
60935dc8aabSLaurent Pinchart {
61035dc8aabSLaurent Pinchart 	struct snd_dw_hdmi *dw = dev_get_drvdata(dev);
61135dc8aabSLaurent Pinchart 
61235dc8aabSLaurent Pinchart 	snd_power_change_state(dw->card, SNDRV_CTL_POWER_D0);
61335dc8aabSLaurent Pinchart 
61435dc8aabSLaurent Pinchart 	return 0;
61535dc8aabSLaurent Pinchart }
61635dc8aabSLaurent Pinchart 
61735dc8aabSLaurent Pinchart static SIMPLE_DEV_PM_OPS(snd_dw_hdmi_pm, snd_dw_hdmi_suspend,
61835dc8aabSLaurent Pinchart 			 snd_dw_hdmi_resume);
61935dc8aabSLaurent Pinchart #define PM_OPS &snd_dw_hdmi_pm
62035dc8aabSLaurent Pinchart #else
62135dc8aabSLaurent Pinchart #define PM_OPS NULL
62235dc8aabSLaurent Pinchart #endif
62335dc8aabSLaurent Pinchart 
62435dc8aabSLaurent Pinchart static struct platform_driver snd_dw_hdmi_driver = {
62535dc8aabSLaurent Pinchart 	.probe	= snd_dw_hdmi_probe,
626*e85f4368SUwe Kleine-König 	.remove_new = snd_dw_hdmi_remove,
62735dc8aabSLaurent Pinchart 	.driver	= {
62835dc8aabSLaurent Pinchart 		.name = DRIVER_NAME,
62935dc8aabSLaurent Pinchart 		.pm = PM_OPS,
63035dc8aabSLaurent Pinchart 	},
63135dc8aabSLaurent Pinchart };
63235dc8aabSLaurent Pinchart 
63335dc8aabSLaurent Pinchart module_platform_driver(snd_dw_hdmi_driver);
63435dc8aabSLaurent Pinchart 
6354f39467eSRussell King MODULE_AUTHOR("Russell King <rmk+kernel@armlinux.org.uk>");
63635dc8aabSLaurent Pinchart MODULE_DESCRIPTION("Synopsis Designware HDMI AHB ALSA interface");
63735dc8aabSLaurent Pinchart MODULE_LICENSE("GPL v2");
63835dc8aabSLaurent Pinchart MODULE_ALIAS("platform:" DRIVER_NAME);
639