📄 m66592-udc.c
字号:
__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 + -