Commit 4f7b018b authored by Ben Zhang's avatar Ben Zhang Committed by Mark Brown

ASoC: rt5677: clear interrupts by polarity flip

The rt5677 jack detection function has a requirement that the polarity
of an interrupt be flipped after it fires in order to clear the
interrupt.

This patch implements an irq_chip with irq_domain directly instead of
using regmap-irq, so that interrupt source line polarities can be
flipped in the irq handler.

The reason that this patch does not add this feature within regmap-irq
is that future patches will add hotword detection support to this irq
handler. Those patches will require adding additional logic that would
not make sense to have in regmap-irq.
Signed-off-by: default avatarBen Zhang <benzh@chromium.org>
Signed-off-by: default avatarFletcher Woodruff <fletcherw@chromium.org>
Signed-off-by: default avatarMark Brown <broonie@kernel.org>
parent 16395cee
...@@ -23,6 +23,10 @@ ...@@ -23,6 +23,10 @@
#include <linux/firmware.h> #include <linux/firmware.h>
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/property.h> #include <linux/property.h>
#include <linux/irq.h>
#include <linux/interrupt.h>
#include <linux/irqdomain.h>
#include <linux/workqueue.h>
#include <sound/core.h> #include <sound/core.h>
#include <sound/pcm.h> #include <sound/pcm.h>
#include <sound/pcm_params.h> #include <sound/pcm_params.h>
...@@ -4620,7 +4624,6 @@ static void rt5677_gpio_config(struct rt5677_priv *rt5677, unsigned offset, ...@@ -4620,7 +4624,6 @@ static void rt5677_gpio_config(struct rt5677_priv *rt5677, unsigned offset,
static int rt5677_to_irq(struct gpio_chip *chip, unsigned offset) static int rt5677_to_irq(struct gpio_chip *chip, unsigned offset)
{ {
struct rt5677_priv *rt5677 = gpiochip_get_data(chip); struct rt5677_priv *rt5677 = gpiochip_get_data(chip);
struct regmap_irq_chip_data *data = rt5677->irq_data;
int irq; int irq;
if ((rt5677->pdata.jd1_gpio == 1 && offset == RT5677_GPIO1) || if ((rt5677->pdata.jd1_gpio == 1 && offset == RT5677_GPIO1) ||
...@@ -4646,7 +4649,7 @@ static int rt5677_to_irq(struct gpio_chip *chip, unsigned offset) ...@@ -4646,7 +4649,7 @@ static int rt5677_to_irq(struct gpio_chip *chip, unsigned offset)
return -ENXIO; return -ENXIO;
} }
return regmap_irq_get_virq(data, irq); return irq_create_mapping(rt5677->domain, irq);
} }
static const struct gpio_chip rt5677_template_chip = { static const struct gpio_chip rt5677_template_chip = {
...@@ -5042,30 +5045,130 @@ static void rt5677_read_device_properties(struct rt5677_priv *rt5677, ...@@ -5042,30 +5045,130 @@ static void rt5677_read_device_properties(struct rt5677_priv *rt5677,
rt5677->pdata.jd3_gpio = val; rt5677->pdata.jd3_gpio = val;
} }
static struct regmap_irq rt5677_irqs[] = { struct rt5677_irq_desc {
unsigned int enable_mask;
unsigned int status_mask;
unsigned int polarity_mask;
};
static const struct rt5677_irq_desc rt5677_irq_descs[] = {
[RT5677_IRQ_JD1] = { [RT5677_IRQ_JD1] = {
.reg_offset = 0, .enable_mask = RT5677_EN_IRQ_GPIO_JD1,
.mask = RT5677_EN_IRQ_GPIO_JD1, .status_mask = RT5677_STA_GPIO_JD1,
.polarity_mask = RT5677_INV_GPIO_JD1,
}, },
[RT5677_IRQ_JD2] = { [RT5677_IRQ_JD2] = {
.reg_offset = 0, .enable_mask = RT5677_EN_IRQ_GPIO_JD2,
.mask = RT5677_EN_IRQ_GPIO_JD2, .status_mask = RT5677_STA_GPIO_JD2,
.polarity_mask = RT5677_INV_GPIO_JD2,
}, },
[RT5677_IRQ_JD3] = { [RT5677_IRQ_JD3] = {
.reg_offset = 0, .enable_mask = RT5677_EN_IRQ_GPIO_JD3,
.mask = RT5677_EN_IRQ_GPIO_JD3, .status_mask = RT5677_STA_GPIO_JD3,
.polarity_mask = RT5677_INV_GPIO_JD3,
}, },
}; };
static struct regmap_irq_chip rt5677_irq_chip = { static irqreturn_t rt5677_irq(int unused, void *data)
.name = RT5677_DRV_NAME, {
.irqs = rt5677_irqs, struct rt5677_priv *rt5677 = data;
.num_irqs = ARRAY_SIZE(rt5677_irqs), int ret = 0, i, reg_irq, virq;
bool irq_fired = false;
mutex_lock(&rt5677->irq_lock);
/* Read interrupt status */
ret = regmap_read(rt5677->regmap, RT5677_IRQ_CTRL1, &reg_irq);
if (ret) {
dev_err(rt5677->dev, "failed reading IRQ status: %d\n", ret);
goto exit;
}
for (i = 0; i < RT5677_IRQ_NUM; i++) {
if (reg_irq & rt5677_irq_descs[i].status_mask) {
irq_fired = true;
virq = irq_find_mapping(rt5677->domain, i);
if (virq)
handle_nested_irq(virq);
/* Clear the interrupt by flipping the polarity of the
* interrupt source line that fired
*/
reg_irq ^= rt5677_irq_descs[i].polarity_mask;
}
}
if (!irq_fired)
goto exit;
ret = regmap_write(rt5677->regmap, RT5677_IRQ_CTRL1, reg_irq);
if (ret) {
dev_err(rt5677->dev, "failed updating IRQ status: %d\n", ret);
goto exit;
}
exit:
mutex_unlock(&rt5677->irq_lock);
if (irq_fired)
return IRQ_HANDLED;
else
return IRQ_NONE;
}
static void rt5677_irq_bus_lock(struct irq_data *data)
{
struct rt5677_priv *rt5677 = irq_data_get_irq_chip_data(data);
mutex_lock(&rt5677->irq_lock);
}
static void rt5677_irq_bus_sync_unlock(struct irq_data *data)
{
struct rt5677_priv *rt5677 = irq_data_get_irq_chip_data(data);
// Set the enable/disable bits for the jack detect IRQs.
regmap_update_bits(rt5677->regmap, RT5677_IRQ_CTRL1,
RT5677_EN_IRQ_GPIO_JD1 | RT5677_EN_IRQ_GPIO_JD2 |
RT5677_EN_IRQ_GPIO_JD3, rt5677->irq_en);
mutex_unlock(&rt5677->irq_lock);
}
static void rt5677_irq_enable(struct irq_data *data)
{
struct rt5677_priv *rt5677 = irq_data_get_irq_chip_data(data);
rt5677->irq_en |= rt5677_irq_descs[data->hwirq].enable_mask;
}
.num_regs = 1, static void rt5677_irq_disable(struct irq_data *data)
.status_base = RT5677_IRQ_CTRL1, {
.mask_base = RT5677_IRQ_CTRL1, struct rt5677_priv *rt5677 = irq_data_get_irq_chip_data(data);
.mask_invert = 1,
rt5677->irq_en &= ~rt5677_irq_descs[data->hwirq].enable_mask;
}
static struct irq_chip rt5677_irq_chip = {
.name = "rt5677_irq_chip",
.irq_bus_lock = rt5677_irq_bus_lock,
.irq_bus_sync_unlock = rt5677_irq_bus_sync_unlock,
.irq_disable = rt5677_irq_disable,
.irq_enable = rt5677_irq_enable,
};
static int rt5677_irq_map(struct irq_domain *h, unsigned int virq,
irq_hw_number_t hw)
{
struct rt5677_priv *rt5677 = h->host_data;
irq_set_chip_data(virq, rt5677);
irq_set_chip(virq, &rt5677_irq_chip);
irq_set_nested_thread(virq, 1);
irq_set_noprobe(virq);
return 0;
}
static const struct irq_domain_ops rt5677_domain_ops = {
.map = rt5677_irq_map,
.xlate = irq_domain_xlate_twocell,
}; };
static int rt5677_init_irq(struct i2c_client *i2c) static int rt5677_init_irq(struct i2c_client *i2c)
...@@ -5084,6 +5187,8 @@ static int rt5677_init_irq(struct i2c_client *i2c) ...@@ -5084,6 +5187,8 @@ static int rt5677_init_irq(struct i2c_client *i2c)
return -EINVAL; return -EINVAL;
} }
mutex_init(&rt5677->irq_lock);
/* /*
* Select RC as the debounce clock so that GPIO works even when * Select RC as the debounce clock so that GPIO works even when
* MCLK is gated which happens when there is no audio stream * MCLK is gated which happens when there is no audio stream
...@@ -5092,7 +5197,6 @@ static int rt5677_init_irq(struct i2c_client *i2c) ...@@ -5092,7 +5197,6 @@ static int rt5677_init_irq(struct i2c_client *i2c)
regmap_update_bits(rt5677->regmap, RT5677_DIG_MISC, regmap_update_bits(rt5677->regmap, RT5677_DIG_MISC,
RT5677_IRQ_DEBOUNCE_SEL_MASK, RT5677_IRQ_DEBOUNCE_SEL_MASK,
RT5677_IRQ_DEBOUNCE_SEL_RC); RT5677_IRQ_DEBOUNCE_SEL_RC);
/* Enable auto power on RC when GPIO states are changed */ /* Enable auto power on RC when GPIO states are changed */
regmap_update_bits(rt5677->regmap, RT5677_GEN_CTRL1, 0xff, 0xff); regmap_update_bits(rt5677->regmap, RT5677_GEN_CTRL1, 0xff, 0xff);
...@@ -5115,24 +5219,21 @@ static int rt5677_init_irq(struct i2c_client *i2c) ...@@ -5115,24 +5219,21 @@ static int rt5677_init_irq(struct i2c_client *i2c)
regmap_update_bits(rt5677->regmap, RT5677_GPIO_CTRL1, regmap_update_bits(rt5677->regmap, RT5677_GPIO_CTRL1,
RT5677_GPIO1_PIN_MASK, RT5677_GPIO1_PIN_IRQ); RT5677_GPIO1_PIN_MASK, RT5677_GPIO1_PIN_IRQ);
ret = regmap_add_irq_chip(rt5677->regmap, i2c->irq, /* Ready to listen for interrupts */
IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING | IRQF_ONESHOT, 0, rt5677->domain = irq_domain_add_linear(i2c->dev.of_node,
&rt5677_irq_chip, &rt5677->irq_data); RT5677_IRQ_NUM, &rt5677_domain_ops, rt5677);
if (!rt5677->domain) {
if (ret != 0) { dev_err(&i2c->dev, "Failed to create IRQ domain\n");
dev_err(&i2c->dev, "Failed to register IRQ chip: %d\n", ret); return -ENOMEM;
return ret;
} }
return 0; ret = devm_request_threaded_irq(&i2c->dev, i2c->irq, NULL, rt5677_irq,
} IRQF_TRIGGER_RISING | IRQF_ONESHOT,
"rt5677", rt5677);
if (ret)
dev_err(&i2c->dev, "Failed to request IRQ: %d\n", ret);
static void rt5677_free_irq(struct i2c_client *i2c) return ret;
{
struct rt5677_priv *rt5677 = i2c_get_clientdata(i2c);
if (rt5677->irq_data)
regmap_del_irq_chip(i2c->irq, rt5677->irq_data);
} }
static int rt5677_i2c_probe(struct i2c_client *i2c) static int rt5677_i2c_probe(struct i2c_client *i2c)
...@@ -5146,6 +5247,7 @@ static int rt5677_i2c_probe(struct i2c_client *i2c) ...@@ -5146,6 +5247,7 @@ static int rt5677_i2c_probe(struct i2c_client *i2c)
if (rt5677 == NULL) if (rt5677 == NULL)
return -ENOMEM; return -ENOMEM;
rt5677->dev = &i2c->dev;
i2c_set_clientdata(i2c, rt5677); i2c_set_clientdata(i2c, rt5677);
if (i2c->dev.of_node) { if (i2c->dev.of_node) {
...@@ -5259,7 +5361,9 @@ static int rt5677_i2c_probe(struct i2c_client *i2c) ...@@ -5259,7 +5361,9 @@ static int rt5677_i2c_probe(struct i2c_client *i2c)
RT5677_MICBIAS1_CTRL_VDD_3_3V); RT5677_MICBIAS1_CTRL_VDD_3_3V);
rt5677_init_gpio(i2c); rt5677_init_gpio(i2c);
rt5677_init_irq(i2c); ret = rt5677_init_irq(i2c);
if (ret)
dev_err(&i2c->dev, "Failed to initialize irq: %d\n", ret);
return devm_snd_soc_register_component(&i2c->dev, return devm_snd_soc_register_component(&i2c->dev,
&soc_component_dev_rt5677, &soc_component_dev_rt5677,
...@@ -5268,7 +5372,6 @@ static int rt5677_i2c_probe(struct i2c_client *i2c) ...@@ -5268,7 +5372,6 @@ static int rt5677_i2c_probe(struct i2c_client *i2c)
static int rt5677_i2c_remove(struct i2c_client *i2c) static int rt5677_i2c_remove(struct i2c_client *i2c)
{ {
rt5677_free_irq(i2c);
rt5677_free_gpio(i2c); rt5677_free_gpio(i2c);
return 0; return 0;
......
...@@ -1749,6 +1749,7 @@ enum { ...@@ -1749,6 +1749,7 @@ enum {
RT5677_IRQ_JD1, RT5677_IRQ_JD1,
RT5677_IRQ_JD2, RT5677_IRQ_JD2,
RT5677_IRQ_JD3, RT5677_IRQ_JD3,
RT5677_IRQ_NUM,
}; };
enum rt5677_type { enum rt5677_type {
...@@ -1827,6 +1828,7 @@ struct rt5677_platform_data { ...@@ -1827,6 +1828,7 @@ struct rt5677_platform_data {
struct rt5677_priv { struct rt5677_priv {
struct snd_soc_component *component; struct snd_soc_component *component;
struct device *dev;
struct rt5677_platform_data pdata; struct rt5677_platform_data pdata;
struct regmap *regmap, *regmap_physical; struct regmap *regmap, *regmap_physical;
const struct firmware *fw1, *fw2; const struct firmware *fw1, *fw2;
...@@ -1847,9 +1849,13 @@ struct rt5677_priv { ...@@ -1847,9 +1849,13 @@ struct rt5677_priv {
struct gpio_chip gpio_chip; struct gpio_chip gpio_chip;
#endif #endif
bool dsp_vad_en; bool dsp_vad_en;
struct regmap_irq_chip_data *irq_data;
bool is_dsp_mode; bool is_dsp_mode;
bool is_vref_slow; bool is_vref_slow;
/* Interrupt handling */
struct irq_domain *domain;
struct mutex irq_lock;
unsigned int irq_en;
}; };
int rt5677_sel_asrc_clk_src(struct snd_soc_component *component, int rt5677_sel_asrc_clk_src(struct snd_soc_component *component,
......
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