Chinaunix首页 | 论坛 | 博客
  • 博客访问: 293479
  • 博文数量: 212
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 37
  • 用 户 组: 普通用户
  • 注册时间: 2013-06-07 11:07
文章分类

全部博文(212)

文章存档

2015年(90)

2013年(122)

我的朋友

分类: 服务器与存储

2015-07-02 23:35:48

学习Gadget比较有效的办法是掌握基本架构后,认真研读例程。其实不单Gadget如此,其他Linux驱动或子系统都是如此。另外Linux下的外设驱动通常有分层的概念,有带有面向对象的思想,因此研读代码是比较有效的领悟办法。

一般一个Linux的USB设备驱动,包括两大部分,一是CPU USB控制器部分的驱动,驱动文件名往往是xxx_udc.c,此部分驱动很多是与硬件CPU相关,包含寄存器设置、DMA设置,同时此部分也柔和了USB Gadget架构的东西,通常此部分代码是比较枯燥且难于理解的,每个CPU平台,此部分代码差异很大,但幸好此部分代码一般厂商会提供。二是gadget功能驱动,如果说USB控制器驱动是肉体,则此部分是灵魂,USB要在通信中是属于什么类型设备(HID?键盘?U盘?),怎样传输等等都是在此部分驱动中实现的。

Linux内核源码driver/usb/gadget/zero.c是一个很好的gadget功能驱动框架,本来内核开发者是想通过这个驱动框架来来方便第三方USB控制器驱动开发者测试UDC驱动的。

先上一个例程,这个例程是华清远见刘洪涛老师在阐述USB驱动部分写的一个例程,尽管例程尚有很多可优化、冗余的东西,尽管例程中描述注册字符设备驱动的方式是陈旧的方式,但是这个例程还是精彩阐述的如何开发一个gadget功能驱动。

