Lines Matching +full:rtd +full:- +full:dwc3
1 // SPDX-License-Identifier: GPL-2.0
3 * core.c - DesignWare USB3 DRD Controller Core file
5 * Copyright (C) 2010-2011 Texas Instruments Incorporated - https://www.ti.com
24 #include <linux/dma-mapping.h>
42 #include "../host/xhci-ext-caps.h"
47 * dwc3_get_dr_mode - Validates and sets dr_mode
50 static int dwc3_get_dr_mode(struct dwc3 *dwc) in dwc3_get_dr_mode()
53 struct device *dev = dwc->dev; in dwc3_get_dr_mode()
56 if (dwc->dr_mode == USB_DR_MODE_UNKNOWN) in dwc3_get_dr_mode()
57 dwc->dr_mode = USB_DR_MODE_OTG; in dwc3_get_dr_mode()
59 mode = dwc->dr_mode; in dwc3_get_dr_mode()
60 hw_mode = DWC3_GHWPARAMS0_MODE(dwc->hwparams.hwparams0); in dwc3_get_dr_mode()
67 return -EINVAL; in dwc3_get_dr_mode()
75 return -EINVAL; in dwc3_get_dr_mode()
90 if (mode == USB_DR_MODE_OTG && !dwc->edev && in dwc3_get_dr_mode()
92 !device_property_read_bool(dwc->dev, "usb-role-switch")) && in dwc3_get_dr_mode()
93 !DWC3_VER_IS_PRIOR(DWC3, 330A)) in dwc3_get_dr_mode()
97 if (mode != dwc->dr_mode) { in dwc3_get_dr_mode()
102 dwc->dr_mode = mode; in dwc3_get_dr_mode()
108 void dwc3_enable_susphy(struct dwc3 *dwc, bool enable) in dwc3_enable_susphy()
113 for (i = 0; i < dwc->num_usb3_ports; i++) { in dwc3_enable_susphy()
114 reg = dwc3_readl(dwc->regs, DWC3_GUSB3PIPECTL(i)); in dwc3_enable_susphy()
115 if (enable && !dwc->dis_u3_susphy_quirk) in dwc3_enable_susphy()
120 dwc3_writel(dwc->regs, DWC3_GUSB3PIPECTL(i), reg); in dwc3_enable_susphy()
123 for (i = 0; i < dwc->num_usb2_ports; i++) { in dwc3_enable_susphy()
124 reg = dwc3_readl(dwc->regs, DWC3_GUSB2PHYCFG(i)); in dwc3_enable_susphy()
125 if (enable && !dwc->dis_u2_susphy_quirk) in dwc3_enable_susphy()
130 dwc3_writel(dwc->regs, DWC3_GUSB2PHYCFG(i), reg); in dwc3_enable_susphy()
134 void dwc3_set_prtcap(struct dwc3 *dwc, u32 mode) in dwc3_set_prtcap()
138 reg = dwc3_readl(dwc->regs, DWC3_GCTL); in dwc3_set_prtcap()
141 dwc3_writel(dwc->regs, DWC3_GCTL, reg); in dwc3_set_prtcap()
143 dwc->current_dr_role = mode; in dwc3_set_prtcap()
148 struct dwc3 *dwc = work_to_dwc(work); in __dwc3_set_mode()
155 mutex_lock(&dwc->mutex); in __dwc3_set_mode()
156 spin_lock_irqsave(&dwc->lock, flags); in __dwc3_set_mode()
157 desired_dr_role = dwc->desired_dr_role; in __dwc3_set_mode()
158 spin_unlock_irqrestore(&dwc->lock, flags); in __dwc3_set_mode()
160 pm_runtime_get_sync(dwc->dev); in __dwc3_set_mode()
162 if (dwc->current_dr_role == DWC3_GCTL_PRTCAP_OTG) in __dwc3_set_mode()
168 if (desired_dr_role == dwc->current_dr_role) in __dwc3_set_mode()
171 if (desired_dr_role == DWC3_GCTL_PRTCAP_OTG && dwc->edev) in __dwc3_set_mode()
174 switch (dwc->current_dr_role) { in __dwc3_set_mode()
184 spin_lock_irqsave(&dwc->lock, flags); in __dwc3_set_mode()
185 dwc->desired_otg_role = DWC3_OTG_ROLE_IDLE; in __dwc3_set_mode()
186 spin_unlock_irqrestore(&dwc->lock, flags); in __dwc3_set_mode()
197 if (dwc->current_dr_role && ((DWC3_IP_IS(DWC3) || in __dwc3_set_mode()
200 reg = dwc3_readl(dwc->regs, DWC3_GCTL); in __dwc3_set_mode()
202 dwc3_writel(dwc->regs, DWC3_GCTL, reg); in __dwc3_set_mode()
212 reg = dwc3_readl(dwc->regs, DWC3_GCTL); in __dwc3_set_mode()
214 dwc3_writel(dwc->regs, DWC3_GCTL, reg); in __dwc3_set_mode()
217 spin_lock_irqsave(&dwc->lock, flags); in __dwc3_set_mode()
221 spin_unlock_irqrestore(&dwc->lock, flags); in __dwc3_set_mode()
227 dev_err(dwc->dev, "failed to initialize host\n"); in __dwc3_set_mode()
229 if (dwc->usb2_phy) in __dwc3_set_mode()
230 otg_set_vbus(dwc->usb2_phy->otg, true); in __dwc3_set_mode()
232 for (i = 0; i < dwc->num_usb2_ports; i++) in __dwc3_set_mode()
233 phy_set_mode(dwc->usb2_generic_phy[i], PHY_MODE_USB_HOST); in __dwc3_set_mode()
234 for (i = 0; i < dwc->num_usb3_ports; i++) in __dwc3_set_mode()
235 phy_set_mode(dwc->usb3_generic_phy[i], PHY_MODE_USB_HOST); in __dwc3_set_mode()
237 if (dwc->dis_split_quirk) { in __dwc3_set_mode()
238 reg = dwc3_readl(dwc->regs, DWC3_GUCTL3); in __dwc3_set_mode()
240 dwc3_writel(dwc->regs, DWC3_GUCTL3, reg); in __dwc3_set_mode()
249 if (dwc->usb2_phy) in __dwc3_set_mode()
250 otg_set_vbus(dwc->usb2_phy->otg, false); in __dwc3_set_mode()
251 phy_set_mode(dwc->usb2_generic_phy[0], PHY_MODE_USB_DEVICE); in __dwc3_set_mode()
252 phy_set_mode(dwc->usb3_generic_phy[0], PHY_MODE_USB_DEVICE); in __dwc3_set_mode()
256 dev_err(dwc->dev, "failed to initialize peripheral\n"); in __dwc3_set_mode()
267 pm_runtime_mark_last_busy(dwc->dev); in __dwc3_set_mode()
268 pm_runtime_put_autosuspend(dwc->dev); in __dwc3_set_mode()
269 mutex_unlock(&dwc->mutex); in __dwc3_set_mode()
272 void dwc3_set_mode(struct dwc3 *dwc, u32 mode) in dwc3_set_mode()
276 if (dwc->dr_mode != USB_DR_MODE_OTG) in dwc3_set_mode()
279 spin_lock_irqsave(&dwc->lock, flags); in dwc3_set_mode()
280 dwc->desired_dr_role = mode; in dwc3_set_mode()
281 spin_unlock_irqrestore(&dwc->lock, flags); in dwc3_set_mode()
283 queue_work(system_freezable_wq, &dwc->drd_work); in dwc3_set_mode()
288 struct dwc3 *dwc = dep->dwc; in dwc3_core_fifo_space()
291 dwc3_writel(dwc->regs, DWC3_GDBGFIFOSPACE, in dwc3_core_fifo_space()
292 DWC3_GDBGFIFOSPACE_NUM(dep->number) | in dwc3_core_fifo_space()
295 reg = dwc3_readl(dwc->regs, DWC3_GDBGFIFOSPACE); in dwc3_core_fifo_space()
301 * dwc3_core_soft_reset - Issues core soft reset and PHY reset
304 int dwc3_core_soft_reset(struct dwc3 *dwc) in dwc3_core_soft_reset()
311 * XHCI driver will reset the host block. If dwc3 was configured for in dwc3_core_soft_reset()
312 * host-only mode, then we can return early. in dwc3_core_soft_reset()
314 if (dwc->current_dr_role == DWC3_GCTL_PRTCAP_HOST) in dwc3_core_soft_reset()
317 reg = dwc3_readl(dwc->regs, DWC3_DCTL); in dwc3_core_soft_reset()
332 reg = dwc3_readl(dwc->regs, DWC3_DCTL); in dwc3_core_soft_reset()
340 } while (--retries); in dwc3_core_soft_reset()
342 dev_warn(dwc->dev, "DWC3 controller soft reset failed.\n"); in dwc3_core_soft_reset()
343 return -ETIMEDOUT; in dwc3_core_soft_reset()
358 * dwc3_frame_length_adjustment - Adjusts frame length if required
359 * @dwc3: Pointer to our controller context structure
361 static void dwc3_frame_length_adjustment(struct dwc3 *dwc) in dwc3_frame_length_adjustment()
366 if (DWC3_VER_IS_PRIOR(DWC3, 250A)) in dwc3_frame_length_adjustment()
369 if (dwc->fladj == 0) in dwc3_frame_length_adjustment()
372 reg = dwc3_readl(dwc->regs, DWC3_GFLADJ); in dwc3_frame_length_adjustment()
374 if (dft != dwc->fladj) { in dwc3_frame_length_adjustment()
376 reg |= DWC3_GFLADJ_30MHZ_SDBND_SEL | dwc->fladj; in dwc3_frame_length_adjustment()
377 dwc3_writel(dwc->regs, DWC3_GFLADJ, reg); in dwc3_frame_length_adjustment()
382 * dwc3_ref_clk_period - Reference clock period configuration
389 static void dwc3_ref_clk_period(struct dwc3 *dwc) in dwc3_ref_clk_period()
397 if (dwc->ref_clk) { in dwc3_ref_clk_period()
398 rate = clk_get_rate(dwc->ref_clk); in dwc3_ref_clk_period()
402 } else if (dwc->ref_clk_per) { in dwc3_ref_clk_period()
403 period = dwc->ref_clk_per; in dwc3_ref_clk_period()
409 reg = dwc3_readl(dwc->regs, DWC3_GUCTL); in dwc3_ref_clk_period()
412 dwc3_writel(dwc->regs, DWC3_GUCTL, reg); in dwc3_ref_clk_period()
414 if (DWC3_VER_IS_PRIOR(DWC3, 250A)) in dwc3_ref_clk_period()
420 * 125000 * (NSEC_PER_SEC / (rate * period) - 1) in dwc3_ref_clk_period()
422 * but rearranged for fixed-point arithmetic. The division must be in dwc3_ref_clk_period()
423 * 64-bit because 125000 * NSEC_PER_SEC doesn't fit in 32 bits (and in dwc3_ref_clk_period()
433 fladj -= 125000; in dwc3_ref_clk_period()
440 reg = dwc3_readl(dwc->regs, DWC3_GFLADJ); in dwc3_ref_clk_period()
448 if (dwc->gfladj_refclk_lpm_sel) in dwc3_ref_clk_period()
451 dwc3_writel(dwc->regs, DWC3_GFLADJ, reg); in dwc3_ref_clk_period()
455 * dwc3_free_one_event_buffer - Frees one event buffer
459 static void dwc3_free_one_event_buffer(struct dwc3 *dwc, in dwc3_free_one_event_buffer()
462 dma_free_coherent(dwc->sysdev, evt->length, evt->buf, evt->dma); in dwc3_free_one_event_buffer()
466 * dwc3_alloc_one_event_buffer - Allocates one event buffer structure
473 static struct dwc3_event_buffer *dwc3_alloc_one_event_buffer(struct dwc3 *dwc, in dwc3_alloc_one_event_buffer()
478 evt = devm_kzalloc(dwc->dev, sizeof(*evt), GFP_KERNEL); in dwc3_alloc_one_event_buffer()
480 return ERR_PTR(-ENOMEM); in dwc3_alloc_one_event_buffer()
482 evt->dwc = dwc; in dwc3_alloc_one_event_buffer()
483 evt->length = length; in dwc3_alloc_one_event_buffer()
484 evt->cache = devm_kzalloc(dwc->dev, length, GFP_KERNEL); in dwc3_alloc_one_event_buffer()
485 if (!evt->cache) in dwc3_alloc_one_event_buffer()
486 return ERR_PTR(-ENOMEM); in dwc3_alloc_one_event_buffer()
488 evt->buf = dma_alloc_coherent(dwc->sysdev, length, in dwc3_alloc_one_event_buffer()
489 &evt->dma, GFP_KERNEL); in dwc3_alloc_one_event_buffer()
490 if (!evt->buf) in dwc3_alloc_one_event_buffer()
491 return ERR_PTR(-ENOMEM); in dwc3_alloc_one_event_buffer()
497 * dwc3_free_event_buffers - frees all allocated event buffers
500 static void dwc3_free_event_buffers(struct dwc3 *dwc) in dwc3_free_event_buffers()
504 evt = dwc->ev_buf; in dwc3_free_event_buffers()
510 * dwc3_alloc_event_buffers - Allocates @num event buffers of size @length
517 static int dwc3_alloc_event_buffers(struct dwc3 *dwc, unsigned int length) in dwc3_alloc_event_buffers()
522 hw_mode = DWC3_GHWPARAMS0_MODE(dwc->hwparams.hwparams0); in dwc3_alloc_event_buffers()
524 dwc->ev_buf = NULL; in dwc3_alloc_event_buffers()
530 dev_err(dwc->dev, "can't allocate event buffer\n"); in dwc3_alloc_event_buffers()
533 dwc->ev_buf = evt; in dwc3_alloc_event_buffers()
539 * dwc3_event_buffers_setup - setup our allocated event buffers
544 int dwc3_event_buffers_setup(struct dwc3 *dwc) in dwc3_event_buffers_setup()
549 if (!dwc->ev_buf) in dwc3_event_buffers_setup()
552 evt = dwc->ev_buf; in dwc3_event_buffers_setup()
553 evt->lpos = 0; in dwc3_event_buffers_setup()
554 dwc3_writel(dwc->regs, DWC3_GEVNTADRLO(0), in dwc3_event_buffers_setup()
555 lower_32_bits(evt->dma)); in dwc3_event_buffers_setup()
556 dwc3_writel(dwc->regs, DWC3_GEVNTADRHI(0), in dwc3_event_buffers_setup()
557 upper_32_bits(evt->dma)); in dwc3_event_buffers_setup()
558 dwc3_writel(dwc->regs, DWC3_GEVNTSIZ(0), in dwc3_event_buffers_setup()
559 DWC3_GEVNTSIZ_SIZE(evt->length)); in dwc3_event_buffers_setup()
562 reg = dwc3_readl(dwc->regs, DWC3_GEVNTCOUNT(0)); in dwc3_event_buffers_setup()
563 dwc3_writel(dwc->regs, DWC3_GEVNTCOUNT(0), reg); in dwc3_event_buffers_setup()
567 void dwc3_event_buffers_cleanup(struct dwc3 *dwc) in dwc3_event_buffers_cleanup()
572 if (!dwc->ev_buf) in dwc3_event_buffers_cleanup()
578 reg = dwc3_readl(dwc->regs, DWC3_DSTS); in dwc3_event_buffers_cleanup()
582 evt = dwc->ev_buf; in dwc3_event_buffers_cleanup()
584 evt->lpos = 0; in dwc3_event_buffers_cleanup()
586 dwc3_writel(dwc->regs, DWC3_GEVNTADRLO(0), 0); in dwc3_event_buffers_cleanup()
587 dwc3_writel(dwc->regs, DWC3_GEVNTADRHI(0), 0); in dwc3_event_buffers_cleanup()
588 dwc3_writel(dwc->regs, DWC3_GEVNTSIZ(0), DWC3_GEVNTSIZ_INTMASK in dwc3_event_buffers_cleanup()
592 reg = dwc3_readl(dwc->regs, DWC3_GEVNTCOUNT(0)); in dwc3_event_buffers_cleanup()
593 dwc3_writel(dwc->regs, DWC3_GEVNTCOUNT(0), reg); in dwc3_event_buffers_cleanup()
596 static void dwc3_core_num_eps(struct dwc3 *dwc) in dwc3_core_num_eps()
598 struct dwc3_hwparams *parms = &dwc->hwparams; in dwc3_core_num_eps()
600 dwc->num_eps = DWC3_NUM_EPS(parms); in dwc3_core_num_eps()
603 static void dwc3_cache_hwparams(struct dwc3 *dwc) in dwc3_cache_hwparams()
605 struct dwc3_hwparams *parms = &dwc->hwparams; in dwc3_cache_hwparams()
607 parms->hwparams0 = dwc3_readl(dwc->regs, DWC3_GHWPARAMS0); in dwc3_cache_hwparams()
608 parms->hwparams1 = dwc3_readl(dwc->regs, DWC3_GHWPARAMS1); in dwc3_cache_hwparams()
609 parms->hwparams2 = dwc3_readl(dwc->regs, DWC3_GHWPARAMS2); in dwc3_cache_hwparams()
610 parms->hwparams3 = dwc3_readl(dwc->regs, DWC3_GHWPARAMS3); in dwc3_cache_hwparams()
611 parms->hwparams4 = dwc3_readl(dwc->regs, DWC3_GHWPARAMS4); in dwc3_cache_hwparams()
612 parms->hwparams5 = dwc3_readl(dwc->regs, DWC3_GHWPARAMS5); in dwc3_cache_hwparams()
613 parms->hwparams6 = dwc3_readl(dwc->regs, DWC3_GHWPARAMS6); in dwc3_cache_hwparams()
614 parms->hwparams7 = dwc3_readl(dwc->regs, DWC3_GHWPARAMS7); in dwc3_cache_hwparams()
615 parms->hwparams8 = dwc3_readl(dwc->regs, DWC3_GHWPARAMS8); in dwc3_cache_hwparams()
618 parms->hwparams9 = dwc3_readl(dwc->regs, DWC3_GHWPARAMS9); in dwc3_cache_hwparams()
621 static void dwc3_config_soc_bus(struct dwc3 *dwc) in dwc3_config_soc_bus()
623 if (dwc->gsbuscfg0_reqinfo != DWC3_GSBUSCFG0_REQINFO_UNSPECIFIED) { in dwc3_config_soc_bus()
626 reg = dwc3_readl(dwc->regs, DWC3_GSBUSCFG0); in dwc3_config_soc_bus()
628 reg |= DWC3_GSBUSCFG0_REQINFO(dwc->gsbuscfg0_reqinfo); in dwc3_config_soc_bus()
629 dwc3_writel(dwc->regs, DWC3_GSBUSCFG0, reg); in dwc3_config_soc_bus()
633 static int dwc3_core_ulpi_init(struct dwc3 *dwc) in dwc3_core_ulpi_init()
638 intf = DWC3_GHWPARAMS3_HSPHY_IFC(dwc->hwparams.hwparams3); in dwc3_core_ulpi_init()
642 dwc->hsphy_interface && in dwc3_core_ulpi_init()
643 !strncmp(dwc->hsphy_interface, "ulpi", 4))) in dwc3_core_ulpi_init()
649 static int dwc3_ss_phy_setup(struct dwc3 *dwc, int index) in dwc3_ss_phy_setup()
653 reg = dwc3_readl(dwc->regs, DWC3_GUSB3PIPECTL(index)); in dwc3_ss_phy_setup()
668 * cleared after power-on reset, and it can be set after core in dwc3_ss_phy_setup()
673 if (dwc->u2ss_inp3_quirk) in dwc3_ss_phy_setup()
676 if (dwc->dis_rxdet_inp3_quirk) in dwc3_ss_phy_setup()
679 if (dwc->req_p1p2p3_quirk) in dwc3_ss_phy_setup()
682 if (dwc->del_p1p2p3_quirk) in dwc3_ss_phy_setup()
685 if (dwc->del_phy_power_chg_quirk) in dwc3_ss_phy_setup()
688 if (dwc->lfps_filter_quirk) in dwc3_ss_phy_setup()
691 if (dwc->rx_detect_poll_quirk) in dwc3_ss_phy_setup()
694 if (dwc->tx_de_emphasis_quirk) in dwc3_ss_phy_setup()
695 reg |= DWC3_GUSB3PIPECTL_TX_DEEPH(dwc->tx_de_emphasis); in dwc3_ss_phy_setup()
697 if (dwc->dis_del_phy_power_chg_quirk) in dwc3_ss_phy_setup()
700 dwc3_writel(dwc->regs, DWC3_GUSB3PIPECTL(index), reg); in dwc3_ss_phy_setup()
705 static int dwc3_hs_phy_setup(struct dwc3 *dwc, int index) in dwc3_hs_phy_setup()
709 reg = dwc3_readl(dwc->regs, DWC3_GUSB2PHYCFG(index)); in dwc3_hs_phy_setup()
712 switch (DWC3_GHWPARAMS3_HSPHY_IFC(dwc->hwparams.hwparams3)) { in dwc3_hs_phy_setup()
714 if (dwc->hsphy_interface && in dwc3_hs_phy_setup()
715 !strncmp(dwc->hsphy_interface, "utmi", 4)) { in dwc3_hs_phy_setup()
718 } else if (dwc->hsphy_interface && in dwc3_hs_phy_setup()
719 !strncmp(dwc->hsphy_interface, "ulpi", 4)) { in dwc3_hs_phy_setup()
721 dwc3_writel(dwc->regs, DWC3_GUSB2PHYCFG(index), reg); in dwc3_hs_phy_setup()
733 switch (dwc->hsphy_mode) { in dwc3_hs_phy_setup()
757 * after power-on reset, and it can be set after core initialization. in dwc3_hs_phy_setup()
761 if (dwc->dis_enblslpm_quirk) in dwc3_hs_phy_setup()
766 if (dwc->dis_u2_freeclk_exists_quirk || dwc->gfladj_refclk_lpm_sel) in dwc3_hs_phy_setup()
776 if (dwc->ulpi_ext_vbus_drv) in dwc3_hs_phy_setup()
779 dwc3_writel(dwc->regs, DWC3_GUSB2PHYCFG(index), reg); in dwc3_hs_phy_setup()
785 * dwc3_phy_setup - Configure USB PHY Interface of DWC3 Core
792 static int dwc3_phy_setup(struct dwc3 *dwc) in dwc3_phy_setup()
797 for (i = 0; i < dwc->num_usb3_ports; i++) { in dwc3_phy_setup()
803 for (i = 0; i < dwc->num_usb2_ports; i++) { in dwc3_phy_setup()
812 static int dwc3_phy_init(struct dwc3 *dwc) in dwc3_phy_init()
818 usb_phy_init(dwc->usb2_phy); in dwc3_phy_init()
819 usb_phy_init(dwc->usb3_phy); in dwc3_phy_init()
821 for (i = 0; i < dwc->num_usb2_ports; i++) { in dwc3_phy_init()
822 ret = phy_init(dwc->usb2_generic_phy[i]); in dwc3_phy_init()
827 for (j = 0; j < dwc->num_usb3_ports; j++) { in dwc3_phy_init()
828 ret = phy_init(dwc->usb3_generic_phy[j]); in dwc3_phy_init()
836 while (--j >= 0) in dwc3_phy_init()
837 phy_exit(dwc->usb3_generic_phy[j]); in dwc3_phy_init()
840 while (--i >= 0) in dwc3_phy_init()
841 phy_exit(dwc->usb2_generic_phy[i]); in dwc3_phy_init()
843 usb_phy_shutdown(dwc->usb3_phy); in dwc3_phy_init()
844 usb_phy_shutdown(dwc->usb2_phy); in dwc3_phy_init()
849 static void dwc3_phy_exit(struct dwc3 *dwc) in dwc3_phy_exit()
853 for (i = 0; i < dwc->num_usb3_ports; i++) in dwc3_phy_exit()
854 phy_exit(dwc->usb3_generic_phy[i]); in dwc3_phy_exit()
856 for (i = 0; i < dwc->num_usb2_ports; i++) in dwc3_phy_exit()
857 phy_exit(dwc->usb2_generic_phy[i]); in dwc3_phy_exit()
859 usb_phy_shutdown(dwc->usb3_phy); in dwc3_phy_exit()
860 usb_phy_shutdown(dwc->usb2_phy); in dwc3_phy_exit()
863 static int dwc3_phy_power_on(struct dwc3 *dwc) in dwc3_phy_power_on()
869 usb_phy_set_suspend(dwc->usb2_phy, 0); in dwc3_phy_power_on()
870 usb_phy_set_suspend(dwc->usb3_phy, 0); in dwc3_phy_power_on()
872 for (i = 0; i < dwc->num_usb2_ports; i++) { in dwc3_phy_power_on()
873 ret = phy_power_on(dwc->usb2_generic_phy[i]); in dwc3_phy_power_on()
878 for (j = 0; j < dwc->num_usb3_ports; j++) { in dwc3_phy_power_on()
879 ret = phy_power_on(dwc->usb3_generic_phy[j]); in dwc3_phy_power_on()
887 while (--j >= 0) in dwc3_phy_power_on()
888 phy_power_off(dwc->usb3_generic_phy[j]); in dwc3_phy_power_on()
891 while (--i >= 0) in dwc3_phy_power_on()
892 phy_power_off(dwc->usb2_generic_phy[i]); in dwc3_phy_power_on()
894 usb_phy_set_suspend(dwc->usb3_phy, 1); in dwc3_phy_power_on()
895 usb_phy_set_suspend(dwc->usb2_phy, 1); in dwc3_phy_power_on()
900 static void dwc3_phy_power_off(struct dwc3 *dwc) in dwc3_phy_power_off()
904 for (i = 0; i < dwc->num_usb3_ports; i++) in dwc3_phy_power_off()
905 phy_power_off(dwc->usb3_generic_phy[i]); in dwc3_phy_power_off()
907 for (i = 0; i < dwc->num_usb2_ports; i++) in dwc3_phy_power_off()
908 phy_power_off(dwc->usb2_generic_phy[i]); in dwc3_phy_power_off()
910 usb_phy_set_suspend(dwc->usb3_phy, 1); in dwc3_phy_power_off()
911 usb_phy_set_suspend(dwc->usb2_phy, 1); in dwc3_phy_power_off()
914 static int dwc3_clk_enable(struct dwc3 *dwc) in dwc3_clk_enable()
918 ret = clk_prepare_enable(dwc->bus_clk); in dwc3_clk_enable()
922 ret = clk_prepare_enable(dwc->ref_clk); in dwc3_clk_enable()
926 ret = clk_prepare_enable(dwc->susp_clk); in dwc3_clk_enable()
930 ret = clk_prepare_enable(dwc->utmi_clk); in dwc3_clk_enable()
934 ret = clk_prepare_enable(dwc->pipe_clk); in dwc3_clk_enable()
941 clk_disable_unprepare(dwc->utmi_clk); in dwc3_clk_enable()
943 clk_disable_unprepare(dwc->susp_clk); in dwc3_clk_enable()
945 clk_disable_unprepare(dwc->ref_clk); in dwc3_clk_enable()
947 clk_disable_unprepare(dwc->bus_clk); in dwc3_clk_enable()
951 static void dwc3_clk_disable(struct dwc3 *dwc) in dwc3_clk_disable()
953 clk_disable_unprepare(dwc->pipe_clk); in dwc3_clk_disable()
954 clk_disable_unprepare(dwc->utmi_clk); in dwc3_clk_disable()
955 clk_disable_unprepare(dwc->susp_clk); in dwc3_clk_disable()
956 clk_disable_unprepare(dwc->ref_clk); in dwc3_clk_disable()
957 clk_disable_unprepare(dwc->bus_clk); in dwc3_clk_disable()
960 static void dwc3_core_exit(struct dwc3 *dwc) in dwc3_core_exit()
966 reset_control_assert(dwc->reset); in dwc3_core_exit()
969 static bool dwc3_core_is_valid(struct dwc3 *dwc) in dwc3_core_is_valid()
973 reg = dwc3_readl(dwc->regs, DWC3_GSNPSID); in dwc3_core_is_valid()
974 dwc->ip = DWC3_GSNPS_ID(reg); in dwc3_core_is_valid()
977 if (DWC3_IP_IS(DWC3)) { in dwc3_core_is_valid()
978 dwc->revision = reg; in dwc3_core_is_valid()
980 dwc->revision = dwc3_readl(dwc->regs, DWC3_VER_NUMBER); in dwc3_core_is_valid()
981 dwc->version_type = dwc3_readl(dwc->regs, DWC3_VER_TYPE); in dwc3_core_is_valid()
989 static void dwc3_core_setup_global_control(struct dwc3 *dwc) in dwc3_core_setup_global_control()
995 reg = dwc3_readl(dwc->regs, DWC3_GCTL); in dwc3_core_setup_global_control()
997 hw_mode = DWC3_GHWPARAMS0_MODE(dwc->hwparams.hwparams0); in dwc3_core_setup_global_control()
998 power_opt = DWC3_GHWPARAMS1_EN_PWROPT(dwc->hwparams.hwparams1); in dwc3_core_setup_global_control()
1003 * WORKAROUND: DWC3 revisions between 2.10a and 2.50a have an in dwc3_core_setup_global_control()
1011 * STAR#9000588375: Clock Gating, SOF Issues when ref_clk-Based in dwc3_core_setup_global_control()
1014 if ((dwc->dr_mode == USB_DR_MODE_HOST || in dwc3_core_setup_global_control()
1015 dwc->dr_mode == USB_DR_MODE_OTG) && in dwc3_core_setup_global_control()
1016 DWC3_VER_IS_WITHIN(DWC3, 210A, 250A)) in dwc3_core_setup_global_control()
1023 * REVISIT Enabling this bit so that host-mode hibernation in dwc3_core_setup_global_control()
1024 * will work. Device-mode hibernation is not yet implemented. in dwc3_core_setup_global_control()
1047 /* check if current dwc3 is on simulation board */ in dwc3_core_setup_global_control()
1048 if (dwc->hwparams.hwparams6 & DWC3_GHWPARAMS6_EN_FPGA) { in dwc3_core_setup_global_control()
1049 dev_info(dwc->dev, "Running with FPGA optimizations\n"); in dwc3_core_setup_global_control()
1050 dwc->is_fpga = true; in dwc3_core_setup_global_control()
1053 WARN_ONCE(dwc->disable_scramble_quirk && !dwc->is_fpga, in dwc3_core_setup_global_control()
1054 "disable_scramble cannot be used on non-FPGA builds\n"); in dwc3_core_setup_global_control()
1056 if (dwc->disable_scramble_quirk && dwc->is_fpga) in dwc3_core_setup_global_control()
1061 if (dwc->u2exit_lfps_quirk) in dwc3_core_setup_global_control()
1065 * WORKAROUND: DWC3 revisions <1.90a have a bug in dwc3_core_setup_global_control()
1067 * and falls back to high-speed mode which causes in dwc3_core_setup_global_control()
1070 if (DWC3_VER_IS_PRIOR(DWC3, 190A)) in dwc3_core_setup_global_control()
1073 dwc3_writel(dwc->regs, DWC3_GCTL, reg); in dwc3_core_setup_global_control()
1076 static int dwc3_core_get_phy(struct dwc3 *dwc);
1077 static int dwc3_core_ulpi_init(struct dwc3 *dwc);
1080 static void dwc3_set_incr_burst_type(struct dwc3 *dwc) in dwc3_set_incr_burst_type()
1082 struct device *dev = dwc->dev; in dwc3_set_incr_burst_type()
1093 cfg = dwc3_readl(dwc->regs, DWC3_GSBUSCFG0); in dwc3_set_incr_burst_type()
1096 * Handle property "snps,incr-burst-type-adjustment". in dwc3_set_incr_burst_type()
1102 ntype = device_property_count_u32(dev, "snps,incr-burst-type-adjustment"); in dwc3_set_incr_burst_type()
1112 "snps,incr-burst-type-adjustment", vals, ntype); in dwc3_set_incr_burst_type()
1168 dwc3_writel(dwc->regs, DWC3_GSBUSCFG0, cfg); in dwc3_set_incr_burst_type()
1171 static void dwc3_set_power_down_clk_scale(struct dwc3 *dwc) in dwc3_set_power_down_clk_scale()
1176 if (!dwc->susp_clk) in dwc3_set_power_down_clk_scale()
1192 scale = DIV_ROUND_UP(clk_get_rate(dwc->susp_clk), 16000); in dwc3_set_power_down_clk_scale()
1193 reg = dwc3_readl(dwc->regs, DWC3_GCTL); in dwc3_set_power_down_clk_scale()
1198 dwc3_writel(dwc->regs, DWC3_GCTL, reg); in dwc3_set_power_down_clk_scale()
1202 static void dwc3_config_threshold(struct dwc3 *dwc) in dwc3_config_threshold()
1214 if (!DWC3_IP_IS(DWC3) && dwc->dr_mode == USB_DR_MODE_HOST) { in dwc3_config_threshold()
1215 rx_thr_num = dwc->rx_thr_num_pkt_prd; in dwc3_config_threshold()
1216 rx_maxburst = dwc->rx_max_burst_prd; in dwc3_config_threshold()
1217 tx_thr_num = dwc->tx_thr_num_pkt_prd; in dwc3_config_threshold()
1218 tx_maxburst = dwc->tx_max_burst_prd; in dwc3_config_threshold()
1221 reg = dwc3_readl(dwc->regs, DWC3_GRXTHRCFG); in dwc3_config_threshold()
1230 dwc3_writel(dwc->regs, DWC3_GRXTHRCFG, reg); in dwc3_config_threshold()
1234 reg = dwc3_readl(dwc->regs, DWC3_GTXTHRCFG); in dwc3_config_threshold()
1243 dwc3_writel(dwc->regs, DWC3_GTXTHRCFG, reg); in dwc3_config_threshold()
1247 rx_thr_num = dwc->rx_thr_num_pkt; in dwc3_config_threshold()
1248 rx_maxburst = dwc->rx_max_burst; in dwc3_config_threshold()
1249 tx_thr_num = dwc->tx_thr_num_pkt; in dwc3_config_threshold()
1250 tx_maxburst = dwc->tx_max_burst; in dwc3_config_threshold()
1252 if (DWC3_IP_IS(DWC3)) { in dwc3_config_threshold()
1254 reg = dwc3_readl(dwc->regs, DWC3_GRXTHRCFG); in dwc3_config_threshold()
1263 dwc3_writel(dwc->regs, DWC3_GRXTHRCFG, reg); in dwc3_config_threshold()
1267 reg = dwc3_readl(dwc->regs, DWC3_GTXTHRCFG); in dwc3_config_threshold()
1276 dwc3_writel(dwc->regs, DWC3_GTXTHRCFG, reg); in dwc3_config_threshold()
1280 reg = dwc3_readl(dwc->regs, DWC3_GRXTHRCFG); in dwc3_config_threshold()
1289 dwc3_writel(dwc->regs, DWC3_GRXTHRCFG, reg); in dwc3_config_threshold()
1293 reg = dwc3_readl(dwc->regs, DWC3_GTXTHRCFG); in dwc3_config_threshold()
1302 dwc3_writel(dwc->regs, DWC3_GTXTHRCFG, reg); in dwc3_config_threshold()
1308 * dwc3_core_init - Low-level initialization of DWC3 Core
1313 static int dwc3_core_init(struct dwc3 *dwc) in dwc3_core_init()
1319 hw_mode = DWC3_GHWPARAMS0_MODE(dwc->hwparams.hwparams0); in dwc3_core_init()
1325 dwc3_writel(dwc->regs, DWC3_GUID, LINUX_VERSION_CODE); in dwc3_core_init()
1331 if (!dwc->ulpi_ready) { in dwc3_core_init()
1334 if (ret == -ETIMEDOUT) { in dwc3_core_init()
1336 ret = -EPROBE_DEFER; in dwc3_core_init()
1340 dwc->ulpi_ready = true; in dwc3_core_init()
1343 if (!dwc->phys_ready) { in dwc3_core_init()
1347 dwc->phys_ready = true; in dwc3_core_init()
1380 dev_err(dwc->dev, "failed to setup event buffers\n"); in dwc3_core_init()
1389 if (DWC3_VER_IS_WITHIN(DWC3, 310A, ANY)) { in dwc3_core_init()
1390 reg = dwc3_readl(dwc->regs, DWC3_GUCTL2); in dwc3_core_init()
1392 dwc3_writel(dwc->regs, DWC3_GUCTL2, reg); in dwc3_core_init()
1404 if (DWC3_VER_IS(DWC3, 320A)) { in dwc3_core_init()
1405 reg = dwc3_readl(dwc->regs, DWC3_GUCTL2); in dwc3_core_init()
1407 dwc3_writel(dwc->regs, DWC3_GUCTL2, reg); in dwc3_core_init()
1414 * re-enumeration of usb device attached. All the termsel, xcvrsel, in dwc3_core_init()
1419 if (dwc->resume_hs_terminations) { in dwc3_core_init()
1420 reg = dwc3_readl(dwc->regs, DWC3_GUCTL1); in dwc3_core_init()
1422 dwc3_writel(dwc->regs, DWC3_GUCTL1, reg); in dwc3_core_init()
1425 if (!DWC3_VER_IS_PRIOR(DWC3, 250A)) { in dwc3_core_init()
1426 reg = dwc3_readl(dwc->regs, DWC3_GUCTL1); in dwc3_core_init()
1432 if (!DWC3_VER_IS_PRIOR(DWC3, 290A)) in dwc3_core_init()
1441 if (!DWC3_VER_IS_PRIOR(DWC3, 300A)) in dwc3_core_init()
1444 if (dwc->dis_tx_ipgap_linecheck_quirk) in dwc3_core_init()
1447 if (dwc->parkmode_disable_ss_quirk) in dwc3_core_init()
1450 if (dwc->parkmode_disable_hs_quirk) in dwc3_core_init()
1453 if (DWC3_VER_IS_WITHIN(DWC3, 290A, ANY)) { in dwc3_core_init()
1454 if (dwc->maximum_speed == USB_SPEED_FULL || in dwc3_core_init()
1455 dwc->maximum_speed == USB_SPEED_HIGH) in dwc3_core_init()
1461 dwc3_writel(dwc->regs, DWC3_GUCTL1, reg); in dwc3_core_init()
1472 dwc->maximum_speed == USB_SPEED_SUPER) { in dwc3_core_init()
1473 reg = dwc3_readl(dwc->regs, DWC3_LLUCTL); in dwc3_core_init()
1475 dwc3_writel(dwc->regs, DWC3_LLUCTL, reg); in dwc3_core_init()
1490 static int dwc3_core_get_phy(struct dwc3 *dwc) in dwc3_core_get_phy()
1492 struct device *dev = dwc->dev; in dwc3_core_get_phy()
1493 struct device_node *node = dev->of_node; in dwc3_core_get_phy()
1499 dwc->usb2_phy = devm_usb_get_phy_by_phandle(dev, "usb-phy", 0); in dwc3_core_get_phy()
1500 dwc->usb3_phy = devm_usb_get_phy_by_phandle(dev, "usb-phy", 1); in dwc3_core_get_phy()
1502 dwc->usb2_phy = devm_usb_get_phy(dev, USB_PHY_TYPE_USB2); in dwc3_core_get_phy()
1503 dwc->usb3_phy = devm_usb_get_phy(dev, USB_PHY_TYPE_USB3); in dwc3_core_get_phy()
1506 if (IS_ERR(dwc->usb2_phy)) { in dwc3_core_get_phy()
1507 ret = PTR_ERR(dwc->usb2_phy); in dwc3_core_get_phy()
1508 if (ret == -ENXIO || ret == -ENODEV) in dwc3_core_get_phy()
1509 dwc->usb2_phy = NULL; in dwc3_core_get_phy()
1514 if (IS_ERR(dwc->usb3_phy)) { in dwc3_core_get_phy()
1515 ret = PTR_ERR(dwc->usb3_phy); in dwc3_core_get_phy()
1516 if (ret == -ENXIO || ret == -ENODEV) in dwc3_core_get_phy()
1517 dwc->usb3_phy = NULL; in dwc3_core_get_phy()
1522 for (i = 0; i < dwc->num_usb2_ports; i++) { in dwc3_core_get_phy()
1523 if (dwc->num_usb2_ports == 1) in dwc3_core_get_phy()
1524 snprintf(phy_name, sizeof(phy_name), "usb2-phy"); in dwc3_core_get_phy()
1526 snprintf(phy_name, sizeof(phy_name), "usb2-%u", i); in dwc3_core_get_phy()
1528 dwc->usb2_generic_phy[i] = devm_phy_get(dev, phy_name); in dwc3_core_get_phy()
1529 if (IS_ERR(dwc->usb2_generic_phy[i])) { in dwc3_core_get_phy()
1530 ret = PTR_ERR(dwc->usb2_generic_phy[i]); in dwc3_core_get_phy()
1531 if (ret == -ENOSYS || ret == -ENODEV) in dwc3_core_get_phy()
1532 dwc->usb2_generic_phy[i] = NULL; in dwc3_core_get_phy()
1539 for (i = 0; i < dwc->num_usb3_ports; i++) { in dwc3_core_get_phy()
1540 if (dwc->num_usb3_ports == 1) in dwc3_core_get_phy()
1541 snprintf(phy_name, sizeof(phy_name), "usb3-phy"); in dwc3_core_get_phy()
1543 snprintf(phy_name, sizeof(phy_name), "usb3-%u", i); in dwc3_core_get_phy()
1545 dwc->usb3_generic_phy[i] = devm_phy_get(dev, phy_name); in dwc3_core_get_phy()
1546 if (IS_ERR(dwc->usb3_generic_phy[i])) { in dwc3_core_get_phy()
1547 ret = PTR_ERR(dwc->usb3_generic_phy[i]); in dwc3_core_get_phy()
1548 if (ret == -ENOSYS || ret == -ENODEV) in dwc3_core_get_phy()
1549 dwc->usb3_generic_phy[i] = NULL; in dwc3_core_get_phy()
1559 static int dwc3_core_init_mode(struct dwc3 *dwc) in dwc3_core_init_mode()
1561 struct device *dev = dwc->dev; in dwc3_core_init_mode()
1565 switch (dwc->dr_mode) { in dwc3_core_init_mode()
1569 if (dwc->usb2_phy) in dwc3_core_init_mode()
1570 otg_set_vbus(dwc->usb2_phy->otg, false); in dwc3_core_init_mode()
1571 phy_set_mode(dwc->usb2_generic_phy[0], PHY_MODE_USB_DEVICE); in dwc3_core_init_mode()
1572 phy_set_mode(dwc->usb3_generic_phy[0], PHY_MODE_USB_DEVICE); in dwc3_core_init_mode()
1581 if (dwc->usb2_phy) in dwc3_core_init_mode()
1582 otg_set_vbus(dwc->usb2_phy->otg, true); in dwc3_core_init_mode()
1583 for (i = 0; i < dwc->num_usb2_ports; i++) in dwc3_core_init_mode()
1584 phy_set_mode(dwc->usb2_generic_phy[i], PHY_MODE_USB_HOST); in dwc3_core_init_mode()
1585 for (i = 0; i < dwc->num_usb3_ports; i++) in dwc3_core_init_mode()
1586 phy_set_mode(dwc->usb3_generic_phy[i], PHY_MODE_USB_HOST); in dwc3_core_init_mode()
1593 INIT_WORK(&dwc->drd_work, __dwc3_set_mode); in dwc3_core_init_mode()
1596 return dev_err_probe(dev, ret, "failed to initialize dual-role\n"); in dwc3_core_init_mode()
1599 dev_err(dev, "Unsupported mode of operation %d\n", dwc->dr_mode); in dwc3_core_init_mode()
1600 return -EINVAL; in dwc3_core_init_mode()
1606 static void dwc3_core_exit_mode(struct dwc3 *dwc) in dwc3_core_exit_mode()
1608 switch (dwc->dr_mode) { in dwc3_core_exit_mode()
1623 /* de-assert DRVVBUS for HOST and OTG mode */ in dwc3_core_exit_mode()
1627 static void dwc3_get_software_properties(struct dwc3 *dwc) in dwc3_get_software_properties()
1633 dwc->gsbuscfg0_reqinfo = DWC3_GSBUSCFG0_REQINFO_UNSPECIFIED; in dwc3_get_software_properties()
1637 * and non-DT (non-ABI) properties. in dwc3_get_software_properties()
1639 for (tmpdev = dwc->dev; tmpdev; tmpdev = tmpdev->parent) { in dwc3_get_software_properties()
1641 "snps,gsbuscfg0-reqinfo", in dwc3_get_software_properties()
1644 dwc->gsbuscfg0_reqinfo = gsbuscfg0_reqinfo; in dwc3_get_software_properties()
1648 static void dwc3_get_properties(struct dwc3 *dwc) in dwc3_get_properties()
1650 struct device *dev = dwc->dev; in dwc3_get_properties()
1669 /* default to -3.5dB de-emphasis */ in dwc3_get_properties()
1685 dwc->maximum_speed = usb_get_maximum_speed(dev); in dwc3_get_properties()
1686 dwc->max_ssp_rate = usb_get_maximum_ssp_rate(dev); in dwc3_get_properties()
1687 dwc->dr_mode = usb_get_dr_mode(dev); in dwc3_get_properties()
1688 dwc->hsphy_mode = of_usb_get_phy_mode(dev->of_node); in dwc3_get_properties()
1690 dwc->sysdev_is_parent = device_property_read_bool(dev, in dwc3_get_properties()
1692 if (dwc->sysdev_is_parent) in dwc3_get_properties()
1693 dwc->sysdev = dwc->dev->parent; in dwc3_get_properties()
1695 dwc->sysdev = dwc->dev; in dwc3_get_properties()
1697 dwc->sys_wakeup = device_may_wakeup(dwc->sysdev); in dwc3_get_properties()
1699 ret = device_property_read_string(dev, "usb-psy-name", &usb_psy_name); in dwc3_get_properties()
1701 dwc->usb_psy = power_supply_get_by_name(usb_psy_name); in dwc3_get_properties()
1702 if (!dwc->usb_psy) in dwc3_get_properties()
1706 dwc->has_lpm_erratum = device_property_read_bool(dev, in dwc3_get_properties()
1707 "snps,has-lpm-erratum"); in dwc3_get_properties()
1708 device_property_read_u8(dev, "snps,lpm-nyet-threshold", in dwc3_get_properties()
1710 dwc->is_utmi_l1_suspend = device_property_read_bool(dev, in dwc3_get_properties()
1711 "snps,is-utmi-l1-suspend"); in dwc3_get_properties()
1712 device_property_read_u8(dev, "snps,hird-threshold", in dwc3_get_properties()
1714 dwc->dis_start_transfer_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1715 "snps,dis-start-transfer-quirk"); in dwc3_get_properties()
1716 dwc->usb3_lpm_capable = device_property_read_bool(dev, in dwc3_get_properties()
1718 dwc->usb2_lpm_disable = device_property_read_bool(dev, in dwc3_get_properties()
1719 "snps,usb2-lpm-disable"); in dwc3_get_properties()
1720 dwc->usb2_gadget_lpm_disable = device_property_read_bool(dev, in dwc3_get_properties()
1721 "snps,usb2-gadget-lpm-disable"); in dwc3_get_properties()
1722 device_property_read_u8(dev, "snps,rx-thr-num-pkt", in dwc3_get_properties()
1724 device_property_read_u8(dev, "snps,rx-max-burst", in dwc3_get_properties()
1726 device_property_read_u8(dev, "snps,tx-thr-num-pkt", in dwc3_get_properties()
1728 device_property_read_u8(dev, "snps,tx-max-burst", in dwc3_get_properties()
1730 device_property_read_u8(dev, "snps,rx-thr-num-pkt-prd", in dwc3_get_properties()
1732 device_property_read_u8(dev, "snps,rx-max-burst-prd", in dwc3_get_properties()
1734 device_property_read_u8(dev, "snps,tx-thr-num-pkt-prd", in dwc3_get_properties()
1736 device_property_read_u8(dev, "snps,tx-max-burst-prd", in dwc3_get_properties()
1738 dwc->do_fifo_resize = device_property_read_bool(dev, in dwc3_get_properties()
1739 "tx-fifo-resize"); in dwc3_get_properties()
1740 if (dwc->do_fifo_resize) in dwc3_get_properties()
1741 device_property_read_u8(dev, "tx-fifo-max-num", in dwc3_get_properties()
1744 dwc->disable_scramble_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1746 dwc->u2exit_lfps_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1748 dwc->u2ss_inp3_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1750 dwc->req_p1p2p3_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1752 dwc->del_p1p2p3_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1754 dwc->del_phy_power_chg_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1756 dwc->lfps_filter_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1758 dwc->rx_detect_poll_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1760 dwc->dis_u3_susphy_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1762 dwc->dis_u2_susphy_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1764 dwc->dis_enblslpm_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1766 dwc->dis_u1_entry_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1767 "snps,dis-u1-entry-quirk"); in dwc3_get_properties()
1768 dwc->dis_u2_entry_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1769 "snps,dis-u2-entry-quirk"); in dwc3_get_properties()
1770 dwc->dis_rxdet_inp3_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1772 dwc->dis_u2_freeclk_exists_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1773 "snps,dis-u2-freeclk-exists-quirk"); in dwc3_get_properties()
1774 dwc->dis_del_phy_power_chg_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1775 "snps,dis-del-phy-power-chg-quirk"); in dwc3_get_properties()
1776 dwc->dis_tx_ipgap_linecheck_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1777 "snps,dis-tx-ipgap-linecheck-quirk"); in dwc3_get_properties()
1778 dwc->resume_hs_terminations = device_property_read_bool(dev, in dwc3_get_properties()
1779 "snps,resume-hs-terminations"); in dwc3_get_properties()
1780 dwc->ulpi_ext_vbus_drv = device_property_read_bool(dev, in dwc3_get_properties()
1781 "snps,ulpi-ext-vbus-drv"); in dwc3_get_properties()
1782 dwc->parkmode_disable_ss_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1783 "snps,parkmode-disable-ss-quirk"); in dwc3_get_properties()
1784 dwc->parkmode_disable_hs_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1785 "snps,parkmode-disable-hs-quirk"); in dwc3_get_properties()
1786 dwc->gfladj_refclk_lpm_sel = device_property_read_bool(dev, in dwc3_get_properties()
1787 "snps,gfladj-refclk-lpm-sel-quirk"); in dwc3_get_properties()
1789 dwc->tx_de_emphasis_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1794 &dwc->hsphy_interface); in dwc3_get_properties()
1795 device_property_read_u32(dev, "snps,quirk-frame-length-adjustment", in dwc3_get_properties()
1796 &dwc->fladj); in dwc3_get_properties()
1797 device_property_read_u32(dev, "snps,ref-clock-period-ns", in dwc3_get_properties()
1798 &dwc->ref_clk_per); in dwc3_get_properties()
1800 dwc->dis_metastability_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1803 dwc->dis_split_quirk = device_property_read_bool(dev, in dwc3_get_properties()
1804 "snps,dis-split-quirk"); in dwc3_get_properties()
1806 dwc->lpm_nyet_threshold = lpm_nyet_threshold; in dwc3_get_properties()
1807 dwc->tx_de_emphasis = tx_de_emphasis; in dwc3_get_properties()
1809 dwc->hird_threshold = hird_threshold; in dwc3_get_properties()
1811 dwc->rx_thr_num_pkt = rx_thr_num_pkt; in dwc3_get_properties()
1812 dwc->rx_max_burst = rx_max_burst; in dwc3_get_properties()
1814 dwc->tx_thr_num_pkt = tx_thr_num_pkt; in dwc3_get_properties()
1815 dwc->tx_max_burst = tx_max_burst; in dwc3_get_properties()
1817 dwc->rx_thr_num_pkt_prd = rx_thr_num_pkt_prd; in dwc3_get_properties()
1818 dwc->rx_max_burst_prd = rx_max_burst_prd; in dwc3_get_properties()
1820 dwc->tx_thr_num_pkt_prd = tx_thr_num_pkt_prd; in dwc3_get_properties()
1821 dwc->tx_max_burst_prd = tx_max_burst_prd; in dwc3_get_properties()
1823 dwc->imod_interval = 0; in dwc3_get_properties()
1825 dwc->tx_fifo_resize_max_num = tx_fifo_resize_max_num; in dwc3_get_properties()
1829 bool dwc3_has_imod(struct dwc3 *dwc) in dwc3_has_imod()
1831 return DWC3_VER_IS_WITHIN(DWC3, 300A, ANY) || in dwc3_has_imod()
1836 static void dwc3_check_params(struct dwc3 *dwc) in dwc3_check_params()
1838 struct device *dev = dwc->dev; in dwc3_check_params()
1840 DWC3_GHWPARAMS3_SSPHY_IFC(dwc->hwparams.hwparams3); in dwc3_check_params()
1843 if (dwc->imod_interval && !dwc3_has_imod(dwc)) { in dwc3_check_params()
1844 dev_warn(dwc->dev, "Interrupt moderation not supported\n"); in dwc3_check_params()
1845 dwc->imod_interval = 0; in dwc3_check_params()
1855 if (!dwc->imod_interval && in dwc3_check_params()
1856 DWC3_VER_IS(DWC3, 300A)) in dwc3_check_params()
1857 dwc->imod_interval = 1; in dwc3_check_params()
1860 switch (dwc->maximum_speed) { in dwc3_check_params()
1877 dwc->maximum_speed); in dwc3_check_params()
1882 dwc->maximum_speed = USB_SPEED_SUPER_PLUS; in dwc3_check_params()
1886 dwc->maximum_speed = USB_SPEED_SUPER_PLUS; in dwc3_check_params()
1888 dwc->maximum_speed = USB_SPEED_SUPER; in dwc3_check_params()
1891 dwc->maximum_speed = USB_SPEED_HIGH; in dwc3_check_params()
1894 dwc->maximum_speed = USB_SPEED_SUPER; in dwc3_check_params()
1904 * set the default to support dual-lane for DWC_usb32 and single-lane in dwc3_check_params()
1905 * for DWC_usb31 for super-speed-plus. in dwc3_check_params()
1907 if (dwc->maximum_speed == USB_SPEED_SUPER_PLUS) { in dwc3_check_params()
1908 switch (dwc->max_ssp_rate) { in dwc3_check_params()
1923 dwc->max_ssp_rate = USB_SSP_GEN_2x2; in dwc3_check_params()
1925 dwc->max_ssp_rate = USB_SSP_GEN_2x1; in dwc3_check_params()
1929 dwc->max_ssp_rate = USB_SSP_GEN_1x2; in dwc3_check_params()
1937 static struct extcon_dev *dwc3_get_extcon(struct dwc3 *dwc) in dwc3_get_extcon()
1939 struct device *dev = dwc->dev; in dwc3_get_extcon()
1953 if (device_property_read_string(dev, "linux,extcon-name", &name) == 0) in dwc3_get_extcon()
1957 * Check explicitly if "usb-role-switch" is used since in dwc3_get_extcon()
1963 device_property_read_bool(dev, "usb-role-switch")) in dwc3_get_extcon()
1970 * device (and "port" node) may be missing in case of "usb-role-switch" in dwc3_get_extcon()
1973 np_phy = of_parse_phandle(dev->of_node, "phys", 0); in dwc3_get_extcon()
1977 np_conn = of_graph_get_remote_node(np_phy, -1, -1); in dwc3_get_extcon()
1987 static int dwc3_get_clocks(struct dwc3 *dwc) in dwc3_get_clocks()
1989 struct device *dev = dwc->dev; in dwc3_get_clocks()
1991 if (!dev->of_node) in dwc3_get_clocks()
1996 * as required by the DT-binding. in dwc3_get_clocks()
2000 dwc->bus_clk = devm_clk_get_optional(dev, "bus_early"); in dwc3_get_clocks()
2001 if (IS_ERR(dwc->bus_clk)) { in dwc3_get_clocks()
2002 return dev_err_probe(dev, PTR_ERR(dwc->bus_clk), in dwc3_get_clocks()
2006 if (dwc->bus_clk == NULL) { in dwc3_get_clocks()
2007 dwc->bus_clk = devm_clk_get_optional(dev, "bus_clk"); in dwc3_get_clocks()
2008 if (IS_ERR(dwc->bus_clk)) { in dwc3_get_clocks()
2009 return dev_err_probe(dev, PTR_ERR(dwc->bus_clk), in dwc3_get_clocks()
2014 dwc->ref_clk = devm_clk_get_optional(dev, "ref"); in dwc3_get_clocks()
2015 if (IS_ERR(dwc->ref_clk)) { in dwc3_get_clocks()
2016 return dev_err_probe(dev, PTR_ERR(dwc->ref_clk), in dwc3_get_clocks()
2020 if (dwc->ref_clk == NULL) { in dwc3_get_clocks()
2021 dwc->ref_clk = devm_clk_get_optional(dev, "ref_clk"); in dwc3_get_clocks()
2022 if (IS_ERR(dwc->ref_clk)) { in dwc3_get_clocks()
2023 return dev_err_probe(dev, PTR_ERR(dwc->ref_clk), in dwc3_get_clocks()
2028 dwc->susp_clk = devm_clk_get_optional(dev, "suspend"); in dwc3_get_clocks()
2029 if (IS_ERR(dwc->susp_clk)) { in dwc3_get_clocks()
2030 return dev_err_probe(dev, PTR_ERR(dwc->susp_clk), in dwc3_get_clocks()
2034 if (dwc->susp_clk == NULL) { in dwc3_get_clocks()
2035 dwc->susp_clk = devm_clk_get_optional(dev, "suspend_clk"); in dwc3_get_clocks()
2036 if (IS_ERR(dwc->susp_clk)) { in dwc3_get_clocks()
2037 return dev_err_probe(dev, PTR_ERR(dwc->susp_clk), in dwc3_get_clocks()
2043 dwc->utmi_clk = devm_clk_get_optional(dev, "utmi"); in dwc3_get_clocks()
2044 if (IS_ERR(dwc->utmi_clk)) { in dwc3_get_clocks()
2045 return dev_err_probe(dev, PTR_ERR(dwc->utmi_clk), in dwc3_get_clocks()
2050 dwc->pipe_clk = devm_clk_get_optional(dev, "pipe"); in dwc3_get_clocks()
2051 if (IS_ERR(dwc->pipe_clk)) { in dwc3_get_clocks()
2052 return dev_err_probe(dev, PTR_ERR(dwc->pipe_clk), in dwc3_get_clocks()
2059 static int dwc3_get_num_ports(struct dwc3 *dwc) in dwc3_get_num_ports()
2070 base = ioremap(dwc->xhci_resources[0].start, in dwc3_get_num_ports()
2071 resource_size(&dwc->xhci_resources[0])); in dwc3_get_num_ports()
2073 return -ENOMEM; in dwc3_get_num_ports()
2087 dwc->num_usb3_ports += XHCI_EXT_PORT_COUNT(val); in dwc3_get_num_ports()
2089 dwc->num_usb2_ports += XHCI_EXT_PORT_COUNT(val); in dwc3_get_num_ports()
2091 dev_warn(dwc->dev, "unrecognized port major revision %d\n", in dwc3_get_num_ports()
2096 dev_dbg(dwc->dev, "hs-ports: %u ss-ports: %u\n", in dwc3_get_num_ports()
2097 dwc->num_usb2_ports, dwc->num_usb3_ports); in dwc3_get_num_ports()
2101 if (dwc->num_usb2_ports > DWC3_USB2_MAX_PORTS || in dwc3_get_num_ports()
2102 dwc->num_usb3_ports > DWC3_USB3_MAX_PORTS) in dwc3_get_num_ports()
2103 return -EINVAL; in dwc3_get_num_ports()
2110 struct device *dev = &pdev->dev; in dwc3_probe()
2114 struct dwc3 *dwc; in dwc3_probe()
2119 return -ENOMEM; in dwc3_probe()
2121 dwc->dev = dev; in dwc3_probe()
2126 return -ENODEV; in dwc3_probe()
2129 dwc->xhci_resources[0].start = res->start; in dwc3_probe()
2130 dwc->xhci_resources[0].end = dwc->xhci_resources[0].start + in dwc3_probe()
2132 dwc->xhci_resources[0].flags = res->flags; in dwc3_probe()
2133 dwc->xhci_resources[0].name = res->name; in dwc3_probe()
2137 * since it will be requested by the xhci-plat driver. in dwc3_probe()
2142 if (dev->of_node) { in dwc3_probe()
2143 struct device_node *parent = of_get_parent(dev->of_node); in dwc3_probe()
2145 if (of_device_is_compatible(parent, "realtek,rtd-dwc3")) { in dwc3_probe()
2146 dwc_res.start -= DWC3_GLOBALS_REGS_START; in dwc3_probe()
2157 dwc->regs = regs; in dwc3_probe()
2158 dwc->regs_size = resource_size(&dwc_res); in dwc3_probe()
2164 dwc->reset = devm_reset_control_array_get_optional_shared(dev); in dwc3_probe()
2165 if (IS_ERR(dwc->reset)) { in dwc3_probe()
2166 ret = PTR_ERR(dwc->reset); in dwc3_probe()
2174 ret = reset_control_deassert(dwc->reset); in dwc3_probe()
2183 dev_err(dwc->dev, "this is not a DesignWare USB3 DRD Core\n"); in dwc3_probe()
2184 ret = -ENODEV; in dwc3_probe()
2191 if (!dwc->sysdev_is_parent && in dwc3_probe()
2192 DWC3_GHWPARAMS0_AWIDTH(dwc->hwparams.hwparams0) == 64) { in dwc3_probe()
2193 ret = dma_set_mask_and_coherent(dwc->sysdev, DMA_BIT_MASK(64)); in dwc3_probe()
2199 * Currently only DWC3 controllers that are host-only capable in dwc3_probe()
2202 hw_mode = DWC3_GHWPARAMS0_MODE(dwc->hwparams.hwparams0); in dwc3_probe()
2208 dwc->num_usb2_ports = 1; in dwc3_probe()
2209 dwc->num_usb3_ports = 1; in dwc3_probe()
2212 spin_lock_init(&dwc->lock); in dwc3_probe()
2213 mutex_init(&dwc->mutex); in dwc3_probe()
2225 dev_err(dwc->dev, "failed to allocate event buffers\n"); in dwc3_probe()
2226 ret = -ENOMEM; in dwc3_probe()
2230 dwc->edev = dwc3_get_extcon(dwc); in dwc3_probe()
2231 if (IS_ERR(dwc->edev)) { in dwc3_probe()
2232 ret = dev_err_probe(dwc->dev, PTR_ERR(dwc->edev), "failed to get extcon\n"); in dwc3_probe()
2276 reset_control_assert(dwc->reset); in dwc3_probe()
2278 if (dwc->usb_psy) in dwc3_probe()
2279 power_supply_put(dwc->usb_psy); in dwc3_probe()
2286 struct dwc3 *dwc = platform_get_drvdata(pdev); in dwc3_remove()
2288 pm_runtime_get_sync(&pdev->dev); in dwc3_remove()
2296 pm_runtime_allow(&pdev->dev); in dwc3_remove()
2297 pm_runtime_disable(&pdev->dev); in dwc3_remove()
2298 pm_runtime_dont_use_autosuspend(&pdev->dev); in dwc3_remove()
2299 pm_runtime_put_noidle(&pdev->dev); in dwc3_remove()
2305 pm_runtime_set_suspended(&pdev->dev); in dwc3_remove()
2309 if (dwc->usb_psy) in dwc3_remove()
2310 power_supply_put(dwc->usb_psy); in dwc3_remove()
2314 static int dwc3_core_init_for_resume(struct dwc3 *dwc) in dwc3_core_init_for_resume()
2318 ret = reset_control_deassert(dwc->reset); in dwc3_core_init_for_resume()
2335 reset_control_assert(dwc->reset); in dwc3_core_init_for_resume()
2340 static int dwc3_suspend_common(struct dwc3 *dwc, pm_message_t msg) in dwc3_suspend_common()
2345 if (!pm_runtime_suspended(dwc->dev) && !PMSG_IS_AUTO(msg)) { in dwc3_suspend_common()
2346 dwc->susphy_state = (dwc3_readl(dwc->regs, DWC3_GUSB2PHYCFG(0)) & in dwc3_suspend_common()
2348 (dwc3_readl(dwc->regs, DWC3_GUSB3PIPECTL(0)) & in dwc3_suspend_common()
2354 if (!dwc->susphy_state) in dwc3_suspend_common()
2358 switch (dwc->current_dr_role) { in dwc3_suspend_common()
2360 if (pm_runtime_suspended(dwc->dev)) in dwc3_suspend_common()
2363 synchronize_irq(dwc->irq_gadget); in dwc3_suspend_common()
2367 if (!PMSG_IS_AUTO(msg) && !device_may_wakeup(dwc->dev)) { in dwc3_suspend_common()
2373 if (dwc->dis_u2_susphy_quirk || in dwc3_suspend_common()
2374 dwc->dis_enblslpm_quirk) { in dwc3_suspend_common()
2375 for (i = 0; i < dwc->num_usb2_ports; i++) { in dwc3_suspend_common()
2376 reg = dwc3_readl(dwc->regs, DWC3_GUSB2PHYCFG(i)); in dwc3_suspend_common()
2379 dwc3_writel(dwc->regs, DWC3_GUSB2PHYCFG(i), reg); in dwc3_suspend_common()
2386 for (i = 0; i < dwc->num_usb2_ports; i++) in dwc3_suspend_common()
2387 phy_pm_runtime_put_sync(dwc->usb2_generic_phy[i]); in dwc3_suspend_common()
2388 for (i = 0; i < dwc->num_usb3_ports; i++) in dwc3_suspend_common()
2389 phy_pm_runtime_put_sync(dwc->usb3_generic_phy[i]); in dwc3_suspend_common()
2396 if (dwc->current_otg_role == DWC3_OTG_ROLE_DEVICE) { in dwc3_suspend_common()
2398 synchronize_irq(dwc->irq_gadget); in dwc3_suspend_common()
2412 static int dwc3_resume_common(struct dwc3 *dwc, pm_message_t msg) in dwc3_resume_common()
2418 switch (dwc->current_dr_role) { in dwc3_resume_common()
2428 if (!PMSG_IS_AUTO(msg) && !device_may_wakeup(dwc->dev)) { in dwc3_resume_common()
2436 for (i = 0; i < dwc->num_usb2_ports; i++) { in dwc3_resume_common()
2437 reg = dwc3_readl(dwc->regs, DWC3_GUSB2PHYCFG(i)); in dwc3_resume_common()
2438 if (dwc->dis_u2_susphy_quirk) in dwc3_resume_common()
2441 if (dwc->dis_enblslpm_quirk) in dwc3_resume_common()
2444 dwc3_writel(dwc->regs, DWC3_GUSB2PHYCFG(i), reg); in dwc3_resume_common()
2447 for (i = 0; i < dwc->num_usb2_ports; i++) in dwc3_resume_common()
2448 phy_pm_runtime_get_sync(dwc->usb2_generic_phy[i]); in dwc3_resume_common()
2449 for (i = 0; i < dwc->num_usb3_ports; i++) in dwc3_resume_common()
2450 phy_pm_runtime_get_sync(dwc->usb3_generic_phy[i]); in dwc3_resume_common()
2461 dwc3_set_prtcap(dwc, dwc->current_dr_role); in dwc3_resume_common()
2464 if (dwc->current_otg_role == DWC3_OTG_ROLE_HOST) { in dwc3_resume_common()
2466 } else if (dwc->current_otg_role == DWC3_OTG_ROLE_DEVICE) { in dwc3_resume_common()
2478 dwc3_enable_susphy(dwc, dwc->susphy_state); in dwc3_resume_common()
2484 static int dwc3_runtime_checks(struct dwc3 *dwc) in dwc3_runtime_checks()
2486 switch (dwc->current_dr_role) { in dwc3_runtime_checks()
2488 if (dwc->connected) in dwc3_runtime_checks()
2489 return -EBUSY; in dwc3_runtime_checks()
2502 struct dwc3 *dwc = dev_get_drvdata(dev); in dwc3_runtime_suspend()
2506 return -EBUSY; in dwc3_runtime_suspend()
2517 struct dwc3 *dwc = dev_get_drvdata(dev); in dwc3_runtime_resume()
2524 switch (dwc->current_dr_role) { in dwc3_runtime_resume()
2526 if (dwc->pending_events) { in dwc3_runtime_resume()
2527 pm_runtime_put(dwc->dev); in dwc3_runtime_resume()
2528 dwc->pending_events = false; in dwc3_runtime_resume()
2529 enable_irq(dwc->irq_gadget); in dwc3_runtime_resume()
2545 struct dwc3 *dwc = dev_get_drvdata(dev); in dwc3_runtime_idle()
2547 switch (dwc->current_dr_role) { in dwc3_runtime_idle()
2550 return -EBUSY; in dwc3_runtime_idle()
2568 struct dwc3 *dwc = dev_get_drvdata(dev); in dwc3_suspend()
2582 struct dwc3 *dwc = dev_get_drvdata(dev); in dwc3_resume()
2601 struct dwc3 *dwc = dev_get_drvdata(dev); in dwc3_complete()
2604 if (dwc->current_dr_role == DWC3_GCTL_PRTCAP_HOST && in dwc3_complete()
2605 dwc->dis_split_quirk) { in dwc3_complete()
2606 reg = dwc3_readl(dwc->regs, DWC3_GUCTL3); in dwc3_complete()
2608 dwc3_writel(dwc->regs, DWC3_GUCTL3, reg); in dwc3_complete()
2631 .compatible = "snps,dwc3"
2634 .compatible = "synopsys,dwc3"
2656 .name = "dwc3",
2665 MODULE_ALIAS("platform:dwc3");