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 + -
显示快捷键?