2
0
mirror of git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git synced 2025-09-04 20:19:47 +08:00

usb: xhci-mtk: supports remote wakeup for mt2712 with two xHCI IPs

The old way of usb wakeup only supports platform with single xHCI IP,
such as mt8173, but mt2712 has two xHCI IPs, so rebuild its flow and
supports the new glue layer of usb wakeup on mt2712 which is different
from mt8173.
Due to there is a hardware bug with the LINE STATE wakeup mode on
mt8173 which causes wakeup failure by low speed devices, and also
because IP SLEEP mode can cover all functions of LINE STATE mode,
it is unused in fact, and will not support it later, so remove it at
the same time.

Signed-off-by: Chunfeng Yun <chunfeng.yun@mediatek.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
Chunfeng Yun 2018-01-03 16:53:20 +08:00 committed by Greg Kroah-Hartman
parent 99ca1f6f3b
commit a2ecc4df9f
2 changed files with 64 additions and 117 deletions

View File

@ -57,26 +57,21 @@
/* u2_phy_pll register */ /* u2_phy_pll register */
#define CTRL_U2_FORCE_PLL_STB BIT(28) #define CTRL_U2_FORCE_PLL_STB BIT(28)
#define PERI_WK_CTRL0 0x400 /* usb remote wakeup registers in syscon */
#define UWK_CTR0_0P_LS_PE BIT(8) /* posedge */ /* mt8173 etc */
#define UWK_CTR0_0P_LS_NE BIT(7) /* negedge for 0p linestate*/ #define PERI_WK_CTRL1 0x4
#define UWK_CTL1_1P_LS_C(x) (((x) & 0xf) << 1) #define WC1_IS_C(x) (((x) & 0xf) << 26) /* cycle debounce */
#define UWK_CTL1_1P_LS_E BIT(0) #define WC1_IS_EN BIT(25)
#define WC1_IS_P BIT(6) /* polarity for ip sleep */
#define PERI_WK_CTRL1 0x404 /* mt2712 etc */
#define UWK_CTL1_IS_C(x) (((x) & 0xf) << 26) #define PERI_SSUSB_SPM_CTRL 0x0
#define UWK_CTL1_IS_E BIT(25) #define SSC_IP_SLEEP_EN BIT(4)
#define UWK_CTL1_0P_LS_C(x) (((x) & 0xf) << 21) #define SSC_SPM_INT_EN BIT(1)
#define UWK_CTL1_0P_LS_E BIT(20)
#define UWK_CTL1_IDDIG_C(x) (((x) & 0xf) << 11) /* cycle debounce */
#define UWK_CTL1_IDDIG_E BIT(10) /* enable debounce */
#define UWK_CTL1_IDDIG_P BIT(9) /* polarity */
#define UWK_CTL1_0P_LS_P BIT(7)
#define UWK_CTL1_IS_P BIT(6) /* polarity for ip sleep */
enum ssusb_wakeup_src { enum ssusb_uwk_vers {
SSUSB_WK_IP_SLEEP = 1, SSUSB_UWK_V1 = 1,
SSUSB_WK_LINE_STATE = 2, SSUSB_UWK_V2,
}; };
static int xhci_mtk_host_enable(struct xhci_hcd_mtk *mtk) static int xhci_mtk_host_enable(struct xhci_hcd_mtk *mtk)
@ -296,112 +291,58 @@ static void xhci_mtk_clks_disable(struct xhci_hcd_mtk *mtk)
} }
/* only clocks can be turn off for ip-sleep wakeup mode */ /* only clocks can be turn off for ip-sleep wakeup mode */
static void usb_wakeup_ip_sleep_en(struct xhci_hcd_mtk *mtk) static void usb_wakeup_ip_sleep_set(struct xhci_hcd_mtk *mtk, bool enable)
{ {
u32 tmp; u32 reg, msk, val;
struct regmap *pericfg = mtk->pericfg;
regmap_read(pericfg, PERI_WK_CTRL1, &tmp); switch (mtk->uwk_vers) {
tmp &= ~UWK_CTL1_IS_P; case SSUSB_UWK_V1:
tmp &= ~(UWK_CTL1_IS_C(0xf)); reg = mtk->uwk_reg_base + PERI_WK_CTRL1;
tmp |= UWK_CTL1_IS_C(0x8); msk = WC1_IS_EN | WC1_IS_C(0xf) | WC1_IS_P;
regmap_write(pericfg, PERI_WK_CTRL1, tmp); val = enable ? (WC1_IS_EN | WC1_IS_C(0x8)) : 0;
regmap_write(pericfg, PERI_WK_CTRL1, tmp | UWK_CTL1_IS_E); break;
case SSUSB_UWK_V2:
regmap_read(pericfg, PERI_WK_CTRL1, &tmp); reg = mtk->uwk_reg_base + PERI_SSUSB_SPM_CTRL;
dev_dbg(mtk->dev, "%s(): WK_CTRL1[P6,E25,C26:29]=%#x\n", msk = SSC_IP_SLEEP_EN | SSC_SPM_INT_EN;
__func__, tmp); val = enable ? msk : 0;
} break;
default:
static void usb_wakeup_ip_sleep_dis(struct xhci_hcd_mtk *mtk) return;
{ };
u32 tmp; regmap_update_bits(mtk->uwk, reg, msk, val);
regmap_read(mtk->pericfg, PERI_WK_CTRL1, &tmp);
tmp &= ~UWK_CTL1_IS_E;
regmap_write(mtk->pericfg, PERI_WK_CTRL1, tmp);
}
/*
* for line-state wakeup mode, phy's power should not power-down
* and only support cable plug in/out
*/
static void usb_wakeup_line_state_en(struct xhci_hcd_mtk *mtk)
{
u32 tmp;
struct regmap *pericfg = mtk->pericfg;
/* line-state of u2-port0 */
regmap_read(pericfg, PERI_WK_CTRL1, &tmp);
tmp &= ~UWK_CTL1_0P_LS_P;
tmp &= ~(UWK_CTL1_0P_LS_C(0xf));
tmp |= UWK_CTL1_0P_LS_C(0x8);
regmap_write(pericfg, PERI_WK_CTRL1, tmp);
regmap_read(pericfg, PERI_WK_CTRL1, &tmp);
regmap_write(pericfg, PERI_WK_CTRL1, tmp | UWK_CTL1_0P_LS_E);
/* line-state of u2-port1 */
regmap_read(pericfg, PERI_WK_CTRL0, &tmp);
tmp &= ~(UWK_CTL1_1P_LS_C(0xf));
tmp |= UWK_CTL1_1P_LS_C(0x8);
regmap_write(pericfg, PERI_WK_CTRL0, tmp);
regmap_write(pericfg, PERI_WK_CTRL0, tmp | UWK_CTL1_1P_LS_E);
}
static void usb_wakeup_line_state_dis(struct xhci_hcd_mtk *mtk)
{
u32 tmp;
struct regmap *pericfg = mtk->pericfg;
/* line-state of u2-port0 */
regmap_read(pericfg, PERI_WK_CTRL1, &tmp);
tmp &= ~UWK_CTL1_0P_LS_E;
regmap_write(pericfg, PERI_WK_CTRL1, tmp);
/* line-state of u2-port1 */
regmap_read(pericfg, PERI_WK_CTRL0, &tmp);
tmp &= ~UWK_CTL1_1P_LS_E;
regmap_write(pericfg, PERI_WK_CTRL0, tmp);
}
static void usb_wakeup_enable(struct xhci_hcd_mtk *mtk)
{
if (mtk->wakeup_src == SSUSB_WK_IP_SLEEP)
usb_wakeup_ip_sleep_en(mtk);
else if (mtk->wakeup_src == SSUSB_WK_LINE_STATE)
usb_wakeup_line_state_en(mtk);
}
static void usb_wakeup_disable(struct xhci_hcd_mtk *mtk)
{
if (mtk->wakeup_src == SSUSB_WK_IP_SLEEP)
usb_wakeup_ip_sleep_dis(mtk);
else if (mtk->wakeup_src == SSUSB_WK_LINE_STATE)
usb_wakeup_line_state_dis(mtk);
} }
static int usb_wakeup_of_property_parse(struct xhci_hcd_mtk *mtk, static int usb_wakeup_of_property_parse(struct xhci_hcd_mtk *mtk,
struct device_node *dn) struct device_node *dn)
{ {
struct device *dev = mtk->dev; struct of_phandle_args args;
int ret;
/* /* Wakeup function is optional */
* wakeup function is optional, so it is not an error if this property mtk->uwk_en = of_property_read_bool(dn, "wakeup-source");
* does not exist, and in such case, no need to get relative if (!mtk->uwk_en)
* properties anymore.
*/
of_property_read_u32(dn, "mediatek,wakeup-src", &mtk->wakeup_src);
if (!mtk->wakeup_src)
return 0; return 0;
mtk->pericfg = syscon_regmap_lookup_by_phandle(dn, ret = of_parse_phandle_with_fixed_args(dn,
"mediatek,syscon-wakeup"); "mediatek,syscon-wakeup", 2, 0, &args);
if (IS_ERR(mtk->pericfg)) { if (ret)
dev_err(dev, "fail to get pericfg regs\n"); return ret;
return PTR_ERR(mtk->pericfg);
}
return 0; mtk->uwk_reg_base = args.args[0];
mtk->uwk_vers = args.args[1];
mtk->uwk = syscon_node_to_regmap(args.np);
of_node_put(args.np);
dev_info(mtk->dev, "uwk - reg:0x%x, version:%d\n",
mtk->uwk_reg_base, mtk->uwk_vers);
return PTR_ERR_OR_ZERO(mtk->uwk);
}
static void usb_wakeup_set(struct xhci_hcd_mtk *mtk, bool enable)
{
if (mtk->uwk_en)
usb_wakeup_ip_sleep_set(mtk, enable);
} }
static int xhci_mtk_setup(struct usb_hcd *hcd); static int xhci_mtk_setup(struct usb_hcd *hcd);
@ -583,8 +524,10 @@ static int xhci_mtk_probe(struct platform_device *pdev)
&mtk->u3p_dis_msk); &mtk->u3p_dis_msk);
ret = usb_wakeup_of_property_parse(mtk, node); ret = usb_wakeup_of_property_parse(mtk, node);
if (ret) if (ret) {
dev_err(dev, "failed to parse uwk property\n");
return ret; return ret;
}
mtk->num_phys = of_count_phandle_with_args(node, mtk->num_phys = of_count_phandle_with_args(node,
"phys", "#phy-cells"); "phys", "#phy-cells");
@ -777,7 +720,7 @@ static int __maybe_unused xhci_mtk_suspend(struct device *dev)
xhci_mtk_host_disable(mtk); xhci_mtk_host_disable(mtk);
xhci_mtk_phy_power_off(mtk); xhci_mtk_phy_power_off(mtk);
xhci_mtk_clks_disable(mtk); xhci_mtk_clks_disable(mtk);
usb_wakeup_enable(mtk); usb_wakeup_set(mtk, true);
return 0; return 0;
} }
@ -787,7 +730,7 @@ static int __maybe_unused xhci_mtk_resume(struct device *dev)
struct usb_hcd *hcd = mtk->hcd; struct usb_hcd *hcd = mtk->hcd;
struct xhci_hcd *xhci = hcd_to_xhci(hcd); struct xhci_hcd *xhci = hcd_to_xhci(hcd);
usb_wakeup_disable(mtk); usb_wakeup_set(mtk, false);
xhci_mtk_clks_enable(mtk); xhci_mtk_clks_enable(mtk);
xhci_mtk_phy_power_on(mtk); xhci_mtk_phy_power_on(mtk);
xhci_mtk_host_enable(mtk); xhci_mtk_host_enable(mtk);

View File

@ -122,8 +122,12 @@ struct xhci_hcd_mtk {
struct regmap *pericfg; struct regmap *pericfg;
struct phy **phys; struct phy **phys;
int num_phys; int num_phys;
int wakeup_src;
bool lpm_support; bool lpm_support;
/* usb remote wakeup */
bool uwk_en;
struct regmap *uwk;
u32 uwk_reg_base;
u32 uwk_vers;
}; };
static inline struct xhci_hcd_mtk *hcd_to_mtk(struct usb_hcd *hcd) static inline struct xhci_hcd_mtk *hcd_to_mtk(struct usb_hcd *hcd)