xref: /rk3399_rockchip-uboot/drivers/usb/host/ehci-zynq.c (revision a821c4af79e4f5ce9b629b20473863397bbe9b10)
1eb8c54bfSMichal Simek /*
2eb8c54bfSMichal Simek  * (C) Copyright 2014, Xilinx, Inc
3eb8c54bfSMichal Simek  *
4eb8c54bfSMichal Simek  * USB Low level initialization(Specific to zynq)
5eb8c54bfSMichal Simek  *
6eb8c54bfSMichal Simek  * SPDX-License-Identifier:	GPL-2.0+
7eb8c54bfSMichal Simek  */
8eb8c54bfSMichal Simek 
9eb8c54bfSMichal Simek #include <common.h>
10dec49e86SSimon Glass #include <dm.h>
11dec49e86SSimon Glass #include <usb.h>
12eb8c54bfSMichal Simek #include <asm/arch/hardware.h>
13eb8c54bfSMichal Simek #include <asm/arch/sys_proto.h>
14eb8c54bfSMichal Simek #include <asm/io.h>
15e162c6b1SMateusz Kulikowski #include <usb/ehci-ci.h>
16eb8c54bfSMichal Simek #include <usb/ulpi.h>
17eb8c54bfSMichal Simek 
18eb8c54bfSMichal Simek #include "ehci.h"
19eb8c54bfSMichal Simek 
20dec49e86SSimon Glass struct zynq_ehci_priv {
21dec49e86SSimon Glass 	struct ehci_ctrl ehcictrl;
22eb8c54bfSMichal Simek 	struct usb_ehci *ehci;
23dec49e86SSimon Glass };
24dec49e86SSimon Glass 
ehci_zynq_ofdata_to_platdata(struct udevice * dev)25dec49e86SSimon Glass static int ehci_zynq_ofdata_to_platdata(struct udevice *dev)
26dec49e86SSimon Glass {
27dec49e86SSimon Glass 	struct zynq_ehci_priv *priv = dev_get_priv(dev);
28dec49e86SSimon Glass 
29*a821c4afSSimon Glass 	priv->ehci = (struct usb_ehci *)devfdt_get_addr_ptr(dev);
30dec49e86SSimon Glass 	if (!priv->ehci)
31dec49e86SSimon Glass 		return -EINVAL;
32dec49e86SSimon Glass 
33dec49e86SSimon Glass 	return 0;
34dec49e86SSimon Glass }
35dec49e86SSimon Glass 
ehci_zynq_probe(struct udevice * dev)36dec49e86SSimon Glass static int ehci_zynq_probe(struct udevice *dev)
37dec49e86SSimon Glass {
38dec49e86SSimon Glass 	struct usb_platdata *plat = dev_get_platdata(dev);
39dec49e86SSimon Glass 	struct zynq_ehci_priv *priv = dev_get_priv(dev);
40dec49e86SSimon Glass 	struct ehci_hccr *hccr;
41dec49e86SSimon Glass 	struct ehci_hcor *hcor;
42eb8c54bfSMichal Simek 	struct ulpi_viewport ulpi_vp;
43eb8c54bfSMichal Simek 	/* Used for writing the ULPI data address */
44eb8c54bfSMichal Simek 	struct ulpi_regs *ulpi = (struct ulpi_regs *)0;
45dec49e86SSimon Glass 	int ret;
46eb8c54bfSMichal Simek 
47dec49e86SSimon Glass 	hccr = (struct ehci_hccr *)((uint32_t)&priv->ehci->caplength);
48dec49e86SSimon Glass 	hcor = (struct ehci_hcor *)((uint32_t) hccr +
49dec49e86SSimon Glass 			HC_LENGTH(ehci_readl(&hccr->cr_capbase)));
50eb8c54bfSMichal Simek 
51dec49e86SSimon Glass 	ulpi_vp.viewport_addr = (u32)&priv->ehci->ulpi_viewpoint;
52eb8c54bfSMichal Simek 	ulpi_vp.port_num = 0;
53eb8c54bfSMichal Simek 
54eb8c54bfSMichal Simek 	ret = ulpi_init(&ulpi_vp);
55eb8c54bfSMichal Simek 	if (ret) {
56eb8c54bfSMichal Simek 		puts("zynq ULPI viewport init failed\n");
57eb8c54bfSMichal Simek 		return -1;
58eb8c54bfSMichal Simek 	}
59eb8c54bfSMichal Simek 
60eb8c54bfSMichal Simek 	/* ULPI set flags */
61eb8c54bfSMichal Simek 	ulpi_write(&ulpi_vp, &ulpi->otg_ctrl,
62eb8c54bfSMichal Simek 		   ULPI_OTG_DP_PULLDOWN | ULPI_OTG_DM_PULLDOWN |
63eb8c54bfSMichal Simek 		   ULPI_OTG_EXTVBUSIND);
64eb8c54bfSMichal Simek 	ulpi_write(&ulpi_vp, &ulpi->function_ctrl,
65eb8c54bfSMichal Simek 		   ULPI_FC_FULL_SPEED | ULPI_FC_OPMODE_NORMAL |
66eb8c54bfSMichal Simek 		   ULPI_FC_SUSPENDM);
67eb8c54bfSMichal Simek 	ulpi_write(&ulpi_vp, &ulpi->iface_ctrl, 0);
68eb8c54bfSMichal Simek 
69eb8c54bfSMichal Simek 	/* Set VBus */
70eb8c54bfSMichal Simek 	ulpi_write(&ulpi_vp, &ulpi->otg_ctrl_set,
71eb8c54bfSMichal Simek 		   ULPI_OTG_DRVVBUS | ULPI_OTG_DRVVBUS_EXT);
72eb8c54bfSMichal Simek 
73dec49e86SSimon Glass 	return ehci_register(dev, hccr, hcor, NULL, 0, plat->init_type);
74eb8c54bfSMichal Simek }
75eb8c54bfSMichal Simek 
76dec49e86SSimon Glass static const struct udevice_id ehci_zynq_ids[] = {
77dec49e86SSimon Glass 	{ .compatible = "xlnx,zynq-usb-2.20a" },
78dec49e86SSimon Glass 	{ }
79dec49e86SSimon Glass };
80dec49e86SSimon Glass 
81dec49e86SSimon Glass U_BOOT_DRIVER(ehci_zynq) = {
82dec49e86SSimon Glass 	.name	= "ehci_zynq",
83dec49e86SSimon Glass 	.id	= UCLASS_USB,
84dec49e86SSimon Glass 	.of_match = ehci_zynq_ids,
85dec49e86SSimon Glass 	.ofdata_to_platdata = ehci_zynq_ofdata_to_platdata,
86dec49e86SSimon Glass 	.probe = ehci_zynq_probe,
8740527342SMasahiro Yamada 	.remove = ehci_deregister,
88dec49e86SSimon Glass 	.ops	= &ehci_usb_ops,
89dec49e86SSimon Glass 	.platdata_auto_alloc_size = sizeof(struct usb_platdata),
90dec49e86SSimon Glass 	.priv_auto_alloc_size = sizeof(struct zynq_ehci_priv),
91dec49e86SSimon Glass 	.flags	= DM_FLAG_ALLOC_PRIV_DMA,
92dec49e86SSimon Glass };
93