Commit 2003c44e authored by Richard Fitzgerald's avatar Richard Fitzgerald Committed by Mark Brown

ASoC: cs42l42: Prevent NULL pointer deref in interrupt handler

The interrupt handling code was getting the struct device* from a
struct snd_soc_component* stored in struct cs42l42_private. If the
interrupt was asserted before ASoC calls component_probe() the
snd_soc_component* will be NULL.

The stored snd_soc_component* is not actually used for anything other
than indirectly getting the struct device*. Remove it, and store the
struct device* in struct cs42l42_private.
Signed-off-by: default avatarRichard Fitzgerald <rf@opensource.cirrus.com>
Link: https://lore.kernel.org/r/20211025112258.9282-1-rf@opensource.cirrus.comSigned-off-by: default avatarMark Brown <broonie@kernel.org>
parent 044c1140
...@@ -526,17 +526,7 @@ static int cs42l42_set_jack(struct snd_soc_component *component, struct snd_soc_ ...@@ -526,17 +526,7 @@ static int cs42l42_set_jack(struct snd_soc_component *component, struct snd_soc_
return 0; return 0;
} }
static int cs42l42_component_probe(struct snd_soc_component *component)
{
struct cs42l42_private *cs42l42 = snd_soc_component_get_drvdata(component);
cs42l42->component = component;
return 0;
}
static const struct snd_soc_component_driver soc_component_dev_cs42l42 = { static const struct snd_soc_component_driver soc_component_dev_cs42l42 = {
.probe = cs42l42_component_probe,
.set_jack = cs42l42_set_jack, .set_jack = cs42l42_set_jack,
.dapm_widgets = cs42l42_dapm_widgets, .dapm_widgets = cs42l42_dapm_widgets,
.num_dapm_widgets = ARRAY_SIZE(cs42l42_dapm_widgets), .num_dapm_widgets = ARRAY_SIZE(cs42l42_dapm_widgets),
...@@ -1207,7 +1197,7 @@ static void cs42l42_process_hs_type_detect(struct cs42l42_private *cs42l42) ...@@ -1207,7 +1197,7 @@ static void cs42l42_process_hs_type_detect(struct cs42l42_private *cs42l42)
*/ */
if (cs42l42->hs_type == CS42L42_PLUG_INVALID || if (cs42l42->hs_type == CS42L42_PLUG_INVALID ||
cs42l42->hs_type == CS42L42_PLUG_HEADPHONE) { cs42l42->hs_type == CS42L42_PLUG_HEADPHONE) {
dev_dbg(cs42l42->component->dev, "Running Manual Detection Fallback\n"); dev_dbg(cs42l42->dev, "Running Manual Detection Fallback\n");
cs42l42_manual_hs_type_detect(cs42l42); cs42l42_manual_hs_type_detect(cs42l42);
} }
...@@ -1506,19 +1496,19 @@ static int cs42l42_handle_button_press(struct cs42l42_private *cs42l42) ...@@ -1506,19 +1496,19 @@ static int cs42l42_handle_button_press(struct cs42l42_private *cs42l42)
switch (bias_level) { switch (bias_level) {
case 1: /* Function C button press */ case 1: /* Function C button press */
bias_level = SND_JACK_BTN_2; bias_level = SND_JACK_BTN_2;
dev_dbg(cs42l42->component->dev, "Function C button press\n"); dev_dbg(cs42l42->dev, "Function C button press\n");
break; break;
case 2: /* Function B button press */ case 2: /* Function B button press */
bias_level = SND_JACK_BTN_1; bias_level = SND_JACK_BTN_1;
dev_dbg(cs42l42->component->dev, "Function B button press\n"); dev_dbg(cs42l42->dev, "Function B button press\n");
break; break;
case 3: /* Function D button press */ case 3: /* Function D button press */
bias_level = SND_JACK_BTN_3; bias_level = SND_JACK_BTN_3;
dev_dbg(cs42l42->component->dev, "Function D button press\n"); dev_dbg(cs42l42->dev, "Function D button press\n");
break; break;
case 4: /* Function A button press */ case 4: /* Function A button press */
bias_level = SND_JACK_BTN_0; bias_level = SND_JACK_BTN_0;
dev_dbg(cs42l42->component->dev, "Function A button press\n"); dev_dbg(cs42l42->dev, "Function A button press\n");
break; break;
default: default:
bias_level = 0; bias_level = 0;
...@@ -1592,7 +1582,6 @@ static const struct cs42l42_irq_params irq_params_table[] = { ...@@ -1592,7 +1582,6 @@ static const struct cs42l42_irq_params irq_params_table[] = {
static irqreturn_t cs42l42_irq_thread(int irq, void *data) static irqreturn_t cs42l42_irq_thread(int irq, void *data)
{ {
struct cs42l42_private *cs42l42 = (struct cs42l42_private *)data; struct cs42l42_private *cs42l42 = (struct cs42l42_private *)data;
struct snd_soc_component *component = cs42l42->component;
unsigned int stickies[12]; unsigned int stickies[12];
unsigned int masks[12]; unsigned int masks[12];
unsigned int current_plug_status; unsigned int current_plug_status;
...@@ -1639,7 +1628,7 @@ static irqreturn_t cs42l42_irq_thread(int irq, void *data) ...@@ -1639,7 +1628,7 @@ static irqreturn_t cs42l42_irq_thread(int irq, void *data)
default: default:
break; break;
} }
dev_dbg(component->dev, "Auto detect done (%d)\n", cs42l42->hs_type); dev_dbg(cs42l42->dev, "Auto detect done (%d)\n", cs42l42->hs_type);
} }
} }
...@@ -1673,7 +1662,7 @@ static irqreturn_t cs42l42_irq_thread(int irq, void *data) ...@@ -1673,7 +1662,7 @@ static irqreturn_t cs42l42_irq_thread(int irq, void *data)
SND_JACK_BTN_0 | SND_JACK_BTN_1 | SND_JACK_BTN_0 | SND_JACK_BTN_1 |
SND_JACK_BTN_2 | SND_JACK_BTN_3); SND_JACK_BTN_2 | SND_JACK_BTN_3);
dev_dbg(component->dev, "Unplug event\n"); dev_dbg(cs42l42->dev, "Unplug event\n");
} }
break; break;
...@@ -1689,7 +1678,7 @@ static irqreturn_t cs42l42_irq_thread(int irq, void *data) ...@@ -1689,7 +1678,7 @@ static irqreturn_t cs42l42_irq_thread(int irq, void *data)
CS42L42_M_HSBIAS_HIZ_MASK)) { CS42L42_M_HSBIAS_HIZ_MASK)) {
if (current_button_status & CS42L42_M_DETECT_TF_MASK) { if (current_button_status & CS42L42_M_DETECT_TF_MASK) {
dev_dbg(component->dev, "Button released\n"); dev_dbg(cs42l42->dev, "Button released\n");
report = 0; report = 0;
} else if (current_button_status & CS42L42_M_DETECT_FT_MASK) { } else if (current_button_status & CS42L42_M_DETECT_FT_MASK) {
report = cs42l42_handle_button_press(cs42l42); report = cs42l42_handle_button_press(cs42l42);
...@@ -2043,6 +2032,7 @@ static int cs42l42_i2c_probe(struct i2c_client *i2c_client, ...@@ -2043,6 +2032,7 @@ static int cs42l42_i2c_probe(struct i2c_client *i2c_client,
if (!cs42l42) if (!cs42l42)
return -ENOMEM; return -ENOMEM;
cs42l42->dev = &i2c_client->dev;
i2c_set_clientdata(i2c_client, cs42l42); i2c_set_clientdata(i2c_client, cs42l42);
cs42l42->regmap = devm_regmap_init_i2c(i2c_client, &cs42l42_regmap); cs42l42->regmap = devm_regmap_init_i2c(i2c_client, &cs42l42_regmap);
......
...@@ -833,7 +833,7 @@ static const char *const cs42l42_supply_names[CS42L42_NUM_SUPPLIES] = { ...@@ -833,7 +833,7 @@ static const char *const cs42l42_supply_names[CS42L42_NUM_SUPPLIES] = {
struct cs42l42_private { struct cs42l42_private {
struct regmap *regmap; struct regmap *regmap;
struct snd_soc_component *component; struct device *dev;
struct regulator_bulk_data supplies[CS42L42_NUM_SUPPLIES]; struct regulator_bulk_data supplies[CS42L42_NUM_SUPPLIES];
struct gpio_desc *reset_gpio; struct gpio_desc *reset_gpio;
struct completion pdn_done; struct completion pdn_done;
......
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