On 25/03/19 4:56 AM, Tony Lindgren wrote: > With the recent regulator changes I noticed new warnings on doing rmmod of > phy-twl4030-usb: > > WARNING: CPU: 0 PID: 1080 at drivers/regulator/core.c:2046 _regulator_put > ... > > Turns out we can currently miss disconnect at least for cases where status > is 0 and linkstat is 0. And in that case doing rmmod phy-twl4030-usb will > produce the regulator_put() warning. > > This is because the missed disconnect causes unbalanced PM runtime calls > and the regulators will be on exit. > > Let's fix the issue by using an atomic flag for the cable state to make > sure that PM runtime won't get out of sync with the cable state. That > way we can also simplify the code a bit. > > Note that we can also drop the old comments, those relate to issues that > the battery charger driver and musb driver is dealing with rather than > the USB PHY driver. > > Cc: NeilBrown <neilb@xxxxxxxx> > Signed-off-by: Tony Lindgren <tony@xxxxxxxxxxx> merged, thanks! -Kishon > --- > drivers/phy/ti/phy-twl4030-usb.c | 35 ++++++++++++-------------------- > 1 file changed, 13 insertions(+), 22 deletions(-) > > diff --git a/drivers/phy/ti/phy-twl4030-usb.c b/drivers/phy/ti/phy-twl4030-usb.c > --- a/drivers/phy/ti/phy-twl4030-usb.c > +++ b/drivers/phy/ti/phy-twl4030-usb.c > @@ -172,6 +172,7 @@ struct twl4030_usb { > > int irq; > enum musb_vbus_id_status linkstat; > + atomic_t connected; > bool vbus_supplied; > bool musb_mailbox_pending; > > @@ -575,39 +576,29 @@ static irqreturn_t twl4030_usb_irq(int irq, void *_twl) > { > struct twl4030_usb *twl = _twl; > enum musb_vbus_id_status status; > - bool status_changed = false; > int err; > > status = twl4030_usb_linkstat(twl); > > mutex_lock(&twl->lock); > - if (status >= 0 && status != twl->linkstat) { > - status_changed = > - cable_present(twl->linkstat) != > - cable_present(status); > - twl->linkstat = status; > - } > + twl->linkstat = status; > mutex_unlock(&twl->lock); > > - if (status_changed) { > - /* FIXME add a set_power() method so that B-devices can > - * configure the charger appropriately. It's not always > - * correct to consume VBUS power, and how much current to > - * consume is a function of the USB configuration chosen > - * by the host. > - * > - * REVISIT usb_gadget_vbus_connect(...) as needed, ditto > - * its disconnect() sibling, when changing to/from the > - * USB_LINK_VBUS state. musb_hdrc won't care until it > - * starts to handle softconnect right. > - */ > - if (cable_present(status)) { > + if (cable_present(status)) { > + if (atomic_add_unless(&twl->connected, 1, 1)) { > + dev_dbg(twl->dev, "%s: cable connected %i\n", > + __func__, status); > pm_runtime_get_sync(twl->dev); > - } else { > + twl->musb_mailbox_pending = true; > + } > + } else { > + if (atomic_add_unless(&twl->connected, -1, 0)) { > + dev_dbg(twl->dev, "%s: cable disconnected %i\n", > + __func__, status); > pm_runtime_mark_last_busy(twl->dev); > pm_runtime_put_autosuspend(twl->dev); > + twl->musb_mailbox_pending = true; > } > - twl->musb_mailbox_pending = true; > } > if (twl->musb_mailbox_pending) { > err = musb_mailbox(status); >