• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /** @file
2 
3   Copyright (c) 2013-2015, ARM Ltd. All rights reserved.<BR>
4 
5   This program and the accompanying materials
6   are licensed and made available under the terms and conditions of the BSD License
7   which accompanies this distribution.  The full text of the license may be found at
8   http://opensource.org/licenses/bsd-license.php
9 
10   THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
11   WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
12 
13 **/
14 
15 #include <Library/DebugLib.h>
16 #include <Library/UefiBootServicesTableLib.h>
17 #include <Library/UefiDriverEntryPoint.h>
18 #include <Library/IoLib.h>
19 #include <Library/MemoryAllocationLib.h>
20 
21 #include <IndustryStandard/Usb.h>
22 
23 #include <Protocol/UsbDevice.h>
24 
25 #include "Isp1761UsbDxe.h"
26 
27 /*
28   Driver for using the NXP ISP1761 as a USB Peripheral controller.
29   Doesn't use USB OTG - just sets it in Pure Peripheral mode.
30 
31   The ISP1582 datasheet has a little more info on the Peripheral controller
32   registers than the ISP1761 datasheet
33 
34   We don't do string descriptors. They're optional.
35   We currently assume the device has one configuration, one interface, one IN
36   endpoint, and one OUT endpoint (plus the default control endpoint).
37 
38   In fact, this driver is the minimum required to implement fastboot.
39 */
40 
41 // TODO Make sure the controller isn't sending empty packets when it shouldn't
42 // (check behaviour in cases when Buffer Length isn't explcitly set)
43 
44 // ISP1582 Datasheet:
45 // "Data transfers preceding the status stage must first be fully
46 // completed before the STATUS bit can be set."
47 // This variable stores whether some control data has been pended in the EP0TX
48 // Tx buffer, so that when an EP0TX interrupt is received we can set the STATUS
49 // bit to go to the Status stage of the control transfer.
50 STATIC BOOLEAN mControlTxPending = FALSE;
51 
52 STATIC USB_DEVICE_DESCRIPTOR    *mDeviceDescriptor;
53 
54 // The config descriptor, interface descriptor, and endpoint descriptors in a
55 // buffer (in that order)
56 STATIC VOID                     *mDescriptors;
57 // Convenience pointers to those descriptors inside the buffer:
58 STATIC USB_INTERFACE_DESCRIPTOR *mInterfaceDescriptor;
59 STATIC USB_CONFIG_DESCRIPTOR    *mConfigDescriptor;
60 STATIC USB_ENDPOINT_DESCRIPTOR  *mEndpointDescriptors;
61 
62 STATIC USB_DEVICE_RX_CALLBACK   mDataReceivedCallback;
63 STATIC USB_DEVICE_TX_CALLBACK   mDataSentCallback;
64 
65 // The time between interrupt polls, in units of 100 nanoseconds
66 // 10 Microseconds
67 #define ISP1761_INTERRUPT_POLL_PERIOD 10000
68 
69 STATIC
70 VOID
SelectEndpoint(IN UINT8 Endpoint)71 SelectEndpoint (
72   IN UINT8 Endpoint
73   )
74 {
75   // The DMA Endpoint Index must not point to the same as the
76   // Endpoint Index Register.
77   WRITE_REG32 (ISP1761_DMA_ENDPOINT_INDEX, ((Endpoint + 2) % ISP1761_NUM_ENDPOINTS));
78   WRITE_REG32 (ISP1761_ENDPOINT_INDEX, Endpoint);
79 }
80 
81 // Enable going to the Data stage of a control transfer
82 STATIC
83 VOID
DataStageEnable(IN UINT8 Endpoint)84 DataStageEnable (
85   IN UINT8 Endpoint
86   )
87 {
88   SelectEndpoint (Endpoint);
89   WRITE_REG32 (ISP1761_CTRL_FUNCTION, ISP1761_CTRL_FUNCTION_DSEN);
90 }
91 
92 // Go to the Status stage of a successful control transfer
93 STATIC
94 VOID
StatusAcknowledge(IN UINT8 Endpoint)95 StatusAcknowledge (
96   IN UINT8 Endpoint
97 )
98 {
99   SelectEndpoint (Endpoint);
100   WRITE_REG32 (ISP1761_CTRL_FUNCTION, ISP1761_CTRL_FUNCTION_STATUS);
101 }
102 
103 // Read the FIFO for the endpoint indexed by Endpoint, into the buffer pointed
104 // at by Buffer, whose size is *Size bytes.
105 //
106 // If *Size is less than the number of bytes in the FIFO, return EFI_BUFFER_TOO_SMALL
107 //
108 // Update *Size with the number of bytes of data in the FIFO.
109 STATIC
110 EFI_STATUS
ReadEndpointBuffer(IN UINT8 Endpoint,IN OUT UINTN * Size,IN OUT VOID * Buffer)111 ReadEndpointBuffer (
112   IN      UINT8   Endpoint,
113   IN OUT  UINTN  *Size,
114   IN OUT  VOID   *Buffer
115   )
116 {
117   UINT16  NumBytesAvailable;
118   UINT32  Val32;
119   UINTN   Index;
120   UINTN   NumBytesRead;
121 
122   SelectEndpoint (Endpoint);
123 
124   NumBytesAvailable = READ_REG16 (ISP1761_BUFFER_LENGTH);
125 
126   if (NumBytesAvailable > *Size) {
127     *Size = NumBytesAvailable;
128     return EFI_BUFFER_TOO_SMALL;
129   }
130   *Size = NumBytesAvailable;
131 
132   /* -- NB! --
133     The datasheet says the Data Port is 16 bits but it actually appears to
134     be 32 bits.
135    */
136 
137   // Read 32-bit chunks
138   for (Index = 0; Index < NumBytesAvailable / 4; Index++) {
139     ((UINT32 *) Buffer)[Index] = READ_REG32 (ISP1761_DATA_PORT);
140   }
141 
142   // Read remaining bytes
143 
144   // Round NumBytesAvailable down to nearest power of 4
145   NumBytesRead = NumBytesAvailable & (~0x3);
146   if (NumBytesRead != NumBytesAvailable) {
147     Val32 = READ_REG32 (ISP1761_DATA_PORT);
148     // Copy each required byte of 32-bit word into buffer
149     for (Index = 0; Index < NumBytesAvailable % 4; Index++) {
150       ((UINT8 *) Buffer)[NumBytesRead + Index] = Val32 >> (Index * 8);
151     }
152   }
153   return EFI_SUCCESS;
154 }
155 
156 /*
157   Write an endpoint buffer. Parameters:
158   Endpoint        Endpoint index (see Endpoint Index Register in datasheet)
159   MaxPacketSize   The MaxPacketSize this endpoint is configured for
160   Size            The size of the Buffer
161   Buffer          The data
162 
163   Assumes MaxPacketSize is a multiple of 4.
164   (It seems that all valid values for MaxPacketSize _are_ multiples of 4)
165 */
166 STATIC
167 EFI_STATUS
WriteEndpointBuffer(IN UINT8 Endpoint,IN UINTN MaxPacketSize,IN UINTN Size,IN CONST VOID * Buffer)168 WriteEndpointBuffer (
169   IN       UINT8   Endpoint,
170   IN       UINTN   MaxPacketSize,
171   IN       UINTN   Size,
172   IN CONST VOID   *Buffer
173   )
174 {
175   UINTN    Index;
176   UINT32  *DwordBuffer;
177 
178   DwordBuffer = (UINT32 *) Buffer;
179   SelectEndpoint (Endpoint);
180 
181   /* -- NB! --
182     The datasheet says the Data Port is 16 bits but it actually appears to
183     be 32 bits.
184    */
185 
186   // Write packets of size MaxPacketSize
187   while (Size > MaxPacketSize) {
188     for (Index = 0; Index < MaxPacketSize / 4; Index++) {
189       WRITE_REG32 (ISP1761_DATA_PORT, DwordBuffer[Index]);
190     }
191     Size -= MaxPacketSize;
192     DwordBuffer += (MaxPacketSize / sizeof (UINT32));
193   }
194 
195   // Write remaining data
196 
197   if (Size > 0) {
198     WRITE_REG32 (ISP1761_BUFFER_LENGTH, Size);
199 
200     while (Size > 4) {
201       WRITE_REG32 (ISP1761_DATA_PORT, DwordBuffer[0]);
202       Size -= 4;
203       DwordBuffer++;
204     }
205 
206     if (Size > 0) {
207       WRITE_REG32 (ISP1761_DATA_PORT, DwordBuffer[0]);
208     }
209   }
210 
211   return EFI_SUCCESS;
212 }
213 
214 STATIC
215 EFI_STATUS
HandleGetDescriptor(IN USB_DEVICE_REQUEST * Request)216 HandleGetDescriptor (
217   IN USB_DEVICE_REQUEST  *Request
218   )
219 {
220   EFI_STATUS  Status;
221   UINT8       DescriptorType;
222   UINTN       ResponseSize;
223   VOID       *ResponseData;
224 
225   ResponseSize = 0;
226   ResponseData = NULL;
227   Status = EFI_SUCCESS;
228 
229   // Pretty confused if bmRequestType is anything but this:
230   ASSERT (Request->RequestType == USB_DEV_GET_DESCRIPTOR_REQ_TYPE);
231 
232   // Choose the response
233   DescriptorType = Request->Value >> 8;
234   switch (DescriptorType) {
235   case USB_DESC_TYPE_DEVICE:
236     DEBUG ((EFI_D_INFO, "USB: Got a request for device descriptor\n"));
237     ResponseSize = sizeof (USB_DEVICE_DESCRIPTOR);
238     ResponseData = mDeviceDescriptor;
239     break;
240   case USB_DESC_TYPE_CONFIG:
241     DEBUG ((EFI_D_INFO, "USB: Got a request for config descriptor\n"));
242     ResponseSize = mConfigDescriptor->TotalLength;
243     ResponseData = mDescriptors;
244     break;
245   case USB_DESC_TYPE_STRING:
246     DEBUG ((EFI_D_INFO, "USB: Got a request for String descriptor %d\n", Request->Value & 0xFF));
247     break;
248   default:
249     DEBUG ((EFI_D_INFO, "USB: Didn't understand request for descriptor 0x%04x\n", Request->Value));
250     Status = EFI_NOT_FOUND;
251     break;
252   }
253 
254   // Send the response
255   if (ResponseData) {
256     ASSERT (ResponseSize != 0);
257 
258     if (Request->Length < ResponseSize) {
259       // Truncate response
260       ResponseSize = Request->Length;
261     } else if (Request->Length > ResponseSize) {
262       DEBUG ((EFI_D_INFO, "USB: Info: ResponseSize < wLength\n"));
263     }
264 
265     DataStageEnable (ISP1761_EP0TX);
266     Status = WriteEndpointBuffer (
267               ISP1761_EP0TX,
268               MAX_PACKET_SIZE_CONTROL,
269               ResponseSize,
270               ResponseData
271               );
272     if (!EFI_ERROR (Status)) {
273       // Setting this value should cause us to go to the Status stage on the
274       // next EP0TX interrupt
275       mControlTxPending = TRUE;
276     }
277   }
278 
279   return EFI_SUCCESS;
280 }
281 
282 STATIC
283 EFI_STATUS
HandleSetAddress(IN USB_DEVICE_REQUEST * Request)284 HandleSetAddress (
285   IN USB_DEVICE_REQUEST  *Request
286   )
287 {
288   // Pretty confused if bmRequestType is anything but this:
289   ASSERT (Request->RequestType == USB_DEV_SET_ADDRESS_REQ_TYPE);
290   // USB Spec: "The USB device does not change its device address until after
291   // the Status stage of this request is completed successfully."
292   // ISP1582 datasheet: "The new device address is activated when the
293   // device receives an acknowledgment from the host for the empty packet
294   // token". (StatusAcknowledge causes an empty packet to be sent).
295   // So, we write the Address register _before_ acking the SET_ADDRESS.
296   DEBUG ((EFI_D_INFO, "USB: Setting address to %d\n", Request->Value));
297   WRITE_REG32 (ISP1761_ADDRESS, Request->Value | ISP1761_ADDRESS_DEVEN);
298   StatusAcknowledge (ISP1761_EP0TX);
299 
300   return EFI_SUCCESS;
301 }
302 
303 // Move the device to the Configured state.
304 // (This code only supports one configuration for a device, so the configuration
305 //  index is ignored)
306 STATIC
307 EFI_STATUS
HandleSetConfiguration(IN USB_DEVICE_REQUEST * Request)308 HandleSetConfiguration (
309   IN USB_DEVICE_REQUEST  *Request
310   )
311 {
312   USB_ENDPOINT_DESCRIPTOR  *EPDesc;
313   UINTN                     Index;
314   UINT8                     EndpointIndex;
315 
316   ASSERT (Request->RequestType == USB_DEV_SET_CONFIGURATION_REQ_TYPE);
317   DEBUG ((EFI_D_INFO, "USB: Setting configuration.\n"));
318 
319   // Configure endpoints
320   for (Index = 0; Index < mInterfaceDescriptor->NumEndpoints; Index++) {
321     EPDesc = &mEndpointDescriptors[Index];
322 
323     // To simplify for now, assume endpoints aren't "sparse", and are in order.
324     ASSERT ((EPDesc->EndpointAddress & 0xF) == ((Index / 2) + 1));
325 
326     // Convert from USB endpoint index to ISP1761 endpoint Index
327     // USB:     Endpoint number is bits [3:0], IN/OUT is bit [7]
328     // ISP1761: Endpoint number is bits [4:1], IN/OUT is bit [0]
329     EndpointIndex = ((EPDesc->EndpointAddress & 0xF) << 1) |
330                     ((EPDesc->EndpointAddress & BIT7) >> 7);
331     SelectEndpoint (EndpointIndex);
332     // Set endpoint type (Bulk/Isochronous/Interrupt)
333     WRITE_REG32 (ISP1761_ENDPOINT_MAX_PACKET_SIZE, EPDesc->MaxPacketSize);
334     // Hardware foible (bug?): Although the datasheet seems to suggest it should
335     // automatically be set to MaxPacketSize, the Buffer Length register appears
336     // to be reset to 0, which causes an empty packet to be sent in response to
337     // the first IN token of the session. The NOEMPKT field of the Endpoint Type
338     // register sounds like it might fix this problem, but it doesn't
339     // (it's "applicable only in the DMA mode").
340     WRITE_REG32 (ISP1761_BUFFER_LENGTH, EPDesc->MaxPacketSize);
341     WRITE_REG32 (ISP1761_ENDPOINT_TYPE, (EPDesc->Attributes & 0x3) |
342                                         ISP1761_ENDPOINT_TYPE_ENABLE);
343   }
344 
345   StatusAcknowledge (ISP1761_EP0TX);
346   return EFI_SUCCESS;
347 }
348 
349 STATIC
350 EFI_STATUS
HandleDeviceRequest(IN USB_DEVICE_REQUEST * Request)351 HandleDeviceRequest (
352   IN USB_DEVICE_REQUEST  *Request
353   )
354 {
355   EFI_STATUS  Status;
356 
357   Status = EFI_SUCCESS;
358 
359   switch (Request->Request) {
360   case USB_DEV_GET_DESCRIPTOR:
361     Status = HandleGetDescriptor (Request);
362     break;
363   case USB_DEV_SET_ADDRESS:
364     Status = HandleSetAddress (Request);
365     break;
366   case USB_DEV_SET_CONFIGURATION:
367     Status = HandleSetConfiguration (Request);
368     break;
369   default:
370     DEBUG ((EFI_D_ERROR,
371       "Didn't understand RequestType 0x%x Request 0x%x\n",
372       Request->RequestType, Request->Request));
373       Status = EFI_INVALID_PARAMETER;
374     break;
375   }
376 
377   return Status;
378 }
379 
380 // Instead of actually registering interrupt handlers, we poll the controller's
381 //  interrupt source register in this function.
382 STATIC
383 VOID
CheckInterrupts(IN EFI_EVENT Event,IN VOID * Context)384 CheckInterrupts (
385   IN EFI_EVENT  Event,
386   IN VOID      *Context
387   )
388 {
389   UINT32      DcInterrupts;
390   UINTN       NumBytes;
391   UINTN       MoreBytes;
392   UINT8       Packet[512];
393   VOID       *DataPacket;
394   UINT32      HandledInterrupts;
395   UINT32      UnhandledInterrupts;
396   EFI_STATUS  Status;
397 
398   // Set bits in HandledInterrupts to mark the interrupt source handled.
399   HandledInterrupts = 0;
400 
401   WRITE_REG32 (ISP1761_DEVICE_UNLOCK, ISP1761_DEVICE_UNLOCK_MAGIC);
402 
403   DcInterrupts = READ_REG32 (ISP1761_DC_INTERRUPT);
404   if (DcInterrupts & ISP1761_DC_INTERRUPT_SUSP) {
405     DEBUG ((EFI_D_INFO, "USB: Suspend\n"));
406     HandledInterrupts |= ISP1761_DC_INTERRUPT_SUSP;
407   }
408   if (DcInterrupts & ISP1761_DC_INTERRUPT_RESUME) {
409     DEBUG ((EFI_D_INFO, "USB: Resume\n"));
410     HandledInterrupts |= ISP1761_DC_INTERRUPT_RESUME;
411   }
412   if (DcInterrupts & ISP1761_DC_INTERRUPT_EP0SETUP) {
413     NumBytes = 512;
414     ReadEndpointBuffer (0x20, &NumBytes, &Packet);
415     ASSERT (NumBytes == 8);
416     HandleDeviceRequest ((USB_DEVICE_REQUEST *) Packet);
417     HandledInterrupts |= ISP1761_DC_INTERRUPT_EP0SETUP;
418   }
419   if (DcInterrupts & ISP1761_DC_INTERRUPT_EP0RX) {
420     HandledInterrupts |= ISP1761_DC_INTERRUPT_EP0RX;
421   }
422   if (DcInterrupts & ISP1761_DC_INTERRUPT_EP0TX) {
423     if (mControlTxPending) {
424       // We previously put some data in the Control Endpoint's IN (Tx) FIFO.
425       // We assume that that data has now been sent in response to the IN token
426       // that triggered this interrupt. We can therefore go to the Status stage
427       // of the control transfer.
428       StatusAcknowledge (ISP1761_EP0TX);
429       mControlTxPending = FALSE;
430     }
431     HandledInterrupts |= ISP1761_DC_INTERRUPT_EP0TX;
432   }
433   if (DcInterrupts & ISP1761_DC_INTERRUPT_EP1RX) {
434     NumBytes = 512;
435     DataPacket = AllocatePool (NumBytes);
436     Status = ReadEndpointBuffer (ISP1761_EP1RX, &NumBytes, DataPacket);
437     if (EFI_ERROR (Status) || NumBytes == 0) {
438       if (EFI_ERROR (Status)) {
439         DEBUG ((EFI_D_ERROR, "Couldn't read EP1RX data: %r\n", Status));
440       }
441       FreePool (DataPacket);
442     } else {
443       // Signal this event again so we poll again ASAP
444       gBS->SignalEvent (Event);
445       mDataReceivedCallback (NumBytes, DataPacket);
446     }
447     HandledInterrupts |= ISP1761_DC_INTERRUPT_EP1RX;
448   }
449   if (DcInterrupts & ISP1761_DC_INTERRUPT_EP1TX) {
450     mDataSentCallback (1);
451     HandledInterrupts |= ISP1761_DC_INTERRUPT_EP1TX;
452   }
453   if (DcInterrupts & (ISP1761_DC_INTERRUPT_SOF | ISP1761_DC_INTERRUPT_PSOF)) {
454     // Don't care about SOFs or pseudo-SOFs
455     HandledInterrupts |= (ISP1761_DC_INTERRUPT_SOF | ISP1761_DC_INTERRUPT_PSOF);
456   }
457   if (ISP1761_DC_INTERRUPT_BRESET) {
458     HandledInterrupts |= ISP1761_DC_INTERRUPT_BRESET;
459   }
460   if (ISP1761_DC_INTERRUPT_HS_STAT) {
461     HandledInterrupts |= ISP1761_DC_INTERRUPT_HS_STAT;
462   }
463   if (ISP1761_DC_INTERRUPT_VBUS) {
464     HandledInterrupts |= ISP1761_DC_INTERRUPT_VBUS;
465   }
466 
467   UnhandledInterrupts = DcInterrupts & (~HandledInterrupts) & ISP1761_DC_INTERRUPT_MASK;
468   if (UnhandledInterrupts) {
469     DEBUG ((EFI_D_ERROR, "USB: Unhandled DC Interrupts: 0x%08x\n",
470       UnhandledInterrupts));
471   }
472 
473   // Check if we received any more data while we were handling the interrupt.
474   SelectEndpoint (ISP1761_EP1RX);
475   MoreBytes = READ_REG16 (ISP1761_BUFFER_LENGTH);
476   if (MoreBytes) {
477     HandledInterrupts &= ~ISP1761_DC_INTERRUPT_EP1RX;
478   }
479 
480   WRITE_REG32 (ISP1761_DC_INTERRUPT, HandledInterrupts);
481 }
482 
483 EFI_STATUS
Isp1761PeriphSend(IN UINT8 EndpointIndex,IN UINTN Size,IN CONST VOID * Buffer)484 Isp1761PeriphSend (
485   IN        UINT8  EndpointIndex,
486   IN        UINTN  Size,
487   IN  CONST VOID  *Buffer
488   )
489 {
490   return WriteEndpointBuffer (
491           (EndpointIndex << 1) | 0x1, //Convert to ISP1761 endpoint index, Tx
492           MAX_PACKET_SIZE_BULK,
493           Size,
494           Buffer
495           );
496 }
497 
498 EFI_STATUS
499 EFIAPI
Isp1761PeriphStart(IN USB_DEVICE_DESCRIPTOR * DeviceDescriptor,IN VOID ** Descriptors,IN USB_DEVICE_RX_CALLBACK RxCallback,IN USB_DEVICE_TX_CALLBACK TxCallback)500 Isp1761PeriphStart (
501   IN USB_DEVICE_DESCRIPTOR   *DeviceDescriptor,
502   IN VOID                   **Descriptors,
503   IN USB_DEVICE_RX_CALLBACK   RxCallback,
504   IN USB_DEVICE_TX_CALLBACK   TxCallback
505   )
506 {
507   UINT16                    OtgStatus;
508   UINT8                    *Ptr;
509   EFI_STATUS                Status;
510   EFI_EVENT                 TimerEvent;
511 
512   ASSERT (DeviceDescriptor != NULL);
513   ASSERT (Descriptors[0] != NULL);
514   ASSERT (RxCallback != NULL);
515   ASSERT (TxCallback != NULL);
516 
517   WRITE_REG32 (ISP1761_DEVICE_UNLOCK, ISP1761_DEVICE_UNLOCK_MAGIC);
518 
519   WRITE_REG32 (ISP1761_SW_RESET_REG, ISP1761_SW_RESET_ALL);
520   while (READ_REG32 (ISP1761_SW_RESET_REG) & ISP1761_SW_RESET_ALL) {
521     //busy wait
522   }
523   WRITE_REG32 (ISP1761_MODE, ISP1761_MODE_SFRESET);
524   while (READ_REG32 (ISP1761_MODE) & ISP1761_MODE_SFRESET) {
525     //busy wait
526   }
527   DEBUG ((EFI_D_INFO, "USB: Software reset done\n"));
528 
529   WRITE_REG32 (ISP1761_DC_INTERRUPT_ENABLE, 0x03FFFFFF);
530   WRITE_REG32 (ISP1761_OTG_INTERRUPT_ENABLE_RISE, 0x07FF);
531 
532   WRITE_REG8 (ISP1761_ADDRESS, ISP1761_ADDRESS_DEVEN);
533   WRITE_REG8 (ISP1761_MODE, ISP1761_MODE_WKUPCS | ISP1761_MODE_CLKAON);
534 
535   // Use port 1 as peripheral controller (magic - disagrees with datasheet)
536   WRITE_REG32 (ISP1761_OTG_CTRL_SET, 0xffff0000);
537   WRITE_REG32 (ISP1761_OTG_CTRL_SET, 0x000014d1);
538 
539   OtgStatus = READ_REG16 (ISP1761_OTG_STATUS);
540   if ((OtgStatus & ISP1761_OTG_STATUS_B_SESS_END) != 0) {
541     DEBUG ((EFI_D_ERROR, "USB: Vbus not powered.\n"));
542   }
543   if ((OtgStatus & ISP1761_OTG_STATUS_A_B_SESS_VLD) == 0) {
544     DEBUG ((EFI_D_ERROR, "USB: Session not valid.\n"));
545   }
546 
547   // Configure Control endpoints
548   SelectEndpoint (0x20);
549   WRITE_REG32 (ISP1761_ENDPOINT_MAX_PACKET_SIZE, MAX_PACKET_SIZE_CONTROL);
550   WRITE_REG32 (ISP1761_ENDPOINT_TYPE, ISP1761_ENDPOINT_TYPE_ENABLE);
551   SelectEndpoint (0x0);
552   WRITE_REG32 (ISP1761_ENDPOINT_MAX_PACKET_SIZE, MAX_PACKET_SIZE_CONTROL);
553   WRITE_REG32 (ISP1761_ENDPOINT_TYPE, ISP1761_ENDPOINT_TYPE_ENABLE);
554   SelectEndpoint (0x1);
555   WRITE_REG32 (ISP1761_ENDPOINT_MAX_PACKET_SIZE, MAX_PACKET_SIZE_CONTROL);
556   WRITE_REG32 (ISP1761_ENDPOINT_TYPE, ISP1761_ENDPOINT_TYPE_ENABLE);
557 
558   // Interrupt on all ACK and NAK
559   WRITE_REG32 (ISP1761_INTERRUPT_CONFIG, ISP1761_INTERRUPT_CONFIG_ACK_ONLY);
560 
561   mDeviceDescriptor = DeviceDescriptor;
562   mDescriptors = Descriptors[0];
563 
564   // Right now we just support one configuration
565   ASSERT (mDeviceDescriptor->NumConfigurations == 1);
566   // ... and one interface
567   mConfigDescriptor = (USB_CONFIG_DESCRIPTOR *)mDescriptors;
568   ASSERT (mConfigDescriptor->NumInterfaces == 1);
569 
570   Ptr = ((UINT8 *) mDescriptors) + sizeof (USB_CONFIG_DESCRIPTOR);
571   mInterfaceDescriptor = (USB_INTERFACE_DESCRIPTOR *) Ptr;
572   Ptr += sizeof (USB_INTERFACE_DESCRIPTOR);
573 
574   mEndpointDescriptors = (USB_ENDPOINT_DESCRIPTOR *) Ptr;
575 
576   mDataReceivedCallback = RxCallback;
577   mDataSentCallback = TxCallback;
578 
579   // Register a timer event so CheckInterupts gets called periodically
580   Status = gBS->CreateEvent (
581                   EVT_TIMER | EVT_NOTIFY_SIGNAL,
582                   TPL_CALLBACK,
583                   CheckInterrupts,
584                   NULL,
585                   &TimerEvent
586                   );
587   ASSERT_EFI_ERROR (Status);
588   if (EFI_ERROR (Status)) {
589     return Status;
590   }
591 
592   Status = gBS->SetTimer (
593                   TimerEvent,
594                   TimerPeriodic,
595                   ISP1761_INTERRUPT_POLL_PERIOD
596                   );
597   ASSERT_EFI_ERROR (Status);
598 
599   return Status;
600 }
601 
602 USB_DEVICE_PROTOCOL mUsbDevice = {
603   Isp1761PeriphStart,
604   Isp1761PeriphSend
605 };
606 
607 
608 EFI_STATUS
609 EFIAPI
Isp1761PeriphEntryPoint(IN EFI_HANDLE ImageHandle,IN EFI_SYSTEM_TABLE * SystemTable)610 Isp1761PeriphEntryPoint (
611   IN EFI_HANDLE                            ImageHandle,
612   IN EFI_SYSTEM_TABLE                      *SystemTable
613   )
614 {
615   UINT32      DeviceId;
616   EFI_HANDLE  Handle;
617 
618   DeviceId = READ_REG32 (ISP1761_DEVICE_ID);
619 
620   if (DeviceId != ISP1761_DEVICE_ID_VAL) {
621     DEBUG ((EFI_D_ERROR,
622       "ERROR: Read incorrect device ID for ISP1761: 0x%08x, expected 0x%08x\n",
623       DeviceId , ISP1761_DEVICE_ID_VAL
624       ));
625     return EFI_DEVICE_ERROR;
626   }
627 
628   Handle = NULL;
629   return gBS->InstallProtocolInterface (
630     &Handle,
631     &gUsbDeviceProtocolGuid,
632     EFI_NATIVE_INTERFACE,
633     &mUsbDevice
634     );
635 }
636