Commit 857a7c42 authored by Pierre-Louis Bossart's avatar Pierre-Louis Bossart Committed by Vinod Koul

soundwire: intel: add multi-link support

The multi-link support is enabled with a hardware gsync signal
connecting all links. All commands and operations which typically are
handled on an SSP boundary will be deferred further and enabled across
all links with the 'syncGo' sequence.
Signed-off-by: default avatarPierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
Signed-off-by: default avatarBard Liao <yung-chuan.liao@linux.intel.com>
Link: https://lore.kernel.org/r/20200901150556.19432-4-yung-chuan.liao@linux.intel.comSigned-off-by: default avatarVinod Koul <vkoul@kernel.org>
parent 0ef2986e
...@@ -34,6 +34,7 @@ ...@@ -34,6 +34,7 @@
#define SDW_INTEL_MASTER_DISABLE_PM_RUNTIME BIT(0) #define SDW_INTEL_MASTER_DISABLE_PM_RUNTIME BIT(0)
#define SDW_INTEL_MASTER_DISABLE_CLOCK_STOP BIT(1) #define SDW_INTEL_MASTER_DISABLE_CLOCK_STOP BIT(1)
#define SDW_INTEL_MASTER_DISABLE_PM_RUNTIME_IDLE BIT(2) #define SDW_INTEL_MASTER_DISABLE_PM_RUNTIME_IDLE BIT(2)
#define SDW_INTEL_MASTER_DISABLE_MULTI_LINK BIT(3)
static int md_flags; static int md_flags;
module_param_named(sdw_md_flags, md_flags, int, 0444); module_param_named(sdw_md_flags, md_flags, int, 0444);
...@@ -555,6 +556,19 @@ static int intel_shim_sync_go_unlocked(struct sdw_intel *sdw) ...@@ -555,6 +556,19 @@ static int intel_shim_sync_go_unlocked(struct sdw_intel *sdw)
return ret; return ret;
} }
static int intel_shim_sync_go(struct sdw_intel *sdw)
{
int ret;
mutex_lock(sdw->link_res->shim_lock);
ret = intel_shim_sync_go_unlocked(sdw);
mutex_unlock(sdw->link_res->shim_lock);
return ret;
}
/* /*
* PDI routines * PDI routines
*/ */
...@@ -1303,10 +1317,7 @@ static int intel_init(struct sdw_intel *sdw) ...@@ -1303,10 +1317,7 @@ static int intel_init(struct sdw_intel *sdw)
intel_shim_init(sdw, clock_stop); intel_shim_init(sdw, clock_stop);
if (clock_stop)
return 0; return 0;
return sdw_cdns_init(&sdw->cdns);
} }
/* /*
...@@ -1372,6 +1383,7 @@ int intel_master_startup(struct platform_device *pdev) ...@@ -1372,6 +1383,7 @@ int intel_master_startup(struct platform_device *pdev)
struct sdw_intel *sdw = cdns_to_intel(cdns); struct sdw_intel *sdw = cdns_to_intel(cdns);
struct sdw_bus *bus = &cdns->bus; struct sdw_bus *bus = &cdns->bus;
int link_flags; int link_flags;
bool multi_link;
u32 clock_stop_quirks; u32 clock_stop_quirks;
int ret; int ret;
...@@ -1382,7 +1394,16 @@ int intel_master_startup(struct platform_device *pdev) ...@@ -1382,7 +1394,16 @@ int intel_master_startup(struct platform_device *pdev)
return 0; return 0;
} }
/* Initialize shim, controller and Cadence IP */ link_flags = md_flags >> (bus->link_id * 8);
multi_link = !(link_flags & SDW_INTEL_MASTER_DISABLE_MULTI_LINK);
if (!multi_link) {
dev_dbg(dev, "Multi-link is disabled\n");
bus->multi_link = false;
} else {
bus->multi_link = true;
}
/* Initialize shim, controller */
ret = intel_init(sdw); ret = intel_init(sdw);
if (ret) if (ret)
goto err_init; goto err_init;
...@@ -1401,12 +1422,33 @@ int intel_master_startup(struct platform_device *pdev) ...@@ -1401,12 +1422,33 @@ int intel_master_startup(struct platform_device *pdev)
goto err_init; goto err_init;
} }
/*
* follow recommended programming flows to avoid timeouts when
* gsync is enabled
*/
if (multi_link)
intel_shim_sync_arm(sdw);
ret = sdw_cdns_init(cdns);
if (ret < 0) {
dev_err(dev, "unable to initialize Cadence IP\n");
goto err_interrupt;
}
ret = sdw_cdns_exit_reset(cdns); ret = sdw_cdns_exit_reset(cdns);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "unable to exit bus reset sequence\n"); dev_err(dev, "unable to exit bus reset sequence\n");
goto err_interrupt; goto err_interrupt;
} }
if (multi_link) {
ret = intel_shim_sync_go(sdw);
if (ret < 0) {
dev_err(dev, "sync go failed: %d\n", ret);
goto err_interrupt;
}
}
/* Register DAIs */ /* Register DAIs */
ret = intel_register_dai(sdw); ret = intel_register_dai(sdw);
if (ret) { if (ret) {
...@@ -1418,7 +1460,6 @@ int intel_master_startup(struct platform_device *pdev) ...@@ -1418,7 +1460,6 @@ int intel_master_startup(struct platform_device *pdev)
intel_debugfs_init(sdw); intel_debugfs_init(sdw);
/* Enable runtime PM */ /* Enable runtime PM */
link_flags = md_flags >> (bus->link_id * 8);
if (!(link_flags & SDW_INTEL_MASTER_DISABLE_PM_RUNTIME)) { if (!(link_flags & SDW_INTEL_MASTER_DISABLE_PM_RUNTIME)) {
pm_runtime_set_autosuspend_delay(dev, pm_runtime_set_autosuspend_delay(dev,
INTEL_MASTER_SUSPEND_DELAY_MS); INTEL_MASTER_SUSPEND_DELAY_MS);
...@@ -1654,6 +1695,7 @@ static int __maybe_unused intel_resume(struct device *dev) ...@@ -1654,6 +1695,7 @@ static int __maybe_unused intel_resume(struct device *dev)
struct sdw_intel *sdw = cdns_to_intel(cdns); struct sdw_intel *sdw = cdns_to_intel(cdns);
struct sdw_bus *bus = &cdns->bus; struct sdw_bus *bus = &cdns->bus;
int link_flags; int link_flags;
bool multi_link;
int ret; int ret;
if (bus->prop.hw_disabled) { if (bus->prop.hw_disabled) {
...@@ -1662,6 +1704,9 @@ static int __maybe_unused intel_resume(struct device *dev) ...@@ -1662,6 +1704,9 @@ static int __maybe_unused intel_resume(struct device *dev)
return 0; return 0;
} }
link_flags = md_flags >> (bus->link_id * 8);
multi_link = !(link_flags & SDW_INTEL_MASTER_DISABLE_MULTI_LINK);
if (pm_runtime_suspended(dev)) { if (pm_runtime_suspended(dev)) {
dev_dbg(dev, "%s: pm_runtime status was suspended, forcing active\n", __func__); dev_dbg(dev, "%s: pm_runtime status was suspended, forcing active\n", __func__);
...@@ -1672,6 +1717,7 @@ static int __maybe_unused intel_resume(struct device *dev) ...@@ -1672,6 +1717,7 @@ static int __maybe_unused intel_resume(struct device *dev)
pm_runtime_enable(dev); pm_runtime_enable(dev);
link_flags = md_flags >> (bus->link_id * 8); link_flags = md_flags >> (bus->link_id * 8);
if (!(link_flags & SDW_INTEL_MASTER_DISABLE_PM_RUNTIME_IDLE)) if (!(link_flags & SDW_INTEL_MASTER_DISABLE_PM_RUNTIME_IDLE))
pm_runtime_idle(dev); pm_runtime_idle(dev);
} }
...@@ -1694,12 +1740,33 @@ static int __maybe_unused intel_resume(struct device *dev) ...@@ -1694,12 +1740,33 @@ static int __maybe_unused intel_resume(struct device *dev)
return ret; return ret;
} }
/*
* follow recommended programming flows to avoid timeouts when
* gsync is enabled
*/
if (multi_link)
intel_shim_sync_arm(sdw);
ret = sdw_cdns_init(&sdw->cdns);
if (ret < 0) {
dev_err(dev, "unable to initialize Cadence IP during resume\n");
return ret;
}
ret = sdw_cdns_exit_reset(cdns); ret = sdw_cdns_exit_reset(cdns);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "unable to exit bus reset sequence during resume\n"); dev_err(dev, "unable to exit bus reset sequence during resume\n");
return ret; return ret;
} }
if (multi_link) {
ret = intel_shim_sync_go(sdw);
if (ret < 0) {
dev_err(dev, "sync go failed during resume\n");
return ret;
}
}
/* /*
* after system resume, the pm_runtime suspend() may kick in * after system resume, the pm_runtime suspend() may kick in
* during the enumeration, before any children device force the * during the enumeration, before any children device force the
...@@ -1722,6 +1789,8 @@ static int intel_resume_runtime(struct device *dev) ...@@ -1722,6 +1789,8 @@ static int intel_resume_runtime(struct device *dev)
struct sdw_bus *bus = &cdns->bus; struct sdw_bus *bus = &cdns->bus;
u32 clock_stop_quirks; u32 clock_stop_quirks;
bool clock_stop0; bool clock_stop0;
int link_flags;
bool multi_link;
int status; int status;
int ret; int ret;
...@@ -1731,6 +1800,9 @@ static int intel_resume_runtime(struct device *dev) ...@@ -1731,6 +1800,9 @@ static int intel_resume_runtime(struct device *dev)
return 0; return 0;
} }
link_flags = md_flags >> (bus->link_id * 8);
multi_link = !(link_flags & SDW_INTEL_MASTER_DISABLE_MULTI_LINK);
clock_stop_quirks = sdw->link_res->clock_stop_quirks; clock_stop_quirks = sdw->link_res->clock_stop_quirks;
if (clock_stop_quirks & SDW_INTEL_CLK_STOP_TEARDOWN) { if (clock_stop_quirks & SDW_INTEL_CLK_STOP_TEARDOWN) {
...@@ -1752,11 +1824,32 @@ static int intel_resume_runtime(struct device *dev) ...@@ -1752,11 +1824,32 @@ static int intel_resume_runtime(struct device *dev)
return ret; return ret;
} }
/*
* follow recommended programming flows to avoid
* timeouts when gsync is enabled
*/
if (multi_link)
intel_shim_sync_arm(sdw);
ret = sdw_cdns_init(&sdw->cdns);
if (ret < 0) {
dev_err(dev, "unable to initialize Cadence IP during resume\n");
return ret;
}
ret = sdw_cdns_exit_reset(cdns); ret = sdw_cdns_exit_reset(cdns);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "unable to exit bus reset sequence during resume\n"); dev_err(dev, "unable to exit bus reset sequence during resume\n");
return ret; return ret;
} }
if (multi_link) {
ret = intel_shim_sync_go(sdw);
if (ret < 0) {
dev_err(dev, "sync go failed during resume\n");
return ret;
}
}
} else if (clock_stop_quirks & SDW_INTEL_CLK_STOP_BUS_RESET) { } else if (clock_stop_quirks & SDW_INTEL_CLK_STOP_BUS_RESET) {
ret = intel_init(sdw); ret = intel_init(sdw);
if (ret) { if (ret) {
...@@ -1773,11 +1866,18 @@ static int intel_resume_runtime(struct device *dev) ...@@ -1773,11 +1866,18 @@ static int intel_resume_runtime(struct device *dev)
*/ */
clock_stop0 = sdw_cdns_is_clock_stop(&sdw->cdns); clock_stop0 = sdw_cdns_is_clock_stop(&sdw->cdns);
if (!clock_stop0) {
/*
* Re-initialize the IP since it was powered-off
*/
sdw_cdns_init(&sdw->cdns);
/* /*
* make sure all Slaves are tagged as UNATTACHED and * make sure all Slaves are tagged as UNATTACHED and
* provide reason for reinitialization * provide reason for reinitialization
*/ */
if (!clock_stop0) {
status = SDW_UNATTACH_REQUEST_MASTER_RESET; status = SDW_UNATTACH_REQUEST_MASTER_RESET;
sdw_clear_slave_status(bus, status); sdw_clear_slave_status(bus, status);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment