/* * (C) Copyright 2009 Stefan Roese <sr@denx.de>, DENX Software Engineering * * Original Author Guenter Gebhardt * Copyright (C) 2006 Micronas GmbH * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA */ #include <common.h> #include "vct.h" int vct_ehci_hcd_init(u32 *hccr, u32 *hcor) { int retval; u32 val; u32 addr; dcgu_set_reset_switch(DCGU_HW_MODULE_USB_24, DCGU_SWITCH_ON); dcgu_set_reset_switch(DCGU_HW_MODULE_USB_60, DCGU_SWITCH_ON); dcgu_set_clk_switch(DCGU_HW_MODULE_USB_24, DCGU_SWITCH_ON); dcgu_set_clk_switch(DCGU_HW_MODULE_USB_PLL, DCGU_SWITCH_ON); dcgu_set_reset_switch(DCGU_HW_MODULE_USB_24, DCGU_SWITCH_OFF); /* Wait until (DCGU_USBPHY_STAT == 7) */ addr = DCGU_USBPHY_STAT(DCGU_BASE); val = reg_read(addr); while (val != 7) val = reg_read(addr); dcgu_set_clk_switch(DCGU_HW_MODULE_USB_60, DCGU_SWITCH_ON); dcgu_set_reset_switch(DCGU_HW_MODULE_USB_60, DCGU_SWITCH_OFF); retval = scc_reset(SCC_USB_RW, 0); if (retval) { printf("scc_reset(SCC_USB_RW, 0) returned: 0x%x\n", retval); return retval; } else { retval = scc_reset(SCC_CPU1_SPDMA_RW, 0); if (retval) { printf("scc_reset(SCC_CPU1_SPDMA_RW, 0) returned: 0x%x\n", retval); return retval; } } if (!retval) { /* * For the AGU bypass, where the SCC client provides full * physical address */ scc_set_usb_address_generation_mode(1); scc_setup_dma(SCC_USB_RW, BCU_USB_BUFFER_1, DMA_LINEAR, USE_NO_FH, DMA_READ, 0); scc_setup_dma(SCC_CPU1_SPDMA_RW, BCU_USB_BUFFER_1, DMA_LINEAR, USE_NO_FH, DMA_WRITE, 0); scc_setup_dma(SCC_USB_RW, BCU_USB_BUFFER_0, DMA_LINEAR, USE_NO_FH, DMA_WRITE, 0); scc_setup_dma(SCC_CPU1_SPDMA_RW, BCU_USB_BUFFER_0, DMA_LINEAR, USE_NO_FH, DMA_READ, 0); /* Enable memory interface */ scc_enable(SCC_USB_RW, 1); /* Start (start_cmd=0) DMAs */ scc_dma_cmd(SCC_USB_RW, DMA_START, 0, DMA_READ); scc_dma_cmd(SCC_USB_RW, DMA_START, 0, DMA_WRITE); } else { printf("Cannot configure USB memory channel.\n"); printf("USB can not access RAM. SCC configuration failed.\n"); return retval; } /* Wait a short while */ udelay(300000); reg_write(USBH_BURSTSIZE(USBH_BASE), 0x00001c1c); /* Set EHCI structures and DATA in RAM */ reg_write(USBH_USBHMISC(USBH_BASE), 0x00840003); /* Set USBMODE to bigendian and set host mode */ reg_write(USBH_USBMODE(USBH_BASE), 0x00000007); /* * USBH_BURSTSIZE MUST EQUAL 0x00001c1c in order for * 512 byte USB transfers on the bulk pipe to work properly. * Set USBH_BURSTSIZE to 0x00001c1c */ reg_write(USBH_BURSTSIZE(USBH_BASE), 0x00001c1c); /* Insert access register addresses */ *hccr = REG_GLOBAL_START_ADDR + USBH_CAPLENGTH(USBH_BASE); *hcor = REG_GLOBAL_START_ADDR + USBH_USBCMD(USBH_BASE); return 0; }