uhci.txt
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:7k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. Specification and Internals for the New UHCI Driver (Whitepaper...)
  2. brought to you by
  3. Georg Acher, acher@in.tum.de (executive slave) (base guitar)
  4. Deti Fliegl, deti@fliegl.de (executive slave) (lead voice)
  5. Thomas Sailer, sailer@ife.ee.ethz.ch (chief consultant) (cheer leader)
  6.  
  7. $Id: README.uhci,v 1.1 1999/12/14 14:03:02 fliegl Exp $
  8. This document and the new uhci sources can be found on
  9. http://hotswap.in.tum.de/usb
  10. 1. General issues
  11. 1.1 Why a new UHCI driver, we already have one?!?
  12. Correct, but its internal structure got more and more mixed up by the (still
  13. ongoing) efforts to get isochronous transfers (ISO) to work.
  14. Since there is an increasing need for reliable ISO-transfers (especially 
  15. for USB-audio needed by TS and for a DAB-USB-Receiver build by GA and DF), 
  16. this state was a bit unsatisfying in our opinion, so we've decided (based
  17. on knowledge and experiences with the old UHCI driver) to start 
  18. from scratch with a new approach, much simpler but at the same time more 
  19. powerful.
  20. It is inspired by the way Win98/Win2000 handles USB requests via URBs,
  21. but it's definitely 100% free of MS-code and doesn't crash while 
  22. unplugging an used ISO-device like Win98 ;-)
  23. Some code for HW setup and root hub management was taken from the 
  24. original UHCI driver, but heavily modified to fit into the new code.
  25. The invention of the basic concept, and major coding were completed in two 
  26. days (and nights) on the 16th and 17th of October 1999, now known as the
  27. great USB-October-Revolution started by GA, DF, and TS ;-)
  28. Since the concept is in no way UHCI dependent, we hope that it will also be 
  29. transferred to the OHCI-driver, so both drivers share a common API.
  30. 1.2. Advantages and disadvantages
  31. + All USB transfer types work now!
  32. + Asynchronous operation
  33. + Simple, but powerful interface (only two calls for start and cancel)
  34. + Easy migration to the new API, simplified by a compatibility API
  35. + Simple usage of ISO transfers
  36. + Automatic linking of requests
  37. + ISO transfers allow variable length for each frame and striping
  38. + No CPU dependent and non-portable atomic memory access, no asm()-inlines
  39. + Tested on x86 and Alpha
  40. - Rewriting for ISO transfers needed
  41. 1.3. Is there some compatibility to the old API?
  42. Yes, but only for control, bulk and interrupt transfers. We've implemented 
  43. some wrapper calls for these transfer types. The usbcore works fine with 
  44. these wrappers. For ISO there's no compatibility, because the old ISO-API 
  45. and its semantics were unnecessary complicated in our opinion.
  46. 1.4. What's really working?
  47. As said above, CTRL and BULK already work fine even with the wrappers,
  48. so legacy code wouldn't notice the change.
  49. Regarding to Thomas, ISO transfers now run stable with USB audio.
  50. INT transfers (e.g. mouse driver) work fine, too.
  51. 1.5. Are there any bugs?
  52. No ;-)
  53. Hm...
  54. Well, of course this implementation needs extensive testing on all available
  55. hardware, but we believe that any fixes shouldn't harm the overall concept.
  56. 1.6. What should be done next?
  57. A large part of the request handling seems to be identical for UHCI and 
  58. OHCI, so it would be a good idea to extract the common parts and have only 
  59. the HW specific stuff in uhci.c. Furthermore, all other USB device drivers
  60. should need URBification, if they use isochronous or interrupt transfers.
  61. One thing missing in the current implementation (and the old UHCI driver) 
  62. is fair queueing for BULK transfers. Since this would need (in principle) 
  63. the alteration of already constructed TD chains (to switch from depth to 
  64. breadth execution), another way has to be found. Maybe some simple 
  65. heuristics work with the same effect.
  66. ---------------------------------------------------------------------------
  67. 2. Internal structure and mechanisms
  68. To get quickly familiar with the internal structures, here's a short
  69. description how the new UHCI driver works. However, the ultimate source of
  70. truth is only uhci.c!
  71. 2.1. Descriptor structure (QHs and TDs)
  72. During initialization, the following skeleton is allocated in init_skel:
  73.  framespecific           |           common chain     
  74. framelist[]
  75. [  0 ]-----> TD --> TD -------
  76. [  1 ]-----> TD --> TD --------> TD ----> QH -------> QH -------> QH ---> NULL
  77.   ...        TD --> TD -------/
  78. [1023]-----> TD --> TD ------/
  79.      
  80.      ^^     ^^           ^^       ^^          ^^          ^^
  81.    1024 TDs for   7 TDs for    1 TD for   Start of    Start of    End Chain
  82.     ISO  INT (2-128ms) 1ms-INT    CTRL Chain  BULK Chain
  83. For each CTRL or BULK transfer a new QH is allocated and the containing data
  84. transfers are appended as (vertical) TDs. After building the whole QH with its
  85. dangling TDs, the QH is inserted before the BULK Chain QH (for CTRL) or
  86. before the End Chain QH (for BULK). Since only the QH->next pointers are
  87. affected, no atomic memory operation is required. The three QHs in the
  88. common chain are never equipped with TDs!
  89. For ISO or INT, the TD for each frame is simply inserted into the appropriate
  90. ISO/INT-TD-chain for the desired frame. The 7 skeleton INT-TDs are scattered
  91. among the 1024 frames similar to the old UHCI driver.
  92. For CTRL/BULK/ISO, the last TD in the transfer has the IOC-bit set. For INT,
  93. every TD (there is only one...) has the IOC-bit set.
  94. Besides the data for the UHCI controller (2 or 4 32bit words), the descriptors
  95. are double-linked through the .vertical and .horizontal elements in the 
  96. SW data of the descriptor (using the double-linked list structures and 
  97. operations), but SW-linking occurs only in closed domains, i.e. for each of
  98. the 1024 ISO-chains and the 8 INT-chains there is a closed cycle. This 
  99. simplifies all insertions and unlinking operations and avoids costly 
  100. bus_to_virt()-calls.
  101. 2.2. URB structure and linking to QH/TDs
  102. During assembly of the QH and TDs of the requested action, these descriptors
  103. are stored in urb->urb_list, so the allocated QH/TD descriptors are bound to
  104. this URB.
  105. If the assembly was successful and the descriptors were added to the HW chain,
  106. the corresponding URB is inserted into a global URB list for this controller.
  107. This list stores all pending URBs.
  108. 2.3. Interrupt processing
  109. Since UHCI provides no means to directly detect completed transactions, the
  110. following is done in each UHCI interrupt (uhci_interrupt()):
  111. For each URB in the pending queue (process_urb()), the ACTIVE-flag of the 
  112. associated TDs are processed (depending on the transfer type 
  113. process_{transfer|interrupt|iso}()). If the TDs are not active anymore, 
  114. they indicate the completion of the transaction and the status is calculated. 
  115. Inactive QH/TDs are removed from the HW chain (since the host controller
  116. already removed the TDs from the QH, no atomic access is needed) and 
  117. eventually the URB is marked as completed (OK or errors) and removed from the 
  118. pending queue. Then the next linked URB is submitted. After (or immediately 
  119. before) that, the completion handler is called.
  120. 2.4. Unlinking URBs
  121. First, all QH/TDs stored in the URB are unlinked from the HW chain. 
  122. To ensure that the host controller really left a vertical TD chain, we 
  123. wait for one frame. After that, the TDs are physically destroyed.
  124. 2.5. URB linking and the consequences
  125. Since URBs can be linked and the corresponding submit_urb is called in
  126. the UHCI-interrupt, all work associated with URB/QH/TD assembly has to be
  127. interrupt save. This forces kmalloc to use GFP_ATOMIC in the interrupt.