Unverified Commit 554f6006 authored by Mark Brown's avatar Mark Brown
Browse files

Improve cs42l43 suspend/IRQ interactions

Merge series from Charles Keepax <ckeepax@opensource.cirrus.com>:

cs42l43 uses pm_runtime_force_suspend() during system suspend, however
this means care must be taken that IRQ handler code isn't running when
entering system suspend as force suspend will ignore that the handler
is holding a pm reference. Typically the result of this is just a few
error messages, but better to improve the handling and ensure that all
IRQ processing is synchronised in before system suspend.
parents d3e39580 dd7ae5b8
Loading
Loading
Loading
Loading
+5 −3
Original line number Diff line number Diff line
@@ -684,7 +684,7 @@ static int cs42l43_run_type_detect(struct cs42l43_codec *priv)
	}
}

static void cs42l43_clear_jack(struct cs42l43_codec *priv)
void cs42l43_clear_jack(struct cs42l43_codec *priv)
{
	struct cs42l43 *cs42l43 = priv->core;

@@ -703,8 +703,6 @@ static void cs42l43_clear_jack(struct cs42l43_codec *priv)
	regmap_update_bits(cs42l43->regmap, CS42L43_HS2,
			   CS42L43_HSDET_MODE_MASK | CS42L43_HSDET_MANUAL_MODE_MASK,
			   0x2 << CS42L43_HSDET_MODE_SHIFT);

	snd_soc_jack_report(priv->jack_hp, 0, 0xFFFF);
}

