usbohci.c 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304
  1. /*
  2. * USB Open Host Controller Interface (OHCI) driver
  3. * from Charles Forsyth's devohci.c, 5 Aug 2006.
  4. */
  5. #include "u.h"
  6. #include "../port/lib.h"
  7. #include "mem.h"
  8. #include "dat.h"
  9. #include "fns.h"
  10. #include "io.h"
  11. #include "../port/error.h"
  12. #include "usb.h"
  13. #define XPRINT if(usbhdebug) print
  14. #define XIPRINT if(usbhdebug) iprint
  15. #define XEPRINT if(usbhdebug || ep->debug) print
  16. #define XEIPRINT if(usbhdebug || ep->debug) iprint
  17. #define IPRINT(x) iprint x
  18. static int usbhdebug = 0;
  19. static int dcls;
  20. enum {
  21. Ned = 63 + 32,
  22. Ntd = 256,
  23. };
  24. /*
  25. * USB packet definitions
  26. */
  27. enum {
  28. Otoksetup = 0,
  29. Otokout = 1,
  30. Otokin = 2,
  31. /* port status - UHCI style */
  32. Suspend = 1<<12,
  33. PortReset = 1<<9,
  34. SlowDevice = 1<<8,
  35. ResumeDetect = 1<<6,
  36. PortEnableChange = 1<<3, /* write 1 to clear */
  37. PortEnable = 1<<2,
  38. ConnectStatusChange = 1<<1, /* write 1 to clear */
  39. DevicePresent = 1<<0,
  40. };
  41. typedef struct Ctlr Ctlr;
  42. typedef struct QTree QTree;
  43. enum {
  44. ED_MPS_MASK = 0x7ff,
  45. ED_MPS_SHIFT = 16,
  46. ED_C_MASK = 1,
  47. ED_C_SHIFT = 1,
  48. ED_F_BIT = 1 << 15,
  49. ED_S_MASK = 1,
  50. ED_S_SHIFT = 13,
  51. ED_D_MASK = 3,
  52. ED_D_SHIFT = 11,
  53. ED_H_MASK = 1,
  54. ED_H_SHIFT = 0,
  55. };
  56. typedef struct Endptx Endptx;
  57. typedef struct TD TD;
  58. struct Endptx
  59. {
  60. Lock; /* for manipulating ed */
  61. ED *ed; /* Single endpoint descriptor */
  62. int ntd; /* Number of TDs in use */
  63. int overruns;
  64. };
  65. struct TD {
  66. ulong ctrl;
  67. ulong cbp;
  68. ulong nexttd;
  69. ulong be;
  70. ushort offsets[8]; /* Iso TDs only */
  71. /* driver specific; pad to multiple of 32 */
  72. TD* next;
  73. Endpt *ep;
  74. Block *bp;
  75. ulong flags;
  76. ulong offset; /* offset associated with end of data */
  77. ulong bytes; /* bytes in this TD */
  78. ulong pad[2];
  79. };
  80. enum {
  81. TD_R_SHIFT = 18,
  82. TD_DP_MASK = 3,
  83. TD_DP_SHIFT = 19,
  84. TD_CC_MASK = 0xf,
  85. TD_CC_SHIFT = 28,
  86. TD_EC_MASK = 3,
  87. TD_EC_SHIFT = 26,
  88. TD_FLAGS_LAST = 1 << 0,
  89. };
  90. typedef struct HCCA HCCA;
  91. struct HCCA {
  92. ulong intrtable[32];
  93. ushort framenumber;
  94. ushort pad1;
  95. ulong donehead;
  96. uchar reserved[116];
  97. };
  98. /* OHCI registers */
  99. typedef struct OHCI OHCI;
  100. struct OHCI {
  101. /* control and status group */
  102. /*00*/ ulong revision;
  103. ulong control;
  104. ulong cmdsts;
  105. ulong intrsts;
  106. /*10*/ ulong intrenable;
  107. ulong intrdisable;
  108. /* memory pointer group */
  109. ulong hcca;
  110. ulong periodcurred;
  111. /*20*/ ulong ctlheaded;
  112. ulong ctlcurred;
  113. ulong bulkheaded;
  114. ulong bulkcurred;
  115. /*30*/ ulong donehead;
  116. /* frame counter group */
  117. ulong fminterval;
  118. ulong fmremaining;
  119. ulong fmnumber;
  120. /*40*/ ulong periodicstart;
  121. ulong lsthreshold;
  122. /* root hub group */
  123. ulong rhdesca;
  124. ulong rhdescb;
  125. /*50*/ ulong rhsts;
  126. ulong rhportsts[15];
  127. /*90*/ ulong pad25[20];
  128. /* unknown */
  129. /*e0*/ ulong hostueaddr;
  130. ulong hostuests;
  131. ulong hosttimeoutctrl;
  132. ulong pad59;
  133. /*f0*/ ulong pad60;
  134. ulong hostrevision;
  135. ulong pad62[2];
  136. /*100*/
  137. };
  138. /*
  139. * software structures
  140. */
  141. static struct {
  142. int bit;
  143. char *name;
  144. } portstatus[] = {
  145. { Suspend, "suspend", },
  146. { PortReset, "reset", },
  147. { SlowDevice, "lowspeed", },
  148. { ResumeDetect, "resume", },
  149. { PortEnableChange, "portchange", },
  150. { PortEnable, "enable", },
  151. { ConnectStatusChange, "statuschange", },
  152. { DevicePresent, "present", },
  153. };
  154. struct QTree {
  155. QLock;
  156. int nel;
  157. int depth;
  158. ulong* bw;
  159. ED **root;
  160. };
  161. /* device parameters */
  162. static char *devstates[] = {
  163. [Disabled] "Disabled",
  164. [Attached] "Attached",
  165. [Enabled] "Enabled",
  166. };
  167. struct Ctlr {
  168. Lock; /* protects state shared with interrupt (eg, free list) */
  169. int active;
  170. Pcidev* pcidev;
  171. int irq;
  172. ulong tbdf;
  173. Ctlr* next;
  174. int nports;
  175. OHCI *base; /* equiv to io in uhci */
  176. HCCA *uchcca;
  177. int idgen; /* version # to distinguish new connections */
  178. QLock resetl; /* lock controller during USB reset */
  179. struct {
  180. Lock;
  181. TD* pool;
  182. TD* free;
  183. int alloced;
  184. } td;
  185. struct {
  186. QLock;
  187. ED* pool;
  188. ED* free;
  189. int alloced;
  190. } ed;
  191. /* TODO: what happened to ctlq, etc. from uhci? */
  192. QTree* tree; /* tree for t Endpt i/o */
  193. struct {
  194. QLock;
  195. Endpt* f;
  196. } activends;
  197. };
  198. enum {
  199. HcRevision = 0x00,
  200. HcControl = 0x01,
  201. HcfsMask = 3 << 6,
  202. HcfsReset = 0 << 6,
  203. HcfsResume = 1 << 6,
  204. HcfsOperational=2 << 6,
  205. HcfsSuspend = 3 << 6,
  206. Ble = 1 << 5,
  207. Cle = 1 << 4,
  208. Ie = 1 << 3,
  209. Ple = 1 << 2,
  210. Cbsr_MASK = 3,
  211. HcCommandStatus = 0x02,
  212. Ocr = 1 << 3,
  213. Blf = 1 << 2,
  214. Clf = 1 << 1,
  215. Hcr = 1 << 0,
  216. HcIntrStatus = 0x03,
  217. HcIntrEnable = 0x04,
  218. Mie = 1 << 31,
  219. Oc = 1 << 30,
  220. Rhsc = 1 << 6,
  221. Fno = 1 << 5,
  222. Ue = 1 << 4,
  223. Rd = 1 << 3,
  224. Sf = 1 << 2,
  225. Wdh = 1 << 1,
  226. So = 1 << 0,
  227. HcIntrDisable = 0x05,
  228. HcFmIntvl = 0x0d,
  229. HcFmIntvl_FSMaxpack_MASK = 0x7fff,
  230. HcFmIntvl_FSMaxpack_SHIFT = 16,
  231. HcFmRemaining = 0x0e,
  232. HcFmNumber = 0x0f,
  233. HcLSThreshold = 0x11,
  234. HcRhDescA = 0x12,
  235. HcRhDescA_POTPGT_MASK = 0xff << 24,
  236. HcRhDescA_POTPGT_SHIFT = 24,
  237. HcRhDescB = 0x13,
  238. HcRhStatus = 0x14,
  239. Lps = 1 << 0,
  240. Cgp = 1 << 0,
  241. Oci = 1 << 1,
  242. Drwe = 1 << 15,
  243. Srwe = 1 << 15,
  244. LpsC = 1 << 16,
  245. Sgp = 1 << 16,
  246. Ccic = 1 << 17,
  247. Crwe = 1 << 31,
  248. HcRhPortStatus1 = 0x15,
  249. Ccs = 1 << 0,
  250. Cpe = 1 << 0,
  251. Pes = 1 << 1,
  252. Spe = 1 << 1,
  253. Pss = 1 << 2,
  254. Poci = 1 << 3,
  255. Prs = 1 << 4,
  256. Spr = 1 << 4,
  257. Pps = 1 << 8,
  258. Spp= 1 << 8,
  259. Lsda = 1 << 9,
  260. Cpp = 1 << 9,
  261. Csc = 1 << 16,
  262. Pesc = 1 << 17,
  263. Pssc = 1 << 18,
  264. Ocic = 1 << 19,
  265. Prsc = 1 << 20,
  266. HcRhPortStatus2 = 0x16,
  267. L2NFRAME = 5,
  268. NFRAME = 1 << L2NFRAME,
  269. /* TODO: from UHCI; correct for OHCI? */
  270. FRAMESIZE = NFRAME*sizeof(ulong), /* fixed by hardware; aligned to same */
  271. };
  272. static char *modename[] = {
  273. [Ctlmode]= "Ctl",
  274. [Bulkmode] = "Bulk",
  275. [Intrmode] = "Intr",
  276. [Isomode] = "Iso",
  277. };
  278. static char *omodename[] = {
  279. [OREAD] = "r",
  280. [OWRITE] = "w",
  281. [ORDWR] = "rw",
  282. };
  283. int ohciinterrupts[Nmodes];
  284. static Ctlr* ctlrhead;
  285. static Ctlr* ctlrtail;
  286. static char Estalled[] = "usb endpoint stalled";
  287. static char EnotWritten[] = "usb write unfinished";
  288. static char EnotRead[] = "usb read unfinished";
  289. static char Eunderrun[] = "usb endpoint underrun";
  290. static QLock usbhstate; /* protects name space state */
  291. static void eptactivate(Ctlr *ub, Endpt *ep);
  292. static void eptdeactivate(Ctlr *ub, Endpt *e);
  293. static long read (Usbhost *, Endpt*, void*, long, vlong);
  294. static void scanpci(void);
  295. static int schedendpt(Ctlr *ub, Endpt *ep, int direction);
  296. static void unschedendpt(Ctlr *ub, Endpt *ep, int);
  297. static long write(Usbhost *, Endpt*, void*, long, vlong, int);
  298. static long qtd(Ctlr*, Endpt*, int, Block*, uchar*, uchar*, int, ulong);
  299. static short
  300. refcnt(Block *b, int i)
  301. {
  302. short v;
  303. static Lock l;
  304. ilock(&l);
  305. v = (b->flag += i);
  306. iunlock(&l);
  307. if(v < 0)
  308. iprint("refcnt 0x%lux %d\n", b, v);
  309. return v;
  310. }
  311. static void
  312. freewb(Block *b)
  313. {
  314. if(b == nil || refcnt(b, -1) > 0)
  315. return;
  316. if(b->base > b->rp || b->rp > b->wp || b->wp > b->lim)
  317. iprint("freebw: %lux %lux %lux %lux\n",
  318. b->base, b->rp, b->wp, b->lim);
  319. /* poison the block in case someone is still holding onto it */
  320. b->next = (Block*)0xdeadcafe;
  321. b->rp = (uchar*)0xdeadcafe;
  322. b->wp = (uchar*)0xdeadcafe;
  323. b->lim = (uchar*)0xdeadcafe;
  324. b->base = (uchar*)0xdeadcafe;
  325. free(b);
  326. }
  327. Block *
  328. allocwb(long size)
  329. {
  330. Block *b;
  331. b = allocb(size);
  332. b->flag = 1;
  333. b->free = freewb;
  334. return b;
  335. }
  336. void
  337. printdata(void *pdata, int itemsize, int nitems)
  338. {
  339. int i;
  340. uchar *p1;
  341. ushort *p2;
  342. ulong *p4;
  343. if(!usbhdebug)
  344. return;
  345. p1 = pdata;
  346. p2 = pdata;
  347. p4 = pdata;
  348. i = 0;
  349. for(;;){
  350. switch(itemsize){
  351. default:
  352. assert(0);
  353. case 1:
  354. print("%2.2ux ", *p1++);
  355. break;
  356. case 2:
  357. print("%4.4ux ", *p2++);
  358. break;
  359. case 4:
  360. print("%8.8lux ", *p4++);
  361. break;
  362. }
  363. if(++i >= nitems || (i & ((0x40 >> itemsize) - 1)) == 0){
  364. print("\n");
  365. if(i >= nitems)
  366. break;
  367. }
  368. }
  369. }
  370. /*
  371. * i left these in so that we could use the same
  372. * driver on several other platforms (in principle).
  373. * the processor on which it was originally developed
  374. * had an IO MMU and thus another address space.
  375. * it's nothing to do with USB as such.
  376. */
  377. ulong
  378. va2hcva(void *va)
  379. {
  380. if(va == nil)
  381. return 0;
  382. return PADDR(va);
  383. }
  384. void *
  385. hcva2va(ulong hcva)
  386. {
  387. if(hcva == 0)
  388. return nil;
  389. return KADDR(hcva);
  390. }
  391. void *
  392. va2ucva(void *va)
  393. {
  394. return va;
  395. }
  396. void *
  397. hcva2ucva(ulong hcva)
  398. {
  399. if(hcva == 0)
  400. return nil;
  401. if(hcva & 0xf0000000){
  402. iprint("hcva2ucva: bad 0x%lux, called from 0x%lux\n",
  403. hcva, getcallerpc(&hcva));
  404. return nil;
  405. }
  406. return KADDR(hcva);
  407. }
  408. #define IOCACHED 0
  409. #define invalidatedcacheva(va)
  410. #define dcclean(p, n)
  411. static void
  412. EDinit(ED *ed, int mps, int f, int k, int s, int d, int en, int fa,
  413. TD *tail, TD *head, int c, int h, ED *next)
  414. {
  415. /* check nothing is running? */
  416. ed->ctrl = (mps & ED_MPS_MASK) << ED_MPS_SHIFT
  417. | (f & 1) << 15
  418. | (k & 1) << 14
  419. | (s & ED_S_MASK) << ED_S_SHIFT
  420. | (d & 3) << 11 /* 00 is obtained from TD (used here) */
  421. | (en & 0xf) << 7
  422. | (fa & 0x7f);
  423. ed->tail = va2hcva(tail) & ~0xF;
  424. ed->head = (va2hcva(head) & ~0xF)
  425. | (c & ED_C_MASK) << ED_C_SHIFT
  426. | (h & ED_H_MASK) << ED_H_SHIFT;
  427. ed->next = va2hcva(next) & ~0xF;
  428. }
  429. static void
  430. EDsetS(ED *ed, int s)
  431. {
  432. XIPRINT("EDsetS: %s speed\n", s == Lowspeed ? "low" : "high");
  433. if(s == Lowspeed)
  434. ed->ctrl |= 1 << ED_S_SHIFT;
  435. else
  436. ed->ctrl &= ~(1 << ED_S_SHIFT);
  437. }
  438. static void
  439. EDsetMPS(ED *ed, int mps)
  440. {
  441. ed->ctrl = (ed->ctrl & ~(ED_MPS_MASK << ED_MPS_SHIFT)) |
  442. (mps & ED_MPS_MASK) << ED_MPS_SHIFT;
  443. }
  444. static void
  445. EDsetC(ED *ed, int c)
  446. {
  447. ed->head = (ed->head & ~(ED_C_MASK << ED_C_SHIFT)) |
  448. (c & ED_C_MASK) << ED_C_SHIFT;
  449. }
  450. static void
  451. EDsetH(ED *ed, int h)
  452. {
  453. ed->head = (ed->head & ~(ED_H_MASK << ED_H_SHIFT)) |
  454. (h & ED_H_MASK) << ED_H_SHIFT;
  455. }
  456. static int
  457. EDgetH(ED *ed)
  458. {
  459. return (ed->head >> ED_H_SHIFT) & ED_H_MASK;
  460. }
  461. static int
  462. EDgetC(ED *ed)
  463. {
  464. return (ed->head >> ED_C_SHIFT) & ED_C_MASK;
  465. }
  466. static void
  467. EDsetnext(ED *ed, void *va)
  468. {
  469. ed->next = va2hcva(va) & ~0xF;
  470. }
  471. static ED *
  472. EDgetnext(ED *ed)
  473. {
  474. return hcva2ucva(ed->next & ~0xF);
  475. }
  476. static void
  477. EDsettail(ED *ed, void *va)
  478. {
  479. ed->tail = va2hcva(va) & ~0xF;
  480. }
  481. static TD *
  482. EDgettail(ED *ed)
  483. {
  484. return hcva2ucva(ed->tail & ~0xF);
  485. }
  486. static void
  487. EDsethead(ED *ed, void *va)
  488. {
  489. ed->head = (ed->head & 0xf) | (va2hcva(va) & ~0xF);
  490. }
  491. static TD *
  492. EDgethead(ED *ed)
  493. {
  494. return hcva2ucva(ed->head & ~0xF);
  495. }
  496. static ED *
  497. EDalloc(Ctlr *ub)
  498. {
  499. ED *t;
  500. qlock(&ub->ed);
  501. t = ub->ed.free;
  502. if(t == nil){
  503. qunlock(&ub->ed);
  504. return nil;
  505. }
  506. ub->ed.free = (ED *)t->next;
  507. ub->ed.alloced++;
  508. if (0)
  509. print("%d endpoints allocated\n", ub->ed.alloced);
  510. qunlock(&ub->ed);
  511. t->next = 0;
  512. return t;
  513. }
  514. void
  515. TDsetnexttd(TD *td, TD *va)
  516. {
  517. td->nexttd = va2hcva(va) & ~0xF;
  518. }
  519. TD *
  520. TDgetnexttd(TD *td)
  521. {
  522. return hcva2ucva(td->nexttd & ~0xF);
  523. }
  524. void
  525. OHCIsetControlHeadED(OHCI *ohci, ED *va)
  526. {
  527. ohci->ctlheaded = va2hcva(va) & ~0xF;
  528. }
  529. ED *
  530. OHCIgetControlHeadED(OHCI *ohci)
  531. {
  532. return hcva2ucva(ohci->ctlheaded);
  533. }
  534. void
  535. OHCIsetBulkHeadED(OHCI *ohci, ED *va)
  536. {
  537. ohci->bulkheaded = va2hcva(va) & ~0xF;
  538. }
  539. ED *
  540. OHCIgetBulkHeadED(OHCI *ohci)
  541. {
  542. return hcva2ucva(ohci->bulkheaded);
  543. }
  544. static TD *
  545. TDalloc(Ctlr *ub, Endpt *ep, int musthave) /* alloctd */
  546. {
  547. TD *t;
  548. Endptx *epx;
  549. for(;;){
  550. ilock(ub);
  551. t = ub->td.free;
  552. if(t)
  553. break;
  554. iunlock(ub);
  555. if(up == nil){
  556. if(musthave)
  557. panic("TDalloc: out of descs");
  558. return nil;
  559. }
  560. tsleep(&up->sleep, return0, 0, 100);
  561. }
  562. ub->td.free = t->next;
  563. epx = ep->private;
  564. epx->ntd++;
  565. ub->td.alloced++;
  566. iunlock(ub);
  567. memset(t, 0, sizeof(TD));
  568. t->ep = ep;
  569. return t;
  570. }
  571. /* call under ilock */
  572. static void
  573. TDfree(Ctlr *ub, TD *t) /* freetd */
  574. {
  575. Endptx *epx;
  576. if(t == 0)
  577. return;
  578. if(t->ep){
  579. epx = t->ep->private;
  580. epx->ntd--;
  581. } else
  582. t->ep = nil; /* redundant? */
  583. t->bp = nil;
  584. t->next = ub->td.free;
  585. ub->td.free = t;
  586. ub->td.alloced--;
  587. }
  588. static void
  589. EDfree(Ctlr *ub, ED *t)
  590. {
  591. TD *td, *next;
  592. if(t == 0)
  593. return;
  594. qlock(&ub->ed);
  595. t->next = (ulong)ub->ed.free;
  596. ub->ed.free = t;
  597. ub->ed.alloced--;
  598. if (0)
  599. print("%d endpoints allocated\n", ub->ed.alloced);
  600. ilock(ub);
  601. for(td = EDgethead(t); td; td = next){
  602. next = TDgetnexttd(td);
  603. TDfree(ub, td);
  604. }
  605. iunlock(ub);
  606. EDsethead(t, 0);
  607. EDsettail(t, 0);
  608. qunlock(&ub->ed);
  609. }
  610. static void
  611. waitSOF(Ctlr *ub)
  612. {
  613. /*
  614. * wait for SOF - interlock with interrupt handler so
  615. * done queue processed first.
  616. */
  617. int frame = ub->uchcca->framenumber & 0x3f;
  618. do {
  619. delay(2);
  620. } while(frame == (ub->uchcca->framenumber & 0x3f));
  621. }
  622. static void
  623. dumptd(TD *td, char *s)
  624. {
  625. int i;
  626. Endpt *ep;
  627. ep = td->ep;
  628. print("\t%s: 0x%.8lux ctrl 0x%.8lux cbp 0x%.8lux "
  629. "nexttd 0x%.8lux be 0x%.8lux, flags %lux\n",
  630. s, td, td->ctrl, td->cbp, td->nexttd, td->be, td->flags);
  631. if(ep->epmode != Isomode){
  632. print("\t\tbytes: %ld\n", td->be + 1 - td->cbp);
  633. return;
  634. }
  635. print("\t\t0x%ux 0x%ux 0x%ux 0x%ux 0x%ux 0x%ux 0x%ux 0x%ux\n",
  636. td->offsets[0], td->offsets[1], td->offsets[2], td->offsets[3],
  637. td->offsets[4], td->offsets[5], td->offsets[6], td->offsets[7]);
  638. print("\t\tbytes:");
  639. for(i = 0; i < td->ctrl >> 24 & 0x7; i++)
  640. print(" %d", (td->offsets[i+1]-td->offsets[i])&0xfff);
  641. print(" %ld\n", (td->be + 1 - td->offsets[i]) & 0xfff);
  642. }
  643. static void
  644. dumped(ED *ed)
  645. {
  646. TD *tailp, *td;
  647. tailp = EDgettail(ed);
  648. td = EDgethead(ed);
  649. print("dumpED 0x%lux: ctrl 0x%lux tail 0x%lux head 0x%lux next 0x%lux\n",
  650. ed, ed->ctrl, ed->tail, ed->head, ed->next);
  651. if(tailp == td)
  652. return;
  653. do {
  654. dumptd(td, "td");
  655. } while((td = TDgetnexttd(td)) != tailp);
  656. }
  657. static void
  658. dumpstatus(Ctlr *ub)
  659. {
  660. ED *ed;
  661. print("dumpstatus 0x%lux, frame 0x%ux:\n", ub, ub->uchcca->framenumber);
  662. print("control 0x%lux, cmdstat 0x%lux, intrsts 0x%lux\n",
  663. ub->base->control, ub->base->cmdsts, ub->base->intrsts);
  664. print("Control:\n");
  665. for(ed = OHCIgetControlHeadED(ub->base); ed; ed = EDgetnext(ed))
  666. dumped(ed);
  667. print("Bulk:\n");
  668. for(ed = OHCIgetBulkHeadED(ub->base); ed; ed = EDgetnext(ed))
  669. dumped(ed);
  670. print("Iso:\n");
  671. for(ed = ub->tree->root[0]; ed; ed = EDgetnext(ed))
  672. dumped(ed);
  673. print("frame 0x%ux:\n", ub->uchcca->framenumber);
  674. }
  675. /*
  676. * halt the ED and free input or output transfer descs
  677. * called when the relevant lock in the enclosing Endpt is held
  678. */
  679. static void
  680. EDcancel(Ctlr *ub, ED *ed, int dirin)
  681. {
  682. int tddir, iso, n;
  683. TD *tailp, *headp, *td, *prev, *next;
  684. Endpt *ep;
  685. if(ed == nil)
  686. return;
  687. /* halt ED if not already halted */
  688. if(EDgetH(ed) != 1){
  689. EDsetH(ed, 1);
  690. waitSOF(ub);
  691. }
  692. SET(tddir);
  693. if((iso = ed->ctrl & ED_F_BIT) != 0)
  694. switch((ed->ctrl >> 11) & 0x3){
  695. default:
  696. panic("ED iso direction unset");
  697. case Otokin: tddir = Dirin; break;
  698. case Otokout: tddir = Dirout; break;
  699. }
  700. /* can now clean up TD list of ED */
  701. tailp = EDgettail(ed);
  702. headp = EDgethead(ed);
  703. n = 0;
  704. prev = nil;
  705. td = headp;
  706. while(td != tailp){
  707. ep = td->ep;
  708. if(iso == 0)
  709. switch((td->ctrl >> TD_DP_SHIFT) & TD_DP_MASK){
  710. default:
  711. panic("TD direction unset");
  712. case Otoksetup: tddir = Dirout; break;
  713. case Otokin: tddir = Dirin; break;
  714. case Otokout: tddir = Dirout; break;
  715. }
  716. else if(usbhdebug || ep->debug)
  717. print("EDcancel: buffered: %d, bytes %ld\n",
  718. ep->buffered, td->bytes);
  719. next = TDgetnexttd(td);
  720. if(dirin == 2 || dirin == tddir){
  721. XEPRINT("%d/%d: EDcancel %d\n", ep->dev->x, ep->x, tddir);
  722. /* Remove this sucker */
  723. ep->buffered -= td->bytes;
  724. if(ep->buffered < 0)
  725. ep->buffered = 0;
  726. ilock(ub);
  727. ep->dir[tddir].queued--;
  728. if(tddir == Dirout){
  729. freeb(td->bp);
  730. td->bp = nil;
  731. }
  732. if(prev)
  733. TDsetnexttd(prev, next);
  734. else
  735. EDsethead(ed, next);
  736. TDfree(ub, td);
  737. n++;
  738. iunlock(ub);
  739. }else{
  740. XEPRINT("%d/%d: EDcancel skip %d\n", ep->dev->x, ep->x,
  741. tddir);
  742. prev = td;
  743. }
  744. td = next;
  745. }
  746. XPRINT("EDcancel: %d\n", n);
  747. }
  748. static void
  749. eptactivate(Ctlr *ub, Endpt *ep)
  750. {
  751. Endptx *epx;
  752. qlock(&ub->activends);
  753. if(ep->active == 0){
  754. epx = ep->private;
  755. XEPRINT("%d/%d: activate\n", ep->dev->x, ep->x);
  756. ep->active = 1;
  757. /*
  758. * set the right speed
  759. */
  760. EDsetS(epx->ed, ep->dev->speed);
  761. switch(ep->epmode){
  762. case Ctlmode:
  763. /*
  764. * chain the two descs together, and
  765. * bind to beginning of control queue
  766. */
  767. EDsetnext(epx->ed, OHCIgetControlHeadED(ub->base));
  768. OHCIsetControlHeadED(ub->base, epx->ed);
  769. /*
  770. * prompt controller to absorb new queue on next pass
  771. */
  772. ub->base->cmdsts |= Clf;
  773. XEPRINT("%d/%d: activated in control queue\n",
  774. ep->dev->x, ep->x);
  775. break;
  776. case Bulkmode:
  777. EDsetnext(epx->ed, OHCIgetBulkHeadED(ub->base));
  778. OHCIsetBulkHeadED(ub->base, epx->ed);
  779. ub->base->cmdsts |= Blf;
  780. XEPRINT("%d/%d: activated %s in bulk input queue\n",
  781. ep->dev->x, ep->x, omodename[ep->mode]);
  782. break;
  783. case Isomode:
  784. if(ep->mode != OWRITE)
  785. schedendpt(ub, ep, Dirin);
  786. if(ep->mode != OREAD)
  787. schedendpt(ub, ep, Dirout);
  788. ep->buffered = 0;
  789. ep->partial = 0;
  790. break;
  791. case Intrmode:
  792. if(ep->mode != OWRITE)
  793. schedendpt(ub, ep, Dirin);
  794. if(ep->mode != OREAD)
  795. schedendpt(ub, ep, Dirout);
  796. break;
  797. case Nomode:
  798. break;
  799. default:
  800. panic("eptactivate: wierd epmode %d\n", ep->epmode);
  801. }
  802. ep->dir[Dirin].xdone = ep->dir[Dirin].xstarted = 0;
  803. ep->dir[Dirout].xdone = ep->dir[Dirout].xstarted = 0;
  804. ep->activef = ub->activends.f;
  805. ub->activends.f = ep;
  806. }
  807. qunlock(&ub->activends);
  808. }
  809. static void
  810. EDpullfrombulk(Ctlr *ub, ED *ed)
  811. {
  812. ED *this, *prev, *next;
  813. this = OHCIgetBulkHeadED(ub->base);
  814. ub->base->bulkcurred = 0;
  815. prev = nil;
  816. while(this != nil && this != ed){
  817. prev = this;
  818. this = EDgetnext(this);
  819. }
  820. if(this == nil){
  821. print("EDpullfrombulk: not found\n");
  822. return;
  823. }
  824. next = EDgetnext(this);
  825. if(prev == nil)
  826. OHCIsetBulkHeadED(ub->base, next);
  827. else
  828. EDsetnext(prev, next);
  829. EDsetnext(ed, nil); /* wipe out next field */
  830. }
  831. static void
  832. EDpullfromctl(Ctlr *ub, ED *ed)
  833. {
  834. ED *this, *prev, *next;
  835. this = OHCIgetControlHeadED(ub->base);
  836. ub->base->ctlcurred = 0;
  837. prev = nil;
  838. while(this != nil && this != ed){
  839. prev = this;
  840. this = EDgetnext(this);
  841. }
  842. if(this == nil)
  843. panic("EDpullfromctl: not found\n");
  844. next = EDgetnext(this);
  845. if(prev == nil)
  846. OHCIsetControlHeadED(ub->base, next);
  847. else
  848. EDsetnext(prev, next);
  849. EDsetnext(ed, nil); /* wipe out next field */
  850. }
  851. static void
  852. eptdeactivate(Ctlr *ub, Endpt *ep)
  853. {
  854. ulong ctrl;
  855. Endpt **l;
  856. Endptx *epx;
  857. /* could be O(1) but not worth it yet */
  858. qlock(&ub->activends);
  859. if(ep->active){
  860. epx = ep->private;
  861. XEPRINT("ohci: eptdeactivate %d/%d\n", ep->dev->x, ep->x);
  862. ep->active = 0;
  863. for(l = &ub->activends.f; *l != ep; l = &(*l)->activef)
  864. if(*l == nil){
  865. qunlock(&ub->activends);
  866. panic("usb eptdeactivate");
  867. }
  868. *l = ep->activef;
  869. /* pull it from the appropriate queue */
  870. ctrl = ub->base->control;
  871. switch(ep->epmode){
  872. case Ctlmode:
  873. if(ctrl & Cle){
  874. ub->base->control &= ~Cle;
  875. waitSOF(ub);
  876. }
  877. EDpullfromctl(ub, epx->ed);
  878. if(ctrl & Cle){
  879. ub->base->control |= Cle;
  880. /*
  881. * don't fill it if there is nothing in it -
  882. * shouldn't be necessary according to the
  883. * spec., but practice is different
  884. */
  885. if(OHCIgetControlHeadED(ub->base))
  886. ub->base->cmdsts |= Clf;
  887. }
  888. break;
  889. case Bulkmode:
  890. if(ctrl & Ble){
  891. ub->base->control &= ~Ble;
  892. waitSOF(ub);
  893. }
  894. EDpullfrombulk(ub, epx->ed);
  895. if(ctrl & Ble){
  896. ub->base->control |= Ble;
  897. /*
  898. * don't fill it if there is nothing in it -
  899. * shouldn't be necessary according to the
  900. * spec., but practice is different
  901. */
  902. if(OHCIgetBulkHeadED(ub->base))
  903. ub->base->cmdsts |= Blf;
  904. }
  905. break;
  906. case Intrmode:
  907. case Isomode:
  908. if(ep->mode != OWRITE)
  909. unschedendpt(ub, ep, Dirin);
  910. if(ep->mode != OREAD)
  911. unschedendpt(ub, ep, Dirout);
  912. waitSOF(ub);
  913. break;
  914. case Nomode:
  915. break;
  916. default:
  917. panic("eptdeactivate: wierd in.epmode %d\n",
  918. ep->epmode);
  919. }
  920. }
  921. qunlock(&ub->activends);
  922. }
  923. static void
  924. kickappropriatequeue(Ctlr *ub, Endpt *ep, int)
  925. {
  926. switch(ep->epmode){
  927. case Nomode:
  928. break;
  929. case Ctlmode:
  930. ub->base->cmdsts |= Clf;
  931. break;
  932. case Bulkmode:
  933. ub->base->cmdsts |= Blf;
  934. break;
  935. case Intrmode:
  936. case Isomode:
  937. /* no kicking required */
  938. break;
  939. default:
  940. panic("wierd epmode %d\n", ep->epmode);
  941. }
  942. }
  943. static void
  944. eptenable(Ctlr *ub, Endpt *ep, int dirin)
  945. {
  946. ED *ed;
  947. Endptx *epx;
  948. epx = ep->private;
  949. ed = epx->ed;
  950. if(EDgetH(ed) == 1){
  951. EDsetH(ed, 0);
  952. kickappropriatequeue(ub, ep, dirin);
  953. if(ep->epmode == Isomode || ep->epmode == Intrmode)
  954. waitSOF(ub);
  955. }
  956. }
  957. /*
  958. * return smallest power of 2 >= n
  959. */
  960. static int
  961. flog2(int n)
  962. {
  963. int i;
  964. for(i = 0; (1 << i) < n; i++)
  965. ;
  966. return i;
  967. }
  968. /*
  969. * return smallest power of 2 <= n
  970. */
  971. static int
  972. flog2lower(int n)
  973. {
  974. int i;
  975. for(i = 0; (1 << (i + 1)) <= n; i++)
  976. ;
  977. return i;
  978. }
  979. static int
  980. pickschedq(QTree *qt, int pollms, ulong bw, ulong limit)
  981. {
  982. int i, j, d, upperb, q;
  983. ulong best, worst, total;
  984. d = flog2lower(pollms);
  985. if(d > qt->depth)
  986. d = qt->depth;
  987. q = -1;
  988. worst = 0;
  989. best = ~0;
  990. upperb = (1 << (d+1)) - 1;
  991. for(i = (1 << d) - 1; i < upperb; i++){
  992. total = qt->bw[0];
  993. for(j = i; j > 0; j = (j - 1) / 2)
  994. total += qt->bw[j];
  995. if(total < best){
  996. best = total;
  997. q = i;
  998. }
  999. if(total > worst)
  1000. worst = total;
  1001. }
  1002. if(worst + bw >= limit)
  1003. return -1;
  1004. return q;
  1005. }
  1006. static int
  1007. schedendpt(Ctlr *ub, Endpt *ep, int dirin)
  1008. {
  1009. int q;
  1010. ED *ed;
  1011. Endptx *epx;
  1012. epx = ep->private;
  1013. qlock(ub->tree);
  1014. /* TO DO: bus bandwidth limit */
  1015. q = pickschedq(ub->tree, ep->pollms, ep->bw, ~0);
  1016. XEPRINT("schedendpt, dir %d Q index %d, ms %d, bw %ld\n",
  1017. dirin, q, ep->pollms, ep->bw);
  1018. if(q < 0){
  1019. qunlock(ub->tree);
  1020. return -1;
  1021. }
  1022. ub->tree->bw[q] += ep->bw;
  1023. ed = ub->tree->root[q];
  1024. ep->sched = q;
  1025. EDsetnext(epx->ed, EDgetnext(ed));
  1026. EDsetnext(ed, epx->ed);
  1027. XEPRINT("%d/%d: sched on q %d pollms %d\n",
  1028. ep->dev->x, ep->x, q, ep->pollms);
  1029. qunlock(ub->tree);
  1030. return 0;
  1031. }
  1032. static void
  1033. unschedendpt(Ctlr *ub, Endpt *ep, int dirin)
  1034. {
  1035. int q;
  1036. ED *prev, *this, *next;
  1037. Endptx *epx;
  1038. epx = ep->private;
  1039. if((q = ep->sched) < 0)
  1040. return;
  1041. qlock(ub->tree);
  1042. ub->tree->bw[q] -= ep->bw;
  1043. prev = ub->tree->root[q];
  1044. this = EDgetnext(prev);
  1045. while(this != nil && this != epx->ed){
  1046. prev = this;
  1047. this = EDgetnext(this);
  1048. }
  1049. if(this == nil)
  1050. print("unschedendpt %d %d: not found\n", dirin, q);
  1051. else{
  1052. next = EDgetnext(this);
  1053. EDsetnext(prev, next);
  1054. }
  1055. qunlock(ub->tree);
  1056. }
  1057. /* at entry, *e is partly populated */
  1058. static void
  1059. epalloc(Usbhost *uh, Endpt *ep)
  1060. {
  1061. int id;
  1062. Endptx *epx;
  1063. Ctlr *ctlr;
  1064. Udev *d;
  1065. TD *dtd;
  1066. XEPRINT("ohci: epalloc from devusb\n");
  1067. ctlr = uh->ctlr;
  1068. id = ep->id;
  1069. d = ep->dev;
  1070. epx = malloc(sizeof(Endptx));
  1071. memset(epx, 0, sizeof(Endptx));
  1072. ep->private = epx;
  1073. dtd = nil;
  1074. if(waserror()){
  1075. XEPRINT("ohci: epalloc error\n");
  1076. EDfree(ctlr, epx->ed);
  1077. epx->ed = nil;
  1078. TDfree(ctlr, dtd);
  1079. nexterror();
  1080. }
  1081. if(epx->ed)
  1082. error("usb: already allocated");
  1083. if((epx->ed = EDalloc(ctlr)) == nil)
  1084. error(Enomem);
  1085. ep->bw = 1; /* all looks the same currently */
  1086. if((dtd = TDalloc(ctlr, ep, 0)) == nil)
  1087. error(Enomem);
  1088. EDinit(epx->ed, ep->maxpkt, 0, 0, 0, 0, id, d->id, dtd, dtd, 0, 0, 0);
  1089. XEPRINT("ohci: epalloc done\n");
  1090. poperror();
  1091. }
  1092. static void
  1093. epfree(Usbhost *uh, Endpt *ep)
  1094. {
  1095. Endptx *epx;
  1096. Ctlr *ctlr;
  1097. epx = ep->private;
  1098. ctlr = uh->ctlr;
  1099. XEPRINT("ohci: epfree %d/%d from devusb\n", ep->dev->x, ep->x);
  1100. if(ep->active)
  1101. panic("epfree: active");
  1102. EDfree(ctlr, epx->ed);
  1103. epx->ed = nil;
  1104. free(epx);
  1105. ep->private = nil;
  1106. }
  1107. static void
  1108. epopen(Usbhost *uh, Endpt *ep)
  1109. {
  1110. Ctlr *ctlr;
  1111. XEPRINT("ohci: epopen %d/%d from devusb\n", ep->dev->x, ep->x);
  1112. ctlr = uh->ctlr;
  1113. if((ep->epmode == Isomode || ep->epmode == Intrmode) && ep->active)
  1114. error("already open");
  1115. eptactivate(ctlr, ep);
  1116. }
  1117. static int
  1118. setfrnum(Ctlr *ub, Endpt *ep)
  1119. {
  1120. short frnum, d;
  1121. static int adj;
  1122. /* adjust frnum as necessary... */
  1123. frnum = ub->base->fmnumber + (ep->buffered*1000)/ep->bw;
  1124. d = frnum - ep->frnum;
  1125. if(d < -100 || d > 100){
  1126. /* We'd play in the past */
  1127. if(0 && d > 1000)
  1128. /* We're more than a second off: */
  1129. print("d %d, done %d, started %d, buffered %d\n", d,
  1130. ep->dir[Dirout].xdone, ep->dir[Dirout].xstarted,
  1131. ep->buffered);
  1132. if(ep->dir[Dirout].xdone == ep->dir[Dirout].xstarted)
  1133. ep->buffered = adj = 0;
  1134. if(0 && (adj++ & 0xff) == 0)
  1135. print("adj %d %d\n", d, ep->buffered);
  1136. ep->frnum = ub->base->fmnumber + 10 + (ep->buffered*1000)/ep->bw;
  1137. ep->partial = 0;
  1138. return 1;
  1139. }
  1140. return 0;
  1141. }
  1142. static int
  1143. ceptdone(void *arg)
  1144. {
  1145. Endpt *ep;
  1146. ep = arg;
  1147. return ep->dir[Dirout].xdone - ep->dir[Dirout].xstarted >= 0
  1148. || ep->dir[Dirout].err;
  1149. }
  1150. static void
  1151. epclose(Usbhost *uh, Endpt *ep)
  1152. {
  1153. Ctlr *ctlr;
  1154. int xdone, part;
  1155. Endptx *epx;
  1156. XEPRINT("ohci: epclose %d/%d from devusb, %d buffered\n",
  1157. ep->dev->x, ep->x, ep->buffered);
  1158. ctlr = uh->ctlr;
  1159. epx = ep->private;
  1160. if(ep->epmode == Isomode && ep->active){
  1161. qlock(&ep->wlock);
  1162. if(ep->partial && setfrnum(ctlr, ep) == 0){
  1163. part = ep->partial;
  1164. memset(ep->bpartial->wp, 0, ep->maxpkt - ep->partial);
  1165. ep->bpartial->wp = ep->bpartial->rp + ep->maxpkt;
  1166. qtd(uh->ctlr, ep, Dirout, nil, ep->bpartial->rp,
  1167. ep->bpartial->wp, Otokout, TD_FLAGS_LAST);
  1168. XEPRINT("epclose: wrote partial block %d\n", part);
  1169. ep->partial = 0;
  1170. }
  1171. qunlock(&ep->wlock);
  1172. XEPRINT("epclose: wait for outstanding TDs, xdone %d"
  1173. ", xstarted %d, buffered %d, queued %d\n",
  1174. ep->dir[Dirout].xdone, ep->dir[Dirout].xstarted,
  1175. ep->buffered, ep->dir[Dirout].queued);
  1176. while(ep->dir[Dirout].err == nil
  1177. && (xdone = ep->dir[Dirout].xdone) - ep->dir[Dirout].xstarted < 0){
  1178. tsleep(&ep->dir[Dirout].rend, ceptdone, ep, 500);
  1179. if(xdone == ep->dir[Dirout].xdone){
  1180. print("no progress\n");
  1181. break;
  1182. }
  1183. }
  1184. if(ep->dir[Dirout].err)
  1185. XEPRINT("error: %s\n", ep->dir[Dirout].err);
  1186. if(ep->buffered)
  1187. XEPRINT("epclose: done waiting, xdone %d, xstarted %d, "
  1188. "buffered %d, queued %d\n",
  1189. ep->dir[Dirout].xdone, ep->dir[Dirout].xstarted,
  1190. ep->buffered, ep->dir[Dirout].queued);
  1191. }
  1192. lock(epx);
  1193. EDcancel(ctlr, epx->ed, 2);
  1194. unlock(epx);
  1195. if(ep->epmode == Isomode && ep->buffered)
  1196. XEPRINT("epclose: after cancelling, xdone %d, xstarted %d"
  1197. ", buffered %d, queued %d\n",
  1198. ep->dir[Dirout].xdone, ep->dir[Dirout].xstarted,
  1199. ep->buffered, ep->dir[Dirout].queued);
  1200. eptdeactivate(ctlr, ep);
  1201. }
  1202. static void
  1203. epmaxpkt(Usbhost *, Endpt *ep)
  1204. {
  1205. Endptx *epx;
  1206. epx = ep->private;
  1207. XEPRINT("ohci: epmaxpkt %d/%d: %d\n",
  1208. ep->dev->x, ep->x, ep->maxpkt);
  1209. EDsetMPS(epx->ed, ep->maxpkt);
  1210. }
  1211. static void
  1212. epmode(Usbhost *uh, Endpt *ep)
  1213. {
  1214. int tok, reactivate;
  1215. Ctlr *ctlr;
  1216. Endptx *epx;
  1217. epx = ep->private;
  1218. ctlr = uh->ctlr;
  1219. XEPRINT("ohci: epmode %d/%d %s → %s\n",
  1220. ep->dev->x, ep->x, modename[ep->epmode], modename[ep->epnewmode]);
  1221. reactivate = 0;
  1222. if(ep->epnewmode != ep->epmode)
  1223. if(reactivate = ep->active){
  1224. XEPRINT("ohci: epmode %d/%d: already open\n",
  1225. ep->dev->x, ep->x);
  1226. eptdeactivate(ctlr, ep);
  1227. }
  1228. EDsetS(epx->ed, ep->dev->speed);
  1229. switch(ep->epnewmode){
  1230. default:
  1231. panic("devusb is sick");
  1232. case Intrmode:
  1233. // ep->debug++;
  1234. ep->bw = ep->maxpkt*1000/ep->pollms; /* bytes/sec */
  1235. XEPRINT("ohci: epmode %d/%d %s, intr: maxpkt %d, pollms %d, bw %ld\n",
  1236. ep->dev->x, ep->x, omodename[ep->mode],
  1237. ep->maxpkt, ep->pollms, ep->bw);
  1238. break;
  1239. case Isomode:
  1240. // ep->debug++;
  1241. ep->rem = 999;
  1242. switch(ep->mode){
  1243. default:
  1244. panic("ep mode");
  1245. case ORDWR:
  1246. error("iso unidirectional only");
  1247. case OREAD:
  1248. tok = Otokin;
  1249. error("iso read not implemented");
  1250. break;
  1251. case OWRITE:
  1252. tok = Otokout;
  1253. break;
  1254. }
  1255. XEPRINT("ohci: epmode %d/%d %s, iso: maxpkt %d, pollms %d, hz %d, samp %d\n",
  1256. ep->dev->x, ep->x, omodename[ep->mode],
  1257. ep->maxpkt, ep->pollms, ep->hz, ep->samplesz);
  1258. ep->bw = ep->hz * ep->samplesz; /* bytes/sec */
  1259. /* Use Iso TDs: */
  1260. epx->ed->ctrl |= ED_F_BIT;
  1261. /* Set direction in ED, no room in an Iso TD for this */
  1262. epx->ed->ctrl &= ~(ED_D_MASK << ED_D_SHIFT);
  1263. epx->ed->ctrl |= tok << ED_D_SHIFT;
  1264. break;
  1265. case Bulkmode:
  1266. // ep->debug++;
  1267. /*
  1268. * Each Bulk device gets a queue head hanging off the
  1269. * bulk queue head
  1270. */
  1271. break;
  1272. case Ctlmode:
  1273. break;
  1274. }
  1275. ep->epmode = ep->epnewmode;
  1276. epmaxpkt(uh, ep);
  1277. if(reactivate){
  1278. XEPRINT("ohci: epmode %d/%d: reactivate\n", ep->dev->x, ep->x);
  1279. eptactivate(ctlr, ep);
  1280. }
  1281. }
  1282. static long
  1283. qtd(Ctlr *ub, Endpt *ep, int dirin , Block *bp, uchar *base, uchar *limit,
  1284. int pid, ulong flags)
  1285. {
  1286. int fc, mps;
  1287. ulong x;
  1288. uchar *p;
  1289. ED *ed;
  1290. Endptx *epx;
  1291. TD *dummytd, *td;
  1292. epx = ep->private;
  1293. ed = epx->ed;
  1294. td = hcva2ucva(ed->tail);
  1295. td->flags = flags;
  1296. if(ep->epmode == Isomode){
  1297. x = va2hcva(base);
  1298. td->cbp = x & ~0xfff;
  1299. x &= 0xfff;
  1300. p = base;
  1301. setfrnum(ub, ep);
  1302. td->ctrl = ep->frnum & 0xffff;
  1303. fc = 0;
  1304. for(;;){
  1305. /* Calculate number of samples in next packet */
  1306. mps = (ep->hz + ep->rem)/1000;
  1307. /* rem is the partial sample left over */
  1308. ep->rem += ep->hz - 1000*mps;
  1309. mps *= ep->samplesz;
  1310. if(mps > ep->maxpkt)
  1311. panic("Packet size");
  1312. if(ep->partial == 0 && mps > limit - p){
  1313. /* Save this data for later ... */
  1314. ep->partial = limit - p;
  1315. if(fc-- == 0)
  1316. return p - base; /* No TD */
  1317. /* We do have a TD, send this one off normally */
  1318. td->flags |= TD_FLAGS_LAST;
  1319. break;
  1320. }else if(mps >= limit - p){
  1321. td->flags |= TD_FLAGS_LAST;
  1322. mps = limit - p;
  1323. ep->partial = 0;
  1324. }
  1325. td->offsets[fc] = 0xe000 | x;
  1326. x += mps;
  1327. p += mps;
  1328. ep->frnum++;
  1329. if(fc == 7 || limit - p == 0)
  1330. break;
  1331. fc++;
  1332. }
  1333. td->ctrl |= fc << 24;
  1334. }else{
  1335. td->cbp = va2hcva(base);
  1336. td->ctrl = (pid & TD_DP_MASK) << TD_DP_SHIFT;
  1337. p = base;
  1338. mps = 0x2000 - ((ulong)p & 0xfff);
  1339. if(mps > ep->maxpkt)
  1340. mps = ep->maxpkt;
  1341. if(mps >= limit - p){
  1342. mps = limit - base;
  1343. td->flags |= TD_FLAGS_LAST;
  1344. }
  1345. p += mps;
  1346. }
  1347. td->be = va2hcva(p == nil? nil: p - 1);
  1348. td->ep = ep;
  1349. td->bytes = p - base;
  1350. ep->buffered += td->bytes;
  1351. td->bp = bp;
  1352. if(td->flags & TD_FLAGS_LAST)
  1353. ep->dir[dirin].xstarted++;
  1354. if(dirin == Dirout && bp)
  1355. refcnt(bp, 1);
  1356. dummytd = TDalloc(ub, ep, 1);
  1357. TDsetnexttd(td, dummytd);
  1358. ep->dir[dirin].queued++;
  1359. EDsettail(ed, dummytd);
  1360. if(usbhdebug || ep->debug)
  1361. dumptd(td, "qtd: before");
  1362. kickappropriatequeue(ub, ep, dirin);
  1363. return p - base;
  1364. }
  1365. Block*
  1366. allocrcvb(long size)
  1367. {
  1368. Block *b;
  1369. int asize;
  1370. asize = ROUND(size, dcls) + dcls - 1;
  1371. /*
  1372. * allocate enough to align rp to dcls, and have an integral number
  1373. * of cache lines in the buffer
  1374. */
  1375. while(waserror())
  1376. tsleep(&up->sleep, return0, 0, 100);
  1377. b = allocb(asize);
  1378. poperror();
  1379. /*
  1380. * align the rp and wp
  1381. */
  1382. b->rp = b->wp = (uchar *)ROUND((ulong)b->rp, dcls);
  1383. /*
  1384. * invalidate the cache lines which enclose the buffer
  1385. */
  1386. if(IOCACHED){
  1387. uchar *p;
  1388. p = b->rp;
  1389. while(size > 0){
  1390. invalidatedcacheva((ulong)p);
  1391. p += dcls;
  1392. size -= dcls;
  1393. }
  1394. }
  1395. return b;
  1396. }
  1397. /*
  1398. * build the periodic scheduling tree:
  1399. * framesize must be a multiple of the tree size
  1400. */
  1401. static QTree *
  1402. mkqhtree(Ctlr *ub)
  1403. {
  1404. int i, n, d, o, leaf0, depth;
  1405. ED **tree;
  1406. QTree *qt;
  1407. depth = flog2(32);
  1408. n = (1 << (depth+1)) - 1;
  1409. qt = mallocz(sizeof(*qt), 1);
  1410. if(qt == nil)
  1411. return nil;
  1412. qt->nel = n;
  1413. qt->depth = depth;
  1414. qt->bw = mallocz(n * sizeof(qt->bw), 1);
  1415. if(qt->bw == nil){
  1416. free(qt);
  1417. return nil;
  1418. }
  1419. tree = mallocz(n * sizeof(ED *), 1);
  1420. if(tree == nil){
  1421. free(qt->bw);
  1422. free(qt);
  1423. return nil;
  1424. }
  1425. for(i = 0; i < n; i++)
  1426. if((tree[i] = EDalloc(ub)) == nil)
  1427. break;
  1428. if(i < n){
  1429. int j;
  1430. for(j = 0; j < i; j++)
  1431. EDfree(ub, tree[j]);
  1432. free(tree);
  1433. free(qt->bw);
  1434. free(qt);
  1435. return nil;
  1436. }
  1437. qt->root = tree;
  1438. EDinit(qt->root[0], 8, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0);
  1439. for(i = 1; i < n; i++)
  1440. EDinit(tree[i], 8, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, tree[(i-1)/2]);
  1441. /* distribute leaves evenly round the frame list */
  1442. leaf0 = n / 2;
  1443. for(i = 0; i < 32; i++){
  1444. o = 0;
  1445. for(d = 0; d < depth; d++){
  1446. o <<= 1;
  1447. if(i & (1 << d))
  1448. o |= 1;
  1449. }
  1450. if(leaf0 + o >= n){
  1451. print("leaf0=%d o=%d i=%d n=%d\n", leaf0, o, i, n);
  1452. break;
  1453. }
  1454. ub->uchcca->intrtable[i] = va2hcva(tree[leaf0 + o]);
  1455. }
  1456. return qt;
  1457. }
  1458. static void
  1459. portreset(Usbhost *uh, int port)
  1460. {
  1461. Ctlr *ctlr;
  1462. XIPRINT("ohci: portreset(port %d) from devusb\n", port);
  1463. ctlr = uh->ctlr;
  1464. /* should check that device not being configured on other port? */
  1465. qlock(&ctlr->resetl);
  1466. if(waserror()){
  1467. qunlock(&ctlr->resetl);
  1468. nexterror();
  1469. }
  1470. ilock(ctlr);
  1471. ctlr->base->rhportsts[port - 1] = Spp | Spr;
  1472. while((ctlr->base->rhportsts[port - 1] & Prsc) == 0){
  1473. iunlock(ctlr);
  1474. XIPRINT("ohci: portreset, wait for reset complete\n");
  1475. ilock(ctlr);
  1476. }
  1477. ctlr->base->rhportsts[port - 1] = Prsc;
  1478. iunlock(ctlr);
  1479. poperror();
  1480. qunlock(&ctlr->resetl);
  1481. }
  1482. static void
  1483. portenable(Usbhost *uh, int port, int on)
  1484. {
  1485. Ctlr *ctlr;
  1486. XIPRINT("ohci: portenable(port %d, on %d) from devusb\n", port, on);
  1487. ctlr = uh->ctlr;
  1488. /* should check that device not being configured on other port? */
  1489. qlock(&ctlr->resetl);
  1490. if(waserror()){
  1491. qunlock(&ctlr->resetl);
  1492. nexterror();
  1493. }
  1494. ilock(ctlr);
  1495. if(on)
  1496. ctlr->base->rhportsts[port - 1] = Spe | Spp;
  1497. else
  1498. ctlr->base->rhportsts[port - 1] = Cpe;
  1499. iunlock(ctlr);
  1500. poperror();
  1501. qunlock(&ctlr->resetl);
  1502. }
  1503. static int
  1504. getportstatus(Ctlr *ub, int port)
  1505. {
  1506. int v;
  1507. ulong ohcistatus;
  1508. ohcistatus = ub->base->rhportsts[port - 1];
  1509. v = 0;
  1510. if(ohcistatus & Ccs)
  1511. v |= DevicePresent;
  1512. if(ohcistatus & Pes)
  1513. v |= PortEnable;
  1514. if(ohcistatus & Pss)
  1515. v |= Suspend;
  1516. if(ohcistatus & Prs)
  1517. v |= PortReset;
  1518. else {
  1519. /* port is not in reset; these potential writes are ok */
  1520. if(ohcistatus & Csc){
  1521. /* TODO: could notify usbd equivalent here */
  1522. v |= ConnectStatusChange;
  1523. ub->base->rhportsts[port - 1] = Csc;
  1524. }
  1525. if(ohcistatus & Pesc){
  1526. /* TODO: could notify usbd equivalent here */
  1527. v |= PortEnableChange;
  1528. ub->base->rhportsts[port - 1] = Pesc;
  1529. }
  1530. }
  1531. if(ohcistatus & Lsda)
  1532. v |= SlowDevice;
  1533. if(ohcistatus & Ccs)
  1534. XIPRINT("portstatus(%d) = OHCI 0x%.8lux UHCI 0x%.8ux\n",
  1535. port, ohcistatus, v);
  1536. return v;
  1537. }
  1538. /* this is called several times every few seconds, possibly due to usbd */
  1539. static void
  1540. portinfo(Usbhost *uh, char *s, char *se)
  1541. {
  1542. int x, i, j;
  1543. Ctlr *ctlr;
  1544. XIPRINT("ohci: portinfo from devusb\n");
  1545. ctlr = uh->ctlr;
  1546. for(i = 1; i <= 4; i++) {
  1547. ilock(ctlr);
  1548. x = getportstatus(ctlr, i);
  1549. iunlock(ctlr);
  1550. s = seprint(s, se, "%d %ux", i, x);
  1551. for(j = 0; j < nelem(portstatus); j++)
  1552. if((x & portstatus[j].bit) != 0)
  1553. s = seprint(s, se, " %s", portstatus[j].name);
  1554. s = seprint(s, se, "\n");
  1555. }
  1556. }
  1557. void
  1558. interrupt(Ureg *, void *arg)
  1559. {
  1560. int dirin, cc;
  1561. ulong ctrl, status;
  1562. uchar *p;
  1563. Block *bp;
  1564. Ctlr *ub;
  1565. Endpt *ep;
  1566. Endptx *epx;
  1567. TD *donetd, *nexttd;
  1568. Usbhost *eh;
  1569. XIPRINT("ohci: interrupt\n");
  1570. eh = arg;
  1571. ub = eh->ctlr;
  1572. status = ub->base->intrsts;
  1573. status &= ub->base->intrenable;
  1574. status &= Oc | Rhsc | Fno
  1575. | Ue
  1576. | Rd | Sf | Wdh
  1577. | So;
  1578. if(status & Wdh){
  1579. /* LSb of donehead has bit that says there are other interrupts */
  1580. donetd = hcva2ucva(ub->uchcca->donehead & ~0xf);
  1581. XIPRINT("donetd 0x%.8lux\n", donetd);
  1582. }else
  1583. donetd = 0;
  1584. ub->base->intrsts = status;
  1585. status &= ~Wdh;
  1586. while(donetd){
  1587. ctrl = donetd->ctrl;
  1588. ep = donetd->ep;
  1589. bp = donetd->bp;
  1590. donetd->bp = nil;
  1591. epx = ep->private;
  1592. ohciinterrupts[ep->epmode]++;
  1593. dirin = ((ctrl >> TD_DP_SHIFT) & TD_DP_MASK) == Otokin;
  1594. ep->buffered -= donetd->bytes;
  1595. if(ep->epmode == Isomode){
  1596. dirin = Dirout;
  1597. if(ep->buffered < 0){
  1598. print("intr: buffered %d bytes %ld\n",
  1599. ep->buffered, donetd->bytes);
  1600. ep->buffered = 0;
  1601. }
  1602. }
  1603. cc = (ctrl >> TD_CC_SHIFT) & TD_CC_MASK;
  1604. if((usbhdebug || ep->debug) && (cc != 0 && cc != 9)){
  1605. print("%d/%d: cc %d frnum 0x%lux\n",
  1606. ep->dev->x, ep->x, cc, ub->base->fmnumber);
  1607. dumptd(donetd, "after");
  1608. }
  1609. switch(cc){
  1610. case 8: /* Overrun, Not an error */
  1611. epx->overruns++;
  1612. /* fall through to no error code */
  1613. case 0: /* noerror */
  1614. if((donetd->flags & TD_FLAGS_LAST) == 0)
  1615. break;
  1616. if(dirin){
  1617. if(bp){
  1618. p = hcva2va(donetd->be + 1);
  1619. if(p < bp->wp)
  1620. print("interrupt: bp: rp 0x%lux"
  1621. ", wp 0x%lux→0x%lux\n",
  1622. bp->rp, bp->wp, p);
  1623. bp->wp = p;
  1624. }
  1625. }
  1626. ep->dir[dirin].xdone++;
  1627. wakeup(&ep->dir[dirin].rend);
  1628. break;
  1629. case 9: /* underrun */
  1630. if(bp){
  1631. p = hcva2va(donetd->cbp);
  1632. XEIPRINT("interrupt: bp: rp 0x%lux, wp "
  1633. "0x%lux→0x%lux\n", bp->rp, bp->wp, p);
  1634. bp->wp = p;
  1635. }
  1636. if((donetd->flags & TD_FLAGS_LAST) == 0){
  1637. XEIPRINT("Underrun\n");
  1638. ep->dir[dirin].err = Eunderrun;
  1639. }
  1640. ep->dir[dirin].xdone++;
  1641. wakeup(&ep->dir[dirin].rend);
  1642. break;
  1643. case 1: /* CRC */
  1644. ep->dir[dirin].err = "CRC error";
  1645. goto error;
  1646. case 2: /* Bitstuff */
  1647. ep->dir[dirin].err = "Bitstuff error";
  1648. goto error;
  1649. case 4: /* Stall */
  1650. ep->dir[dirin].err = Estalled;
  1651. goto error;
  1652. case 5: /* No response */
  1653. ep->dir[dirin].err = "No response";
  1654. goto error;
  1655. case 6: /* PIDcheck */
  1656. ep->dir[dirin].err = "PIDcheck";
  1657. goto error;
  1658. case 7: /* UnexpectedPID */
  1659. ep->dir[dirin].err = "badPID";
  1660. goto error;
  1661. error:
  1662. XEPRINT("fail %d (%lud)\n", cc,
  1663. (ctrl >> TD_EC_SHIFT) & TD_EC_MASK);
  1664. ep->dir[dirin].xdone++;
  1665. wakeup(&ep->dir[dirin].rend);
  1666. break;
  1667. default:
  1668. panic("cc %lud unimplemented\n",
  1669. (ctrl >> TD_CC_SHIFT) & TD_CC_MASK);
  1670. }
  1671. ep->dir[dirin].queued--;
  1672. /* Clean up blocks used for transfers */
  1673. if(dirin == Dirout)
  1674. freeb(bp);
  1675. nexttd = TDgetnexttd(donetd);
  1676. TDfree(ub, donetd);
  1677. donetd = nexttd;
  1678. }
  1679. if(status & Sf){
  1680. if (0)
  1681. XIPRINT(("sof!!\n"));
  1682. // wakeup(&ub->sofr); /* sofr doesn't exist anywhere! */
  1683. status &= ~Sf;
  1684. }
  1685. if(status & Ue){
  1686. ulong curred;
  1687. // usbhdbg(); /* TODO */
  1688. curred = ub->base->periodcurred;
  1689. print("usbh: unrecoverable error frame 0x%.8lux ed 0x%.8lux, "
  1690. "ints %d %d %d %d\n",
  1691. ub->base->fmnumber, curred,
  1692. ohciinterrupts[1], ohciinterrupts[2],
  1693. ohciinterrupts[3], ohciinterrupts[4]);
  1694. if(curred)
  1695. dumped(hcva2ucva(curred));
  1696. }
  1697. if(status)
  1698. IPRINT(("interrupt: unhandled interrupt 0x%.8lux\n", status));
  1699. }
  1700. static void
  1701. usbhattach(Ctlr *ub) /* TODO: is unused now, but it fiddles ctlr */
  1702. {
  1703. ulong ctrl;
  1704. if(ub == nil || ub->base == 0)
  1705. error(Enodev);
  1706. ctrl = ub->base->control;
  1707. if((ctrl & HcfsMask) != HcfsOperational){
  1708. ctrl = (ctrl & ~HcfsMask) | HcfsOperational;
  1709. ub->base->control = ctrl;
  1710. ub->base->rhsts = Sgp;
  1711. }
  1712. }
  1713. static int
  1714. reptdone(void *arg)
  1715. {
  1716. Endpt *ep;
  1717. ep = arg;
  1718. return ep->dir[Dirin].err
  1719. /* Expression crafted to deal with wrap around: */
  1720. || ep->dir[Dirin].xdone - ep->dir[Dirin].xstarted >= 0;
  1721. }
  1722. Block *
  1723. breadusbh(Ctlr *ub, Endpt *ep, long n) /* guts of read() */
  1724. {
  1725. long in, l;
  1726. uchar *p;
  1727. Block *bp;
  1728. Endptx *epx;
  1729. epx = ep->private;
  1730. qlock(&ep->rlock);
  1731. EDsetC(epx->ed, ep->rdata01);
  1732. XEPRINT("breadusbh(%d/%d, %ld, dt %d)\n", ep->dev->x, ep->x, n, ep->rdata01);
  1733. eptenable(ub, ep, Dirin);
  1734. if(waserror()){
  1735. EDcancel(ub, epx->ed, Dirin);
  1736. ep->dir[Dirin].err = nil;
  1737. qunlock(&ep->rlock);
  1738. nexterror();
  1739. }
  1740. if(ep->dir[Dirin].err != nil)
  1741. error("usb: can't happen");
  1742. bp = allocrcvb(n);
  1743. in = n;
  1744. if(in > bp->lim - bp->wp){
  1745. print("usb: read larger than block\n");
  1746. in = bp->lim - bp->wp;
  1747. }
  1748. p = bp->rp;
  1749. do{
  1750. l = qtd(ub, ep, Dirin, bp, p, p+in, Otokin, 0);
  1751. p += l;
  1752. in -= l;
  1753. }while(in > 0);
  1754. sleep(&ep->dir[Dirin].rend, reptdone, ep);
  1755. if(ep->dir[Dirin].err){
  1756. EDcancel(ub, epx->ed, Dirin);
  1757. if(ep->dir[Dirin].err == Eunderrun)
  1758. ep->dir[Dirin].err = nil;
  1759. else
  1760. error(ep->dir[Dirin].err);
  1761. }
  1762. XEPRINT("breadusbh(%d/%d, %ld) returned %ld\n", ep->dev->x, ep->x, n,
  1763. BLEN(bp));
  1764. poperror();
  1765. qunlock(&ep->rlock);
  1766. ep->rdata01 = EDgetC(epx->ed);
  1767. return bp;
  1768. }
  1769. static long
  1770. read(Usbhost *uh, Endpt *ep, void *a, long n, vlong off) /* TODO off */
  1771. {
  1772. long l;
  1773. Block *bp;
  1774. Ctlr *ub;
  1775. XEPRINT("ohci: read from devusb\n");
  1776. USED(off);
  1777. ub = uh->ctlr;
  1778. XEPRINT("%d/%d: read 0x%.8lux %ld\n", ep->dev->x, ep->x, a, n);
  1779. bp = breadusbh(ub, ep, n);
  1780. l = BLEN(bp);
  1781. memmove(a, bp->rp, l);
  1782. printdata(bp->rp, 1, l);
  1783. XEPRINT("ohci: read %ld\n\n", l);
  1784. freeb(bp);
  1785. return l;
  1786. }
  1787. static int
  1788. weptdone(void *arg)
  1789. {
  1790. Endpt *ep;
  1791. ep = arg;
  1792. /*
  1793. * success when all operations are done or when less than
  1794. * a second is buffered in iso connections
  1795. */
  1796. return ep->dir[Dirout].xdone - ep->dir[Dirout].xstarted >= 0
  1797. || (ep->epmode == Isomode && ep->buffered <= ep->bw)
  1798. || ep->dir[Dirout].err;
  1799. }
  1800. /* TODO: honour off */
  1801. static long
  1802. write(Usbhost *uh, Endpt *ep, void *a, long n, vlong off, int tok)
  1803. {
  1804. long m;
  1805. short frnum;
  1806. uchar *p = a;
  1807. Block *b;
  1808. Ctlr *ub;
  1809. Endptx *epx;
  1810. XEPRINT("ohci: write(addr %p, bytes %ld, off %lld, tok %d) from devusb\n",
  1811. a, n, off, tok);
  1812. epx = ep->private;
  1813. ub = uh->ctlr;
  1814. qlock(&ep->wlock);
  1815. XEPRINT("%d/%d: write 0x%.8lux %ld %s\n", ep->dev->x, ep->x, a, n,
  1816. tok == Otoksetup? "setup": "out");
  1817. if(ep->dir[Dirout].xdone - ep->dir[Dirout].xstarted > 0){
  1818. print("done > started, %d %d\n",
  1819. ep->dir[Dirout].xdone, ep->dir[Dirout].xstarted);
  1820. ep->dir[Dirout].xdone = ep->dir[Dirout].xstarted;
  1821. }
  1822. if(waserror()){
  1823. lock(epx);
  1824. EDcancel(ub, epx->ed, Dirout);
  1825. unlock(epx);
  1826. ep->dir[Dirout].err = nil;
  1827. qunlock(&ep->wlock);
  1828. nexterror();
  1829. }
  1830. eptenable(ub, ep, Dirout);
  1831. EDsetC(epx->ed, ep->wdata01);
  1832. if(ep->dir[Dirout].err)
  1833. error(ep->dir[Dirout].err);
  1834. if((m = n) == 0 || p == nil)
  1835. qtd(ub, ep, Dirout, 0, 0, 0, tok, TD_FLAGS_LAST);
  1836. else{
  1837. b = allocwb(m+ep->partial);
  1838. if(ep->partial){
  1839. memmove(b->wp, ep->bpartial->rp, ep->partial);
  1840. b->wp += ep->partial;
  1841. ep->partial = 0;
  1842. }
  1843. validaddr((uintptr)p, m, 0); /* DEBUG */
  1844. memmove(b->wp, a, m);
  1845. b->wp += m;
  1846. printdata(b->rp, 1, m);
  1847. m = BLEN(b);
  1848. dcclean(b->rp, m);
  1849. if(ep->epmode == Isomode && ep->buffered <= ep->bw<<1){
  1850. sleep(&ep->dir[Dirout].rend, weptdone, ep);
  1851. if(ep->dir[Dirout].err)
  1852. error(ep->dir[Dirout].err);
  1853. }
  1854. while(m > 0){
  1855. int l;
  1856. l = qtd(ub, ep, Dirout, b, b->rp, b->wp, tok, 0);
  1857. b->rp += l;
  1858. m -= l;
  1859. tok = Otokout;
  1860. if(ep->partial){
  1861. /* We have some data to save */
  1862. if(ep->bpartial == nil)
  1863. ep->bpartial = allocb(ep->maxpkt);
  1864. if(ep->partial != m)
  1865. print("curious: %d != %ld\n",
  1866. ep->partial, m);
  1867. memmove(ep->bpartial->rp, b->rp, ep->partial);
  1868. ep->bpartial->wp = ep->bpartial->rp + ep->partial;
  1869. break;
  1870. }
  1871. }
  1872. freeb(b);
  1873. }
  1874. if(ep->epmode != Isomode){
  1875. sleep(&ep->dir[Dirout].rend, weptdone, ep);
  1876. if(ep->dir[Dirout].err)
  1877. error(ep->dir[Dirout].err);
  1878. }else if(0 && (frnum = ep->frnum - ub->base->fmnumber) < 0)
  1879. print("too late %d\n", frnum);
  1880. poperror();
  1881. qunlock(&ep->wlock);
  1882. XEPRINT("ohci: wrote %ld\n\n", n);
  1883. ep->wdata01 = EDgetC(epx->ed);
  1884. return n;
  1885. }
  1886. static void
  1887. init(Usbhost*)
  1888. {
  1889. XIPRINT("ohci: init from devusb\n");
  1890. }
  1891. static void
  1892. scanpci(void)
  1893. {
  1894. ulong mem;
  1895. Ctlr *ctlr;
  1896. Pcidev *p;
  1897. static int already = 0;
  1898. if(already)
  1899. return;
  1900. already = 1;
  1901. p = nil;
  1902. while(p = pcimatch(p, 0, 0)) {
  1903. /*
  1904. * Find OHCI controllers (Programming Interface = 0x10).
  1905. */
  1906. if(p->ccrb != Pcibcserial || p->ccru != Pciscusb ||
  1907. p->ccrp != 0x10)
  1908. continue;
  1909. mem = p->mem[0].bar & ~0x0F;
  1910. XPRINT("usbohci: %x/%x port 0x%lux size 0x%x irq %d\n",
  1911. p->vid, p->did, mem, p->mem[0].size, p->intl);
  1912. if(mem == 0){
  1913. print("usbohci: failed to map registers\n");
  1914. continue;
  1915. }
  1916. if(p->intl == 0xFF || p->intl == 0) {
  1917. print("usbohci: no irq assigned for port %#lux\n", mem);
  1918. continue;
  1919. }
  1920. ctlr = malloc(sizeof(Ctlr));
  1921. ctlr->pcidev = p;
  1922. ctlr->base = vmap(mem, p->mem[0].size);
  1923. XPRINT("scanpci: ctlr 0x%lux, base 0x%lux\n", ctlr, ctlr->base);
  1924. pcisetbme(p);
  1925. pcisetpms(p, 0);
  1926. if(ctlrhead != nil)
  1927. ctlrtail->next = ctlr;
  1928. else
  1929. ctlrhead = ctlr;
  1930. ctlrtail = ctlr;
  1931. }
  1932. }
  1933. static int
  1934. reset(Usbhost *uh)
  1935. {
  1936. int i, linesize;
  1937. ulong io, fminterval, ctrl;
  1938. Ctlr *ctlr;
  1939. HCCA *atmp;
  1940. OHCI *ohci;
  1941. Pcidev *p;
  1942. QTree *qt;
  1943. /*
  1944. * data cache line size; probably doesn't matter on pc
  1945. * except that it must be a power of 2 for xspanalloc.
  1946. */
  1947. dcls = 32;
  1948. scanpci();
  1949. /*
  1950. * Any adapter matches if no uh->port is supplied,
  1951. * otherwise the ports must match.
  1952. */
  1953. for(ctlr = ctlrhead; ctlr != nil; ctlr = ctlr->next){
  1954. if(ctlr->active)
  1955. continue;
  1956. if(uh->port == 0 || uh->port == (uintptr)ctlr->base){
  1957. ctlr->active = 1;
  1958. break;
  1959. }
  1960. }
  1961. if(ctlr == nil)
  1962. return -1;
  1963. io = (uintptr)ctlr->base; /* TODO: correct? */
  1964. ohci = ctlr->base;
  1965. XPRINT("OHCI ctlr 0x%lux, base 0x%lux\n", ctlr, ohci);
  1966. p = ctlr->pcidev;
  1967. uh->ctlr = ctlr;
  1968. uh->port = io;
  1969. uh->irq = p->intl;
  1970. uh->tbdf = p->tbdf;
  1971. XPRINT("OHCI revision %ld.%ld\n", (ohci->revision >> 4) & 0xf,
  1972. ohci->revision & 0xf);
  1973. XPRINT("Host revision %ld.%ld\n", (ohci->hostrevision >> 4) & 0xf,
  1974. ohci->hostrevision & 0xf);
  1975. ctlr->nports = ohci->rhdesca & 0xff;
  1976. XPRINT("HcControl 0x%.8lux, %d ports\n", ohci->control, ctlr->nports);
  1977. delay(100); /* anything greater than 50 should ensure reset is done */
  1978. if(ohci->control == ~0){
  1979. ctlrhead = nil;
  1980. return -1;
  1981. }
  1982. /*
  1983. * usually enter here in reset, wait till its through,
  1984. * then do our own so we are on known timing conditions.
  1985. */
  1986. ohci->control = 0;
  1987. delay(100);
  1988. fminterval = ohci->fminterval;
  1989. /* legacy support register: turn off lunacy mode */
  1990. pcicfgw16(p, 0xc0, 0x2000);
  1991. /*
  1992. * transfer descs need at least 16 byte alignment, but
  1993. * align to dcache line size since the access will always be uncached.
  1994. * linesize must be a power of 2 for xspanalloc.
  1995. */
  1996. linesize = dcls;
  1997. if(linesize < 0x20)
  1998. linesize = 0x20;
  1999. ctlr->td.pool = va2ucva(xspanalloc(Ntd * sizeof(TD), linesize, 0));
  2000. if(ctlr->td.pool == nil)
  2001. panic("usbohci: no memory for TD pool");
  2002. for(i = Ntd - 1; --i >= 0;){
  2003. ctlr->td.pool[i].next = ctlr->td.free;
  2004. ctlr->td.free = &ctlr->td.pool[i];
  2005. }
  2006. ctlr->td.alloced = 0;
  2007. ctlr->ed.pool = va2ucva(xspanalloc(Ned*sizeof(ED), linesize, 0));
  2008. if(ctlr->ed.pool == nil)
  2009. panic("usbohci: no memory for ED pool");
  2010. for(i = Ned - 1; --i >= 0;){
  2011. ctlr->ed.pool[i].next = (ulong)ctlr->ed.free;
  2012. ctlr->ed.free = &ctlr->ed.pool[i];
  2013. }
  2014. ctlr->ed.alloced = 0;
  2015. atmp = xspanalloc(sizeof(HCCA), 256, 0);
  2016. if(atmp == nil)
  2017. panic("usbhreset: no memory for HCCA");
  2018. memset(atmp, 0, sizeof(*atmp));
  2019. ctlr->uchcca = atmp;
  2020. qt = mkqhtree(ctlr);
  2021. if(qt == nil){
  2022. panic("usb: can't allocate scheduling tree");
  2023. return -1;
  2024. }
  2025. ctlr->tree = qt;
  2026. ctlr->base = ohci;
  2027. /* time to move to rest then suspend mode. */
  2028. ohci->cmdsts = 1; /* reset the block */
  2029. while(ohci->cmdsts == 1)
  2030. continue; /* wait till reset complete, OHCI says 10us max. */
  2031. /*
  2032. * now that soft reset is done we are in suspend state.
  2033. * Setup registers which take in suspend state
  2034. * (will only be here for 2ms).
  2035. */
  2036. ohci->hcca = va2hcva(ctlr->uchcca);
  2037. OHCIsetControlHeadED(ctlr->base, 0);
  2038. ctlr->base->ctlcurred = 0;
  2039. OHCIsetBulkHeadED(ctlr->base, 0);
  2040. ctlr->base->bulkcurred = 0;
  2041. ohci->intrenable = Mie | Wdh | Ue;
  2042. ohci->control |= Cle | Ble | Ple | Ie | HcfsOperational;
  2043. /* set frame after operational */
  2044. ohci->fminterval = (fminterval &
  2045. ~(HcFmIntvl_FSMaxpack_MASK << HcFmIntvl_FSMaxpack_SHIFT)) |
  2046. 5120 << HcFmIntvl_FSMaxpack_SHIFT;
  2047. ohci->rhdesca = 1 << 9;
  2048. for(i = 0; i < ctlr->nports; i++)
  2049. ohci->rhportsts[i] = Spp | Spr;
  2050. delay(100);
  2051. ctrl = ohci->control;
  2052. if((ctrl & HcfsMask) != HcfsOperational){
  2053. XIPRINT("ohci: reset, take ctlr out of Suspend\n");
  2054. ctrl = (ctrl & ~HcfsMask) | HcfsOperational;
  2055. ohci->control = ctrl;
  2056. ohci->rhsts = Sgp;
  2057. }
  2058. p = ctlr->pcidev;
  2059. ctlr->irq = p->intl;
  2060. ctlr->tbdf = p->tbdf;
  2061. /*
  2062. * Linkage to the generic USB driver.
  2063. */
  2064. uh->init = init;
  2065. uh->interrupt = interrupt;
  2066. uh->portinfo = portinfo;
  2067. uh->portreset = portreset;
  2068. uh->portenable = portenable;
  2069. uh->epalloc = epalloc;
  2070. uh->epfree = epfree;
  2071. uh->epopen = epopen;
  2072. uh->epclose = epclose;
  2073. uh->epmode = epmode;
  2074. uh->epmaxpkt = epmaxpkt;
  2075. uh->read = read;
  2076. uh->write = write;
  2077. uh->tokin = Otokin;
  2078. uh->tokout = Otokout;
  2079. uh->toksetup = Otoksetup;
  2080. return 0;
  2081. }
  2082. void
  2083. usbohcilink(void)
  2084. {
  2085. addusbtype("ohci", reset);
  2086. }