• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /** @file
2   Debug Port Library implementation based on usb3 debug port.
3 
4   Copyright (c) 2014 - 2015, Intel Corporation. All rights reserved.<BR>
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 #include "DebugCommunicationLibUsb3Internal.h"
15 
16 /**
17   Synchronize the specified transfer ring to update the enqueue and dequeue pointer.
18 
19   @param  Handle      Debug port handle.
20   @param  TrsRing     The transfer ring to sync.
21 
22   @retval EFI_SUCCESS The transfer ring is synchronized successfully.
23 
24 **/
25 EFI_STATUS
26 EFIAPI
XhcSyncTrsRing(IN USB3_DEBUG_PORT_HANDLE * Handle,IN TRANSFER_RING * TrsRing)27 XhcSyncTrsRing (
28   IN USB3_DEBUG_PORT_HANDLE    *Handle,
29   IN TRANSFER_RING             *TrsRing
30   )
31 {
32   UINTN               Index;
33   TRB_TEMPLATE        *TrsTrb;
34   UINT32              CycleBit;
35 
36   ASSERT (TrsRing != NULL);
37 
38   //
39   // Calculate the latest RingEnqueue and RingPCS
40   //
41   TrsTrb = (TRB_TEMPLATE *)(UINTN) TrsRing->RingEnqueue;
42 
43   ASSERT (TrsTrb != NULL);
44 
45   for (Index = 0; Index < TrsRing->TrbNumber; Index++) {
46     if (TrsTrb->CycleBit != (TrsRing->RingPCS & BIT0)) {
47       break;
48     }
49     TrsTrb++;
50     if ((UINT8) TrsTrb->Type == TRB_TYPE_LINK) {
51       ASSERT (((LINK_TRB*)TrsTrb)->TC != 0);
52       //
53       // set cycle bit in Link TRB as normal
54       //
55       ((LINK_TRB*)TrsTrb)->CycleBit = TrsRing->RingPCS & BIT0;
56       //
57       // Toggle PCS maintained by software
58       //
59       TrsRing->RingPCS = (TrsRing->RingPCS & BIT0) ? 0 : 1;
60       TrsTrb           = (TRB_TEMPLATE *)(UINTN)((TrsTrb->Parameter1 | LShiftU64 ((UINT64)TrsTrb->Parameter2, 32)) & ~0x0F);
61     }
62   }
63   ASSERT (Index != TrsRing->TrbNumber);
64 
65   if ((EFI_PHYSICAL_ADDRESS)(UINTN) TrsTrb != TrsRing->RingEnqueue) {
66     TrsRing->RingEnqueue = (EFI_PHYSICAL_ADDRESS)(UINTN) TrsTrb;
67   }
68 
69   //
70   // Clear the Trb context for enqueue, but reserve the PCS bit which indicates free Trb.
71   //
72   CycleBit = TrsTrb->CycleBit;
73   ZeroMem (TrsTrb, sizeof (TRB_TEMPLATE));
74   TrsTrb->CycleBit = CycleBit;
75 
76   return EFI_SUCCESS;
77 }
78 
79 /**
80   Synchronize the specified event ring to update the enqueue and dequeue pointer.
81 
82   @param  Handle      Debug port handle.
83   @param  EvtRing     The event ring to sync.
84 
85   @retval EFI_SUCCESS The event ring is synchronized successfully.
86 
87 **/
88 EFI_STATUS
89 EFIAPI
XhcSyncEventRing(IN USB3_DEBUG_PORT_HANDLE * Handle,IN EVENT_RING * EvtRing)90 XhcSyncEventRing (
91   IN USB3_DEBUG_PORT_HANDLE  *Handle,
92   IN EVENT_RING                *EvtRing
93   )
94 {
95   UINTN               Index;
96   TRB_TEMPLATE        *EvtTrb1;
97 
98   ASSERT (EvtRing != NULL);
99 
100   //
101   // Calculate the EventRingEnqueue and EventRingCCS.
102   // Note: only support single Segment
103   //
104   EvtTrb1 = (TRB_TEMPLATE *)(UINTN) EvtRing->EventRingDequeue;
105 
106   for (Index = 0; Index < EvtRing->TrbNumber; Index++) {
107     if (EvtTrb1->CycleBit != EvtRing->EventRingCCS) {
108       break;
109     }
110 
111     EvtTrb1++;
112 
113     if ((UINTN)EvtTrb1 >= ((UINTN) EvtRing->EventRingSeg0 + sizeof (TRB_TEMPLATE) * EvtRing->TrbNumber)) {
114       EvtTrb1 = (TRB_TEMPLATE *)(UINTN) EvtRing->EventRingSeg0;
115       EvtRing->EventRingCCS = (EvtRing->EventRingCCS) ? 0 : 1;
116     }
117   }
118 
119   if (Index < EvtRing->TrbNumber) {
120     EvtRing->EventRingEnqueue = (EFI_PHYSICAL_ADDRESS)(UINTN)EvtTrb1;
121   } else {
122     ASSERT (FALSE);
123   }
124 
125   return EFI_SUCCESS;
126 }
127 
128 /**
129   Check if there is a new generated event.
130 
131   @param  Handle        Debug port handle.
132   @param  EvtRing       The event ring to check.
133   @param  NewEvtTrb     The new event TRB found.
134 
135   @retval EFI_SUCCESS   Found a new event TRB at the event ring.
136   @retval EFI_NOT_READY The event ring has no new event.
137 
138 **/
139 EFI_STATUS
140 EFIAPI
XhcCheckNewEvent(IN USB3_DEBUG_PORT_HANDLE * Handle,IN EVENT_RING * EvtRing,OUT TRB_TEMPLATE ** NewEvtTrb)141 XhcCheckNewEvent (
142   IN  USB3_DEBUG_PORT_HANDLE   *Handle,
143   IN  EVENT_RING               *EvtRing,
144   OUT TRB_TEMPLATE             **NewEvtTrb
145   )
146 {
147   EFI_STATUS          Status;
148   TRB_TEMPLATE        *EvtTrb;
149 
150   ASSERT (EvtRing != NULL);
151 
152   EvtTrb     = (TRB_TEMPLATE *)(UINTN) EvtRing->EventRingDequeue;
153   *NewEvtTrb = (TRB_TEMPLATE *)(UINTN) EvtRing->EventRingDequeue;
154 
155   if (EvtRing->EventRingDequeue == EvtRing->EventRingEnqueue) {
156     return EFI_NOT_READY;
157   }
158 
159   Status = EFI_SUCCESS;
160 
161   EvtRing->EventRingDequeue += sizeof (TRB_TEMPLATE);
162   //
163   // If the dequeue pointer is beyond the ring, then roll-back it to the begining of the ring.
164   //
165   if ((UINTN)EvtRing->EventRingDequeue >= ((UINTN) EvtRing->EventRingSeg0 + sizeof (TRB_TEMPLATE) * EvtRing->TrbNumber)) {
166     EvtRing->EventRingDequeue = EvtRing->EventRingSeg0;
167   }
168 
169   return Status;
170 }
171 
172 /**
173   Check if the Trb is a transaction of the URB.
174 
175   @param Ring   The transfer ring to be checked.
176   @param Trb    The TRB to be checked.
177 
178   @retval TRUE  It is a transaction of the URB.
179   @retval FALSE It is not any transaction of the URB.
180 
181 **/
182 BOOLEAN
IsTrbInTrsRing(IN TRANSFER_RING * Ring,IN TRB_TEMPLATE * Trb)183 IsTrbInTrsRing (
184   IN  TRANSFER_RING       *Ring,
185   IN  TRB_TEMPLATE        *Trb
186   )
187 {
188   TRB_TEMPLATE  *CheckedTrb;
189   UINTN         Index;
190 
191   CheckedTrb = (TRB_TEMPLATE *)(UINTN) Ring->RingSeg0;
192 
193   ASSERT (Ring->TrbNumber == TR_RING_TRB_NUMBER);
194 
195   for (Index = 0; Index < Ring->TrbNumber; Index++) {
196     if (Trb == CheckedTrb) {
197       return TRUE;
198     }
199     CheckedTrb++;
200   }
201 
202   return FALSE;
203 }
204 
205 /**
206   Check the URB's execution result and update the URB's
207   result accordingly.
208 
209   @param  Handle          Debug port handle.
210   @param  Urb             The URB to check result.
211 
212 **/
213 VOID
XhcCheckUrbResult(IN USB3_DEBUG_PORT_HANDLE * Handle,IN URB * Urb)214 XhcCheckUrbResult (
215   IN  USB3_DEBUG_PORT_HANDLE *Handle,
216   IN  URB                      *Urb
217   )
218 {
219   EVT_TRB_TRANSFER        *EvtTrb;
220   TRB_TEMPLATE            *TRBPtr;
221   UINTN                   Index;
222   EFI_STATUS              Status;
223   URB                     *CheckedUrb;
224   UINT64                  XhcDequeue;
225   UINT32                  High;
226   UINT32                  Low;
227 
228   ASSERT ((Handle != NULL) && (Urb != NULL));
229 
230   if (Urb->Finished) {
231     goto EXIT;
232   }
233 
234   EvtTrb = NULL;
235 
236   //
237   // Traverse the event ring to find out all new events from the previous check.
238   //
239   XhcSyncEventRing (Handle, &Handle->EventRing);
240 
241   for (Index = 0; Index < Handle->EventRing.TrbNumber; Index++) {
242 
243     Status = XhcCheckNewEvent (Handle, &Handle->EventRing, ((TRB_TEMPLATE **)&EvtTrb));
244     if (Status == EFI_NOT_READY) {
245       //
246       // All new events are handled, return directly.
247       //
248       goto EXIT;
249     }
250 
251     if ((EvtTrb->Type != TRB_TYPE_COMMAND_COMPLT_EVENT) && (EvtTrb->Type != TRB_TYPE_TRANS_EVENT)) {
252       continue;
253     }
254 
255     TRBPtr = (TRB_TEMPLATE *)(UINTN)(EvtTrb->TRBPtrLo | LShiftU64 ((UINT64) EvtTrb->TRBPtrHi, 32));
256 
257     if (IsTrbInTrsRing ((TRANSFER_RING *)(UINTN)(Urb->Ring), TRBPtr)) {
258       CheckedUrb = Urb;
259     } else if (IsTrbInTrsRing ((TRANSFER_RING *)(UINTN)(Handle->UrbIn.Ring), TRBPtr)) {
260       //
261       // If it is read event and it should be generated by poll, and current operation is write, we need save data into internal buffer.
262       // Internal buffer is used by next read.
263       //
264       Handle->DataCount = (UINT8) (Handle->UrbIn.DataLen - EvtTrb->Length);
265       CopyMem ((VOID *)(UINTN)Handle->Data, (VOID *)(UINTN)Handle->UrbIn.Data, Handle->DataCount);
266       //
267       // Fill this TRB complete with CycleBit, otherwise next read will fail with old TRB.
268       //
269       TRBPtr->CycleBit = (TRBPtr->CycleBit & BIT0) ? 0 : 1;
270       continue;
271     } else {
272       continue;
273     }
274 
275     if ((EvtTrb->Completecode == TRB_COMPLETION_SHORT_PACKET) ||
276         (EvtTrb->Completecode == TRB_COMPLETION_SUCCESS)) {
277       //
278       // The length of data which were transferred.
279       //
280       CheckedUrb->Completed += (((TRANSFER_TRB_NORMAL*)TRBPtr)->Length - EvtTrb->Length);
281     } else {
282       CheckedUrb->Result  |= EFI_USB_ERR_TIMEOUT;
283     }
284     //
285     // This Urb has been processed
286     //
287     CheckedUrb->Finished = TRUE;
288   }
289 
290 EXIT:
291   //
292   // Advance event ring to last available entry
293   //
294   // Some 3rd party XHCI external cards don't support single 64-bytes width register access,
295   // So divide it to two 32-bytes width register access.
296   //
297   Low  = XhcReadDebugReg (Handle, XHC_DC_DCERDP);
298   High = XhcReadDebugReg (Handle, XHC_DC_DCERDP + 4);
299   XhcDequeue = (UINT64)(LShiftU64((UINT64)High, 32) | Low);
300 
301   if ((XhcDequeue & (~0x0F)) != ((UINT64)(UINTN)Handle->EventRing.EventRingDequeue & (~0x0F))) {
302     //
303     // Some 3rd party XHCI external cards don't support single 64-bytes width register access,
304     // So divide it to two 32-bytes width register access.
305     //
306     XhcWriteDebugReg (Handle, XHC_DC_DCERDP, XHC_LOW_32BIT (Handle->EventRing.EventRingDequeue));
307     XhcWriteDebugReg (Handle, XHC_DC_DCERDP + 4, XHC_HIGH_32BIT (Handle->EventRing.EventRingDequeue));
308   }
309 }
310 
311 /**
312   Ring the door bell to notify XHCI there is a transaction to be executed.
313 
314   @param  Handle        Debug port handle.
315   @param  Urb           The pointer to URB.
316 
317   @retval EFI_SUCCESS   Successfully ring the door bell.
318 
319 **/
320 EFI_STATUS
321 EFIAPI
XhcRingDoorBell(IN USB3_DEBUG_PORT_HANDLE * Handle,IN URB * Urb)322 XhcRingDoorBell (
323   IN USB3_DEBUG_PORT_HANDLE    *Handle,
324   IN URB                       *Urb
325   )
326 {
327   UINT32      Dcdb;
328 
329   //
330   // 7.6.8.2 DCDB Register
331   //
332   Dcdb = (Urb->Direction == EfiUsbDataIn) ? 0x100 : 0x0;
333 
334   XhcWriteDebugReg (
335     Handle,
336     XHC_DC_DCDB,
337     Dcdb
338     );
339 
340   return EFI_SUCCESS;
341 }
342 
343 /**
344   Execute the transfer by polling the URB. This is a synchronous operation.
345 
346   @param  Handle            Debug port handle.
347   @param  Urb               The URB to execute.
348   @param  Timeout           The time to wait before abort, in microsecond.
349 
350 **/
351 VOID
XhcExecTransfer(IN USB3_DEBUG_PORT_HANDLE * Handle,IN URB * Urb,IN UINTN Timeout)352 XhcExecTransfer (
353   IN  USB3_DEBUG_PORT_HANDLE   *Handle,
354   IN  URB                      *Urb,
355   IN  UINTN                    Timeout
356   )
357 {
358   TRANSFER_RING           *Ring;
359   TRB_TEMPLATE            *Trb;
360   UINTN                   Loop;
361   UINTN                   Index;
362 
363   Loop = Timeout / XHC_DEBUG_PORT_1_MILLISECOND;
364   if (Timeout == 0) {
365     Loop = 0xFFFFFFFF;
366   }
367   XhcRingDoorBell (Handle, Urb);
368   //
369   // Event Ring Not Empty bit can only be set to 1 by XHC after ringing door bell with some delay.
370   //
371   for (Index = 0; Index < Loop; Index++) {
372     XhcCheckUrbResult (Handle, Urb);
373     if (Urb->Finished) {
374       break;
375     }
376     MicroSecondDelay (XHC_DEBUG_PORT_1_MILLISECOND);
377   }
378   if (Index == Loop) {
379     //
380     // If time out occurs.
381     //
382     Urb->Result |= EFI_USB_ERR_TIMEOUT;
383   }
384   //
385   // If URB transfer is error, restore transfer ring to original value before URB transfer
386   // This will make the current transfer TRB is always at the latest unused one in transfer ring.
387   //
388   Ring = (TRANSFER_RING *)(UINTN) Urb->Ring;
389   if ((Urb->Result != EFI_USB_NOERROR) && (Urb->Direction == EfiUsbDataIn)) {
390     //
391     // Adjust Enqueue pointer
392     //
393     Ring->RingEnqueue = Urb->Trb;
394     //
395     // Clear CCS flag for next use
396     //
397     Trb = (TRB_TEMPLATE *)(UINTN) Urb->Trb;
398     Trb->CycleBit = ((~Ring->RingPCS) & BIT0);
399   } else {
400     //
401     // Update transfer ring for next transfer.
402     //
403     XhcSyncTrsRing (Handle, Ring);
404   }
405 }
406 
407 /**
408   Create a transfer TRB.
409 
410   @param  Handle  Debug port handle.
411   @param  Urb     The urb used to construct the transfer TRB.
412 
413   @return Created TRB or NULL
414 
415 **/
416 EFI_STATUS
XhcCreateTransferTrb(IN USB3_DEBUG_PORT_HANDLE * Handle,IN URB * Urb)417 XhcCreateTransferTrb (
418   IN USB3_DEBUG_PORT_HANDLE   *Handle,
419   IN URB                        *Urb
420   )
421 {
422   TRANSFER_RING                 *EPRing;
423   TRB                           *Trb;
424 
425   if (Urb->Direction == EfiUsbDataIn) {
426     EPRing = &Handle->TransferRingIn;
427   } else {
428     EPRing = &Handle->TransferRingOut;
429   }
430 
431   Urb->Ring = (EFI_PHYSICAL_ADDRESS)(UINTN) EPRing;
432   XhcSyncTrsRing (Handle, EPRing);
433 
434   Urb->Trb = EPRing->RingEnqueue;
435   Trb = (TRB *)(UINTN)EPRing->RingEnqueue;
436   Trb->TrbNormal.TRBPtrLo  = XHC_LOW_32BIT (Urb->Data);
437   Trb->TrbNormal.TRBPtrHi  = XHC_HIGH_32BIT (Urb->Data);
438   Trb->TrbNormal.Length    = Urb->DataLen;
439   Trb->TrbNormal.TDSize    = 0;
440   Trb->TrbNormal.IntTarget = 0;
441   Trb->TrbNormal.ISP       = 1;
442   Trb->TrbNormal.IOC       = 1;
443   Trb->TrbNormal.Type      = TRB_TYPE_NORMAL;
444 
445   //
446   // Update the cycle bit to indicate this TRB has been consumed.
447   //
448   Trb->TrbNormal.CycleBit = EPRing->RingPCS & BIT0;
449 
450   return EFI_SUCCESS;
451 }
452 
453 /**
454   Create a new URB for a new transaction.
455 
456   @param  Handle     Debug port handle.
457   @param  Direction  The direction of data flow.
458   @param  Data       The user data to transfer
459   @param  DataLen    The length of data buffer
460 
461   @return Created URB or NULL
462 
463 **/
464 URB*
XhcCreateUrb(IN USB3_DEBUG_PORT_HANDLE * Handle,IN EFI_USB_DATA_DIRECTION Direction,IN VOID * Data,IN UINTN DataLen)465 XhcCreateUrb (
466   IN USB3_DEBUG_PORT_HANDLE             *Handle,
467   IN EFI_USB_DATA_DIRECTION             Direction,
468   IN VOID                               *Data,
469   IN UINTN                              DataLen
470   )
471 {
472   EFI_STATUS                    Status;
473   URB                           *Urb;
474   EFI_PHYSICAL_ADDRESS          UrbData;
475 
476   if (Direction == EfiUsbDataIn) {
477     Urb = &Handle->UrbIn;
478   } else {
479     Urb = &Handle->UrbOut;
480   }
481 
482   UrbData  = Urb->Data;
483 
484   ZeroMem (Urb, sizeof (URB));
485   Urb->Direction = Direction;
486 
487   //
488   // Allocate memory to move data from CAR or SMRAM to normal memory
489   // to make XHCI DMA successfully
490   // re-use the pre-allocate buffer in PEI to avoid DXE memory service or gBS are not ready
491   //
492   Urb->Data  = UrbData;
493 
494   if (Direction == EfiUsbDataIn) {
495     //
496     // Do not break URB data in buffer as it may contain the data which were just put in via DMA by XHC
497     //
498     Urb->DataLen  = (UINT32) DataLen;
499   } else {
500     //
501     // Put data into URB data out buffer which will create TRBs
502     //
503     ZeroMem ((VOID*)(UINTN) Urb->Data, DataLen);
504     CopyMem ((VOID*)(UINTN) Urb->Data, Data, DataLen);
505     Urb->DataLen  = (UINT32) DataLen;
506   }
507 
508   Status = XhcCreateTransferTrb (Handle, Urb);
509   ASSERT_EFI_ERROR (Status);
510 
511   return Urb;
512 }
513 
514 /**
515   Submits bulk transfer to a bulk endpoint of a USB device.
516 
517   @param  Handle                Debug port handle.
518   @param  Direction             The direction of data transfer.
519   @param  Data                  Array of pointers to the buffers of data to transmit
520                                 from or receive into.
521   @param  DataLength            The lenght of the data buffer.
522   @param  Timeout               Indicates the maximum time, in microsecond, which
523                                 the transfer is allowed to complete.
524 
525   @retval EFI_SUCCESS           The transfer was completed successfully.
526   @retval EFI_OUT_OF_RESOURCES  The transfer failed due to lack of resource.
527   @retval EFI_INVALID_PARAMETER Some parameters are invalid.
528   @retval EFI_TIMEOUT           The transfer failed due to timeout.
529   @retval EFI_DEVICE_ERROR      The transfer failed due to host controller error.
530 
531 **/
532 EFI_STATUS
533 EFIAPI
XhcDataTransfer(IN USB3_DEBUG_PORT_HANDLE * Handle,IN EFI_USB_DATA_DIRECTION Direction,IN OUT VOID * Data,IN OUT UINTN * DataLength,IN UINTN Timeout)534 XhcDataTransfer (
535   IN     USB3_DEBUG_PORT_HANDLE              *Handle,
536   IN     EFI_USB_DATA_DIRECTION              Direction,
537   IN OUT VOID                                *Data,
538   IN OUT UINTN                               *DataLength,
539   IN     UINTN                               Timeout
540   )
541 {
542   URB                     *Urb;
543   EFI_STATUS              Status;
544 
545   //
546   // Validate the parameters
547   //
548   if ((DataLength == NULL) || (*DataLength == 0) || (Data == NULL)) {
549     return EFI_INVALID_PARAMETER;
550   }
551 
552   //
553   // Create a new URB, insert it into the asynchronous
554   // schedule list, then poll the execution status.
555   //
556   Urb = XhcCreateUrb (Handle, Direction, Data, *DataLength);
557   ASSERT (Urb != NULL);
558 
559   XhcExecTransfer (Handle, Urb, Timeout);
560 
561   *DataLength     = Urb->Completed;
562 
563   Status = EFI_TIMEOUT;
564   if (Urb->Result == EFI_USB_NOERROR) {
565     Status = EFI_SUCCESS;
566   }
567 
568   if (Direction == EfiUsbDataIn) {
569     //
570     // Move data from internal buffer to outside buffer (outside buffer may be in SMRAM...)
571     // SMRAM does not allow to do DMA, so we create an internal buffer.
572     //
573     CopyMem (Data, (VOID *)(UINTN)Urb->Data, *DataLength);
574   }
575 
576   return Status;
577 }
578 
579