Linux gadget驅動分析2------設備識別過程

設備連上主機以後,設備驅動作了的事.c#

設備連上host的port以後,主機端會有一套策略發送請求獲取device的一系列描述符.進行枚舉過程.找到適合該device的驅動. 這樣就能夠與device進行通訊.數據結構

 

usb device這邊收到主機的請求,會進入中斷處理函數.能夠說usb device 控制器的活動都是從中斷開始的.函數

#號之間的都是udc驅動中作的事,大部分與硬件相關,原理應該差很少.oop

########################nuc900_udc##########################this

nuc9xx系列的中斷處理函數:spa

 1 static irqreturn_t nuc900_udc_irq(int irq, void *_dev)
 2 {
 3         struct nuc900_udc *dev;
 4         struct nuc900_ep *ep;
 5         u32 volatile IrqStL, IrqEnL;
 6         u32 volatile  IrqSt, IrqEn;
 7         int i=0, j;
 8 
 9         dev=(struct nuc900_udc *)(_dev);
10 
11         IrqStL = __raw_readl(controller.reg + REG_USBD_IRQ_STAT_L);    /* 0x000 register get interrupt status */
12         IrqEnL = __raw_readl(controller.reg + REG_USBD_IRQ_ENB_L);
13 
14         IrqStL = IrqStL & IrqEnL ;
15         if (!IrqStL) {
16                 printk("Not our interrupt !\n");
17                 return IRQ_HANDLED;
18         }
19 
20         if (IrqStL & IRQ_USB_STAT) {
21                 IrqSt = __raw_readl(controller.reg + REG_USBD_IRQ_STAT);
22                 IrqEn = __raw_readl(controller.reg + REG_USBD_IRQ_ENB);
23                 __raw_writel(IrqSt, controller.reg + REG_USBD_IRQ_STAT);
24 
25                 IrqSt = IrqSt & IrqEn ;
26 
27                 if (IrqSt && dev->driver) {
28                         for (i=0; i<6; i++) {
29                                 if (IrqSt&(1<<i)) {
30                                         paser_irq_stat(1<<i,dev);
31                                         break;
32                                 }
33                         }
34                 }
35 
36         }//end IRQ_USB_STAT
37 
38 
39         if (IrqStL & IRQ_CEP) {
40                 IrqSt = __raw_readl(controller.reg + REG_USBD_CEP_IRQ_STAT);
41                 IrqEn = __raw_readl(controller.reg + REG_USBD_CEP_IRQ_ENB);
42                 //printk("cep:%x, %x\n", IrqSt, IrqEn);
43                 IrqSt = IrqSt & IrqEn ;
44                 __raw_writel(IrqSt, controller.reg + REG_USBD_CEP_IRQ_STAT);
45 
46                 if (IrqSt && dev->driver) {
47                         //for(i=12;i>=0;i--)
48                         if (IrqSt&CEP_STS_END) { //deal with STS END
49                                 if (dev->ep0state == EP0_OUT_DATA_PHASE)
50                                         IrqSt &= 0x1BF7;
51                                 paser_irq_cep(CEP_STS_END,dev,IrqSt);
52                         }
53                         for (i=0; i<13; i++) {
54                                 if (i == 10)
55                                         continue;
56                                 if (IrqSt&(1<<i)) {
57                                         paser_irq_cep(1<<i,dev,IrqSt);
58                                         //break;
59                                 }
60                         }
61                 }
62         }
63 
64         if (IrqStL & IRQ_NCEP) {
65                 IrqStL >>= 2;
66 
67                 for (j = 0; j < 6; j++) { //6 endpoints
68                         if (IrqStL & (1 << j)) {
69                                 //in-token and out token interrupt can deal with one only
70                                 IrqSt = __raw_readl(controller.reg + REG_USBD_EPA_IRQ_STAT + 0x28 * j);
71                                 IrqEn = __raw_readl(controller.reg + REG_USBD_EPA_IRQ_ENB + 0x28 * j);
72 
73                                 IrqSt = IrqSt & IrqEn ;
74                                 if (IrqSt && dev->driver) {
75                                         ep = &dev->ep[j+1];
76 
77                                         for (i=12; i>=0; i--) {
78                                                 if (IrqSt&(1<<i)) {
79                                                         if ((1<<i) == EP_BO_SHORT_PKT)
80                                                                 IrqSt &= 0x1FCF;//clear out token/RxED intr
81                                                         if ((ep->EP_Type == EP_TYPE_BLK) || (ep->EP_Type == EP_TYPE_ISO))
82                                                                 paser_irq_nep(1<<i, ep, IrqSt);
83                                                         else if (ep->EP_Type == EP_TYPE_INT)
84                                                                 paser_irq_nepint(1<<i, ep, IrqSt);
85                                                         break;
86                                                 }
87                                         }
88                                 }
89                         }
90                 }
91         }//if end
92 
93         return IRQ_HANDLED;
94 
95 
96 }