void cs42l43_tip_sense_work(struct work_struct *work)
@@ -753,6 +751,8 @@ void cs42l43_tip_sense_work(struct work_struct *work)

		cs42l43_clear_jack(priv);

		snd_soc_jack_report(priv->jack_hp, 0, 0xFFFF);

		if (cs42l43->sdw && priv->jack_present) {
			pm_runtime_put(priv->dev);
			priv->jack_present = false;
@@ -903,6 +903,8 @@ int cs42l43_jack_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *u

	cs42l43_clear_jack(priv);

	snd_soc_jack_report(priv->jack_hp, 0, 0xFFFF);

	if (!override) {
		queue_delayed_work(system_long_wq, &priv->tip_sense_work, 0);
	} else {
+78 −31
Original line number Diff line number Diff line
@@ -2210,13 +2210,12 @@ static const struct cs42l43_irq cs42l43_irqs[] = {
};

static int cs42l43_request_irq(struct cs42l43_codec *priv,
			       struct irq_domain *dom, const char * const name,
			       unsigned int irq, irq_handler_t handler,
			       unsigned long flags)
			       const char * const name, unsigned int irq,
			       irq_handler_t handler, unsigned long flags)
{
	int ret;

	ret = irq_create_mapping(dom, irq);
	ret = irq_create_mapping(priv->dom, irq);
	if (ret < 0)
		return dev_err_probe(priv->dev, ret, "Failed to map IRQ %s\n", name);

@@ -2230,13 +2229,29 @@ static int cs42l43_request_irq(struct cs42l43_codec *priv,
	return 0;
}

static int cs42l43_shutter_irq(struct cs42l43_codec *priv,
			       struct irq_domain *dom, unsigned int shutter,
			       const char * const open_name,
			       const char * const close_name,
static void cs42l43_disable_irq(struct cs42l43_codec *priv, unsigned int irq)
{
	int ret;

	ret = irq_find_mapping(priv->dom, irq);
	if (ret > 0)
		disable_irq(ret);
}

static void cs42l43_enable_irq(struct cs42l43_codec *priv, unsigned int irq)
{
	int ret;

	ret = irq_find_mapping(priv->dom, irq);
	if (ret > 0)
		enable_irq(ret);
}

static int cs42l43_shutter_irq(struct cs42l43_codec *priv, unsigned int shutter,
			       const char * const open_name, unsigned int *open_irq,
			       const char * const close_name, unsigned int *close_irq,
			       irq_handler_t handler)
{
	unsigned int open_irq, close_irq;
	int ret;

	switch (shutter) {
@@ -2244,40 +2259,35 @@ static int cs42l43_shutter_irq(struct cs42l43_codec *priv,
		dev_warn(priv->dev, "Manual shutters, notifications not available\n");
		return 0;
	case 0x2:
		open_irq = CS42L43_GPIO1_RISE;
		close_irq = CS42L43_GPIO1_FALL;
		*open_irq = CS42L43_GPIO1_RISE;
		*close_irq = CS42L43_GPIO1_FALL;
		break;
	case 0x4:
		open_irq = CS42L43_GPIO2_RISE;
		close_irq = CS42L43_GPIO2_FALL;
		*open_irq = CS42L43_GPIO2_RISE;
		*close_irq = CS42L43_GPIO2_FALL;
		break;
	case 0x8:
		open_irq = CS42L43_GPIO3_RISE;
		close_irq = CS42L43_GPIO3_FALL;
		*open_irq = CS42L43_GPIO3_RISE;
		*close_irq = CS42L43_GPIO3_FALL;
		break;
	default:
		return 0;
	}

	ret = cs42l43_request_irq(priv, dom, close_name, close_irq, handler, IRQF_SHARED);
	ret = cs42l43_request_irq(priv, close_name, *close_irq, handler, IRQF_SHARED);
	if (ret)
		return ret;

	return cs42l43_request_irq(priv, dom, open_name, open_irq, handler, IRQF_SHARED);
	return cs42l43_request_irq(priv, open_name, *open_irq, handler, IRQF_SHARED);
}

static int cs42l43_codec_probe(struct platform_device *pdev)
{
	struct cs42l43 *cs42l43 = dev_get_drvdata(pdev->dev.parent);
	struct cs42l43_codec *priv;
	struct irq_domain *dom;
	unsigned int val;
	int i, ret;

	dom = irq_find_matching_fwnode(dev_fwnode(cs42l43->dev), DOMAIN_BUS_ANY);
	if (!dom)
		return -EPROBE_DEFER;

	priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL);
	if (!priv)
		return -ENOMEM;
@@ -2285,6 +2295,10 @@ static int cs42l43_codec_probe(struct platform_device *pdev)
	priv->dev = &pdev->dev;
	priv->core = cs42l43;

	priv->dom = irq_find_matching_fwnode(dev_fwnode(cs42l43->dev), DOMAIN_BUS_ANY);
	if (!priv->dom)
		return -EPROBE_DEFER;

	platform_set_drvdata(pdev, priv);

	mutex_init(&priv->jack_lock);
@@ -2314,7 +2328,7 @@ static int cs42l43_codec_probe(struct platform_device *pdev)
		goto err_pm;

	for (i = 0; i < ARRAY_SIZE(cs42l43_irqs); i++) {
		ret = cs42l43_request_irq(priv, dom, cs42l43_irqs[i].name,
		ret = cs42l43_request_irq(priv, cs42l43_irqs[i].name,
					  cs42l43_irqs[i].irq,
					  cs42l43_irqs[i].handler, 0);
		if (ret)
@@ -2327,15 +2341,17 @@ static int cs42l43_codec_probe(struct platform_device *pdev)
		goto err_pm;
	}

	ret = cs42l43_shutter_irq(priv, dom, val & CS42L43_MIC_SHUTTER_CFG_MASK,
				  "mic shutter open", "mic shutter close",
	ret = cs42l43_shutter_irq(priv, val & CS42L43_MIC_SHUTTER_CFG_MASK,
				  "mic shutter open", &priv->shutter_irqs[0],
				  "mic shutter close", &priv->shutter_irqs[1],
				  cs42l43_mic_shutter);
	if (ret)
		goto err_pm;

	ret = cs42l43_shutter_irq(priv, dom, (val & CS42L43_SPK_SHUTTER_CFG_MASK) >>
	ret = cs42l43_shutter_irq(priv, (val & CS42L43_SPK_SHUTTER_CFG_MASK) >>
				  CS42L43_SPK_SHUTTER_CFG_SHIFT,
				  "spk shutter open", "spk shutter close",
				  "spk shutter open", &priv->shutter_irqs[2],
				  "spk shutter close", &priv->shutter_irqs[3],
				  cs42l43_spk_shutter);
	if (ret)
		goto err_pm;
@@ -2386,22 +2402,53 @@ static int cs42l43_codec_runtime_resume(struct device *dev)
	return 0;
}

static int cs42l43_codec_runtime_force_suspend(struct device *dev)
static int cs42l43_codec_suspend(struct device *dev)
{
	struct cs42l43_codec *priv = dev_get_drvdata(dev);
	int i;

	dev_dbg(priv->dev, "Runtime suspend\n");
	dev_dbg(priv->dev, "System suspend\n");

	priv->suspend_jack_debounce = true;

	pm_runtime_force_suspend(dev);
	for (i = 0; i < ARRAY_SIZE(cs42l43_irqs); i++)
		cs42l43_disable_irq(priv, cs42l43_irqs[i].irq);

	for (i = 0; i < ARRAY_SIZE(priv->shutter_irqs); i++)
		if (priv->shutter_irqs[i])
			cs42l43_disable_irq(priv, priv->shutter_irqs[i]);

	cancel_delayed_work_sync(&priv->bias_sense_timeout);
	cancel_delayed_work_sync(&priv->tip_sense_work);
	cancel_delayed_work_sync(&priv->hp_ilimit_clear_work);

	cs42l43_clear_jack(priv);

	return pm_runtime_force_suspend(dev);
}

static int cs42l43_codec_resume(struct device *dev)
{
	struct cs42l43_codec *priv = dev_get_drvdata(dev);
	int ret, i;

	ret = pm_runtime_force_resume(dev);
	if (ret)
		return ret;

	for (i = 0; i < ARRAY_SIZE(cs42l43_irqs); i++)
		cs42l43_enable_irq(priv, cs42l43_irqs[i].irq);

	for (i = 0; i < ARRAY_SIZE(priv->shutter_irqs); i++)
		if (priv->shutter_irqs[i])
			cs42l43_enable_irq(priv, priv->shutter_irqs[i]);

	return 0;
}

static const struct dev_pm_ops cs42l43_codec_pm_ops = {
	RUNTIME_PM_OPS(NULL, cs42l43_codec_runtime_resume, NULL)
	SYSTEM_SLEEP_PM_OPS(cs42l43_codec_runtime_force_suspend, pm_runtime_force_resume)
	SYSTEM_SLEEP_PM_OPS(cs42l43_codec_suspend, cs42l43_codec_resume)
};

static const struct platform_device_id cs42l43_codec_id_table[] = {
+3 −0
Original line number Diff line number Diff line
@@ -44,6 +44,8 @@ struct cs42l43_codec {
	struct device *dev;
	struct cs42l43 *core;
	struct snd_soc_component *component;
	struct irq_domain *dom;
	unsigned int shutter_irqs[4];

	struct clk *mclk;

@@ -130,6 +132,7 @@ static inline int cs42l43_sdw_add_peripheral(struct snd_pcm_substream *substream
int cs42l43_set_jack(struct snd_soc_component *component,
		     struct snd_soc_jack *jack, void *d);
void cs42l43_bias_sense_timeout(struct work_struct *work);
void cs42l43_clear_jack(struct cs42l43_codec *priv);
void cs42l43_tip_sense_work(struct work_struct *work);
irqreturn_t cs42l43_bias_detect_clamp(int irq, void *data);
irqreturn_t cs42l43_button_press(int irq, void *data);