usb: dwc2: Allow entering hibernation from USB_SUSPEND interrupt
authorVardan Mikayelyan <mvardan@synopsys.com>
Fri, 16 Feb 2018 10:11:07 +0000 (14:11 +0400)
committerFelipe Balbi <felipe.balbi@linux.intel.com>
Tue, 13 Mar 2018 08:47:56 +0000 (10:47 +0200)
Do changes to allow entering hibernated state from USB_SUSPEND
interrupt. All code is added under if conditions and mustn't impact
existing functionality.

Signed-off-by: Vardan Mikayelyan <mvardan@synopsys.com>
Signed-off-by: John Youn <johnyoun@synopsys.com>
Signed-off-by: Grigor Tovmasyan <tovmasya@synopsys.com>
Signed-off-by: Felipe Balbi <felipe.balbi@linux.intel.com>
drivers/usb/dwc2/core_intr.c

index 41d7dda40cb15aa463ea2ccbfebe1e4f30d673e1..d01581594ce5a88458b9d6b567d00228415489b2 100644 (file)
@@ -484,32 +484,44 @@ static void dwc2_handle_usb_suspend_intr(struct dwc2_hsotg *hsotg)
                 * state is active
                 */
                dsts = dwc2_readl(hsotg->regs + DSTS);
-               dev_dbg(hsotg->dev, "DSTS=0x%0x\n", dsts);
+               dev_dbg(hsotg->dev, "%s: DSTS=0x%0x\n", __func__, dsts);
                dev_dbg(hsotg->dev,
-                       "DSTS.Suspend Status=%d HWCFG4.Power Optimize=%d\n",
+                       "DSTS.Suspend Status=%d HWCFG4.Power Optimize=%d HWCFG4.Hibernation=%d\n",
                        !!(dsts & DSTS_SUSPSTS),
-                       hsotg->hw_params.power_optimized);
-               if ((dsts & DSTS_SUSPSTS) && hsotg->hw_params.power_optimized) {
-                       /* Ignore suspend request before enumeration */
-                       if (!dwc2_is_device_connected(hsotg)) {
-                               dev_dbg(hsotg->dev,
-                                       "ignore suspend request before enumeration\n");
-                               return;
+                       hsotg->hw_params.power_optimized,
+                       hsotg->hw_params.hibernation);
+
+               /* Ignore suspend request before enumeration */
+               if (!dwc2_is_device_connected(hsotg)) {
+                       dev_dbg(hsotg->dev,
+                               "ignore suspend request before enumeration\n");
+                       return;
+               }
+               if (dsts & DSTS_SUSPSTS) {
+                       if (hsotg->hw_params.power_optimized) {
+                               ret = dwc2_enter_partial_power_down(hsotg);
+                               if (ret) {
+                                       if (ret != -ENOTSUPP)
+                                               dev_err(hsotg->dev,
+                                                       "%s: enter partial_power_down failed\n",
+                                                       __func__);
+                                       goto skip_power_saving;
+                               }
+
+                               udelay(100);
+
+                               /* Ask phy to be suspended */
+                               if (!IS_ERR_OR_NULL(hsotg->uphy))
+                                       usb_phy_set_suspend(hsotg->uphy, true);
                        }
 
-                       ret = dwc2_enter_partial_power_down(hsotg);
-                       if (ret) {
-                               if (ret != -ENOTSUPP)
+                       if (hsotg->hw_params.hibernation) {
+                               ret = dwc2_enter_hibernation(hsotg, 0);
+                               if (ret && ret != -ENOTSUPP)
                                        dev_err(hsotg->dev,
-                                               "enter power_down failed\n");
-                               goto skip_power_saving;
+                                               "%s: enter hibernation failed\n",
+                                               __func__);
                        }
-
-                       udelay(100);
-
-                       /* Ask phy to be suspended */
-                       if (!IS_ERR_OR_NULL(hsotg->uphy))
-                               usb_phy_set_suspend(hsotg->uphy, true);
 skip_power_saving:
                        /*
                         * Change to L2 (suspend) state before releasing