基本原理就是從一些相關寄存器中讀取出中斷的類型,分別進入相應的處理流程.code

20行  if (IrqStL & IRQ_USB_STAT) {blog

  就是usb Suspend Resume 等中斷髮生時會進入該iftoken

39行  if (IrqStL & IRQ_CEP) {ip

  中斷是控制端口中斷,進入該if語句.

  usb host會經過控制端口獲取一系列描述符. 因此device這邊會先進入這裏.

  而後再讀取中斷的子類型

  進入      51或57行的                            

   paser_irq_cep(1<<i,dev,IrqSt);

  

  1 void paser_irq_cep(int irq, struct nuc900_udc *dev, u32 IrqSt)
  2 {
  3         struct nuc900_ep    *ep = &dev->ep[0];
  4         struct nuc900_request    *req;
  5         int        is_last=1;
  6 
  7         if (list_empty(&ep->queue)) {
  8                 req = 0;
  9         } else {
 10                 req = list_entry(ep->queue.next, struct nuc900_request, queue);
 11         }
 12 
 13         switch (irq) {
 14         case CEP_SUPPKT://receive setup packet
 15                 dev->ep0state=EP0_IDLE;
 16                 dev->setup_ret = 0;
 17 
 18                 udc_isr_ctrl_pkt(dev);
 19                 break;
 20 
 21         case CEP_DATA_RXD:
 22 
 23 
 24                 if (dev->ep0state == EP0_OUT_DATA_PHASE) {
 25                         if (req)
 26                                 is_last = read_fifo(ep,req, 0);
 27 
 28                         __raw_writel(0x400, controller.reg + REG_USBD_CEP_IRQ_STAT);
 29 
 30                         if (!is_last)
 31                                 __raw_writel(0x440, controller.reg + REG_USBD_CEP_IRQ_ENB);//enable out token and status complete int
 32                         else { //transfer finished
 33                                 __raw_writel(0x04C, controller.reg + REG_USBD_CEP_IRQ_STAT);
 34                                 __raw_writel(CEP_NAK_CLEAR, controller.reg + REG_USBD_CEP_CTRL_STAT);    // clear nak so that sts stage is complete
 35                                 __raw_writel(0x400, controller.reg + REG_USBD_CEP_IRQ_ENB);        // suppkt int//enb sts completion int
 36                                 dev->ep0state = EP0_END_XFER;
 37                         }
 38                 }
 39 
 40                 return;
 41 
 42         case CEP_IN_TOK:
 43 
 44                 if ((IrqSt & CEP_STS_END))
 45                         dev->ep0state=EP0_IDLE;
 46 
 47                 if (dev->setup_ret < 0) { // == -EOPNOTSUPP)
 48                         printk("CEP send zero pkt\n");
 49                         __raw_writel(CEP_ZEROLEN, controller.reg + REG_USBD_CEP_CTRL_STAT);
 50                         __raw_writel(0x400, controller.reg + REG_USBD_CEP_IRQ_ENB);        //enb sts completion int
 51                 }
 52 
 53                 else if (dev->ep0state == EP0_IN_DATA_PHASE) {
 54 
 55 
 56                         if (req) {
 57                                 is_last = write_fifo(ep,req);
 58                         }
 59 
 60 
 61                         if (!is_last)
 62                                 __raw_writel(0x408, controller.reg + REG_USBD_CEP_IRQ_ENB);
 63                         else {
 64                                 if (dev->setup_ret >= 0)
 65                                         __raw_writel(CEP_NAK_CLEAR, controller.reg + REG_USBD_CEP_CTRL_STAT);    // clear nak so that sts stage is complete
 66                                 __raw_writel(0x402, controller.reg + REG_USBD_CEP_IRQ_ENB);        // suppkt int//enb sts completion int
 67 
 68                                 if (dev->setup_ret < 0)//== -EOPNOTSUPP)
 69                                         dev->ep0state=EP0_IDLE;
 70                                 else if (dev->ep0state != EP0_IDLE)
 71                                         dev->ep0state=EP0_END_XFER;
 72                         }
 73                 }
 74 
 75                 return;
 76 
 77         case CEP_PING_TOK:
 78 
 79                 __raw_writel(0x402, controller.reg + REG_USBD_CEP_IRQ_ENB);        // suppkt int//enb sts completion int
 80                 return;
 81 
 82         case CEP_DATA_TXD:
 83                 return;
 84 
 85         case CEP_STS_END:
 86 
 87                 __raw_writel(0x4A, controller.reg + REG_USBD_CEP_IRQ_ENB);
 88                 udc_isr_update_dev(dev);
 89                 dev->ep0state=EP0_IDLE;
 90                 dev->setup_ret = 0;
 91 
 92                 break;
 93 
 94         default:
 95                 break;
 96 
 97         }
 98 
 99         return ;
100 
101 }

很簡單的switch語句,device控制器硬件會根據host發來的包類型設置相應寄存器. 中斷處理函數再從該寄存器中得到類型,進入不一樣的switch分支.

 

host請求設備描述符是usb控制傳輸,並經過device的控制端口進行,接下去會進入哪一個case分支跟控制傳輸的流程有關

大致上是 :

1,host發送 setup 到device         .device收到後準備相關data

2,host發送 in      到device     device收到後發送data到host

3,host 發送 out   到device       host 把狀態數據發到device

詳細流程可參考網上文章.

case setup中

18行  udc_isr_ctrl_pkt(dev);

  1 static void udc_isr_ctrl_pkt(struct nuc900_udc *dev)
  2 {
  3         u32    temp;
  4         u32    ReqErr=0;
  5         struct nuc900_ep *ep = &dev->ep[0];
  6         struct usb_ctrlrequest    crq;
  7         struct nuc900_request    *req;
  8         int ret;
  9 
 10         if (list_empty(&ep->queue)) {
 11                 //printk("ctrl ep->queue is empty\n");
 12                 req = 0;
 13         } else {
 14                 req = list_entry(ep->queue.next, struct nuc900_request, queue);
 15                 //printk("req = %x\n", req);
 16         }
 17 
 18         temp = __raw_readl(controller.reg + REG_USBD_SETUP1_0);
 19 
 20         Get_SetupPacket(&crq,temp);
 21 
 22         dev->crq = crq;
 23 
 24         switch (dev->ep0state) {
 25         case EP0_IDLE:
 26                 switch (crq.bRequest) {
 27 
 28                 case USBR_SET_ADDRESS:
 29                         ReqErr = ((crq.bRequestType == 0) && ((crq.wValue & 0xff00) == 0)
 30                                   && (crq.wIndex == 0) && (crq.wLength == 0)) ? 0 : 1;
 31 
 32                         if ((crq.wValue & 0xffff) > 0x7f) {    //within 7f
 33                                 ReqErr=1;    //Devaddr > 127
 34                         }
 35 
 36                         if (dev->usb_devstate == 3) {
 37                                 ReqErr=1;    //Dev is configured
 38                         }
 39 
 40                         if (ReqErr==1) {
 41                                 break;        //break this switch loop
 42                         }
 43 
 44                         if (dev->usb_devstate == 2) {
 45                                 if (crq.wValue == 0)
 46                                         dev->usb_devstate = 1;        //enter default state
 47                                 dev->usb_address = crq.wValue;    //if wval !=0,use new address
 48                         }
 49 
 50                         if (dev->usb_devstate == 1) {
 51                                 if (crq.wValue != 0) {
 52                                         dev->usb_address = crq.wValue;
 53                                         dev->usb_devstate = 2;
 54                                 }
 55                         }
 56 
 57                         break;
 58 
 59                 case USBR_SET_CONFIGURATION:
 60                         ReqErr = ((crq.bRequestType == 0) && ((crq.wValue & 0xff00) == 0) &&
 61                                   ((crq.wValue & 0x80) == 0) && (crq.wIndex == 0) &&
 62                                   (crq.wLength == 0)) ? 0 : 1;
 63 
 64                         if (dev->usb_devstate == 1) {
 65                                 ReqErr=1;
 66                         }
 67 
 68                         if (ReqErr==1) {
 69                                 break;    //break this switch loop
 70                         }
 71 
 72                         if (crq.wValue == 0)
 73                                 dev->usb_devstate = 2;
 74                         else
 75                                 dev->usb_devstate = 3;
 76                         break;
 77 
 78                 case USBR_SET_INTERFACE:
 79                         ReqErr = ((crq.bRequestType == 0x1) && ((crq.wValue & 0xff80) == 0)
 80                                   && ((crq.wIndex & 0xfff0) == 0) && (crq.wLength == 0)) ? 0 : 1;
 81 
 82                         if (!((dev->usb_devstate == 0x3) && (crq.wIndex == 0x0) && (crq.wValue == 0x0)))
 83                                 ReqErr=1;
 84 
 85                         if (ReqErr == 1) {
 86                                 break;    //break this switch loop
 87                         }
 88 
 89                         break;
 90 
 91                 default:
 92                         break;
 93                 }//switch end
 94 
 95                 if (crq.bRequestType & USB_DIR_IN) {
 96                         dev->ep0state = EP0_IN_DATA_PHASE;
 97                         __raw_writel(0x08, controller.reg + REG_USBD_CEP_IRQ_ENB);
 98                 } else {
 99                         dev->ep0state = EP0_OUT_DATA_PHASE;
100                         __raw_writel(0x40, controller.reg + REG_USBD_CEP_IRQ_ENB);
101                 }
102 
103                 ret = dev->driver->setup(&dev->gadget, &crq);
104                 dev->setup_ret = ret;
105                 if (ret < 0) {
106 
107                         __raw_writel(0x400, controller.reg + REG_USBD_CEP_IRQ_STAT);
108                         __raw_writel(0x448, controller.reg + REG_USBD_CEP_IRQ_ENB);        // enable in/RxED/status complete interrupt
109                         __raw_writel(CEP_NAK_CLEAR, controller.reg + REG_USBD_CEP_CTRL_STAT);    //clear nak so that sts stage is complete
110 
111 
112                         if (ret == -EOPNOTSUPP)
113                                 printk("Operation %x not supported\n", crq.bRequest);
114                         else {
115                                 printk("dev->driver->setup failed. (%d)\n",ret);
116                         }
117                 } else if (ret > 1000) { //DELAYED_STATUS
118                         printk("DELAYED_STATUS:%p\n", req);
119                         dev->ep0state = EP0_END_XFER;
120                         __raw_writel(0, controller.reg + REG_USBD_CEP_IRQ_ENB);
121                 }
122 
123                 break;
124 
125         case EP0_STALL:
126                 break;
127         default:
128                 break;
129         }
130 
131         if (ReqErr == 1) {
132                 __raw_writel(CEP_SEND_STALL, controller.reg + REG_USBD_CEP_CTRL_STAT);
133                 dev->ep0state = EP0_STALL;
134         }
135 
136 }

 

18 20行就是從寄存器拿出來setup 包的數據,放到  struct usb_ctrlrequest    crq;

struct usb_ctrlrequest {
 __u8 bRequestType;
 __u8 bRequest;
 __le16 wValue;
 __le16 wIndex;
 __le16 wLength;
} __attribute__ ((packed));

 這個數據結構與usb協議裏的定義相一致.

下面的操做全是圍繞這個結構進行的.

若是請bRequest是USBR_SET_ADDRESS, USBR_SET_CONFIGURATION,USBR_SET_INTERFACE.就地處理.

若是不是,進入103行的 調用gadget驅動的setup函數.

########################nuc900_udc##########################

 

這個setup就是上一篇

static struct usb_gadget_driver composite_driver = {
 .speed  = USB_SPEED_HIGH,

 .bind  = composite_bind,
 .unbind  = composite_unbind,

 .setup  = composite_setup,
 .disconnect = composite_disconnect,

 .suspend = composite_suspend,
 .resume  = composite_resume,

 .driver = {
  .owner  = THIS_MODULE,
 },
};

中的setup,就是composite_setup

  1 /*
  2  * The setup() callback implements all the ep0 functionality that's
  3  * not handled lower down, in hardware or the hardware driver(like
  4  * device and endpoint feature flags, and their status).  It's all
  5  * housekeeping for the gadget function we're implementing.  Most of
  6  * the work is in config and function specific setup.
  7  */
  8 static int
  9 composite_setup(struct usb_gadget *gadget, const struct usb_ctrlrequest *ctrl)
 10 {
 11     struct usb_composite_dev    *cdev = get_gadget_data(gadget);
 12     struct usb_request        *req = cdev->req;
 13     int                value = -EOPNOTSUPP;
 14     u16                w_index = le16_to_cpu(ctrl->wIndex);
 15     u8                intf = w_index & 0xFF;
 16     u16                w_value = le16_to_cpu(ctrl->wValue);
 17     u16                w_length = le16_to_cpu(ctrl->wLength);
 18     struct usb_function        *f = NULL;
 19     u8                endp;
 20 
 21     /* partial re-init of the response message; the function or the
 22      * gadget might need to intercept e.g. a control-OUT completion
 23      * when we delegate to it.
 24      */
 25     req->zero = 0;
 26     req->complete = composite_setup_complete;
 27     req->length = USB_BUFSIZ;
 28     gadget->ep0->driver_data = cdev;
 29 
 30     switch (ctrl->bRequest) {
 31 
 32     /* we handle all standard USB descriptors */
 33     case USB_REQ_GET_DESCRIPTOR:
 34         if (ctrl->bRequestType != USB_DIR_IN)
 35             goto unknown;
 36         switch (w_value >> 8) {
 37 
 38         case USB_DT_DEVICE:  // 1
 39             cdev->desc.bNumConfigurations =
 40                 count_configs(cdev, USB_DT_DEVICE);
 41             value = min(w_length, (u16) sizeof cdev->desc);
 42             memcpy(req->buf, &cdev->desc, value);
 43             break;
 44         case USB_DT_DEVICE_QUALIFIER:
 45             if (!gadget_is_dualspeed(gadget))
 46                 break;
 47             device_qual(cdev);
 48             value = min_t(int, w_length,
 49                 sizeof(struct usb_qualifier_descriptor));
 50             break;
 51         case USB_DT_OTHER_SPEED_CONFIG:
 52             if (!gadget_is_dualspeed(gadget))
 53                 break;
 54             /* FALLTHROUGH */
 55         case USB_DT_CONFIG: // 2 若干次跟config個數有關
 56             value = config_desc(cdev, w_value);
 57             if (value >= 0)
 58                 value = min(w_length, (u16) value);
 59             break;
 60         case USB_DT_STRING: // 3 若干次跟function個數有關
 61             value = get_string(cdev, req->buf,
 62                     w_index, w_value & 0xff);
 63             if (value >= 0)
 64                 value = min(w_length, (u16) value);
 65             break;
 66         }
 67         break;
 68 
 69     /* any number of configs can work */
 70     case USB_REQ_SET_CONFIGURATION:  // 4 
 71         if (ctrl->bRequestType != 0)
 72             goto unknown;
 73         if (gadget_is_otg(gadget)) {
 74             if (gadget->a_hnp_support)
 75                 DBG(cdev, "HNP available\n");
 76             else if (gadget->a_alt_hnp_support)
 77                 DBG(cdev, "HNP on another port\n");
 78             else
 79                 VDBG(cdev, "HNP inactive\n");
 80         }
 81         spin_lock(&cdev->lock);
 82         value = set_config(cdev, ctrl, w_value);
 83         spin_unlock(&cdev->lock);
 84         break;
 85     case USB_REQ_GET_CONFIGURATION:
 86         if (ctrl->bRequestType != USB_DIR_IN)
 87             goto unknown;
 88         if (cdev->config)
 89             *(u8 *)req->buf = cdev->config->bConfigurationValue;
 90         else
 91             *(u8 *)req->buf = 0;
 92         value = min(w_length, (u16) 1);
 93         break;
 94 
 95     /* function drivers must handle get/set altsetting; if there's
 96      * no get() method, we know only altsetting zero works.
 97      */
 98     case USB_REQ_SET_INTERFACE:
 99         if (ctrl->bRequestType != USB_RECIP_INTERFACE)
100             goto unknown;
101         if (!cdev->config || w_index >= MAX_CONFIG_INTERFACES)
102             break;
103         f = cdev->config->interface[intf];
104         if (!f)
105             break;
106         if (w_value && !f->set_alt)
107             break;
108         value = f->set_alt(f, w_index, w_value);
109         break;
110     case USB_REQ_GET_INTERFACE:
111         if (ctrl->bRequestType != (USB_DIR_IN|USB_RECIP_INTERFACE))
112             goto unknown;
113         if (!cdev->config || w_index >= MAX_CONFIG_INTERFACES)
114             break;
115         f = cdev->config->interface[intf];
116         if (!f)
117             break;
118         /* lots of interfaces only need altsetting zero... */
119         value = f->get_alt ? f->get_alt(f, w_index) : 0;
120         if (value < 0)
121             break;
122         *((u8 *)req->buf) = value;
123         value = min(w_length, (u16) 1);
124         break;
125     default:
126 unknown:
127         VDBG(cdev,
128             "non-core control req%02x.%02x v%04x i%04x l%d\n",
129             ctrl->bRequestType, ctrl->bRequest,
130             w_value, w_index, w_length);
131 
132         /* functions always handle their interfaces and endpoints...
133          * punt other recipients (other, WUSB, ...) to the current
134          * configuration code.
135          *
136          * REVISIT it could make sense to let the composite device
137          * take such requests too, if that's ever needed:  to work
138          * in config 0, etc.
139          */
140         switch (ctrl->bRequestType & USB_RECIP_MASK) {
141         case USB_RECIP_INTERFACE:
142             f = cdev->config->interface[intf];
143             break;
144 
145         case USB_RECIP_ENDPOINT:
146             endp = ((w_index & 0x80) >> 3) | (w_index & 0x0f);
147             list_for_each_entry(f, &cdev->config->functions, list) {
148                 if (test_bit(endp, f->endpoints))
149                     break;
150             }
151             if (&f->list == &cdev->config->functions)
152                 f = NULL;
153             break;
154         }
155 
156         if (f && f->setup)
157             value = f->setup(f, ctrl);
158         else {
159             struct usb_configuration    *c;
160 
161             c = cdev->config;
162             if (c && c->setup)
163                 value = c->setup(c, ctrl);
164         }
165 
166         goto done;
167     }
168 
169     /* respond with data transfer before status phase? */
170     if (value >= 0) {
171         req->length = value;
172         req->zero = value < w_length;
173         value = usb_ep_queue(gadget->ep0, req, GFP_ATOMIC);
174         if (value < 0) {
175             DBG(cdev, "ep_queue --> %d\n", value);
176             req->status = 0;
177             composite_setup_complete(gadget->ep0, req);
178         }
179     }
180 
181 done:
182     /* device either stalls (value < 0) or reports success */
183     return value;
184 }

 這個函數作的就是根據host發來的請求,填充相應的req->buf,而後在最後173行,把req掛到req list上.

 

按照usb控制傳輸,setup階段以後,是datain階段. host會發送intoken包, device接到該包,會進到上面貼的paser_irq_cep函數的42行的case中調用 write_fifo.

write_fifo把數據寫到緩衝區,並把req從req list上拿掉.

這個函數157行調用function的setup,處理特殊的setup請求.

以前的內核沒有composite這一層,每一種gadget設備,都要有一個setup函數,後來的內核把通用的內容都放到了composite.c. 寫gadget 驅動就簡化了些.

設備識別時device這邊是跟着host的請求作相應的動做,函數大致流程就是這樣,細節部分之後再慢慢研究

相關文章
相關標籤/搜索