// SPDX-License-Identifier: (GPL-2.0-only OR BSD-3-Clause) // Copyright(c) 2015-2023 Intel Corporation #include #include #include #include #include "cadence_master.h" #include "bus.h" #include "intel.h" int intel_start_bus(struct sdw_intel *sdw) { struct device *dev = sdw->cdns.dev; struct sdw_cdns *cdns = &sdw->cdns; struct sdw_bus *bus = &cdns->bus; int ret; ret = sdw_cdns_soft_reset(cdns); if (ret < 0) { dev_err(dev, "%s: unable to soft-reset Cadence IP: %d\n", __func__, ret); return ret; } /* * follow recommended programming flows to avoid timeouts when * gsync is enabled */ if (bus->multi_link) sdw_intel_sync_arm(sdw); ret = sdw_cdns_init(cdns); if (ret < 0) { dev_err(dev, "%s: unable to initialize Cadence IP: %d\n", __func__, ret); return ret; } sdw_cdns_config_update(cdns); if (bus->multi_link) { ret = sdw_intel_sync_go(sdw); if (ret < 0) { dev_err(dev, "%s: sync go failed: %d\n", __func__, ret); return ret; } } ret = sdw_cdns_config_update_set_wait(cdns); if (ret < 0) { dev_err(dev, "%s: CONFIG_UPDATE BIT still set\n", __func__); return ret; } ret = sdw_cdns_enable_interrupt(cdns, true); if (ret < 0) { dev_err(dev, "%s: cannot enable interrupts: %d\n", __func__, ret); return ret; } ret = sdw_cdns_exit_reset(cdns); if (ret < 0) { dev_err(dev, "%s: unable to exit bus reset sequence: %d\n", __func__, ret); return ret; } sdw_cdns_check_self_clearing_bits(cdns, __func__, true, INTEL_MASTER_RESET_ITERATIONS); schedule_delayed_work(&cdns->attach_dwork, msecs_to_jiffies(SDW_INTEL_DELAYED_ENUMERATION_MS)); return 0; } int intel_start_bus_after_reset(struct sdw_intel *sdw) { struct device *dev = sdw->cdns.dev; struct sdw_cdns *cdns = &sdw->cdns; struct sdw_bus *bus = &cdns->bus; bool clock_stop0; int status; int ret; /* * An exception condition occurs for the CLK_STOP_BUS_RESET * case if one or more masters remain active. In this condition, * all the masters are powered on for they are in the same power * domain. Master can preserve its context for clock stop0, so * there is no need to clear slave status and reset bus. */ clock_stop0 = sdw_cdns_is_clock_stop(&sdw->cdns); if (!clock_stop0) { /* * make sure all Slaves are tagged as UNATTACHED and * provide reason for reinitialization */ status = SDW_UNATTACH_REQUEST_MASTER_RESET; sdw_clear_slave_status(bus, status); /* * follow recommended programming flows to avoid * timeouts when gsync is enabled */ if (bus->multi_link) sdw_intel_sync_arm(sdw); /* * Re-initialize the IP since it was powered-off */ sdw_cdns_init(&sdw->cdns); } else { ret = sdw_cdns_enable_interrupt(cdns, true); if (ret < 0) { dev_err(dev, "cannot enable interrupts during resume\n"); return ret; } } ret = sdw_cdns_clock_restart(cdns, !clock_stop0); if (ret < 0) { dev_err(dev, "unable to restart clock during resume\n"); if (!clock_stop0) sdw_cdns_enable_interrupt(cdns, false); return ret; } if (!clock_stop0) { sdw_cdns_config_update(cdns); if (bus->multi_link) { ret = sdw_intel_sync_go(sdw); if (ret < 0) { dev_err(sdw->cdns.dev, "sync go failed during resume\n"); return ret; } } ret = sdw_cdns_config_update_set_wait(cdns); if (ret < 0) { dev_err(dev, "%s: CONFIG_UPDATE BIT still set\n", __func__); return ret; } ret = sdw_cdns_enable_interrupt(cdns, true); if (ret < 0) { dev_err(dev, "cannot enable interrupts during resume\n"); return ret; } ret = sdw_cdns_exit_reset(cdns); if (ret < 0) { dev_err(dev, "unable to exit bus reset sequence during resume\n"); return ret; } } sdw_cdns_check_self_clearing_bits(cdns, __func__, true, INTEL_MASTER_RESET_ITERATIONS); schedule_delayed_work(&cdns->attach_dwork, msecs_to_jiffies(SDW_INTEL_DELAYED_ENUMERATION_MS)); return 0; } void intel_check_clock_stop(struct sdw_intel *sdw) { struct device *dev = sdw->cdns.dev; bool clock_stop0; clock_stop0 = sdw_cdns_is_clock_stop(&sdw->cdns); if (!clock_stop0) dev_err(dev, "%s: invalid configuration, clock was not stopped\n", __func__); } int intel_start_bus_after_clock_stop(struct sdw_intel *sdw) { struct device *dev = sdw->cdns.dev; struct sdw_cdns *cdns = &sdw->cdns; int ret; ret = sdw_cdns_clock_restart(cdns, false); if (ret < 0) { dev_err(dev, "%s: unable to restart clock: %d\n", __func__, ret); return ret; } ret = sdw_cdns_enable_interrupt(cdns, true); if (ret < 0) { dev_err(dev, "%s: cannot enable interrupts: %d\n", __func__, ret); return ret; } sdw_cdns_check_self_clearing_bits(cdns, __func__, true, INTEL_MASTER_RESET_ITERATIONS); schedule_delayed_work(&cdns->attach_dwork, msecs_to_jiffies(SDW_INTEL_DELAYED_ENUMERATION_MS)); return 0; } int intel_stop_bus(struct sdw_intel *sdw, bool clock_stop) { struct device *dev = sdw->cdns.dev; struct sdw_cdns *cdns = &sdw->cdns; bool wake_enable = false; int ret; cancel_delayed_work_sync(&cdns->attach_dwork); if (clock_stop) { ret = sdw_cdns_clock_stop(cdns, true); if (ret < 0) dev_err(dev, "%s: cannot stop clock: %d\n", __func__, ret); else wake_enable = true; } ret = sdw_cdns_enable_interrupt(cdns, false); if (ret < 0) { dev_err(dev, "%s: cannot disable interrupts: %d\n", __func__, ret); return ret; } ret = sdw_intel_link_power_down(sdw); if (ret) { dev_err(dev, "%s: Link power down failed: %d\n", __func__, ret); return ret; } sdw_intel_shim_wake(sdw, wake_enable); return 0; } /* * bank switch routines */ int intel_pre_bank_switch(struct sdw_intel *sdw) { struct sdw_cdns *cdns = &sdw->cdns; struct sdw_bus *bus = &cdns->bus; /* Write to register only for multi-link */ if (!bus->multi_link) return 0; sdw_intel_sync_arm(sdw); return 0; } int intel_post_bank_switch(struct sdw_intel *sdw) { struct sdw_cdns *cdns = &sdw->cdns; struct sdw_bus *bus = &cdns->bus; int ret = 0; /* Write to register only for multi-link */ if (!bus->multi_link) return 0; mutex_lock(sdw->link_res->shim_lock); /* * post_bank_switch() ops is called from the bus in loop for * all the Masters in the steam with the expectation that * we trigger the bankswitch for the only first Master in the list * and do nothing for the other Masters * * So, set the SYNCGO bit only if CMDSYNC bit is set for any Master. */ if (sdw_intel_sync_check_cmdsync_unlocked(sdw)) ret = sdw_intel_sync_go_unlocked(sdw); mutex_unlock(sdw->link_res->shim_lock); if (ret < 0) dev_err(sdw->cdns.dev, "Post bank switch failed: %d\n", ret); return ret; }