⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 m66592-udc.c

📁 linux下面gadget设备驱动
💻 C
📖 第 1 页 / 共 3 页
字号:
__releases(m66592->lock)__acquires(m66592->lock){	struct usb_ctrlrequest ctrl;	u16 ctsq;	ctsq = m66592_read(m66592, M66592_INTSTS0) & M66592_CTSQ;	m66592_write(m66592, ~M66592_CTRT, M66592_INTSTS0);	switch (ctsq) {	case M66592_CS_IDST: {		struct m66592_ep *ep;		struct m66592_request *req;		ep = &m66592->ep[0];		req = list_entry(ep->queue.next, struct m66592_request, queue);		transfer_complete(ep, req, 0);		}		break;	case M66592_CS_RDDS:	case M66592_CS_WRDS:	case M66592_CS_WRND:		if (setup_packet(m66592, &ctrl)) {			spin_unlock(&m66592->lock);			if (m66592->driver->setup(&m66592->gadget, &ctrl) < 0)				pipe_stall(m66592, 0);			spin_lock(&m66592->lock);		}		break;	case M66592_CS_RDSS:	case M66592_CS_WRSS:		control_end(m66592, 0);		break;	default:		printk(KERN_ERR "ctrl_stage: unexpect ctsq(%x)\n", ctsq);		break;	}}static irqreturn_t m66592_irq(int irq, void *_m66592){	struct m66592 *m66592 = _m66592;	u16 intsts0;	u16 intenb0;	u16 brdysts, nrdysts, bempsts;	u16 brdyenb, nrdyenb, bempenb;	u16 savepipe;	u16 mask0;	spin_lock(&m66592->lock);	intsts0 = m66592_read(m66592, M66592_INTSTS0);	intenb0 = m66592_read(m66592, M66592_INTENB0);	savepipe = m66592_read(m66592, M66592_CFIFOSEL);	mask0 = intsts0 & intenb0;	if (mask0) {		brdysts = m66592_read(m66592, M66592_BRDYSTS);		nrdysts = m66592_read(m66592, M66592_NRDYSTS);		bempsts = m66592_read(m66592, M66592_BEMPSTS);		brdyenb = m66592_read(m66592, M66592_BRDYENB);		nrdyenb = m66592_read(m66592, M66592_NRDYENB);		bempenb = m66592_read(m66592, M66592_BEMPENB);		if (mask0 & M66592_VBINT) {			m66592_write(m66592,  0xffff & ~M66592_VBINT,					M66592_INTSTS0);			m66592_start_xclock(m66592);			/* start vbus sampling */			m66592->old_vbus = m66592_read(m66592, M66592_INTSTS0)					& M66592_VBSTS;			m66592->scount = M66592_MAX_SAMPLING;			mod_timer(&m66592->timer,					jiffies + msecs_to_jiffies(50));		}		if (intsts0 & M66592_DVSQ)			irq_device_state(m66592);		if ((intsts0 & M66592_BRDY) && (intenb0 & M66592_BRDYE)				&& (brdysts & brdyenb)) {			irq_pipe_ready(m66592, brdysts, brdyenb);		}		if ((intsts0 & M66592_BEMP) && (intenb0 & M66592_BEMPE)				&& (bempsts & bempenb)) {			irq_pipe_empty(m66592, bempsts, bempenb);		}		if (intsts0 & M66592_CTRT)			irq_control_stage(m66592);	}	m66592_write(m66592, savepipe, M66592_CFIFOSEL);	spin_unlock(&m66592->lock);	return IRQ_HANDLED;}static void m66592_timer(unsigned long _m66592){	struct m66592 *m66592 = (struct m66592 *)_m66592;	unsigned long flags;	u16 tmp;	spin_lock_irqsave(&m66592->lock, flags);	tmp = m66592_read(m66592, M66592_SYSCFG);	if (!(tmp & M66592_RCKE)) {		m66592_bset(m66592, M66592_RCKE | M66592_PLLC, M66592_SYSCFG);		udelay(10);		m66592_bset(m66592, M66592_SCKE, M66592_SYSCFG);	}	if (m66592->scount > 0) {		tmp = m66592_read(m66592, M66592_INTSTS0) & M66592_VBSTS;		if (tmp == m66592->old_vbus) {			m66592->scount--;			if (m66592->scount == 0) {				if (tmp == M66592_VBSTS)					m66592_usb_connect(m66592);				else					m66592_usb_disconnect(m66592);			} else {				mod_timer(&m66592->timer,					jiffies + msecs_to_jiffies(50));			}		} else {			m66592->scount = M66592_MAX_SAMPLING;			m66592->old_vbus = tmp;			mod_timer(&m66592->timer,					jiffies + msecs_to_jiffies(50));		}	}	spin_unlock_irqrestore(&m66592->lock, flags);}/*-------------------------------------------------------------------------*/static int m66592_enable(struct usb_ep *_ep,			 const struct usb_endpoint_descriptor *desc){	struct m66592_ep *ep;	ep = container_of(_ep, struct m66592_ep, ep);	return alloc_pipe_config(ep, desc);}static int m66592_disable(struct usb_ep *_ep){	struct m66592_ep *ep;	struct m66592_request *req;	unsigned long flags;	ep = container_of(_ep, struct m66592_ep, ep);	BUG_ON(!ep);	while (!list_empty(&ep->queue)) {		req = list_entry(ep->queue.next, struct m66592_request, queue);		spin_lock_irqsave(&ep->m66592->lock, flags);		transfer_complete(ep, req, -ECONNRESET);		spin_unlock_irqrestore(&ep->m66592->lock, flags);	}	pipe_irq_disable(ep->m66592, ep->pipenum);	return free_pipe_config(ep);}static struct usb_request *m66592_alloc_request(struct usb_ep *_ep,						gfp_t gfp_flags){	struct m66592_request *req;	req = kzalloc(sizeof(struct m66592_request), gfp_flags);	if (!req)		return NULL;	INIT_LIST_HEAD(&req->queue);	return &req->req;}static void m66592_free_request(struct usb_ep *_ep, struct usb_request *_req){	struct m66592_request *req;	req = container_of(_req, struct m66592_request, req);	kfree(req);}static int m66592_queue(struct usb_ep *_ep, struct usb_request *_req,			gfp_t gfp_flags){	struct m66592_ep *ep;	struct m66592_request *req;	unsigned long flags;	int request = 0;	ep = container_of(_ep, struct m66592_ep, ep);	req = container_of(_req, struct m66592_request, req);	if (ep->m66592->gadget.speed == USB_SPEED_UNKNOWN)		return -ESHUTDOWN;	spin_lock_irqsave(&ep->m66592->lock, flags);	if (list_empty(&ep->queue))		request = 1;	list_add_tail(&req->queue, &ep->queue);	req->req.actual = 0;	req->req.status = -EINPROGRESS;	if (ep->desc == NULL)	/* control */		start_ep0(ep, req);	else {		if (request && !ep->busy)			start_packet(ep, req);	}	spin_unlock_irqrestore(&ep->m66592->lock, flags);	return 0;}static int m66592_dequeue(struct usb_ep *_ep, struct usb_request *_req){	struct m66592_ep *ep;	struct m66592_request *req;	unsigned long flags;	ep = container_of(_ep, struct m66592_ep, ep);	req = container_of(_req, struct m66592_request, req);	spin_lock_irqsave(&ep->m66592->lock, flags);	if (!list_empty(&ep->queue))		transfer_complete(ep, req, -ECONNRESET);	spin_unlock_irqrestore(&ep->m66592->lock, flags);	return 0;}static int m66592_set_halt(struct usb_ep *_ep, int value){	struct m66592_ep *ep;	struct m66592_request *req;	unsigned long flags;	int ret = 0;	ep = container_of(_ep, struct m66592_ep, ep);	req = list_entry(ep->queue.next, struct m66592_request, queue);	spin_lock_irqsave(&ep->m66592->lock, flags);	if (!list_empty(&ep->queue)) {		ret = -EAGAIN;		goto out;	}	if (value) {		ep->busy = 1;		pipe_stall(ep->m66592, ep->pipenum);	} else {		ep->busy = 0;		pipe_stop(ep->m66592, ep->pipenum);	}out:	spin_unlock_irqrestore(&ep->m66592->lock, flags);	return ret;}static void m66592_fifo_flush(struct usb_ep *_ep){	struct m66592_ep *ep;	unsigned long flags;	ep = container_of(_ep, struct m66592_ep, ep);	spin_lock_irqsave(&ep->m66592->lock, flags);	if (list_empty(&ep->queue) && !ep->busy) {		pipe_stop(ep->m66592, ep->pipenum);		m66592_bclr(ep->m66592, M66592_BCLR, ep->fifoctr);	}	spin_unlock_irqrestore(&ep->m66592->lock, flags);}static struct usb_ep_ops m66592_ep_ops = {	.enable		= m66592_enable,	.disable	= m66592_disable,	.alloc_request	= m66592_alloc_request,	.free_request	= m66592_free_request,	.queue		= m66592_queue,	.dequeue	= m66592_dequeue,	.set_halt	= m66592_set_halt,	.fifo_flush	= m66592_fifo_flush,};/*-------------------------------------------------------------------------*/static struct m66592 *the_controller;int usb_gadget_register_driver(struct usb_gadget_driver *driver){	struct m66592 *m66592 = the_controller;	int retval;	if (!driver			|| driver->speed != USB_SPEED_HIGH			|| !driver->bind			|| !driver->setup)		return -EINVAL;	if (!m66592)		return -ENODEV;	if (m66592->driver)		return -EBUSY;	/* hook up the driver */	driver->driver.bus = NULL;	m66592->driver = driver;	m66592->gadget.dev.driver = &driver->driver;	retval = device_add(&m66592->gadget.dev);	if (retval) {		printk(KERN_ERR "device_add error (%d)\n", retval);		goto error;	}	retval = driver->bind (&m66592->gadget);	if (retval) {		printk(KERN_ERR "bind to driver error (%d)\n", retval);		device_del(&m66592->gadget.dev);		goto error;	}	m66592_bset(m66592, M66592_VBSE | M66592_URST, M66592_INTENB0);	if (m66592_read(m66592, M66592_INTSTS0) & M66592_VBSTS) {		m66592_start_xclock(m66592);		/* start vbus sampling */		m66592->old_vbus = m66592_read(m66592,					 M66592_INTSTS0) & M66592_VBSTS;		m66592->scount = M66592_MAX_SAMPLING;		mod_timer(&m66592->timer, jiffies + msecs_to_jiffies(50));	}	return 0;error:	m66592->driver = NULL;	m66592->gadget.dev.driver = NULL;	return retval;}EXPORT_SYMBOL(usb_gadget_register_driver);int usb_gadget_unregister_driver(struct usb_gadget_driver *driver){	struct m66592 *m66592 = the_controller;	unsigned long flags;	if (driver != m66592->driver || !driver->unbind)		return -EINVAL;	spin_lock_irqsave(&m66592->lock, flags);	if (m66592->gadget.speed != USB_SPEED_UNKNOWN)		m66592_usb_disconnect(m66592);	spin_unlock_irqrestore(&m66592->lock, flags);	m66592_bclr(m66592, M66592_VBSE | M66592_URST, M66592_INTENB0);	driver->unbind(&m66592->gadget);	init_controller(m66592);	disable_controller(m66592);	device_del(&m66592->gadget.dev);	m66592->driver = NULL;	return 0;}EXPORT_SYMBOL(usb_gadget_unregister_driver);/*-------------------------------------------------------------------------*/static int m66592_get_frame(struct usb_gadget *_gadget){	struct m66592 *m66592 = gadget_to_m66592(_gadget);	return m66592_read(m66592, M66592_FRMNUM) & 0x03FF;}static struct usb_gadget_ops m66592_gadget_ops = {	.get_frame		= m66592_get_frame,};static int __exit m66592_remove(struct platform_device *pdev){	struct m66592		*m66592 = dev_get_drvdata(&pdev->dev);	del_timer_sync(&m66592->timer);	iounmap(m66592->reg);	free_irq(platform_get_irq(pdev, 0), m66592);	m66592_free_request(&m66592->ep[0].ep, m66592->ep0_req);	kfree(m66592);	return 0;}static void nop_completion(struct usb_ep *ep, struct usb_request *r){}#define resource_len(r) (((r)->end - (r)->start) + 1)static int __init m66592_probe(struct platform_device *pdev){	struct resource *res;	int irq;	void __iomem *reg = NULL;	struct m66592 *m66592 = NULL;	int ret = 0;	int i;	res = platform_get_resource_byname(pdev, IORESOURCE_MEM,			(char *)udc_name);	if (!res) {		ret = -ENODEV;		printk(KERN_ERR "platform_get_resource_byname error.\n");		goto clean_up;	}	irq = platform_get_irq(pdev, 0);	if (irq < 0) {		ret = -ENODEV;		printk(KERN_ERR "platform_get_irq error.\n");		goto clean_up;	}	reg = ioremap(res->start, resource_len(res));	if (reg == NULL) {		ret = -ENOMEM;		printk(KERN_ERR "ioremap error.\n");		goto clean_up;	}	/* initialize ucd */	m66592 = kzalloc(sizeof(struct m66592), GFP_KERNEL);	if (m66592 == NULL) {		printk(KERN_ERR "kzalloc error\n");		goto clean_up;	}	spin_lock_init(&m66592->lock);	dev_set_drvdata(&pdev->dev, m66592);	m66592->gadget.ops = &m66592_gadget_ops;	device_initialize(&m66592->gadget.dev);	strcpy(m66592->gadget.dev.bus_id, "gadget");	m66592->gadget.is_dualspeed = 1;	m66592->gadget.dev.parent = &pdev->dev;	m66592->gadget.dev.dma_mask = pdev->dev.dma_mask;	m66592->gadget.dev.release = pdev->dev.release;	m66592->gadget.name = udc_name;	init_timer(&m66592->timer);	m66592->timer.function = m66592_timer;	m66592->timer.data = (unsigned long)m66592;	m66592->reg = reg;	m66592->bi_bufnum = M66592_BASE_BUFNUM;	ret = request_irq(irq, m66592_irq, IRQF_DISABLED | IRQF_SHARED,			udc_name, m66592);	if (ret < 0) {		printk(KERN_ERR "request_irq error (%d)\n", ret);		goto clean_up;	}	INIT_LIST_HEAD(&m66592->gadget.ep_list);	m66592->gadget.ep0 = &m66592->ep[0].ep;	INIT_LIST_HEAD(&m66592->gadget.ep0->ep_list);	for (i = 0; i < M66592_MAX_NUM_PIPE; i++) {		struct m66592_ep *ep = &m66592->ep[i];		if (i != 0) {			INIT_LIST_HEAD(&m66592->ep[i].ep.ep_list);			list_add_tail(&m66592->ep[i].ep.ep_list,					&m66592->gadget.ep_list);		}		ep->m66592 = m66592;		INIT_LIST_HEAD(&ep->queue);		ep->ep.name = m66592_ep_name[i];		ep->ep.ops = &m66592_ep_ops;		ep->ep.maxpacket = 512;	}	m66592->ep[0].ep.maxpacket = 64;	m66592->ep[0].pipenum = 0;	m66592->ep[0].fifoaddr = M66592_CFIFO;	m66592->ep[0].fifosel = M66592_CFIFOSEL;	m66592->ep[0].fifoctr = M66592_CFIFOCTR;	m66592->ep[0].fifotrn = 0;	m66592->ep[0].pipectr = get_pipectr_addr(0);	m66592->pipenum2ep[0] = &m66592->ep[0];	m66592->epaddr2ep[0] = &m66592->ep[0];	the_controller = m66592;	m66592->ep0_req = m66592_alloc_request(&m66592->ep[0].ep, GFP_KERNEL);	if (m66592->ep0_req == NULL)		goto clean_up2;	m66592->ep0_req->complete = nop_completion;	init_controller(m66592);	dev_info(&pdev->dev, "version %s\n", DRIVER_VERSION);	return 0;clean_up2:	free_irq(irq, m66592);clean_up:	if (m66592) {		if (m66592->ep0_req)			m66592_free_request(&m66592->ep[0].ep, m66592->ep0_req);		kfree(m66592);	}	if (reg)		iounmap(reg);	return ret;}/*-------------------------------------------------------------------------*/static struct platform_driver m66592_driver = {	.remove =	__exit_p(m66592_remove),	.driver		= {		.name =	(char *) udc_name,	},};static int __init m66592_udc_init(void){	return platform_driver_probe(&m66592_driver, m66592_probe);}module_init(m66592_udc_init);static void __exit m66592_udc_cleanup(void){	platform_driver_unregister(&m66592_driver);}module_exit(m66592_udc_cleanup);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -