>-----Original Message----- >From: Takashi Iwai [mailto:tiwai@xxxxxxx] >Sent: Monday, February 26, 2018 3:55 PM >To: Ughreja, Rakesh A <rakesh.a.ughreja@xxxxxxxxx> >Cc: alsa-devel@xxxxxxxxxxxxxxxx; broonie@xxxxxxxxxx; >liam.r.girdwood@xxxxxxxxxxxxxxx; pierre-louis.bossart@xxxxxxxxxxxxxxx; Koul, >Vinod <vinod.koul@xxxxxxxxx>; Patches Audio <patches.audio@xxxxxxxxx> >Subject: Re: [PATCH v1 8/9] ASoC: hdac_hda: add asoc extension for legacy HDA >codec drivers > >On Mon, 26 Feb 2018 11:11:44 +0100, >Ughreja, Rakesh A wrote: >> >> >> >> >-----Original Message----- >> >From: Takashi Iwai [mailto:tiwai@xxxxxxx] >> >Sent: Friday, February 23, 2018 3:48 PM >> >To: Ughreja, Rakesh A <rakesh.a.ughreja@xxxxxxxxx> >> >Cc: alsa-devel@xxxxxxxxxxxxxxxx; broonie@xxxxxxxxxx; >> >liam.r.girdwood@xxxxxxxxxxxxxxx; pierre-louis.bossart@xxxxxxxxxxxxxxx; Koul, >> >Vinod <vinod.koul@xxxxxxxxx>; Patches Audio <patches.audio@xxxxxxxxx> >> >Subject: Re: [PATCH v1 8/9] ASoC: hdac_hda: add asoc extension for legacy >HDA >> >codec drivers >> > >> >On Fri, 23 Feb 2018 09:12:29 +0100, >> >Rakesh Ughreja wrote: >> >> >> >> +static int hdac_hda_codec_probe(struct snd_soc_codec *codec) >> >> +{ >> >> + struct hdac_hda_priv *hda_pvt = snd_soc_codec_get_drvdata(codec); >> >> + struct snd_soc_dapm_context *dapm = >> >> + snd_soc_component_get_dapm(&codec- >> >>component); >> >> + struct hdac_device *hdev = &hda_pvt->codec.core; >> >> + struct hda_codec *hcodec = &hda_pvt->codec; >> >> + struct hdac_ext_link *hlink = NULL; >> >> + hda_codec_patch_t patch; >> >> + int ret, i = 0; >> >> + u16 codec_mask; >> >> + >> >> + hda_pvt->scodec = codec; >> >> + hlink = snd_hdac_ext_bus_get_link(hdev->bus, dev_name(&hdev->dev)); >> >> + if (!hlink) { >> >> + dev_err(&hdev->dev, "hdac link not found\n"); >> >> + return -EIO; >> >> + } >> >> + >> >> + ret = snd_hdac_ext_bus_link_get(hdev->bus, hlink); >> > >> >So this is the essential part for the ext-hda stuff. But... >> > >> >> + >> >> + udelay(1000); >> >> + do { >> >> + codec_mask = snd_hdac_chip_readw(hdev->bus, STATESTS); >> >> + if (codec_mask) >> >> + break; >> >> + i++; >> >> + udelay(100); >> >> + } while (i < 100); >> >> + >> >> + if (!codec_mask) { >> >> + dev_err(&hdev->dev, "No codecs found after link reset\n"); >> >> + return -EIO; >> >> + } >> >> + >> >> + snd_hdac_chip_writew(hdev->bus, STATESTS, STATESTS_INT_MASK); >> > >> >Why do you need this? The callback gets called by the HD-audio >> >controller driver and it already should have checked the codec mask >> >bits. >> >> With the multiple link support on the same HDA controller, when the link >> gets turned ON immediately codec may not indicate its presence. As per >> the HDA spec, it may take up to 521uSec before the codec responds. >> >> In the legacy HDA the link was turned ON/OFF only during CRST time, but >> now it can happen anytime. > >This must be documented. Otherwise no one understands it. > >And I still don't think such stuff belonging to the codec driver. >It's rather a job of the controller driver to assure the codec access. >If any, we should fix that side instead. Yes, I can move this into the API implementation. So when we call snd_hdac_ext_bus_link_get it will return only after making sure that the codecs have responded with the status in the STS register. > > >> > >> >> + ret = snd_hda_codec_device_new(hcodec->bus, >> >> + codec->component.card->snd_card, hdev->addr, >> >hcodec); >> >> + if (ret < 0) { >> >> + dev_err(codec->dev, "failed to create hda codec %d\n", ret); >> >> + return ret; >> >> + } >> >> + >> >> + /* >> >> + * snd_hda_codec_new1 decrements the usage count and so get the pm >> >> + * count else the device will be powered off >> >> + */ >> >> + pm_runtime_get_noresume(&hdev->dev); >> >> + >> >> + hcodec->bus->card = dapm->card->snd_card; >> >> + >> >> + patch = (hda_codec_patch_t)hcodec->preset->driver_data; >> >> + if (patch) { >> >> + ret = patch(hcodec); >> >> + if (ret < 0) { >> >> + dev_err(codec->dev, "patch failed %d\n", ret); >> >> + return ret; >> >> + } >> >> + } else { >> >> + dev_dbg(&hdev->dev, "no patch file found\n"); >> >> + } >> >> + >> >> + ret = snd_hda_codec_set_name(hcodec, hcodec->preset->name); >> >> + if (ret < 0) { >> >> + dev_err(codec->dev, "name failed %s\n", hcodec->preset- >> >>name); >> >> + return ret; >> >> + } >> >> + >> >> + ret = snd_hdac_regmap_init(&hcodec->core); >> >> + if (ret < 0) { >> >> + dev_err(codec->dev, "regmap init failed\n"); >> >> + return ret; >> >> + } >> > >> >The regmap and the codec name must be initialized before calling the >> >patch (i.e. the real probe stuff). >> >> Okay. >> >> > >> >> + >> >> + ret = snd_hda_codec_parse_pcms(hcodec); >> >> + if (ret < 0) { >> >> + dev_err(&hdev->dev, "unable to map pcms to dai %d\n", ret); >> >> + return ret; >> >> + } >> >> + >> >> + ret = snd_hda_codec_build_controls(hcodec); >> >> + if (ret < 0) { >> >> + dev_err(&hdev->dev, "unable to create controls %d\n", ret); >> >> + return ret; >> >> + } >> >> + >> >> + hcodec->core.lazy_cache = true; >> >> + >> >> + /* >> >> + * hdac_device core already sets the state to active and calls >> >> + * get_noresume. So enable runtime and set the device to suspend. >> >> + * pm_runtime_enable is also called during codec registeration >> >> + */ >> >> + pm_runtime_put(&hdev->dev); >> >> + pm_runtime_suspend(&hdev->dev); >> >> + >> >> + return 0; >> >> +} >> >> + >> >> +static int hdac_hda_codec_remove(struct snd_soc_codec *codec) >> >> +{ >> >> + struct hdac_hda_priv *hda_pvt = snd_soc_codec_get_drvdata(codec); >> >> + struct hdac_device *hdev = &hda_pvt->codec.core; >> >> + struct hdac_ext_link *hlink = NULL; >> >> + >> >> + hlink = snd_hdac_ext_bus_get_link(hdev->bus, dev_name(&hdev->dev)); >> >> + if (!hlink) { >> >> + dev_err(&hdev->dev, "hdac link not found\n"); >> >> + return -EIO; >> >> + } >> >> + >> >> + snd_hdac_ext_bus_link_put(hdev->bus, hlink); >> >> + pm_runtime_disable(&hdev->dev); >> >> + >> >> + return 0; >> >> +} >> > >> >.... and I see no proper error paths there, and some cleanups seem >> >missing, too. >> >> Surely, I can correct it. Do you mind giving some more hints. >> >> > >> >Now I think what you need is rather not to open-code again the mostly >> >same procedure from hda_codec_driver_probe() / _remove(), but to let >> >hda_codec_driver_probe() and remove() skip some unnecessary steps for >> >the ext codec (e.g. registration), in addition to the hlink setup. >> >> I think you gave this suggestion in the last series and I tried that [1]. >> But I think we didn't conclude on that. So let's do it now. >> >> All the functions exception regmap_init which are called in the >> hda_codec_driver_probe requies snd_card and I don't have that until >> machine driver creates it. So we will end up in skipping/not calling all the >> functions except the regmap_init(). >> >> If that looks okay to you, I can do that. > >Ah crap, now I see the point. The confusion comes from that you have >two probe and two remove functions. How about renaming the ext_ops >ones? Oh okay, instead of "probe" and "remove", shall I call them "hdev_probe" and "hdev_remove" to indicate that its related to HDA Device probe at the bus level and not the ASoC porbe. Regards, Rakesh _______________________________________________ Alsa-devel mailing list Alsa-devel@xxxxxxxxxxxxxxxx http://mailman.alsa-project.org/mailman/listinfo/alsa-devel