blob: 034593bf2cd6fcb2f65099d1bc2e85c7df0b26ad [file] [log] [blame]
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301/*
2 * hdac_hdmi.c - ASoc HDA-HDMI codec driver for Intel platforms
3 *
4 * Copyright (C) 2014-2015 Intel Corp
5 * Author: Samreen Nilofer <samreen.nilofer@intel.com>
6 * Subhransu S. Prusty <subhransu.s.prusty@intel.com>
7 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
8 *
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; version 2 of the License.
12 *
13 * This program is distributed in the hope that it will be useful, but
14 * WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * General Public License for more details.
17 *
18 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
19 */
20#include <linux/init.h>
21#include <linux/delay.h>
22#include <linux/module.h>
23#include <linux/pm_runtime.h>
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +053024#include <linux/hdmi.h>
Subhransu S. Prusty2428bca2016-02-12 07:46:02 +053025#include <drm/drm_edid.h>
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053026#include <sound/pcm_params.h>
Jeeja KP4a3478d2016-02-12 07:46:06 +053027#include <sound/jack.h>
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053028#include <sound/soc.h>
29#include <sound/hdaudio_ext.h>
Subhransu S. Prusty07f083a2015-11-10 18:42:10 +053030#include <sound/hda_i915.h>
Subhransu S. Prusty2428bca2016-02-12 07:46:02 +053031#include <sound/pcm_drm_eld.h>
Subhransu S. Prustybcced702016-04-14 10:07:30 +053032#include <sound/hda_chmap.h>
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053033#include "../../hda/local.h"
Jeeja KP4a3478d2016-02-12 07:46:06 +053034#include "hdac_hdmi.h"
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053035
Subhransu S. Prusty17a42c42016-02-12 07:46:04 +053036#define NAME_SIZE 32
37
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +053038#define AMP_OUT_MUTE 0xb080
39#define AMP_OUT_UNMUTE 0xb000
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053040#define PIN_OUT (AC_PINCTL_OUT_EN)
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +053041
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053042#define HDA_MAX_CONNECTIONS 32
43
Subhransu S. Prusty148569f2016-02-12 07:46:07 +053044#define HDA_MAX_CVTS 3
45
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +053046#define ELD_MAX_SIZE 256
47#define ELD_FIXED_BYTES 20
48
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053049struct hdac_hdmi_cvt_params {
50 unsigned int channels_min;
51 unsigned int channels_max;
52 u32 rates;
53 u64 formats;
54 unsigned int maxbps;
55};
56
57struct hdac_hdmi_cvt {
Subhransu S. Prusty15b91442015-12-09 21:46:10 +053058 struct list_head head;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053059 hda_nid_t nid;
Jeeja KP4a3478d2016-02-12 07:46:06 +053060 const char *name;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053061 struct hdac_hdmi_cvt_params params;
62};
63
Subhransu S. Prustyb7756ed2016-04-14 10:07:28 +053064/* Currently only spk_alloc, more to be added */
65struct hdac_hdmi_parsed_eld {
66 u8 spk_alloc;
67};
68
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +053069struct hdac_hdmi_eld {
70 bool monitor_present;
71 bool eld_valid;
72 int eld_size;
73 char eld_buffer[ELD_MAX_SIZE];
Subhransu S. Prustyb7756ed2016-04-14 10:07:28 +053074 struct hdac_hdmi_parsed_eld info;
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +053075};
76
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053077struct hdac_hdmi_pin {
Subhransu S. Prusty15b91442015-12-09 21:46:10 +053078 struct list_head head;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053079 hda_nid_t nid;
80 int num_mux_nids;
81 hda_nid_t mux_nids[HDA_MAX_CONNECTIONS];
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +053082 struct hdac_hdmi_eld eld;
83 struct hdac_ext_device *edev;
84 int repoll_count;
85 struct delayed_work work;
Subhransu S. Prustybcced702016-04-14 10:07:30 +053086 struct mutex lock;
87 bool chmap_set;
88 unsigned char chmap[8]; /* ALSA API channel-map */
89 int channels; /* current number of channels */
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +053090};
91
Jeeja KP4a3478d2016-02-12 07:46:06 +053092struct hdac_hdmi_pcm {
93 struct list_head head;
94 int pcm_id;
95 struct hdac_hdmi_pin *pin;
96 struct hdac_hdmi_cvt *cvt;
97 struct snd_jack *jack;
98};
99
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530100struct hdac_hdmi_dai_pin_map {
101 int dai_id;
Subhransu S. Prusty15b91442015-12-09 21:46:10 +0530102 struct hdac_hdmi_pin *pin;
103 struct hdac_hdmi_cvt *cvt;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530104};
105
106struct hdac_hdmi_priv {
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530107 struct hdac_hdmi_dai_pin_map dai_map[HDA_MAX_CVTS];
Subhransu S. Prusty15b91442015-12-09 21:46:10 +0530108 struct list_head pin_list;
109 struct list_head cvt_list;
Jeeja KP4a3478d2016-02-12 07:46:06 +0530110 struct list_head pcm_list;
Subhransu S. Prusty15b91442015-12-09 21:46:10 +0530111 int num_pin;
112 int num_cvt;
Jeeja KP4a3478d2016-02-12 07:46:06 +0530113 struct mutex pin_mutex;
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530114 struct hdac_chmap chmap;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530115};
116
Subhransu S. Prustye342ac02015-11-10 18:42:07 +0530117static inline struct hdac_ext_device *to_hda_ext_device(struct device *dev)
118{
Geliang Tang51b2c422015-12-28 22:47:13 +0800119 struct hdac_device *hdac = dev_to_hdac_dev(dev);
Subhransu S. Prustye342ac02015-11-10 18:42:07 +0530120
Geliang Tang51b2c422015-12-28 22:47:13 +0800121 return to_ehdac_device(hdac);
Subhransu S. Prustye342ac02015-11-10 18:42:07 +0530122}
123
Subhransu S. Prusty2428bca2016-02-12 07:46:02 +0530124static unsigned int sad_format(const u8 *sad)
125{
126 return ((sad[0] >> 0x3) & 0x1f);
127}
128
129static unsigned int sad_sample_bits_lpcm(const u8 *sad)
130{
131 return (sad[2] & 7);
132}
133
134static int hdac_hdmi_eld_limit_formats(struct snd_pcm_runtime *runtime,
135 void *eld)
136{
137 u64 formats = SNDRV_PCM_FMTBIT_S16;
138 int i;
139 const u8 *sad, *eld_buf = eld;
140
141 sad = drm_eld_sad(eld_buf);
142 if (!sad)
143 goto format_constraint;
144
145 for (i = drm_eld_sad_count(eld_buf); i > 0; i--, sad += 3) {
146 if (sad_format(sad) == 1) { /* AUDIO_CODING_TYPE_LPCM */
147
148 /*
149 * the controller support 20 and 24 bits in 32 bit
150 * container so we set S32
151 */
152 if (sad_sample_bits_lpcm(sad) & 0x6)
153 formats |= SNDRV_PCM_FMTBIT_S32;
154 }
155 }
156
157format_constraint:
158 return snd_pcm_hw_constraint_mask64(runtime, SNDRV_PCM_HW_PARAM_FORMAT,
159 formats);
160
161}
162
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +0530163 /* HDMI ELD routines */
164static unsigned int hdac_hdmi_get_eld_data(struct hdac_device *codec,
165 hda_nid_t nid, int byte_index)
166{
167 unsigned int val;
168
169 val = snd_hdac_codec_read(codec, nid, 0, AC_VERB_GET_HDMI_ELDD,
170 byte_index);
171
172 dev_dbg(&codec->dev, "HDMI: ELD data byte %d: 0x%x\n",
173 byte_index, val);
174
175 return val;
176}
177
178static int hdac_hdmi_get_eld_size(struct hdac_device *codec, hda_nid_t nid)
179{
180 return snd_hdac_codec_read(codec, nid, 0, AC_VERB_GET_HDMI_DIP_SIZE,
181 AC_DIPSIZE_ELD_BUF);
182}
183
184/*
185 * This function queries the ELD size and ELD data and fills in the buffer
186 * passed by user
187 */
188static int hdac_hdmi_get_eld(struct hdac_device *codec, hda_nid_t nid,
189 unsigned char *buf, int *eld_size)
190{
191 int i, size, ret = 0;
192
193 /*
194 * ELD size is initialized to zero in caller function. If no errors and
195 * ELD is valid, actual eld_size is assigned.
196 */
197
198 size = hdac_hdmi_get_eld_size(codec, nid);
199 if (size < ELD_FIXED_BYTES || size > ELD_MAX_SIZE) {
200 dev_err(&codec->dev, "HDMI: invalid ELD buf size %d\n", size);
201 return -ERANGE;
202 }
203
204 /* set ELD buffer */
205 for (i = 0; i < size; i++) {
206 unsigned int val = hdac_hdmi_get_eld_data(codec, nid, i);
207 /*
208 * Graphics driver might be writing to ELD buffer right now.
209 * Just abort. The caller will repoll after a while.
210 */
211 if (!(val & AC_ELDD_ELD_VALID)) {
212 dev_err(&codec->dev,
213 "HDMI: invalid ELD data byte %d\n", i);
214 ret = -EINVAL;
215 goto error;
216 }
217 val &= AC_ELDD_ELD_DATA;
218 /*
219 * The first byte cannot be zero. This can happen on some DVI
220 * connections. Some Intel chips may also need some 250ms delay
221 * to return non-zero ELD data, even when the graphics driver
222 * correctly writes ELD content before setting ELD_valid bit.
223 */
224 if (!val && !i) {
225 dev_err(&codec->dev, "HDMI: 0 ELD data\n");
226 ret = -EINVAL;
227 goto error;
228 }
229 buf[i] = val;
230 }
231
232 *eld_size = size;
233error:
234 return ret;
235}
236
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530237static int hdac_hdmi_setup_stream(struct hdac_ext_device *hdac,
238 hda_nid_t cvt_nid, hda_nid_t pin_nid,
239 u32 stream_tag, int format)
240{
241 unsigned int val;
242
243 dev_dbg(&hdac->hdac.dev, "cvt nid %d pnid %d stream %d format 0x%x\n",
244 cvt_nid, pin_nid, stream_tag, format);
245
246 val = (stream_tag << 4);
247
248 snd_hdac_codec_write(&hdac->hdac, cvt_nid, 0,
249 AC_VERB_SET_CHANNEL_STREAMID, val);
250 snd_hdac_codec_write(&hdac->hdac, cvt_nid, 0,
251 AC_VERB_SET_STREAM_FORMAT, format);
252
253 return 0;
254}
255
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +0530256static void
257hdac_hdmi_set_dip_index(struct hdac_ext_device *hdac, hda_nid_t pin_nid,
258 int packet_index, int byte_index)
259{
260 int val;
261
262 val = (packet_index << 5) | (byte_index & 0x1f);
263
264 snd_hdac_codec_write(&hdac->hdac, pin_nid, 0,
265 AC_VERB_SET_HDMI_DIP_INDEX, val);
266}
267
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530268struct dp_audio_infoframe {
269 u8 type; /* 0x84 */
270 u8 len; /* 0x1b */
271 u8 ver; /* 0x11 << 2 */
272
273 u8 CC02_CT47; /* match with HDMI infoframe from this on */
274 u8 SS01_SF24;
275 u8 CXT04;
276 u8 CA;
277 u8 LFEPBL01_LSV36_DM_INH7;
278};
279
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +0530280static int hdac_hdmi_setup_audio_infoframe(struct hdac_ext_device *hdac,
281 hda_nid_t cvt_nid, hda_nid_t pin_nid)
282{
283 uint8_t buffer[HDMI_INFOFRAME_HEADER_SIZE + HDMI_AUDIO_INFOFRAME_SIZE];
284 struct hdmi_audio_infoframe frame;
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530285 struct dp_audio_infoframe dp_ai;
286 struct hdac_hdmi_priv *hdmi = hdac->private_data;
287 struct hdac_hdmi_pin *pin;
288 u8 *dip;
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +0530289 int ret;
290 int i;
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530291 const u8 *eld_buf;
292 u8 conn_type;
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530293 int channels, ca;
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +0530294
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530295 list_for_each_entry(pin, &hdmi->pin_list, head) {
296 if (pin->nid == pin_nid)
297 break;
298 }
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +0530299
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530300 ca = snd_hdac_channel_allocation(&hdac->hdac, pin->eld.info.spk_alloc,
301 pin->channels, pin->chmap_set, true, pin->chmap);
302
303 channels = snd_hdac_get_active_channels(ca);
304 hdmi->chmap.ops.set_channel_count(&hdac->hdac, cvt_nid, channels);
305
306 snd_hdac_setup_channel_mapping(&hdmi->chmap, pin->nid, false, ca,
307 pin->channels, pin->chmap, pin->chmap_set);
308
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530309 eld_buf = pin->eld.eld_buffer;
310 conn_type = drm_eld_get_conn_type(eld_buf);
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +0530311
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530312 switch (conn_type) {
313 case DRM_ELD_CONN_TYPE_HDMI:
314 hdmi_audio_infoframe_init(&frame);
315
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530316 frame.channels = channels;
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530317 frame.channel_allocation = ca;
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530318
319 ret = hdmi_audio_infoframe_pack(&frame, buffer, sizeof(buffer));
320 if (ret < 0)
321 return ret;
322
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530323 break;
324
325 case DRM_ELD_CONN_TYPE_DP:
326 memset(&dp_ai, 0, sizeof(dp_ai));
327 dp_ai.type = 0x84;
328 dp_ai.len = 0x1b;
329 dp_ai.ver = 0x11 << 2;
330 dp_ai.CC02_CT47 = channels - 1;
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530331 dp_ai.CA = ca;
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530332
333 dip = (u8 *)&dp_ai;
334 break;
335
336 default:
337 dev_err(&hdac->hdac.dev, "Invalid connection type: %d\n",
338 conn_type);
339 return -EIO;
340 }
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +0530341
342 /* stop infoframe transmission */
343 hdac_hdmi_set_dip_index(hdac, pin_nid, 0x0, 0x0);
344 snd_hdac_codec_write(&hdac->hdac, pin_nid, 0,
345 AC_VERB_SET_HDMI_DIP_XMIT, AC_DIPXMIT_DISABLE);
346
347
348 /* Fill infoframe. Index auto-incremented */
349 hdac_hdmi_set_dip_index(hdac, pin_nid, 0x0, 0x0);
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530350 if (conn_type == DRM_ELD_CONN_TYPE_HDMI) {
Subhransu S. Prusty391005e2016-03-10 09:04:07 +0530351 for (i = 0; i < sizeof(buffer); i++)
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530352 snd_hdac_codec_write(&hdac->hdac, pin_nid, 0,
Subhransu S. Prusty391005e2016-03-10 09:04:07 +0530353 AC_VERB_SET_HDMI_DIP_DATA, buffer[i]);
Subhransu S. Prusty478f544e2016-02-12 07:46:09 +0530354 } else {
355 for (i = 0; i < sizeof(dp_ai); i++)
356 snd_hdac_codec_write(&hdac->hdac, pin_nid, 0,
357 AC_VERB_SET_HDMI_DIP_DATA, dip[i]);
358 }
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +0530359
360 /* Start infoframe */
361 hdac_hdmi_set_dip_index(hdac, pin_nid, 0x0, 0x0);
362 snd_hdac_codec_write(&hdac->hdac, pin_nid, 0,
363 AC_VERB_SET_HDMI_DIP_XMIT, AC_DIPXMIT_BEST);
364
365 return 0;
366}
367
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530368static void hdac_hdmi_set_power_state(struct hdac_ext_device *edev,
369 struct hdac_hdmi_dai_pin_map *dai_map, unsigned int pwr_state)
370{
371 /* Power up pin widget */
Subhransu S. Prusty15b91442015-12-09 21:46:10 +0530372 if (!snd_hdac_check_power_state(&edev->hdac, dai_map->pin->nid,
373 pwr_state))
374 snd_hdac_codec_write(&edev->hdac, dai_map->pin->nid, 0,
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530375 AC_VERB_SET_POWER_STATE, pwr_state);
376
377 /* Power up converter */
Subhransu S. Prusty15b91442015-12-09 21:46:10 +0530378 if (!snd_hdac_check_power_state(&edev->hdac, dai_map->cvt->nid,
379 pwr_state))
380 snd_hdac_codec_write(&edev->hdac, dai_map->cvt->nid, 0,
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530381 AC_VERB_SET_POWER_STATE, pwr_state);
382}
383
384static int hdac_hdmi_playback_prepare(struct snd_pcm_substream *substream,
385 struct snd_soc_dai *dai)
386{
387 struct hdac_ext_device *hdac = snd_soc_dai_get_drvdata(dai);
388 struct hdac_hdmi_priv *hdmi = hdac->private_data;
389 struct hdac_hdmi_dai_pin_map *dai_map;
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530390 struct hdac_hdmi_pin *pin;
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530391 struct hdac_ext_dma_params *dd;
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +0530392 int ret;
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530393
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530394 dai_map = &hdmi->dai_map[dai->id];
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530395 pin = dai_map->pin;
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530396
397 dd = (struct hdac_ext_dma_params *)snd_soc_dai_get_dma_data(dai, substream);
398 dev_dbg(&hdac->hdac.dev, "stream tag from cpu dai %d format in cvt 0x%x\n",
399 dd->stream_tag, dd->format);
400
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530401 mutex_lock(&pin->lock);
402 pin->channels = substream->runtime->channels;
403
Subhransu S. Prusty15b91442015-12-09 21:46:10 +0530404 ret = hdac_hdmi_setup_audio_infoframe(hdac, dai_map->cvt->nid,
405 dai_map->pin->nid);
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530406 mutex_unlock(&pin->lock);
Subhransu S. Prustya657f1d2015-11-10 18:42:09 +0530407 if (ret < 0)
408 return ret;
409
Subhransu S. Prusty15b91442015-12-09 21:46:10 +0530410 return hdac_hdmi_setup_stream(hdac, dai_map->cvt->nid,
411 dai_map->pin->nid, dd->stream_tag, dd->format);
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530412}
413
414static int hdac_hdmi_set_hw_params(struct snd_pcm_substream *substream,
415 struct snd_pcm_hw_params *hparams, struct snd_soc_dai *dai)
416{
417 struct hdac_ext_device *hdac = snd_soc_dai_get_drvdata(dai);
Subhransu S. Prusty54dfa1e2016-02-17 21:34:00 +0530418 struct hdac_hdmi_priv *hdmi = hdac->private_data;
419 struct hdac_hdmi_dai_pin_map *dai_map;
420 struct hdac_hdmi_pin *pin;
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530421 struct hdac_ext_dma_params *dd;
422
Subhransu S. Prusty54dfa1e2016-02-17 21:34:00 +0530423 dai_map = &hdmi->dai_map[dai->id];
424 pin = dai_map->pin;
425
426 if (!pin)
427 return -ENODEV;
428
429 if ((!pin->eld.monitor_present) || (!pin->eld.eld_valid)) {
430 dev_err(&hdac->hdac.dev, "device is not configured for this pin: %d\n",
431 pin->nid);
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530432 return -ENODEV;
433 }
434
Subhransu S. Prusty6793a3d72016-02-17 21:33:59 +0530435 dd = snd_soc_dai_get_dma_data(dai, substream);
436 if (!dd) {
437 dd = kzalloc(sizeof(*dd), GFP_KERNEL);
438 if (!dd)
439 return -ENOMEM;
440 }
441
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530442 dd->format = snd_hdac_calc_stream_format(params_rate(hparams),
443 params_channels(hparams), params_format(hparams),
444 24, 0);
445
446 snd_soc_dai_set_dma_data(dai, substream, (void *)dd);
447
448 return 0;
449}
450
451static int hdac_hdmi_playback_cleanup(struct snd_pcm_substream *substream,
452 struct snd_soc_dai *dai)
453{
454 struct hdac_ext_device *edev = snd_soc_dai_get_drvdata(dai);
455 struct hdac_ext_dma_params *dd;
456 struct hdac_hdmi_priv *hdmi = edev->private_data;
457 struct hdac_hdmi_dai_pin_map *dai_map;
458
459 dai_map = &hdmi->dai_map[dai->id];
460
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530461 dd = (struct hdac_ext_dma_params *)snd_soc_dai_get_dma_data(dai, substream);
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530462
Subhransu S. Prusty6793a3d72016-02-17 21:33:59 +0530463 if (dd) {
464 snd_soc_dai_set_dma_data(dai, substream, NULL);
465 kfree(dd);
466 }
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530467
468 return 0;
469}
470
Subhransu S. Prustyab85f5b2016-02-17 21:34:02 +0530471static void hdac_hdmi_enable_cvt(struct hdac_ext_device *edev,
472 struct hdac_hdmi_dai_pin_map *dai_map)
473{
474 /* Enable transmission */
475 snd_hdac_codec_write(&edev->hdac, dai_map->cvt->nid, 0,
476 AC_VERB_SET_DIGI_CONVERT_1, 1);
477
478 /* Category Code (CC) to zero */
479 snd_hdac_codec_write(&edev->hdac, dai_map->cvt->nid, 0,
480 AC_VERB_SET_DIGI_CONVERT_2, 0);
481}
482
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530483static int hdac_hdmi_enable_pin(struct hdac_ext_device *hdac,
484 struct hdac_hdmi_dai_pin_map *dai_map)
485{
486 int mux_idx;
487 struct hdac_hdmi_pin *pin = dai_map->pin;
488
489 for (mux_idx = 0; mux_idx < pin->num_mux_nids; mux_idx++) {
490 if (pin->mux_nids[mux_idx] == dai_map->cvt->nid) {
491 snd_hdac_codec_write(&hdac->hdac, pin->nid, 0,
492 AC_VERB_SET_CONNECT_SEL, mux_idx);
493 break;
494 }
495 }
496
497 if (mux_idx == pin->num_mux_nids)
498 return -EIO;
499
500 /* Enable out path for this pin widget */
501 snd_hdac_codec_write(&hdac->hdac, pin->nid, 0,
502 AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
503
504 hdac_hdmi_set_power_state(hdac, dai_map, AC_PWRST_D0);
505
506 snd_hdac_codec_write(&hdac->hdac, pin->nid, 0,
507 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_UNMUTE);
508
509 return 0;
510}
511
512static int hdac_hdmi_query_pin_connlist(struct hdac_ext_device *hdac,
513 struct hdac_hdmi_pin *pin)
514{
515 if (!(get_wcaps(&hdac->hdac, pin->nid) & AC_WCAP_CONN_LIST)) {
516 dev_warn(&hdac->hdac.dev,
517 "HDMI: pin %d wcaps %#x does not support connection list\n",
518 pin->nid, get_wcaps(&hdac->hdac, pin->nid));
519 return -EINVAL;
520 }
521
522 pin->num_mux_nids = snd_hdac_get_connections(&hdac->hdac, pin->nid,
523 pin->mux_nids, HDA_MAX_CONNECTIONS);
524 if (pin->num_mux_nids == 0)
525 dev_warn(&hdac->hdac.dev, "No connections found for pin: %d\n",
526 pin->nid);
527
528 dev_dbg(&hdac->hdac.dev, "num_mux_nids %d for pin: %d\n",
529 pin->num_mux_nids, pin->nid);
530
531 return pin->num_mux_nids;
532}
533
534/*
535 * Query pcm list and return pin widget to which stream is routed.
536 *
537 * Also query connection list of the pin, to validate the cvt to pin map.
538 *
539 * Same stream rendering to multiple pins simultaneously can be done
540 * possibly, but not supported for now in driver. So return the first pin
541 * connected.
542 */
543static struct hdac_hdmi_pin *hdac_hdmi_get_pin_from_cvt(
544 struct hdac_ext_device *edev,
545 struct hdac_hdmi_priv *hdmi,
546 struct hdac_hdmi_cvt *cvt)
547{
548 struct hdac_hdmi_pcm *pcm;
549 struct hdac_hdmi_pin *pin = NULL;
550 int ret, i;
551
552 list_for_each_entry(pcm, &hdmi->pcm_list, head) {
553 if (pcm->cvt == cvt) {
554 pin = pcm->pin;
555 break;
556 }
557 }
558
559 if (pin) {
560 ret = hdac_hdmi_query_pin_connlist(edev, pin);
561 if (ret < 0)
562 return NULL;
563
564 for (i = 0; i < pin->num_mux_nids; i++) {
565 if (pin->mux_nids[i] == cvt->nid)
566 return pin;
567 }
568 }
569
570 return NULL;
571}
572
Subhransu S. Prusty54dfa1e2016-02-17 21:34:00 +0530573/*
574 * This tries to get a valid pin and set the HW constraints based on the
575 * ELD. Even if a valid pin is not found return success so that device open
576 * doesn't fail.
577 */
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530578static int hdac_hdmi_pcm_open(struct snd_pcm_substream *substream,
579 struct snd_soc_dai *dai)
580{
581 struct hdac_ext_device *hdac = snd_soc_dai_get_drvdata(dai);
582 struct hdac_hdmi_priv *hdmi = hdac->private_data;
583 struct hdac_hdmi_dai_pin_map *dai_map;
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530584 struct hdac_hdmi_cvt *cvt;
585 struct hdac_hdmi_pin *pin;
Subhransu S. Prusty2428bca2016-02-12 07:46:02 +0530586 int ret;
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530587
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530588 dai_map = &hdmi->dai_map[dai->id];
589
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530590 cvt = dai_map->cvt;
591 pin = hdac_hdmi_get_pin_from_cvt(hdac, hdmi, cvt);
Subhransu S. Prusty54dfa1e2016-02-17 21:34:00 +0530592
593 /*
594 * To make PA and other userland happy.
595 * userland scans devices so returning error does not help.
596 */
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530597 if (!pin)
Subhransu S. Prusty54dfa1e2016-02-17 21:34:00 +0530598 return 0;
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530599
600 if ((!pin->eld.monitor_present) ||
601 (!pin->eld.eld_valid)) {
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530602
Subhransu S. Prusty54dfa1e2016-02-17 21:34:00 +0530603 dev_warn(&hdac->hdac.dev,
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530604 "Failed: montior present? %d ELD valid?: %d for pin: %d\n",
605 pin->eld.monitor_present, pin->eld.eld_valid, pin->nid);
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +0530606
Subhransu S. Prusty54dfa1e2016-02-17 21:34:00 +0530607 return 0;
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530608 }
609
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530610 dai_map->pin = pin;
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530611
Subhransu S. Prustyab85f5b2016-02-17 21:34:02 +0530612 hdac_hdmi_enable_cvt(hdac, dai_map);
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530613 ret = hdac_hdmi_enable_pin(hdac, dai_map);
614 if (ret < 0)
615 return ret;
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530616
Subhransu S. Prusty2428bca2016-02-12 07:46:02 +0530617 ret = hdac_hdmi_eld_limit_formats(substream->runtime,
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530618 pin->eld.eld_buffer);
Subhransu S. Prusty2428bca2016-02-12 07:46:02 +0530619 if (ret < 0)
620 return ret;
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530621
Subhransu S. Prusty2428bca2016-02-12 07:46:02 +0530622 return snd_pcm_hw_constraint_eld(substream->runtime,
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530623 pin->eld.eld_buffer);
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530624}
625
Jeeja KP571d5072016-02-22 07:50:33 +0530626static int hdac_hdmi_trigger(struct snd_pcm_substream *substream, int cmd,
627 struct snd_soc_dai *dai)
628{
629 struct hdac_hdmi_dai_pin_map *dai_map;
630 struct hdac_ext_device *hdac = snd_soc_dai_get_drvdata(dai);
631 struct hdac_hdmi_priv *hdmi = hdac->private_data;
632 int ret;
633
634 dai_map = &hdmi->dai_map[dai->id];
635 if (cmd == SNDRV_PCM_TRIGGER_RESUME) {
636 ret = hdac_hdmi_enable_pin(hdac, dai_map);
637 if (ret < 0)
638 return ret;
639
640 return hdac_hdmi_playback_prepare(substream, dai);
641 }
642
643 return 0;
644}
645
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530646static void hdac_hdmi_pcm_close(struct snd_pcm_substream *substream,
647 struct snd_soc_dai *dai)
648{
649 struct hdac_ext_device *hdac = snd_soc_dai_get_drvdata(dai);
650 struct hdac_hdmi_priv *hdmi = hdac->private_data;
651 struct hdac_hdmi_dai_pin_map *dai_map;
652
653 dai_map = &hdmi->dai_map[dai->id];
654
Subhransu S. Prusty54dfa1e2016-02-17 21:34:00 +0530655 if (dai_map->pin) {
656 snd_hdac_codec_write(&hdac->hdac, dai_map->cvt->nid, 0,
657 AC_VERB_SET_CHANNEL_STREAMID, 0);
658 snd_hdac_codec_write(&hdac->hdac, dai_map->cvt->nid, 0,
659 AC_VERB_SET_STREAM_FORMAT, 0);
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530660
Subhransu S. Prusty54dfa1e2016-02-17 21:34:00 +0530661 hdac_hdmi_set_power_state(hdac, dai_map, AC_PWRST_D3);
662
663 snd_hdac_codec_write(&hdac->hdac, dai_map->pin->nid, 0,
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530664 AC_VERB_SET_AMP_GAIN_MUTE, AMP_OUT_MUTE);
Subhransu S. Prusty148569f2016-02-12 07:46:07 +0530665
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530666 mutex_lock(&dai_map->pin->lock);
667 dai_map->pin->channels = 0;
668 mutex_unlock(&dai_map->pin->lock);
669
Subhransu S. Prusty54dfa1e2016-02-17 21:34:00 +0530670 dai_map->pin = NULL;
671 }
Subhransu S. Prustyb0362ad2015-11-10 18:42:08 +0530672}
673
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530674static int
675hdac_hdmi_query_cvt_params(struct hdac_device *hdac, struct hdac_hdmi_cvt *cvt)
676{
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530677 unsigned int chans;
678 struct hdac_ext_device *edev = to_ehdac_device(hdac);
679 struct hdac_hdmi_priv *hdmi = edev->private_data;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530680 int err;
681
Subhransu S. Prustybcced702016-04-14 10:07:30 +0530682 chans = get_wcaps(hdac, cvt->nid);
683 chans = get_wcaps_channels(chans);
684
685 cvt->params.channels_min = 2;
686
687 cvt->params.channels_max = chans;
688 if (chans > hdmi->chmap.channels_max)
689 hdmi->chmap.channels_max = chans;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530690
691 err = snd_hdac_query_supported_pcm(hdac, cvt->nid,
692 &cvt->params.rates,
693 &cvt->params.formats,
694 &cvt->params.maxbps);
695 if (err < 0)
696 dev_err(&hdac->dev,
697 "Failed to query pcm params for nid %d: %d\n",
698 cvt->nid, err);
699
700 return err;
701}
702
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530703static int hdac_hdmi_fill_widget_info(struct device *dev,
704 struct snd_soc_dapm_widget *w,
705 enum snd_soc_dapm_type id, void *priv,
706 const char *wname, const char *stream,
707 struct snd_kcontrol_new *wc, int numkc)
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530708{
709 w->id = id;
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530710 w->name = devm_kstrdup(dev, wname, GFP_KERNEL);
711 if (!w->name)
712 return -ENOMEM;
713
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530714 w->sname = stream;
715 w->reg = SND_SOC_NOPM;
716 w->shift = 0;
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530717 w->kcontrol_news = wc;
718 w->num_kcontrols = numkc;
719 w->priv = priv;
720
721 return 0;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530722}
723
724static void hdac_hdmi_fill_route(struct snd_soc_dapm_route *route,
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530725 const char *sink, const char *control, const char *src,
726 int (*handler)(struct snd_soc_dapm_widget *src,
727 struct snd_soc_dapm_widget *sink))
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530728{
729 route->sink = sink;
730 route->source = src;
731 route->control = control;
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530732 route->connected = handler;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530733}
734
Jeeja KP4a3478d2016-02-12 07:46:06 +0530735static struct hdac_hdmi_pcm *hdac_hdmi_get_pcm(struct hdac_ext_device *edev,
736 struct hdac_hdmi_pin *pin)
737{
738 struct hdac_hdmi_priv *hdmi = edev->private_data;
739 struct hdac_hdmi_pcm *pcm = NULL;
740
741 list_for_each_entry(pcm, &hdmi->pcm_list, head) {
742 if (pcm->pin == pin)
743 return pcm;
744 }
745
746 return NULL;
747}
748
749/*
750 * Based on user selection, map the PINs with the PCMs.
751 */
752static int hdac_hdmi_set_pin_mux(struct snd_kcontrol *kcontrol,
753 struct snd_ctl_elem_value *ucontrol)
754{
755 int ret;
756 struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
757 struct snd_soc_dapm_widget *w = snd_soc_dapm_kcontrol_widget(kcontrol);
758 struct snd_soc_dapm_context *dapm = w->dapm;
759 struct hdac_hdmi_pin *pin = w->priv;
760 struct hdac_ext_device *edev = to_hda_ext_device(dapm->dev);
761 struct hdac_hdmi_priv *hdmi = edev->private_data;
762 struct hdac_hdmi_pcm *pcm = NULL;
763 const char *cvt_name = e->texts[ucontrol->value.enumerated.item[0]];
764
765 ret = snd_soc_dapm_put_enum_double(kcontrol, ucontrol);
766 if (ret < 0)
767 return ret;
768
769 mutex_lock(&hdmi->pin_mutex);
770 list_for_each_entry(pcm, &hdmi->pcm_list, head) {
771 if (pcm->pin == pin)
772 pcm->pin = NULL;
773
774 /*
775 * Jack status is not reported during device probe as the
776 * PCMs are not registered by then. So report it here.
777 */
778 if (!strcmp(cvt_name, pcm->cvt->name) && !pcm->pin) {
779 pcm->pin = pin;
780 if (pin->eld.monitor_present && pin->eld.eld_valid) {
781 dev_dbg(&edev->hdac.dev,
782 "jack report for pcm=%d\n",
783 pcm->pcm_id);
784
785 snd_jack_report(pcm->jack, SND_JACK_AVOUT);
786 }
787 mutex_unlock(&hdmi->pin_mutex);
788 return ret;
789 }
790 }
791 mutex_unlock(&hdmi->pin_mutex);
792
793 return ret;
794}
795
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530796/*
797 * Ideally the Mux inputs should be based on the num_muxs enumerated, but
798 * the display driver seem to be programming the connection list for the pin
799 * widget runtime.
800 *
801 * So programming all the possible inputs for the mux, the user has to take
802 * care of selecting the right one and leaving all other inputs selected to
803 * "NONE"
804 */
805static int hdac_hdmi_create_pin_muxs(struct hdac_ext_device *edev,
806 struct hdac_hdmi_pin *pin,
807 struct snd_soc_dapm_widget *widget,
808 const char *widget_name)
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530809{
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530810 struct hdac_hdmi_priv *hdmi = edev->private_data;
811 struct snd_kcontrol_new *kc;
812 struct hdac_hdmi_cvt *cvt;
813 struct soc_enum *se;
814 char kc_name[NAME_SIZE];
815 char mux_items[NAME_SIZE];
816 /* To hold inputs to the Pin mux */
817 char *items[HDA_MAX_CONNECTIONS];
818 int i = 0;
819 int num_items = hdmi->num_cvt + 1;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530820
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530821 kc = devm_kzalloc(&edev->hdac.dev, sizeof(*kc), GFP_KERNEL);
822 if (!kc)
823 return -ENOMEM;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530824
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530825 se = devm_kzalloc(&edev->hdac.dev, sizeof(*se), GFP_KERNEL);
826 if (!se)
827 return -ENOMEM;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530828
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530829 sprintf(kc_name, "Pin %d Input", pin->nid);
830 kc->name = devm_kstrdup(&edev->hdac.dev, kc_name, GFP_KERNEL);
831 if (!kc->name)
832 return -ENOMEM;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530833
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530834 kc->private_value = (long)se;
835 kc->iface = SNDRV_CTL_ELEM_IFACE_MIXER;
836 kc->access = 0;
837 kc->info = snd_soc_info_enum_double;
Jeeja KP4a3478d2016-02-12 07:46:06 +0530838 kc->put = hdac_hdmi_set_pin_mux;
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530839 kc->get = snd_soc_dapm_get_enum_double;
840
841 se->reg = SND_SOC_NOPM;
842
843 /* enum texts: ["NONE", "cvt #", "cvt #", ...] */
844 se->items = num_items;
845 se->mask = roundup_pow_of_two(se->items) - 1;
846
847 sprintf(mux_items, "NONE");
848 items[i] = devm_kstrdup(&edev->hdac.dev, mux_items, GFP_KERNEL);
849 if (!items[i])
850 return -ENOMEM;
851
852 list_for_each_entry(cvt, &hdmi->cvt_list, head) {
853 i++;
854 sprintf(mux_items, "cvt %d", cvt->nid);
855 items[i] = devm_kstrdup(&edev->hdac.dev, mux_items, GFP_KERNEL);
856 if (!items[i])
857 return -ENOMEM;
858 }
859
860 se->texts = devm_kmemdup(&edev->hdac.dev, items,
861 (num_items * sizeof(char *)), GFP_KERNEL);
862 if (!se->texts)
863 return -ENOMEM;
864
865 return hdac_hdmi_fill_widget_info(&edev->hdac.dev, widget,
Jeeja KP4a3478d2016-02-12 07:46:06 +0530866 snd_soc_dapm_mux, pin, widget_name, NULL, kc, 1);
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +0530867}
868
869/* Add cvt <- input <- mux route map */
870static void hdac_hdmi_add_pinmux_cvt_route(struct hdac_ext_device *edev,
871 struct snd_soc_dapm_widget *widgets,
872 struct snd_soc_dapm_route *route, int rindex)
873{
874 struct hdac_hdmi_priv *hdmi = edev->private_data;
875 const struct snd_kcontrol_new *kc;
876 struct soc_enum *se;
877 int mux_index = hdmi->num_cvt + hdmi->num_pin;
878 int i, j;
879
880 for (i = 0; i < hdmi->num_pin; i++) {
881 kc = widgets[mux_index].kcontrol_news;
882 se = (struct soc_enum *)kc->private_value;
883 for (j = 0; j < hdmi->num_cvt; j++) {
884 hdac_hdmi_fill_route(&route[rindex],
885 widgets[mux_index].name,
886 se->texts[j + 1],
887 widgets[j].name, NULL);
888
889 rindex++;
890 }
891
892 mux_index++;
893 }
894}
895
896/*
897 * Widgets are added in the below sequence
898 * Converter widgets for num converters enumerated
899 * Pin widgets for num pins enumerated
900 * Pin mux widgets to represent connenction list of pin widget
901 *
902 * Total widgets elements = num_cvt + num_pin + num_pin;
903 *
904 * Routes are added as below:
905 * pin mux -> pin (based on num_pins)
906 * cvt -> "Input sel control" -> pin_mux
907 *
908 * Total route elements:
909 * num_pins + (pin_muxes * num_cvt)
910 */
911static int create_fill_widget_route_map(struct snd_soc_dapm_context *dapm)
912{
913 struct snd_soc_dapm_widget *widgets;
914 struct snd_soc_dapm_route *route;
915 struct hdac_ext_device *edev = to_hda_ext_device(dapm->dev);
916 struct hdac_hdmi_priv *hdmi = edev->private_data;
917 struct snd_soc_dai_driver *dai_drv = dapm->component->dai_drv;
918 char widget_name[NAME_SIZE];
919 struct hdac_hdmi_cvt *cvt;
920 struct hdac_hdmi_pin *pin;
921 int ret, i = 0, num_routes = 0;
922
923 if (list_empty(&hdmi->cvt_list) || list_empty(&hdmi->pin_list))
924 return -EINVAL;
925
926 widgets = devm_kzalloc(dapm->dev,
927 (sizeof(*widgets) * ((2 * hdmi->num_pin) + hdmi->num_cvt)),
928 GFP_KERNEL);
929
930 if (!widgets)
931 return -ENOMEM;
932
933 /* DAPM widgets to represent each converter widget */
934 list_for_each_entry(cvt, &hdmi->cvt_list, head) {
935 sprintf(widget_name, "Converter %d", cvt->nid);
936 ret = hdac_hdmi_fill_widget_info(dapm->dev, &widgets[i],
937 snd_soc_dapm_aif_in, &cvt->nid,
938 widget_name, dai_drv[i].playback.stream_name, NULL, 0);
939 if (ret < 0)
940 return ret;
941 i++;
942 }
943
944 list_for_each_entry(pin, &hdmi->pin_list, head) {
945 sprintf(widget_name, "hif%d Output", pin->nid);
946 ret = hdac_hdmi_fill_widget_info(dapm->dev, &widgets[i],
947 snd_soc_dapm_output, &pin->nid,
948 widget_name, NULL, NULL, 0);
949 if (ret < 0)
950 return ret;
951 i++;
952 }
953
954 /* DAPM widgets to represent the connection list to pin widget */
955 list_for_each_entry(pin, &hdmi->pin_list, head) {
956 sprintf(widget_name, "Pin %d Mux", pin->nid);
957 ret = hdac_hdmi_create_pin_muxs(edev, pin, &widgets[i],
958 widget_name);
959 if (ret < 0)
960 return ret;
961 i++;
962
963 /* For cvt to pin_mux mapping */
964 num_routes += hdmi->num_cvt;
965
966 /* For pin_mux to pin mapping */
967 num_routes++;
968 }
969
970 route = devm_kzalloc(dapm->dev, (sizeof(*route) * num_routes),
971 GFP_KERNEL);
972 if (!route)
973 return -ENOMEM;
974
975 i = 0;
976 /* Add pin <- NULL <- mux route map */
977 list_for_each_entry(pin, &hdmi->pin_list, head) {
978 int sink_index = i + hdmi->num_cvt;
979 int src_index = sink_index + hdmi->num_pin;
980
981 hdac_hdmi_fill_route(&route[i],
982 widgets[sink_index].name, NULL,
983 widgets[src_index].name, NULL);
984 i++;
985
986 }
987
988 hdac_hdmi_add_pinmux_cvt_route(edev, widgets, route, i);
989
990 snd_soc_dapm_new_controls(dapm, widgets,
991 ((2 * hdmi->num_pin) + hdmi->num_cvt));
992
993 snd_soc_dapm_add_routes(dapm, route, num_routes);
994 snd_soc_dapm_new_widgets(dapm->card);
995
996 return 0;
997
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +0530998}
999
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301000static int hdac_hdmi_init_dai_map(struct hdac_ext_device *edev)
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301001{
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301002 struct hdac_hdmi_priv *hdmi = edev->private_data;
Subhransu S. Prusty148569f2016-02-12 07:46:07 +05301003 struct hdac_hdmi_dai_pin_map *dai_map;
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301004 struct hdac_hdmi_cvt *cvt;
Subhransu S. Prusty148569f2016-02-12 07:46:07 +05301005 int dai_id = 0;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301006
Subhransu S. Prusty148569f2016-02-12 07:46:07 +05301007 if (list_empty(&hdmi->cvt_list))
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301008 return -EINVAL;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301009
Subhransu S. Prusty148569f2016-02-12 07:46:07 +05301010 list_for_each_entry(cvt, &hdmi->cvt_list, head) {
1011 dai_map = &hdmi->dai_map[dai_id];
1012 dai_map->dai_id = dai_id;
1013 dai_map->cvt = cvt;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301014
Subhransu S. Prusty148569f2016-02-12 07:46:07 +05301015 dai_id++;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301016
Subhransu S. Prusty148569f2016-02-12 07:46:07 +05301017 if (dai_id == HDA_MAX_CVTS) {
1018 dev_warn(&edev->hdac.dev,
1019 "Max dais supported: %d\n", dai_id);
1020 break;
1021 }
1022 }
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301023
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301024 return 0;
1025}
1026
1027static int hdac_hdmi_add_cvt(struct hdac_ext_device *edev, hda_nid_t nid)
1028{
1029 struct hdac_hdmi_priv *hdmi = edev->private_data;
1030 struct hdac_hdmi_cvt *cvt;
Jeeja KP4a3478d2016-02-12 07:46:06 +05301031 char name[NAME_SIZE];
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301032
1033 cvt = kzalloc(sizeof(*cvt), GFP_KERNEL);
1034 if (!cvt)
1035 return -ENOMEM;
1036
1037 cvt->nid = nid;
Jeeja KP4a3478d2016-02-12 07:46:06 +05301038 sprintf(name, "cvt %d", cvt->nid);
1039 cvt->name = kstrdup(name, GFP_KERNEL);
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301040
1041 list_add_tail(&cvt->head, &hdmi->cvt_list);
1042 hdmi->num_cvt++;
1043
1044 return hdac_hdmi_query_cvt_params(&edev->hdac, cvt);
1045}
1046
Subhransu S. Prustyb7756ed2016-04-14 10:07:28 +05301047static void hdac_hdmi_parse_eld(struct hdac_ext_device *edev,
1048 struct hdac_hdmi_pin *pin)
1049{
1050 pin->eld.info.spk_alloc = pin->eld.eld_buffer[DRM_ELD_SPEAKER];
1051}
1052
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301053static void hdac_hdmi_present_sense(struct hdac_hdmi_pin *pin, int repoll)
1054{
1055 struct hdac_ext_device *edev = pin->edev;
Jeeja KP4a3478d2016-02-12 07:46:06 +05301056 struct hdac_hdmi_priv *hdmi = edev->private_data;
1057 struct hdac_hdmi_pcm *pcm;
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301058 int val;
1059
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301060 pin->repoll_count = repoll;
1061
1062 pm_runtime_get_sync(&edev->hdac.dev);
1063 val = snd_hdac_codec_read(&edev->hdac, pin->nid, 0,
1064 AC_VERB_GET_PIN_SENSE, 0);
1065
1066 dev_dbg(&edev->hdac.dev, "Pin sense val %x for pin: %d\n",
1067 val, pin->nid);
1068
Jeeja KP4a3478d2016-02-12 07:46:06 +05301069
1070 mutex_lock(&hdmi->pin_mutex);
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301071 pin->eld.monitor_present = !!(val & AC_PINSENSE_PRESENCE);
1072 pin->eld.eld_valid = !!(val & AC_PINSENSE_ELDV);
1073
Jeeja KP4a3478d2016-02-12 07:46:06 +05301074 pcm = hdac_hdmi_get_pcm(edev, pin);
1075
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301076 if (!pin->eld.monitor_present || !pin->eld.eld_valid) {
1077
1078 dev_dbg(&edev->hdac.dev, "%s: disconnect for pin %d\n",
1079 __func__, pin->nid);
Jeeja KP4a3478d2016-02-12 07:46:06 +05301080
1081 /*
1082 * PCMs are not registered during device probe, so don't
1083 * report jack here. It will be done in usermode mux
1084 * control select.
1085 */
1086 if (pcm) {
1087 dev_dbg(&edev->hdac.dev,
1088 "jack report for pcm=%d\n", pcm->pcm_id);
1089
1090 snd_jack_report(pcm->jack, 0);
1091 }
1092
1093 mutex_unlock(&hdmi->pin_mutex);
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301094 goto put_hdac_device;
1095 }
1096
1097 if (pin->eld.monitor_present && pin->eld.eld_valid) {
1098 /* TODO: use i915 component for reading ELD later */
1099 if (hdac_hdmi_get_eld(&edev->hdac, pin->nid,
1100 pin->eld.eld_buffer,
1101 &pin->eld.eld_size) == 0) {
1102
Jeeja KP4a3478d2016-02-12 07:46:06 +05301103 if (pcm) {
1104 dev_dbg(&edev->hdac.dev,
1105 "jack report for pcm=%d\n",
1106 pcm->pcm_id);
1107
1108 snd_jack_report(pcm->jack, SND_JACK_AVOUT);
1109 }
Subhransu S. Prustyb7756ed2016-04-14 10:07:28 +05301110 hdac_hdmi_parse_eld(edev, pin);
Jeeja KP4a3478d2016-02-12 07:46:06 +05301111
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301112 print_hex_dump_bytes("ELD: ", DUMP_PREFIX_OFFSET,
1113 pin->eld.eld_buffer, pin->eld.eld_size);
1114 } else {
1115 pin->eld.monitor_present = false;
1116 pin->eld.eld_valid = false;
Jeeja KP4a3478d2016-02-12 07:46:06 +05301117
1118 if (pcm) {
1119 dev_dbg(&edev->hdac.dev,
1120 "jack report for pcm=%d\n",
1121 pcm->pcm_id);
1122
1123 snd_jack_report(pcm->jack, 0);
1124 }
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301125 }
1126 }
1127
Jeeja KP4a3478d2016-02-12 07:46:06 +05301128 mutex_unlock(&hdmi->pin_mutex);
1129
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301130 /*
1131 * Sometimes the pin_sense may present invalid monitor
1132 * present and eld_valid. If ELD data is not valid, loop few
1133 * more times to get correct pin sense and valid ELD.
1134 */
1135 if ((!pin->eld.monitor_present || !pin->eld.eld_valid) && repoll)
1136 schedule_delayed_work(&pin->work, msecs_to_jiffies(300));
1137
1138put_hdac_device:
1139 pm_runtime_put_sync(&edev->hdac.dev);
1140}
1141
1142static void hdac_hdmi_repoll_eld(struct work_struct *work)
1143{
1144 struct hdac_hdmi_pin *pin =
1145 container_of(to_delayed_work(work), struct hdac_hdmi_pin, work);
1146
1147 /* picked from legacy HDA driver */
1148 if (pin->repoll_count++ > 6)
1149 pin->repoll_count = 0;
1150
1151 hdac_hdmi_present_sense(pin, pin->repoll_count);
1152}
1153
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301154static int hdac_hdmi_add_pin(struct hdac_ext_device *edev, hda_nid_t nid)
1155{
1156 struct hdac_hdmi_priv *hdmi = edev->private_data;
1157 struct hdac_hdmi_pin *pin;
1158
1159 pin = kzalloc(sizeof(*pin), GFP_KERNEL);
1160 if (!pin)
1161 return -ENOMEM;
1162
1163 pin->nid = nid;
1164
1165 list_add_tail(&pin->head, &hdmi->pin_list);
1166 hdmi->num_pin++;
1167
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301168 pin->edev = edev;
Subhransu S. Prustybcced702016-04-14 10:07:30 +05301169 mutex_init(&pin->lock);
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301170 INIT_DELAYED_WORK(&pin->work, hdac_hdmi_repoll_eld);
1171
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301172 return 0;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301173}
1174
Subhransu S. Prusty211caab2016-02-12 07:46:03 +05301175#define INTEL_VENDOR_NID 0x08
1176#define INTEL_GET_VENDOR_VERB 0xf81
1177#define INTEL_SET_VENDOR_VERB 0x781
1178#define INTEL_EN_DP12 0x02 /* enable DP 1.2 features */
1179#define INTEL_EN_ALL_PIN_CVTS 0x01 /* enable 2nd & 3rd pins and convertors */
1180
1181static void hdac_hdmi_skl_enable_all_pins(struct hdac_device *hdac)
1182{
1183 unsigned int vendor_param;
1184
1185 vendor_param = snd_hdac_codec_read(hdac, INTEL_VENDOR_NID, 0,
1186 INTEL_GET_VENDOR_VERB, 0);
1187 if (vendor_param == -1 || vendor_param & INTEL_EN_ALL_PIN_CVTS)
1188 return;
1189
1190 vendor_param |= INTEL_EN_ALL_PIN_CVTS;
1191 vendor_param = snd_hdac_codec_read(hdac, INTEL_VENDOR_NID, 0,
1192 INTEL_SET_VENDOR_VERB, vendor_param);
1193 if (vendor_param == -1)
1194 return;
1195}
1196
1197static void hdac_hdmi_skl_enable_dp12(struct hdac_device *hdac)
1198{
1199 unsigned int vendor_param;
1200
1201 vendor_param = snd_hdac_codec_read(hdac, INTEL_VENDOR_NID, 0,
1202 INTEL_GET_VENDOR_VERB, 0);
1203 if (vendor_param == -1 || vendor_param & INTEL_EN_DP12)
1204 return;
1205
1206 /* enable DP1.2 mode */
1207 vendor_param |= INTEL_EN_DP12;
1208 vendor_param = snd_hdac_codec_read(hdac, INTEL_VENDOR_NID, 0,
1209 INTEL_SET_VENDOR_VERB, vendor_param);
1210 if (vendor_param == -1)
1211 return;
1212
1213}
1214
Subhransu S. Prusty17a42c42016-02-12 07:46:04 +05301215static struct snd_soc_dai_ops hdmi_dai_ops = {
1216 .startup = hdac_hdmi_pcm_open,
1217 .shutdown = hdac_hdmi_pcm_close,
1218 .hw_params = hdac_hdmi_set_hw_params,
1219 .prepare = hdac_hdmi_playback_prepare,
Jeeja KP571d5072016-02-22 07:50:33 +05301220 .trigger = hdac_hdmi_trigger,
Subhransu S. Prusty17a42c42016-02-12 07:46:04 +05301221 .hw_free = hdac_hdmi_playback_cleanup,
1222};
1223
1224/*
1225 * Each converter can support a stream independently. So a dai is created
1226 * based on the number of converter queried.
1227 */
1228static int hdac_hdmi_create_dais(struct hdac_device *hdac,
1229 struct snd_soc_dai_driver **dais,
1230 struct hdac_hdmi_priv *hdmi, int num_dais)
1231{
1232 struct snd_soc_dai_driver *hdmi_dais;
1233 struct hdac_hdmi_cvt *cvt;
1234 char name[NAME_SIZE], dai_name[NAME_SIZE];
1235 int i = 0;
1236 u32 rates, bps;
1237 unsigned int rate_max = 384000, rate_min = 8000;
1238 u64 formats;
1239 int ret;
1240
1241 hdmi_dais = devm_kzalloc(&hdac->dev,
1242 (sizeof(*hdmi_dais) * num_dais),
1243 GFP_KERNEL);
1244 if (!hdmi_dais)
1245 return -ENOMEM;
1246
1247 list_for_each_entry(cvt, &hdmi->cvt_list, head) {
1248 ret = snd_hdac_query_supported_pcm(hdac, cvt->nid,
1249 &rates, &formats, &bps);
1250 if (ret)
1251 return ret;
1252
1253 sprintf(dai_name, "intel-hdmi-hifi%d", i+1);
1254 hdmi_dais[i].name = devm_kstrdup(&hdac->dev,
1255 dai_name, GFP_KERNEL);
1256
1257 if (!hdmi_dais[i].name)
1258 return -ENOMEM;
1259
1260 snprintf(name, sizeof(name), "hifi%d", i+1);
1261 hdmi_dais[i].playback.stream_name =
1262 devm_kstrdup(&hdac->dev, name, GFP_KERNEL);
1263 if (!hdmi_dais[i].playback.stream_name)
1264 return -ENOMEM;
1265
1266 /*
1267 * Set caps based on capability queried from the converter.
1268 * It will be constrained runtime based on ELD queried.
1269 */
1270 hdmi_dais[i].playback.formats = formats;
1271 hdmi_dais[i].playback.rates = rates;
1272 hdmi_dais[i].playback.rate_max = rate_max;
1273 hdmi_dais[i].playback.rate_min = rate_min;
1274 hdmi_dais[i].playback.channels_min = 2;
1275 hdmi_dais[i].playback.channels_max = 2;
1276 hdmi_dais[i].ops = &hdmi_dai_ops;
1277
1278 i++;
1279 }
1280
1281 *dais = hdmi_dais;
1282
1283 return 0;
1284}
1285
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301286/*
1287 * Parse all nodes and store the cvt/pin nids in array
1288 * Add one time initialization for pin and cvt widgets
1289 */
Subhransu S. Prusty17a42c42016-02-12 07:46:04 +05301290static int hdac_hdmi_parse_and_map_nid(struct hdac_ext_device *edev,
1291 struct snd_soc_dai_driver **dais, int *num_dais)
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301292{
1293 hda_nid_t nid;
Sudip Mukherjee3c83ac22015-12-01 14:29:35 +05301294 int i, num_nodes;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301295 struct hdac_device *hdac = &edev->hdac;
1296 struct hdac_hdmi_priv *hdmi = edev->private_data;
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301297 int ret;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301298
Subhransu S. Prusty211caab2016-02-12 07:46:03 +05301299 hdac_hdmi_skl_enable_all_pins(hdac);
1300 hdac_hdmi_skl_enable_dp12(hdac);
1301
Sudip Mukherjee3c83ac22015-12-01 14:29:35 +05301302 num_nodes = snd_hdac_get_sub_nodes(hdac, hdac->afg, &nid);
Subhransu S. Prusty541140d2015-12-09 21:46:08 +05301303 if (!nid || num_nodes <= 0) {
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301304 dev_warn(&hdac->dev, "HDMI: failed to get afg sub nodes\n");
1305 return -EINVAL;
1306 }
1307
Sudip Mukherjee3c83ac22015-12-01 14:29:35 +05301308 hdac->num_nodes = num_nodes;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301309 hdac->start_nid = nid;
1310
1311 for (i = 0; i < hdac->num_nodes; i++, nid++) {
1312 unsigned int caps;
1313 unsigned int type;
1314
1315 caps = get_wcaps(hdac, nid);
1316 type = get_wcaps_type(caps);
1317
1318 if (!(caps & AC_WCAP_DIGITAL))
1319 continue;
1320
1321 switch (type) {
1322
1323 case AC_WID_AUD_OUT:
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301324 ret = hdac_hdmi_add_cvt(edev, nid);
1325 if (ret < 0)
1326 return ret;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301327 break;
1328
1329 case AC_WID_PIN:
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301330 ret = hdac_hdmi_add_pin(edev, nid);
1331 if (ret < 0)
1332 return ret;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301333 break;
1334 }
1335 }
1336
1337 hdac->end_nid = nid;
1338
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301339 if (!hdmi->num_pin || !hdmi->num_cvt)
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301340 return -EIO;
1341
Subhransu S. Prusty17a42c42016-02-12 07:46:04 +05301342 ret = hdac_hdmi_create_dais(hdac, dais, hdmi, hdmi->num_cvt);
1343 if (ret) {
1344 dev_err(&hdac->dev, "Failed to create dais with err: %d\n",
1345 ret);
1346 return ret;
1347 }
1348
1349 *num_dais = hdmi->num_cvt;
1350
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301351 return hdac_hdmi_init_dai_map(edev);
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301352}
1353
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301354static void hdac_hdmi_eld_notify_cb(void *aptr, int port)
1355{
1356 struct hdac_ext_device *edev = aptr;
1357 struct hdac_hdmi_priv *hdmi = edev->private_data;
1358 struct hdac_hdmi_pin *pin;
1359 struct snd_soc_codec *codec = edev->scodec;
1360
1361 /* Don't know how this mapping is derived */
1362 hda_nid_t pin_nid = port + 0x04;
1363
1364 dev_dbg(&edev->hdac.dev, "%s: for pin: %d\n", __func__, pin_nid);
1365
1366 /*
1367 * skip notification during system suspend (but not in runtime PM);
1368 * the state will be updated at resume. Also since the ELD and
1369 * connection states are updated in anyway at the end of the resume,
1370 * we can skip it when received during PM process.
1371 */
1372 if (snd_power_get_state(codec->component.card->snd_card) !=
1373 SNDRV_CTL_POWER_D0)
1374 return;
1375
1376 if (atomic_read(&edev->hdac.in_pm))
1377 return;
1378
1379 list_for_each_entry(pin, &hdmi->pin_list, head) {
1380 if (pin->nid == pin_nid)
1381 hdac_hdmi_present_sense(pin, 1);
1382 }
1383}
1384
1385static struct i915_audio_component_audio_ops aops = {
1386 .pin_eld_notify = hdac_hdmi_eld_notify_cb,
1387};
1388
Jeeja KP4a3478d2016-02-12 07:46:06 +05301389int hdac_hdmi_jack_init(struct snd_soc_dai *dai, int device)
1390{
1391 char jack_name[NAME_SIZE];
1392 struct snd_soc_codec *codec = dai->codec;
1393 struct hdac_ext_device *edev = snd_soc_codec_get_drvdata(codec);
1394 struct snd_soc_dapm_context *dapm =
1395 snd_soc_component_get_dapm(&codec->component);
1396 struct hdac_hdmi_priv *hdmi = edev->private_data;
1397 struct hdac_hdmi_pcm *pcm;
1398
1399 /*
1400 * this is a new PCM device, create new pcm and
1401 * add to the pcm list
1402 */
1403 pcm = kzalloc(sizeof(*pcm), GFP_KERNEL);
1404 if (!pcm)
1405 return -ENOMEM;
1406 pcm->pcm_id = device;
1407 pcm->cvt = hdmi->dai_map[dai->id].cvt;
1408
1409 list_add_tail(&pcm->head, &hdmi->pcm_list);
1410
1411 sprintf(jack_name, "HDMI/DP, pcm=%d Jack", device);
1412
1413 return snd_jack_new(dapm->card->snd_card, jack_name,
1414 SND_JACK_AVOUT, &pcm->jack, true, false);
1415}
1416EXPORT_SYMBOL_GPL(hdac_hdmi_jack_init);
1417
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301418static int hdmi_codec_probe(struct snd_soc_codec *codec)
1419{
1420 struct hdac_ext_device *edev = snd_soc_codec_get_drvdata(codec);
1421 struct hdac_hdmi_priv *hdmi = edev->private_data;
1422 struct snd_soc_dapm_context *dapm =
1423 snd_soc_component_get_dapm(&codec->component);
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301424 struct hdac_hdmi_pin *pin;
1425 int ret;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301426
1427 edev->scodec = codec;
1428
Subhransu S. Prusty79f4e922016-02-12 07:46:05 +05301429 ret = create_fill_widget_route_map(dapm);
1430 if (ret < 0)
1431 return ret;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301432
Subhransu S. Prustyb8a54542016-02-12 07:46:01 +05301433 aops.audio_ptr = edev;
1434 ret = snd_hdac_i915_register_notifier(&aops);
1435 if (ret < 0) {
1436 dev_err(&edev->hdac.dev, "notifier register failed: err: %d\n",
1437 ret);
1438 return ret;
1439 }
1440
1441 list_for_each_entry(pin, &hdmi->pin_list, head)
1442 hdac_hdmi_present_sense(pin, 1);
1443
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301444 /* Imp: Store the card pointer in hda_codec */
1445 edev->card = dapm->card->snd_card;
1446
Subhransu S. Prustye342ac02015-11-10 18:42:07 +05301447 /*
1448 * hdac_device core already sets the state to active and calls
1449 * get_noresume. So enable runtime and set the device to suspend.
1450 */
1451 pm_runtime_enable(&edev->hdac.dev);
1452 pm_runtime_put(&edev->hdac.dev);
1453 pm_runtime_suspend(&edev->hdac.dev);
1454
1455 return 0;
1456}
1457
1458static int hdmi_codec_remove(struct snd_soc_codec *codec)
1459{
1460 struct hdac_ext_device *edev = snd_soc_codec_get_drvdata(codec);
1461
1462 pm_runtime_disable(&edev->hdac.dev);
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301463 return 0;
1464}
1465
Jeeja KP571d5072016-02-22 07:50:33 +05301466#ifdef CONFIG_PM
1467static int hdmi_codec_resume(struct snd_soc_codec *codec)
1468{
1469 struct hdac_ext_device *edev = snd_soc_codec_get_drvdata(codec);
1470 struct hdac_hdmi_priv *hdmi = edev->private_data;
1471 struct hdac_hdmi_pin *pin;
1472 struct hdac_device *hdac = &edev->hdac;
1473 struct hdac_bus *bus = hdac->bus;
1474 int err;
1475 unsigned long timeout;
1476
1477 hdac_hdmi_skl_enable_all_pins(&edev->hdac);
1478 hdac_hdmi_skl_enable_dp12(&edev->hdac);
1479
1480 /* Power up afg */
1481 if (!snd_hdac_check_power_state(hdac, hdac->afg, AC_PWRST_D0)) {
1482
1483 snd_hdac_codec_write(hdac, hdac->afg, 0,
1484 AC_VERB_SET_POWER_STATE, AC_PWRST_D0);
1485
1486 /* Wait till power state is set to D0 */
1487 timeout = jiffies + msecs_to_jiffies(1000);
1488 while (!snd_hdac_check_power_state(hdac, hdac->afg, AC_PWRST_D0)
1489 && time_before(jiffies, timeout)) {
1490 msleep(50);
1491 }
1492 }
1493
1494 /*
1495 * As the ELD notify callback request is not entertained while the
1496 * device is in suspend state. Need to manually check detection of
1497 * all pins here.
1498 */
1499 list_for_each_entry(pin, &hdmi->pin_list, head)
1500 hdac_hdmi_present_sense(pin, 1);
1501
1502 /*
1503 * Codec power is turned ON during controller resume.
1504 * Turn it OFF here
1505 */
1506 err = snd_hdac_display_power(bus, false);
1507 if (err < 0) {
1508 dev_err(bus->dev,
1509 "Cannot turn OFF display power on i915, err: %d\n",
1510 err);
1511 return err;
1512 }
1513
1514 return 0;
1515}
1516#else
1517#define hdmi_codec_resume NULL
1518#endif
1519
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301520static struct snd_soc_codec_driver hdmi_hda_codec = {
1521 .probe = hdmi_codec_probe,
Subhransu S. Prustye342ac02015-11-10 18:42:07 +05301522 .remove = hdmi_codec_remove,
Jeeja KP571d5072016-02-22 07:50:33 +05301523 .resume = hdmi_codec_resume,
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301524 .idle_bias_off = true,
1525};
1526
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301527static int hdac_hdmi_dev_probe(struct hdac_ext_device *edev)
1528{
1529 struct hdac_device *codec = &edev->hdac;
1530 struct hdac_hdmi_priv *hdmi_priv;
Subhransu S. Prusty17a42c42016-02-12 07:46:04 +05301531 struct snd_soc_dai_driver *hdmi_dais = NULL;
1532 int num_dais = 0;
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301533 int ret = 0;
1534
1535 hdmi_priv = devm_kzalloc(&codec->dev, sizeof(*hdmi_priv), GFP_KERNEL);
1536 if (hdmi_priv == NULL)
1537 return -ENOMEM;
1538
1539 edev->private_data = hdmi_priv;
Subhransu S. Prustybcced702016-04-14 10:07:30 +05301540 snd_hdac_register_chmap_ops(codec, &hdmi_priv->chmap);
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301541
1542 dev_set_drvdata(&codec->dev, edev);
1543
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301544 INIT_LIST_HEAD(&hdmi_priv->pin_list);
1545 INIT_LIST_HEAD(&hdmi_priv->cvt_list);
Jeeja KP4a3478d2016-02-12 07:46:06 +05301546 INIT_LIST_HEAD(&hdmi_priv->pcm_list);
1547 mutex_init(&hdmi_priv->pin_mutex);
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301548
Ramesh Babuaeaccef2016-02-17 21:34:01 +05301549 /*
1550 * Turned off in the runtime_suspend during the first explicit
1551 * pm_runtime_suspend call.
1552 */
1553 ret = snd_hdac_display_power(edev->hdac.bus, true);
1554 if (ret < 0) {
1555 dev_err(&edev->hdac.dev,
1556 "Cannot turn on display power on i915 err: %d\n",
1557 ret);
1558 return ret;
1559 }
1560
Subhransu S. Prusty17a42c42016-02-12 07:46:04 +05301561 ret = hdac_hdmi_parse_and_map_nid(edev, &hdmi_dais, &num_dais);
1562 if (ret < 0) {
1563 dev_err(&codec->dev,
1564 "Failed in parse and map nid with err: %d\n", ret);
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301565 return ret;
Subhransu S. Prusty17a42c42016-02-12 07:46:04 +05301566 }
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301567
1568 /* ASoC specific initialization */
1569 return snd_soc_register_codec(&codec->dev, &hdmi_hda_codec,
Subhransu S. Prusty17a42c42016-02-12 07:46:04 +05301570 hdmi_dais, num_dais);
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301571}
1572
1573static int hdac_hdmi_dev_remove(struct hdac_ext_device *edev)
1574{
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301575 struct hdac_hdmi_priv *hdmi = edev->private_data;
1576 struct hdac_hdmi_pin *pin, *pin_next;
1577 struct hdac_hdmi_cvt *cvt, *cvt_next;
Jeeja KP4a3478d2016-02-12 07:46:06 +05301578 struct hdac_hdmi_pcm *pcm, *pcm_next;
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301579
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301580 snd_soc_unregister_codec(&edev->hdac.dev);
1581
Jeeja KP4a3478d2016-02-12 07:46:06 +05301582 list_for_each_entry_safe(pcm, pcm_next, &hdmi->pcm_list, head) {
1583 pcm->cvt = NULL;
1584 pcm->pin = NULL;
1585 list_del(&pcm->head);
1586 kfree(pcm);
1587 }
1588
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301589 list_for_each_entry_safe(cvt, cvt_next, &hdmi->cvt_list, head) {
1590 list_del(&cvt->head);
Jeeja KP4a3478d2016-02-12 07:46:06 +05301591 kfree(cvt->name);
Subhransu S. Prusty15b91442015-12-09 21:46:10 +05301592 kfree(cvt);
1593 }
1594
1595 list_for_each_entry_safe(pin, pin_next, &hdmi->pin_list, head) {
1596 list_del(&pin->head);
1597 kfree(pin);
1598 }
1599
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301600 return 0;
1601}
1602
Subhransu S. Prustye342ac02015-11-10 18:42:07 +05301603#ifdef CONFIG_PM
1604static int hdac_hdmi_runtime_suspend(struct device *dev)
1605{
1606 struct hdac_ext_device *edev = to_hda_ext_device(dev);
1607 struct hdac_device *hdac = &edev->hdac;
Subhransu S. Prusty07f083a2015-11-10 18:42:10 +05301608 struct hdac_bus *bus = hdac->bus;
Subhransu S. Prusty7ed49702016-02-22 07:50:32 +05301609 unsigned long timeout;
Subhransu S. Prusty07f083a2015-11-10 18:42:10 +05301610 int err;
Subhransu S. Prustye342ac02015-11-10 18:42:07 +05301611
1612 dev_dbg(dev, "Enter: %s\n", __func__);
1613
Subhransu S. Prusty07f083a2015-11-10 18:42:10 +05301614 /* controller may not have been initialized for the first time */
1615 if (!bus)
1616 return 0;
1617
Subhransu S. Prustye342ac02015-11-10 18:42:07 +05301618 /* Power down afg */
Subhransu S. Prusty7ed49702016-02-22 07:50:32 +05301619 if (!snd_hdac_check_power_state(hdac, hdac->afg, AC_PWRST_D3)) {
Subhransu S. Prustye342ac02015-11-10 18:42:07 +05301620 snd_hdac_codec_write(hdac, hdac->afg, 0,
1621 AC_VERB_SET_POWER_STATE, AC_PWRST_D3);
1622
Subhransu S. Prusty7ed49702016-02-22 07:50:32 +05301623 /* Wait till power state is set to D3 */
1624 timeout = jiffies + msecs_to_jiffies(1000);
1625 while (!snd_hdac_check_power_state(hdac, hdac->afg, AC_PWRST_D3)
1626 && time_before(jiffies, timeout)) {
1627
1628 msleep(50);
1629 }
1630 }
1631
Subhransu S. Prusty07f083a2015-11-10 18:42:10 +05301632 err = snd_hdac_display_power(bus, false);
1633 if (err < 0) {
1634 dev_err(bus->dev, "Cannot turn on display power on i915\n");
1635 return err;
1636 }
1637
Subhransu S. Prustye342ac02015-11-10 18:42:07 +05301638 return 0;
1639}
1640
1641static int hdac_hdmi_runtime_resume(struct device *dev)
1642{
1643 struct hdac_ext_device *edev = to_hda_ext_device(dev);
1644 struct hdac_device *hdac = &edev->hdac;
Subhransu S. Prusty07f083a2015-11-10 18:42:10 +05301645 struct hdac_bus *bus = hdac->bus;
1646 int err;
Subhransu S. Prustye342ac02015-11-10 18:42:07 +05301647
1648 dev_dbg(dev, "Enter: %s\n", __func__);
1649
Subhransu S. Prusty07f083a2015-11-10 18:42:10 +05301650 /* controller may not have been initialized for the first time */
1651 if (!bus)
1652 return 0;
1653
1654 err = snd_hdac_display_power(bus, true);
1655 if (err < 0) {
1656 dev_err(bus->dev, "Cannot turn on display power on i915\n");
1657 return err;
1658 }
1659
Subhransu S. Prustyab85f5b2016-02-17 21:34:02 +05301660 hdac_hdmi_skl_enable_all_pins(&edev->hdac);
1661 hdac_hdmi_skl_enable_dp12(&edev->hdac);
1662
Subhransu S. Prustye342ac02015-11-10 18:42:07 +05301663 /* Power up afg */
1664 if (!snd_hdac_check_power_state(hdac, hdac->afg, AC_PWRST_D0))
1665 snd_hdac_codec_write(hdac, hdac->afg, 0,
1666 AC_VERB_SET_POWER_STATE, AC_PWRST_D0);
1667
1668 return 0;
1669}
1670#else
1671#define hdac_hdmi_runtime_suspend NULL
1672#define hdac_hdmi_runtime_resume NULL
1673#endif
1674
1675static const struct dev_pm_ops hdac_hdmi_pm = {
1676 SET_RUNTIME_PM_OPS(hdac_hdmi_runtime_suspend, hdac_hdmi_runtime_resume, NULL)
1677};
1678
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301679static const struct hda_device_id hdmi_list[] = {
1680 HDA_CODEC_EXT_ENTRY(0x80862809, 0x100000, "Skylake HDMI", 0),
Jeeja KPe2304802016-03-11 10:12:55 +05301681 HDA_CODEC_EXT_ENTRY(0x8086280a, 0x100000, "Broxton HDMI", 0),
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301682 {}
1683};
1684
1685MODULE_DEVICE_TABLE(hdaudio, hdmi_list);
1686
1687static struct hdac_ext_driver hdmi_driver = {
1688 . hdac = {
1689 .driver = {
1690 .name = "HDMI HDA Codec",
Subhransu S. Prustye342ac02015-11-10 18:42:07 +05301691 .pm = &hdac_hdmi_pm,
Subhransu S. Prusty18382ea2015-11-10 18:42:06 +05301692 },
1693 .id_table = hdmi_list,
1694 },
1695 .probe = hdac_hdmi_dev_probe,
1696 .remove = hdac_hdmi_dev_remove,
1697};
1698
1699static int __init hdmi_init(void)
1700{
1701 return snd_hda_ext_driver_register(&hdmi_driver);
1702}
1703
1704static void __exit hdmi_exit(void)
1705{
1706 snd_hda_ext_driver_unregister(&hdmi_driver);
1707}
1708
1709module_init(hdmi_init);
1710module_exit(hdmi_exit);
1711
1712MODULE_LICENSE("GPL v2");
1713MODULE_DESCRIPTION("HDMI HD codec");
1714MODULE_AUTHOR("Samreen Nilofer<samreen.nilofer@intel.com>");
1715MODULE_AUTHOR("Subhransu S. Prusty<subhransu.s.prusty@intel.com>");