162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci *  hdac-ext-controller.c - HD-audio extended controller functions.
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci *  Copyright (C) 2014-2015 Intel Corp
662306a36Sopenharmony_ci *  Author: Jeeja KP <jeeja.kp@intel.com>
762306a36Sopenharmony_ci *  ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
1062306a36Sopenharmony_ci */
1162306a36Sopenharmony_ci
1262306a36Sopenharmony_ci#include <linux/delay.h>
1362306a36Sopenharmony_ci#include <linux/slab.h>
1462306a36Sopenharmony_ci#include <sound/hda_register.h>
1562306a36Sopenharmony_ci#include <sound/hdaudio_ext.h>
1662306a36Sopenharmony_ci
1762306a36Sopenharmony_ci/*
1862306a36Sopenharmony_ci * processing pipe helpers - these helpers are useful for dealing with HDA
1962306a36Sopenharmony_ci * new capability of processing pipelines
2062306a36Sopenharmony_ci */
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_ci/**
2362306a36Sopenharmony_ci * snd_hdac_ext_bus_ppcap_enable - enable/disable processing pipe capability
2462306a36Sopenharmony_ci * @bus: the pointer to HDAC bus object
2562306a36Sopenharmony_ci * @enable: flag to turn on/off the capability
2662306a36Sopenharmony_ci */
2762306a36Sopenharmony_civoid snd_hdac_ext_bus_ppcap_enable(struct hdac_bus *bus, bool enable)
2862306a36Sopenharmony_ci{
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci	if (!bus->ppcap) {
3162306a36Sopenharmony_ci		dev_err(bus->dev, "Address of PP capability is NULL");
3262306a36Sopenharmony_ci		return;
3362306a36Sopenharmony_ci	}
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_ci	if (enable)
3662306a36Sopenharmony_ci		snd_hdac_updatel(bus->ppcap, AZX_REG_PP_PPCTL,
3762306a36Sopenharmony_ci				 AZX_PPCTL_GPROCEN, AZX_PPCTL_GPROCEN);
3862306a36Sopenharmony_ci	else
3962306a36Sopenharmony_ci		snd_hdac_updatel(bus->ppcap, AZX_REG_PP_PPCTL,
4062306a36Sopenharmony_ci				 AZX_PPCTL_GPROCEN, 0);
4162306a36Sopenharmony_ci}
4262306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_ppcap_enable);
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_ci/**
4562306a36Sopenharmony_ci * snd_hdac_ext_bus_ppcap_int_enable - ppcap interrupt enable/disable
4662306a36Sopenharmony_ci * @bus: the pointer to HDAC bus object
4762306a36Sopenharmony_ci * @enable: flag to enable/disable interrupt
4862306a36Sopenharmony_ci */
4962306a36Sopenharmony_civoid snd_hdac_ext_bus_ppcap_int_enable(struct hdac_bus *bus, bool enable)
5062306a36Sopenharmony_ci{
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci	if (!bus->ppcap) {
5362306a36Sopenharmony_ci		dev_err(bus->dev, "Address of PP capability is NULL\n");
5462306a36Sopenharmony_ci		return;
5562306a36Sopenharmony_ci	}
5662306a36Sopenharmony_ci
5762306a36Sopenharmony_ci	if (enable)
5862306a36Sopenharmony_ci		snd_hdac_updatel(bus->ppcap, AZX_REG_PP_PPCTL,
5962306a36Sopenharmony_ci				 AZX_PPCTL_PIE, AZX_PPCTL_PIE);
6062306a36Sopenharmony_ci	else
6162306a36Sopenharmony_ci		snd_hdac_updatel(bus->ppcap, AZX_REG_PP_PPCTL,
6262306a36Sopenharmony_ci				 AZX_PPCTL_PIE, 0);
6362306a36Sopenharmony_ci}
6462306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_ppcap_int_enable);
6562306a36Sopenharmony_ci
6662306a36Sopenharmony_ci/*
6762306a36Sopenharmony_ci * Multilink helpers - these helpers are useful for dealing with HDA
6862306a36Sopenharmony_ci * new multilink capability
6962306a36Sopenharmony_ci */
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci/**
7262306a36Sopenharmony_ci * snd_hdac_ext_bus_get_ml_capabilities - get multilink capability
7362306a36Sopenharmony_ci * @bus: the pointer to HDAC bus object
7462306a36Sopenharmony_ci *
7562306a36Sopenharmony_ci * This will parse all links and read the mlink capabilities and add them
7662306a36Sopenharmony_ci * in hlink_list of extended hdac bus
7762306a36Sopenharmony_ci * Note: this will be freed on bus exit by driver
7862306a36Sopenharmony_ci */
7962306a36Sopenharmony_ciint snd_hdac_ext_bus_get_ml_capabilities(struct hdac_bus *bus)
8062306a36Sopenharmony_ci{
8162306a36Sopenharmony_ci	int idx;
8262306a36Sopenharmony_ci	u32 link_count;
8362306a36Sopenharmony_ci	struct hdac_ext_link *hlink;
8462306a36Sopenharmony_ci
8562306a36Sopenharmony_ci	link_count = readl(bus->mlcap + AZX_REG_ML_MLCD) + 1;
8662306a36Sopenharmony_ci
8762306a36Sopenharmony_ci	dev_dbg(bus->dev, "In %s Link count: %d\n", __func__, link_count);
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_ci	for (idx = 0; idx < link_count; idx++) {
9062306a36Sopenharmony_ci		hlink  = kzalloc(sizeof(*hlink), GFP_KERNEL);
9162306a36Sopenharmony_ci		if (!hlink)
9262306a36Sopenharmony_ci			return -ENOMEM;
9362306a36Sopenharmony_ci		hlink->index = idx;
9462306a36Sopenharmony_ci		hlink->bus = bus;
9562306a36Sopenharmony_ci		hlink->ml_addr = bus->mlcap + AZX_ML_BASE +
9662306a36Sopenharmony_ci					(AZX_ML_INTERVAL * idx);
9762306a36Sopenharmony_ci		hlink->lcaps  = readl(hlink->ml_addr + AZX_REG_ML_LCAP);
9862306a36Sopenharmony_ci		hlink->lsdiid = readw(hlink->ml_addr + AZX_REG_ML_LSDIID);
9962306a36Sopenharmony_ci
10062306a36Sopenharmony_ci		/* since link in On, update the ref */
10162306a36Sopenharmony_ci		hlink->ref_count = 1;
10262306a36Sopenharmony_ci
10362306a36Sopenharmony_ci		list_add_tail(&hlink->list, &bus->hlink_list);
10462306a36Sopenharmony_ci	}
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ci	return 0;
10762306a36Sopenharmony_ci}
10862306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_get_ml_capabilities);
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_ci/**
11162306a36Sopenharmony_ci * snd_hdac_ext_link_free_all- free hdac extended link objects
11262306a36Sopenharmony_ci *
11362306a36Sopenharmony_ci * @bus: the pointer to HDAC bus object
11462306a36Sopenharmony_ci */
11562306a36Sopenharmony_ci
11662306a36Sopenharmony_civoid snd_hdac_ext_link_free_all(struct hdac_bus *bus)
11762306a36Sopenharmony_ci{
11862306a36Sopenharmony_ci	struct hdac_ext_link *hlink;
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_ci	while (!list_empty(&bus->hlink_list)) {
12162306a36Sopenharmony_ci		hlink = list_first_entry(&bus->hlink_list, struct hdac_ext_link, list);
12262306a36Sopenharmony_ci		list_del(&hlink->list);
12362306a36Sopenharmony_ci		kfree(hlink);
12462306a36Sopenharmony_ci	}
12562306a36Sopenharmony_ci}
12662306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_link_free_all);
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci/**
12962306a36Sopenharmony_ci * snd_hdac_ext_bus_get_hlink_by_addr - get hlink at specified address
13062306a36Sopenharmony_ci * @bus: hlink's parent bus device
13162306a36Sopenharmony_ci * @addr: codec device address
13262306a36Sopenharmony_ci *
13362306a36Sopenharmony_ci * Returns hlink object or NULL if matching hlink is not found.
13462306a36Sopenharmony_ci */
13562306a36Sopenharmony_cistruct hdac_ext_link *snd_hdac_ext_bus_get_hlink_by_addr(struct hdac_bus *bus, int addr)
13662306a36Sopenharmony_ci{
13762306a36Sopenharmony_ci	struct hdac_ext_link *hlink;
13862306a36Sopenharmony_ci
13962306a36Sopenharmony_ci	list_for_each_entry(hlink, &bus->hlink_list, list)
14062306a36Sopenharmony_ci		if (hlink->lsdiid & (0x1 << addr))
14162306a36Sopenharmony_ci			return hlink;
14262306a36Sopenharmony_ci	return NULL;
14362306a36Sopenharmony_ci}
14462306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_get_hlink_by_addr);
14562306a36Sopenharmony_ci
14662306a36Sopenharmony_ci/**
14762306a36Sopenharmony_ci * snd_hdac_ext_bus_get_hlink_by_name - get hlink based on codec name
14862306a36Sopenharmony_ci * @bus: the pointer to HDAC bus object
14962306a36Sopenharmony_ci * @codec_name: codec name
15062306a36Sopenharmony_ci */
15162306a36Sopenharmony_cistruct hdac_ext_link *snd_hdac_ext_bus_get_hlink_by_name(struct hdac_bus *bus,
15262306a36Sopenharmony_ci							 const char *codec_name)
15362306a36Sopenharmony_ci{
15462306a36Sopenharmony_ci	int bus_idx, addr;
15562306a36Sopenharmony_ci
15662306a36Sopenharmony_ci	if (sscanf(codec_name, "ehdaudio%dD%d", &bus_idx, &addr) != 2)
15762306a36Sopenharmony_ci		return NULL;
15862306a36Sopenharmony_ci	if (bus->idx != bus_idx)
15962306a36Sopenharmony_ci		return NULL;
16062306a36Sopenharmony_ci	if (addr < 0 || addr > 31)
16162306a36Sopenharmony_ci		return NULL;
16262306a36Sopenharmony_ci
16362306a36Sopenharmony_ci	return snd_hdac_ext_bus_get_hlink_by_addr(bus, addr);
16462306a36Sopenharmony_ci}
16562306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_get_hlink_by_name);
16662306a36Sopenharmony_ci
16762306a36Sopenharmony_cistatic int check_hdac_link_power_active(struct hdac_ext_link *hlink, bool enable)
16862306a36Sopenharmony_ci{
16962306a36Sopenharmony_ci	int timeout;
17062306a36Sopenharmony_ci	u32 val;
17162306a36Sopenharmony_ci	int mask = (1 << AZX_ML_LCTL_CPA_SHIFT);
17262306a36Sopenharmony_ci
17362306a36Sopenharmony_ci	udelay(3);
17462306a36Sopenharmony_ci	timeout = 150;
17562306a36Sopenharmony_ci
17662306a36Sopenharmony_ci	do {
17762306a36Sopenharmony_ci		val = readl(hlink->ml_addr + AZX_REG_ML_LCTL);
17862306a36Sopenharmony_ci		if (enable) {
17962306a36Sopenharmony_ci			if (((val & mask) >> AZX_ML_LCTL_CPA_SHIFT))
18062306a36Sopenharmony_ci				return 0;
18162306a36Sopenharmony_ci		} else {
18262306a36Sopenharmony_ci			if (!((val & mask) >> AZX_ML_LCTL_CPA_SHIFT))
18362306a36Sopenharmony_ci				return 0;
18462306a36Sopenharmony_ci		}
18562306a36Sopenharmony_ci		udelay(3);
18662306a36Sopenharmony_ci	} while (--timeout);
18762306a36Sopenharmony_ci
18862306a36Sopenharmony_ci	return -EIO;
18962306a36Sopenharmony_ci}
19062306a36Sopenharmony_ci
19162306a36Sopenharmony_ci/**
19262306a36Sopenharmony_ci * snd_hdac_ext_bus_link_power_up -power up hda link
19362306a36Sopenharmony_ci * @hlink: HD-audio extended link
19462306a36Sopenharmony_ci */
19562306a36Sopenharmony_ciint snd_hdac_ext_bus_link_power_up(struct hdac_ext_link *hlink)
19662306a36Sopenharmony_ci{
19762306a36Sopenharmony_ci	snd_hdac_updatel(hlink->ml_addr, AZX_REG_ML_LCTL,
19862306a36Sopenharmony_ci			 AZX_ML_LCTL_SPA, AZX_ML_LCTL_SPA);
19962306a36Sopenharmony_ci
20062306a36Sopenharmony_ci	return check_hdac_link_power_active(hlink, true);
20162306a36Sopenharmony_ci}
20262306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_link_power_up);
20362306a36Sopenharmony_ci
20462306a36Sopenharmony_ci/**
20562306a36Sopenharmony_ci * snd_hdac_ext_bus_link_power_down -power down hda link
20662306a36Sopenharmony_ci * @hlink: HD-audio extended link
20762306a36Sopenharmony_ci */
20862306a36Sopenharmony_ciint snd_hdac_ext_bus_link_power_down(struct hdac_ext_link *hlink)
20962306a36Sopenharmony_ci{
21062306a36Sopenharmony_ci	snd_hdac_updatel(hlink->ml_addr, AZX_REG_ML_LCTL, AZX_ML_LCTL_SPA, 0);
21162306a36Sopenharmony_ci
21262306a36Sopenharmony_ci	return check_hdac_link_power_active(hlink, false);
21362306a36Sopenharmony_ci}
21462306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_link_power_down);
21562306a36Sopenharmony_ci
21662306a36Sopenharmony_ci/**
21762306a36Sopenharmony_ci * snd_hdac_ext_bus_link_power_up_all -power up all hda link
21862306a36Sopenharmony_ci * @bus: the pointer to HDAC bus object
21962306a36Sopenharmony_ci */
22062306a36Sopenharmony_ciint snd_hdac_ext_bus_link_power_up_all(struct hdac_bus *bus)
22162306a36Sopenharmony_ci{
22262306a36Sopenharmony_ci	struct hdac_ext_link *hlink = NULL;
22362306a36Sopenharmony_ci	int ret;
22462306a36Sopenharmony_ci
22562306a36Sopenharmony_ci	list_for_each_entry(hlink, &bus->hlink_list, list) {
22662306a36Sopenharmony_ci		ret = snd_hdac_ext_bus_link_power_up(hlink);
22762306a36Sopenharmony_ci		if (ret < 0)
22862306a36Sopenharmony_ci			return ret;
22962306a36Sopenharmony_ci	}
23062306a36Sopenharmony_ci
23162306a36Sopenharmony_ci	return 0;
23262306a36Sopenharmony_ci}
23362306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_link_power_up_all);
23462306a36Sopenharmony_ci
23562306a36Sopenharmony_ci/**
23662306a36Sopenharmony_ci * snd_hdac_ext_bus_link_power_down_all -power down all hda link
23762306a36Sopenharmony_ci * @bus: the pointer to HDAC bus object
23862306a36Sopenharmony_ci */
23962306a36Sopenharmony_ciint snd_hdac_ext_bus_link_power_down_all(struct hdac_bus *bus)
24062306a36Sopenharmony_ci{
24162306a36Sopenharmony_ci	struct hdac_ext_link *hlink = NULL;
24262306a36Sopenharmony_ci	int ret;
24362306a36Sopenharmony_ci
24462306a36Sopenharmony_ci	list_for_each_entry(hlink, &bus->hlink_list, list) {
24562306a36Sopenharmony_ci		ret = snd_hdac_ext_bus_link_power_down(hlink);
24662306a36Sopenharmony_ci		if (ret < 0)
24762306a36Sopenharmony_ci			return ret;
24862306a36Sopenharmony_ci	}
24962306a36Sopenharmony_ci
25062306a36Sopenharmony_ci	return 0;
25162306a36Sopenharmony_ci}
25262306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_link_power_down_all);
25362306a36Sopenharmony_ci
25462306a36Sopenharmony_ci/**
25562306a36Sopenharmony_ci * snd_hdac_ext_bus_link_set_stream_id - maps stream id to link output
25662306a36Sopenharmony_ci * @link: HD-audio ext link to set up
25762306a36Sopenharmony_ci * @stream: stream id
25862306a36Sopenharmony_ci */
25962306a36Sopenharmony_civoid snd_hdac_ext_bus_link_set_stream_id(struct hdac_ext_link *link,
26062306a36Sopenharmony_ci					 int stream)
26162306a36Sopenharmony_ci{
26262306a36Sopenharmony_ci	snd_hdac_updatew(link->ml_addr, AZX_REG_ML_LOSIDV, (1 << stream), 1 << stream);
26362306a36Sopenharmony_ci}
26462306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_link_set_stream_id);
26562306a36Sopenharmony_ci
26662306a36Sopenharmony_ci/**
26762306a36Sopenharmony_ci * snd_hdac_ext_bus_link_clear_stream_id - maps stream id to link output
26862306a36Sopenharmony_ci * @link: HD-audio ext link to set up
26962306a36Sopenharmony_ci * @stream: stream id
27062306a36Sopenharmony_ci */
27162306a36Sopenharmony_civoid snd_hdac_ext_bus_link_clear_stream_id(struct hdac_ext_link *link,
27262306a36Sopenharmony_ci					   int stream)
27362306a36Sopenharmony_ci{
27462306a36Sopenharmony_ci	snd_hdac_updatew(link->ml_addr, AZX_REG_ML_LOSIDV, (1 << stream), 0);
27562306a36Sopenharmony_ci}
27662306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_link_clear_stream_id);
27762306a36Sopenharmony_ci
27862306a36Sopenharmony_ciint snd_hdac_ext_bus_link_get(struct hdac_bus *bus,
27962306a36Sopenharmony_ci				struct hdac_ext_link *hlink)
28062306a36Sopenharmony_ci{
28162306a36Sopenharmony_ci	unsigned long codec_mask;
28262306a36Sopenharmony_ci	int ret = 0;
28362306a36Sopenharmony_ci
28462306a36Sopenharmony_ci	mutex_lock(&bus->lock);
28562306a36Sopenharmony_ci
28662306a36Sopenharmony_ci	/*
28762306a36Sopenharmony_ci	 * if we move from 0 to 1, count will be 1 so power up this link
28862306a36Sopenharmony_ci	 * as well, also check the dma status and trigger that
28962306a36Sopenharmony_ci	 */
29062306a36Sopenharmony_ci	if (++hlink->ref_count == 1) {
29162306a36Sopenharmony_ci		if (!bus->cmd_dma_state) {
29262306a36Sopenharmony_ci			snd_hdac_bus_init_cmd_io(bus);
29362306a36Sopenharmony_ci			bus->cmd_dma_state = true;
29462306a36Sopenharmony_ci		}
29562306a36Sopenharmony_ci
29662306a36Sopenharmony_ci		ret = snd_hdac_ext_bus_link_power_up(hlink);
29762306a36Sopenharmony_ci
29862306a36Sopenharmony_ci		/*
29962306a36Sopenharmony_ci		 * clear the register to invalidate all the output streams
30062306a36Sopenharmony_ci		 */
30162306a36Sopenharmony_ci		snd_hdac_updatew(hlink->ml_addr, AZX_REG_ML_LOSIDV,
30262306a36Sopenharmony_ci				 AZX_ML_LOSIDV_STREAM_MASK, 0);
30362306a36Sopenharmony_ci		/*
30462306a36Sopenharmony_ci		 *  wait for 521usec for codec to report status
30562306a36Sopenharmony_ci		 *  HDA spec section 4.3 - Codec Discovery
30662306a36Sopenharmony_ci		 */
30762306a36Sopenharmony_ci		udelay(521);
30862306a36Sopenharmony_ci		codec_mask = snd_hdac_chip_readw(bus, STATESTS);
30962306a36Sopenharmony_ci		dev_dbg(bus->dev, "codec_mask = 0x%lx\n", codec_mask);
31062306a36Sopenharmony_ci		snd_hdac_chip_writew(bus, STATESTS, codec_mask);
31162306a36Sopenharmony_ci		if (!bus->codec_mask)
31262306a36Sopenharmony_ci			bus->codec_mask = codec_mask;
31362306a36Sopenharmony_ci	}
31462306a36Sopenharmony_ci
31562306a36Sopenharmony_ci	mutex_unlock(&bus->lock);
31662306a36Sopenharmony_ci	return ret;
31762306a36Sopenharmony_ci}
31862306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_link_get);
31962306a36Sopenharmony_ci
32062306a36Sopenharmony_ciint snd_hdac_ext_bus_link_put(struct hdac_bus *bus,
32162306a36Sopenharmony_ci			      struct hdac_ext_link *hlink)
32262306a36Sopenharmony_ci{
32362306a36Sopenharmony_ci	int ret = 0;
32462306a36Sopenharmony_ci	struct hdac_ext_link *hlink_tmp;
32562306a36Sopenharmony_ci	bool link_up = false;
32662306a36Sopenharmony_ci
32762306a36Sopenharmony_ci	mutex_lock(&bus->lock);
32862306a36Sopenharmony_ci
32962306a36Sopenharmony_ci	/*
33062306a36Sopenharmony_ci	 * if we move from 1 to 0, count will be 0
33162306a36Sopenharmony_ci	 * so power down this link as well
33262306a36Sopenharmony_ci	 */
33362306a36Sopenharmony_ci	if (--hlink->ref_count == 0) {
33462306a36Sopenharmony_ci		ret = snd_hdac_ext_bus_link_power_down(hlink);
33562306a36Sopenharmony_ci
33662306a36Sopenharmony_ci		/*
33762306a36Sopenharmony_ci		 * now check if all links are off, if so turn off
33862306a36Sopenharmony_ci		 * cmd dma as well
33962306a36Sopenharmony_ci		 */
34062306a36Sopenharmony_ci		list_for_each_entry(hlink_tmp, &bus->hlink_list, list) {
34162306a36Sopenharmony_ci			if (hlink_tmp->ref_count) {
34262306a36Sopenharmony_ci				link_up = true;
34362306a36Sopenharmony_ci				break;
34462306a36Sopenharmony_ci			}
34562306a36Sopenharmony_ci		}
34662306a36Sopenharmony_ci
34762306a36Sopenharmony_ci		if (!link_up) {
34862306a36Sopenharmony_ci			snd_hdac_bus_stop_cmd_io(bus);
34962306a36Sopenharmony_ci			bus->cmd_dma_state = false;
35062306a36Sopenharmony_ci		}
35162306a36Sopenharmony_ci	}
35262306a36Sopenharmony_ci
35362306a36Sopenharmony_ci	mutex_unlock(&bus->lock);
35462306a36Sopenharmony_ci	return ret;
35562306a36Sopenharmony_ci}
35662306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_link_put);
35762306a36Sopenharmony_ci
35862306a36Sopenharmony_cistatic void hdac_ext_codec_link_up(struct hdac_device *codec)
35962306a36Sopenharmony_ci{
36062306a36Sopenharmony_ci	const char *devname = dev_name(&codec->dev);
36162306a36Sopenharmony_ci	struct hdac_ext_link *hlink =
36262306a36Sopenharmony_ci		snd_hdac_ext_bus_get_hlink_by_name(codec->bus, devname);
36362306a36Sopenharmony_ci
36462306a36Sopenharmony_ci	if (hlink)
36562306a36Sopenharmony_ci		snd_hdac_ext_bus_link_get(codec->bus, hlink);
36662306a36Sopenharmony_ci}
36762306a36Sopenharmony_ci
36862306a36Sopenharmony_cistatic void hdac_ext_codec_link_down(struct hdac_device *codec)
36962306a36Sopenharmony_ci{
37062306a36Sopenharmony_ci	const char *devname = dev_name(&codec->dev);
37162306a36Sopenharmony_ci	struct hdac_ext_link *hlink =
37262306a36Sopenharmony_ci		snd_hdac_ext_bus_get_hlink_by_name(codec->bus, devname);
37362306a36Sopenharmony_ci
37462306a36Sopenharmony_ci	if (hlink)
37562306a36Sopenharmony_ci		snd_hdac_ext_bus_link_put(codec->bus, hlink);
37662306a36Sopenharmony_ci}
37762306a36Sopenharmony_ci
37862306a36Sopenharmony_civoid snd_hdac_ext_bus_link_power(struct hdac_device *codec, bool enable)
37962306a36Sopenharmony_ci{
38062306a36Sopenharmony_ci	struct hdac_bus *bus = codec->bus;
38162306a36Sopenharmony_ci	bool oldstate = test_bit(codec->addr, &bus->codec_powered);
38262306a36Sopenharmony_ci
38362306a36Sopenharmony_ci	if (enable == oldstate)
38462306a36Sopenharmony_ci		return;
38562306a36Sopenharmony_ci
38662306a36Sopenharmony_ci	snd_hdac_bus_link_power(codec, enable);
38762306a36Sopenharmony_ci
38862306a36Sopenharmony_ci	if (enable)
38962306a36Sopenharmony_ci		hdac_ext_codec_link_up(codec);
39062306a36Sopenharmony_ci	else
39162306a36Sopenharmony_ci		hdac_ext_codec_link_down(codec);
39262306a36Sopenharmony_ci}
39362306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(snd_hdac_ext_bus_link_power);
394