在放这个例程前,为方便阅读,我更改部分代码格式。


  1. /*  
  2.  * zero.c -- Gadget Zero, for simple USB development 
  3.  * All rights reserved. 
  4.  */  
  5.    
  6. /* #define VERBOSE_DEBUG */  
  7.   
  8. #include   
  9. #include   
  10. #include   
  11. #include   
  12. #include   
  13. #include "gadget_chips.h"  
  14. #include   
  15. #include   
  16. #include   
  17. #include   
  18. #include   
  19. #include   
  20. #include   
  21. #include   
  22. #include  /* size_t */  
  23. #include  /* error codes */  
  24. #include   
  25. #include   
  26. #include   
  27.   
  28. /*-------------------------------------------------------------------------*/  
  29. static const char shortname[] = "zero";  
  30. static const char loopback[] = "loop input to output";  
  31. static const char longname[] = "Gadget Zero";  
  32. static const char source_sink[] = "source and sink data";  
  33. #define STRING_MANUFACTURER 25  
  34. #define STRING_PRODUCT 42  
  35. #define STRING_SERIAL 101  
  36. #define STRING_SOURCE_SINK 250  
  37. #define STRING_LOOPBACK 251  
  38.   
  39. //#define DRIVER_VENDOR_NUM 0x0525 /* NetChip */  
  40. //#define DRIVER_PRODUCT_NUM 0xa4a0 /* Linux-USB "Gadget Zero" */  
  41. #define DRIVER_VENDOR_NUM 0x5345 /* NetChip */  
  42. #define DRIVER_PRODUCT_NUM 0x1234 /* Linux-USB "Gadget Zero" */  
  43.   
  44. static int usb_zero_major = 251;  
  45. /*-------------------------------------------------------------------------*/  
  46. static const char *EP_OUT_NAME; /* sink */  
  47. /*-------------------------------------------------------------------------*/  
  48.   
  49. /* big enough to hold our biggest descriptor */  
  50. #define USB_BUFSIZ 256  
  51. struct zero_dev   
  52. {  
  53.     //zero设备结构  
  54.     spinlock_t lock;  
  55.     struct usb_gadget *gadget;  
  56.     struct usb_request *req; /* for control responses */  
  57.     struct usb_ep *out_ep;  
  58.     struct cdev cdev;  
  59.     unsigned char data[128];  
  60.     unsigned int data_size;  
  61.     wait_queue_head_t bulkrq;   
  62. };  
  63. #define CONFIG_LOOPBACK 2  
  64. static struct usb_device_descriptor device_desc =   
  65. {   
  66.     //设备描述符  
  67.     .bLength = sizeof device_desc,  
  68.     .bDescriptorType = USB_DT_DEVICE,  
  69.     .bcdUSB = __constant_cpu_to_le16(0x0110),  
  70.     .bDeviceClass = USB_CLASS_VENDOR_SPEC,  
  71.     .idVendor = __constant_cpu_to_le16(DRIVER_VENDOR_NUM),  
  72.     .idProduct = __constant_cpu_to_le16(DRIVER_PRODUCT_NUM),  
  73.     .iManufacturer = STRING_MANUFACTURER,  
  74.     .iProduct = STRING_PRODUCT,  
  75.     .iSerialNumber = STRING_SERIAL,  
  76.     .bNumConfigurations = 1,  
  77. };  
  78. static struct usb_endpoint_descriptor fs_sink_desc =   
  79. {   
  80.     //端点描述符  
  81.     .bLength = USB_DT_ENDPOINT_SIZE,  
  82.     .bDescriptorType = USB_DT_ENDPOINT,  
  83.     .bEndpointAddress = USB_DIR_OUT, //对主机端来说,输出  
  84.     .bmAttributes = USB_ENDPOINT_XFER_BULK,  
  85.  };  
  86.   
  87. static struct usb_config_descriptor loopback_config =   
  88. {   
  89.     //配置描述符  
  90.     .bLength = sizeof loopback_config,  
  91.     .bDescriptorType = USB_DT_CONFIG,  
  92.     /* compute wTotalLength on the fly */  
  93.     .bNumInterfaces = 1,  
  94.     .bConfigurationValue = CONFIG_LOOPBACK,  
  95.     .iConfiguration = STRING_LOOPBACK,  
  96.     .bmAttributes = USB_CONFIG_ATT_ONE | USB_CONFIG_ATT_SELFPOWER,  
  97.     .bMaxPower = 1, /* self-powered */  
  98.  };  
  99.    
  100. static const struct usb_interface_descriptor loopback_intf =   
  101. {   
  102.     //接口描述符  
  103.     .bLength = sizeof loopback_intf,  
  104.     .bDescriptorType = USB_DT_INTERFACE,  
  105.     .bNumEndpoints = 1,  
  106.     .bInterfaceClass = USB_CLASS_VENDOR_SPEC,  
  107.     .iInterface = STRING_LOOPBACK,  
  108. };  
  109.   
  110. /* static strings, in UTF-8 */  
  111. #define STRING_MANUFACTURER         25  
  112. #define STRING_PRODUCT              42  
  113. #define STRING_SERIAL               101  
  114. #define STRING_SOURCE_SINK          250  
  115. #define STRING_LOOPBACK             251  
  116. static char manufacturer[50];  
  117. /* default serial number takes at least two packets */  
  118. static char serial[] = "0123456789.0123456789.0123456789";  
  119. static struct usb_string strings[] =   
  120. {   
  121.     //字符串描述符  
  122.     { STRING_MANUFACTURER, manufacturer, },  
  123.     { STRING_PRODUCT, longname, },  
  124.     { STRING_SERIAL, serial, },  
  125.     { STRING_LOOPBACK, loopback, },  
  126.     { STRING_SOURCE_SINK, source_sink, },  
  127.     { } /* end of list */  
  128. };  
  129.   
  130. static struct usb_gadget_strings stringtab =   
  131. {  
  132.     .language = 0x0409, /* en-us */  
  133.     .strings = strings,  
  134. };  
  135.   
  136. static const struct usb_descriptor_header *fs_loopback_function[] =   
  137. {  
  138.     (struct usb_descriptor_header *) &loopback_intf,  
  139.     (struct usb_descriptor_header *) &fs_sink_desc,  
  140.     NULL,  
  141. };  
  142.   
  143.   
  144. static void free_ep_req(struct usb_ep *ep, struct usb_request *req)  
  145. {  
  146.     kfree(req->buf);  
  147.     usb_ep_free_request(ep, req);  
  148. }  
  149.   
  150. static struct usb_request *alloc_ep_req(struct usb_ep *ep, unsigned length)//分配请求  
  151. {  
  152.     struct usb_request *req;  
  153.     req = usb_ep_alloc_request(ep, GFP_ATOMIC);  
  154.     if (req)  
  155.     {  
  156.         req->length = length;  
  157.         req->buf = kmalloc(length, GFP_ATOMIC);  
  158.         if (!req->buf)   
  159.         {  
  160.             usb_ep_free_request(ep, req);  
  161.             req = NULL;  
  162.         }  
  163.     }  
  164.     return req;  
  165. }  
  166.   
  167. static void source_sink_complete(struct usb_ep *ep, struct usb_request *req)//请求完成函数  
  168. {  
  169.     struct zero_dev *dev = ep->driver_data;  
  170.     int status = req->status;  
  171.     switch (status)   
  172.     {  
  173.         case 0: /* normal completion */  
  174.                 if (ep == dev->out_ep)   
  175.                 {  
  176.                     memcpy(dev->data, req->buf, req->actual);//返回数据拷贝到req->buf中, //dev->data_size=req->length;   
  177.                     dev->data_size=req->actual; //实际长度为req->actual;需要确认req –>short_not_ok为0。参考gadget.h中关于usb_request结构的注释  
  178.                 }   
  179.             break;  
  180.         /* this endpoint is normally active while we're configured */  
  181.         case -ECONNABORTED: /* hardware forced ep reset */  
  182.         case -ECONNRESET:   /* request dequeued */  
  183.         case -ESHUTDOWN:    /* disconnect from host */  
  184.                 printk("%s gone (%d), %d/%d/n", ep->name, status,req->actual, req->length);  
  185.         case -EOVERFLOW:   
  186.         /* buffer overrun on read means that 
  187.          * we didn't provide a big enough 
  188.          * buffer. 
  189.          */  
  190.         default:  
  191.         #if 1  
  192.             printk("%s complete --> %d, %d/%d/n", ep->name,  
  193.             status, req->actual, req->length);  
  194.         #endif  
  195.         case -EREMOTEIO: /* short read */  
  196.             break;  
  197.     }  
  198.     free_ep_req(ep, req);  
  199.     wake_up_interruptible (&dev->bulkrq); //唤醒读函数  
  200. }  
  201.   
  202. static struct usb_request *source_sink_start_ep(struct usb_ep *ep)//构造并发送读请求  
  203. {  
  204.     struct usb_request *req;  
  205.     int status;  
  206.     //printk("in %s/n",__FUNCTION__);  
  207.     req = alloc_ep_req(ep, 128);  
  208.     if (!req)  
  209.         return NULL;  
  210.     memset(req->buf, 0, req->length);  
  211.     req->complete = source_sink_complete; //请求完成函数  
  212.     status = usb_ep_queue(ep, req, GFP_ATOMIC); //递交请求  
  213.     if (status)   
  214.     {  
  215.          struct zero_dev *dev = ep->driver_data;  
  216.          printk("start %s --> %d/n", ep->name, status);  
  217.          free_ep_req(ep, req);  
  218.          req = NULL;  
  219.     }  
  220.     return req;  
  221. }  
  222.   
  223. static int usb_zero_open (struct inode *inode, struct file *file) //打开设备  
  224. {  
  225.     struct zero_dev *dev = container_of (inode->i_cdev, struct zero_dev, cdev);  
  226.     file->private_data = dev;  
  227.     init_waitqueue_head (&dev->bulkrq);  
  228.     return 0;  
  229. }  
  230.   
  231. static int usb_zero_release (struct inode *inode, struct file *file) //关闭设备  
  232. {  
  233.     return 0;  
  234. }  
  235.   
  236. ssize_t usb_zero_read (struct file * file, const char __user * buf, size_t count,loff_t * f_pos) //读设备  
  237. {  
  238.     struct zero_dev *dev =file->private_data;  
  239.     struct usb_request *req;  
  240.     int status;  
  241.     struct usb_ep *ep;  
  242.     struct usb_gadget *gadget = dev->gadget;  
  243.     ssize_t ret = 0;  
  244.     int result;  
  245.     ep=dev->out_ep;  
  246.     source_sink_start_ep(ep);//构造、递交读请求  
  247.     if (count < 0)  
  248.          return -EINVAL;  
  249.     interruptible_sleep_on (&dev->bulkrq);//睡眠,等到请求完成  
  250.     if (copy_to_user (buf,dev->data,dev->data_size)) //拷贝读取的数据到用户空间  
  251.     {  
  252.         ret = -EFAULT;  
  253.     }  
  254.     else  
  255.     {  
  256.         ret = dev->data_size;  
  257.     }  
  258.     return ret;  
  259. }  
  260.   
  261. struct file_operations usb_zero_fops =   
  262. {  
  263.     .owner = THIS_MODULE,  
  264.     .read = usb_zero_read,  
  265.     .open = usb_zero_open,  
  266.     .release = usb_zero_release,  
  267. };  
  268.   
  269. static void usb_zero_setup_cdev (struct zero_dev *dev, int minor)//注册字符设备驱动  
  270. {  
  271.     int err, devno = MKDEV (usb_zero_major, minor);  
  272.     cdev_init(&dev->cdev, &usb_zero_fops);  
  273.     dev->cdev.owner = THIS_MODULE;  
  274.     err = cdev_add (&dev->cdev, devno, 1);  
  275.     if (err)  
  276.         printk ("Error adding usb_rcv/n");  
  277. }  
  278.   
  279. static void zero_setup_complete(struct usb_ep *ep, struct usb_request *req)//配置端点0的请求完成处理  
  280. {  
  281.     if (req->status || req->actual != req->length)  
  282.         printk("setup complete --> %d, %d/%d/n",req->status, req->actual, req->length);  
  283. }  
  284. static void zero_reset_config(struct zero_dev *dev) //复位配置  
  285. {  
  286.     usb_ep_disable(dev->out_ep);  
  287.     dev->out_ep = NULL;  
  288. }  
  289.   
  290. static void zero_disconnect(struct usb_gadget *gadget)//卸载驱动时被调用,做一些注销工作  
  291. {  
  292.     struct zero_dev *dev = get_gadget_data(gadget);  
  293.     unsigned long flags;  
  294.     unregister_chrdev_region (MKDEV (usb_zero_major, 0), 1);  
  295.     cdev_del (&(dev->cdev));  
  296.     zero_reset_config(dev);  
  297.     printk("in %s/n",__FUNCTION__);  
  298. }  
  299.   
  300. static int config_buf(struct usb_gadget *gadget,u8 *buf, u8 type, unsigned index)  
  301. {  
  302.     //int is_source_sink;  
  303.     int len;  
  304.     const struct usb_descriptor_header **function;  
  305.     int hs = 0;  
  306.     function =fs_loopback_function;//根据fs_loopback_function,得到长度,//此处len=配置(9)+1个接口(9)+1个端点(7)=25  
  307.     len = usb_gadget_config_buf(&loopback_config,buf, USB_BUFSIZ, function);  
  308.     if (len < 0)  
  309.         return len;  
  310.     ((struct usb_config_descriptor *) buf)->bDescriptorType = type;  
  311.     return len;  
  312. }  
  313.   
  314. static int set_loopback_config(struct zero_dev *dev)  
  315. {  
  316.     int result = 0;  
  317.     struct usb_ep *ep;  
  318.     struct usb_gadget *gadget = dev->gadget;  
  319.     ep=dev->out_ep;  
  320.     const struct usb_endpoint_descriptor *d;  
  321.     d = &fs_sink_desc;  
  322.     result = usb_ep_enable(ep, d); //激活端点  
  323.     //printk("");  
  324.     if (result == 0)   
  325.     {  
  326.             printk("connected/n"); //如果成功,打印“connected”  
  327.     }   
  328.     else  
  329.             printk("can't enable %s, result %d/n", ep->name, result);  
  330.     return result;  
  331. }  
  332.   
  333. static int zero_set_config(struct zero_dev *dev, unsigned number)  
  334. {  
  335.     int result = 0;  
  336.     struct usb_gadget *gadget = dev->gadget;  
  337.     result = set_loopback_config(dev);//激活设备  
  338.     if (result)  
  339.         zero_reset_config(dev); //复位设备  
  340.     else   
  341.     {  
  342.         char *speed;  
  343.   
  344.         switch (gadget->speed)   
  345.         {  
  346.             case USB_SPEED_LOW: speed = "low"break;  
  347.             case USB_SPEED_FULL: speed = "full"break;  
  348.             case USB_SPEED_HIGH: speed = "high"break;  
  349.             default: speed = " "break;  
  350.         }  
  351.     }  
  352.     return result;  
  353. }  
  354.   
  355. /* 
  356.  *zero_setup完成USB设置阶段和具体功能相关的交互部分 
  357.  */  
  358. static int zero_setup(struct usb_gadget *gadget, const struct usb_ctrlrequest *ctrl)  
  359. {  
  360.     struct zero_dev *dev = get_gadget_data(gadget);  
  361.     struct usb_request *req = dev->req;  
  362.     int value = -EOPNOTSUPP;  
  363.     u16 w_index = le16_to_cpu(ctrl->wIndex);  
  364.     u16 w_value = le16_to_cpu(ctrl->wValue);  
  365.     u16 w_length = le16_to_cpu(ctrl->wLength);  
  366.   
  367. /* usually this stores reply data in the pre-allocated ep0 buffer, 
  368.  * but config change events will reconfigure hardware. 
  369.  */  
  370.     req->zero = 0;  
  371.   
  372.     switch (ctrl->bRequest)   
  373.     {  
  374.         case USB_REQ_GET_DESCRIPTOR: //获取描述符  
  375.             if (ctrl->bRequestType != USB_DIR_IN)  
  376.                 goto unknown;  
  377.             switch (w_value >> 8)   
  378.             {  
  379.                 case USB_DT_DEVICE: //获取设备描述符  
  380.                     value = min(w_length, (u16) sizeof device_desc);  
  381.                     memcpy(req->buf, &device_desc, value);  
  382.                     break;  
  383.                 case USB_DT_CONFIG: //获取配置,注意:会根据fs_loopback_function读取到接口、端点描述符,注意通过config_buf完成读取数据及数量的统计。  
  384.                     value = config_buf(gadget, req->buf,  
  385.                     w_value >> 8,  
  386.                     w_value & 0xff);  
  387.                     if (value >= 0)  
  388.                         value = min(w_length, (u16) value);  
  389.                     break;  
  390.                 case USB_DT_STRING:  
  391.                     value = usb_gadget_get_string(&stringtab,w_value & 0xff, req->buf);  
  392.                     if (value >= 0)  
  393.                         value = min(w_length, (u16) value);  
  394.                     break;  
  395.             }  
  396.             break;  
  397.   
  398.         case USB_REQ_SET_CONFIGURATION:  
  399.             if (ctrl->bRequestType != 0)  
  400.                 goto unknown;  
  401.             spin_lock(&dev->lock);  
  402.             value = zero_set_config(dev, w_value);//激活相应的端点  
  403.             spin_unlock(&dev->lock);  
  404.             break;  
  405.         default:  
  406.             unknown:  
  407.         printk(  
  408.                 "unknown control req%02x.%02x v%04x i%04x l%d/n",  
  409.                 ctrl->bRequestType, ctrl->bRequest,  
  410.                 w_value, w_index, w_length);  
  411.     }  
  412. /* respond with data transfer before status phase */  
  413.     if (value >= 0)   
  414.     {  
  415.         req->length = value;  
  416.         req->zero = value < w_length;  
  417.         value = usb_ep_queue(gadget->ep0, req, GFP_ATOMIC);//通过端点0完成setup  
  418.         if (value < 0)   
  419.         {  
  420.             printk("ep_queue --> %d/n", value);  
  421.             req->status = 0;  
  422.             zero_setup_complete(gadget->ep0, req);  
  423.         }  
  424.     }  
  425.   /* device either stalls (value < 0) or reports success */  
  426.     return value;  
  427. }  
  428. static void zero_unbind(struct usb_gadget *gadget) //解除绑定  
  429. {  
  430.     struct zero_dev *dev = get_gadget_data(gadget);  
  431.     printk("unbind/n");  
  432.     unregister_chrdev_region (MKDEV (usb_zero_major, 0), 1);  
  433.     cdev_del (&(dev->cdev));  
  434.     /* we've already been disconnected ... no i/o is active */  
  435.     if (dev->req)   
  436.     {  
  437.         dev->req->length = USB_BUFSIZ;  
  438.         free_ep_req(gadget->ep0, dev->req);  
  439.     }  
  440.     kfree(dev);  
  441.     set_gadget_data(gadget, NULL);  
  442. }  
  443.   
  444. static int __init zero_bind(struct usb_gadget *gadget) //绑定过程   
  445. {  
  446.     struct zero_dev *dev;  
  447.     struct usb_ep *ep;  
  448.     int gcnum;  
  449.     usb_ep_autoconfig_reset(gadget);  
  450.     ep = usb_ep_autoconfig(gadget, &fs_sink_desc);//根据端点描述符及控制器端点情况,分配一个合适的端点。  
  451.     if (!ep)  
  452.         goto enomem;  
  453.     EP_OUT_NAME = ep->name; //记录名称  
  454.     gcnum = usb_gadget_controller_number(gadget);//获得控制器代号  
  455.     if (gcnum >= 0)  
  456.         device_desc.bcdDevice = cpu_to_le16(0x0200 + gcnum);//赋值设备描述符  
  457.     else   
  458.     {  
  459.         pr_warning("%s: controller '%s' not recognized/n",shortname, gadget->name);  
  460.         device_desc.bcdDevice = __constant_cpu_to_le16(0x9999);  
  461.     }  
  462.     dev = kzalloc(sizeof(*dev), GFP_KERNEL); //分配设备结构体  
  463.     if (!dev)  
  464.         return -ENOMEM;  
  465.     spin_lock_init(&dev->lock);  
  466.     dev->gadget = gadget;  
  467.     set_gadget_data(gadget, dev);  
  468.     dev->req = usb_ep_alloc_request(gadget->ep0, GFP_KERNEL);//分配一个请求  
  469.     if (!dev->req)  
  470.         goto enomem;  
  471.     dev->req->buf = kmalloc(USB_BUFSIZ, GFP_KERNEL);  
  472.     if (!dev->req->buf)  
  473.         goto enomem;  
  474.     dev->req->complete = zero_setup_complete;  
  475.     dev->out_ep=ep; //记录端点(就是接收host端数据的端点)  
  476.     printk("name=%s/n",dev->out_ep->name); //打印出这个端点的名称  
  477.     ep->driver_data=dev;  
  478.     device_desc.bMaxPacketSize0 = gadget->ep0->maxpacket;  
  479.     usb_gadget_set_selfpowered(gadget);  
  480.     gadget->ep0->driver_data = dev;  
  481.     snprintf(manufacturer, sizeof manufacturer, "%s %s with %s",init_utsname()->sysname, init_utsname()->release,gadget->name);  
  482.     /******************* 字符设备注册 ******************/  
  483.     dev_t usb_zero_dev = MKDEV (usb_zero_major, 0);   
  484.     int result = register_chrdev_region (usb_zero_dev, 1, "usb_zero");  
  485.     if (result < 0)  
  486.     {  
  487.         printk (KERN_NOTICE "Unable to get usb_transfer region, error %d/n",result);  
  488.         return 0;  
  489.     }  
  490.     usb_zero_setup_cdev (dev, 0);   
  491.     return 0;  
  492. enomem:  
  493.     zero_unbind(gadget);  
  494.     return -ENOMEM;  
  495. }  
  496. /*-------------------------------------------------------------------------*/  
  497. static struct usb_gadget_driver zero_driver =   
  498. {   
  499.     //gadget驱动的核心数据结构  
  500. #ifdef CONFIG_USB_GADGET_DUALSPEED  
  501.     .speed = USB_SPEED_HIGH,  
  502. #else  
  503.     .speed = USB_SPEED_FULL,  
  504. #endif  
  505.     .function = (char *) longname,  
  506.     .bind = zero_bind,  
  507.     .unbind = __exit_p(zero_unbind),  
  508.     .setup = zero_setup,  
  509.     .disconnect = zero_disconnect,  
  510.     //.suspend = zero_suspend, //不考虑电源管理的功能   
  511.     //.resume = zero_resume,  
  512.     .driver = {  
  513.         .name = (char *) shortname,  
  514.         .owner = THIS_MODULE,  
  515.     },  
  516. };  
  517. MODULE_AUTHOR("David Brownell");  
  518. MODULE_LICENSE("GPL");  
  519. static int __init init(void)  
  520. {  
  521.     return usb_gadget_register_driver(&zero_driver); //注册驱动,调用bind绑定到控制器   
  522. }  
  523. module_init(init);  
  524.   
  525. static void __exit cleanup(void)  
  526. {  
  527.     usb_gadget_unregister_driver(&zero_driver); //注销驱动,通常会调用到unbind解除绑定, //在s3c2410_udc.c中调用的是disconnect方法   
  528. }  
  529. module_exit(cleanup);  
阅读(1525) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~