1 // SPDX-License-Identifier: (GPL-2.0-only OR BSD-3-Clause)
2 // Copyright(c) 2015-2023 Intel Corporation. All rights reserved.
4 #include <linux/acpi.h>
5 #include <linux/soundwire/sdw_registers.h>
6 #include <linux/soundwire/sdw.h>
7 #include <linux/soundwire/sdw_intel.h>
8 #include "cadence_master.h"
12 int intel_start_bus(struct sdw_intel *sdw)
14 struct device *dev = sdw->cdns.dev;
15 struct sdw_cdns *cdns = &sdw->cdns;
16 struct sdw_bus *bus = &cdns->bus;
20 * follow recommended programming flows to avoid timeouts when
24 sdw_intel_sync_arm(sdw);
26 ret = sdw_cdns_init(cdns);
28 dev_err(dev, "%s: unable to initialize Cadence IP: %d\n", __func__, ret);
32 sdw_cdns_config_update(cdns);
34 if (bus->multi_link) {
35 ret = sdw_intel_sync_go(sdw);
37 dev_err(dev, "%s: sync go failed: %d\n", __func__, ret);
42 ret = sdw_cdns_config_update_set_wait(cdns);
44 dev_err(dev, "%s: CONFIG_UPDATE BIT still set\n", __func__);
48 ret = sdw_cdns_exit_reset(cdns);
50 dev_err(dev, "%s: unable to exit bus reset sequence: %d\n", __func__, ret);
54 ret = sdw_cdns_enable_interrupt(cdns, true);
56 dev_err(dev, "%s: cannot enable interrupts: %d\n", __func__, ret);
60 sdw_cdns_check_self_clearing_bits(cdns, __func__,
61 true, INTEL_MASTER_RESET_ITERATIONS);
66 int intel_start_bus_after_reset(struct sdw_intel *sdw)
68 struct device *dev = sdw->cdns.dev;
69 struct sdw_cdns *cdns = &sdw->cdns;
70 struct sdw_bus *bus = &cdns->bus;
76 * An exception condition occurs for the CLK_STOP_BUS_RESET
77 * case if one or more masters remain active. In this condition,
78 * all the masters are powered on for they are in the same power
79 * domain. Master can preserve its context for clock stop0, so
80 * there is no need to clear slave status and reset bus.
82 clock_stop0 = sdw_cdns_is_clock_stop(&sdw->cdns);
87 * make sure all Slaves are tagged as UNATTACHED and
88 * provide reason for reinitialization
91 status = SDW_UNATTACH_REQUEST_MASTER_RESET;
92 sdw_clear_slave_status(bus, status);
95 * follow recommended programming flows to avoid
96 * timeouts when gsync is enabled
99 sdw_intel_sync_arm(sdw);
102 * Re-initialize the IP since it was powered-off
104 sdw_cdns_init(&sdw->cdns);
107 ret = sdw_cdns_enable_interrupt(cdns, true);
109 dev_err(dev, "cannot enable interrupts during resume\n");
114 ret = sdw_cdns_clock_restart(cdns, !clock_stop0);
116 dev_err(dev, "unable to restart clock during resume\n");
118 sdw_cdns_enable_interrupt(cdns, false);
123 sdw_cdns_config_update(cdns);
125 if (bus->multi_link) {
126 ret = sdw_intel_sync_go(sdw);
128 dev_err(sdw->cdns.dev, "sync go failed during resume\n");
133 ret = sdw_cdns_config_update_set_wait(cdns);
135 dev_err(dev, "%s: CONFIG_UPDATE BIT still set\n", __func__);
139 ret = sdw_cdns_exit_reset(cdns);
141 dev_err(dev, "unable to exit bus reset sequence during resume\n");
145 ret = sdw_cdns_enable_interrupt(cdns, true);
147 dev_err(dev, "cannot enable interrupts during resume\n");
152 sdw_cdns_check_self_clearing_bits(cdns, __func__, true, INTEL_MASTER_RESET_ITERATIONS);
157 void intel_check_clock_stop(struct sdw_intel *sdw)
159 struct device *dev = sdw->cdns.dev;
162 clock_stop0 = sdw_cdns_is_clock_stop(&sdw->cdns);
164 dev_err(dev, "%s: invalid configuration, clock was not stopped\n", __func__);
167 int intel_start_bus_after_clock_stop(struct sdw_intel *sdw)
169 struct device *dev = sdw->cdns.dev;
170 struct sdw_cdns *cdns = &sdw->cdns;
173 ret = sdw_cdns_clock_restart(cdns, false);
175 dev_err(dev, "%s: unable to restart clock: %d\n", __func__, ret);
179 ret = sdw_cdns_enable_interrupt(cdns, true);
181 dev_err(dev, "%s: cannot enable interrupts: %d\n", __func__, ret);
185 sdw_cdns_check_self_clearing_bits(cdns, __func__, true, INTEL_MASTER_RESET_ITERATIONS);
190 int intel_stop_bus(struct sdw_intel *sdw, bool clock_stop)
192 struct device *dev = sdw->cdns.dev;
193 struct sdw_cdns *cdns = &sdw->cdns;
194 bool wake_enable = false;
198 ret = sdw_cdns_clock_stop(cdns, true);
200 dev_err(dev, "%s: cannot stop clock: %d\n", __func__, ret);
205 ret = sdw_cdns_enable_interrupt(cdns, false);
207 dev_err(dev, "%s: cannot disable interrupts: %d\n", __func__, ret);
211 ret = sdw_intel_link_power_down(sdw);
213 dev_err(dev, "%s: Link power down failed: %d\n", __func__, ret);
217 sdw_intel_shim_wake(sdw, wake_enable);
223 * bank switch routines
226 int intel_pre_bank_switch(struct sdw_intel *sdw)
228 struct sdw_cdns *cdns = &sdw->cdns;
229 struct sdw_bus *bus = &cdns->bus;
231 /* Write to register only for multi-link */
232 if (!bus->multi_link)
235 sdw_intel_sync_arm(sdw);
240 int intel_post_bank_switch(struct sdw_intel *sdw)
242 struct sdw_cdns *cdns = &sdw->cdns;
243 struct sdw_bus *bus = &cdns->bus;
246 /* Write to register only for multi-link */
247 if (!bus->multi_link)
250 mutex_lock(sdw->link_res->shim_lock);
253 * post_bank_switch() ops is called from the bus in loop for
254 * all the Masters in the steam with the expectation that
255 * we trigger the bankswitch for the only first Master in the list
256 * and do nothing for the other Masters
258 * So, set the SYNCGO bit only if CMDSYNC bit is set for any Master.
260 if (sdw_intel_sync_check_cmdsync_unlocked(sdw))
261 ret = sdw_intel_sync_go_unlocked(sdw);
263 mutex_unlock(sdw->link_res->shim_lock);
266 dev_err(sdw->cdns.dev, "Post bank switch failed: %d\n", ret);