Hi, On 6/9/21 9:49 PM, Jonathan Cameron wrote: > On Wed, 26 May 2021 17:55:41 +0100 > Jonathan Cameron <jic23@xxxxxxxxxx> wrote: > >> On Sun, 23 May 2021 19:00:56 +0200 >> Hans de Goede <hdegoede@xxxxxxxxxx> wrote: >> >>> On machines with dual accelerometers described in a single ACPI fwnode, >>> the bmc150_accel_probe() instantiates a second i2c-client for the second >>> accelerometer. >>> >>> A pointer to this manually instantiated second i2c-client is stored >>> inside the iio_dev's private-data through bmc150_set_second_device(), >>> so that the i2c-client can be unregistered from bmc150_accel_remove(). >>> >>> Before this commit bmc150_set_second_device() took only 1 argument so it >>> would store the pointer in private-data of the iio_dev belonging to the >>> manually instantiated i2c-client, leading to the bmc150_accel_remove() >>> call for the second_dev trying to unregister *itself* while it was >>> being removed, leading to a deadlock and rmmod hanging. >>> >>> Change bmc150_set_second_device() to take 2 arguments: 1. The i2c-client >>> which is instantiating the second i2c-client for the 2nd accelerometer and >>> 2. The second-device pointer itself (which also is an i2c-client). >>> >>> This will store the second_device pointer in the private data of the >>> iio_dev belonging to the (ACPI instantiated) i2c-client for the first >>> accelerometer and will make bmc150_accel_remove() unregister the >>> second_device i2c-client when called for the first client, >>> avoiding the deadlock. >>> >>> Fixes: 5bfb3a4bd8f6 ("iio: accel: bmc150: Check for a second ACPI device for BOSC0200") >>> Cc: Jeremy Cline <jeremy@xxxxxxxxxx> >>> Signed-off-by: Hans de Goede <hdegoede@xxxxxxxxxx> >> Patches 1 and 2 applied to the fixes-togreg branch of iio.git and marked for stable. >> The rest will have to wait for now. > Cycle has gone past rather quicker than expected, so I've changed > my mind on these two and pulled them across to the togreg branch > targeting 5.14. > That will mean the hit stable a little later (after 5.14-rc1) Ok, note the problems fixed in the first 2 patches are only hit on a rmmod (or an unbind), so there is no big hurry in getting them in to the stable series. > but pretty much ensures the rest of the series makes it into 5.14 Getting the rest of the series into 5.14 without issues is much appreciated, thank you. Regards, Hans >>> --- >>> drivers/iio/accel/bmc150-accel-core.c | 4 ++-- >>> drivers/iio/accel/bmc150-accel-i2c.c | 2 +- >>> drivers/iio/accel/bmc150-accel.h | 2 +- >>> 3 files changed, 4 insertions(+), 4 deletions(-) >>> >>> diff --git a/drivers/iio/accel/bmc150-accel-core.c b/drivers/iio/accel/bmc150-accel-core.c >>> index 3a3f67930165..8ff358c37a81 100644 >>> --- a/drivers/iio/accel/bmc150-accel-core.c >>> +++ b/drivers/iio/accel/bmc150-accel-core.c >>> @@ -1815,11 +1815,11 @@ struct i2c_client *bmc150_get_second_device(struct i2c_client *client) >>> } >>> EXPORT_SYMBOL_GPL(bmc150_get_second_device); >>> >>> -void bmc150_set_second_device(struct i2c_client *client) >>> +void bmc150_set_second_device(struct i2c_client *client, struct i2c_client *second_dev) >>> { >>> struct bmc150_accel_data *data = iio_priv(i2c_get_clientdata(client)); >>> >>> - data->second_device = client; >>> + data->second_device = second_dev; >>> } >>> EXPORT_SYMBOL_GPL(bmc150_set_second_device); >>> >>> diff --git a/drivers/iio/accel/bmc150-accel-i2c.c b/drivers/iio/accel/bmc150-accel-i2c.c >>> index 69f709319484..2afaae0294ee 100644 >>> --- a/drivers/iio/accel/bmc150-accel-i2c.c >>> +++ b/drivers/iio/accel/bmc150-accel-i2c.c >>> @@ -70,7 +70,7 @@ static int bmc150_accel_probe(struct i2c_client *client, >>> >>> second_dev = i2c_acpi_new_device(&client->dev, 1, &board_info); >>> if (!IS_ERR(second_dev)) >>> - bmc150_set_second_device(second_dev); >>> + bmc150_set_second_device(client, second_dev); >>> } >>> #endif >>> >>> diff --git a/drivers/iio/accel/bmc150-accel.h b/drivers/iio/accel/bmc150-accel.h >>> index 6024f15b9700..e30c1698f6fb 100644 >>> --- a/drivers/iio/accel/bmc150-accel.h >>> +++ b/drivers/iio/accel/bmc150-accel.h >>> @@ -18,7 +18,7 @@ int bmc150_accel_core_probe(struct device *dev, struct regmap *regmap, int irq, >>> const char *name, bool block_supported); >>> int bmc150_accel_core_remove(struct device *dev); >>> struct i2c_client *bmc150_get_second_device(struct i2c_client *second_device); >>> -void bmc150_set_second_device(struct i2c_client *second_device); >>> +void bmc150_set_second_device(struct i2c_client *client, struct i2c_client *second_dev); >>> extern const struct dev_pm_ops bmc150_accel_pm_ops; >>> extern const struct regmap_config bmc150_regmap_conf; >>> >> >