VirtualBox

source: vbox/trunk/src/VBox/Devices/USB/DevOHCI.cpp@ 37641

Last change on this file since 37641 was 37636, checked in by vboxsync, 14 years ago

Changed FNIOMMMIOWRITE to take a const buffer pointer.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 196.5 KB
Line 
1/* $Id: DevOHCI.cpp 37636 2011-06-24 14:59:59Z vboxsync $ */
2/** @file
3 * DevOHCI - Open Host Controller Interface for USB.
4 */
5
6/*
7 * Copyright (C) 2006-2009 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/** @page pg_dev_ohci OHCI - Open Host Controller Interface Emulation.
19 *
20 * This component implements an OHCI USB controller. It is split roughly in
21 * to two main parts, the first part implements the register level
22 * specification of USB OHCI and the second part maintains the root hub (which
23 * is an integrated component of the device).
24 *
25 * The OHCI registers are used for the usual stuff like enabling and disabling
26 * interrupts. Since the USB time is divided in to 1ms frames and various
27 * interrupts may need to be triggered at frame boundary time, a timer-based
28 * approach was taken. Whenever the bus is enabled ohci->eof_timer will be set.
29 *
30 * The actual USB transfers are stored in main memory (along with endpoint and
31 * transfer descriptors). The ED's for all the control and bulk endpoints are
32 * found by consulting the HcControlHeadED and HcBulkHeadED registers
33 * respectively. Interrupt ED's are different, they are found by looking
34 * in the HCCA (another communication area in main memory).
35 *
36 * At the start of every frame (in function ohci_sof) we traverse all enabled
37 * ED lists and queue up as many transfers as possible. No attention is paid
38 * to control/bulk service ratios or bandwidth requirements since our USB
39 * could conceivably contain a dozen high speed busses so this would
40 * artificially limit the performance.
41 *
42 * Once we have a transfer ready to go (in function ohciServiceTd) we
43 * allocate an URB on the stack, fill in all the relevant fields and submit
44 * it using the VUSBIRhSubmitUrb function. The roothub device and the virtual
45 * USB core code (vusb.c) coordinates everything else from this point onwards.
46 *
47 * When the URB has been successfully handed to the lower level driver, our
48 * prepare callback gets called and we can remove the TD from the ED transfer
49 * list. This stops us queueing it twice while it completes.
50 * bird: no, we don't remove it because that confuses the guest! (=> crashes)
51 *
52 * Completed URBs are reaped at the end of every frame (in function
53 * ohci_frame_boundary). Our completion routine makes use of the ED and TD
54 * fields in the URB to store the physical addresses of the descriptors so
55 * that they may be modified in the roothub callbacks. Our completion
56 * routine (ohciRhXferComplete) carries out a number of tasks:
57 * -# Retires the TD associated with the transfer, setting the
58 * relevant error code etc.
59 * -# Updates done-queue interrupt timer and potentially causes
60 * a writeback of the done-queue.
61 * -# If the transfer was device-to-host, we copy the data in to
62 * the host memory.
63 *
64 * As for error handling OHCI allows for 3 retries before failing a transfer,
65 * an error count is stored in each transfer descriptor. A halt flag is also
66 * stored in the transfer descriptor. That allows for ED's to be disabled
67 * without stopping the bus and de-queuing them.
68 *
69 * When the bus is started and stopped we call VUSBIDevPowerOn/Off() on our
70 * roothub to indicate it's powering up and powering down. Whenever we power
71 * down, the USB core makes sure to synchronously complete all outstanding
72 * requests so that the OHCI is never seen in an inconsistent state by the
73 * guest OS (Transfers are not meant to be unlinked until they've actually
74 * completed, but we can't do that unless we work synchronously, so we just
75 * have to fake it).
76 * bird: we do work synchronously now, anything causes guest crashes.
77 */
78
79
80/*******************************************************************************
81* Header Files *
82*******************************************************************************/
83#define LOG_GROUP LOG_GROUP_DEV_USB
84#include <VBox/pci.h>
85#include <VBox/vmm/pdm.h>
86#include <VBox/vmm/mm.h>
87#include <VBox/err.h>
88#include <VBox/log.h>
89#include <iprt/assert.h>
90#include <iprt/string.h>
91#include <iprt/asm.h>
92#include <iprt/asm-math.h>
93#ifdef IN_RING3
94# include <iprt/alloca.h>
95# include <iprt/mem.h>
96# include <iprt/thread.h>
97# include <iprt/uuid.h>
98#endif
99#include <VBox/vusb.h>
100#include "VBoxDD.h"
101
102
103/*******************************************************************************
104* Structures and Typedefs *
105*******************************************************************************/
106/** The saved state version. */
107#define OHCI_SAVED_STATE_VERSION 4
108/** The saved state version used in 3.0 and earlier.
109 *
110 * @remarks Because of the SSMR3MemPut/Get laziness we ended up with an
111 * accidental format change between 2.0 and 2.1 that didn't get its own
112 * version number. It is therefore not possible to restore states from
113 * 2.0 and earlier with 2.1 and later. */
114#define OHCI_SAVED_STATE_VERSION_MEM_HELL 3
115
116
117/* Number of Downstream Ports on the root hub, if you change this
118 * you need to add more status register words to the 'opreg' array
119 */
120#define OHCI_NDP 8
121
122/** Pointer to OHCI device data. */
123typedef struct OHCI *POHCI;
124
125
126/**
127 * An OHCI root hub port.
128 */
129typedef struct OHCIHUBPORT
130{
131 /** The port register. */
132 uint32_t fReg;
133#if HC_ARCH_BITS == 64
134 uint32_t Alignment0; /**< Align the pointer correctly. */
135#endif
136 /** The device attached to the port. */
137 R3PTRTYPE(PVUSBIDEVICE) pDev;
138} OHCIHUBPORT;
139#if HC_ARCH_BITS == 64
140AssertCompile(sizeof(OHCIHUBPORT) == 16); /* saved state */
141#endif
142/** Pointer to an OHCI hub port. */
143typedef OHCIHUBPORT *POHCIHUBPORT;
144
145/**
146 * The OHCI root hub.
147 *
148 * @implements PDMIBASE
149 * @implements VUSBIROOTHUBPORT
150 * @implements PDMILEDPORTS
151 */
152typedef struct ohci_roothub
153{
154 /** Pointer to the base interface of the VUSB RootHub. */
155 R3PTRTYPE(PPDMIBASE) pIBase;
156 /** Pointer to the connector interface of the VUSB RootHub. */
157 R3PTRTYPE(PVUSBIROOTHUBCONNECTOR) pIRhConn;
158 /** Pointer to the device interface of the VUSB RootHub. */
159 R3PTRTYPE(PVUSBIDEVICE) pIDev;
160 /** The base interface exposed to the roothub driver. */
161 PDMIBASE IBase;
162 /** The roothub port interface exposed to the roothub driver. */
163 VUSBIROOTHUBPORT IRhPort;
164
165 /** The LED. */
166 PDMLED Led;
167 /** The LED ports. */
168 PDMILEDPORTS ILeds;
169 /** Partner of ILeds. */
170 R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector;
171
172 uint32_t status;
173 uint32_t desc_a;
174 uint32_t desc_b;
175#if HC_ARCH_BITS == 64
176 uint32_t Alignment0; /**< Align aPorts on a 8 byte boundary. */
177#endif
178 OHCIHUBPORT aPorts[OHCI_NDP];
179 R3PTRTYPE(POHCI) pOhci;
180} OHCIROOTHUB;
181#if HC_ARCH_BITS == 64
182AssertCompile(sizeof(OHCIROOTHUB) == 280); /* saved state */
183#endif
184/** Pointer to the OHCI root hub. */
185typedef OHCIROOTHUB *POHCIROOTHUB;
186
187
188/**
189 * Data used for reattaching devices on a state load.
190 */
191typedef struct ohci_load {
192 /** Timer used once after state load to inform the guest about new devices.
193 * We do this to be sure the guest get any disconnect / reconnect on the
194 * same port. */
195 PTMTIMERR3 pTimer;
196 /** Number of detached devices. */
197 unsigned cDevs;
198 /** Array of devices which were detached. */
199 PVUSBIDEVICE apDevs[OHCI_NDP];
200} OHCILOAD;
201/** Pointer to an OHCILOAD structure. */
202typedef OHCILOAD *POHCILOAD;
203
204
205/**
206 * OHCI device data.
207 */
208typedef struct OHCI
209{
210 /** The PCI device. */
211 PCIDEVICE PciDev;
212
213 /** Pointer to the device instance - R3 ptr. */
214 PPDMDEVINSR3 pDevInsR3;
215 /** The End-Of-Frame timer - R3 Ptr. */
216 PTMTIMERR3 pEndOfFrameTimerR3;
217
218 /** Pointer to the device instance - R0 ptr */
219 PPDMDEVINSR0 pDevInsR0;
220 /** The End-Of-Frame timer - R0 Ptr. */
221 PTMTIMERR0 pEndOfFrameTimerR0;
222
223 /** Pointer to the device instance - RC ptr. */
224 PPDMDEVINSRC pDevInsRC;
225 /** The End-Of-Frame timer - RC Ptr. */
226 PTMTIMERRC pEndOfFrameTimerRC;
227
228 /** Start of current frame. */
229 uint64_t SofTime;
230 /* done queue interrupt counter */
231 uint32_t dqic : 3;
232 /** frame number overflow. */
233 uint32_t fno : 1;
234 /** Address of the MMIO region assigned by PCI. */
235 RTGCPHYS32 MMIOBase;
236
237 /* Root hub device */
238 OHCIROOTHUB RootHub;
239
240 /* OHCI registers */
241
242 /** @name Control partition
243 * @{ */
244 /** HcControl. */
245 uint32_t ctl;
246 /** HcCommandStatus. */
247 uint32_t status;
248 /** HcInterruptStatus. */
249 uint32_t intr_status;
250 /** HcInterruptEnabled. */
251 uint32_t intr;
252 /** @} */
253
254 /** @name Memory pointer partition
255 * @{ */
256 /** HcHCCA. */
257 uint32_t hcca;
258 /** HcPeriodCurrentEd. */
259 uint32_t per_cur;
260 /** HcControlCurrentED. */
261 uint32_t ctrl_cur;
262 /** HcControlHeadED. */
263 uint32_t ctrl_head;
264 /** HcBlockCurrendED. */
265 uint32_t bulk_cur;
266 /** HcBlockHeadED. */
267 uint32_t bulk_head;
268 /** HcDoneHead. */
269 uint32_t done;
270 /** @} */
271
272 /** @name Frame counter partition
273 * @{ */
274 /** HcFmInterval.FSMPS - FSLargestDataPacket */
275 uint32_t fsmps : 15;
276 /** HcFmInterval.FIT - FrameItervalToggle */
277 uint32_t fit : 1;
278 /** HcFmInterval.FI - FrameInterval */
279 uint32_t fi : 14;
280 /** HcFmRemaining.FRT - toggle bit. */
281 uint32_t frt : 1;
282 /** HcFmNumber.
283 * @remark The register size is 16-bit, but for debugging and performance
284 * reasons we maintain a 32-bit counter. */
285 uint32_t HcFmNumber;
286 /** HcPeriodicStart */
287 uint32_t pstart;
288 /** @} */
289
290 /** The number of virtual time ticks per frame. */
291 uint64_t cTicksPerFrame;
292 /** The number of virtual time ticks per USB bus tick. */
293 uint64_t cTicksPerUsbTick;
294
295 /** Number of in-flight TDs. */
296 unsigned cInFlight;
297 unsigned Alignment1; /**< Align aInFlight on a 8 byte boundary. */
298 /** Array of in-flight TDs. */
299 struct ohci_td_in_flight
300 {
301 /** Address of the transport descriptor. */
302 uint32_t GCPhysTD;
303#if HC_ARCH_BITS == 64
304 uint32_t Alignment0; /**< Alignment pUrb correctly. */
305#endif
306 /** Pointer to the URB. */
307 R3PTRTYPE(PVUSBURB) pUrb;
308 } aInFlight[257];
309
310 /** Number of in-done-queue TDs. */
311 unsigned cInDoneQueue;
312 /** Array of in-done-queue TDs. */
313 struct ohci_td_in_done_queue
314 {
315 /** Address of the transport descriptor. */
316 uint32_t GCPhysTD;
317 } aInDoneQueue[64];
318 /** When the tail of the done queue was added.
319 * Used to calculate the age of the done queue. */
320 uint32_t u32FmDoneQueueTail;
321#if R3_ARCH_BITS == 32
322 /** Align pLoad, the stats and the struct size correctly. */
323 uint32_t Alignment2;
324#endif
325 /** Pointer to state load data. */
326 R3PTRTYPE(POHCILOAD) pLoad;
327
328 /** Detected canceled isochronous URBs. */
329 STAMCOUNTER StatCanceledIsocUrbs;
330 /** Detected canceled general URBs. */
331 STAMCOUNTER StatCanceledGenUrbs;
332 /** Dropped URBs (endpoint halted, or URB canceled). */
333 STAMCOUNTER StatDroppedUrbs;
334 /** Profiling ohciFrameBoundaryTimer. */
335 STAMPROFILE StatTimer;
336
337 /** This member and all the following are not part of saved state. */
338 uint64_t SavedStateEnd;
339
340 /** VM timer frequency used for frame timer calculations. */
341 uint64_t u64TimerHz;
342 /** Number of USB work cycles with no transfers. */
343 uint32_t cIdleCycles;
344 /** Current frame timer rate (default 1000). */
345 uint32_t uFrameRate;
346 /** Idle detection flag; must be cleared at start of frame */
347 bool fIdle;
348 /** A flag indicating that the bulk list may have in-flight URBs. */
349 bool fBulkNeedsCleaning;
350
351 uint32_t Alignment3; /**< Align size on a 8 byte boundary. */
352} OHCI;
353
354/* Standard OHCI bus speed */
355#define OHCI_DEFAULT_TIMER_FREQ 1000
356
357/* Host Controller Communications Area */
358#define OHCI_HCCA_NUM_INTR 32
359#define OHCI_HCCA_OFS (OHCI_HCCA_NUM_INTR * sizeof(uint32_t))
360struct ohci_hcca
361{
362 uint16_t frame;
363 uint16_t pad;
364 uint32_t done;
365};
366AssertCompileSize(ohci_hcca, 8);
367
368/** @name OHCI Endpoint Descriptor
369 * @{ */
370
371#define ED_PTR_MASK (~(uint32_t)0xf)
372#define ED_HWINFO_MPS 0x07ff0000
373#define ED_HWINFO_ISO RT_BIT(15)
374#define ED_HWINFO_SKIP RT_BIT(14)
375#define ED_HWINFO_LOWSPEED RT_BIT(13)
376#define ED_HWINFO_IN RT_BIT(12)
377#define ED_HWINFO_OUT RT_BIT(11)
378#define ED_HWINFO_DIR (RT_BIT(11) | RT_BIT(12))
379#define ED_HWINFO_ENDPOINT 0x780 /* 4 bits */
380#define ED_HWINFO_ENDPOINT_SHIFT 7
381#define ED_HWINFO_FUNCTION 0x7f /* 7 bits */
382#define ED_HEAD_CARRY RT_BIT(1)
383#define ED_HEAD_HALTED RT_BIT(0)
384
385/**
386 * OHCI Endpoint Descriptor.
387 */
388typedef struct OHCIED
389{
390 /** Flags and stuff. */
391 uint32_t hwinfo;
392 /** TailP - TD Queue Tail pointer. Bits 0-3 ignored / preserved. */
393 uint32_t TailP;
394 /** HeadP - TD Queue head pointer. Bit 0 - Halted, Bit 1 - toggleCarry. Bit 2&3 - 0. */
395 uint32_t HeadP;
396 /** NextED - Next Endpoint Descriptor. Bits 0-3 ignored / preserved. */
397 uint32_t NextED;
398} OHCIED, *POHCIED;
399typedef const OHCIED *PCOHCIED;
400AssertCompileSize(OHCIED, 16);
401
402/** @} */
403
404
405/** @name Completion Codes
406 * @{ */
407#define OHCI_CC_NO_ERROR (UINT32_C(0x00) << 28)
408#define OHCI_CC_CRC (UINT32_C(0x01) << 28)
409#define OHCI_CC_STALL (UINT32_C(0x04) << 28)
410#define OHCI_CC_DEVICE_NOT_RESPONDING (UINT32_C(0x05) << 28)
411#define OHCI_CC_DNR OHCI_CC_DEVICE_NOT_RESPONDING
412#define OHCI_CC_PID_CHECK_FAILURE (UINT32_C(0x06) << 28)
413#define OHCI_CC_UNEXPECTED_PID (UINT32_C(0x07) << 28)
414#define OHCI_CC_DATA_OVERRUN (UINT32_C(0x08) << 28)
415#define OHCI_CC_DATA_UNDERRUN (UINT32_C(0x09) << 28)
416/* 0x0a..0x0b - reserved */
417#define OHCI_CC_BUFFER_OVERRUN (UINT32_C(0x0c) << 28)
418#define OHCI_CC_BUFFER_UNDERRUN (UINT32_C(0x0d) << 28)
419#define OHCI_CC_NOT_ACCESSED_0 (UINT32_C(0x0e) << 28)
420#define OHCI_CC_NOT_ACCESSED_1 (UINT32_C(0x0f) << 28)
421/** @} */
422
423
424/** @name OHCI General transfer descriptor
425 * @{ */
426
427/** Error count (EC) shift. */
428#define TD_ERRORS_SHIFT 26
429/** Error count max. (One greater than what the EC field can hold.) */
430#define TD_ERRORS_MAX 4
431
432/** CC - Condition code mask. */
433#define TD_HWINFO_CC (UINT32_C(0xf0000000))
434#define TD_HWINFO_CC_SHIFT 28
435/** EC - Error count. */
436#define TD_HWINFO_ERRORS (RT_BIT(26) | RT_BIT(27))
437/** T - Data toggle. */
438#define TD_HWINFO_TOGGLE (RT_BIT(24) | RT_BIT(25))
439#define TD_HWINFO_TOGGLE_HI (RT_BIT(25))
440#define TD_HWINFO_TOGGLE_LO (RT_BIT(24))
441/** DI - Delay interrupt. */
442#define TD_HWINFO_DI (RT_BIT(21) | RT_BIT(22) | RT_BIT(23))
443#define TD_HWINFO_IN (RT_BIT(20))
444#define TD_HWINFO_OUT (RT_BIT(19))
445/** DP - Direction / PID. */
446#define TD_HWINFO_DIR (RT_BIT(19) | RT_BIT(20))
447/** R - Buffer rounding. */
448#define TD_HWINFO_ROUNDING (RT_BIT(18))
449/** Bits that are reserved / unknown. */
450#define TD_HWINFO_UNKNOWN_MASK (UINT32_C(0x0003ffff))
451
452/** SETUP - to endpoint. */
453#define OHCI_TD_DIR_SETUP 0x0
454/** OUT - to endpoint. */
455#define OHCI_TD_DIR_OUT 0x1
456/** IN - from endpoint. */
457#define OHCI_TD_DIR_IN 0x2
458/** Reserved. */
459#define OHCI_TD_DIR_RESERVED 0x3
460
461/**
462 * OHCI general transfer descriptor
463 */
464typedef struct OHCITD
465{
466 uint32_t hwinfo;
467 /** CBP - Current Buffer Pointer. (32-bit physical address) */
468 uint32_t cbp;
469 /** NextTD - Link to the next transfer descriptor. (32-bit physical address, dword aligned) */
470 uint32_t NextTD;
471 /** BE - Buffer End (inclusive). (32-bit physical address) */
472 uint32_t be;
473} OHCITD, *POHCITD;
474typedef const OHCITD *PCOHCITD;
475AssertCompileSize(OHCIED, 16);
476/** @} */
477
478
479/** @name OHCI isochronous transfer descriptor.
480 * @{ */
481/** SF - Start frame number. */
482#define ITD_HWINFO_SF 0xffff
483/** DI - Delay interrupt. (TD_HWINFO_DI) */
484#define ITD_HWINFO_DI (RT_BIT(21) | RT_BIT(22) | RT_BIT(23))
485#define ITD_HWINFO_DI_SHIFT 21
486/** FC - Frame count. */
487#define ITD_HWINFO_FC (RT_BIT(24) | RT_BIT(25) | RT_BIT(26))
488#define ITD_HWINFO_FC_SHIFT 24
489/** CC - Condition code mask. (=TD_HWINFO_CC) */
490#define ITD_HWINFO_CC UINT32_C(0xf0000000)
491#define ITD_HWINFO_CC_SHIFT 28
492/** The buffer page 0 mask (lower 12 bits are ignored). */
493#define ITD_BP0_MASK UINT32_C(0xfffff000)
494
495#define ITD_NUM_PSW 8
496/** OFFSET - offset of the package into the buffer page.
497 * (Only valid when CC set to Not Accessed.)
498 *
499 * Note that the top bit of the OFFSET field is overlapping with the
500 * first bit in the CC field. This is ok because both 0xf and 0xe are
501 * defined as "Not Accessed".
502 */
503#define ITD_PSW_OFFSET 0x1fff
504/** SIZE field mask for IN bound transfers.
505 * (Only valid when CC isn't Not Accessed.)*/
506#define ITD_PSW_SIZE 0x07ff
507/** CC field mask.
508 * USed to indicate the format of SIZE (Not Accessed -> OFFSET). */
509#define ITD_PSW_CC 0xf000
510#define ITD_PSW_CC_SHIFT 12
511
512/**
513 * OHCI isochronous transfer descriptor.
514 */
515typedef struct OHCIITD
516{
517 uint32_t HwInfo;
518 /** BP0 - Buffer Page 0. The lower 12 bits are ignored. */
519 uint32_t BP0;
520 /** NextTD - Link to the next transfer descriptor. (32-bit physical address, dword aligned) */
521 uint32_t NextTD;
522 /** BE - Buffer End (inclusive). (32-bit physical address) */
523 uint32_t BE;
524 /** (OffsetN/)PSWN - package status word array (0..7).
525 * The format varies depending on whether the package has been completed or not. */
526 uint16_t aPSW[ITD_NUM_PSW];
527} OHCIITD, *POHCIITD;
528typedef const OHCIITD *PCOHCIITD;
529AssertCompileSize(OHCIITD, 32);
530/** @} */
531
532/**
533 * OHCI register operator.
534 */
535typedef struct ohci_opreg
536{
537 const char *pszName;
538 int (*pfnRead )(POHCI ohci, uint32_t iReg, uint32_t *pu32Value);
539 int (*pfnWrite)(POHCI ohci, uint32_t iReg, uint32_t u32Value);
540} OHCIOPREG;
541
542
543/* OHCI Local stuff */
544#define OHCI_CTL_CBSR ((1<<0)|(1<<1))
545#define OHCI_CTL_PLE (1<<2)
546#define OHCI_CTL_IE (1<<3)
547#define OHCI_CTL_CLE (1<<4)
548#define OHCI_CTL_BLE (1<<5)
549#define OHCI_CTL_HCFS ((1<<6)|(1<<7))
550#define OHCI_USB_RESET 0x00
551#define OHCI_USB_RESUME 0x40
552#define OHCI_USB_OPERATIONAL 0x80
553#define OHCI_USB_SUSPEND 0xc0
554#define OHCI_CTL_IR (1<<8)
555#define OHCI_CTL_RWC (1<<9)
556#define OHCI_CTL_RWE (1<<10)
557
558#define OHCI_STATUS_HCR (1<<0)
559#define OHCI_STATUS_CLF (1<<1)
560#define OHCI_STATUS_BLF (1<<2)
561#define OHCI_STATUS_OCR (1<<3)
562#define OHCI_STATUS_SOC ((1<<6)|(1<<7))
563
564/** @name Interrupt Status and Enabled/Disabled Flags
565 * @{ */
566/** SO - Scheduling overrun. */
567#define OHCI_INTR_SCHEDULEING_OVERRUN RT_BIT(0)
568/** WDH - HcDoneHead writeback. */
569#define OHCI_INTR_WRITE_DONE_HEAD RT_BIT(1)
570/** SF - Start of frame. */
571#define OHCI_INTR_START_OF_FRAME RT_BIT(2)
572/** RD - Resume detect. */
573#define OHCI_INTR_RESUME_DETECT RT_BIT(3)
574/** UE - Unrecoverable error. */
575#define OHCI_INTR_UNRECOVERABLE_ERROR RT_BIT(4)
576/** FNO - Frame number overflow. */
577#define OHCI_INTR_FRAMENUMBER_OVERFLOW RT_BIT(5)
578/** RHSC- Root hub status change. */
579#define OHCI_INTR_ROOT_HUB_STATUS_CHANGE RT_BIT(6)
580/** OC - Ownership change. */
581#define OHCI_INTR_OWNERSHIP_CHANGE RT_BIT(30)
582/** MIE - Master interrupt enable. */
583#define OHCI_INTR_MASTER_INTERRUPT_ENABLED RT_BIT(31)
584/** @} */
585
586#define OHCI_HCCA_SIZE 0x100
587#define OHCI_HCCA_MASK UINT32_C(0xffffff00)
588
589#define OHCI_FMI_FI UINT32_C(0x00003fff)
590#define OHCI_FMI_FSMPS UINT32_C(0x7fff0000)
591#define OHCI_FMI_FSMPS_SHIFT 16
592#define OHCI_FMI_FIT UINT32_C(0x80000000)
593#define OHCI_FMI_FIT_SHIFT 31
594
595#define OHCI_FR_RT RT_BIT_32(31)
596
597#define OHCI_LS_THRESH 0x628
598
599#define OHCI_RHA_NDP (0xff)
600#define OHCI_RHA_PSM RT_BIT_32(8)
601#define OHCI_RHA_NPS RT_BIT_32(9)
602#define OHCI_RHA_DT RT_BIT_32(10)
603#define OHCI_RHA_OCPM RT_BIT_32(11)
604#define OHCI_RHA_NOCP RT_BIT_32(12)
605#define OHCI_RHA_POTPGP UINT32_C(0xff000000)
606
607#define OHCI_RHS_LPS RT_BIT_32(0)
608#define OHCI_RHS_OCI RT_BIT_32(1)
609#define OHCI_RHS_DRWE RT_BIT_32(15)
610#define OHCI_RHS_LPSC RT_BIT_32(16)
611#define OHCI_RHS_OCIC RT_BIT_32(17)
612#define OHCI_RHS_CRWE RT_BIT_32(31)
613
614/** @name HcRhPortStatus[n] - RH Port Status register (read).
615 * @{ */
616/** CCS - CurrentConnectionStatus - 0 = no device, 1 = device. */
617#define OHCI_PORT_CCS RT_BIT(0)
618/** PES - PortEnableStatus. */
619#define OHCI_PORT_PES RT_BIT(1)
620/** PSS - PortSuspendStatus */
621#define OHCI_PORT_PSS RT_BIT(2)
622/** POCI- PortOverCurrentIndicator. */
623#define OHCI_PORT_POCI RT_BIT(3)
624/** PRS - PortResetStatus */
625#define OHCI_PORT_PRS RT_BIT(4)
626/** PPS - PortPowerStatus */
627#define OHCI_PORT_PPS RT_BIT(8)
628/** LSDA - LowSpeedDeviceAttached */
629#define OHCI_PORT_LSDA RT_BIT(9)
630/** CSC - ConnectStatusChange */
631#define OHCI_PORT_CSC RT_BIT(16)
632/** PESC - PortEnableStatusChange */
633#define OHCI_PORT_PESC RT_BIT(17)
634/** PSSC - PortSuspendStatusChange */
635#define OHCI_PORT_PSSC RT_BIT(18)
636/** OCIC - OverCurrentIndicatorChange */
637#define OHCI_PORT_OCIC RT_BIT(19)
638/** PRSC - PortResetStatusChange */
639#define OHCI_PORT_PRSC RT_BIT(20)
640/** @} */
641
642
643/** @name HcRhPortStatus[n] - Root Hub Port Status Registers - read.
644 * @{ */
645/** CCS - CurrentConnectStatus - 0 = no device, 1 = device. */
646#define OHCI_PORT_R_CURRENT_CONNECT_STATUS RT_BIT(0)
647/** PES - PortEnableStatus. */
648#define OHCI_PORT_R_ENABLE_STATUS RT_BIT(1)
649/** PSS - PortSuspendStatus */
650#define OHCI_PORT_R_SUSPEND_STATUS RT_BIT(2)
651/** POCI- PortOverCurrentIndicator. */
652#define OHCI_PORT_R_OVER_CURRENT_INDICATOR RT_BIT(3)
653/** PRS - PortResetStatus */
654#define OHCI_PORT_R_RESET_STATUS RT_BIT(4)
655/** PPS - PortPowerStatus */
656#define OHCI_PORT_R_POWER_STATUS RT_BIT(8)
657/** LSDA - LowSpeedDeviceAttached */
658#define OHCI_PORT_R_LOW_SPEED_DEVICE_ATTACHED RT_BIT(9)
659/** CSC - ConnectStatusChange */
660#define OHCI_PORT_R_CONNECT_STATUS_CHANGE RT_BIT(16)
661/** PESC - PortEnableStatusChange */
662#define OHCI_PORT_R_ENABLE_STATUS_CHANGE RT_BIT(17)
663/** PSSC - PortSuspendStatusChange */
664#define OHCI_PORT_R_SUSPEND_STATUS_CHANGE RT_BIT(18)
665/** OCIC - OverCurrentIndicatorChange */
666#define OHCI_PORT_R_OVER_CURRENT_INDICATOR_CHANGE RT_BIT(19)
667/** PRSC - PortResetStatusChange */
668#define OHCI_PORT_R_RESET_STATUS_CHANGE RT_BIT(20)
669/** @} */
670
671/** @name HcRhPortStatus[n] - Root Hub Port Status Registers - write.
672 * @{ */
673/** CCS - ClearPortEnable. */
674#define OHCI_PORT_W_CLEAR_ENABLE RT_BIT(0)
675/** PES - SetPortEnable. */
676#define OHCI_PORT_W_SET_ENABLE RT_BIT(1)
677/** PSS - SetPortSuspend */
678#define OHCI_PORT_W_SET_SUSPEND RT_BIT(2)
679/** POCI- ClearSuspendStatus. */
680#define OHCI_PORT_W_CLEAR_SUSPEND_STATUS RT_BIT(3)
681/** PRS - SetPortReset */
682#define OHCI_PORT_W_SET_RESET RT_BIT(4)
683/** PPS - SetPortPower */
684#define OHCI_PORT_W_SET_POWER RT_BIT(8)
685/** LSDA - ClearPortPower */
686#define OHCI_PORT_W_CLEAR_POWER RT_BIT(9)
687/** CSC - ClearConnectStatusChange */
688#define OHCI_PORT_W_CLEAR_CSC RT_BIT(16)
689/** PESC - PortEnableStatusChange */
690#define OHCI_PORT_W_CLEAR_PESC RT_BIT(17)
691/** PSSC - PortSuspendStatusChange */
692#define OHCI_PORT_W_CLEAR_PSSC RT_BIT(18)
693/** OCIC - OverCurrentIndicatorChange */
694#define OHCI_PORT_W_CLEAR_OCIC RT_BIT(19)
695/** PRSC - PortResetStatusChange */
696#define OHCI_PORT_W_CLEAR_PRSC RT_BIT(20)
697/** The mask of bit which are used to clear themselves. */
698#define OHCI_PORT_W_CLEAR_CHANGE_MASK ( OHCI_PORT_W_CLEAR_CSC | OHCI_PORT_W_CLEAR_PESC | OHCI_PORT_W_CLEAR_PSSC \
699 | OHCI_PORT_W_CLEAR_OCIC | OHCI_PORT_W_CLEAR_PRSC)
700/** @} */
701
702
703#ifndef VBOX_DEVICE_STRUCT_TESTCASE
704/*******************************************************************************
705* Global Variables *
706*******************************************************************************/
707#if defined(LOG_ENABLED) && defined(IN_RING3)
708static bool g_fLogBulkEPs = false;
709static bool g_fLogControlEPs = false;
710static bool g_fLogInterruptEPs = false;
711#endif
712#ifdef IN_RING3
713/**
714 * SSM descriptor table for the OHCI structure.
715 */
716static SSMFIELD const g_aOhciFields[] =
717{
718 SSMFIELD_ENTRY( OHCI, SofTime),
719 SSMFIELD_ENTRY_CUSTOM( dpic+fno, RT_OFFSETOF(OHCI, SofTime) + RT_SIZEOFMEMB(OHCI, SofTime), 4),
720 SSMFIELD_ENTRY( OHCI, RootHub.status),
721 SSMFIELD_ENTRY( OHCI, RootHub.desc_a),
722 SSMFIELD_ENTRY( OHCI, RootHub.desc_b),
723 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[0].fReg),
724 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[1].fReg),
725 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[2].fReg),
726 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[3].fReg),
727 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[4].fReg),
728 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[5].fReg),
729 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[6].fReg),
730 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[7].fReg),
731 SSMFIELD_ENTRY( OHCI, ctl),
732 SSMFIELD_ENTRY( OHCI, status),
733 SSMFIELD_ENTRY( OHCI, intr_status),
734 SSMFIELD_ENTRY( OHCI, intr),
735 SSMFIELD_ENTRY( OHCI, hcca),
736 SSMFIELD_ENTRY( OHCI, per_cur),
737 SSMFIELD_ENTRY( OHCI, ctrl_cur),
738 SSMFIELD_ENTRY( OHCI, ctrl_head),
739 SSMFIELD_ENTRY( OHCI, bulk_cur),
740 SSMFIELD_ENTRY( OHCI, bulk_head),
741 SSMFIELD_ENTRY( OHCI, done),
742 SSMFIELD_ENTRY_CUSTOM( fsmps+fit+fi+frt, RT_OFFSETOF(OHCI, done) + RT_SIZEOFMEMB(OHCI, done), 4),
743 SSMFIELD_ENTRY( OHCI, HcFmNumber),
744 SSMFIELD_ENTRY( OHCI, pstart),
745 SSMFIELD_ENTRY_TERM()
746};
747#endif
748
749
750/*******************************************************************************
751* Internal Functions *
752*******************************************************************************/
753RT_C_DECLS_BEGIN
754#ifdef IN_RING3
755/* Update host controller state to reflect a device attach */
756static void rhport_power(POHCIROOTHUB pRh, unsigned iPort, bool fPowerUp);
757static void ohciBusResume(POHCI ohci, bool fHardware);
758
759static DECLCALLBACK(void) ohciRhXferCompletion(PVUSBIROOTHUBPORT pInterface, PVUSBURB pUrb);
760static DECLCALLBACK(bool) ohciRhXferError(PVUSBIROOTHUBPORT pInterface, PVUSBURB pUrb);
761
762static int ohci_in_flight_find(POHCI pOhci, uint32_t GCPhysTD);
763# if defined(VBOX_STRICT) || defined(LOG_ENABLED)
764static int ohci_in_done_queue_find(POHCI pOhci, uint32_t GCPhysTD);
765# endif
766static DECLCALLBACK(void) ohciR3LoadReattachDevices(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser);
767#endif /* IN_RING3 */
768RT_C_DECLS_END
769
770
771/**
772 * Update PCI IRQ levels
773 */
774static void ohciUpdateInterrupt(POHCI ohci, const char *msg)
775{
776 int level = 0;
777
778 if ( (ohci->intr & OHCI_INTR_MASTER_INTERRUPT_ENABLED)
779 && (ohci->intr_status & ohci->intr)
780 && !(ohci->ctl & OHCI_CTL_IR))
781 level = 1;
782
783 PDMDevHlpPCISetIrq(ohci->CTX_SUFF(pDevIns), 0, level);
784 if (level)
785 {
786 uint32_t val = ohci->intr_status & ohci->intr;
787 Log2(("ohci: Fired off interrupt %#010x - SO=%d WDH=%d SF=%d RD=%d UE=%d FNO=%d RHSC=%d OC=%d - %s\n",
788 val, val & 1, (val >> 1) & 1, (val >> 2) & 1, (val >> 3) & 1, (val >> 4) & 1, (val >> 5) & 1,
789 (val >> 6) & 1, (val >> 30) & 1, msg)); NOREF(val); NOREF(msg);
790 }
791}
792
793/**
794 * Set an interrupt, use the wrapper ohciSetInterrupt.
795 */
796DECLINLINE(void) ohciSetInterruptInt(POHCI ohci, uint32_t intr, const char *msg)
797{
798 if ( (ohci->intr_status & intr) == intr )
799 return;
800 ohci->intr_status |= intr;
801 ohciUpdateInterrupt(ohci, msg);
802}
803
804/**
805 * Set an interrupt wrapper macro for logging purposes.
806 */
807#define ohciSetInterrupt(ohci, intr) ohciSetInterruptInt(ohci, intr, #intr)
808
809
810#ifdef IN_RING3
811
812/* Carry out a hardware remote wakeup */
813static void ohci_remote_wakeup(POHCI pOhci)
814{
815 if ((pOhci->ctl & OHCI_CTL_HCFS) != OHCI_USB_SUSPEND)
816 return;
817 if (!(pOhci->RootHub.status & OHCI_RHS_DRWE))
818 return;
819 ohciBusResume(pOhci, true /* hardware */);
820}
821
822
823/**
824 * Query interface method for the roothub LUN.
825 */
826static DECLCALLBACK(void *) ohciRhQueryInterface(PPDMIBASE pInterface, const char *pszIID)
827{
828 POHCI pThis = RT_FROM_MEMBER(pInterface, OHCI, RootHub.IBase);
829 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->RootHub.IBase);
830 PDMIBASE_RETURN_INTERFACE(pszIID, VUSBIROOTHUBPORT, &pThis->RootHub.IRhPort);
831 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThis->RootHub.ILeds);
832 return NULL;
833}
834
835/**
836 * Gets the pointer to the status LED of a unit.
837 *
838 * @returns VBox status code.
839 * @param pInterface Pointer to the interface structure containing the called function pointer.
840 * @param iLUN The unit which status LED we desire.
841 * @param ppLed Where to store the LED pointer.
842 */
843static DECLCALLBACK(int) ohciRhQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
844{
845 POHCI pOhci = (POHCI)((uintptr_t)pInterface - RT_OFFSETOF(OHCI, RootHub.ILeds));
846 if (iLUN == 0)
847 {
848 *ppLed = &pOhci->RootHub.Led;
849 return VINF_SUCCESS;
850 }
851 return VERR_PDM_LUN_NOT_FOUND;
852}
853
854
855/** Converts a OHCI.roothub.IRhPort pointer to a POHCI. */
856#define VUSBIROOTHUBPORT_2_OHCI(pInterface) ((POHCI)( (uintptr_t)(pInterface) - RT_OFFSETOF(OHCI, RootHub.IRhPort) ))
857
858
859/**
860 * Get the number of available ports in the hub.
861 *
862 * @returns The number of ports available.
863 * @param pInterface Pointer to this structure.
864 * @param pAvailable Bitmap indicating the available ports. Set bit == available port.
865 */
866static DECLCALLBACK(unsigned) ohciRhGetAvailablePorts(PVUSBIROOTHUBPORT pInterface, PVUSBPORTBITMAP pAvailable)
867{
868 POHCI pOhci = VUSBIROOTHUBPORT_2_OHCI(pInterface);
869 unsigned iPort;
870 unsigned cPorts = 0;
871
872 memset(pAvailable, 0, sizeof(*pAvailable));
873
874 PDMCritSectEnter(pOhci->pDevInsR3->pCritSectRoR3, VERR_IGNORED);
875 for (iPort = 0; iPort < RT_ELEMENTS(pOhci->RootHub.aPorts); iPort++)
876 {
877 if (!pOhci->RootHub.aPorts[iPort].pDev)
878 {
879 cPorts++;
880 ASMBitSet(pAvailable, iPort + 1);
881 }
882 }
883 PDMCritSectLeave(pOhci->pDevInsR3->pCritSectRoR3);
884
885 return cPorts;
886}
887
888
889/**
890 * Gets the supported USB versions.
891 *
892 * @returns The mask of supported USB versions.
893 * @param pInterface Pointer to this structure.
894 */
895static DECLCALLBACK(uint32_t) ohciRhGetUSBVersions(PVUSBIROOTHUBPORT pInterface)
896{
897 return VUSB_STDVER_11;
898}
899
900
901/**
902 * A device is being attached to a port in the roothub.
903 *
904 * @param pInterface Pointer to this structure.
905 * @param pDev Pointer to the device being attached.
906 * @param uPort The port number assigned to the device.
907 */
908static DECLCALLBACK(int) ohciRhAttach(PVUSBIROOTHUBPORT pInterface, PVUSBIDEVICE pDev, unsigned uPort)
909{
910 POHCI pOhci = VUSBIROOTHUBPORT_2_OHCI(pInterface);
911 LogFlow(("ohciRhAttach: pDev=%p uPort=%u\n", pDev, uPort));
912 PDMCritSectEnter(pOhci->pDevInsR3->pCritSectRoR3, VERR_IGNORED);
913
914 /*
915 * Validate and adjust input.
916 */
917 Assert(uPort >= 1 && uPort <= RT_ELEMENTS(pOhci->RootHub.aPorts));
918 uPort--;
919 Assert(!pOhci->RootHub.aPorts[uPort].pDev);
920
921 /*
922 * Attach it.
923 */
924 pOhci->RootHub.aPorts[uPort].fReg = OHCI_PORT_R_CURRENT_CONNECT_STATUS | OHCI_PORT_R_CONNECT_STATUS_CHANGE;
925 pOhci->RootHub.aPorts[uPort].pDev = pDev;
926 rhport_power(&pOhci->RootHub, uPort, 1 /* power on */);
927
928 ohci_remote_wakeup(pOhci);
929 ohciSetInterrupt(pOhci, OHCI_INTR_ROOT_HUB_STATUS_CHANGE);
930
931 PDMCritSectLeave(pOhci->pDevInsR3->pCritSectRoR3);
932 return VINF_SUCCESS;
933}
934
935
936/**
937 * A device is being detached from a port in the roothub.
938 *
939 * @param pInterface Pointer to this structure.
940 * @param pDev Pointer to the device being detached.
941 * @param uPort The port number assigned to the device.
942 */
943static DECLCALLBACK(void) ohciRhDetach(PVUSBIROOTHUBPORT pInterface, PVUSBIDEVICE pDev, unsigned uPort)
944{
945 POHCI pOhci = VUSBIROOTHUBPORT_2_OHCI(pInterface);
946 LogFlow(("ohciRhDetach: pDev=%p uPort=%u\n", pDev, uPort));
947 PDMCritSectEnter(pOhci->pDevInsR3->pCritSectRoR3, VERR_IGNORED);
948
949 /*
950 * Validate and adjust input.
951 */
952 Assert(uPort >= 1 && uPort <= RT_ELEMENTS(pOhci->RootHub.aPorts));
953 uPort--;
954 Assert(pOhci->RootHub.aPorts[uPort].pDev == pDev);
955
956 /*
957 * Detach it.
958 */
959 pOhci->RootHub.aPorts[uPort].pDev = NULL;
960 if (pOhci->RootHub.aPorts[uPort].fReg & OHCI_PORT_PES)
961 pOhci->RootHub.aPorts[uPort].fReg = OHCI_PORT_R_CONNECT_STATUS_CHANGE | OHCI_PORT_PESC;
962 else
963 pOhci->RootHub.aPorts[uPort].fReg = OHCI_PORT_R_CONNECT_STATUS_CHANGE;
964
965 ohci_remote_wakeup(pOhci);
966 ohciSetInterrupt(pOhci, OHCI_INTR_ROOT_HUB_STATUS_CHANGE);
967
968 PDMCritSectLeave(pOhci->pDevInsR3->pCritSectRoR3);
969}
970
971
972#ifdef IN_RING3
973/**
974 * One of the roothub devices has completed its reset operation.
975 *
976 * Currently, we don't think anything is required to be done here
977 * so it's just a stub for forcing async resetting of the devices
978 * during a root hub reset.
979 *
980 * @param pDev The root hub device.
981 * @param rc The result of the operation.
982 * @param pvUser Pointer to the controller.
983 */
984static DECLCALLBACK(void) ohciRhResetDoneOneDev(PVUSBIDEVICE pDev, int rc, void *pvUser)
985{
986 LogRel(("OHCI: root hub reset completed with %Rrc\n", rc));
987 NOREF(pDev); NOREF(rc); NOREF(pvUser);
988}
989#endif
990
991
992/**
993 * Reset the root hub.
994 *
995 * @returns VBox status code.
996 * @param pInterface Pointer to this structure.
997 * @param fResetOnLinux This is used to indicate whether we're at VM reset time and
998 * can do real resets or if we're at any other time where that
999 * isn't such a good idea.
1000 * @remark Do NOT call VUSBIDevReset on the root hub in an async fashion!
1001 * @thread EMT
1002 */
1003static DECLCALLBACK(int) ohciRhReset(PVUSBIROOTHUBPORT pInterface, bool fResetOnLinux)
1004{
1005 POHCI pOhci = VUSBIROOTHUBPORT_2_OHCI(pInterface);
1006 PDMCritSectEnter(pOhci->pDevInsR3->pCritSectRoR3, VERR_IGNORED);
1007
1008 pOhci->RootHub.status = 0;
1009 pOhci->RootHub.desc_a = OHCI_RHA_NPS | OHCI_NDP;
1010 pOhci->RootHub.desc_b = 0x0; /* Impl. specific */
1011
1012 /*
1013 * We're pending to _reattach_ the device without resetting them.
1014 * Except, during VM reset where we use the opportunity to do a proper
1015 * reset before the guest comes along and expect things.
1016 *
1017 * However, it's very very likely that we're not doing the right thing
1018 * here if coming from the guest (USB Reset state). The docs talks about
1019 * root hub resetting, however what exact behaviour in terms of root hub
1020 * status and changed bits, and HC interrupts aren't stated clearly. IF we
1021 * get trouble and see the guest doing "USB Resets" we will have to look
1022 * into this. For the time being we stick with simple.
1023 */
1024 for (unsigned iPort = 0; iPort < RT_ELEMENTS(pOhci->RootHub.aPorts); iPort++)
1025 {
1026 if (pOhci->RootHub.aPorts[iPort].pDev)
1027 {
1028 pOhci->RootHub.aPorts[iPort].fReg = OHCI_PORT_R_CURRENT_CONNECT_STATUS | OHCI_PORT_R_CONNECT_STATUS_CHANGE;
1029 if (fResetOnLinux)
1030 {
1031 PVM pVM = PDMDevHlpGetVM(pOhci->CTX_SUFF(pDevIns));
1032 VUSBIDevReset(pOhci->RootHub.aPorts[iPort].pDev, fResetOnLinux, ohciRhResetDoneOneDev, pOhci, pVM);
1033 }
1034 }
1035 else
1036 pOhci->RootHub.aPorts[iPort].fReg = 0;
1037 }
1038
1039 PDMCritSectLeave(pOhci->pDevInsR3->pCritSectRoR3);
1040 return VINF_SUCCESS;
1041}
1042
1043
1044/**
1045 * Does a software or hardware reset of the controller.
1046 *
1047 * This is called in response to setting HcCommandStatus.HCR, hardware reset,
1048 * and device construction.
1049 *
1050 * @param pOhci The ohci instance data.
1051 * @param fNewMode The new mode of operation. This is UsbSuspend if it's a
1052 * software reset, and UsbReset if it's a hardware reset / cold boot.
1053 * @param fResetOnLinux Set if we can do a real reset of the devices attached to the root hub.
1054 * This is really a just a hack for the non-working linux device reset.
1055 * Linux has this feature called 'logical disconnect' if device reset fails
1056 * which prevents us from doing resets when the guest asks for it - the guest
1057 * will get confused when the device seems to be reconnected everytime it tries
1058 * to reset it. But if we're at hardware reset time, we can allow a device to
1059 * be 'reconnected' without upsetting the guest.
1060 *
1061 * @remark This hasn't got anything to do with software setting the mode to UsbReset.
1062 */
1063static void ohciDoReset(POHCI pOhci, uint32_t fNewMode, bool fResetOnLinux)
1064{
1065 Log(("ohci: %s reset%s\n", fNewMode == OHCI_USB_RESET ? "hardware" : "software",
1066 fResetOnLinux ? " (reset on linux)" : ""));
1067
1068 /*
1069 * Cancel all outstanding URBs.
1070 *
1071 * We can't, and won't, deal with URBs until we're moved out of the
1072 * suspend/reset state. Also, a real HC isn't going to send anything
1073 * any more when a reset has been signaled.
1074 */
1075 pOhci->RootHub.pIRhConn->pfnCancelAllUrbs(pOhci->RootHub.pIRhConn);
1076
1077 /*
1078 * Reset the hardware registers.
1079 */
1080 if (fNewMode == OHCI_USB_RESET)
1081 pOhci->ctl |= OHCI_CTL_RWC; /* We're the firmware, set RemoteWakeupConnected. */
1082 else
1083 pOhci->ctl &= OHCI_CTL_IR | OHCI_CTL_RWC; /* IR and RWC are preserved on software reset. */
1084 pOhci->ctl |= fNewMode;
1085 pOhci->status = 0;
1086 pOhci->intr_status = 0;
1087 pOhci->intr = OHCI_INTR_MASTER_INTERRUPT_ENABLED; /* (We follow the text and the not reset value column,) */
1088
1089 pOhci->hcca = 0;
1090 pOhci->per_cur = 0;
1091 pOhci->ctrl_head = pOhci->ctrl_cur = 0;
1092 pOhci->bulk_head = pOhci->bulk_cur = 0;
1093 pOhci->done = 0;
1094
1095 pOhci->fsmps = 0x2778; /* To-Be-Defined, use the value linux sets...*/
1096 pOhci->fit = 0;
1097 pOhci->fi = 11999; /* (12MHz ticks, one frame is 1ms) */
1098 pOhci->frt = 0;
1099 pOhci->HcFmNumber = 0;
1100 pOhci->pstart = 0;
1101
1102 pOhci->dqic = 0x7;
1103 pOhci->fno = 0;
1104
1105 /*
1106 * If this is a hardware reset, we will initialize the root hub too.
1107 * Software resets doesn't do this according to the specs.
1108 * (It's not possible to have device connected at the time of the
1109 * device construction, so nothing to worry about there.)
1110 */
1111 if (fNewMode == OHCI_USB_RESET)
1112 VUSBIDevReset(pOhci->RootHub.pIDev, fResetOnLinux, NULL, NULL, NULL);
1113}
1114#endif /* IN_RING3 */
1115
1116/**
1117 * Reads physical memory.
1118 */
1119DECLINLINE(void) ohciPhysRead(POHCI pOhci, uint32_t Addr, void *pvBuf, size_t cbBuf)
1120{
1121 if (cbBuf)
1122 PDMDevHlpPhysRead(pOhci->CTX_SUFF(pDevIns), Addr, pvBuf, cbBuf);
1123}
1124
1125/**
1126 * Writes physical memory.
1127 */
1128DECLINLINE(void) ohciPhysWrite(POHCI pOhci, uint32_t Addr, const void *pvBuf, size_t cbBuf)
1129{
1130 if (cbBuf)
1131 PDMDevHlpPhysWrite(pOhci->CTX_SUFF(pDevIns), Addr, pvBuf, cbBuf);
1132}
1133
1134/**
1135 * Read an array of dwords from physical memory and correct endianness.
1136 */
1137DECLINLINE(void) ohciGetDWords(POHCI pOhci, uint32_t Addr, uint32_t *pau32s, int c32s)
1138{
1139 ohciPhysRead(pOhci, Addr, pau32s, c32s * sizeof(uint32_t));
1140#if BYTE_ORDER != LITTLE_ENDIAN
1141 for(int i = 0; i < c32s; i++)
1142 pau32s[i] = RT_H2LE_U32(pau32s[i]);
1143#endif
1144}
1145
1146/**
1147 * Write an array of dwords from physical memory and correct endianness.
1148 */
1149DECLINLINE(void) ohciPutDWords(POHCI pOhci, uint32_t Addr, const uint32_t *pau32s, int cu32s)
1150{
1151#if BYTE_ORDER == LITTLE_ENDIAN
1152 ohciPhysWrite(pOhci, Addr, pau32s, cu32s << 2);
1153#else
1154 for (int i = 0; i < c32s; i++, pau32s++, Addr += sizeof(*pau32s))
1155 {
1156 uint32_t u32Tmp = RT_H2LE_U32(*pau32s);
1157 ohciPhysWrite(pOhci, Addr, (uint8_t *)&u32Tmp, sizeof(u32Tmp));
1158 }
1159#endif
1160}
1161
1162
1163#ifdef IN_RING3
1164
1165/**
1166 * Reads an OHCIED.
1167 */
1168DECLINLINE(void) ohciReadEd(POHCI pOhci, uint32_t EdAddr, POHCIED pEd)
1169{
1170 ohciGetDWords(pOhci, EdAddr, (uint32_t *)pEd, sizeof(*pEd) >> 2);
1171}
1172
1173/**
1174 * Reads an OHCITD.
1175 */
1176DECLINLINE(void) ohciReadTd(POHCI pOhci, uint32_t TdAddr, POHCITD pTd)
1177{
1178 ohciGetDWords(pOhci, TdAddr, (uint32_t *)pTd, sizeof(*pTd) >> 2);
1179#ifdef LOG_ENABLED
1180 if (LogIs3Enabled())
1181 {
1182 uint32_t hichg;
1183 hichg = pTd->hwinfo;
1184 Log3(("ohciReadTd(,%#010x,): R=%d DP=%d DI=%d T=%d EC=%d CC=%#x CBP=%#010x NextTD=%#010x BE=%#010x UNK=%#x\n",
1185 TdAddr,
1186 (pTd->hwinfo >> 18) & 1,
1187 (pTd->hwinfo >> 19) & 3,
1188 (pTd->hwinfo >> 21) & 7,
1189 (pTd->hwinfo >> 24) & 3,
1190 (pTd->hwinfo >> 26) & 3,
1191 (pTd->hwinfo >> 28) &15,
1192 pTd->cbp,
1193 pTd->NextTD,
1194 pTd->be,
1195 pTd->hwinfo & TD_HWINFO_UNKNOWN_MASK));
1196#if 0
1197 if (LogIs3Enabled())
1198 {
1199 /*
1200 * usbohci.sys (32-bit XP) allocates 0x80 bytes per TD:
1201 * 0x00-0x0f is the OHCI TD.
1202 * 0x10-0x1f for isochronous TDs
1203 * 0x20 is the physical address of this TD.
1204 * 0x24 is initialized with 0x64745948, probably a magic.
1205 * 0x28 is some kind of flags. the first bit begin the allocated / not allocated indicator.
1206 * 0x30 is a pointer to something. endpoint? interface? device?
1207 * 0x38 is initialized to 0xdeadface. but is changed into a pointer or something.
1208 * 0x40 looks like a pointer.
1209 * The rest is unknown and initialized with zeros.
1210 */
1211 uint8_t abXpTd[0x80];
1212 ohciPhysRead(pOhci, TdAddr, abXpTd, sizeof(abXpTd));
1213 Log3(("WinXpTd: alloc=%d PhysSelf=%RX32 s2=%RX32 magic=%RX32 s4=%RX32 s5=%RX32\n"
1214 "%.*Rhxd\n",
1215 abXpTd[28] & RT_BIT(0),
1216 *((uint32_t *)&abXpTd[0x20]), *((uint32_t *)&abXpTd[0x30]),
1217 *((uint32_t *)&abXpTd[0x24]), *((uint32_t *)&abXpTd[0x38]),
1218 *((uint32_t *)&abXpTd[0x40]),
1219 sizeof(abXpTd), &abXpTd[0]));
1220 }
1221#endif
1222 }
1223#endif
1224}
1225
1226/**
1227 * Reads an OHCIITD.
1228 */
1229DECLINLINE(void) ohciReadITd(POHCI pOhci, uint32_t ITdAddr, POHCIITD pITd)
1230{
1231 ohciGetDWords(pOhci, ITdAddr, (uint32_t *)pITd, sizeof(*pITd) / sizeof(uint32_t));
1232#ifdef LOG_ENABLED
1233 if (LogIs3Enabled())
1234 {
1235 Log3(("ohciReadITd(,%#010x,): SF=%#06x (%#RX32) DI=%#x FC=%d CC=%#x BP0=%#010x NextTD=%#010x BE=%#010x\n",
1236 ITdAddr,
1237 pITd->HwInfo & 0xffff, pOhci->HcFmNumber,
1238 (pITd->HwInfo >> 21) & 7,
1239 (pITd->HwInfo >> 24) & 7,
1240 (pITd->HwInfo >> 28) &15,
1241 pITd->BP0,
1242 pITd->NextTD,
1243 pITd->BE));
1244 Log3(("psw0=%x:%03x psw1=%x:%03x psw2=%x:%03x psw3=%x:%03x psw4=%x:%03x psw5=%x:%03x psw6=%x:%03x psw7=%x:%03x\n",
1245 pITd->aPSW[0] >> 12, pITd->aPSW[0] & 0xfff,
1246 pITd->aPSW[1] >> 12, pITd->aPSW[1] & 0xfff,
1247 pITd->aPSW[2] >> 12, pITd->aPSW[2] & 0xfff,
1248 pITd->aPSW[3] >> 12, pITd->aPSW[3] & 0xfff,
1249 pITd->aPSW[4] >> 12, pITd->aPSW[4] & 0xfff,
1250 pITd->aPSW[5] >> 12, pITd->aPSW[5] & 0xfff,
1251 pITd->aPSW[6] >> 12, pITd->aPSW[6] & 0xfff,
1252 pITd->aPSW[7] >> 12, pITd->aPSW[7] & 0xfff));
1253 }
1254#endif
1255}
1256
1257
1258/**
1259 * Writes an OHCIED.
1260 */
1261DECLINLINE(void) ohciWriteEd(POHCI pOhci, uint32_t EdAddr, PCOHCIED pEd)
1262{
1263#ifdef LOG_ENABLED
1264 if (LogIs3Enabled())
1265 {
1266 OHCIED EdOld;
1267 uint32_t hichg;
1268
1269 ohciGetDWords(pOhci, EdAddr, (uint32_t *)&EdOld, sizeof(EdOld) >> 2);
1270 hichg = EdOld.hwinfo ^ pEd->hwinfo;
1271 Log3(("ohciWriteEd(,%#010x,): %sFA=%#x %sEN=%#x %sD=%#x %sS=%d %sK=%d %sF=%d %sMPS=%#x %sTailP=%#010x %sHeadP=%#010x %sH=%d %sC=%d %sNextED=%#010x\n",
1272 EdAddr,
1273 (hichg >> 0) & 0x7f ? "*" : "", (pEd->hwinfo >> 0) & 0x7f,
1274 (hichg >> 7) & 0xf ? "*" : "", (pEd->hwinfo >> 7) & 0xf,
1275 (hichg >> 11) & 3 ? "*" : "", (pEd->hwinfo >> 11) & 3,
1276 (hichg >> 13) & 1 ? "*" : "", (pEd->hwinfo >> 13) & 1,
1277 (hichg >> 14) & 1 ? "*" : "", (pEd->hwinfo >> 14) & 1,
1278 (hichg >> 15) & 1 ? "*" : "", (pEd->hwinfo >> 15) & 1,
1279 (hichg >> 24) &0x3ff ? "*" : "", (pEd->hwinfo >> 16) &0x3ff,
1280 EdOld.TailP != pEd->TailP ? "*" : "", pEd->TailP,
1281 (EdOld.HeadP & ~3) != (pEd->HeadP & ~3) ? "*" : "", pEd->HeadP & ~3,
1282 (EdOld.HeadP ^ pEd->HeadP) & 1 ? "*" : "", pEd->HeadP & 1,
1283 (EdOld.HeadP ^ pEd->HeadP) & 2 ? "*" : "", (pEd->HeadP >> 1) & 1,
1284 EdOld.NextED != pEd->NextED ? "*" : "", pEd->NextED));
1285 }
1286#endif
1287
1288 ohciPutDWords(pOhci, EdAddr, (uint32_t *)pEd, sizeof(*pEd) >> 2);
1289}
1290
1291
1292/**
1293 * Writes an OHCITD.
1294 */
1295DECLINLINE(void) ohciWriteTd(POHCI pOhci, uint32_t TdAddr, PCOHCITD pTd, const char *pszLogMsg)
1296{
1297#ifdef LOG_ENABLED
1298 if (LogIs3Enabled())
1299 {
1300 OHCITD TdOld;
1301 ohciGetDWords(pOhci, TdAddr, (uint32_t *)&TdOld, sizeof(TdOld) >> 2);
1302 uint32_t hichg = TdOld.hwinfo ^ pTd->hwinfo;
1303 Log3(("ohciWriteTd(,%#010x,): %sR=%d %sDP=%d %sDI=%#x %sT=%d %sEC=%d %sCC=%#x %sCBP=%#010x %sNextTD=%#010x %sBE=%#010x (%s)\n",
1304 TdAddr,
1305 (hichg >> 18) & 1 ? "*" : "", (pTd->hwinfo >> 18) & 1,
1306 (hichg >> 19) & 3 ? "*" : "", (pTd->hwinfo >> 19) & 3,
1307 (hichg >> 21) & 7 ? "*" : "", (pTd->hwinfo >> 21) & 7,
1308 (hichg >> 24) & 3 ? "*" : "", (pTd->hwinfo >> 24) & 3,
1309 (hichg >> 26) & 3 ? "*" : "", (pTd->hwinfo >> 26) & 3,
1310 (hichg >> 28) &15 ? "*" : "", (pTd->hwinfo >> 28) &15,
1311 TdOld.cbp != pTd->cbp ? "*" : "", pTd->cbp,
1312 TdOld.NextTD != pTd->NextTD ? "*" : "", pTd->NextTD,
1313 TdOld.be != pTd->be ? "*" : "", pTd->be,
1314 pszLogMsg));
1315 }
1316#endif
1317 ohciPutDWords(pOhci, TdAddr, (uint32_t *)pTd, sizeof(*pTd) >> 2);
1318}
1319
1320/**
1321 * Writes an OHCIITD.
1322 */
1323DECLINLINE(void) ohciWriteITd(POHCI pOhci, uint32_t ITdAddr, PCOHCIITD pITd, const char *pszLogMsg)
1324{
1325#ifdef LOG_ENABLED
1326 if (LogIs3Enabled())
1327 {
1328 OHCIITD ITdOld;
1329 ohciGetDWords(pOhci, ITdAddr, (uint32_t *)&ITdOld, sizeof(ITdOld) / sizeof(uint32_t));
1330 uint32_t HIChg = ITdOld.HwInfo ^ pITd->HwInfo;
1331 Log3(("ohciWriteITd(,%#010x,): %sSF=%#x (now=%#RX32) %sDI=%#x %sFC=%d %sCC=%#x %sBP0=%#010x %sNextTD=%#010x %sBE=%#010x (%s)\n",
1332 ITdAddr,
1333 (HIChg & 0xffff) & 1 ? "*" : "", pITd->HwInfo & 0xffff, pOhci->HcFmNumber,
1334 (HIChg >> 21) & 7 ? "*" : "", (pITd->HwInfo >> 21) & 7,
1335 (HIChg >> 24) & 7 ? "*" : "", (pITd->HwInfo >> 24) & 7,
1336 (HIChg >> 28) &15 ? "*" : "", (pITd->HwInfo >> 28) &15,
1337 ITdOld.BP0 != pITd->BP0 ? "*" : "", pITd->BP0,
1338 ITdOld.NextTD != pITd->NextTD ? "*" : "", pITd->NextTD,
1339 ITdOld.BE != pITd->BE ? "*" : "", pITd->BE,
1340 pszLogMsg));
1341 Log3(("psw0=%s%x:%s%03x psw1=%s%x:%s%03x psw2=%s%x:%s%03x psw3=%s%x:%s%03x psw4=%s%x:%s%03x psw5=%s%x:%s%03x psw6=%s%x:%s%03x psw7=%s%x:%s%03x\n",
1342 (ITdOld.aPSW[0] >> 12) != (pITd->aPSW[0] >> 12) ? "*" : "", pITd->aPSW[0] >> 12, (ITdOld.aPSW[0] & 0xfff) != (pITd->aPSW[0] & 0xfff) ? "*" : "", pITd->aPSW[0] & 0xfff,
1343 (ITdOld.aPSW[1] >> 12) != (pITd->aPSW[1] >> 12) ? "*" : "", pITd->aPSW[1] >> 12, (ITdOld.aPSW[1] & 0xfff) != (pITd->aPSW[1] & 0xfff) ? "*" : "", pITd->aPSW[1] & 0xfff,
1344 (ITdOld.aPSW[2] >> 12) != (pITd->aPSW[2] >> 12) ? "*" : "", pITd->aPSW[2] >> 12, (ITdOld.aPSW[2] & 0xfff) != (pITd->aPSW[2] & 0xfff) ? "*" : "", pITd->aPSW[2] & 0xfff,
1345 (ITdOld.aPSW[3] >> 12) != (pITd->aPSW[3] >> 12) ? "*" : "", pITd->aPSW[3] >> 12, (ITdOld.aPSW[3] & 0xfff) != (pITd->aPSW[3] & 0xfff) ? "*" : "", pITd->aPSW[3] & 0xfff,
1346 (ITdOld.aPSW[4] >> 12) != (pITd->aPSW[4] >> 12) ? "*" : "", pITd->aPSW[4] >> 12, (ITdOld.aPSW[4] & 0xfff) != (pITd->aPSW[4] & 0xfff) ? "*" : "", pITd->aPSW[4] & 0xfff,
1347 (ITdOld.aPSW[5] >> 12) != (pITd->aPSW[5] >> 12) ? "*" : "", pITd->aPSW[5] >> 12, (ITdOld.aPSW[5] & 0xfff) != (pITd->aPSW[5] & 0xfff) ? "*" : "", pITd->aPSW[5] & 0xfff,
1348 (ITdOld.aPSW[6] >> 12) != (pITd->aPSW[6] >> 12) ? "*" : "", pITd->aPSW[6] >> 12, (ITdOld.aPSW[6] & 0xfff) != (pITd->aPSW[6] & 0xfff) ? "*" : "", pITd->aPSW[6] & 0xfff,
1349 (ITdOld.aPSW[7] >> 12) != (pITd->aPSW[7] >> 12) ? "*" : "", pITd->aPSW[7] >> 12, (ITdOld.aPSW[7] & 0xfff) != (pITd->aPSW[7] & 0xfff) ? "*" : "", pITd->aPSW[7] & 0xfff));
1350 }
1351#endif
1352 ohciPutDWords(pOhci, ITdAddr, (uint32_t *)pITd, sizeof(*pITd) / sizeof(uint32_t));
1353}
1354
1355
1356#ifdef LOG_ENABLED
1357
1358/**
1359 * Core TD queue dumper. LOG_ENABLED builds only.
1360 */
1361DECLINLINE(void) ohciDumpTdQueueCore(POHCI pOhci, uint32_t GCPhysHead, uint32_t GCPhysTail, bool fFull)
1362{
1363 uint32_t GCPhys = GCPhysHead;
1364 int cMax = 100;
1365 for (;;)
1366 {
1367 OHCITD Td;
1368 Log4(("%#010x%s%s", GCPhys,
1369 GCPhys && ohci_in_flight_find(pOhci, GCPhys) >= 0 ? "~" : "",
1370 GCPhys && ohci_in_done_queue_find(pOhci, GCPhys) >= 0 ? "^" : ""));
1371 if (GCPhys == 0 || GCPhys == GCPhysTail)
1372 break;
1373
1374 /* can't use ohciReadTd() because of Log4. */
1375 ohciGetDWords(pOhci, GCPhys, (uint32_t *)&Td, sizeof(Td) >> 2);
1376 if (fFull)
1377 Log4((" [R=%d DP=%d DI=%d T=%d EC=%d CC=%#x CBP=%#010x NextTD=%#010x BE=%#010x] -> ",
1378 (Td.hwinfo >> 18) & 1,
1379 (Td.hwinfo >> 19) & 3,
1380 (Td.hwinfo >> 21) & 7,
1381 (Td.hwinfo >> 24) & 3,
1382 (Td.hwinfo >> 26) & 3,
1383 (Td.hwinfo >> 28) &15,
1384 Td.cbp,
1385 Td.NextTD,
1386 Td.be));
1387 else
1388 Log4((" -> "));
1389 GCPhys = Td.NextTD & ED_PTR_MASK;
1390 Assert(GCPhys != GCPhysHead);
1391 Assert(cMax-- > 0); NOREF(cMax);
1392 }
1393}
1394
1395/**
1396 * Dumps a TD queue. LOG_ENABLED builds only.
1397 */
1398DECLINLINE(void) ohciDumpTdQueue(POHCI pOhci, uint32_t GCPhysHead, const char *pszMsg)
1399{
1400 if (pszMsg)
1401 Log4(("%s: ", pszMsg));
1402 ohciDumpTdQueueCore(pOhci, GCPhysHead, 0, true);
1403 Log4(("\n"));
1404}
1405
1406/**
1407 * Core ITD queue dumper. LOG_ENABLED builds only.
1408 */
1409DECLINLINE(void) ohciDumpITdQueueCore(POHCI pOhci, uint32_t GCPhysHead, uint32_t GCPhysTail, bool fFull)
1410{
1411 uint32_t GCPhys = GCPhysHead;
1412 int cMax = 100;
1413 for (;;)
1414 {
1415 OHCIITD ITd;
1416 Log4(("%#010x%s%s", GCPhys,
1417 GCPhys && ohci_in_flight_find(pOhci, GCPhys) >= 0 ? "~" : "",
1418 GCPhys && ohci_in_done_queue_find(pOhci, GCPhys) >= 0 ? "^" : ""));
1419 if (GCPhys == 0 || GCPhys == GCPhysTail)
1420 break;
1421
1422 /* can't use ohciReadTd() because of Log4. */
1423 ohciGetDWords(pOhci, GCPhys, (uint32_t *)&ITd, sizeof(ITd) / sizeof(uint32_t));
1424 /*if (fFull)
1425 Log4((" [R=%d DP=%d DI=%d T=%d EC=%d CC=%#x CBP=%#010x NextTD=%#010x BE=%#010x] -> ",
1426 (Td.hwinfo >> 18) & 1,
1427 (Td.hwinfo >> 19) & 3,
1428 (Td.hwinfo >> 21) & 7,
1429 (Td.hwinfo >> 24) & 3,
1430 (Td.hwinfo >> 26) & 3,
1431 (Td.hwinfo >> 28) &15,
1432 Td.cbp,
1433 Td.NextTD,
1434 Td.be));
1435 else*/
1436 Log4((" -> "));
1437 GCPhys = ITd.NextTD & ED_PTR_MASK;
1438 Assert(GCPhys != GCPhysHead);
1439 Assert(cMax-- > 0); NOREF(cMax);
1440 }
1441}
1442
1443/**
1444 * Dumps a ED list. LOG_ENABLED builds only.
1445 */
1446DECLINLINE(void) ohciDumpEdList(POHCI pOhci, uint32_t GCPhysHead, const char *pszMsg, bool fTDs)
1447{
1448 uint32_t GCPhys = GCPhysHead;
1449 if (pszMsg)
1450 Log4(("%s:", pszMsg));
1451 for (;;)
1452 {
1453 OHCIED Ed;
1454
1455 /* ED */
1456 Log4((" %#010x={", GCPhys));
1457 if (!GCPhys)
1458 {
1459 Log4(("END}\n"));
1460 return;
1461 }
1462
1463 /* TDs */
1464 ohciReadEd(pOhci, GCPhys, &Ed);
1465 if (Ed.hwinfo & ED_HWINFO_ISO)
1466 Log4(("[I]"));
1467 if ((Ed.HeadP & ED_HEAD_HALTED) || (Ed.hwinfo & ED_HWINFO_SKIP))
1468 {
1469 if ((Ed.HeadP & ED_HEAD_HALTED) && (Ed.hwinfo & ED_HWINFO_SKIP))
1470 Log4(("SH}"));
1471 else if (Ed.hwinfo & ED_HWINFO_SKIP)
1472 Log4(("S-}"));
1473 else
1474 Log4(("-H}"));
1475 }
1476 else
1477 {
1478 if (Ed.hwinfo & ED_HWINFO_ISO)
1479 ohciDumpITdQueueCore(pOhci, Ed.HeadP & ED_PTR_MASK, Ed.TailP & ED_PTR_MASK, false);
1480 else
1481 ohciDumpTdQueueCore(pOhci, Ed.HeadP & ED_PTR_MASK, Ed.TailP & ED_PTR_MASK, false);
1482 Log4(("}"));
1483 }
1484
1485 /* next */
1486 GCPhys = Ed.NextED & ED_PTR_MASK;
1487 Assert(GCPhys != GCPhysHead);
1488 }
1489 Log4(("\n"));
1490}
1491
1492#endif /* LOG_ENABLED */
1493
1494
1495DECLINLINE(int) ohci_in_flight_find_free(POHCI pOhci, const int iStart)
1496{
1497 unsigned i = iStart;
1498 while (i < RT_ELEMENTS(pOhci->aInFlight))
1499 {
1500 if (pOhci->aInFlight[i].GCPhysTD == 0)
1501 return i;
1502 i++;
1503 }
1504 i = iStart;
1505 while (i-- > 0)
1506 {
1507 if (pOhci->aInFlight[i].GCPhysTD == 0)
1508 return i;
1509 }
1510 return -1;
1511}
1512
1513
1514/**
1515 * Record an in-flight TD.
1516 *
1517 * @param pOhci OHCI instance data.
1518 * @param GCPhysTD Physical address of the TD.
1519 * @param pUrb The URB.
1520 */
1521static void ohci_in_flight_add(POHCI pOhci, uint32_t GCPhysTD, PVUSBURB pUrb)
1522{
1523 int i = ohci_in_flight_find_free(pOhci, (GCPhysTD >> 4) % RT_ELEMENTS(pOhci->aInFlight));
1524 if (i >= 0)
1525 {
1526#ifdef LOG_ENABLED
1527 pUrb->Hci.u32FrameNo = pOhci->HcFmNumber;
1528#endif
1529 pOhci->aInFlight[i].GCPhysTD = GCPhysTD;
1530 pOhci->aInFlight[i].pUrb = pUrb;
1531 pOhci->cInFlight++;
1532 return;
1533 }
1534 AssertMsgFailed(("Out of space cInFlight=%d!\n", pOhci->cInFlight));
1535}
1536
1537
1538/**
1539 * Record in-flight TDs for an URB.
1540 *
1541 * @param pOhci OHCI instance data.
1542 * @param pUrb The URB.
1543 */
1544static void ohci_in_flight_add_urb(POHCI pOhci, PVUSBURB pUrb)
1545{
1546 for (unsigned iTd = 0; iTd < pUrb->Hci.cTds; iTd++)
1547 ohci_in_flight_add(pOhci, pUrb->Hci.paTds[iTd].TdAddr, pUrb);
1548}
1549
1550
1551/**
1552 * Finds a in-flight TD.
1553 *
1554 * @returns Index of the record.
1555 * @returns -1 if not found.
1556 * @param pOhci OHCI instance data.
1557 * @param GCPhysTD Physical address of the TD.
1558 * @remark This has to be fast.
1559 */
1560static int ohci_in_flight_find(POHCI pOhci, uint32_t GCPhysTD)
1561{
1562 unsigned cLeft = pOhci->cInFlight;
1563 unsigned i = (GCPhysTD >> 4) % RT_ELEMENTS(pOhci->aInFlight);
1564 const int iLast = i;
1565 while (i < RT_ELEMENTS(pOhci->aInFlight))
1566 {
1567 if (pOhci->aInFlight[i].GCPhysTD == GCPhysTD)
1568 return i;
1569 if (pOhci->aInFlight[i].GCPhysTD)
1570 if (cLeft-- <= 1)
1571 return -1;
1572 i++;
1573 }
1574 i = iLast;
1575 while (i-- > 0)
1576 {
1577 if (pOhci->aInFlight[i].GCPhysTD == GCPhysTD)
1578 return i;
1579 if (pOhci->aInFlight[i].GCPhysTD)
1580 if (cLeft-- <= 1)
1581 return -1;
1582 }
1583 return -1;
1584}
1585
1586
1587/**
1588 * Checks if a TD is in-flight.
1589 *
1590 * @returns true if in flight, false if not.
1591 * @param pOhci OHCI instance data.
1592 * @param GCPhysTD Physical address of the TD.
1593 */
1594static bool ohciIsTdInFlight(POHCI pOhci, uint32_t GCPhysTD)
1595{
1596 return ohci_in_flight_find(pOhci, GCPhysTD) >= 0;
1597}
1598
1599/**
1600 * Returns a URB associated with an in-flight TD, if any.
1601 *
1602 * @returns pointer to URB if TD is in flight.
1603 * @returns NULL if not in flight.
1604 * @param pOhci OHCI instance data.
1605 * @param GCPhysTD Physical address of the TD.
1606 */
1607static PVUSBURB ohciTdInFlightUrb(POHCI pOhci, uint32_t GCPhysTD)
1608{
1609 int i;
1610
1611 i = ohci_in_flight_find(pOhci, GCPhysTD);
1612 if ( i >= 0 )
1613 return pOhci->aInFlight[i].pUrb;
1614 return NULL;
1615}
1616
1617/**
1618 * Removes a in-flight TD.
1619 *
1620 * @returns 0 if found. For logged builds this is the number of frames the TD has been in-flight.
1621 * @returns -1 if not found.
1622 * @param pOhci OHCI instance data.
1623 * @param GCPhysTD Physical address of the TD.
1624 */
1625static int ohci_in_flight_remove(POHCI pOhci, uint32_t GCPhysTD)
1626{
1627 int i = ohci_in_flight_find(pOhci, GCPhysTD);
1628 if (i >= 0)
1629 {
1630#ifdef LOG_ENABLED
1631 const int cFramesInFlight = pOhci->HcFmNumber - pOhci->aInFlight[i].pUrb->Hci.u32FrameNo;
1632#else
1633 const int cFramesInFlight = 0;
1634#endif
1635 Log2(("ohci_in_flight_remove: reaping TD=%#010x %d frames (%#010x-%#010x)\n",
1636 GCPhysTD, cFramesInFlight, pOhci->aInFlight[i].pUrb->Hci.u32FrameNo, pOhci->HcFmNumber));
1637 pOhci->aInFlight[i].GCPhysTD = 0;
1638 pOhci->aInFlight[i].pUrb = NULL;
1639 pOhci->cInFlight--;
1640 return cFramesInFlight;
1641 }
1642 AssertMsgFailed(("TD %#010x is not in flight\n", GCPhysTD));
1643 return -1;
1644}
1645
1646
1647/**
1648 * Removes all TDs associated with a URB from the in-flight tracking.
1649 *
1650 * @returns 0 if found. For logged builds this is the number of frames the TD has been in-flight.
1651 * @returns -1 if not found.
1652 * @param pOhci OHCI instance data.
1653 * @param pUrb The URB.
1654 */
1655static int ohci_in_flight_remove_urb(POHCI pOhci, PVUSBURB pUrb)
1656{
1657 int cFramesInFlight = ohci_in_flight_remove(pOhci, pUrb->Hci.paTds[0].TdAddr);
1658 if (pUrb->Hci.cTds > 1)
1659 {
1660 for (unsigned iTd = 1; iTd < pUrb->Hci.cTds; iTd++)
1661 if (ohci_in_flight_remove(pOhci, pUrb->Hci.paTds[iTd].TdAddr) < 0)
1662 cFramesInFlight = -1;
1663 }
1664 return cFramesInFlight;
1665}
1666
1667
1668#if defined(VBOX_STRICT) || defined(LOG_ENABLED)
1669
1670/**
1671 * Empties the in-done-queue.
1672 * @param pOhci OHCI instance data.
1673 */
1674static void ohci_in_done_queue_zap(POHCI pOhci)
1675{
1676 pOhci->cInDoneQueue = 0;
1677}
1678
1679/**
1680 * Finds a TD in the in-done-queue.
1681 * @returns >= 0 on success.
1682 * @returns -1 if not found.
1683 * @param pOhci OHCI instance data.
1684 * @param GCPhysTD Physical address of the TD.
1685 */
1686static int ohci_in_done_queue_find(POHCI pOhci, uint32_t GCPhysTD)
1687{
1688 unsigned i = pOhci->cInDoneQueue;
1689 while (i-- > 0)
1690 if (pOhci->aInDoneQueue[i].GCPhysTD == GCPhysTD)
1691 return i;
1692 return -1;
1693}
1694
1695/**
1696 * Checks that the specified TD is not in the done queue.
1697 * @param pOhci OHCI instance data.
1698 * @param GCPhysTD Physical address of the TD.
1699 */
1700static bool ohci_in_done_queue_check(POHCI pOhci, uint32_t GCPhysTD)
1701{
1702 int i = ohci_in_done_queue_find(pOhci, GCPhysTD);
1703 AssertMsg(i < 0, ("TD %#010x (i=%d)\n", GCPhysTD, i));
1704 return i < 0;
1705}
1706
1707
1708# ifdef VBOX_STRICT
1709/**
1710 * Adds a TD to the in-done-queue tracking, checking that it's not there already.
1711 * @param pOhci OHCI instance data.
1712 * @param GCPhysTD Physical address of the TD.
1713 */
1714static void ohci_in_done_queue_add(POHCI pOhci, uint32_t GCPhysTD)
1715{
1716 Assert(pOhci->cInDoneQueue + 1 <= RT_ELEMENTS(pOhci->aInDoneQueue));
1717 if (ohci_in_done_queue_check(pOhci, GCPhysTD))
1718 pOhci->aInDoneQueue[pOhci->cInDoneQueue++].GCPhysTD = GCPhysTD;
1719}
1720# endif /* VBOX_STRICT */
1721#endif /* defined(VBOX_STRICT) || defined(LOG_ENABLED) */
1722
1723
1724/**
1725 * OHCI Transport Buffer - represents a OHCI Transport Descriptor (TD).
1726 * A TD may be split over max 2 pages.
1727 */
1728typedef struct OHCIBUF
1729{
1730 /** Pages involved. */
1731 struct OHCIBUFVEC
1732 {
1733 /** The 32-bit physical address of this part. */
1734 uint32_t Addr;
1735 /** The length. */
1736 uint32_t cb;
1737 } aVecs[2];
1738 /** Number of valid entries in aVecs. */
1739 uint32_t cVecs;
1740 /** The total length. */
1741 uint32_t cbTotal;
1742} OHCIBUF, *POHCIBUF;
1743
1744
1745/**
1746 * Sets up a OHCI transport buffer.
1747 *
1748 * @param pBuf Ohci buffer.
1749 * @param cbp Current buffer pointer. 32-bit physical address.
1750 * @param be Last byte in buffer (BufferEnd). 32-bit physical address.
1751 */
1752static void ohciBufInit(POHCIBUF pBuf, uint32_t cbp, uint32_t be)
1753{
1754 if (!cbp || !be)
1755 {
1756 pBuf->cVecs = 0;
1757 pBuf->cbTotal = 0;
1758 Log2(("ohci: cbp=%#010x be=%#010x cbTotal=0 EMPTY\n", cbp, be));
1759 }
1760 else if ((cbp & ~0xfff) == (be & ~0xfff))
1761 {
1762 pBuf->aVecs[0].Addr = cbp;
1763 pBuf->aVecs[0].cb = (be - cbp) + 1;
1764 pBuf->cVecs = 1;
1765 pBuf->cbTotal = pBuf->aVecs[0].cb;
1766 Log2(("ohci: cbp=%#010x be=%#010x cbTotal=%u\n", cbp, be, pBuf->cbTotal));
1767 }
1768 else
1769 {
1770 pBuf->aVecs[0].Addr = cbp;
1771 pBuf->aVecs[0].cb = 0x1000 - (cbp & 0xfff);
1772 pBuf->aVecs[1].Addr = be & ~0xfff;
1773 pBuf->aVecs[1].cb = (be & 0xfff) + 1;
1774 pBuf->cVecs = 2;
1775 pBuf->cbTotal = pBuf->aVecs[0].cb + pBuf->aVecs[1].cb;
1776 Log2(("ohci: cbp=%#010x be=%#010x cbTotal=%u PAGE FLIP\n", cbp, be, pBuf->cbTotal));
1777 }
1778}
1779
1780/**
1781 * Updates a OHCI transport buffer.
1782 *
1783 * This is called upon completion to adjust the sector lengths if
1784 * the total length has changed. (received less then we had space for
1785 * or a partial transfer.)
1786 *
1787 * @param pBuf The buffer to update. cbTotal contains the new total on input.
1788 * While the aVecs[*].cb members is updated upon return.
1789 */
1790static void ohciBufUpdate(POHCIBUF pBuf)
1791{
1792 for (uint32_t i = 0, cbCur = 0; i < pBuf->cVecs; i++)
1793 {
1794 if (cbCur + pBuf->aVecs[i].cb > pBuf->cbTotal)
1795 {
1796 pBuf->aVecs[i].cb = pBuf->cbTotal - cbCur;
1797 pBuf->cVecs = i + 1;
1798 return;
1799 }
1800 cbCur += pBuf->aVecs[i].cb;
1801 }
1802}
1803
1804
1805/** A worker for ohciUnlinkTds(). */
1806static bool ohciUnlinkIsochronousTdInList(POHCI pOhci, uint32_t TdAddr, POHCIITD pITd, POHCIED pEd)
1807{
1808 const uint32_t LastTdAddr = pEd->TailP & ED_PTR_MASK;
1809 Log(("ohciUnlinkIsocTdInList: Unlinking non-head ITD! TdAddr=%#010RX32 HeadTdAddr=%#010RX32 LastEdAddr=%#010RX32\n",
1810 TdAddr, pEd->HeadP & ED_PTR_MASK, LastTdAddr));
1811 AssertMsgReturn(LastTdAddr != TdAddr, ("TdAddr=%#010RX32\n", TdAddr), false);
1812
1813 uint32_t cMax = 256;
1814 uint32_t CurTdAddr = pEd->HeadP & ED_PTR_MASK;
1815 while ( CurTdAddr != LastTdAddr
1816 && cMax-- > 0)
1817 {
1818 OHCIITD ITd;
1819 ohciReadITd(pOhci, CurTdAddr, &ITd);
1820 if ((ITd.NextTD & ED_PTR_MASK) == TdAddr)
1821 {
1822 ITd.NextTD = (pITd->NextTD & ED_PTR_MASK) | (ITd.NextTD & ~ED_PTR_MASK);
1823 ohciWriteITd(pOhci, CurTdAddr, &ITd, "ohciUnlinkIsocTdInList");
1824 pITd->NextTD &= ~ED_PTR_MASK;
1825 return true;
1826 }
1827
1828 /* next */
1829 CurTdAddr = ITd.NextTD & ED_PTR_MASK;
1830 }
1831
1832 Log(("ohciUnlinkIsocTdInList: TdAddr=%#010RX32 wasn't found in the list!!! (cMax=%d)\n", TdAddr, cMax));
1833 return false;
1834}
1835
1836
1837/** A worker for ohciUnlinkTds(). */
1838static bool ohciUnlinkGeneralTdInList(POHCI pOhci, uint32_t TdAddr, POHCITD pTd, POHCIED pEd)
1839{
1840 const uint32_t LastTdAddr = pEd->TailP & ED_PTR_MASK;
1841 Log(("ohciUnlinkGeneralTdInList: Unlinking non-head TD! TdAddr=%#010RX32 HeadTdAddr=%#010RX32 LastEdAddr=%#010RX32\n",
1842 TdAddr, pEd->HeadP & ED_PTR_MASK, LastTdAddr));
1843 AssertMsgReturn(LastTdAddr != TdAddr, ("TdAddr=%#010RX32\n", TdAddr), false);
1844
1845 uint32_t cMax = 256;
1846 uint32_t CurTdAddr = pEd->HeadP & ED_PTR_MASK;
1847 while ( CurTdAddr != LastTdAddr
1848 && cMax-- > 0)
1849 {
1850 OHCITD Td;
1851 ohciReadTd(pOhci, CurTdAddr, &Td);
1852 if ((Td.NextTD & ED_PTR_MASK) == TdAddr)
1853 {
1854 Td.NextTD = (pTd->NextTD & ED_PTR_MASK) | (Td.NextTD & ~ED_PTR_MASK);
1855 ohciWriteTd(pOhci, CurTdAddr, &Td, "ohciUnlinkGeneralTdInList");
1856 pTd->NextTD &= ~ED_PTR_MASK;
1857 return true;
1858 }
1859
1860 /* next */
1861 CurTdAddr = Td.NextTD & ED_PTR_MASK;
1862 }
1863
1864 Log(("ohciUnlinkGeneralTdInList: TdAddr=%#010RX32 wasn't found in the list!!! (cMax=%d)\n", TdAddr, cMax));
1865 return false;
1866}
1867
1868
1869/**
1870 * Unlinks the TDs that makes up the URB from the ED.
1871 *
1872 * @returns success indicator. true if successfully unlinked.
1873 * @returns false if the TD was not found in the list.
1874 */
1875static bool ohciUnlinkTds(POHCI pOhci, PVUSBURB pUrb, POHCIED pEd)
1876{
1877 /*
1878 * Don't unlink more than once.
1879 */
1880 if (pUrb->Hci.fUnlinked)
1881 return true;
1882 pUrb->Hci.fUnlinked = true;
1883
1884 if (pUrb->enmType == VUSBXFERTYPE_ISOC)
1885 {
1886 for (unsigned iTd = 0; iTd < pUrb->Hci.cTds; iTd++)
1887 {
1888 POHCIITD pITd = (POHCIITD)&pUrb->Hci.paTds[iTd].TdCopy[0];
1889 const uint32_t ITdAddr = pUrb->Hci.paTds[iTd].TdAddr;
1890
1891 /*
1892 * Unlink the TD from the ED list.
1893 * The normal case is that it's at the head of the list.
1894 */
1895 Assert((ITdAddr & ED_PTR_MASK) == ITdAddr);
1896 if ((pEd->HeadP & ED_PTR_MASK) == ITdAddr)
1897 {
1898 pEd->HeadP = (pITd->NextTD & ED_PTR_MASK) | (pEd->HeadP & ~ED_PTR_MASK);
1899 pITd->NextTD &= ~ED_PTR_MASK;
1900 }
1901 else
1902 {
1903 /*
1904 * It's probably somewhere in the list, not a unlikely situation with
1905 * the current isochronous code.
1906 */
1907 if (!ohciUnlinkIsochronousTdInList(pOhci, ITdAddr, pITd, pEd))
1908 return false;
1909 }
1910 }
1911 }
1912 else
1913 {
1914 for (unsigned iTd = 0; iTd < pUrb->Hci.cTds; iTd++)
1915 {
1916 POHCITD pTd = (POHCITD)&pUrb->Hci.paTds[iTd].TdCopy[0];
1917 const uint32_t TdAddr = pUrb->Hci.paTds[iTd].TdAddr;
1918
1919 /** @todo r=bird: Messing with the toggle flag in prepare is probably not correct
1920 * when we encounter a STALL error, 4.3.1.3.7.2: "If an endpoint returns a STALL
1921 * PID, the Host Controller retires the General TD with the ConditionCode set
1922 * to STALL and halts the endpoint. The CurrentBufferPointer, ErrorCount, and
1923 * dataToggle fields retain the values that they had at the start of the
1924 * transaction." */
1925
1926 /* update toggle and set data toggle carry */
1927 pTd->hwinfo &= ~TD_HWINFO_TOGGLE;
1928 if ( pTd->hwinfo & TD_HWINFO_TOGGLE_HI )
1929 {
1930 if ( !!(pTd->hwinfo & TD_HWINFO_TOGGLE_LO) ) /** @todo r=bird: is it just me or doesn't this make sense at all? */
1931 pTd->hwinfo |= TD_HWINFO_TOGGLE_LO;
1932 else
1933 pTd->hwinfo &= ~TD_HWINFO_TOGGLE_LO;
1934 }
1935 else
1936 {
1937 if ( !!(pEd->HeadP & ED_HEAD_CARRY) ) /** @todo r=bird: is it just me or doesn't this make sense at all? */
1938 pEd->HeadP |= ED_HEAD_CARRY;
1939 else
1940 pEd->HeadP &= ~ED_HEAD_CARRY;
1941 }
1942
1943 /*
1944 * Unlink the TD from the ED list.
1945 * The normal case is that it's at the head of the list.
1946 */
1947 Assert((TdAddr & ED_PTR_MASK) == TdAddr);
1948 if ((pEd->HeadP & ED_PTR_MASK) == TdAddr)
1949 {
1950 pEd->HeadP = (pTd->NextTD & ED_PTR_MASK) | (pEd->HeadP & ~ED_PTR_MASK);
1951 pTd->NextTD &= ~ED_PTR_MASK;
1952 }
1953 else
1954 {
1955 /*
1956 * The TD is probably somewhere in the list.
1957 *
1958 * This shouldn't ever happen unless there was a failure! Even on failure,
1959 * we can screw up the HCD state by picking out a TD from within the list
1960 * like this! If this turns out to be a problem, we have to find a better
1961 * solution. For now we'll hope the HCD handles it...
1962 */
1963 if (!ohciUnlinkGeneralTdInList(pOhci, TdAddr, pTd, pEd))
1964 return false;
1965 }
1966
1967 /*
1968 * Only unlink the first TD on error.
1969 * See comment in ohciRhXferCompleteGeneralURB().
1970 */
1971 if (pUrb->enmStatus != VUSBSTATUS_OK)
1972 break;
1973 }
1974 }
1975
1976 return true;
1977}
1978
1979
1980/**
1981 * Checks that the transport descriptors associated with the URB
1982 * hasn't been changed in any way indicating that they may have been canceled.
1983 *
1984 * This rountine also updates the TD copies contained within the URB.
1985 *
1986 * @returns true if the URB has been canceled, otherwise false.
1987 * @param pOhci The OHCI instance.
1988 * @param pUrb The URB in question.
1989 * @param pEd The ED pointer (optional).
1990 */
1991static bool ohciHasUrbBeenCanceled(POHCI pOhci, PVUSBURB pUrb, PCOHCIED pEd)
1992{
1993 if (!pUrb)
1994 return true;
1995
1996 /*
1997 * Make sure we've got an endpoint descriptor so we can
1998 * check for tail TDs.
1999 */
2000 OHCIED Ed;
2001 if (!pEd)
2002 {
2003 ohciReadEd(pOhci, pUrb->Hci.EdAddr, &Ed);
2004 pEd = &Ed;
2005 }
2006
2007 if (pUrb->enmType == VUSBXFERTYPE_ISOC)
2008 {
2009 for (unsigned iTd = 0; iTd < pUrb->Hci.cTds; iTd++)
2010 {
2011 union
2012 {
2013 OHCIITD ITd;
2014 uint32_t au32[8];
2015 } u;
2016 if ( (pUrb->Hci.paTds[iTd].TdAddr & ED_PTR_MASK)
2017 == (pEd->TailP & ED_PTR_MASK))
2018 {
2019 Log(("%s: ohciHasUrbBeenCanceled: iTd=%d cTds=%d TdAddr=%#010RX32 canceled (tail)! [iso]\n",
2020 pUrb->pszDesc, iTd, pUrb->Hci.cTds, pUrb->Hci.paTds[iTd].TdAddr));
2021 STAM_COUNTER_INC(&pOhci->StatCanceledIsocUrbs);
2022 return true;
2023 }
2024 ohciReadITd(pOhci, pUrb->Hci.paTds[iTd].TdAddr, &u.ITd);
2025 if ( u.au32[0] != pUrb->Hci.paTds[iTd].TdCopy[0] /* hwinfo */
2026 || u.au32[1] != pUrb->Hci.paTds[iTd].TdCopy[1] /* bp0 */
2027 || u.au32[3] != pUrb->Hci.paTds[iTd].TdCopy[3] /* be */
2028 || ( u.au32[2] != pUrb->Hci.paTds[iTd].TdCopy[2] /* NextTD */
2029 && iTd + 1 < pUrb->Hci.cTds /* ignore the last one */)
2030 || u.au32[4] != pUrb->Hci.paTds[iTd].TdCopy[4] /* psw0&1 */
2031 || u.au32[5] != pUrb->Hci.paTds[iTd].TdCopy[5] /* psw2&3 */
2032 || u.au32[6] != pUrb->Hci.paTds[iTd].TdCopy[6] /* psw4&5 */
2033 || u.au32[7] != pUrb->Hci.paTds[iTd].TdCopy[7] /* psw6&7 */
2034 )
2035 {
2036 Log(("%s: ohciHasUrbBeenCanceled: iTd=%d cTds=%d TdAddr=%#010RX32 canceled! [iso]\n",
2037 pUrb->pszDesc, iTd, pUrb->Hci.cTds, pUrb->Hci.paTds[iTd].TdAddr));
2038 Log2((" %.*Rhxs (cur)\n"
2039 "!= %.*Rhxs (copy)\n",
2040 sizeof(u.ITd), &u.ITd, sizeof(u.ITd), &pUrb->Hci.paTds[iTd].TdCopy[0]));
2041 STAM_COUNTER_INC(&pOhci->StatCanceledIsocUrbs);
2042 return true;
2043 }
2044 pUrb->Hci.paTds[iTd].TdCopy[2] = u.au32[2];
2045 }
2046 }
2047 else
2048 {
2049 for (unsigned iTd = 0; iTd < pUrb->Hci.cTds; iTd++)
2050 {
2051 union
2052 {
2053 OHCITD Td;
2054 uint32_t au32[4];
2055 } u;
2056 if ( (pUrb->Hci.paTds[iTd].TdAddr & ED_PTR_MASK)
2057 == (pEd->TailP & ED_PTR_MASK))
2058 {
2059 Log(("%s: ohciHasUrbBeenCanceled: iTd=%d cTds=%d TdAddr=%#010RX32 canceled (tail)!\n",
2060 pUrb->pszDesc, iTd, pUrb->Hci.cTds, pUrb->Hci.paTds[iTd].TdAddr));
2061 STAM_COUNTER_INC(&pOhci->StatCanceledGenUrbs);
2062 return true;
2063 }
2064 ohciReadTd(pOhci, pUrb->Hci.paTds[iTd].TdAddr, &u.Td);
2065 if ( u.au32[0] != pUrb->Hci.paTds[iTd].TdCopy[0] /* hwinfo */
2066 || u.au32[1] != pUrb->Hci.paTds[iTd].TdCopy[1] /* cbp */
2067 || u.au32[3] != pUrb->Hci.paTds[iTd].TdCopy[3] /* be */
2068 || ( u.au32[2] != pUrb->Hci.paTds[iTd].TdCopy[2] /* NextTD */
2069 && iTd + 1 < pUrb->Hci.cTds /* ignore the last one */)
2070 )
2071 {
2072 Log(("%s: ohciHasUrbBeenCanceled: iTd=%d cTds=%d TdAddr=%#010RX32 canceled!\n",
2073 pUrb->pszDesc, iTd, pUrb->Hci.cTds, pUrb->Hci.paTds[iTd].TdAddr));
2074 Log2((" %.*Rhxs (cur)\n"
2075 "!= %.*Rhxs (copy)\n",
2076 sizeof(u.Td), &u.Td, sizeof(u.Td), &pUrb->Hci.paTds[iTd].TdCopy[0]));
2077 STAM_COUNTER_INC(&pOhci->StatCanceledGenUrbs);
2078 return true;
2079 }
2080 pUrb->Hci.paTds[iTd].TdCopy[2] = u.au32[2];
2081 }
2082 }
2083 return false;
2084}
2085
2086
2087/**
2088 * Returns the OHCI_CC_* corresponding to the VUSB status code.
2089 *
2090 * @returns OHCI_CC_* value.
2091 * @param enmStatus The VUSB status code.
2092 */
2093static uint32_t ohciVUsbStatus2OhciStatus(VUSBSTATUS enmStatus)
2094{
2095 switch (enmStatus)
2096 {
2097 case VUSBSTATUS_OK: return OHCI_CC_NO_ERROR;
2098 case VUSBSTATUS_STALL: return OHCI_CC_STALL;
2099 case VUSBSTATUS_CRC: return OHCI_CC_CRC;
2100 case VUSBSTATUS_DATA_UNDERRUN: return OHCI_CC_DATA_UNDERRUN;
2101 case VUSBSTATUS_DATA_OVERRUN: return OHCI_CC_DATA_OVERRUN;
2102 case VUSBSTATUS_DNR: return OHCI_CC_DNR;
2103 case VUSBSTATUS_NOT_ACCESSED: return OHCI_CC_NOT_ACCESSED_1;
2104 default:
2105 Log(("pUrb->enmStatus=%#x!!!\n", enmStatus));
2106 return OHCI_CC_DNR;
2107 }
2108}
2109
2110/**
2111 * Worker for ohciRhXferCompletion that handles the completion of
2112 * a URB made up of isochronous TDs.
2113 *
2114 * In general, all URBs should have status OK.
2115 */
2116static void ohciRhXferCompleteIsochronousURB(POHCI pOhci, PVUSBURB pUrb, POHCIED pEd, int cFmAge)
2117{
2118 /*
2119 * Copy the data back (if IN operation) and update the TDs.
2120 */
2121 for (unsigned iTd = 0; iTd < pUrb->Hci.cTds; iTd++)
2122 {
2123 POHCIITD pITd = (POHCIITD)&pUrb->Hci.paTds[iTd].TdCopy[0];
2124 const uint32_t ITdAddr = pUrb->Hci.paTds[iTd].TdAddr;
2125 const unsigned cFrames = ((pITd->HwInfo & ITD_HWINFO_FC) >> ITD_HWINFO_FC_SHIFT) + 1;
2126 unsigned R = (pUrb->Hci.u32FrameNo & ITD_HWINFO_SF) - (pITd->HwInfo & ITD_HWINFO_SF);
2127 if (R >= 8)
2128 R = 0; /* submitted ahead of time. */
2129
2130 /*
2131 * Only one case of TD level condition code is document, so
2132 * just set NO_ERROR here to reduce number duplicate code.
2133 */
2134 pITd->HwInfo &= ~TD_HWINFO_CC;
2135 AssertCompile(OHCI_CC_NO_ERROR == 0);
2136
2137 if (pUrb->enmStatus == VUSBSTATUS_OK)
2138 {
2139 /*
2140 * Update the frames and copy back the data.
2141 * We assume that we don't get incorrect lengths here.
2142 */
2143 for (unsigned i = 0; i < cFrames; i++)
2144 {
2145 if ( i < R
2146 || pUrb->aIsocPkts[i - R].enmStatus == VUSBSTATUS_NOT_ACCESSED)
2147 {
2148 /* It should already be NotAccessed. */
2149 pITd->aPSW[i] |= 0xe000; /* (Don't touch the 12th bit.) */
2150 continue;
2151 }
2152
2153 /* Update the PSW (save the offset first in case of a IN). */
2154 uint32_t off = pITd->aPSW[i] & ITD_PSW_OFFSET;
2155 pITd->aPSW[i] = ohciVUsbStatus2OhciStatus(pUrb->aIsocPkts[i - R].enmStatus)
2156 >> (TD_HWINFO_CC_SHIFT - ITD_PSW_CC_SHIFT);
2157
2158 if ( pUrb->enmDir == VUSBDIRECTION_IN
2159 && ( pUrb->aIsocPkts[i - R].enmStatus == VUSBSTATUS_OK
2160 || pUrb->aIsocPkts[i - R].enmStatus == VUSBSTATUS_DATA_UNDERRUN
2161 || pUrb->aIsocPkts[i - R].enmStatus == VUSBSTATUS_DATA_OVERRUN))
2162 {
2163 /* Set the size. */
2164 const unsigned cb = pUrb->aIsocPkts[i - R].cb;
2165 pITd->aPSW[i] |= cb & ITD_PSW_SIZE;
2166 /* Copy data. */
2167 if (cb)
2168 {
2169 uint8_t *pb = &pUrb->abData[pUrb->aIsocPkts[i - R].off];
2170 if (off + cb > 0x1000)
2171 {
2172 if (off < 0x1000)
2173 {
2174 /* both */
2175 const unsigned cb0 = 0x1000 - off;
2176 ohciPhysWrite(pOhci, (pITd->BP0 & ITD_BP0_MASK) + off, pb, cb0);
2177 ohciPhysWrite(pOhci, pITd->BE & ITD_BP0_MASK, pb + cb0, cb - cb0);
2178 }
2179 else /* only in the 2nd page */
2180 ohciPhysWrite(pOhci, (pITd->BE & ITD_BP0_MASK) + (off & ITD_BP0_MASK), pb, cb);
2181 }
2182 else /* only in the 1st page */
2183 ohciPhysWrite(pOhci, (pITd->BP0 & ITD_BP0_MASK) + off, pb, cb);
2184 Log5(("packet %d: off=%#x cb=%#x pb=%p (%#x)\n"
2185 "%.*Rhxd\n",
2186 i + R, off, cb, pb, pb - &pUrb->abData[0], cb, pb));
2187 //off += cb;
2188 }
2189 }
2190 }
2191
2192 /*
2193 * If the last package ended with a NotAccessed status, set ITD CC
2194 * to DataOverrun to indicate scheduling overrun.
2195 */
2196 if (pUrb->aIsocPkts[pUrb->cIsocPkts - 1].enmStatus == VUSBSTATUS_NOT_ACCESSED)
2197 pITd->HwInfo |= OHCI_CC_DATA_OVERRUN;
2198 }
2199 else
2200 {
2201 Log(("DevOHCI: Taking untested code path at line %d...\n", __LINE__));
2202 /*
2203 * Most status codes only applies to the individual packets.
2204 *
2205 * If we get a URB level error code of this kind, we'll distribute
2206 * it to all the packages unless some other status is available for
2207 * a package. This is a bit fuzzy, and we will get rid of this code
2208 * before long!
2209 */
2210 //if (pUrb->enmStatus != VUSBSTATUS_DATA_OVERRUN)
2211 {
2212 const unsigned uCC = ohciVUsbStatus2OhciStatus(pUrb->enmStatus)
2213 >> (TD_HWINFO_CC_SHIFT - ITD_PSW_CC_SHIFT);
2214 for (unsigned i = 0; i < cFrames; i++)
2215 pITd->aPSW[i] = uCC;
2216 }
2217 //else
2218 // pITd->HwInfo |= ohciVUsbStatus2OhciStatus(pUrb->enmStatus);
2219 }
2220
2221 /*
2222 * Update the done queue interrupt timer.
2223 */
2224 uint32_t DoneInt = (pITd->HwInfo & ITD_HWINFO_DI) >> ITD_HWINFO_DI_SHIFT;
2225 if ((pITd->HwInfo & TD_HWINFO_CC) != OHCI_CC_NO_ERROR)
2226 DoneInt = 0; /* It's cleared on error. */
2227 if ( DoneInt != 0x7
2228 && DoneInt < pOhci->dqic)
2229 pOhci->dqic = DoneInt;
2230
2231 /*
2232 * Move on to the done list and write back the modified TD.
2233 */
2234#ifdef LOG_ENABLED
2235 if (!pOhci->done)
2236 pOhci->u32FmDoneQueueTail = pOhci->HcFmNumber;
2237# ifdef VBOX_STRICT
2238 ohci_in_done_queue_add(pOhci, ITdAddr);
2239# endif
2240#endif
2241 pITd->NextTD = pOhci->done;
2242 pOhci->done = ITdAddr;
2243
2244 Log(("%s: ohciRhXferCompleteIsochronousURB: ITdAddr=%#010x EdAddr=%#010x SF=%#x (%#x) CC=%#x FC=%d "
2245 "psw0=%x:%x psw1=%x:%x psw2=%x:%x psw3=%x:%x psw4=%x:%x psw5=%x:%x psw6=%x:%x psw7=%x:%x R=%d\n",
2246 pUrb->pszDesc, ITdAddr,
2247 pUrb->Hci.EdAddr,
2248 pITd->HwInfo & ITD_HWINFO_SF, pOhci->HcFmNumber,
2249 (pITd->HwInfo & ITD_HWINFO_CC) >> ITD_HWINFO_CC_SHIFT,
2250 (pITd->HwInfo & ITD_HWINFO_FC) >> ITD_HWINFO_FC_SHIFT,
2251 pITd->aPSW[0] >> ITD_PSW_CC_SHIFT, pITd->aPSW[0] & ITD_PSW_SIZE,
2252 pITd->aPSW[1] >> ITD_PSW_CC_SHIFT, pITd->aPSW[1] & ITD_PSW_SIZE,
2253 pITd->aPSW[2] >> ITD_PSW_CC_SHIFT, pITd->aPSW[2] & ITD_PSW_SIZE,
2254 pITd->aPSW[3] >> ITD_PSW_CC_SHIFT, pITd->aPSW[3] & ITD_PSW_SIZE,
2255 pITd->aPSW[4] >> ITD_PSW_CC_SHIFT, pITd->aPSW[4] & ITD_PSW_SIZE,
2256 pITd->aPSW[5] >> ITD_PSW_CC_SHIFT, pITd->aPSW[5] & ITD_PSW_SIZE,
2257 pITd->aPSW[6] >> ITD_PSW_CC_SHIFT, pITd->aPSW[6] & ITD_PSW_SIZE,
2258 pITd->aPSW[7] >> ITD_PSW_CC_SHIFT, pITd->aPSW[7] & ITD_PSW_SIZE,
2259 R));
2260 ohciWriteITd(pOhci, ITdAddr, pITd, "retired");
2261 }
2262}
2263
2264
2265/**
2266 * Worker for ohciRhXferCompletion that handles the completion of
2267 * a URB made up of general TDs.
2268 */
2269static void ohciRhXferCompleteGeneralURB(POHCI pOhci, PVUSBURB pUrb, POHCIED pEd, int cFmAge)
2270{
2271 /*
2272 * Copy the data back (if IN operation) and update the TDs.
2273 */
2274 unsigned cbLeft = pUrb->cbData;
2275 uint8_t *pb = &pUrb->abData[0];
2276 for (unsigned iTd = 0; iTd < pUrb->Hci.cTds; iTd++)
2277 {
2278 POHCITD pTd = (POHCITD)&pUrb->Hci.paTds[iTd].TdCopy[0];
2279 const uint32_t TdAddr = pUrb->Hci.paTds[iTd].TdAddr;
2280
2281 /*
2282 * Setup a ohci transfer buffer and calc the new cbp value.
2283 */
2284 OHCIBUF Buf;
2285 ohciBufInit(&Buf, pTd->cbp, pTd->be);
2286 uint32_t NewCbp;
2287 if (cbLeft >= Buf.cbTotal)
2288 NewCbp = 0;
2289 else
2290 {
2291 /* (len may have changed for short transfers) */
2292 Buf.cbTotal = cbLeft;
2293 ohciBufUpdate(&Buf);
2294 Assert(Buf.cVecs >= 1);
2295 NewCbp = Buf.aVecs[Buf.cVecs-1].Addr + Buf.aVecs[Buf.cVecs-1].cb;
2296 }
2297
2298 /*
2299 * Write back IN buffers.
2300 */
2301 if ( pUrb->enmDir == VUSBDIRECTION_IN
2302 && ( pUrb->enmStatus == VUSBSTATUS_OK
2303 || pUrb->enmStatus == VUSBSTATUS_DATA_OVERRUN
2304 || pUrb->enmStatus == VUSBSTATUS_DATA_UNDERRUN)
2305 && Buf.cbTotal > 0)
2306 {
2307 Assert(Buf.cVecs > 0);
2308 ohciPhysWrite(pOhci, Buf.aVecs[0].Addr, pb, Buf.aVecs[0].cb);
2309 if (Buf.cVecs > 1)
2310 ohciPhysWrite(pOhci, Buf.aVecs[1].Addr, pb + Buf.aVecs[0].cb, Buf.aVecs[1].cb);
2311 }
2312
2313 /* advance the data buffer. */
2314 cbLeft -= Buf.cbTotal;
2315 pb += Buf.cbTotal;
2316
2317 /*
2318 * Set writeback field.
2319 */
2320 /* zero out writeback fields for retirement */
2321 pTd->hwinfo &= ~TD_HWINFO_CC;
2322 /* always update the CurrentBufferPointer; essential for underrun/overrun errors */
2323 pTd->cbp = NewCbp;
2324
2325 if (pUrb->enmStatus == VUSBSTATUS_OK)
2326 {
2327 pTd->hwinfo &= ~TD_HWINFO_ERRORS;
2328
2329 /* update done queue interrupt timer */
2330 uint32_t DoneInt = (pTd->hwinfo & TD_HWINFO_DI) >> 21;
2331 if ( DoneInt != 0x7
2332 && DoneInt < pOhci->dqic)
2333 pOhci->dqic = DoneInt;
2334 Log(("%s: ohciRhXferCompleteGeneralURB: ED=%#010x TD=%#010x Age=%d cbTotal=%#x NewCbp=%#010RX32 dqic=%d\n",
2335 pUrb->pszDesc, pUrb->Hci.EdAddr, TdAddr, cFmAge, pUrb->enmStatus, Buf.cbTotal, NewCbp, pOhci->dqic));
2336 }
2337 else
2338 {
2339 Log(("%s: ohciRhXferCompleteGeneralURB: HALTED ED=%#010x TD=%#010x (age %d) pUrb->enmStatus=%d\n",
2340 pUrb->pszDesc, pUrb->Hci.EdAddr, TdAddr, cFmAge, pUrb->enmStatus));
2341 pEd->HeadP |= ED_HEAD_HALTED;
2342 pOhci->dqic = 0; /* "If the Transfer Descriptor is being retired with an error,
2343 * then the Done Queue Interrupt Counter is cleared as if the
2344 * InterruptDelay field were zero."
2345 */
2346 switch (pUrb->enmStatus)
2347 {
2348 case VUSBSTATUS_STALL:
2349 pTd->hwinfo |= OHCI_CC_STALL;
2350 break;
2351 case VUSBSTATUS_CRC:
2352 pTd->hwinfo |= OHCI_CC_CRC;
2353 break;
2354 case VUSBSTATUS_DATA_UNDERRUN:
2355 pTd->hwinfo |= OHCI_CC_DATA_UNDERRUN;
2356 break;
2357 case VUSBSTATUS_DATA_OVERRUN:
2358 pTd->hwinfo |= OHCI_CC_DATA_OVERRUN;
2359 break;
2360 default: /* what the hell */
2361 Log(("pUrb->enmStatus=%#x!!!\n", pUrb->enmStatus));
2362 case VUSBSTATUS_DNR:
2363 pTd->hwinfo |= OHCI_CC_DNR;
2364 break;
2365 }
2366 }
2367
2368 /*
2369 * Move on to the done list and write back the modified TD.
2370 */
2371#ifdef LOG_ENABLED
2372 if (!pOhci->done)
2373 pOhci->u32FmDoneQueueTail = pOhci->HcFmNumber;
2374# ifdef VBOX_STRICT
2375 ohci_in_done_queue_add(pOhci, TdAddr);
2376# endif
2377#endif
2378 pTd->NextTD = pOhci->done;
2379 pOhci->done = TdAddr;
2380
2381 ohciWriteTd(pOhci, TdAddr, pTd, "retired");
2382
2383 /*
2384 * If we've halted the endpoint, we stop here.
2385 * ohciUnlinkTds() will make sure we've only unliked the first TD.
2386 *
2387 * The reason for this is that while we can have more than one TD in a URB, real
2388 * OHCI hardware will only deal with one TD at the time and it's therefore incorrect
2389 * to retire TDs after the endpoint has been halted. Win2k will crash or enter infinite
2390 * kernel loop if we don't behave correctly. (See #1646.)
2391 */
2392 if (pEd->HeadP & ED_HEAD_HALTED)
2393 break;
2394 }
2395}
2396
2397
2398/**
2399 * Transfer completion callback routine.
2400 *
2401 * VUSB will call this when a transfer have been completed
2402 * in a one or another way.
2403 *
2404 * @param pInterface Pointer to OHCI::ROOTHUB::IRhPort.
2405 * @param pUrb Pointer to the URB in question.
2406 */
2407static DECLCALLBACK(void) ohciRhXferCompletion(PVUSBIROOTHUBPORT pInterface, PVUSBURB pUrb)
2408{
2409 POHCI pOhci = VUSBIROOTHUBPORT_2_OHCI(pInterface);
2410 LogFlow(("%s: ohciRhXferCompletion: EdAddr=%#010RX32 cTds=%d TdAddr0=%#010RX32\n",
2411 pUrb->pszDesc, pUrb->Hci.EdAddr, pUrb->Hci.cTds, pUrb->Hci.paTds[0].TdAddr));
2412 Assert(PDMCritSectIsOwner(pOhci->pDevInsR3->pCritSectRoR3));
2413
2414 pOhci->fIdle = false; /* Mark as active */
2415
2416 /* get the current end point descriptor. */
2417 OHCIED Ed;
2418 ohciReadEd(pOhci, pUrb->Hci.EdAddr, &Ed);
2419
2420 /*
2421 * Check that the URB hasn't been canceled and then try unlink the TDs.
2422 *
2423 * We drop the URB if the ED is marked halted/skip ASSUMING that this
2424 * means the HCD has canceled the URB.
2425 *
2426 * If we succeed here (i.e. not dropping the URB), the TdCopy members will
2427 * be updated but not yet written. We will delay the writing till we're done
2428 * with the data copying, buffer pointer advancing and error handling.
2429 */
2430 int cFmAge = ohci_in_flight_remove_urb(pOhci, pUrb);
2431 if (pUrb->enmStatus == VUSBSTATUS_UNDO)
2432 {
2433 /* Leave the TD alone - the HCD doesn't want us talking to the device. */
2434 Log(("%s: ohciRhXferCompletion: CANCELED {ED=%#010x cTds=%d TD0=%#010x age %d}\n",
2435 pUrb->pszDesc, pUrb->Hci.EdAddr, pUrb->Hci.cTds, pUrb->Hci.paTds[0].TdAddr, cFmAge));
2436 STAM_COUNTER_INC(&pOhci->StatDroppedUrbs);
2437 return;
2438 }
2439 bool fHasBeenCanceled = false;
2440 if ( (Ed.HeadP & ED_HEAD_HALTED)
2441 || (Ed.hwinfo & ED_HWINFO_SKIP)
2442 || cFmAge < 0
2443 || (fHasBeenCanceled = ohciHasUrbBeenCanceled(pOhci, pUrb, &Ed))
2444 || !ohciUnlinkTds(pOhci, pUrb, &Ed)
2445 )
2446 {
2447 Log(("%s: ohciRhXferCompletion: DROPPED {ED=%#010x cTds=%d TD0=%#010x age %d} because:%s%s%s%s%s!!!\n",
2448 pUrb->pszDesc, pUrb->Hci.EdAddr, pUrb->Hci.cTds, pUrb->Hci.paTds[0].TdAddr, cFmAge,
2449 (Ed.HeadP & ED_HEAD_HALTED) ? " ep halted" : "",
2450 (Ed.hwinfo & ED_HWINFO_SKIP) ? " ep skip" : "",
2451 (Ed.HeadP & ED_PTR_MASK) != pUrb->Hci.paTds[0].TdAddr ? " ep head-changed" : "",
2452 cFmAge < 0 ? " td not-in-flight" : "",
2453 fHasBeenCanceled ? " td canceled" : ""));
2454 NOREF(fHasBeenCanceled);
2455 STAM_COUNTER_INC(&pOhci->StatDroppedUrbs);
2456 return;
2457 }
2458
2459 /*
2460 * Complete the TD updating and write the back.
2461 * When appropriate also copy data back to the guest memory.
2462 */
2463 if (pUrb->enmType == VUSBXFERTYPE_ISOC)
2464 ohciRhXferCompleteIsochronousURB(pOhci, pUrb, &Ed, cFmAge);
2465 else
2466 ohciRhXferCompleteGeneralURB(pOhci, pUrb, &Ed, cFmAge);
2467
2468 /* finally write back the endpoint descriptor. */
2469 ohciWriteEd(pOhci, pUrb->Hci.EdAddr, &Ed);
2470}
2471
2472
2473/**
2474 * Handle transfer errors.
2475 *
2476 * VUSB calls this when a transfer attempt failed. This function will respond
2477 * indicating whether to retry or complete the URB with failure.
2478 *
2479 * @returns true if the URB should be retired.
2480 * @returns false if the URB should be retried.
2481 * @param pInterface Pointer to OHCI::ROOTHUB::IRhPort.
2482 * @param pUrb Pointer to the URB in question.
2483 */
2484static DECLCALLBACK(bool) ohciRhXferError(PVUSBIROOTHUBPORT pInterface, PVUSBURB pUrb)
2485{
2486 POHCI pOhci = VUSBIROOTHUBPORT_2_OHCI(pInterface);
2487 Assert(PDMCritSectIsOwner(pOhci->pDevInsR3->pCritSectRoR3));
2488
2489 /*
2490 * Isochronous URBs can't be retried.
2491 */
2492 if (pUrb->enmType == VUSBXFERTYPE_ISOC)
2493 return true;
2494
2495 /*
2496 * Don't retry on stall.
2497 */
2498 if (pUrb->enmStatus == VUSBSTATUS_STALL)
2499 {
2500 Log2(("%s: ohciRhXferError: STALL, giving up.\n", pUrb->pszDesc, pUrb->enmStatus));
2501 return true;
2502 }
2503
2504 /*
2505 * Check if the TDs still are valid.
2506 * This will make sure the TdCopy is up to date.
2507 */
2508 const uint32_t TdAddr = pUrb->Hci.paTds[0].TdAddr;
2509/** @todo IMPORTANT! we must check if the ED is still valid at this point!!! */
2510 if (ohciHasUrbBeenCanceled(pOhci, pUrb, NULL))
2511 {
2512 Log(("%s: ohciRhXferError: TdAddr0=%#x canceled!\n", pUrb->pszDesc, TdAddr));
2513 return true;
2514 }
2515
2516 /*
2517 * Get and update the error counter.
2518 */
2519 POHCITD pTd = (POHCITD)&pUrb->Hci.paTds[0].TdCopy[0];
2520 unsigned cErrs = (pTd->hwinfo & TD_HWINFO_ERRORS) >> TD_ERRORS_SHIFT;
2521 pTd->hwinfo &= ~TD_HWINFO_ERRORS;
2522 cErrs++;
2523 pTd->hwinfo |= (cErrs % TD_ERRORS_MAX) << TD_ERRORS_SHIFT;
2524 ohciWriteTd(pOhci, TdAddr, pTd, "ohciRhXferError");
2525
2526 if (cErrs >= TD_ERRORS_MAX - 1)
2527 {
2528 Log2(("%s: ohciRhXferError: too many errors, giving up!\n", pUrb->pszDesc));
2529 return true;
2530 }
2531 Log2(("%s: ohciRhXferError: cErrs=%d: retrying...\n", pUrb->pszDesc, cErrs));
2532 return false;
2533}
2534
2535
2536/**
2537 * Service a general transport descriptor.
2538 */
2539static bool ohciServiceTd(POHCI pOhci, VUSBXFERTYPE enmType, PCOHCIED pEd, uint32_t EdAddr, uint32_t TdAddr, uint32_t *pNextTdAddr, const char *pszListName)
2540{
2541 /*
2542 * Read the TD and setup the buffer data.
2543 */
2544 OHCITD Td;
2545 ohciReadTd(pOhci, TdAddr, &Td);
2546 OHCIBUF Buf;
2547 ohciBufInit(&Buf, Td.cbp, Td.be);
2548
2549 *pNextTdAddr = Td.NextTD & ED_PTR_MASK;
2550
2551 /*
2552 * Determine the direction.
2553 */
2554 VUSBDIRECTION enmDir;
2555 switch (pEd->hwinfo & ED_HWINFO_DIR)
2556 {
2557 case ED_HWINFO_OUT: enmDir = VUSBDIRECTION_OUT; break;
2558 case ED_HWINFO_IN: enmDir = VUSBDIRECTION_IN; break;
2559 default:
2560 switch (Td.hwinfo & TD_HWINFO_DIR)
2561 {
2562 case TD_HWINFO_OUT: enmDir = VUSBDIRECTION_OUT; break;
2563 case TD_HWINFO_IN: enmDir = VUSBDIRECTION_IN; break;
2564 case 0: enmDir = VUSBDIRECTION_SETUP; break;
2565 default:
2566 Log(("ohciServiceTd: Invalid direction!!!! Td.hwinfo=%#x Ed.hwdinfo=%#x\n", Td.hwinfo, pEd->hwinfo));
2567 /* TODO: Do the correct thing here */
2568 return false;
2569 }
2570 break;
2571 }
2572
2573 pOhci->fIdle = false; /* Mark as active */
2574
2575 /*
2576 * Allocate and initialize a new URB.
2577 */
2578 PVUSBURB pUrb = VUSBIRhNewUrb(pOhci->RootHub.pIRhConn, pEd->hwinfo & ED_HWINFO_FUNCTION, Buf.cbTotal, 1);
2579 if (!pUrb)
2580 return false; /* retry later... */
2581 Assert(pUrb->Hci.cTds == 1);
2582
2583 pUrb->enmType = enmType;
2584 pUrb->EndPt = (pEd->hwinfo & ED_HWINFO_ENDPOINT) >> ED_HWINFO_ENDPOINT_SHIFT;
2585 pUrb->enmDir = enmDir;
2586 pUrb->fShortNotOk = !(Td.hwinfo & TD_HWINFO_ROUNDING);
2587 pUrb->enmStatus = VUSBSTATUS_OK;
2588 pUrb->Hci.EdAddr = EdAddr;
2589 pUrb->Hci.fUnlinked = false;
2590 pUrb->Hci.paTds[0].TdAddr = TdAddr;
2591 pUrb->Hci.u32FrameNo = pOhci->HcFmNumber;
2592 AssertCompile(sizeof(pUrb->Hci.paTds[0].TdCopy) >= sizeof(Td));
2593 memcpy(pUrb->Hci.paTds[0].TdCopy, &Td, sizeof(Td));
2594#ifdef LOG_ENABLED
2595 static unsigned s_iSerial = 0;
2596 s_iSerial = (s_iSerial + 1) % 10000;
2597 RTStrAPrintf(&pUrb->pszDesc, "URB %p %10s/s%c%04d", pUrb, pszListName,
2598 enmDir == VUSBDIRECTION_IN ? '<' : enmDir == VUSBDIRECTION_OUT ? '>' : '-', s_iSerial);
2599#endif
2600
2601 /* copy data if out bound transfer. */
2602 pUrb->cbData = Buf.cbTotal;
2603 if ( Buf.cbTotal
2604 && Buf.cVecs > 0
2605 && enmDir != VUSBDIRECTION_IN)
2606 {
2607 ohciPhysRead(pOhci, Buf.aVecs[0].Addr, pUrb->abData, Buf.aVecs[0].cb);
2608 if (Buf.cVecs > 1)
2609 ohciPhysRead(pOhci, Buf.aVecs[1].Addr, &pUrb->abData[Buf.aVecs[0].cb], Buf.aVecs[1].cb);
2610 }
2611
2612 /*
2613 * Submit the URB.
2614 */
2615 ohci_in_flight_add(pOhci, TdAddr, pUrb);
2616 Log(("%s: ohciServiceTd: submitting TdAddr=%#010x EdAddr=%#010x cbData=%#x\n",
2617 pUrb->pszDesc, TdAddr, EdAddr, pUrb->cbData));
2618
2619 int rc = VUSBIRhSubmitUrb(pOhci->RootHub.pIRhConn, pUrb, &pOhci->RootHub.Led);
2620 if (RT_SUCCESS(rc))
2621 return true;
2622
2623 /* Failure cleanup. Can happen if we're still resetting the device or out of resources. */
2624 Log(("ohciServiceTd: failed submitting TdAddr=%#010x EdAddr=%#010x pUrb=%p!!\n",
2625 TdAddr, EdAddr, pUrb));
2626 ohci_in_flight_remove(pOhci, TdAddr);
2627 return false;
2628}
2629
2630
2631/**
2632 * Service a the head TD of an endpoint.
2633 */
2634static bool ohciServiceHeadTd(POHCI pOhci, VUSBXFERTYPE enmType, PCOHCIED pEd, uint32_t EdAddr, const char *pszListName)
2635{
2636 /*
2637 * Read the TD, after first checking if it's already in-flight.
2638 */
2639 uint32_t TdAddr = pEd->HeadP & ED_PTR_MASK;
2640 if (ohciIsTdInFlight(pOhci, TdAddr))
2641 return false;
2642#if defined(VBOX_STRICT) || defined(LOG_ENABLED)
2643 ohci_in_done_queue_check(pOhci, TdAddr);
2644#endif
2645 return ohciServiceTd(pOhci, enmType, pEd, EdAddr, TdAddr, &TdAddr, pszListName);
2646}
2647
2648
2649/**
2650 * Service one or more general transport descriptors (bulk or interrupt).
2651 */
2652static bool ohciServiceTdMultiple(POHCI pOhci, VUSBXFERTYPE enmType, PCOHCIED pEd, uint32_t EdAddr,
2653 uint32_t TdAddr, uint32_t *pNextTdAddr, const char *pszListName)
2654{
2655 /*
2656 * Read the TDs involved in this URB.
2657 */
2658 struct OHCITDENTRY
2659 {
2660 /** The TD. */
2661 OHCITD Td;
2662 /** The associated OHCI buffer tracker. */
2663 OHCIBUF Buf;
2664 /** The TD address. */
2665 uint32_t TdAddr;
2666 /** Pointer to the next element in the chain (stack). */
2667 struct OHCITDENTRY *pNext;
2668 } Head;
2669
2670 /* read the head */
2671 ohciReadTd(pOhci, TdAddr, &Head.Td);
2672 ohciBufInit(&Head.Buf, Head.Td.cbp, Head.Td.be);
2673 Head.TdAddr = TdAddr;
2674 Head.pNext = NULL;
2675
2676 /* combine with more TDs. */
2677 struct OHCITDENTRY *pTail = &Head;
2678 unsigned cbTotal = pTail->Buf.cbTotal;
2679 unsigned cTds = 1;
2680 while ( (pTail->Buf.cbTotal == 0x1000 || pTail->Buf.cbTotal == 0x2000)
2681 && !(pTail->Td.hwinfo & TD_HWINFO_ROUNDING) /* This isn't right for *BSD, but let's not . */
2682 && (pTail->Td.NextTD & ED_PTR_MASK) != (pEd->TailP & ED_PTR_MASK)
2683 && cTds < 128)
2684 {
2685 struct OHCITDENTRY *pCur = (struct OHCITDENTRY *)alloca(sizeof(*pCur));
2686
2687 pCur->pNext = NULL;
2688 pCur->TdAddr = pTail->Td.NextTD & ED_PTR_MASK;
2689 ohciReadTd(pOhci, pCur->TdAddr, &pCur->Td);
2690 ohciBufInit(&pCur->Buf, pCur->Td.cbp, pCur->Td.be);
2691
2692 /* don't combine if the direction doesn't match up. */
2693 if ( (pCur->Td.hwinfo & (TD_HWINFO_DIR))
2694 != (pCur->Td.hwinfo & (TD_HWINFO_DIR)))
2695 break;
2696
2697 pTail->pNext = pCur;
2698 pTail = pCur;
2699 cbTotal += pCur->Buf.cbTotal;
2700 cTds++;
2701 }
2702
2703 /* calc next TD address */
2704 *pNextTdAddr = pTail->Td.NextTD & ED_PTR_MASK;
2705
2706 /*
2707 * Determine the direction.
2708 */
2709 VUSBDIRECTION enmDir;
2710 switch (pEd->hwinfo & ED_HWINFO_DIR)
2711 {
2712 case ED_HWINFO_OUT: enmDir = VUSBDIRECTION_OUT; break;
2713 case ED_HWINFO_IN: enmDir = VUSBDIRECTION_IN; break;
2714 default:
2715 Log(("ohciServiceTdMultiple: WARNING! Ed.hwdinfo=%#x bulk or interrupt EP shouldn't rely on the TD for direction...\n", pEd->hwinfo));
2716 switch (Head.Td.hwinfo & TD_HWINFO_DIR)
2717 {
2718 case TD_HWINFO_OUT: enmDir = VUSBDIRECTION_OUT; break;
2719 case TD_HWINFO_IN: enmDir = VUSBDIRECTION_IN; break;
2720 default:
2721 Log(("ohciServiceTdMultiple: Invalid direction!!!! Head.Td.hwinfo=%#x Ed.hwdinfo=%#x\n", Head.Td.hwinfo, pEd->hwinfo));
2722 /* TODO: Do the correct thing here */
2723 return false;
2724 }
2725 break;
2726 }
2727
2728 pOhci->fIdle = false; /* Mark as active */
2729
2730 /*
2731 * Allocate and initialize a new URB.
2732 */
2733 PVUSBURB pUrb = VUSBIRhNewUrb(pOhci->RootHub.pIRhConn, pEd->hwinfo & ED_HWINFO_FUNCTION, cbTotal, cTds);
2734 if (!pUrb)
2735 /* retry later... */
2736 return false;
2737 Assert(pUrb->Hci.cTds == cTds);
2738 Assert(pUrb->cbData == cbTotal);
2739
2740 pUrb->enmType = enmType;
2741 pUrb->EndPt = (pEd->hwinfo & ED_HWINFO_ENDPOINT) >> ED_HWINFO_ENDPOINT_SHIFT;
2742 pUrb->enmDir = enmDir;
2743 pUrb->fShortNotOk = !(pTail->Td.hwinfo & TD_HWINFO_ROUNDING);
2744 pUrb->enmStatus = VUSBSTATUS_OK;
2745 pUrb->Hci.EdAddr = EdAddr;
2746 pUrb->Hci.fUnlinked = false;
2747 pUrb->Hci.u32FrameNo = pOhci->HcFmNumber;
2748#ifdef LOG_ENABLED
2749 static unsigned s_iSerial = 0;
2750 s_iSerial = (s_iSerial + 1) % 10000;
2751 RTStrAPrintf(&pUrb->pszDesc, "URB %p %10s/m%c%04d", pUrb, pszListName,
2752 enmDir == VUSBDIRECTION_IN ? '<' : enmDir == VUSBDIRECTION_OUT ? '>' : '-', s_iSerial);
2753#endif
2754
2755 /* Copy data and TD information. */
2756 unsigned iTd = 0;
2757 uint8_t *pb = &pUrb->abData[0];
2758 for (struct OHCITDENTRY *pCur = &Head; pCur; pCur = pCur->pNext, iTd++)
2759 {
2760 /* data */
2761 if ( cbTotal
2762 && enmDir != VUSBDIRECTION_IN
2763 && pCur->Buf.cVecs > 0)
2764 {
2765 ohciPhysRead(pOhci, pCur->Buf.aVecs[0].Addr, pb, pCur->Buf.aVecs[0].cb);
2766 if (pCur->Buf.cVecs > 1)
2767 ohciPhysRead(pOhci, pCur->Buf.aVecs[1].Addr, pb + pCur->Buf.aVecs[0].cb, pCur->Buf.aVecs[1].cb);
2768 }
2769 pb += pCur->Buf.cbTotal;
2770
2771 /* TD info */
2772 pUrb->Hci.paTds[iTd].TdAddr = pCur->TdAddr;
2773 AssertCompile(sizeof(pUrb->Hci.paTds[iTd].TdCopy) >= sizeof(pCur->Td));
2774 memcpy(pUrb->Hci.paTds[iTd].TdCopy, &pCur->Td, sizeof(pCur->Td));
2775 }
2776
2777 /*
2778 * Submit the URB.
2779 */
2780 ohci_in_flight_add_urb(pOhci, pUrb);
2781 Log(("%s: ohciServiceTdMultiple: submitting cbData=%#x EdAddr=%#010x cTds=%d TdAddr0=%#010x\n",
2782 pUrb->pszDesc, pUrb->cbData, EdAddr, cTds, TdAddr));
2783 int rc = VUSBIRhSubmitUrb(pOhci->RootHub.pIRhConn, pUrb, &pOhci->RootHub.Led);
2784 if (RT_SUCCESS(rc))
2785 return true;
2786
2787 /* Failure cleanup. Can happen if we're still resetting the device or out of resources. */
2788 Log(("ohciServiceTdMultiple: failed submitting pUrb=%p cbData=%#x EdAddr=%#010x cTds=%d TdAddr0=%#010x - rc=%Rrc\n",
2789 pUrb, cbTotal, EdAddr, cTds, TdAddr, rc));
2790 for (struct OHCITDENTRY *pCur = &Head; pCur; pCur = pCur->pNext, iTd++)
2791 ohci_in_flight_remove(pOhci, pCur->TdAddr);
2792 return false;
2793}
2794
2795
2796/**
2797 * Service the head TD of an endpoint.
2798 */
2799static bool ohciServiceHeadTdMultiple(POHCI pOhci, VUSBXFERTYPE enmType, PCOHCIED pEd, uint32_t EdAddr, const char *pszListName)
2800{
2801 /*
2802 * First, check that it's not already in-flight.
2803 */
2804 uint32_t TdAddr = pEd->HeadP & ED_PTR_MASK;
2805 if (ohciIsTdInFlight(pOhci, TdAddr))
2806 return false;
2807#if defined(VBOX_STRICT) || defined(LOG_ENABLED)
2808 ohci_in_done_queue_check(pOhci, TdAddr);
2809#endif
2810 return ohciServiceTdMultiple(pOhci, enmType, pEd, EdAddr, TdAddr, &TdAddr, pszListName);
2811}
2812
2813
2814/**
2815 * A worker for ohciServiceIsochronousEndpoint which unlinks a ITD
2816 * that belongs to the past.
2817 */
2818static bool ohciServiceIsochronousTdUnlink(POHCI pOhci, POHCIITD pITd, uint32_t ITdAddr, uint32_t ITdAddrPrev,
2819 PVUSBURB pUrb, POHCIED pEd, uint32_t EdAddr)
2820{
2821 LogFlow(("%s%sohciServiceIsochronousTdUnlink: Unlinking ITD: ITdAddr=%#010x EdAddr=%#010x ITdAddrPrev=%#010x\n",
2822 pUrb ? pUrb->pszDesc : "", pUrb ? ": " : "", ITdAddr, EdAddr, ITdAddrPrev));
2823
2824 /*
2825 * Do the unlinking.
2826 */
2827 const uint32_t ITdAddrNext = pITd->NextTD & ED_PTR_MASK;
2828 if (ITdAddrPrev)
2829 {
2830 /* Get validate the previous TD */
2831 int iInFlightPrev = ohci_in_flight_find(pOhci, ITdAddr);
2832 AssertMsgReturn(iInFlightPrev >= 0, ("ITdAddr=%#RX32\n", ITdAddr), false);
2833 PVUSBURB pUrbPrev = pOhci->aInFlight[iInFlightPrev].pUrb;
2834 if (ohciHasUrbBeenCanceled(pOhci, pUrbPrev, pEd)) /* ensures the copy is correct. */
2835 return false;
2836
2837 /* Update the copy and write it back. */
2838 POHCIITD pITdPrev = ((POHCIITD)pUrbPrev->Hci.paTds[0].TdCopy);
2839 pITdPrev->NextTD = (pITdPrev->NextTD & ~ED_PTR_MASK) | ITdAddrNext;
2840 ohciWriteITd(pOhci, ITdAddrPrev, pITdPrev, "ohciServiceIsochronousEndpoint");
2841 }
2842 else
2843 {
2844 /* It's the head node. update the copy from the caller and write it back. */
2845 pEd->HeadP = (pEd->HeadP & ~ED_PTR_MASK) | ITdAddrNext;
2846 ohciWriteEd(pOhci, EdAddr, pEd);
2847 }
2848
2849 /*
2850 * If it's in flight, just mark the URB as unlinked (there is only one ITD per URB atm).
2851 * Otherwise, retire it to the done queue with an error and cause a done line interrupt (?).
2852 */
2853 if (pUrb)
2854 {
2855 pUrb->Hci.fUnlinked = true;
2856 if (ohciHasUrbBeenCanceled(pOhci, pUrb, pEd)) /* ensures the copy is correct (paranoia). */
2857 return false;
2858
2859 POHCIITD pITdCopy = ((POHCIITD)pUrb->Hci.paTds[0].TdCopy);
2860 pITd->NextTD = pITdCopy->NextTD &= ~ED_PTR_MASK;
2861 }
2862 else
2863 {
2864 pITd->HwInfo &= ~ITD_HWINFO_CC;
2865 pITd->HwInfo |= OHCI_CC_DATA_OVERRUN;
2866
2867 pITd->NextTD = pOhci->done;
2868 pOhci->done = ITdAddr;
2869
2870 pOhci->dqic = 0;
2871 }
2872
2873 ohciWriteITd(pOhci, ITdAddr, pITd, "ohciServiceIsochronousTdUnlink");
2874 return true;
2875}
2876
2877
2878/**
2879 * A worker for ohciServiceIsochronousEndpoint which submits the specified TD.
2880 *
2881 * @returns true on success.
2882 * @returns false on failure to submit.
2883 * @param R The start packet (frame) relative to the start of frame in HwInfo.
2884 */
2885static bool ohciServiceIsochronousTd(POHCI pOhci, POHCIITD pITd, uint32_t ITdAddr, const unsigned R, PCOHCIED pEd, uint32_t EdAddr)
2886{
2887 /*
2888 * Determine the endpoint direction.
2889 */
2890 VUSBDIRECTION enmDir;
2891 switch (pEd->hwinfo & ED_HWINFO_DIR)
2892 {
2893 case ED_HWINFO_OUT: enmDir = VUSBDIRECTION_OUT; break;
2894 case ED_HWINFO_IN: enmDir = VUSBDIRECTION_IN; break;
2895 default:
2896 Log(("ohciServiceIsochronousTd: Invalid direction!!!! Ed.hwdinfo=%#x\n", pEd->hwinfo));
2897 /* Should probably raise an unrecoverable HC error here */
2898 return false;
2899 }
2900
2901 /*
2902 * Extract the packet sizes and calc the total URB size.
2903 */
2904 struct
2905 {
2906 uint16_t cb;
2907 uint16_t off;
2908 } aPkts[ITD_NUM_PSW];
2909
2910 /* first entry (R) */
2911 uint32_t cbTotal = 0;
2912 if (((uint32_t)pITd->aPSW[R] >> ITD_PSW_CC_SHIFT) < (OHCI_CC_NOT_ACCESSED_0 >> TD_HWINFO_CC_SHIFT))
2913 Log(("ITdAddr=%RX32 PSW%d.CC=%#x < 'Not Accessed'!\n", ITdAddr, R, pITd->aPSW[R] >> ITD_PSW_CC_SHIFT)); /* => Unrecoverable Error*/
2914 uint16_t offPrev = aPkts[0].off = (pITd->aPSW[R] & ITD_PSW_OFFSET);
2915
2916 /* R+1..cFrames */
2917 const unsigned cFrames = ((pITd->HwInfo & ITD_HWINFO_FC) >> ITD_HWINFO_FC_SHIFT) + 1;
2918 for (unsigned iR = R + 1; iR < cFrames; iR++)
2919 {
2920 const uint16_t PSW = pITd->aPSW[iR];
2921 const uint16_t off = aPkts[iR - R].off = (PSW & ITD_PSW_OFFSET);
2922 cbTotal += aPkts[iR - R - 1].cb = off - offPrev;
2923 if (off < offPrev)
2924 Log(("ITdAddr=%RX32 PSW%d.offset=%#x < offPrev=%#x!\n", ITdAddr, iR, off, offPrev)); /* => Unrecoverable Error*/
2925 if (((uint32_t)PSW >> ITD_PSW_CC_SHIFT) < (OHCI_CC_NOT_ACCESSED_0 >> TD_HWINFO_CC_SHIFT))
2926 Log(("ITdAddr=%RX32 PSW%d.CC=%#x < 'Not Accessed'!\n", ITdAddr, iR, PSW >> ITD_PSW_CC_SHIFT)); /* => Unrecoverable Error*/
2927 offPrev = off;
2928 }
2929
2930 /* calc offEnd and figure out the size of the last packet. */
2931 const uint32_t offEnd = (pITd->BE & 0xfff)
2932 + (((pITd->BE & ITD_BP0_MASK) != (pITd->BP0 & ITD_BP0_MASK)) << 12)
2933 + 1 /* BE is inclusive */;
2934 if (offEnd < offPrev)
2935 Log(("ITdAddr=%RX32 offEnd=%#x < offPrev=%#x!\n", ITdAddr, offEnd, offPrev)); /* => Unrecoverable Error*/
2936 cbTotal += aPkts[cFrames - 1 - R].cb = offEnd - offPrev;
2937 Assert(cbTotal <= 0x2000);
2938
2939 pOhci->fIdle = false; /* Mark as active */
2940
2941 /*
2942 * Allocate and initialize a new URB.
2943 */
2944 PVUSBURB pUrb = VUSBIRhNewUrb(pOhci->RootHub.pIRhConn, pEd->hwinfo & ED_HWINFO_FUNCTION, cbTotal, 1);
2945 if (!pUrb)
2946 /* retry later... */
2947 return false;
2948
2949 pUrb->enmType = VUSBXFERTYPE_ISOC;
2950 pUrb->EndPt = (pEd->hwinfo & ED_HWINFO_ENDPOINT) >> ED_HWINFO_ENDPOINT_SHIFT;
2951 pUrb->enmDir = enmDir;
2952 pUrb->fShortNotOk = false;
2953 pUrb->enmStatus = VUSBSTATUS_OK;
2954 pUrb->Hci.EdAddr = EdAddr;
2955 pUrb->Hci.fUnlinked = false;
2956 pUrb->Hci.u32FrameNo = pOhci->HcFmNumber;
2957 pUrb->Hci.paTds[0].TdAddr = ITdAddr;
2958 AssertCompile(sizeof(pUrb->Hci.paTds[0].TdCopy) >= sizeof(*pITd));
2959 memcpy(pUrb->Hci.paTds[0].TdCopy, pITd, sizeof(*pITd));
2960#if 0 /* color the data */
2961 memset(pUrb->abData, 0xfe, cbTotal);
2962#endif
2963#ifdef LOG_ENABLED
2964 static unsigned s_iSerial = 0;
2965 s_iSerial = (s_iSerial + 1) % 10000;
2966 RTStrAPrintf(&pUrb->pszDesc, "URB %p isoc%c%04d", pUrb, enmDir == VUSBDIRECTION_IN ? '<' : '>', s_iSerial);
2967#endif
2968
2969 /* copy the data */
2970 if ( cbTotal
2971 && enmDir != VUSBDIRECTION_IN)
2972 {
2973 const uint32_t off0 = pITd->aPSW[R] & ITD_PSW_OFFSET;
2974 if (off0 < 0x1000)
2975 {
2976 if (offEnd > 0x1000)
2977 {
2978 /* both pages. */
2979 const unsigned cb0 = 0x1000 - off0;
2980 ohciPhysRead(pOhci, (pITd->BP0 & ITD_BP0_MASK) + off0, &pUrb->abData[0], cb0);
2981 ohciPhysRead(pOhci, pITd->BE & ITD_BP0_MASK, &pUrb->abData[cb0], offEnd & 0xfff);
2982 }
2983 else /* a portion of the 1st page. */
2984 ohciPhysRead(pOhci, (pITd->BP0 & ITD_BP0_MASK) + off0, pUrb->abData, offEnd - off0);
2985 }
2986 else /* a portion of the 2nd page. */
2987 ohciPhysRead(pOhci, (pITd->BE & UINT32_C(0xfffff000)) + (off0 & 0xfff), pUrb->abData, cbTotal);
2988 }
2989
2990 /* setup the packets */
2991 pUrb->cIsocPkts = cFrames - R;
2992 unsigned off = 0;
2993 for (unsigned i = 0; i < pUrb->cIsocPkts; i++)
2994 {
2995 pUrb->aIsocPkts[i].enmStatus = VUSBSTATUS_NOT_ACCESSED;
2996 pUrb->aIsocPkts[i].off = off;
2997 off += pUrb->aIsocPkts[i].cb = aPkts[i].cb;
2998 }
2999 Assert(off == cbTotal);
3000
3001 /*
3002 * Submit the URB.
3003 */
3004 ohci_in_flight_add_urb(pOhci, pUrb);
3005 Log(("%s: ohciServiceIsochronousTd: submitting cbData=%#x cIsocPkts=%d EdAddr=%#010x TdAddr=%#010x SF=%#x (%#x)\n",
3006 pUrb->pszDesc, pUrb->cbData, pUrb->cIsocPkts, EdAddr, ITdAddr, pITd->HwInfo & ITD_HWINFO_SF, pOhci->HcFmNumber));
3007 int rc = VUSBIRhSubmitUrb(pOhci->RootHub.pIRhConn, pUrb, &pOhci->RootHub.Led);
3008 if (RT_SUCCESS(rc))
3009 return true;
3010
3011 /* Failure cleanup. Can happen if we're still resetting the device or out of resources. */
3012 Log(("ohciServiceIsochronousTd: failed submitting pUrb=%p cbData=%#x EdAddr=%#010x cTds=%d ITdAddr0=%#010x - rc=%Rrc\n",
3013 pUrb, cbTotal, EdAddr, 1, ITdAddr, rc));
3014 ohci_in_flight_remove(pOhci, ITdAddr);
3015 return false;
3016}
3017
3018
3019/**
3020 * Service an isochronous endpoint.
3021 */
3022static void ohciServiceIsochronousEndpoint(POHCI pOhci, POHCIED pEd, uint32_t EdAddr)
3023{
3024 /*
3025 * We currently process this as if the guest follows the interrupt end point chaining
3026 * hierarchy described in the documenation. This means that for an isochronous endpoint
3027 * with a 1 ms interval we expect to find in-flight TDs at the head of the list. We will
3028 * skip over all in-flight TDs which timeframe has been exceed. Those which aren't in
3029 * flight but which are too late will be retired (possibly out of order, but, we don't
3030 * care right now).
3031 *
3032 * When we reach a TD which still has a buffer which is due for take off, we will
3033 * stop iterating TDs. If it's in-flight, there isn't anything to be done. Otherwise
3034 * we will push it onto the runway for immediate take off. In this process we
3035 * might have to complete buffers which didn't make it on time, something which
3036 * complicates the kind of status info we need to keep around for the TD.
3037 *
3038 * Note: We're currently not making any attempt at reassembling ITDs into URBs.
3039 * However, this will become necessary because of EMT scheduling and guest
3040 * like linux using one TD for each frame (simple but inefficient for us).
3041 */
3042 OHCIITD ITd;
3043 uint32_t ITdAddr = pEd->HeadP & ED_PTR_MASK;
3044 uint32_t ITdAddrPrev = 0;
3045 uint32_t u32NextFrame = UINT32_MAX;
3046 const uint16_t u16CurFrame = pOhci->HcFmNumber;
3047 for (;;)
3048 {
3049 /* check for end-of-chain. */
3050 if ( ITdAddr == (pEd->TailP & ED_PTR_MASK)
3051 || !ITdAddr)
3052 break;
3053
3054 /*
3055 * If isochronous endpoints are around, don't slow down the timer. Getting the timing right
3056 * is difficult enough as it is.
3057 */
3058 pOhci->fIdle = false;
3059
3060 /*
3061 * Read the current ITD and check what we're supposed to do about it.
3062 */
3063 ohciReadITd(pOhci, ITdAddr, &ITd);
3064 const uint32_t ITdAddrNext = ITd.NextTD & ED_PTR_MASK;
3065 const int16_t R = u16CurFrame - (uint16_t)(ITd.HwInfo & ITD_HWINFO_SF); /* 4.3.2.3 */
3066 const int16_t cFrames = ((ITd.HwInfo & ITD_HWINFO_FC) >> ITD_HWINFO_FC_SHIFT) + 1;
3067
3068 if (R < cFrames)
3069 {
3070 /*
3071 * It's inside the current or a future launch window.
3072 *
3073 * We will try maximize the TD in flight here to deal with EMT scheduling
3074 * issues and similar stuff which will screw up the time. So, we will only
3075 * stop submitting TD when we reach a gap (in time) or end of the list.
3076 */
3077 if ( R < 0 /* (a future frame) */
3078 && (uint16_t)u32NextFrame != (uint16_t)(ITd.HwInfo & ITD_HWINFO_SF))
3079 break;
3080 if (ohci_in_flight_find(pOhci, ITdAddr) < 0)
3081 if (!ohciServiceIsochronousTd(pOhci, &ITd, ITdAddr, R < 0 ? 0 : R, pEd, EdAddr))
3082 break;
3083
3084 ITdAddrPrev = ITdAddr;
3085 }
3086 else
3087 {
3088#if 1
3089 /*
3090 * Ok, the launch window for this TD has passed.
3091 * If it's not in flight it should be retired with a DataOverrun status (TD).
3092 *
3093 * Don't remove in-flight TDs before they complete.
3094 * Windows will, upon the completion of another ITD it seems, check for if
3095 * any other TDs has been unlinked. If we unlink them before they really
3096 * complete all the packet status codes will be NotAccessed and Windows
3097 * will fail the URB with status USBD_STATUS_ISOCH_REQUEST_FAILED.
3098 *
3099 * I don't know if unlinking TDs out of order could cause similar problems,
3100 * time will show.
3101 */
3102 int iInFlight = ohci_in_flight_find(pOhci, ITdAddr);
3103 if (iInFlight >= 0)
3104 ITdAddrPrev = ITdAddr;
3105 else if (!ohciServiceIsochronousTdUnlink(pOhci, &ITd, ITdAddr, ITdAddrPrev,
3106 NULL, pEd, EdAddr))
3107 {
3108 Log(("ohciServiceIsochronousEndpoint: Failed unlinking old ITD.\n"));
3109 break;
3110 }
3111#else /* BAD IDEA: */
3112 /*
3113 * Ok, the launch window for this TD has passed.
3114 * If it's not in flight it should be retired with a DataOverrun status (TD).
3115 *
3116 * If it's in flight we will try unlink it from the list prematurely to
3117 * help the guest to move on and shorten the list we have to walk. We currently
3118 * are successful with the first URB but then it goes too slowly...
3119 */
3120 int iInFlight = ohci_in_flight_find(pOhci, ITdAddr);
3121 if (!ohciServiceIsochronousTdUnlink(pOhci, &ITd, ITdAddr, ITdAddrPrev,
3122 iInFlight < 0 ? NULL : pOhci->aInFlight[iInFlight].pUrb,
3123 pEd, EdAddr))
3124 {
3125 Log(("ohciServiceIsochronousEndpoint: Failed unlinking old ITD.\n"));
3126 break;
3127 }
3128#endif
3129 }
3130
3131 /* advance to the next ITD */
3132 ITdAddr = ITdAddrNext;
3133 u32NextFrame = (ITd.HwInfo & ITD_HWINFO_SF) + cFrames;
3134 }
3135}
3136
3137
3138/**
3139 * Checks if a endpoints has TDs queued and is ready to have them processed.
3140 *
3141 * @returns true if it's ok to process TDs.
3142 * @param pEd The endpoint data.
3143 */
3144DECLINLINE(bool) ohciIsEdReady(PCOHCIED pEd)
3145{
3146 return (pEd->HeadP & ED_PTR_MASK) != (pEd->TailP & ED_PTR_MASK)
3147 && !(pEd->HeadP & ED_HEAD_HALTED)
3148 && !(pEd->hwinfo & ED_HWINFO_SKIP);
3149}
3150
3151
3152/**
3153 * Checks if an endpoint has TDs queued (not necessarily ready to have them processed).
3154 *
3155 * @returns true if endpoint may have TDs queued.
3156 * @param pEd The endpoint data.
3157 */
3158DECLINLINE(bool) ohciIsEdPresent(PCOHCIED pEd)
3159{
3160 return (pEd->HeadP & ED_PTR_MASK) != (pEd->TailP & ED_PTR_MASK)
3161 && !(pEd->HeadP & ED_HEAD_HALTED);
3162}
3163
3164
3165/**
3166 * Services the bulk list.
3167 *
3168 * On the bulk list we must reassemble URBs from multiple TDs using heuristics
3169 * derived from USB tracing done in the guests and guest source code (when available).
3170 */
3171static void ohciServiceBulkList(POHCI pOhci)
3172{
3173#ifdef LOG_ENABLED
3174 if (g_fLogBulkEPs)
3175 ohciDumpEdList(pOhci, pOhci->bulk_head, "Bulk before", true);
3176 if (pOhci->bulk_cur)
3177 Log(("ohciServiceBulkList: bulk_cur=%#010x before listprocessing!!! HCD have positioned us!!!\n", pOhci->bulk_cur));
3178#endif
3179
3180 /*
3181 * ", HC will start processing the Bulk list and will set BF [BulkListFilled] to 0"
3182 * - We've simplified and are always starting at the head of the list and working
3183 * our way thru to the end each time.
3184 */
3185 pOhci->status &= ~OHCI_STATUS_BLF;
3186 pOhci->fBulkNeedsCleaning = false;
3187 pOhci->bulk_cur = 0;
3188
3189 uint32_t EdAddr = pOhci->bulk_head;
3190 while (EdAddr)
3191 {
3192 OHCIED Ed;
3193 ohciReadEd(pOhci, EdAddr, &Ed);
3194 Assert(!(Ed.hwinfo & ED_HWINFO_ISO)); /* the guest is screwing us */
3195 if (ohciIsEdReady(&Ed))
3196 {
3197 pOhci->status |= OHCI_STATUS_BLF;
3198 pOhci->fBulkNeedsCleaning = true;
3199
3200#if 1
3201 /*
3202
3203 * After we figured out that all the TDs submitted for dealing with MSD
3204 * read/write data really makes up on single URB, and that we must
3205 * reassemble these TDs into an URB before submitting it, there is no
3206 * longer any need for servicing anything other than the head *URB*
3207 * on a bulk endpoint.
3208 */
3209 ohciServiceHeadTdMultiple(pOhci, VUSBXFERTYPE_BULK, &Ed, EdAddr, "Bulk");
3210#else
3211 /*
3212 * This alternative code was used before we started reassembling URBs from
3213 * multiple TDs. We keep it handy for debugging.
3214 */
3215 uint32_t TdAddr = Ed.HeadP & ED_PTR_MASK;
3216 if (!ohciIsTdInFlight(pOhci, TdAddr))
3217 {
3218 do
3219 {
3220 if (!ohciServiceTdMultiple(pOhci, VUSBXFERTYPE_BULK, &Ed, EdAddr, TdAddr, &TdAddr, "Bulk"))
3221 {
3222 LogFlow(("ohciServiceBulkList: ohciServiceTdMultiple -> false\n"));
3223 break;
3224 }
3225 if ( (TdAddr & ED_PTR_MASK) == (Ed.TailP & ED_PTR_MASK)
3226 || !TdAddr /* paranoia */)
3227 {
3228 LogFlow(("ohciServiceBulkList: TdAddr=%#010RX32 Ed.TailP=%#010RX32\n", TdAddr, Ed.TailP));
3229 break;
3230 }
3231
3232 ohciReadEd(pOhci, EdAddr, &Ed); /* It might have been updated on URB completion. */
3233 } while (ohciIsEdReady(&Ed));
3234 }
3235#endif
3236 }
3237 else
3238 {
3239 if (Ed.hwinfo & ED_HWINFO_SKIP)
3240 {
3241 LogFlow(("ohciServiceBulkList: Ed=%#010RX32 Ed.TailP=%#010RX32 SKIP\n", EdAddr, Ed.TailP));
3242 /* If the ED is in 'skip' state, no transactions on it are allowed and we must
3243 * cancel outstanding URBs, if any.
3244 */
3245 uint32_t TdAddr = Ed.HeadP & ED_PTR_MASK;
3246 PVUSBURB pUrb = ohciTdInFlightUrb(pOhci, TdAddr);
3247 if (pUrb)
3248 pOhci->RootHub.pIRhConn->pfnCancelUrbsEp(pOhci->RootHub.pIRhConn, pUrb);
3249 }
3250 }
3251
3252 /* next end point */
3253 EdAddr = Ed.NextED & ED_PTR_MASK;
3254
3255 }
3256
3257#ifdef LOG_ENABLED
3258 if (g_fLogBulkEPs)
3259 ohciDumpEdList(pOhci, pOhci->bulk_head, "Bulk after ", true);
3260#endif
3261}
3262
3263/**
3264 * Abort outstanding transfers on the bulk list.
3265 *
3266 * If the guest disabled bulk list processing, we must abort any outstanding transfers
3267 * (that is, cancel in-flight URBs associated with the list). This is required because
3268 * there may be outstanding read URBs that will never get a response from the device
3269 * and would block further communication.
3270 */
3271static void ohciUndoBulkList(POHCI pOhci)
3272{
3273#ifdef LOG_ENABLED
3274 if (g_fLogBulkEPs)
3275 ohciDumpEdList(pOhci, pOhci->bulk_head, "Bulk before", true);
3276 if (pOhci->bulk_cur)
3277 Log(("ohciUndoBulkList: bulk_cur=%#010x before list processing!!! HCD has positioned us!!!\n", pOhci->bulk_cur));
3278#endif
3279
3280 /* This flag follows OHCI_STATUS_BLF, but BLF doesn't change when list processing is disabled. */
3281 pOhci->fBulkNeedsCleaning = false;
3282
3283 uint32_t EdAddr = pOhci->bulk_head;
3284 while (EdAddr)
3285 {
3286 OHCIED Ed;
3287 ohciReadEd(pOhci, EdAddr, &Ed);
3288 Assert(!(Ed.hwinfo & ED_HWINFO_ISO)); /* the guest is screwing us */
3289 if (ohciIsEdPresent(&Ed))
3290 {
3291 uint32_t TdAddr = Ed.HeadP & ED_PTR_MASK;
3292 if (ohciIsTdInFlight(pOhci, TdAddr))
3293 {
3294 LogFlow(("ohciUndoBulkList: Ed=%#010RX32 Ed.TailP=%#010RX32 UNDO\n", EdAddr, Ed.TailP));
3295 PVUSBURB pUrb = ohciTdInFlightUrb(pOhci, TdAddr);
3296 if (pUrb)
3297 pOhci->RootHub.pIRhConn->pfnCancelUrbsEp(pOhci->RootHub.pIRhConn, pUrb);
3298 }
3299 }
3300 /* next endpoint */
3301 EdAddr = Ed.NextED & ED_PTR_MASK;
3302 }
3303}
3304
3305
3306/**
3307 * Services the control list.
3308 *
3309 * The control list has complex URB assembling, but that's taken
3310 * care of at VUSB level (unlike the other transfer types).
3311 */
3312static void ohciServiceCtrlList(POHCI pOhci)
3313{
3314#ifdef LOG_ENABLED
3315 if (g_fLogControlEPs)
3316 ohciDumpEdList(pOhci, pOhci->ctrl_head, "Ctrl before", true);
3317 if (pOhci->ctrl_cur)
3318 Log(("ohciServiceCtrlList: ctrl_cur=%010x before list processing!!! HCD have positioned us!!!\n", pOhci->ctrl_cur));
3319#endif
3320
3321 /*
3322 * ", HC will start processing the list and will set ControlListFilled to 0"
3323 * - We've simplified and are always starting at the head of the list and working
3324 * our way thru to the end each time.
3325 */
3326 pOhci->status &= ~OHCI_STATUS_CLF;
3327 pOhci->ctrl_cur = 0;
3328
3329 uint32_t EdAddr = pOhci->ctrl_head;
3330 while (EdAddr)
3331 {
3332 OHCIED Ed;
3333 ohciReadEd(pOhci, EdAddr, &Ed);
3334 Assert(!(Ed.hwinfo & ED_HWINFO_ISO)); /* the guest is screwing us */
3335 if (ohciIsEdReady(&Ed))
3336 {
3337#if 1
3338 /*
3339 * Control TDs depends on order and stage. Only one can be in-flight
3340 * at any given time. OTOH, some stages are completed immediately,
3341 * so we process the list until we've got a head which is in-fligth
3342 * or reach the end of the list.
3343 */
3344 do
3345 {
3346 if ( !ohciServiceHeadTd(pOhci, VUSBXFERTYPE_CTRL, &Ed, EdAddr, "Control")
3347 || ohciIsTdInFlight(pOhci, Ed.HeadP & ED_PTR_MASK))
3348 {
3349 pOhci->status |= OHCI_STATUS_CLF;
3350 break;
3351 }
3352 ohciReadEd(pOhci, EdAddr, &Ed); /* It might have been updated on URB completion. */
3353 } while (ohciIsEdReady(&Ed));
3354#else
3355 /* Simplistic, for debugging. */
3356 ohciServiceHeadTd(pOhci, VUSBXFERTYPE_CTRL, &Ed, EdAddr, "Control");
3357 pOhci->status |= OHCI_STATUS_CLF;
3358#endif
3359 }
3360
3361 /* next end point */
3362 EdAddr = Ed.NextED & ED_PTR_MASK;
3363 }
3364
3365#ifdef LOG_ENABLED
3366 if (g_fLogControlEPs)
3367 ohciDumpEdList(pOhci, pOhci->ctrl_head, "Ctrl after ", true);
3368#endif
3369}
3370
3371
3372/**
3373 * Services the periodic list.
3374 *
3375 * On the interrupt portion of the periodic list we must reassemble URBs from multiple
3376 * TDs using heuristics derived from USB tracing done in the guests and guest source
3377 * code (when available).
3378 */
3379static void ohciServicePeriodicList(POHCI pOhci)
3380{
3381 /*
3382 * Read the list head from the HCCA.
3383 */
3384 const unsigned iList = pOhci->HcFmNumber % OHCI_HCCA_NUM_INTR;
3385 uint32_t EdAddr;
3386 ohciGetDWords(pOhci, pOhci->hcca + iList * sizeof(EdAddr), &EdAddr, 1);
3387
3388#ifdef LOG_ENABLED
3389 const uint32_t EdAddrHead = EdAddr;
3390 if (g_fLogInterruptEPs)
3391 {
3392 char sz[48];
3393 RTStrPrintf(sz, sizeof(sz), "Int%02x before", iList);
3394 ohciDumpEdList(pOhci, EdAddrHead, sz, true);
3395 }
3396#endif
3397
3398 /*
3399 * Iterate the endpoint list.
3400 */
3401 while (EdAddr)
3402 {
3403 OHCIED Ed;
3404 ohciReadEd(pOhci, EdAddr, &Ed);
3405
3406 if (ohciIsEdReady(&Ed))
3407 {
3408 /*
3409 * "There is no separate head pointer of isochronous transfers. The first
3410 * isochronous Endpoint Descriptor simply links to the last interrupt
3411 * Endpoint Descriptor."
3412 */
3413 if (!(Ed.hwinfo & ED_HWINFO_ISO))
3414 {
3415 /*
3416 * Presently we will only process the head URB on an interrupt endpoint.
3417 */
3418 ohciServiceHeadTdMultiple(pOhci, VUSBXFERTYPE_INTR, &Ed, EdAddr, "Periodic");
3419 }
3420 else if (pOhci->ctl & OHCI_CTL_IE)
3421 {
3422 /*
3423 * Presently only the head ITD.
3424 */
3425 ohciServiceIsochronousEndpoint(pOhci, &Ed, EdAddr);
3426 }
3427 else
3428 break;
3429 }
3430
3431 /* next end point */
3432 EdAddr = Ed.NextED & ED_PTR_MASK;
3433 }
3434
3435#ifdef LOG_ENABLED
3436 if (g_fLogInterruptEPs)
3437 {
3438 char sz[48];
3439 RTStrPrintf(sz, sizeof(sz), "Int%02x after ", iList);
3440 ohciDumpEdList(pOhci, EdAddrHead, sz, true);
3441 }
3442#endif
3443}
3444
3445
3446/**
3447 * Update the HCCA.
3448 *
3449 * @param pOhci The OHCI instance data.
3450 */
3451static void ohciUpdateHCCA(POHCI pOhci)
3452{
3453 struct ohci_hcca hcca;
3454 ohciPhysRead(pOhci, pOhci->hcca + OHCI_HCCA_OFS, &hcca, sizeof(hcca));
3455
3456 hcca.frame = RT_H2LE_U16((uint16_t)pOhci->HcFmNumber);
3457 hcca.pad = 0;
3458
3459 bool fWriteDoneHeadInterrupt = false;
3460 if ( pOhci->dqic == 0
3461 && (pOhci->intr_status & OHCI_INTR_WRITE_DONE_HEAD) == 0)
3462 {
3463 uint32_t done = pOhci->done;
3464
3465 if (pOhci->intr_status & ~( OHCI_INTR_MASTER_INTERRUPT_ENABLED | OHCI_INTR_OWNERSHIP_CHANGE
3466 | OHCI_INTR_WRITE_DONE_HEAD) )
3467 done |= 0x1;
3468
3469 hcca.done = RT_H2LE_U32(done);
3470 pOhci->done = 0;
3471 pOhci->dqic = 0x7;
3472
3473 Log(("ohci: Writeback Done (%#010x) on frame %#x (age %#x)\n", hcca.done,
3474 pOhci->HcFmNumber, pOhci->HcFmNumber - pOhci->u32FmDoneQueueTail));
3475#ifdef LOG_ENABLED
3476 ohciDumpTdQueue(pOhci, hcca.done & ED_PTR_MASK, "DoneQueue");
3477#endif
3478 Assert(RT_OFFSETOF(struct ohci_hcca, done) == 4);
3479#if defined(VBOX_STRICT) || defined(LOG_ENABLED)
3480 ohci_in_done_queue_zap(pOhci);
3481#endif
3482 fWriteDoneHeadInterrupt = true;
3483 }
3484
3485 ohciPhysWrite(pOhci, pOhci->hcca + OHCI_HCCA_OFS, (uint8_t *)&hcca, sizeof(hcca));
3486 if (fWriteDoneHeadInterrupt)
3487 ohciSetInterrupt(pOhci, OHCI_INTR_WRITE_DONE_HEAD);
3488}
3489
3490
3491/**
3492 * Calculate frame timer variables given a frame rate (1,000 Hz is the full speed).
3493 */
3494static void ohciCalcTimerIntervals(POHCI pOhci, uint32_t u32FrameRate)
3495{
3496 Assert(u32FrameRate <= OHCI_DEFAULT_TIMER_FREQ);
3497
3498
3499 pOhci->cTicksPerFrame = pOhci->u64TimerHz / u32FrameRate;
3500 if (!pOhci->cTicksPerFrame)
3501 pOhci->cTicksPerFrame = 1;
3502 pOhci->cTicksPerUsbTick = pOhci->u64TimerHz >= VUSB_BUS_HZ ? pOhci->u64TimerHz / VUSB_BUS_HZ : 1;
3503 pOhci->uFrameRate = u32FrameRate;
3504}
3505
3506
3507/**
3508 * Generate a Start-Of-Frame event, and set a timer for End-Of-Frame.
3509 */
3510static void ohciStartOfFrame(POHCI pOhci)
3511{
3512 uint32_t uNewFrameRate = pOhci->uFrameRate;
3513#ifdef LOG_ENABLED
3514 const uint32_t status_old = pOhci->status;
3515#endif
3516
3517 /*
3518 * Update HcFmRemaining.FRT and re-arm the timer.
3519 */
3520 pOhci->frt = pOhci->fit;
3521#if 1 /* This is required for making the quickcam work on the mac. Should really look
3522 into that adaptive polling stuff... */
3523 pOhci->SofTime += pOhci->cTicksPerFrame;
3524 const uint64_t u64Now = TMTimerGet(pOhci->CTX_SUFF(pEndOfFrameTimer));
3525 if (pOhci->SofTime + pOhci->cTicksPerFrame < u64Now)
3526 pOhci->SofTime = u64Now - pOhci->cTicksPerFrame / 2;
3527#else
3528 pOhci->SofTime = TMTimerGet(pOhci->CTX_SUFF(pEndOfFrameTimer));
3529#endif
3530 TMTimerSet(pOhci->CTX_SUFF(pEndOfFrameTimer), pOhci->SofTime + pOhci->cTicksPerFrame);
3531
3532 /*
3533 * Check that the HCCA address isn't bogus. Linux 2.4.x is known to start
3534 * the bus with a hcca of 0 to work around problem with a specific controller.
3535 */
3536 bool fValidHCCA = !( pOhci->hcca >= OHCI_HCCA_MASK
3537 || pOhci->hcca < ~OHCI_HCCA_MASK);
3538
3539#if 0 /* moved down for higher speed. */
3540 /*
3541 * Update the HCCA.
3542 * Should be done after SOF but before HC read first ED in this frame.
3543 */
3544 if (fValidHCCA)
3545 ohciUpdateHCCA(pOhci);
3546#endif
3547
3548 /* "After writing to HCCA, HC will set SF in HcInterruptStatus" - guest isn't executing, so ignore the order! */
3549 ohciSetInterrupt(pOhci, OHCI_INTR_START_OF_FRAME);
3550
3551 if (pOhci->fno)
3552 {
3553 ohciSetInterrupt(pOhci, OHCI_INTR_FRAMENUMBER_OVERFLOW);
3554 pOhci->fno = 0;
3555 }
3556
3557 /* If the HCCA address is invalid, we're quitting here to avoid doing something which cannot be reported to the HCD. */
3558 if (!fValidHCCA)
3559 {
3560 Log(("ohciStartOfFrame: skipping hcca part because hcca=%RX32 (our 'valid' range: %RX32-%RX32)\n",
3561 pOhci->hcca, ~OHCI_HCCA_MASK, OHCI_HCCA_MASK));
3562 return;
3563 }
3564
3565 /*
3566 * Periodic EPs.
3567 */
3568 if (pOhci->ctl & OHCI_CTL_PLE)
3569 ohciServicePeriodicList(pOhci);
3570
3571 /*
3572 * Control EPs.
3573 */
3574 if ( (pOhci->ctl & OHCI_CTL_CLE)
3575 && (pOhci->status & OHCI_STATUS_CLF) )
3576 ohciServiceCtrlList(pOhci);
3577
3578 /*
3579 * Bulk EPs.
3580 */
3581 if ( (pOhci->ctl & OHCI_CTL_BLE)
3582 && (pOhci->status & OHCI_STATUS_BLF))
3583 ohciServiceBulkList(pOhci);
3584 else if ((pOhci->status & OHCI_STATUS_BLF)
3585 && pOhci->fBulkNeedsCleaning)
3586 ohciUndoBulkList(pOhci); /* If list disabled but not empty, abort endpoints. */
3587
3588#if 1
3589 /*
3590 * Update the HCCA after processing the lists and everything. A bit experimental.
3591 *
3592 * ASSUME the guest won't be very upset if a TD is completed, retired and handed
3593 * back immediately. The idea is to be able to retire the data and/or status stages
3594 * of a control transfer together with the setup stage, thus saving a frame. This
3595 * behaviour is should be perfectly ok, since the setup (and maybe data) stages
3596 * have already taken at least one frame to complete.
3597 *
3598 * But, when implementing the first synchronous virtual USB devices, we'll have to
3599 * verify that the guest doesn't choke when having a TD returned in the same frame
3600 * as it was submitted.
3601 */
3602 ohciUpdateHCCA(pOhci);
3603#endif
3604
3605#ifdef LOG_ENABLED
3606 if (pOhci->status ^ status_old)
3607 {
3608 uint32_t val = pOhci->status;
3609 uint32_t chg = val ^ status_old; NOREF(chg);
3610 Log2(("ohciStartOfFrame: HcCommandStatus=%#010x: %sHCR=%d %sCLF=%d %sBLF=%d %sOCR=%d %sSOC=%d\n",
3611 val,
3612 chg & RT_BIT(0) ? "*" : "", val & 1,
3613 chg & RT_BIT(1) ? "*" : "", (val >> 1) & 1,
3614 chg & RT_BIT(2) ? "*" : "", (val >> 2) & 1,
3615 chg & RT_BIT(3) ? "*" : "", (val >> 3) & 1,
3616 chg & (3<<16)? "*" : "", (val >> 16) & 3));
3617 }
3618#endif
3619
3620 /*
3621 * Adjust the frame timer interval based on idle detection.
3622 */
3623 if (pOhci->fIdle)
3624 {
3625 pOhci->cIdleCycles++;
3626 /* Set the new frame rate based on how long we've been idle. Tunable. */
3627 switch (pOhci->cIdleCycles)
3628 {
3629 case 4: uNewFrameRate = 500; break; /* 2ms interval */
3630 case 16:uNewFrameRate = 125; break; /* 8ms interval */
3631 case 24:uNewFrameRate = 50; break; /* 20ms interval */
3632 default: break;
3633 }
3634 /* Avoid overflow. */
3635 if (pOhci->cIdleCycles > 60000)
3636 pOhci->cIdleCycles = 20000;
3637 }
3638 else
3639 {
3640 if (pOhci->cIdleCycles)
3641 {
3642 pOhci->cIdleCycles = 0;
3643 uNewFrameRate = OHCI_DEFAULT_TIMER_FREQ;
3644 }
3645 }
3646 if (uNewFrameRate != pOhci->uFrameRate)
3647 {
3648 ohciCalcTimerIntervals(pOhci, uNewFrameRate);
3649 if (uNewFrameRate == OHCI_DEFAULT_TIMER_FREQ)
3650 {
3651 /* If we're switching back to full speed, re-program the timer immediately to minimize latency. */
3652 TMTimerSet(pOhci->CTX_SUFF(pEndOfFrameTimer), pOhci->SofTime + pOhci->cTicksPerFrame);
3653 }
3654 }
3655}
3656
3657/**
3658 * Updates the HcFmNumber and FNO registers.
3659 */
3660static void bump_frame_number(POHCI pOhci)
3661{
3662 const uint16_t u16OldFmNumber = pOhci->HcFmNumber++;
3663 if ((u16OldFmNumber ^ pOhci->HcFmNumber) & RT_BIT(15))
3664 pOhci->fno = 1;
3665}
3666
3667/**
3668 * Do frame processing on frame boundary
3669 */
3670static void ohciFrameBoundaryTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
3671{
3672 POHCI pOhci = (POHCI)pvUser;
3673 STAM_PROFILE_START(&pOhci->StatTimer, a);
3674
3675 /* Reset idle detection flag */
3676 pOhci->fIdle = true;
3677
3678 VUSBIRhReapAsyncUrbs(pOhci->RootHub.pIRhConn, 0);
3679
3680 /* Frame boundary, so do EOF stuff here */
3681 bump_frame_number(pOhci);
3682 if ( (pOhci->dqic != 0x7) && (pOhci->dqic != 0) )
3683 pOhci->dqic--;
3684
3685 /* Start the next frame */
3686 ohciStartOfFrame(pOhci);
3687
3688 STAM_PROFILE_STOP(&pOhci->StatTimer, a);
3689}
3690
3691/**
3692 * Start sending SOF tokens across the USB bus, lists are processed in
3693 * next frame
3694 */
3695static void ohciBusStart(POHCI pOhci)
3696{
3697 VUSBIDevPowerOn(pOhci->RootHub.pIDev);
3698 bump_frame_number(pOhci);
3699 pOhci->dqic = 0x7;
3700
3701 Log(("ohci: %s: Bus started\n", pOhci->PciDev.name));
3702
3703 pOhci->SofTime = TMTimerGet(pOhci->CTX_SUFF(pEndOfFrameTimer)) - pOhci->cTicksPerFrame;
3704 pOhci->fIdle = false; /* Assume we won't be idle */
3705 ohciStartOfFrame(pOhci);
3706}
3707
3708/**
3709 * Stop sending SOF tokens on the bus
3710 */
3711static void ohciBusStop(POHCI pOhci)
3712{
3713 if (pOhci->CTX_SUFF(pEndOfFrameTimer))
3714 TMTimerStop(pOhci->CTX_SUFF(pEndOfFrameTimer));
3715 VUSBIDevPowerOff(pOhci->RootHub.pIDev);
3716}
3717
3718/**
3719 * Move in to resume state
3720 */
3721static void ohciBusResume(POHCI pOhci, bool fHardware)
3722{
3723 pOhci->ctl &= ~OHCI_CTL_HCFS;
3724 pOhci->ctl |= OHCI_USB_RESUME;
3725
3726 Log(("pOhci: ohciBusResume fHardware=%RTbool RWE=%s\n",
3727 fHardware, (pOhci->ctl & OHCI_CTL_RWE) ? "on" : "off"));
3728
3729 if (fHardware && (pOhci->ctl & OHCI_CTL_RWE))
3730 ohciSetInterrupt(pOhci, OHCI_INTR_RESUME_DETECT);
3731
3732 ohciBusStart(pOhci);
3733}
3734
3735
3736/* Power a port up or down */
3737static void rhport_power(POHCIROOTHUB pRh, unsigned iPort, bool fPowerUp)
3738{
3739 POHCIHUBPORT pPort = &pRh->aPorts[iPort];
3740 bool fOldPPS = !!(pPort->fReg & OHCI_PORT_PPS);
3741 if (fPowerUp)
3742 {
3743 /* power up */
3744 if (pPort->pDev)
3745 pPort->fReg |= OHCI_PORT_R_CURRENT_CONNECT_STATUS;
3746 if (pPort->fReg & OHCI_PORT_R_CURRENT_CONNECT_STATUS)
3747 pPort->fReg |= OHCI_PORT_R_POWER_STATUS;
3748 if (pPort->pDev && !fOldPPS)
3749 VUSBIDevPowerOn(pPort->pDev);
3750 }
3751 else
3752 {
3753 /* power down */
3754 pPort->fReg &= ~( OHCI_PORT_R_POWER_STATUS
3755 | OHCI_PORT_R_CURRENT_CONNECT_STATUS
3756 | OHCI_PORT_R_SUSPEND_STATUS
3757 | OHCI_PORT_R_RESET_STATUS);
3758 if (pPort->pDev && fOldPPS)
3759 VUSBIDevPowerOff(pPort->pDev);
3760 }
3761}
3762
3763#endif /* IN_RING3 */
3764
3765/**
3766 * Read the HcRevision register.
3767 */
3768static int HcRevision_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
3769{
3770 Log2(("HcRevision_r() -> 0x10\n"));
3771 *pu32Value = 0x10; /* OHCI revision 1.0, no emulation. */
3772 return VINF_SUCCESS;
3773}
3774
3775/**
3776 * Write to the HcRevision register.
3777 */
3778static int HcRevision_w(POHCI pOhci, uint32_t iReg, uint32_t u32Value)
3779{
3780 Log2(("HcRevision_w(%#010x) - denied\n", u32Value));
3781 AssertMsgFailed(("Invalid operation!!! u32Value=%#010x\n", u32Value));
3782 return VINF_SUCCESS;
3783}
3784
3785/**
3786 * Read the HcControl register.
3787 */
3788static int HcControl_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
3789{
3790 uint32_t ctl = pOhci->ctl;
3791 Log2(("HcControl_r -> %#010x - CBSR=%d PLE=%d IE=%d CLE=%d BLE=%d HCFS=%#x IR=%d RWC=%d RWE=%d\n",
3792 ctl, ctl & 3, (ctl >> 2) & 1, (ctl >> 3) & 1, (ctl >> 4) & 1, (ctl >> 5) & 1, (ctl >> 6) & 3, (ctl >> 8) & 1,
3793 (ctl >> 9) & 1, (ctl >> 10) & 1));
3794 *pu32Value = ctl;
3795 return VINF_SUCCESS;
3796}
3797
3798/**
3799 * Write the HcControl register.
3800 */
3801static int HcControl_w(POHCI pOhci, uint32_t iReg, uint32_t val)
3802{
3803 /* log it. */
3804 uint32_t chg = pOhci->ctl ^ val; NOREF(chg);
3805 Log2(("HcControl_w(%#010x) => %sCBSR=%d %sPLE=%d %sIE=%d %sCLE=%d %sBLE=%d %sHCFS=%#x %sIR=%d %sRWC=%d %sRWE=%d\n",
3806 val,
3807 chg & 3 ? "*" : "", val & 3,
3808 chg & RT_BIT(2) ? "*" : "", (val >> 2) & 1,
3809 chg & RT_BIT(3) ? "*" : "", (val >> 3) & 1,
3810 chg & RT_BIT(4) ? "*" : "", (val >> 4) & 1,
3811 chg & RT_BIT(5) ? "*" : "", (val >> 5) & 1,
3812 chg & (3 << 6)? "*" : "", (val >> 6) & 3,
3813 chg & RT_BIT(8) ? "*" : "", (val >> 8) & 1,
3814 chg & RT_BIT(9) ? "*" : "", (val >> 9) & 1,
3815 chg & RT_BIT(10) ? "*" : "", (val >> 10) & 1));
3816 if (val & ~0x07ff)
3817 Log2(("Unknown bits %#x are set!!!\n", val & ~0x07ff));
3818
3819 /* see what changed and take action on that. */
3820 uint32_t old_state = pOhci->ctl & OHCI_CTL_HCFS;
3821 uint32_t new_state = val & OHCI_CTL_HCFS;
3822
3823#ifdef IN_RING3
3824 pOhci->ctl = val;
3825 if (new_state != old_state)
3826 {
3827 switch (new_state)
3828 {
3829 case OHCI_USB_OPERATIONAL:
3830 LogRel(("OHCI: USB Operational\n"));
3831 ohciBusStart(pOhci);
3832 break;
3833 case OHCI_USB_SUSPEND:
3834 ohciBusStop(pOhci);
3835 LogRel(("OHCI: USB Suspended\n"));
3836 break;
3837 case OHCI_USB_RESUME:
3838 LogRel(("OHCI: USB Resume\n"));
3839 ohciBusResume(pOhci, false /* not hardware */);
3840 break;
3841 case OHCI_USB_RESET:
3842 {
3843 LogRel(("OHCI: USB Reset\n"));
3844 ohciBusStop(pOhci);
3845 /** @todo This should probably do a real reset, but we don't implement
3846 * that correctly in the roothub reset callback yet. check it's
3847 * comments and argument for more details. */
3848 VUSBIDevReset(pOhci->RootHub.pIDev, false /* don't do a real reset */, NULL, NULL, NULL);
3849 break;
3850 }
3851 }
3852 }
3853#else /* !IN_RING3 */
3854 if ( new_state != old_state )
3855 {
3856 Log2(("HcControl_w: state changed -> VINF_IOM_HC_MMIO_WRITE\n"));
3857 return VINF_IOM_HC_MMIO_WRITE;
3858 }
3859 pOhci->ctl = val;
3860#endif /* !IN_RING3 */
3861
3862 return VINF_SUCCESS;
3863}
3864
3865/**
3866 * Read the HcCommandStatus register.
3867 */
3868static int HcCommandStatus_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
3869{
3870 uint32_t status = pOhci->status;
3871 Log2(("HcCommandStatus_r() -> %#010x - HCR=%d CLF=%d BLF=%d OCR=%d SOC=%d\n",
3872 status, status & 1, (status >> 1) & 1, (status >> 2) & 1, (status >> 3) & 1, (status >> 16) & 3));
3873 *pu32Value = status;
3874 return VINF_SUCCESS;
3875}
3876
3877/**
3878 * Write to the HcCommandStatus register.
3879 */
3880static int HcCommandStatus_w(POHCI pOhci, uint32_t iReg, uint32_t val)
3881{
3882 /* log */
3883 uint32_t chg = pOhci->status ^ val; NOREF(chg);
3884 Log2(("HcCommandStatus_w(%#010x) => %sHCR=%d %sCLF=%d %sBLF=%d %sOCR=%d %sSOC=%d\n",
3885 val,
3886 chg & RT_BIT(0) ? "*" : "", val & 1,
3887 chg & RT_BIT(1) ? "*" : "", (val >> 1) & 1,
3888 chg & RT_BIT(2) ? "*" : "", (val >> 2) & 1,
3889 chg & RT_BIT(3) ? "*" : "", (val >> 3) & 1,
3890 chg & (3<<16)? "!!!":"", (pOhci->status >> 16) & 3));
3891 if (val & ~0x0003000f)
3892 Log2(("Unknown bits %#x are set!!!\n", val & ~0x0003000f));
3893
3894 /* SOC is read-only */
3895 val = (val & ~OHCI_STATUS_SOC);
3896
3897#ifdef IN_RING3
3898 /* "bits written as '0' remain unchanged in the register" */
3899 pOhci->status |= val;
3900 if (pOhci->status & OHCI_STATUS_HCR)
3901 {
3902 LogRel(("OHCI: Software reset\n"));
3903 ohciDoReset(pOhci, OHCI_USB_SUSPEND, false /* N/A */);
3904 }
3905#else
3906 if ((pOhci->status | val) & OHCI_STATUS_HCR)
3907 {
3908 LogFlow(("HcCommandStatus_w: reset -> VINF_IOM_HC_MMIO_WRITE\n"));
3909 return VINF_IOM_HC_MMIO_WRITE;
3910 }
3911 pOhci->status |= val;
3912#endif
3913 return VINF_SUCCESS;
3914}
3915
3916/**
3917 * Read the HcInterruptStatus register.
3918 */
3919static int HcInterruptStatus_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
3920{
3921 uint32_t val = pOhci->intr_status;
3922 Log2(("HcInterruptStatus_r() -> %#010x - SO=%d WDH=%d SF=%d RD=%d UE=%d FNO=%d RHSC=%d OC=%d\n",
3923 val, val & 1, (val >> 1) & 1, (val >> 2) & 1, (val >> 3) & 1, (val >> 4) & 1, (val >> 5) & 1,
3924 (val >> 6) & 1, (val >> 30) & 1));
3925 *pu32Value = val;
3926 return VINF_SUCCESS;
3927}
3928
3929/**
3930 * Write to the HcInterruptStatus register.
3931 */
3932static int HcInterruptStatus_w(POHCI pOhci, uint32_t iReg, uint32_t val)
3933{
3934 uint32_t res = pOhci->intr_status & ~val;
3935 uint32_t chg = pOhci->intr_status ^ res; NOREF(chg);
3936 Log2(("HcInterruptStatus_w(%#010x) => %sSO=%d %sWDH=%d %sSF=%d %sRD=%d %sUE=%d %sFNO=%d %sRHSC=%d %sOC=%d\n",
3937 val,
3938 chg & RT_BIT(0) ? "*" : "", res & 1,
3939 chg & RT_BIT(1) ? "*" : "", (res >> 1) & 1,
3940 chg & RT_BIT(2) ? "*" : "", (res >> 2) & 1,
3941 chg & RT_BIT(3) ? "*" : "", (res >> 3) & 1,
3942 chg & RT_BIT(4) ? "*" : "", (res >> 4) & 1,
3943 chg & RT_BIT(5) ? "*" : "", (res >> 5) & 1,
3944 chg & RT_BIT(6) ? "*" : "", (res >> 6) & 1,
3945 chg & RT_BIT(30)? "*" : "", (res >> 30) & 1));
3946 if ( (val & ~0xc000007f)
3947 && val != 0xffffffff /* ignore clear-all-like requests from xp. */)
3948 Log2(("Unknown bits %#x are set!!!\n", val & ~0xc000007f));
3949
3950 /* "The Host Controller Driver may clear specific bits in this
3951 * register by writing '1' to bit positions to be cleared"
3952 */
3953 pOhci->intr_status &= ~val;
3954 ohciUpdateInterrupt(pOhci, "HcInterruptStatus_w");
3955 return VINF_SUCCESS;
3956}
3957
3958/**
3959 * Read the HcInterruptEnable register
3960 */
3961static int HcInterruptEnable_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
3962{
3963 uint32_t val = pOhci->intr;
3964 Log2(("HcInterruptEnable_r() -> %#010x - SO=%d WDH=%d SF=%d RD=%d UE=%d FNO=%d RHSC=%d OC=%d MIE=%d\n",
3965 val, val & 1, (val >> 1) & 1, (val >> 2) & 1, (val >> 3) & 1, (val >> 4) & 1, (val >> 5) & 1,
3966 (val >> 6) & 1, (val >> 30) & 1, (val >> 31) & 1));
3967 *pu32Value = val;
3968 return VINF_SUCCESS;
3969}
3970
3971/**
3972 * Writes to the HcInterruptEnable register.
3973 */
3974static int HcInterruptEnable_w(POHCI pOhci, uint32_t iReg, uint32_t val)
3975{
3976 uint32_t res = pOhci->intr | val;
3977 uint32_t chg = pOhci->intr ^ res; NOREF(chg);
3978 Log2(("HcInterruptEnable_w(%#010x) => %sSO=%d %sWDH=%d %sSF=%d %sRD=%d %sUE=%d %sFNO=%d %sRHSC=%d %sOC=%d %sMIE=%d\n",
3979 val,
3980 chg & RT_BIT(0) ? "*" : "", res & 1,
3981 chg & RT_BIT(1) ? "*" : "", (res >> 1) & 1,
3982 chg & RT_BIT(2) ? "*" : "", (res >> 2) & 1,
3983 chg & RT_BIT(3) ? "*" : "", (res >> 3) & 1,
3984 chg & RT_BIT(4) ? "*" : "", (res >> 4) & 1,
3985 chg & RT_BIT(5) ? "*" : "", (res >> 5) & 1,
3986 chg & RT_BIT(6) ? "*" : "", (res >> 6) & 1,
3987 chg & RT_BIT(30) ? "*" : "", (res >> 30) & 1,
3988 chg & RT_BIT(31) ? "*" : "", (res >> 31) & 1));
3989 if (val & ~0xc000007f)
3990 Log2(("Uknown bits %#x are set!!!\n", val & ~0xc000007f));
3991
3992 pOhci->intr |= val;
3993 ohciUpdateInterrupt(pOhci, "HcInterruptEnable_w");
3994 return VINF_SUCCESS;
3995}
3996
3997/**
3998 * Reads the HcInterruptDisable register.
3999 */
4000static int HcInterruptDisable_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4001{
4002#if 1 /** @todo r=bird: "On read, the current value of the HcInterruptEnable register is returned." */
4003 uint32_t val = pOhci->intr;
4004#else /* old code. */
4005 uint32_t val = ~pOhci->intr;
4006#endif
4007 Log2(("HcInterruptDisable_r() -> %#010x - SO=%d WDH=%d SF=%d RD=%d UE=%d FNO=%d RHSC=%d OC=%d MIE=%d\n",
4008 val, val & 1, (val >> 1) & 1, (val >> 2) & 1, (val >> 3) & 1, (val >> 4) & 1, (val >> 5) & 1,
4009 (val >> 6) & 1, (val >> 30) & 1, (val >> 31) & 1));
4010
4011 *pu32Value = val;
4012 return VINF_SUCCESS;
4013}
4014
4015/**
4016 * Writes to the HcInterruptDisable register.
4017 */
4018static int HcInterruptDisable_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4019{
4020 uint32_t res = pOhci->intr & ~val;
4021 uint32_t chg = pOhci->intr ^ res; NOREF(chg);
4022 Log2(("HcInterruptDisable_w(%#010x) => %sSO=%d %sWDH=%d %sSF=%d %sRD=%d %sUE=%d %sFNO=%d %sRHSC=%d %sOC=%d %sMIE=%d\n",
4023 val,
4024 chg & RT_BIT(0) ? "*" : "", res & 1,
4025 chg & RT_BIT(1) ? "*" : "", (res >> 1) & 1,
4026 chg & RT_BIT(2) ? "*" : "", (res >> 2) & 1,
4027 chg & RT_BIT(3) ? "*" : "", (res >> 3) & 1,
4028 chg & RT_BIT(4) ? "*" : "", (res >> 4) & 1,
4029 chg & RT_BIT(5) ? "*" : "", (res >> 5) & 1,
4030 chg & RT_BIT(6) ? "*" : "", (res >> 6) & 1,
4031 chg & RT_BIT(30) ? "*" : "", (res >> 30) & 1,
4032 chg & RT_BIT(31) ? "*" : "", (res >> 31) & 1));
4033 /* Don't bitch about invalid bits here since it makes sense to disable
4034 * interrupts you don't know about. */
4035
4036 pOhci->intr &= ~val;
4037 ohciUpdateInterrupt(pOhci, "HcInterruptDisable_w");
4038 return VINF_SUCCESS;
4039}
4040
4041/**
4042 * Read the HcHCCA register (Host Controller Communications Area physical address).
4043 */
4044static int HcHCCA_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4045{
4046 Log2(("HcHCCA_r() -> %#010x\n", pOhci->hcca));
4047 *pu32Value = pOhci->hcca;
4048 return VINF_SUCCESS;
4049}
4050
4051/**
4052 * Write to the HcHCCA register (Host Controller Communications Area physical address).
4053 */
4054static int HcHCCA_w(POHCI pOhci, uint32_t iReg, uint32_t Value)
4055{
4056 Log2(("HcHCCA_w(%#010x) - old=%#010x new=%#010x\n", Value, pOhci->hcca, Value & OHCI_HCCA_MASK));
4057 pOhci->hcca = Value & OHCI_HCCA_MASK;
4058 return VINF_SUCCESS;
4059}
4060
4061/**
4062 * Read the HcPeriodCurrentED register.
4063 */
4064static int HcPeriodCurrentED_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4065{
4066 Log2(("HcPeriodCurrentED_r() -> %#010x\n", pOhci->per_cur));
4067 *pu32Value = pOhci->per_cur;
4068 return VINF_SUCCESS;
4069}
4070
4071/**
4072 * Write to the HcPeriodCurrentED register.
4073 */
4074static int HcPeriodCurrentED_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4075{
4076 Log(("HcPeriodCurrentED_w(%#010x) - old=%#010x new=%#010x (This is a read only register, only the linux guys don't respect that!)\n",
4077 val, pOhci->per_cur, val & ~7));
4078 //AssertMsgFailed(("HCD (Host Controller Driver) should not write to HcPeriodCurrentED! val=%#010x (old=%#010x)\n", val, pOhci->per_cur));
4079 AssertMsg(!(val & 7), ("Invalid alignment, val=%#010x\n", val));
4080 pOhci->per_cur = val & ~7;
4081 return VINF_SUCCESS;
4082}
4083
4084/**
4085 * Read the HcControlHeadED register.
4086 */
4087static int HcControlHeadED_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4088{
4089 Log2(("HcControlHeadED_r() -> %#010x\n", pOhci->ctrl_head));
4090 *pu32Value = pOhci->ctrl_head;
4091 return VINF_SUCCESS;
4092}
4093
4094/**
4095 * Write to the HcControlHeadED register.
4096 */
4097static int HcControlHeadED_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4098{
4099 Log2(("HcControlHeadED_w(%#010x) - old=%#010x new=%#010x\n", val, pOhci->ctrl_head, val & ~7));
4100 AssertMsg(!(val & 7), ("Invalid alignment, val=%#010x\n", val));
4101 pOhci->ctrl_head = val & ~7;
4102 return VINF_SUCCESS;
4103}
4104
4105/**
4106 * Read the HcControlCurrentED register.
4107 */
4108static int HcControlCurrentED_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4109{
4110 Log2(("HcControlCurrentED_r() -> %#010x\n", pOhci->ctrl_cur));
4111 *pu32Value = pOhci->ctrl_cur;
4112 return VINF_SUCCESS;
4113}
4114
4115/**
4116 * Write to the HcControlCurrentED register.
4117 */
4118static int HcControlCurrentED_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4119{
4120 Log2(("HcControlCurrentED_w(%#010x) - old=%#010x new=%#010x\n", val, pOhci->ctrl_cur, val & ~7));
4121 AssertMsg(!(pOhci->ctl & OHCI_CTL_CLE), ("Illegal write! HcControl.ControlListEnabled is set! val=%#010x\n", val));
4122 AssertMsg(!(val & 7), ("Invalid alignment, val=%#010x\n", val));
4123 pOhci->ctrl_cur = val & ~7;
4124 return VINF_SUCCESS;
4125}
4126
4127/**
4128 * Read the HcBulkHeadED register.
4129 */
4130static int HcBulkHeadED_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4131{
4132 Log2(("HcBulkHeadED_r() -> %#010x\n", pOhci->bulk_head));
4133 *pu32Value = pOhci->bulk_head;
4134 return VINF_SUCCESS;
4135}
4136
4137/**
4138 * Write to the HcBulkHeadED register.
4139 */
4140static int HcBulkHeadED_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4141{
4142 Log2(("HcBulkHeadED_w(%#010x) - old=%#010x new=%#010x\n", val, pOhci->bulk_head, val & ~7));
4143 AssertMsg(!(val & 7), ("Invalid alignment, val=%#010x\n", val));
4144 pOhci->bulk_head = val & ~7;
4145 return VINF_SUCCESS;
4146}
4147
4148/**
4149 * Read the HcBulkCurrentED register.
4150 */
4151static int HcBulkCurrentED_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4152{
4153 Log2(("HcBulkCurrentED_r() -> %#010x\n", pOhci->bulk_cur));
4154 *pu32Value = pOhci->bulk_cur;
4155 return VINF_SUCCESS;
4156}
4157
4158/**
4159 * Write to the HcBulkCurrentED register.
4160 */
4161static int HcBulkCurrentED_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4162{
4163 Log2(("HcBulkCurrentED_w(%#010x) - old=%#010x new=%#010x\n", val, pOhci->bulk_cur, val & ~7));
4164 AssertMsg(!(pOhci->ctl & OHCI_CTL_BLE), ("Illegal write! HcControl.BulkListEnabled is set! val=%#010x\n", val));
4165 AssertMsg(!(val & 7), ("Invalid alignment, val=%#010x\n", val));
4166 pOhci->bulk_cur = val & ~7;
4167 return VINF_SUCCESS;
4168}
4169
4170
4171/**
4172 * Read the HcDoneHead register.
4173 */
4174static int HcDoneHead_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4175{
4176 Log2(("HcDoneHead_r() -> 0x%#08x\n", pOhci->done));
4177 *pu32Value = pOhci->done;
4178 return VINF_SUCCESS;
4179}
4180
4181/**
4182 * Write to the HcDoneHead register.
4183 */
4184static int HcDoneHead_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4185{
4186 Log2(("HcDoneHead_w(0x%#08x) - denied!!!\n", val));
4187 AssertMsgFailed(("Illegal operation!!! val=%#010x\n", val));
4188 return VINF_SUCCESS;
4189}
4190
4191
4192/**
4193 * Read the HcFmInterval (Fm=Frame) register.
4194 */
4195static int HcFmInterval_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4196{
4197 uint32_t val = (pOhci->fit << 31) | (pOhci->fsmps << 16) | (pOhci->fi);
4198 Log2(("HcFmInterval_r() -> 0x%#08x - FI=%d FSMPS=%d FIT=%d\n",
4199 val, val & 0x3fff, (val >> 16) & 0x7fff, val >> 31));
4200 *pu32Value = val;
4201 return VINF_SUCCESS;
4202}
4203
4204/**
4205 * Write to the HcFmInterval (Fm = Frame) register.
4206 */
4207static int HcFmInterval_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4208{
4209 /* log */
4210 uint32_t chg = val ^ ((pOhci->fit << 31) | (pOhci->fsmps << 16) | pOhci->fi); NOREF(chg);
4211 Log2(("HcFmInterval_w(%#010x) => %sFI=%d %sFSMPS=%d %sFIT=%d\n",
4212 val,
4213 chg & 0x00003fff ? "*" : "", val & 0x3fff,
4214 chg & 0x7fff0000 ? "*" : "", (val >> 16) & 0x7fff,
4215 chg >> 31 ? "*" : "", (val >> 31) & 1));
4216 if ( pOhci->fi != (val & OHCI_FMI_FI) )
4217 {
4218 Log(("ohci: FrameInterval: %#010x -> %#010x\n", pOhci->fi, val & OHCI_FMI_FI));
4219 AssertMsg(pOhci->fit != ((val >> OHCI_FMI_FIT_SHIFT) & 1), ("HCD didn't toggle the FIT bit!!!\n"));
4220 }
4221
4222 /* update */
4223 pOhci->fi = val & OHCI_FMI_FI;
4224 pOhci->fit = (val & OHCI_FMI_FIT) >> OHCI_FMI_FIT_SHIFT;
4225 pOhci->fsmps = (val & OHCI_FMI_FSMPS) >> OHCI_FMI_FSMPS_SHIFT;
4226 return VINF_SUCCESS;
4227}
4228
4229/**
4230 * Read the HcFmRemaining (Fm = Frame) register.
4231 */
4232static int HcFmRemaining_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4233{
4234 uint32_t Value = pOhci->frt << 31;
4235 if ((pOhci->ctl & OHCI_CTL_HCFS) == OHCI_USB_OPERATIONAL)
4236 {
4237 /*
4238 * Being in USB operational state guarantees SofTime was set already.
4239 */
4240 uint64_t tks = TMTimerGet(pOhci->CTX_SUFF(pEndOfFrameTimer)) - pOhci->SofTime;
4241 if (tks < pOhci->cTicksPerFrame) /* avoid muldiv if possible */
4242 {
4243 uint16_t fr;
4244 tks = ASMMultU64ByU32DivByU32(1, tks, pOhci->cTicksPerUsbTick);
4245 fr = (uint16_t)(pOhci->fi - tks);
4246 Value |= fr;
4247 }
4248 }
4249
4250 Log2(("HcFmRemaining_r() -> %#010x - FR=%d FRT=%d\n", Value, Value & 0x3fff, Value >> 31));
4251 *pu32Value = Value;
4252 return VINF_SUCCESS;
4253}
4254
4255/**
4256 * Write to the HcFmRemaining (Fm = Frame) register.
4257 */
4258static int HcFmRemaining_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4259{
4260 Log2(("HcFmRemaining_w(%#010x) - denied\n", val));
4261 AssertMsgFailed(("Invalid operation!!! val=%#010x\n", val));
4262 return VINF_SUCCESS;
4263}
4264
4265/**
4266 * Read the HcFmNumber (Fm = Frame) register.
4267 */
4268static int HcFmNumber_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4269{
4270 uint32_t val = (uint16_t)pOhci->HcFmNumber;
4271 Log2(("HcFmNumber_r() -> %#010x - FN=%#x(%d) (32-bit=%#x(%d))\n", val, val, val, pOhci->HcFmNumber, pOhci->HcFmNumber));
4272 *pu32Value = val;
4273 return VINF_SUCCESS;
4274}
4275
4276/**
4277 * Write to the HcFmNumber (Fm = Frame) register.
4278 */
4279static int HcFmNumber_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4280{
4281 Log2(("HcFmNumber_w(%#010x) - denied\n", val));
4282 AssertMsgFailed(("Invalid operation!!! val=%#010x\n", val));
4283 return VINF_SUCCESS;
4284}
4285
4286/**
4287 * Read the HcPeriodicStart register.
4288 * The register determines when in a frame to switch from control&bulk to periodic lists.
4289 */
4290static int HcPeriodicStart_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4291{
4292 Log2(("HcPeriodicStart_r() -> %#010x - PS=%d\n", pOhci->pstart, pOhci->pstart & 0x3fff));
4293 *pu32Value = pOhci->pstart;
4294 return VINF_SUCCESS;
4295}
4296
4297/**
4298 * Write to the HcPeriodicStart register.
4299 * The register determines when in a frame to switch from control&bulk to periodic lists.
4300 */
4301static int HcPeriodicStart_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4302{
4303 Log2(("HcPeriodicStart_w(%#010x) => PS=%d\n", val, val & 0x3fff));
4304 if (val & ~0x3fff)
4305 Log2(("Unknown bits %#x are set!!!\n", val & ~0x3fff));
4306 pOhci->pstart = val; /** @todo r=bird: should we support setting the other bits? */
4307 return VINF_SUCCESS;
4308}
4309
4310/**
4311 * Read the HcLSThreshold register.
4312 */
4313static int HcLSThreshold_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4314{
4315 Log2(("HcLSThreshold_r() -> %#010x\n", OHCI_LS_THRESH));
4316 *pu32Value = OHCI_LS_THRESH;
4317 return VINF_SUCCESS;
4318}
4319
4320/**
4321 * Write to the HcLSThreshold register.
4322 *
4323 * Docs are inconsistent here:
4324 *
4325 * "Neither the Host Controller nor the Host Controller Driver are allowed to change this value."
4326 *
4327 * "This value is calculated by HCD with the consideration of transmission and setup overhead."
4328 *
4329 * The register is marked "R/W" the HCD column.
4330 *
4331 */
4332static int HcLSThreshold_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4333{
4334 Log2(("HcLSThreshold_w(%#010x) => LST=0x%03x(%d)\n", val, val & 0x0fff, val & 0x0fff));
4335 AssertMsg(val == OHCI_LS_THRESH,
4336 ("HCD tried to write bad LS threshold: 0x%x (see function header)\n", val));
4337 return VINF_SUCCESS;
4338}
4339
4340/**
4341 * Read the HcRhDescriptorA register.
4342 */
4343static int HcRhDescriptorA_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4344{
4345 uint32_t val = pOhci->RootHub.desc_a;
4346#if 0 /* annoying */
4347 Log2(("HcRhDescriptorA_r() -> %#010x - NDP=%d PSM=%d NPS=%d DT=%d OCPM=%d NOCP=%d POTGT=%#x\n",
4348 val, val & 0xff, (val >> 8) & 1, (val >> 9) & 1, (val >> 10) & 1, (val >> 11) & 1,
4349 (val >> 12) & 1, (val >> 24) & 0xff));
4350#endif
4351 *pu32Value = val;
4352 return VINF_SUCCESS;
4353}
4354
4355/**
4356 * Write to the HcRhDescriptorA register.
4357 */
4358static int HcRhDescriptorA_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4359{
4360 uint32_t chg = val ^ pOhci->RootHub.desc_a; NOREF(chg);
4361 Log2(("HcRhDescriptorA_w(%#010x) => %sNDP=%d %sPSM=%d %sNPS=%d %sDT=%d %sOCPM=%d %sNOCP=%d %sPOTGT=%#x - %sPowerSwitching Set%sPower\n",
4362 val,
4363 chg & 0xff ?"!!!": "", OHCI_NDP,
4364 (chg >> 8) & 1 ? "*" : "", (val >> 8) & 1,
4365 (chg >> 9) & 1 ? "*" : "", (val >> 9) & 1,
4366 (chg >> 10) & 1 ?"!!!": "", 0,
4367 (chg >> 11) & 1 ? "*" : "", (val >> 11) & 1,
4368 (chg >> 12) & 1 ? "*" : "", (val >> 12) & 1,
4369 (chg >> 24)&0xff? "*" : "", (val >> 24) & 0xff,
4370 val & OHCI_RHA_NPS ? "No" : "",
4371 val & OHCI_RHA_PSM ? "Port" : "Global"));
4372 if (val & ~0xff001fff)
4373 Log2(("Unknown bits %#x are set!!!\n", val & ~0xff001fff));
4374
4375
4376 if ((val & (OHCI_RHA_NDP | OHCI_RHA_DT)) != OHCI_NDP)
4377 {
4378 Log(("ohci: %s: invalid write to NDP or DT in roothub descriptor A!!! val=0x%.8x\n",
4379 pOhci->PciDev.name, val));
4380 val &= ~(OHCI_RHA_NDP | OHCI_RHA_DT);
4381 val |= OHCI_NDP;
4382 }
4383
4384 pOhci->RootHub.desc_a = val;
4385 return VINF_SUCCESS;
4386}
4387
4388/**
4389 * Read the HcRhDescriptorB register.
4390 */
4391static int HcRhDescriptorB_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4392{
4393 uint32_t val = pOhci->RootHub.desc_b;
4394 Log2(("HcRhDescriptorB_r() -> %#010x - DR=0x%04x PPCM=0x%04x\n",
4395 val, val & 0xffff, val >> 16));
4396 *pu32Value = val;
4397 return VINF_SUCCESS;
4398}
4399
4400/**
4401 * Write to the HcRhDescriptorB register.
4402 */
4403static int HcRhDescriptorB_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4404{
4405 uint32_t chg = pOhci->RootHub.desc_b ^ val; NOREF(chg);
4406 Log2(("HcRhDescriptorB_w(%#010x) => %sDR=0x%04x %sPPCM=0x%04x\n",
4407 val,
4408 chg & 0xffff ? "!!!" : "", val & 0xffff,
4409 chg >> 16 ? "!!!" : "", val >> 16));
4410
4411 if ( pOhci->RootHub.desc_b != val )
4412 Log(("ohci: %s: unsupported write to root descriptor B!!! 0x%.8x -> 0x%.8x\n",
4413 pOhci->PciDev.name,
4414 pOhci->RootHub.desc_b, val));
4415 pOhci->RootHub.desc_b = val;
4416 return VINF_SUCCESS;
4417}
4418
4419/**
4420 * Read the HcRhStatus (Rh = Root Hub) register.
4421 */
4422static int HcRhStatus_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4423{
4424 uint32_t val = pOhci->RootHub.status;
4425 if (val & (OHCI_RHS_LPSC | OHCI_RHS_OCIC))
4426 Log2(("HcRhStatus_r() -> %#010x - LPS=%d OCI=%d DRWE=%d LPSC=%d OCIC=%d CRWE=%d\n",
4427 val, val & 1, (val >> 1) & 1, (val >> 15) & 1, (val >> 16) & 1, (val >> 17) & 1, (val >> 31) & 1));
4428 *pu32Value = val;
4429 return VINF_SUCCESS;
4430}
4431
4432/**
4433 * Write to the HcRhStatus (Rh = Root Hub) register.
4434 */
4435static int HcRhStatus_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4436{
4437#ifdef IN_RING3
4438 /* log */
4439 uint32_t old = pOhci->RootHub.status;
4440 uint32_t chg;
4441 if (val & ~0x80038003)
4442 Log2(("HcRhStatus_w: Unknown bits %#x are set!!!\n", val & ~0x80038003));
4443 if ( (val & OHCI_RHS_LPSC) && (val & OHCI_RHS_LPS) )
4444 Log2(("HcRhStatus_w: Warning both CGP and SGP are set! (Clear/Set Global Power)\n"));
4445 if ( (val & OHCI_RHS_DRWE) && (val & OHCI_RHS_CRWE) )
4446 Log2(("HcRhStatus_w: Warning both CRWE and SRWE are set! (Clear/Set Remote Wakeup Enable)\n"));
4447
4448
4449 /* write 1 to clear OCIC */
4450 if ( val & OHCI_RHS_OCIC )
4451 pOhci->RootHub.status &= ~OHCI_RHS_OCIC;
4452
4453 /* SetGlobalPower */
4454 if ( val & OHCI_RHS_LPSC )
4455 {
4456 int i;
4457 Log2(("ohci: %s: global power up\n", pOhci->PciDev.name));
4458 for (i = 0; i < OHCI_NDP; i++)
4459 rhport_power(&pOhci->RootHub, i, true /* power up */);
4460 }
4461
4462 /* ClearGlobalPower */
4463 if ( val & OHCI_RHS_LPS )
4464 {
4465 int i;
4466 Log2(("ohci: %s: global power down\n", pOhci->PciDev.name));
4467 for (i = 0; i < OHCI_NDP; i++)
4468 rhport_power(&pOhci->RootHub, i, false /* power down */);
4469 }
4470
4471 if ( val & OHCI_RHS_DRWE )
4472 pOhci->RootHub.status |= OHCI_RHS_DRWE;
4473
4474 if ( val & OHCI_RHS_CRWE )
4475 pOhci->RootHub.status &= ~OHCI_RHS_DRWE;
4476
4477 chg = pOhci->RootHub.status ^ old;
4478 Log2(("HcRhStatus_w(%#010x) => %sCGP=%d %sOCI=%d %sSRWE=%d %sSGP=%d %sOCIC=%d %sCRWE=%d\n",
4479 val,
4480 chg & 1 ? "*" : "", val & 1,
4481 (chg >> 1) & 1 ?"!!!": "", (val >> 1) & 1,
4482 (chg >> 15) & 1 ? "*" : "", (val >> 15) & 1,
4483 (chg >> 16) & 1 ? "*" : "", (val >> 16) & 1,
4484 (chg >> 17) & 1 ? "*" : "", (val >> 17) & 1,
4485 (chg >> 31) & 1 ? "*" : "", (val >> 31) & 1));
4486 return VINF_SUCCESS;
4487#else /* !IN_RING3 */
4488 return VINF_IOM_HC_MMIO_WRITE;
4489#endif /* !IN_RING3 */
4490}
4491
4492/**
4493 * Read the HcRhPortStatus register of a port.
4494 */
4495static int HcRhPortStatus_r(POHCI pOhci, uint32_t iReg, uint32_t *pu32Value)
4496{
4497 const unsigned i = iReg - 21;
4498 uint32_t val = pOhci->RootHub.aPorts[i].fReg | OHCI_PORT_R_POWER_STATUS; /* PortPowerStatus: see todo on power in _w function. */
4499 if (val & OHCI_PORT_R_RESET_STATUS)
4500 {
4501#ifdef IN_RING3
4502 RTThreadYield();
4503#else
4504 Log2(("HcRhPortStatus_r: yield -> VINF_IOM_HC_MMIO_READ\n"));
4505 return VINF_IOM_HC_MMIO_READ;
4506#endif
4507 }
4508 if (val & (OHCI_PORT_R_RESET_STATUS | OHCI_PORT_CSC | OHCI_PORT_PESC | OHCI_PORT_PSSC | OHCI_PORT_OCIC | OHCI_PORT_PRSC))
4509 Log2(("HcRhPortStatus_r(): port %u: -> %#010x - CCS=%d PES=%d PSS=%d POCI=%d RRS=%d PPS=%d LSDA=%d CSC=%d PESC=%d PSSC=%d OCIC=%d PRSC=%d\n",
4510 i, val, val & 1, (val >> 1) & 1, (val >> 2) & 1, (val >> 3) & 1, (val >> 4) & 1, (val >> 8) & 1, (val >> 9) & 1,
4511 (val >> 16) & 1, (val >> 17) & 1, (val >> 18) & 1, (val >> 19) & 1, (val >> 20) & 1));
4512 *pu32Value = val;
4513 return VINF_SUCCESS;
4514}
4515
4516#ifdef IN_RING3
4517/**
4518 * Completion callback for the vusb_dev_reset() operation.
4519 * @thread EMT.
4520 */
4521static DECLCALLBACK(void) uchi_port_reset_done(PVUSBIDEVICE pDev, int rc, void *pvUser)
4522{
4523 POHCI pOhci = (POHCI)pvUser;
4524
4525 /*
4526 * Find the port in question
4527 */
4528 POHCIHUBPORT pPort = NULL;
4529 unsigned iPort;
4530 for (iPort = 0; iPort < RT_ELEMENTS(pOhci->RootHub.aPorts); iPort++) /* lazy bird */
4531 if (pOhci->RootHub.aPorts[iPort].pDev == pDev)
4532 {
4533 pPort = &pOhci->RootHub.aPorts[iPort];
4534 break;
4535 }
4536 if (!pPort)
4537 {
4538 Assert(pPort); /* sometimes happens because of #1510 */
4539 return;
4540 }
4541
4542 if (RT_SUCCESS(rc))
4543 {
4544 /*
4545 * Successful reset.
4546 */
4547 Log2(("uchi_port_reset_done: Reset completed.\n"));
4548 pPort->fReg &= ~(OHCI_PORT_R_RESET_STATUS | OHCI_PORT_R_SUSPEND_STATUS | OHCI_PORT_R_SUSPEND_STATUS_CHANGE);
4549 pPort->fReg |= OHCI_PORT_R_ENABLE_STATUS | OHCI_PORT_R_RESET_STATUS_CHANGE;
4550 }
4551 else
4552 {
4553 /* desperate measures. */
4554 if ( pPort->pDev
4555 && VUSBIDevGetState(pPort->pDev) == VUSB_DEVICE_STATE_ATTACHED)
4556 {
4557 /*
4558 * Damn, something weird happened during reset. We'll pretend the user did an
4559 * incredible fast reconnect or something. (probably not gonna work)
4560 */
4561 Log2(("uchi_port_reset_done: The reset failed (rc=%Rrc)!!! Pretending reconnect at the speed of light.\n", rc));
4562 pPort->fReg = OHCI_PORT_R_CURRENT_CONNECT_STATUS | OHCI_PORT_R_CONNECT_STATUS_CHANGE;
4563 }
4564 else
4565 {
4566 /*
4567 * The device have / will be disconnected.
4568 */
4569 Log2(("uchi_port_reset_done: Disconnected (rc=%Rrc)!!!\n", rc));
4570 pPort->fReg &= ~(OHCI_PORT_R_RESET_STATUS | OHCI_PORT_R_SUSPEND_STATUS | OHCI_PORT_R_SUSPEND_STATUS_CHANGE | OHCI_PORT_R_RESET_STATUS_CHANGE);
4571 pPort->fReg |= OHCI_PORT_R_CONNECT_STATUS_CHANGE;
4572 }
4573 }
4574
4575 /* Raise roothub status change interrupt. */
4576 ohciSetInterrupt(pOhci, OHCI_INTR_ROOT_HUB_STATUS_CHANGE);
4577}
4578
4579/**
4580 * Sets a flag in a port status register but only set it if a device is
4581 * connected, if not set ConnectStatusChange flag to force HCD to reevaluate
4582 * connect status.
4583 *
4584 * @returns true if device was connected and the flag was cleared.
4585 */
4586static bool rhport_set_if_connected(POHCIROOTHUB pRh, int iPort, uint32_t fValue)
4587{
4588 /*
4589 * Writing a 0 has no effect
4590 */
4591 if (fValue == 0)
4592 return false;
4593
4594 /*
4595 * If CurrentConnectStatus is cleared we set ConnectStatusChange.
4596 */
4597 if (!(pRh->aPorts[iPort].fReg & OHCI_PORT_R_CURRENT_CONNECT_STATUS))
4598 {
4599 pRh->aPorts[iPort].fReg |= OHCI_PORT_R_CONNECT_STATUS_CHANGE;
4600 ohciSetInterrupt(pRh->pOhci, OHCI_INTR_ROOT_HUB_STATUS_CHANGE);
4601 return false;
4602 }
4603
4604 bool fRc = !(pRh->aPorts[iPort].fReg & fValue);
4605
4606 /* set the bit */
4607 pRh->aPorts[iPort].fReg |= fValue;
4608
4609 return fRc;
4610}
4611#endif /* IN_RING3 */
4612
4613/**
4614 * Write to the HcRhPortStatus register of a port.
4615 */
4616static int HcRhPortStatus_w(POHCI pOhci, uint32_t iReg, uint32_t val)
4617{
4618#ifdef IN_RING3
4619 const unsigned i = iReg - 21;
4620 POHCIHUBPORT p = &pOhci->RootHub.aPorts[i];
4621 uint32_t old_state = p->fReg;
4622
4623#ifdef LOG_ENABLED
4624 /*
4625 * Log it.
4626 */
4627 static const char *apszCmdNames[32] =
4628 {
4629 "ClearPortEnable", "SetPortEnable", "SetPortSuspend", "!!!ClearSuspendStatus",
4630 "SetPortReset", "!!!5", "!!!6", "!!!7",
4631 "SetPortPower", "ClearPortPower", "!!!10", "!!!11",
4632 "!!!12", "!!!13", "!!!14", "!!!15",
4633 "ClearCSC", "ClearPESC", "ClearPSSC", "ClearOCIC",
4634 "ClearPRSC", "!!!21", "!!!22", "!!!23",
4635 "!!!24", "!!!25", "!!!26", "!!!27",
4636 "!!!28", "!!!29", "!!!30", "!!!31"
4637 };
4638 Log2(("HcRhPortStatus_w(%#010x): port %u:", val, i));
4639 for (unsigned j = 0; j < RT_ELEMENTS(apszCmdNames); j++)
4640 if (val & (1 << j))
4641 Log2((" %s", apszCmdNames[j]));
4642 Log2(("\n"));
4643#endif
4644
4645 /* Write to clear any of the change bits: CSC, PESC, PSSC, OCIC and PRSC */
4646 if (val & OHCI_PORT_W_CLEAR_CHANGE_MASK)
4647 p->fReg &= ~(val & OHCI_PORT_W_CLEAR_CHANGE_MASK);
4648
4649 if (val & OHCI_PORT_W_CLEAR_ENABLE)
4650 {
4651 p->fReg &= ~OHCI_PORT_R_ENABLE_STATUS;
4652 Log2(("HcRhPortStatus_w(): port %u: DISABLE\n", i));
4653 }
4654
4655 if (rhport_set_if_connected(&pOhci->RootHub, i, val & OHCI_PORT_W_SET_ENABLE))
4656 Log2(("HcRhPortStatus_w(): port %u: ENABLE\n", i));
4657
4658 if (rhport_set_if_connected(&pOhci->RootHub, i, val & OHCI_PORT_W_SET_SUSPEND))
4659 Log2(("HcRhPortStatus_w(): port %u: SUSPEND - not implemented correctly!!!\n", i));
4660
4661 if (val & OHCI_PORT_W_SET_RESET)
4662 {
4663 if (rhport_set_if_connected(&pOhci->RootHub, i, val & OHCI_PORT_W_SET_RESET))
4664 {
4665 PVM pVM = PDMDevHlpGetVM(pOhci->CTX_SUFF(pDevIns));
4666 p->fReg &= ~OHCI_PORT_R_RESET_STATUS_CHANGE;
4667 VUSBIDevReset(p->pDev, false /* don't reset on linux */, uchi_port_reset_done, pOhci, pVM);
4668 }
4669 else if (p->fReg & OHCI_PORT_R_RESET_STATUS)
4670 {
4671 /* the guest is getting impatient. */
4672 Log2(("HcRhPortStatus_w(): port %u: Impatient guest!\n"));
4673 RTThreadYield();
4674 }
4675 }
4676
4677 if (!(pOhci->RootHub.desc_a & OHCI_RHA_NPS))
4678 {
4679 /** @todo To implement per-device power-switching
4680 * we need to check PortPowerControlMask to make
4681 * sure it isn't gang powered
4682 */
4683 if (val & OHCI_PORT_W_CLEAR_POWER)
4684 rhport_power(&pOhci->RootHub, i, false /* power down */);
4685 if (val & OHCI_PORT_W_SET_POWER)
4686 rhport_power(&pOhci->RootHub, i, true /* power up */);
4687 }
4688
4689 /** @todo r=frank: ClearSuspendStatus. Timing? */
4690 if (val & OHCI_PORT_W_CLEAR_SUSPEND_STATUS)
4691 {
4692 rhport_power(&pOhci->RootHub, i, true /* power up */);
4693 pOhci->RootHub.aPorts[i].fReg &= ~OHCI_PORT_R_SUSPEND_STATUS;
4694 pOhci->RootHub.aPorts[i].fReg |= OHCI_PORT_R_SUSPEND_STATUS_CHANGE;
4695 ohciSetInterrupt(pOhci, OHCI_INTR_ROOT_HUB_STATUS_CHANGE);
4696 }
4697
4698 if (p->fReg != old_state)
4699 {
4700 uint32_t res = p->fReg;
4701 uint32_t chg = res ^ old_state; NOREF(chg);
4702 Log2(("HcRhPortStatus_w(%#010x): port %u: => %sCCS=%d %sPES=%d %sPSS=%d %sPOCI=%d %sRRS=%d %sPPS=%d %sLSDA=%d %sCSC=%d %sPESC=%d %sPSSC=%d %sOCIC=%d %sPRSC=%d\n",
4703 val, i,
4704 chg & 1 ? "*" : "", res & 1,
4705 (chg >> 1) & 1 ? "*" : "", (res >> 1) & 1,
4706 (chg >> 2) & 1 ? "*" : "", (res >> 2) & 1,
4707 (chg >> 3) & 1 ? "*" : "", (res >> 3) & 1,
4708 (chg >> 4) & 1 ? "*" : "", (res >> 4) & 1,
4709 (chg >> 8) & 1 ? "*" : "", (res >> 8) & 1,
4710 (chg >> 9) & 1 ? "*" : "", (res >> 9) & 1,
4711 (chg >> 16) & 1 ? "*" : "", (res >> 16) & 1,
4712 (chg >> 17) & 1 ? "*" : "", (res >> 17) & 1,
4713 (chg >> 18) & 1 ? "*" : "", (res >> 18) & 1,
4714 (chg >> 19) & 1 ? "*" : "", (res >> 19) & 1,
4715 (chg >> 20) & 1 ? "*" : "", (res >> 20) & 1));
4716 }
4717 return VINF_SUCCESS;
4718#else /* !IN_RING3 */
4719 return VINF_IOM_HC_MMIO_WRITE;
4720#endif /* !IN_RING3 */
4721}
4722
4723/**
4724 * Register descriptor table
4725 */
4726static const OHCIOPREG g_aOpRegs[] =
4727{
4728 {"HcRevision", HcRevision_r, HcRevision_w},
4729 {"HcControl", HcControl_r, HcControl_w},
4730 {"HcCommandStatus", HcCommandStatus_r, HcCommandStatus_w},
4731 {"HcInterruptStatus", HcInterruptStatus_r, HcInterruptStatus_w},
4732 {"HcInterruptEnable", HcInterruptEnable_r, HcInterruptEnable_w},
4733 {"HcInterruptDisable", HcInterruptDisable_r, HcInterruptDisable_w},
4734 {"HcHCCA", HcHCCA_r, HcHCCA_w},
4735 {"HcPeriodCurrentED", HcPeriodCurrentED_r, HcPeriodCurrentED_w},
4736 {"HcControlHeadED", HcControlHeadED_r, HcControlHeadED_w},
4737 {"HcControlCurrentED", HcControlCurrentED_r, HcControlCurrentED_w},
4738 {"HcBulkHeadED", HcBulkHeadED_r, HcBulkHeadED_w},
4739 {"HcBulkCurrentED", HcBulkCurrentED_r, HcBulkCurrentED_w},
4740 {"HcDoneHead", HcDoneHead_r, HcDoneHead_w},
4741 {"HcFmInterval", HcFmInterval_r, HcFmInterval_w},
4742 {"HcFmRemaining", HcFmRemaining_r, HcFmRemaining_w},
4743 {"HcFmNumber", HcFmNumber_r, HcFmNumber_w},
4744 {"HcPeriodicStart", HcPeriodicStart_r, HcPeriodicStart_w},
4745 {"HcLSThreshold", HcLSThreshold_r, HcLSThreshold_w},
4746 {"HcRhDescriptorA", HcRhDescriptorA_r, HcRhDescriptorA_w},
4747 {"HcRhDescriptorB", HcRhDescriptorB_r, HcRhDescriptorB_w},
4748 {"HcRhStatus", HcRhStatus_r, HcRhStatus_w},
4749
4750 /* The number of port status register depends on the definition
4751 * of OHCI_NDP macro
4752 */
4753 {"HcRhPortStatus[0]", HcRhPortStatus_r, HcRhPortStatus_w},
4754 {"HcRhPortStatus[1]", HcRhPortStatus_r, HcRhPortStatus_w},
4755 {"HcRhPortStatus[2]", HcRhPortStatus_r, HcRhPortStatus_w},
4756 {"HcRhPortStatus[3]", HcRhPortStatus_r, HcRhPortStatus_w},
4757 {"HcRhPortStatus[4]", HcRhPortStatus_r, HcRhPortStatus_w},
4758 {"HcRhPortStatus[5]", HcRhPortStatus_r, HcRhPortStatus_w},
4759 {"HcRhPortStatus[6]", HcRhPortStatus_r, HcRhPortStatus_w},
4760 {"HcRhPortStatus[7]", HcRhPortStatus_r, HcRhPortStatus_w},
4761};
4762
4763
4764/**
4765 * Read a MMIO register.
4766 *
4767 * We only accept 32-bit writes that are 32-bit aligned.
4768 *
4769 * @returns VBox status code suitable for scheduling.
4770 * @param pDevIns The device instance.
4771 * @param pvUser A user argument (ignored).
4772 * @param GCPhysAddr The physical address being written to. (This is within our MMIO memory range.)
4773 * @param pv Where to put the data we read.
4774 * @param cb The size of the read.
4775 */
4776PDMBOTHCBDECL(int) ohciRead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
4777{
4778 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
4779
4780 /*
4781 * Validate the access.
4782 */
4783 if (cb != sizeof(uint32_t))
4784 {
4785 Log2(("ohciRead: Bad read size!!! GCPhysAddr=%RGp cb=%d\n", GCPhysAddr, cb));
4786 return VINF_IOM_MMIO_UNUSED_FF; /* No idea what really would happen... */
4787 }
4788 if (GCPhysAddr & 0x3)
4789 {
4790 Log2(("ohciRead: Unaligned read!!! GCPhysAddr=%RGp cb=%d\n", GCPhysAddr, cb));
4791 return VINF_IOM_MMIO_UNUSED_FF;
4792 }
4793
4794 /*
4795 * Validate the register and call the read operator.
4796 */
4797 int rc;
4798 const uint32_t iReg = (GCPhysAddr - pOhci->MMIOBase) >> 2;
4799 if (iReg < RT_ELEMENTS(g_aOpRegs))
4800 {
4801 const OHCIOPREG *pReg = &g_aOpRegs[iReg];
4802 rc = pReg->pfnRead(pOhci, iReg, (uint32_t *)pv);
4803 }
4804 else
4805 {
4806 Log(("ohci: Trying to read register %u/%u!!!\n", iReg, RT_ELEMENTS(g_aOpRegs)));
4807 rc = VINF_IOM_MMIO_UNUSED_FF;
4808 }
4809 return rc;
4810}
4811
4812
4813/**
4814 * Write to a MMIO register.
4815 *
4816 * We only accept 32-bit writes that are 32-bit aligned.
4817 *
4818 * @returns VBox status code suitable for scheduling.
4819 * @param pDevIns The device instance.
4820 * @param pvUser A user argument (ignored).
4821 * @param GCPhysAddr The physical address being written to. (This is within our MMIO memory range.)
4822 * @param pv Pointer to the data being written.
4823 * @param cb The size of the data being written.
4824 */
4825PDMBOTHCBDECL(int) ohciWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS GCPhysAddr, void const *pv, unsigned cb)
4826{
4827 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
4828
4829 /*
4830 * Validate the access.
4831 */
4832 if (cb != sizeof(uint32_t))
4833 {
4834 Log2(("ohciWrite: Bad write size!!! GCPhysAddr=%RGp cb=%d\n", GCPhysAddr, cb));
4835 return VINF_SUCCESS;
4836 }
4837 if (GCPhysAddr & 0x3)
4838 {
4839 Log2(("ohciWrite: Unaligned write!!! GCPhysAddr=%RGp cb=%d\n", GCPhysAddr, cb));
4840 return VINF_SUCCESS;
4841 }
4842
4843 /*
4844 * Validate the register and call the read operator.
4845 */
4846 int rc;
4847 const uint32_t iReg = (GCPhysAddr - pOhci->MMIOBase) >> 2;
4848 if (iReg < RT_ELEMENTS(g_aOpRegs))
4849 {
4850 const OHCIOPREG *pReg = &g_aOpRegs[iReg];
4851 rc = pReg->pfnWrite(pOhci, iReg, *(uint32_t *)pv);
4852 }
4853 else
4854 {
4855 Log(("ohci: Trying to write to register %u/%u!!!\n", iReg, RT_ELEMENTS(g_aOpRegs)));
4856 rc = VINF_SUCCESS;
4857 }
4858 return rc;
4859}
4860
4861#ifdef IN_RING3
4862
4863static DECLCALLBACK(int)
4864ohciR3Map(PPCIDEVICE pPciDev, int iRegion, RTGCPHYS GCPhysAddress, uint32_t cb, PCIADDRESSSPACE enmType)
4865{
4866 POHCI pOhci = (POHCI)pPciDev;
4867 int rc = PDMDevHlpMMIORegister(pOhci->CTX_SUFF(pDevIns),
4868 GCPhysAddress,
4869 cb,
4870 NULL,
4871 ohciWrite,
4872 ohciRead,
4873 NULL,
4874 "USB OHCI");
4875 if (RT_FAILURE(rc))
4876 return rc;
4877
4878# if 1 /* this enabled / disabled GC/R0 stuff */
4879 rc = PDMDevHlpMMIORegisterRC(pOhci->CTX_SUFF(pDevIns),
4880 GCPhysAddress,
4881 cb,
4882 0,
4883 "ohciWrite",
4884 "ohciRead",
4885 NULL);
4886 if (RT_FAILURE(rc))
4887 return rc;
4888
4889 rc = PDMDevHlpMMIORegisterR0(pOhci->CTX_SUFF(pDevIns),
4890 GCPhysAddress,
4891 cb,
4892 0,
4893 "ohciWrite",
4894 "ohciRead",
4895 NULL);
4896 if (RT_FAILURE(rc))
4897 return rc;
4898
4899# endif
4900
4901 pOhci->MMIOBase = GCPhysAddress;
4902 return VINF_SUCCESS;
4903}
4904
4905/**
4906 * Prepares for state saving.
4907 * All URBs needs to be canceled.
4908 *
4909 * @returns VBox status code.
4910 * @param pDevIns The device instance.
4911 * @param pSSM The handle to save the state to.
4912 */
4913static DECLCALLBACK(int) ohciR3SavePrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4914{
4915 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
4916 POHCIROOTHUB pRh = &pOhci->RootHub;
4917 unsigned i;
4918 LogFlow(("ohciR3SavePrep: \n"));
4919
4920 /*
4921 * Detach all proxied devices.
4922 */
4923 /** @todo we a) can't tell which are proxied, and b) this won't work well when continuing after saving! */
4924 for (i = 0; i < RT_ELEMENTS(pRh->aPorts); i++)
4925 {
4926 PVUSBIDEVICE pDev = pRh->aPorts[i].pDev;
4927 if (pDev)
4928 {
4929 VUSBIRhDetachDevice(pRh->pIRhConn, pDev);
4930 /*
4931 * Save the device pointers here so we can reattach them afterwards.
4932 * This will work fine even if the save fails since the Done handler is
4933 * called unconditionally if the Prep handler was called.
4934 */
4935 pRh->aPorts[i].pDev = pDev;
4936 }
4937 }
4938
4939 /*
4940 * Kill old load data which might be hanging around.
4941 */
4942 if (pOhci->pLoad)
4943 {
4944 TMR3TimerDestroy(pOhci->pLoad->pTimer);
4945 MMR3HeapFree(pOhci->pLoad);
4946 pOhci->pLoad = NULL;
4947 }
4948 return VINF_SUCCESS;
4949}
4950
4951/**
4952 * Saves the state of the OHCI device.
4953 *
4954 * @returns VBox status code.
4955 * @param pDevIns The device instance.
4956 * @param pSSM The handle to save the state to.
4957 */
4958static DECLCALLBACK(int) ohciR3SaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4959{
4960 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
4961 LogFlow(("ohciR3SaveExec: \n"));
4962
4963 int rc = SSMR3PutStructEx(pSSM, pOhci, sizeof(*pOhci), 0 /*fFlags*/, &g_aOhciFields[0], NULL);
4964 if (RT_SUCCESS(rc))
4965 rc = TMR3TimerSave(pOhci->CTX_SUFF(pEndOfFrameTimer), pSSM);
4966 return rc;
4967}
4968
4969
4970/**
4971 * Done state save operation.
4972 *
4973 * @returns VBox load code.
4974 * @param pDevIns Device instance of the device which registered the data unit.
4975 * @param pSSM SSM operation handle.
4976 */
4977static DECLCALLBACK(int) ohciR3SaveDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4978{
4979 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
4980 POHCIROOTHUB pRh = &pOhci->RootHub;
4981 OHCIROOTHUB Rh;
4982 unsigned i;
4983 LogFlow(("ohciR3SavePrep: \n"));
4984
4985 /*
4986 * NULL the dev pointers.
4987 */
4988 Rh = *pRh;
4989 for (i = 0; i < RT_ELEMENTS(pRh->aPorts); i++)
4990 pRh->aPorts[i].pDev = NULL;
4991
4992 /*
4993 * Attach the devices.
4994 */
4995 for (i = 0; i < RT_ELEMENTS(pRh->aPorts); i++)
4996 {
4997 PVUSBIDEVICE pDev = Rh.aPorts[i].pDev;
4998 if (pDev)
4999 VUSBIRhAttachDevice(pRh->pIRhConn, pDev);
5000 }
5001
5002 return VINF_SUCCESS;
5003}
5004
5005
5006/**
5007 * Prepare loading the state of the OHCI device.
5008 * This must detach the devices currently attached and save
5009 * the up for reconnect after the state load have been completed
5010 *
5011 * @returns VBox status code.
5012 * @param pDevIns The device instance.
5013 * @param pSSM The handle to the saved state.
5014 * @param u32Version The data unit version number.
5015 */
5016static DECLCALLBACK(int) ohciR3LoadPrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
5017{
5018 int rc = VINF_SUCCESS;
5019 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
5020 LogFlow(("ohciR3LoadPrep:\n"));
5021 if (!pOhci->pLoad)
5022 {
5023 POHCIROOTHUB pRh = &pOhci->RootHub;
5024 OHCILOAD Load;
5025 unsigned i;
5026
5027 /*
5028 * Detach all devices which are present in this session. Save them in the load
5029 * structure so we can reattach them after restoring the guest.
5030 */
5031 Load.pTimer = NULL;
5032 Load.cDevs = 0;
5033 for (i = 0; i < RT_ELEMENTS(pRh->aPorts); i++)
5034 {
5035 PVUSBIDEVICE pDev = pRh->aPorts[i].pDev;
5036 if (pDev)
5037 {
5038 Load.apDevs[Load.cDevs++] = pDev;
5039 VUSBIRhDetachDevice(pRh->pIRhConn, pDev);
5040 Assert(!pRh->aPorts[i].pDev);
5041 }
5042 }
5043
5044 /*
5045 * Any devices to reattach, if so duplicate the Load struct.
5046 */
5047 if (Load.cDevs)
5048 {
5049 pOhci->pLoad = (POHCILOAD)PDMDevHlpMMHeapAlloc(pDevIns, sizeof(Load));
5050 if (!pOhci->pLoad)
5051 return VERR_NO_MEMORY;
5052 *pOhci->pLoad = Load;
5053 }
5054 }
5055 /* else: we ASSUME no device can be attached or detach in the period
5056 * between a state load and the pLoad stuff is processed. */
5057 return rc;
5058}
5059
5060
5061/**
5062 * Loads the state of the OHCI device.
5063 *
5064 * @returns VBox status code.
5065 * @param pDevIns The device instance.
5066 * @param pSSM The handle to the saved state.
5067 * @param uVersion The data unit version number.
5068 * @param uPass The data pass.
5069 */
5070static DECLCALLBACK(int) ohciR3LoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
5071{
5072 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
5073 int rc;
5074 LogFlow(("ohciR3LoadExec:\n"));
5075 Assert(uPass == SSM_PASS_FINAL); NOREF(uPass);
5076
5077 if (uVersion == OHCI_SAVED_STATE_VERSION)
5078 {
5079 rc = SSMR3GetStructEx(pSSM, pOhci, sizeof(*pOhci), 0 /*fFlags*/, &g_aOhciFields[0], NULL);
5080 if (RT_FAILURE(rc))
5081 return rc;
5082 }
5083 else if (uVersion == OHCI_SAVED_STATE_VERSION_MEM_HELL)
5084 {
5085 static SSMFIELD const s_aOhciFields22[] =
5086 {
5087 SSMFIELD_ENTRY_OLD( PciDev.config, 256), /* DevPCI restores this. */
5088 SSMFIELD_ENTRY_OLD( PciDev.Int, 224),
5089 SSMFIELD_ENTRY_OLD( PciDev.devfn, 4),
5090 SSMFIELD_ENTRY_OLD( PciDev.Alignment0, 4),
5091 SSMFIELD_ENTRY_OLD_HCPTR( PciDev.name),
5092 SSMFIELD_ENTRY_OLD_HCPTR( PciDev.pDevIns),
5093 SSMFIELD_ENTRY_OLD_HCPTR( pDevInsR3),
5094 SSMFIELD_ENTRY_OLD_HCPTR( pEndOfFrameTimerR3),
5095 SSMFIELD_ENTRY_OLD_HCPTR( pDevInsR0),
5096 SSMFIELD_ENTRY_OLD_HCPTR( pEndOfFrameTimerR0),
5097 SSMFIELD_ENTRY_OLD_RCPTR( pDevInsRC),
5098 SSMFIELD_ENTRY_OLD_RCPTR( pEndOfFrameTimerRC),
5099 SSMFIELD_ENTRY( OHCI, SofTime),
5100 SSMFIELD_ENTRY_CUSTOM( dpic+fno, RT_OFFSETOF(OHCI, SofTime) + RT_SIZEOFMEMB(OHCI, SofTime), 4),
5101 SSMFIELD_ENTRY_OLD( MMIOBase, 4), /* DevPCI implicitly restores this. */
5102 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.pIBase),
5103 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.pIRhConn),
5104 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.pIDev),
5105 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.IBase.pfnQueryInterface),
5106 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.IRhPort.pfnGetAvailablePorts),
5107 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.IRhPort.pfnGetUSBVersions),
5108 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.IRhPort.pfnAttach),
5109 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.IRhPort.pfnDetach),
5110 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.IRhPort.pfnReset),
5111 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.IRhPort.pfnXferCompletion),
5112 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.IRhPort.pfnXferError),
5113 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.IRhPort.Alignment),
5114 SSMFIELD_ENTRY_OLD( RootHub.Led, 16), /* No device restored. */
5115 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.ILeds.pfnQueryStatusLed),
5116 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.pLedsConnector),
5117 SSMFIELD_ENTRY( OHCI, RootHub.status),
5118 SSMFIELD_ENTRY( OHCI, RootHub.desc_a),
5119 SSMFIELD_ENTRY( OHCI, RootHub.desc_b),
5120 SSMFIELD_ENTRY_OLD_PAD_HC64( RootHub.Alignment0, 4),
5121 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[0].fReg),
5122 SSMFIELD_ENTRY_OLD_PAD_HC64( RootHub.aPorts[0].Alignment0, 4),
5123 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.aPorts[0].pDev),
5124 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[1].fReg),
5125 SSMFIELD_ENTRY_OLD_PAD_HC64( RootHub.aPorts[1].Alignment0, 4),
5126 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.aPorts[1].pDev),
5127 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[2].fReg),
5128 SSMFIELD_ENTRY_OLD_PAD_HC64( RootHub.aPorts[2].Alignment0, 4),
5129 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.aPorts[2].pDev),
5130 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[3].fReg),
5131 SSMFIELD_ENTRY_OLD_PAD_HC64( RootHub.aPorts[3].Alignment0, 4),
5132 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.aPorts[3].pDev),
5133 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[4].fReg),
5134 SSMFIELD_ENTRY_OLD_PAD_HC64( RootHub.aPorts[4].Alignment0, 4),
5135 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.aPorts[4].pDev),
5136 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[5].fReg),
5137 SSMFIELD_ENTRY_OLD_PAD_HC64( RootHub.aPorts[5].Alignment0, 4),
5138 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.aPorts[5].pDev),
5139 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[6].fReg),
5140 SSMFIELD_ENTRY_OLD_PAD_HC64( RootHub.aPorts[6].Alignment0, 4),
5141 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.aPorts[6].pDev),
5142 SSMFIELD_ENTRY( OHCI, RootHub.aPorts[7].fReg),
5143 SSMFIELD_ENTRY_OLD_PAD_HC64( RootHub.aPorts[7].Alignment0, 4),
5144 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.aPorts[7].pDev),
5145 SSMFIELD_ENTRY_OLD_HCPTR( RootHub.pOhci),
5146 SSMFIELD_ENTRY( OHCI, ctl),
5147 SSMFIELD_ENTRY( OHCI, status),
5148 SSMFIELD_ENTRY( OHCI, intr_status),
5149 SSMFIELD_ENTRY( OHCI, intr),
5150 SSMFIELD_ENTRY( OHCI, hcca),
5151 SSMFIELD_ENTRY( OHCI, per_cur),
5152 SSMFIELD_ENTRY( OHCI, ctrl_cur),
5153 SSMFIELD_ENTRY( OHCI, ctrl_head),
5154 SSMFIELD_ENTRY( OHCI, bulk_cur),
5155 SSMFIELD_ENTRY( OHCI, bulk_head),
5156 SSMFIELD_ENTRY( OHCI, done),
5157 SSMFIELD_ENTRY_CUSTOM( fsmps+fit+fi+frt, RT_OFFSETOF(OHCI, done) + RT_SIZEOFMEMB(OHCI, done), 4),
5158 SSMFIELD_ENTRY( OHCI, HcFmNumber),
5159 SSMFIELD_ENTRY( OHCI, pstart),
5160 SSMFIELD_ENTRY_OLD( cTicksPerFrame, 8), /* done by the constructor */
5161 SSMFIELD_ENTRY_OLD( cTicksPerUsbTick, 8), /* ditto */
5162 SSMFIELD_ENTRY_OLD( cInFlight, 4), /* no in-flight stuff when saving. */
5163 SSMFIELD_ENTRY_OLD( Alignment1, 4),
5164 SSMFIELD_ENTRY_OLD( aInFlight, 257 * 8),
5165 SSMFIELD_ENTRY_OLD_PAD_HC64( aInFlight, 257 * 8),
5166 SSMFIELD_ENTRY_OLD( cInDoneQueue, 4), /* strict builds only, so don't bother. */
5167 SSMFIELD_ENTRY_OLD( aInDoneQueue, 4*64),
5168 SSMFIELD_ENTRY_OLD( u32FmDoneQueueTail, 4), /* logging only */
5169 SSMFIELD_ENTRY_OLD_PAD_HC32( Alignment2, 4),
5170 SSMFIELD_ENTRY_OLD_HCPTR( pLoad),
5171 SSMFIELD_ENTRY_OLD( StatCanceledIsocUrbs, 8),
5172 SSMFIELD_ENTRY_OLD( StatCanceledGenUrbs, 8),
5173 SSMFIELD_ENTRY_OLD( StatDroppedUrbs, 8),
5174 SSMFIELD_ENTRY_OLD( StatTimer, 32),
5175 SSMFIELD_ENTRY_TERM()
5176 };
5177
5178 /* deserialize the struct */
5179 rc = SSMR3GetStructEx(pSSM, pOhci, sizeof(*pOhci), SSMSTRUCT_FLAGS_NO_MARKERS /*fFlags*/, &s_aOhciFields22[0], NULL);
5180 if (RT_FAILURE(rc))
5181 return rc;
5182
5183 /* check delimiter */
5184 uint32_t u32;
5185 rc = SSMR3GetU32(pSSM, &u32);
5186 if (RT_FAILURE(rc))
5187 return rc;
5188 AssertMsgReturn(u32 == ~0U, ("%#x\n", u32), VERR_SSM_DATA_UNIT_FORMAT_CHANGED);
5189 }
5190 else
5191 AssertMsgFailedReturn(("%d\n", uVersion), VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION);
5192
5193 /*
5194 * Finally restore the timer.
5195 */
5196 return TMR3TimerLoad(pOhci->pEndOfFrameTimerR3, pSSM);
5197}
5198
5199
5200/**
5201 * Done state load operation.
5202 *
5203 * @returns VBox load code.
5204 * @param pDevIns Device instance of the device which registered the data unit.
5205 * @param pSSM SSM operation handle.
5206 */
5207static DECLCALLBACK(int) ohciR3LoadDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
5208{
5209 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
5210 LogFlow(("ohciR3LoadDone:\n"));
5211
5212 /*
5213 * Start a timer if we've got devices to reattach
5214 */
5215 if (pOhci->pLoad)
5216 {
5217 int rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, ohciR3LoadReattachDevices, pOhci,
5218 TMTIMER_FLAGS_NO_CRIT_SECT, "OHCI reattach devices on load",
5219 &pOhci->pLoad->pTimer);
5220 if (RT_SUCCESS(rc))
5221 rc = TMTimerSetMillies(pOhci->pLoad->pTimer, 250);
5222 return rc;
5223 }
5224
5225 return VINF_SUCCESS;
5226}
5227
5228
5229/**
5230 * Reattaches devices after a saved state load.
5231 */
5232static DECLCALLBACK(void) ohciR3LoadReattachDevices(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
5233{
5234 POHCI pOhci = (POHCI)pvUser;
5235 POHCILOAD pLoad = pOhci->pLoad;
5236 POHCIROOTHUB pRh = &pOhci->RootHub;
5237 LogFlow(("ohciR3LoadReattachDevices:\n"));
5238
5239 /*
5240 * Reattach devices.
5241 */
5242 for (unsigned i = 0; i < pLoad->cDevs; i++)
5243 VUSBIRhAttachDevice(pRh->pIRhConn, pLoad->apDevs[i]);
5244
5245 /*
5246 * Cleanup.
5247 */
5248 TMR3TimerDestroy(pTimer);
5249 MMR3HeapFree(pLoad);
5250 pOhci->pLoad = NULL;
5251}
5252
5253
5254/**
5255 * Reset notification.
5256 *
5257 * @returns VBox status.
5258 * @param pDevIns The device instance data.
5259 */
5260static DECLCALLBACK(void) ohciR3Reset(PPDMDEVINS pDevIns)
5261{
5262 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
5263 LogFlow(("ohciR3Reset:\n"));
5264
5265 /*
5266 * There is no distinction between cold boot, warm reboot and software reboots,
5267 * all of these are treated as cold boots. We are also doing the initialization
5268 * job of a BIOS or SMM driver.
5269 *
5270 * Important: Don't confuse UsbReset with hardware reset. Hardware reset is
5271 * just one way of getting into the UsbReset state.
5272 */
5273 ohciBusStop(pOhci);
5274 ohciDoReset(pOhci, OHCI_USB_RESET, true /* reset devices */);
5275}
5276
5277
5278/**
5279 * Info handler, device version. Dumps OHCI control registers.
5280 *
5281 * @param pDevIns Device instance which registered the info.
5282 * @param pHlp Callback functions for doing output.
5283 * @param pszArgs Argument string. Optional and specific to the handler.
5284 */
5285static DECLCALLBACK(void) ohciR3InfoRegs(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
5286{
5287 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
5288 uint32_t val, ctl, status;
5289
5290 /* Control register */
5291 ctl = pOhci->ctl;
5292 pHlp->pfnPrintf(pHlp, "HcControl: %08x - CBSR=%d PLE=%d IE=%d CLE=%d BLE=%d HCFS=%#x IR=%d RWC=%d RWE=%d\n",
5293 ctl, ctl & 3, (ctl >> 2) & 1, (ctl >> 3) & 1, (ctl >> 4) & 1, (ctl >> 5) & 1, (ctl >> 6) & 3, (ctl >> 8) & 1,
5294 (ctl >> 9) & 1, (ctl >> 10) & 1);
5295
5296 /* Command status register */
5297 status = pOhci->status;
5298 pHlp->pfnPrintf(pHlp, "HcCommandStatus: %08x - HCR=%d CLF=%d BLF=%d OCR=%d SOC=%d\n",
5299 status, status & 1, (status >> 1) & 1, (status >> 2) & 1, (status >> 3) & 1, (status >> 16) & 3);
5300
5301 /* Interrupt status register */
5302 val = pOhci->intr_status;
5303 pHlp->pfnPrintf(pHlp, "HcInterruptStatus: %08x - SO=%d WDH=%d SF=%d RD=%d UE=%d FNO=%d RHSC=%d OC=%d\n",
5304 val, val & 1, (val >> 1) & 1, (val >> 2) & 1, (val >> 3) & 1, (val >> 4) & 1, (val >> 5) & 1,
5305 (val >> 6) & 1, (val >> 30) & 1);
5306
5307 /* Interrupt enable register */
5308 val = pOhci->intr;
5309 pHlp->pfnPrintf(pHlp, "HcInterruptEnable: %08x - SO=%d WDH=%d SF=%d RD=%d UE=%d FNO=%d RHSC=%d OC=%d MIE=%d\n",
5310 val, val & 1, (val >> 1) & 1, (val >> 2) & 1, (val >> 3) & 1, (val >> 4) & 1, (val >> 5) & 1,
5311 (val >> 6) & 1, (val >> 30) & 1, (val >> 31) & 1);
5312
5313 /* HCCA address register */
5314 pHlp->pfnPrintf(pHlp, "HcHCCA: %08x\n", pOhci->hcca);
5315
5316 /* Current periodic ED register */
5317 pHlp->pfnPrintf(pHlp, "HcPeriodCurrentED: %08x\n", pOhci->per_cur);
5318
5319 /* Control ED registers */
5320 pHlp->pfnPrintf(pHlp, "HcControlHeadED: %08x\n", pOhci->ctrl_head);
5321 pHlp->pfnPrintf(pHlp, "HcControlCurrentED: %08x\n", pOhci->ctrl_cur);
5322
5323 /* Bulk ED registers */
5324 pHlp->pfnPrintf(pHlp, "HcBulkHeadED: %08x\n", pOhci->bulk_head);
5325 pHlp->pfnPrintf(pHlp, "HcBulkCurrentED: %08x\n", pOhci->bulk_cur);
5326
5327 /* Done head register */
5328 pHlp->pfnPrintf(pHlp, "HcDoneHead: %08x\n", pOhci->done);
5329
5330 pHlp->pfnPrintf(pHlp, "\n");
5331}
5332
5333
5334/**
5335 * Relocate device instance data.
5336 *
5337 * @returns VBox status.
5338 * @param pDevIns The device instance data.
5339 * @param offDelta The relocation delta.
5340 */
5341static DECLCALLBACK(void) ohciR3Relocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
5342{
5343 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
5344 pOhci->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
5345 pOhci->pEndOfFrameTimerRC = TMTimerRCPtr(pOhci->pEndOfFrameTimerR3);
5346}
5347
5348
5349/**
5350 * Destruct a device instance.
5351 *
5352 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
5353 * resources can be freed correctly.
5354 *
5355 * @returns VBox status.
5356 * @param pDevIns The device instance data.
5357 */
5358static DECLCALLBACK(int) ohciR3Destruct(PPDMDEVINS pDevIns)
5359{
5360 PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
5361
5362 /*
5363 * Tear down the per endpoint in-flight tracking...
5364 */
5365
5366 return VINF_SUCCESS;
5367}
5368
5369
5370/**
5371 * @interface_method_impl{PDMDEVREG,pfnConstruct,OHCI constructor}
5372 */
5373static DECLCALLBACK(int) ohciR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
5374{
5375 POHCI pOhci = PDMINS_2_DATA(pDevIns, POHCI);
5376 int rc;
5377 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
5378
5379 /*
5380 * Read configuration. No configuration keys are currently supported.
5381 */
5382 if (!CFGMR3AreValuesValid(pCfg, "\0"))
5383 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
5384 N_("Configuration error: Unknown config key"));
5385
5386 /*
5387 * Init instance data.
5388 */
5389 pOhci->pDevInsR3 = pDevIns;
5390 pOhci->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
5391 pOhci->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
5392
5393 PCIDevSetVendorId (&pOhci->PciDev, 0x106b);
5394 PCIDevSetDeviceId (&pOhci->PciDev, 0x003f);
5395 PCIDevSetClassProg (&pOhci->PciDev, 0x10); /* OHCI */
5396 PCIDevSetClassSub (&pOhci->PciDev, 0x03);
5397 PCIDevSetClassBase (&pOhci->PciDev, 0x0c);
5398 PCIDevSetInterruptPin (&pOhci->PciDev, 0x01);
5399#ifdef VBOX_WITH_MSI_DEVICES
5400 PCIDevSetStatus (&pOhci->PciDev, VBOX_PCI_STATUS_CAP_LIST);
5401 PCIDevSetCapabilityList(&pOhci->PciDev, 0x80);
5402#endif
5403
5404 pOhci->RootHub.pOhci = pOhci;
5405 pOhci->RootHub.IBase.pfnQueryInterface = ohciRhQueryInterface;
5406 pOhci->RootHub.IRhPort.pfnGetAvailablePorts = ohciRhGetAvailablePorts;
5407 pOhci->RootHub.IRhPort.pfnGetUSBVersions = ohciRhGetUSBVersions;
5408 pOhci->RootHub.IRhPort.pfnAttach = ohciRhAttach;
5409 pOhci->RootHub.IRhPort.pfnDetach = ohciRhDetach;
5410 pOhci->RootHub.IRhPort.pfnReset = ohciRhReset;
5411 pOhci->RootHub.IRhPort.pfnXferCompletion = ohciRhXferCompletion;
5412 pOhci->RootHub.IRhPort.pfnXferError = ohciRhXferError;
5413
5414 /* USB LED */
5415 pOhci->RootHub.Led.u32Magic = PDMLED_MAGIC;
5416 pOhci->RootHub.ILeds.pfnQueryStatusLed = ohciRhQueryStatusLed;
5417
5418 /*
5419 * Register PCI device and I/O region.
5420 */
5421 rc = PDMDevHlpPCIRegister(pDevIns, &pOhci->PciDev);
5422 if (RT_FAILURE(rc))
5423 return rc;
5424
5425#ifdef VBOX_WITH_MSI_DEVICES
5426 PDMMSIREG aMsiReg;
5427 RT_ZERO(aMsiReg);
5428 aMsiReg.cMsiVectors = 1;
5429 aMsiReg.iMsiCapOffset = 0x80;
5430 aMsiReg.iMsiNextOffset = 0x0;
5431 rc = PDMDevHlpPCIRegisterMsi(pDevIns, &aMsiReg);
5432 if (RT_FAILURE (rc))
5433 {
5434 PCIDevSetCapabilityList(&pOhci->PciDev, 0x0);
5435 /* That's OK, we can work without MSI */
5436 }
5437#endif
5438
5439 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 0, 4096, PCI_ADDRESS_SPACE_MEM, ohciR3Map);
5440 if (RT_FAILURE(rc))
5441 return rc;
5442
5443 /*
5444 * Create the end-of-frame timer.
5445 */
5446 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, ohciFrameBoundaryTimer, pOhci,
5447 TMTIMER_FLAGS_DEFAULT_CRIT_SECT, "USB Frame Timer",
5448 &pOhci->pEndOfFrameTimerR3);
5449 if (RT_FAILURE(rc))
5450 return rc;
5451 pOhci->pEndOfFrameTimerR0 = TMTimerR0Ptr(pOhci->pEndOfFrameTimerR3);
5452 pOhci->pEndOfFrameTimerRC = TMTimerRCPtr(pOhci->pEndOfFrameTimerR3);
5453
5454 /*
5455 * Register the saved state data unit.
5456 */
5457 rc = PDMDevHlpSSMRegisterEx(pDevIns, OHCI_SAVED_STATE_VERSION, sizeof(*pOhci), NULL,
5458 NULL, NULL, NULL,
5459 ohciR3SavePrep, ohciR3SaveExec, ohciR3SaveDone,
5460 ohciR3LoadPrep, ohciR3LoadExec, ohciR3LoadDone);
5461 if (RT_FAILURE(rc))
5462 return rc;
5463
5464 /*
5465 * Attach to the VBox USB RootHub Driver on LUN #0.
5466 */
5467 rc = PDMDevHlpDriverAttach(pDevIns, 0, &pOhci->RootHub.IBase, &pOhci->RootHub.pIBase, "RootHub");
5468 if (RT_FAILURE(rc))
5469 {
5470 AssertMsgFailed(("Configuration error: No roothub driver attached to LUN #0!\n"));
5471 return rc;
5472 }
5473 pOhci->RootHub.pIRhConn = PDMIBASE_QUERY_INTERFACE(pOhci->RootHub.pIBase, VUSBIROOTHUBCONNECTOR);
5474 AssertMsgReturn(pOhci->RootHub.pIRhConn,
5475 ("Configuration error: The driver doesn't provide the VUSBIROOTHUBCONNECTOR interface!\n"),
5476 VERR_PDM_MISSING_INTERFACE);
5477 pOhci->RootHub.pIDev = PDMIBASE_QUERY_INTERFACE(pOhci->RootHub.pIBase, VUSBIDEVICE);
5478 AssertMsgReturn(pOhci->RootHub.pIDev,
5479 ("Configuration error: The driver doesn't provide the VUSBIDEVICE interface!\n"),
5480 VERR_PDM_MISSING_INTERFACE);
5481
5482 /*
5483 * Attach status driver (optional).
5484 */
5485 PPDMIBASE pBase;
5486 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pOhci->RootHub.IBase, &pBase, "Status Port");
5487 if (RT_SUCCESS(rc))
5488 pOhci->RootHub.pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS);
5489 else if (rc != VERR_PDM_NO_ATTACHED_DRIVER)
5490 {
5491 AssertMsgFailed(("Failed to attach to status driver. rc=%Rrc\n", rc));
5492 return rc;
5493 }
5494
5495 /*
5496 * Calculate the timer intervals.
5497 * This assumes that the VM timer doesn't change frequency during the run.
5498 */
5499 pOhci->u64TimerHz = TMTimerGetFreq(pOhci->CTX_SUFF(pEndOfFrameTimer));
5500 ohciCalcTimerIntervals(pOhci, OHCI_DEFAULT_TIMER_FREQ);
5501 Log(("ohci: cTicksPerFrame=%RU64 cTicksPerUsbTick=%RU64\n",
5502 pOhci->cTicksPerFrame, pOhci->cTicksPerUsbTick));
5503
5504 /*
5505 * Do a hardware reset.
5506 */
5507 ohciDoReset(pOhci, OHCI_USB_RESET, false /* don't reset devices */);
5508
5509#ifdef VBOX_WITH_STATISTICS
5510 /*
5511 * Register statistics.
5512 */
5513 PDMDevHlpSTAMRegister(pDevIns, &pOhci->StatCanceledIsocUrbs, STAMTYPE_COUNTER, "/Devices/OHCI/CanceledIsocUrbs", STAMUNIT_OCCURENCES, "Detected canceled isochronous URBs.");
5514 PDMDevHlpSTAMRegister(pDevIns, &pOhci->StatCanceledGenUrbs, STAMTYPE_COUNTER, "/Devices/OHCI/CanceledGenUrbs", STAMUNIT_OCCURENCES, "Detected canceled general URBs.");
5515 PDMDevHlpSTAMRegister(pDevIns, &pOhci->StatDroppedUrbs, STAMTYPE_COUNTER, "/Devices/OHCI/DroppedUrbs", STAMUNIT_OCCURENCES, "Dropped URBs (endpoint halted, or URB canceled).");
5516 PDMDevHlpSTAMRegister(pDevIns, &pOhci->StatTimer, STAMTYPE_PROFILE, "/Devices/OHCI/Timer", STAMUNIT_TICKS_PER_CALL, "Profiling ohciFrameBoundaryTimer.");
5517#endif
5518
5519 /*
5520 * Register debugger info callbacks.
5521 */
5522 PDMDevHlpDBGFInfoRegister(pDevIns, "ohci", "OHCI control registers.", ohciR3InfoRegs);
5523
5524#if 0/*def DEBUG_bird*/
5525// g_fLogInterruptEPs = true;
5526 g_fLogControlEPs = true;
5527 g_fLogBulkEPs = true;
5528#endif
5529
5530 return VINF_SUCCESS;
5531}
5532
5533
5534const PDMDEVREG g_DeviceOHCI =
5535{
5536 /* u32version */
5537 PDM_DEVREG_VERSION,
5538 /* szName */
5539 "usb-ohci",
5540 /* szRCMod */
5541 "VBoxDDGC.gc",
5542 /* szR0Mod */
5543 "VBoxDDR0.r0",
5544 /* pszDescription */
5545 "OHCI USB controller.\n",
5546 /* fFlags */
5547 PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0,
5548 /* fClass */
5549 PDM_DEVREG_CLASS_BUS_USB,
5550 /* cMaxInstances */
5551 ~0,
5552 /* cbInstance */
5553 sizeof(OHCI),
5554 /* pfnConstruct */
5555 ohciR3Construct,
5556 /* pfnDestruct */
5557 ohciR3Destruct,
5558 /* pfnRelocate */
5559 ohciR3Relocate,
5560 /* pfnIOCtl */
5561 NULL,
5562 /* pfnPowerOn */
5563 NULL,
5564 /* pfnReset */
5565 ohciR3Reset,
5566 /* pfnSuspend */
5567 NULL,
5568 /* pfnResume */
5569 NULL,
5570 /* pfnAttach */
5571 NULL,
5572 /* pfnDetach */
5573 NULL,
5574 /* pfnQueryInterface */
5575 NULL,
5576 /* pfnInitComplete */
5577 NULL,
5578 /* pfnPowerOff */
5579 NULL,
5580 /* pfnSoftReset */
5581 NULL,
5582 /* u32VersionEnd */
5583 PDM_DEVREG_VERSION
5584};
5585
5586#endif /* IN_RING3 */
5587#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette