blob: 1338b1021c644facb8243cd1caba0b0fb32f097c [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Stefan Roese07faf112016-07-14 11:39:20 +02002/*
3 * Copyright (C) 2015 Marvell International Ltd.
4 *
5 * MVEBU USB HOST xHCI Controller
Stefan Roese07faf112016-07-14 11:39:20 +02006 */
7
Stefan Roese07faf112016-07-14 11:39:20 +02008#include <dm.h>
9#include <fdtdec.h>
Simon Glass0f2af882020-05-10 11:40:05 -060010#include <log.h>
Stefan Roese07faf112016-07-14 11:39:20 +020011#include <usb.h>
Konstantin Porotchkin1b5ed4d2017-02-12 11:10:30 +020012#include <power/regulator.h>
Stefan Roese07faf112016-07-14 11:39:20 +020013#include <asm/gpio.h>
14
Jean-Jacques Hiblotad4142b2019-09-11 11:33:46 +020015#include <usb/xhci.h>
Stefan Roese07faf112016-07-14 11:39:20 +020016
Simon Glassb75b15b2020-12-03 16:55:23 -070017struct mvebu_xhci_plat {
Stefan Roese07faf112016-07-14 11:39:20 +020018 fdt_addr_t hcd_base;
19};
20
21/**
22 * Contains pointers to register base addresses
23 * for the usb controller.
24 */
25struct mvebu_xhci {
26 struct xhci_ctrl ctrl; /* Needs to come first in this struct! */
Simon Glassb75b15b2020-12-03 16:55:23 -070027 struct usb_plat usb_plat;
Stefan Roese07faf112016-07-14 11:39:20 +020028 struct xhci_hccr *hcd;
29};
30
31/*
32 * Dummy implementation that can be overwritten by a board
33 * specific function
34 */
Jon Nettletona81f47c2017-11-06 10:33:19 +020035__weak int board_xhci_enable(fdt_addr_t base)
Stefan Roese07faf112016-07-14 11:39:20 +020036{
37 return 0;
38}
39
40static int xhci_usb_probe(struct udevice *dev)
41{
Simon Glassb75b15b2020-12-03 16:55:23 -070042 struct mvebu_xhci_plat *plat = dev_get_plat(dev);
Stefan Roese07faf112016-07-14 11:39:20 +020043 struct mvebu_xhci *ctx = dev_get_priv(dev);
44 struct xhci_hcor *hcor;
Konstantin Porotchkin1b5ed4d2017-02-12 11:10:30 +020045 int len, ret;
46 struct udevice *regulator;
Stefan Roese07faf112016-07-14 11:39:20 +020047
48 ctx->hcd = (struct xhci_hccr *)plat->hcd_base;
49 len = HC_LENGTH(xhci_readl(&ctx->hcd->cr_capbase));
50 hcor = (struct xhci_hcor *)((uintptr_t)ctx->hcd + len);
51
Konstantin Porotchkin1b5ed4d2017-02-12 11:10:30 +020052 ret = device_get_supply_regulator(dev, "vbus-supply", &regulator);
53 if (!ret) {
54 ret = regulator_set_enable(regulator, true);
55 if (ret) {
56 printf("Failed to turn ON the VBUS regulator\n");
57 return ret;
58 }
59 }
60
Stefan Roese07faf112016-07-14 11:39:20 +020061 /* Enable USB xHCI (VBUS, reset etc) in board specific code */
Jon Nettletona81f47c2017-11-06 10:33:19 +020062 board_xhci_enable(devfdt_get_addr_index(dev, 1));
Stefan Roese07faf112016-07-14 11:39:20 +020063
64 return xhci_register(dev, ctx->hcd, hcor);
65}
66
Simon Glassaad29ae2020-12-03 16:55:21 -070067static int xhci_usb_of_to_plat(struct udevice *dev)
Stefan Roese07faf112016-07-14 11:39:20 +020068{
Simon Glassb75b15b2020-12-03 16:55:23 -070069 struct mvebu_xhci_plat *plat = dev_get_plat(dev);
Stefan Roese07faf112016-07-14 11:39:20 +020070
71 /*
72 * Get the base address for XHCI controller from the device node
73 */
Masahiro Yamadaa89b4de2020-07-17 14:36:48 +090074 plat->hcd_base = dev_read_addr(dev);
Stefan Roese07faf112016-07-14 11:39:20 +020075 if (plat->hcd_base == FDT_ADDR_T_NONE) {
76 debug("Can't get the XHCI register base address\n");
77 return -ENXIO;
78 }
79
80 return 0;
81}
82
83static const struct udevice_id xhci_usb_ids[] = {
84 { .compatible = "marvell,armada3700-xhci" },
Jon Nettletona81f47c2017-11-06 10:33:19 +020085 { .compatible = "marvell,armada-380-xhci" },
Stefan Roese755e3f52016-08-31 06:48:56 +020086 { .compatible = "marvell,armada-8k-xhci" },
Stefan Roese07faf112016-07-14 11:39:20 +020087 { }
88};
89
90U_BOOT_DRIVER(usb_xhci) = {
91 .name = "xhci_mvebu",
92 .id = UCLASS_USB,
93 .of_match = xhci_usb_ids,
Simon Glassaad29ae2020-12-03 16:55:21 -070094 .of_to_plat = xhci_usb_of_to_plat,
Stefan Roese07faf112016-07-14 11:39:20 +020095 .probe = xhci_usb_probe,
Masahiro Yamada91b09aa2016-10-14 10:30:01 +090096 .remove = xhci_deregister,
Stefan Roese07faf112016-07-14 11:39:20 +020097 .ops = &xhci_usb_ops,
Simon Glassb75b15b2020-12-03 16:55:23 -070098 .plat_auto = sizeof(struct mvebu_xhci_plat),
Simon Glass8a2b47f2020-12-03 16:55:17 -070099 .priv_auto = sizeof(struct mvebu_xhci),
Stefan Roese07faf112016-07-14 11:39:20 +0200100 .flags = DM_FLAG_ALLOC_PRIV_DMA,
101};