blob: b5ec2969188bb5f39f0887e5d4854391f3d7d8d9 [file] [log] [blame]
Masahiro Yamada21b88902014-11-07 18:48:33 +09001/*
2 * Copyright (C) 2014 Panasonic Corporation
3 * Author: Masahiro Yamada <yamada.m@jp.panasonic.com>
4 *
5 * SPDX-License-Identifier: GPL-2.0+
6 */
7
8#include <common.h>
Masahiro Yamada9a724622014-11-26 18:34:01 +09009#include <linux/err.h>
Masahiro Yamadab2584c52015-02-27 02:26:55 +090010#include <asm/io.h>
Masahiro Yamada21b88902014-11-07 18:48:33 +090011#include <usb.h>
Masahiro Yamadab2584c52015-02-27 02:26:55 +090012#include <mach/mio-regs.h>
Masahiro Yamada9a1dff52015-02-27 02:26:54 +090013#include <fdtdec.h>
Masahiro Yamada21b88902014-11-07 18:48:33 +090014#include "ehci.h"
15
Masahiro Yamada9a724622014-11-26 18:34:01 +090016DECLARE_GLOBAL_DATA_PTR;
17
18#define FDT gd->fdt_blob
19#define COMPAT "panasonic,uniphier-ehci"
20
21static int get_uniphier_ehci_base(int index, struct ehci_hccr **base)
22{
23 int offset;
24
25 for (offset = fdt_node_offset_by_compatible(FDT, 0, COMPAT);
26 offset >= 0;
27 offset = fdt_node_offset_by_compatible(FDT, offset, COMPAT)) {
28 if (index == 0) {
29 *base = (struct ehci_hccr *)
30 fdtdec_get_addr(FDT, offset, "reg");
31 return 0;
32 }
33 index--;
34 }
35
36 return -ENODEV; /* not found */
37}
Masahiro Yamada9a724622014-11-26 18:34:01 +090038
Masahiro Yamadab2584c52015-02-27 02:26:55 +090039static void uniphier_ehci_reset(int index, int on)
40{
41 u32 tmp;
42
43 tmp = readl(MIO_USB_RSTCTRL(index));
44 if (on)
45 tmp &= ~MIO_USB_RSTCTRL_XRST;
46 else
47 tmp |= MIO_USB_RSTCTRL_XRST;
48 writel(tmp, MIO_USB_RSTCTRL(index));
49}
50
Masahiro Yamada21b88902014-11-07 18:48:33 +090051int ehci_hcd_init(int index, enum usb_init_type init, struct ehci_hccr **hccr,
52 struct ehci_hcor **hcor)
53{
Masahiro Yamada9a724622014-11-26 18:34:01 +090054 int ret;
Masahiro Yamada21b88902014-11-07 18:48:33 +090055 struct ehci_hccr *cr;
56 struct ehci_hcor *or;
57
58 uniphier_ehci_reset(index, 0);
59
Masahiro Yamada9a724622014-11-26 18:34:01 +090060 ret = get_uniphier_ehci_base(index, &cr);
61 if (ret < 0)
62 return ret;
Masahiro Yamada21b88902014-11-07 18:48:33 +090063 or = (void *)cr + HC_LENGTH(ehci_readl(&cr->cr_capbase));
64
65 *hccr = cr;
66 *hcor = or;
67
68 return 0;
69}
70
71int ehci_hcd_stop(int index)
72{
73 uniphier_ehci_reset(index, 1);
74
75 return 0;
76}