r8a66597-hcd.c
来自「linux 内核源代码」· C语言 代码 · 共 2,241 行 · 第 1/4 页
C
2,241 行
/* * R8A66597 HCD (Host Controller Driver) * * Copyright (C) 2006-2007 Renesas Solutions Corp. * Portions Copyright (C) 2004 Psion Teklogix (for NetBook PRO) * Portions Copyright (C) 2004-2005 David Brownell * Portions Copyright (C) 1999 Roman Weissgaerber * * Author : Yoshihiro Shimoda <shimoda.yoshihiro@renesas.com> * * 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; version 2 of the License. * * 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * */#include <linux/module.h>#include <linux/kernel.h>#include <linux/sched.h>#include <linux/smp_lock.h>#include <linux/errno.h>#include <linux/init.h>#include <linux/timer.h>#include <linux/delay.h>#include <linux/list.h>#include <linux/interrupt.h>#include <linux/usb.h>#include <linux/platform_device.h>#include <linux/io.h>#include <linux/irq.h>#include "../core/hcd.h"#include "r8a66597.h"MODULE_DESCRIPTION("R8A66597 USB Host Controller Driver");MODULE_LICENSE("GPL");MODULE_AUTHOR("Yoshihiro Shimoda");#define DRIVER_VERSION "29 May 2007"static const char hcd_name[] = "r8a66597_hcd";/* module parameters */static unsigned short clock = XTAL12;module_param(clock, ushort, 0644);MODULE_PARM_DESC(clock, "input clock: 48MHz=32768, 24MHz=16384, 12MHz=0 " "(default=0)");static unsigned short vif = LDRV;module_param(vif, ushort, 0644);MODULE_PARM_DESC(vif, "input VIF: 3.3V=32768, 1.5V=0(default=32768)");static unsigned short endian;module_param(endian, ushort, 0644);MODULE_PARM_DESC(endian, "data endian: big=256, little=0 (default=0)");static unsigned short irq_sense = INTL;module_param(irq_sense, ushort, 0644);MODULE_PARM_DESC(irq_sense, "IRQ sense: low level=32, falling edge=0 " "(default=32)");static void packet_write(struct r8a66597 *r8a66597, u16 pipenum);static int r8a66597_get_frame(struct usb_hcd *hcd);/* this function must be called with interrupt disabled */static void enable_pipe_irq(struct r8a66597 *r8a66597, u16 pipenum, unsigned long reg){ u16 tmp; tmp = r8a66597_read(r8a66597, INTENB0); r8a66597_bclr(r8a66597, BEMPE | NRDYE | BRDYE, INTENB0); r8a66597_bset(r8a66597, 1 << pipenum, reg); r8a66597_write(r8a66597, tmp, INTENB0);}/* this function must be called with interrupt disabled */static void disable_pipe_irq(struct r8a66597 *r8a66597, u16 pipenum, unsigned long reg){ u16 tmp; tmp = r8a66597_read(r8a66597, INTENB0); r8a66597_bclr(r8a66597, BEMPE | NRDYE | BRDYE, INTENB0); r8a66597_bclr(r8a66597, 1 << pipenum, reg); r8a66597_write(r8a66597, tmp, INTENB0);}static void set_devadd_reg(struct r8a66597 *r8a66597, u8 r8a66597_address, u16 usbspd, u8 upphub, u8 hubport, int port){ u16 val; unsigned long devadd_reg = get_devadd_addr(r8a66597_address); val = (upphub << 11) | (hubport << 8) | (usbspd << 6) | (port & 0x0001); r8a66597_write(r8a66597, val, devadd_reg);}static int enable_controller(struct r8a66597 *r8a66597){ u16 tmp; int i = 0; do { r8a66597_write(r8a66597, USBE, SYSCFG0); tmp = r8a66597_read(r8a66597, SYSCFG0); if (i++ > 1000) { err("register access fail."); return -ENXIO; } } while ((tmp & USBE) != USBE); r8a66597_bclr(r8a66597, USBE, SYSCFG0); r8a66597_mdfy(r8a66597, clock, XTAL, SYSCFG0); i = 0; r8a66597_bset(r8a66597, XCKE, SYSCFG0); do { msleep(1); tmp = r8a66597_read(r8a66597, SYSCFG0); if (i++ > 500) { err("register access fail."); return -ENXIO; } } while ((tmp & SCKE) != SCKE); r8a66597_bset(r8a66597, DCFM | DRPD, SYSCFG0); r8a66597_bset(r8a66597, DRPD, SYSCFG1); r8a66597_bset(r8a66597, vif & LDRV, PINCFG); r8a66597_bset(r8a66597, HSE, SYSCFG0); r8a66597_bset(r8a66597, HSE, SYSCFG1); r8a66597_bset(r8a66597, USBE, SYSCFG0); r8a66597_bset(r8a66597, BEMPE | NRDYE | BRDYE, INTENB0); r8a66597_bset(r8a66597, irq_sense & INTL, SOFCFG); r8a66597_bset(r8a66597, BRDY0, BRDYENB); r8a66597_bset(r8a66597, BEMP0, BEMPENB); r8a66597_write(r8a66597, BURST | CPU_ADR_RD_WR, DMA0CFG); r8a66597_write(r8a66597, BURST | CPU_ADR_RD_WR, DMA1CFG); r8a66597_bset(r8a66597, endian & BIGEND, CFIFOSEL); r8a66597_bset(r8a66597, endian & BIGEND, D0FIFOSEL); r8a66597_bset(r8a66597, endian & BIGEND, D1FIFOSEL); r8a66597_bset(r8a66597, TRNENSEL, SOFCFG); r8a66597_bset(r8a66597, SIGNE | SACKE, INTENB1); r8a66597_bclr(r8a66597, DTCHE, INTENB1); r8a66597_bset(r8a66597, ATTCHE, INTENB1); r8a66597_bclr(r8a66597, DTCHE, INTENB2); r8a66597_bset(r8a66597, ATTCHE, INTENB2); return 0;}static void disable_controller(struct r8a66597 *r8a66597){ u16 tmp; r8a66597_write(r8a66597, 0, INTENB0); r8a66597_write(r8a66597, 0, INTENB1); r8a66597_write(r8a66597, 0, INTENB2); r8a66597_write(r8a66597, 0, INTSTS0); r8a66597_write(r8a66597, 0, INTSTS1); r8a66597_write(r8a66597, 0, INTSTS2); r8a66597_port_power(r8a66597, 0, 0); r8a66597_port_power(r8a66597, 1, 0); do { tmp = r8a66597_read(r8a66597, SOFCFG) & EDGESTS; udelay(640); } while (tmp == EDGESTS); r8a66597_bclr(r8a66597, DCFM | DRPD, SYSCFG0); r8a66597_bclr(r8a66597, DRPD, SYSCFG1); r8a66597_bclr(r8a66597, HSE, SYSCFG0); r8a66597_bclr(r8a66597, HSE, SYSCFG1); r8a66597_bclr(r8a66597, SCKE, SYSCFG0); udelay(1); r8a66597_bclr(r8a66597, PLLC, SYSCFG0); r8a66597_bclr(r8a66597, XCKE, SYSCFG0); r8a66597_bclr(r8a66597, USBE, SYSCFG0);}static int get_parent_r8a66597_address(struct r8a66597 *r8a66597, struct usb_device *udev){ struct r8a66597_device *dev; if (udev->parent && udev->parent->devnum != 1) udev = udev->parent; dev = dev_get_drvdata(&udev->dev); if (dev) return dev->address; else return 0;}static int is_child_device(char *devpath){ return (devpath[2] ? 1 : 0);}static int is_hub_limit(char *devpath){ return ((strlen(devpath) >= 4) ? 1 : 0);}static void get_port_number(char *devpath, u16 *root_port, u16 *hub_port){ if (root_port) { *root_port = (devpath[0] & 0x0F) - 1; if (*root_port >= R8A66597_MAX_ROOT_HUB) err("illegal root port number"); } if (hub_port) *hub_port = devpath[2] & 0x0F;}static u16 get_r8a66597_usb_speed(enum usb_device_speed speed){ u16 usbspd = 0; switch (speed) { case USB_SPEED_LOW: usbspd = LSMODE; break; case USB_SPEED_FULL: usbspd = FSMODE; break; case USB_SPEED_HIGH: usbspd = HSMODE; break; default: err("unknown speed"); break; } return usbspd;}static void set_child_connect_map(struct r8a66597 *r8a66597, int address){ int idx; idx = address / 32; r8a66597->child_connect_map[idx] |= 1 << (address % 32);}static void put_child_connect_map(struct r8a66597 *r8a66597, int address){ int idx; idx = address / 32; r8a66597->child_connect_map[idx] &= ~(1 << (address % 32));}static void set_pipe_reg_addr(struct r8a66597_pipe *pipe, u8 dma_ch){ u16 pipenum = pipe->info.pipenum; unsigned long fifoaddr[] = {D0FIFO, D1FIFO, CFIFO}; unsigned long fifosel[] = {D0FIFOSEL, D1FIFOSEL, CFIFOSEL}; unsigned long fifoctr[] = {D0FIFOCTR, D1FIFOCTR, CFIFOCTR}; if (dma_ch > R8A66597_PIPE_NO_DMA) /* dma fifo not use? */ dma_ch = R8A66597_PIPE_NO_DMA; pipe->fifoaddr = fifoaddr[dma_ch]; pipe->fifosel = fifosel[dma_ch]; pipe->fifoctr = fifoctr[dma_ch]; if (pipenum == 0) pipe->pipectr = DCPCTR; else pipe->pipectr = get_pipectr_addr(pipenum); if (check_bulk_or_isoc(pipenum)) { pipe->pipetre = get_pipetre_addr(pipenum); pipe->pipetrn = get_pipetrn_addr(pipenum); } else { pipe->pipetre = 0; pipe->pipetrn = 0; }}static struct r8a66597_device *get_urb_to_r8a66597_dev(struct r8a66597 *r8a66597, struct urb *urb){ if (usb_pipedevice(urb->pipe) == 0) return &r8a66597->device0; return dev_get_drvdata(&urb->dev->dev);}static int make_r8a66597_device(struct r8a66597 *r8a66597, struct urb *urb, u8 addr){ struct r8a66597_device *dev; int usb_address = urb->setup_packet[2]; /* urb->pipe is address 0 */ dev = kzalloc(sizeof(struct r8a66597_device), GFP_ATOMIC); if (dev == NULL) return -ENOMEM; dev_set_drvdata(&urb->dev->dev, dev); dev->udev = urb->dev; dev->address = addr; dev->usb_address = usb_address; dev->state = USB_STATE_ADDRESS; dev->ep_in_toggle = 0; dev->ep_out_toggle = 0; INIT_LIST_HEAD(&dev->device_list); list_add_tail(&dev->device_list, &r8a66597->child_device); get_port_number(urb->dev->devpath, &dev->root_port, &dev->hub_port); if (!is_child_device(urb->dev->devpath)) r8a66597->root_hub[dev->root_port].dev = dev; set_devadd_reg(r8a66597, dev->address, get_r8a66597_usb_speed(urb->dev->speed), get_parent_r8a66597_address(r8a66597, urb->dev), dev->hub_port, dev->root_port); return 0;}/* this function must be called with interrupt disabled */static u8 alloc_usb_address(struct r8a66597 *r8a66597, struct urb *urb){ u8 addr; /* R8A66597's address */ struct r8a66597_device *dev; if (is_hub_limit(urb->dev->devpath)) { err("Externel hub limit reached."); return 0; } dev = get_urb_to_r8a66597_dev(r8a66597, urb); if (dev && dev->state >= USB_STATE_ADDRESS) return dev->address; for (addr = 1; addr <= R8A66597_MAX_DEVICE; addr++) { if (r8a66597->address_map & (1 << addr)) continue; dbg("alloc_address: r8a66597_addr=%d", addr); r8a66597->address_map |= 1 << addr; if (make_r8a66597_device(r8a66597, urb, addr) < 0) return 0; return addr; } err("cannot communicate with a USB device more than 10.(%x)", r8a66597->address_map); return 0;}/* this function must be called with interrupt disabled */static void free_usb_address(struct r8a66597 *r8a66597, struct r8a66597_device *dev){ int port; if (!dev) return; dbg("free_addr: addr=%d", dev->address); dev->state = USB_STATE_DEFAULT; r8a66597->address_map &= ~(1 << dev->address); dev->address = 0; dev_set_drvdata(&dev->udev->dev, NULL); list_del(&dev->device_list); kfree(dev); for (port = 0; port < R8A66597_MAX_ROOT_HUB; port++) { if (r8a66597->root_hub[port].dev == dev) { r8a66597->root_hub[port].dev = NULL; break; } }}static void r8a66597_reg_wait(struct r8a66597 *r8a66597, unsigned long reg, u16 mask, u16 loop){ u16 tmp; int i = 0; do { tmp = r8a66597_read(r8a66597, reg); if (i++ > 1000000) { err("register%lx, loop %x is timeout", reg, loop); break; } ndelay(1); } while ((tmp & mask) != loop);}/* this function must be called with interrupt disabled */static void pipe_start(struct r8a66597 *r8a66597, struct r8a66597_pipe *pipe){ u16 tmp; tmp = r8a66597_read(r8a66597, pipe->pipectr) & PID; if ((pipe->info.pipenum != 0) & ((tmp & PID_STALL) != 0)) /* stall? */ r8a66597_mdfy(r8a66597, PID_NAK, PID, pipe->pipectr); r8a66597_mdfy(r8a66597, PID_BUF, PID, pipe->pipectr);}/* this function must be called with interrupt disabled */static void pipe_stop(struct r8a66597 *r8a66597, struct r8a66597_pipe *pipe){ u16 tmp; tmp = r8a66597_read(r8a66597, pipe->pipectr) & PID; if ((tmp & PID_STALL11) != PID_STALL11) /* force stall? */ r8a66597_mdfy(r8a66597, PID_STALL, PID, pipe->pipectr); r8a66597_mdfy(r8a66597, PID_NAK, PID, pipe->pipectr); r8a66597_reg_wait(r8a66597, pipe->pipectr, PBUSY, 0);}/* this function must be called with interrupt disabled */static void clear_all_buffer(struct r8a66597 *r8a66597, struct r8a66597_pipe *pipe){ u16 tmp; if (!pipe || pipe->info.pipenum == 0) return; pipe_stop(r8a66597, pipe); r8a66597_bset(r8a66597, ACLRM, pipe->pipectr); tmp = r8a66597_read(r8a66597, pipe->pipectr); tmp = r8a66597_read(r8a66597, pipe->pipectr); tmp = r8a66597_read(r8a66597, pipe->pipectr); r8a66597_bclr(r8a66597, ACLRM, pipe->pipectr);}/* this function must be called with interrupt disabled */static void r8a66597_pipe_toggle(struct r8a66597 *r8a66597, struct r8a66597_pipe *pipe, int toggle){ if (toggle) r8a66597_bset(r8a66597, SQSET, pipe->pipectr); else r8a66597_bset(r8a66597, SQCLR, pipe->pipectr);}/* this function must be called with interrupt disabled */static inline void cfifo_change(struct r8a66597 *r8a66597, u16 pipenum){ r8a66597_mdfy(r8a66597, MBW | pipenum, MBW | CURPIPE, CFIFOSEL); r8a66597_reg_wait(r8a66597, CFIFOSEL, CURPIPE, pipenum);}/* this function must be called with interrupt disabled */static inline void fifo_change_from_pipe(struct r8a66597 *r8a66597, struct r8a66597_pipe *pipe){ cfifo_change(r8a66597, 0); r8a66597_mdfy(r8a66597, MBW | 0, MBW | CURPIPE, D0FIFOSEL); r8a66597_mdfy(r8a66597, MBW | 0, MBW | CURPIPE, D1FIFOSEL); r8a66597_mdfy(r8a66597, MBW | pipe->info.pipenum, MBW | CURPIPE, pipe->fifosel); r8a66597_reg_wait(r8a66597, pipe->fifosel, CURPIPE, pipe->info.pipenum);}static u16 r8a66597_get_pipenum(struct urb *urb, struct usb_host_endpoint *hep){ struct r8a66597_pipe *pipe = hep->hcpriv; if (usb_pipeendpoint(urb->pipe) == 0) return 0; else return pipe->info.pipenum;}static u16 get_urb_to_r8a66597_addr(struct r8a66597 *r8a66597, struct urb *urb){ struct r8a66597_device *dev = get_urb_to_r8a66597_dev(r8a66597, urb); return (usb_pipedevice(urb->pipe) == 0) ? 0 : dev->address;}static unsigned short *get_toggle_pointer(struct r8a66597_device *dev, int urb_pipe){ if (!dev) return NULL; return usb_pipein(urb_pipe) ? &dev->ep_in_toggle : &dev->ep_out_toggle;}/* this function must be called with interrupt disabled */static void pipe_toggle_set(struct r8a66597 *r8a66597, struct r8a66597_pipe *pipe, struct urb *urb, int set){ struct r8a66597_device *dev = get_urb_to_r8a66597_dev(r8a66597, urb); unsigned char endpoint = usb_pipeendpoint(urb->pipe); unsigned short *toggle = get_toggle_pointer(dev, urb->pipe); if (!toggle) return; if (set) *toggle |= 1 << endpoint; else *toggle &= ~(1 << endpoint);}/* this function must be called with interrupt disabled */static void pipe_toggle_save(struct r8a66597 *r8a66597, struct r8a66597_pipe *pipe, struct urb *urb){ if (r8a66597_read(r8a66597, pipe->pipectr) & SQMON) pipe_toggle_set(r8a66597, pipe, urb, 1); else pipe_toggle_set(r8a66597, pipe, urb, 0);}/* this function must be called with interrupt disabled */static void pipe_toggle_restore(struct r8a66597 *r8a66597, struct r8a66597_pipe *pipe, struct urb *urb){ struct r8a66597_device *dev = get_urb_to_r8a66597_dev(r8a66597, urb); unsigned char endpoint = usb_pipeendpoint(urb->pipe); unsigned short *toggle = get_toggle_pointer(dev, urb->pipe); if (!toggle) return; r8a66597_pipe_toggle(r8a66597, pipe, *toggle & (1 << endpoint));}/* this function must be called with interrupt disabled */static void pipe_buffer_setting(struct r8a66597 *r8a66597, struct r8a66597_pipe_info *info){ u16 val = 0;
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?