Update contrib.
1 // Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of the License "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
20 @return The name of the driver
24 const TDesC& RUsbInterface::Name()
26 _LIT(KDriverName,"USBDI");
33 @return The version number of the driver
37 TVersion RUsbInterface::VersionRequired()
39 const TInt KMajorVersionNumber=1;
40 const TInt KMinorVersionNumber=0;
41 const TInt KBuildVersionNumber=KE32BuildVersionNumber;
42 return TVersion(KMajorVersionNumber,KMinorVersionNumber,KBuildVersionNumber);
47 #ifndef __KERNEL_MODE__
48 RUsbInterface::RUsbInterface()
49 : iHeadInterfaceDescriptor(NULL)
50 , iInterfaceDescriptorData(NULL)
51 , iTransferStrategy(NULL)
52 , iAlternateSetting(0)
57 Signals to the hub driver that this interface is idle and may be suspended.
58 As suspend operates at the device level, this will only trigger a removal of bus activity if all interfaces
59 associated with the device are marked as suspended.
61 @param [in] aResumeSignal The TRequestStatus that will be completed when the interface is resumed.
63 void RUsbInterface::PermitSuspendAndWaitForResume(TRequestStatus& aResumeSignal)
65 DoRequest(ESuspend, aResumeSignal);
69 Cancel the outstanding permission to suspend.
71 void RUsbInterface::CancelPermitSuspend()
73 DoCancel(ECancelSuspend);
77 Cancel the outstanding PermitSuspendAndWaitForResume request with KErrCancel
79 void RUsbInterface::CancelWaitForResume()
81 DoCancel(ECancelWaitForResume);
85 Request or clear the interface's remote wakeup flag. If any interface on the device
86 has this flag set, suspending the device will cause it to have remote wakeup capability
87 enabled. This function may only be called when the interface is active -- the device will
88 not be woken to change the status if it is currently suspended.
89 Note that clearing this flag will not prevent a device from using remote wakeup -- this
90 will happen only if all interfaces on the device do not require it.
91 By default the device will not have remote wakeup enabled.
93 @param aAllowed ETrue if remote wakeup should be permitted, EFalse if this interface does
96 @return KErrNotReady if two calls have been made in succession with the same parameter.
97 KErrUsbDeviceSuspended if the interface is currently marked as suspended.
99 TInt RUsbInterface::PermitRemoteWakeup(TBool aAllowed)
101 return DoControl(EPermitRemoteWakeup, (TAny*)aAllowed);
106 Select the specified alternate interface.
108 Asserts that all open pipes have been closed.
110 @param [in] aAlternateInterface The alternate interface to select.
112 @return KErrArgument if the specified alternate interface does not exist.
113 @return KErrOverflow if selecting this alternate interface would overcommit the bus' bandwidth.
115 TInt RUsbInterface::SelectAlternateInterface(TInt aAlternateInterface)
117 TInt err = DoControl(ESelectAlternateInterface, (TAny*)aAlternateInterface);
120 iAlternateSetting = aAlternateInterface;
125 TInt RUsbInterface::GetStringDescriptor(TDes8& aStringDescriptor, TUint8 aIndex, TUint16 aLangId)
128 params = aIndex | (aLangId << 16);
129 return DoControl(EGetStringDescriptor, &aStringDescriptor, ¶ms);
134 Performs an Endpoint 0 transfer.
136 void RUsbInterface::Ep0Transfer(TUsbTransferRequestDetails& aDetails, const TDesC8& aSend, TDes8& aRecv, TRequestStatus& aRequest)
138 aDetails.iSend = &aSend;
139 aDetails.iRecv = &aRecv;
140 DoRequest(EEp0Transfer, aRequest, (TAny*)&aDetails);
144 Cancel an Endpoint 0 transfer.
146 void RUsbInterface::CancelEP0Transfer()
148 DoCancel(ECancelEp0Transfer);
152 Establish a pipe between host and device. The RUsbPipe object returned is ready for use.
154 @param aPipe The pipe to connect to the remote endpoint. [out]
155 @param aEndpoint The endpoint on the remote device to connect to. [in]
156 @param aUseDMA In future implementations where DMA is supported this flag indicates DMA must be used. The Open attempt will fail if DMA cannot be offered on the pipe. [in]
158 @return KErrArgument if the specified endpoint does not exist.
159 @see SelectAlternateInterface
161 TInt RUsbInterface::OpenPipeForEndpoint(RUsbPipe& aPipe, TInt aEndpoint, TBool /*aUseDMA*/)
168 TInt err = GetEndpointDescriptor(iAlternateSetting, aEndpoint, aPipe.iHeadEndpointDescriptor);
169 // Allow KErrNotFound as audio endpoint descriptors are not valid endpoint descriptors
170 if ((err == KErrNone) || (err == KErrNotFound))
173 err = DoControl(EOpenPipe, &pipeHandle, reinterpret_cast<TAny*>(aEndpoint));
176 aPipe.iHandle = pipeHandle;
177 aPipe.iInterface = this;
187 TInt RUsbInterface::AllocateSharedChunk(RChunk& aChunk, TInt aSize, TInt& aOffset)
189 TInt chunkHandle = 0;
190 RUsbInterface::TChunkRequestDetails details;
191 details.iRequestSize = aSize;
192 details.iChunkHandle = &chunkHandle;
193 details.iOffset = &aOffset;
194 TInt err = DoControl(EAllocChunk, &details);
197 aChunk.SetHandle(chunkHandle);
203 Return the section of the USB Configuration Descriptor under this interface, including any alternate
206 Note: the supplied TUsbInterfaceDescriptor is owned by the caller, but any descriptor objects linked to it
207 remain the property of the RUsbInterface object. Memory leaks will result if the head pointer is not
208 cleaned up, but the pointed to objects should not be destroyed.
210 @param [out] aDescriptor The supplied TUsbInterfaceDescriptor object will be populated from the data retrieved from
211 the device. Note that the caller owns the head of the list, but not any children or peers.
213 @return System wide error code.
215 TInt RUsbInterface::GetInterfaceDescriptor(TUsbInterfaceDescriptor& aDescriptor)
217 if (!iHeadInterfaceDescriptor)
222 aDescriptor = *iHeadInterfaceDescriptor;
227 Find and return the section of the USB Configuration Descriptor under the supplied alternate interface.
229 Note: the supplied TUsbInterfaceDescriptor is owned by the caller, but any descriptor objects linked to it
230 remain the property of the RUsbInterface object. Memory leaks will result if the head pointer is not
231 cleaned up, but the pointed to objects should not be destroyed.
233 @param aAlternateInterface The alternate interface number to return the descriptor for. [in]
234 @param aDescriptor The supplied TUsbInterfaceDescriptor object will be populated from the data retrieved from
235 the device. Note that the caller owns the head of the list, but not any children or peers. [out]
237 @return KErrArgument if the specified alternate interface does not exist.
239 TInt RUsbInterface::GetAlternateInterfaceDescriptor(TInt aAlternateInterface, TUsbInterfaceDescriptor& aDescriptor)
241 if (!iHeadInterfaceDescriptor)
246 TUsbGenericDescriptor* descriptor = iHeadInterfaceDescriptor;
249 TUsbInterfaceDescriptor* interface = TUsbInterfaceDescriptor::Cast(descriptor);
252 if (interface->AlternateSetting() == aAlternateInterface)
254 aDescriptor = *interface;
258 // we must check any Interface Association Descriptors for
259 // Alternate Interface settings. The spec is abiguous on how these may be organised so we
260 // presume the worst and do a full search
261 TUsbInterfaceAssociationDescriptor* iad = TUsbInterfaceAssociationDescriptor::Cast(descriptor);
264 TUsbGenericDescriptor* assocDes = iad->iFirstChild;
267 interface = TUsbInterfaceDescriptor::Cast(assocDes);
270 if (interface->AlternateSetting() == aAlternateInterface)
272 aDescriptor = *interface;
276 assocDes = assocDes->iNextPeer;
279 descriptor = descriptor->iNextPeer;
286 Find and return the section of the USB Configuration Descriptor under the supplied endpoint.
288 Note: the supplied TUsbEndpointDescriptor is owned by the caller, but any descriptor objects linked to it
289 remain the property of the RUsbInterface object. Memory leaks will result if the head pointer is not
290 cleaned up, but the pointed to objects should not be destroyed.
292 @param aAlternateInterface The alternate interface number to return the descriptor for. [in]
293 @param aEndpoint The endpoint number to return the descriptor for. [in]
294 @param aDescriptor The supplied TUsbEndpointDescriptor object will be populated from the data retrieved from
295 the device. Note that the caller owns the head of the list, but not any children or peers. [out]
297 @return KErrArgument if the specified alternate interface does not exist, or KErrNotFound if the specified
298 endpoint cannot be found on the alternate interface.
300 TInt RUsbInterface::GetEndpointDescriptor(TInt aAlternateInterface, TInt aEndpoint, TUsbEndpointDescriptor& aDescriptor)
302 TUsbEndpointDescriptor* descriptor = &aDescriptor;
303 TInt err = GetEndpointDescriptor(aAlternateInterface, aEndpoint, descriptor);
304 if ((err == KErrNone) && descriptor)
306 aDescriptor = *descriptor;
311 TInt RUsbInterface::GetEndpointDescriptor(TInt aAlternateInterface, TInt aEndpoint, TUsbEndpointDescriptor*& aDescriptor)
315 TUsbInterfaceDescriptor alternate;
316 TInt err = GetAlternateInterfaceDescriptor(aAlternateInterface, alternate);
322 TUsbGenericDescriptor* descriptor = alternate.iFirstChild;
325 TUsbEndpointDescriptor* endpoint = TUsbEndpointDescriptor::Cast(descriptor);
326 if (endpoint && (endpoint->EndpointAddress() == aEndpoint))
328 aDescriptor = endpoint;
332 descriptor = descriptor->iNextPeer;
339 @return Number of alternate interface options on this interface.
341 TInt RUsbInterface::GetAlternateInterfaceCount()
343 if (!iHeadInterfaceDescriptor)
350 // Don't need to look for children of the interface -- all the alternates
352 TUsbGenericDescriptor* descriptor = iHeadInterfaceDescriptor;
355 TUsbInterfaceDescriptor* interface = TUsbInterfaceDescriptor::Cast(descriptor);
362 // we must check any Interface Association Descriptors for
363 // Alternate Interface settings. The spec is abiguous on how these may be organised so we
364 // presume the worst and do a full search
365 TUsbInterfaceAssociationDescriptor* iad = TUsbInterfaceAssociationDescriptor::Cast(descriptor);
368 TUsbGenericDescriptor* assocDes = iad->iFirstChild;
371 interface = TUsbInterfaceDescriptor::Cast(assocDes);
376 assocDes = assocDes->iNextPeer;
380 descriptor = descriptor->iNextPeer;
387 Count the endpoints on an alternate interface.
389 @param [in] aAlternateInterface The alternate interface to count endpoints on.
390 @return Number of endpoionts on the requested alternate interface or an error code.
392 TInt RUsbInterface::EnumerateEndpointsOnInterface(TInt aAlternateInterface)
394 TUsbInterfaceDescriptor alternate;
395 TInt err = GetAlternateInterfaceDescriptor(aAlternateInterface, alternate);
401 return alternate.NumEndpoints();
405 Returns an identifier that is unique for the bus that the device that provides this interface is on.
406 @param aBusId On success provides an identifier that is unique for the bus this interface is on.
407 @return KErrNone on success, otherwise a system-wide error code.
409 TInt RUsbInterface::GetBusId(TUsbBusId& aBusId)
411 return DoControl(EGetBusId, &aBusId);
415 Returns the size of pages used by the HCD.
417 @return The HCD's page size.
419 TInt RUsbInterface::HcdPageSize()
421 return DoControl(EHcdPageSize);
425 Returns the speed the remote device is connected at.
426 @param aDeviceSpeed On sucess an enumeration value describing the current speed of the remote device.
427 @return KErrNone on success, otherwise a system-wide error code.
429 TInt RUsbInterface::GetDeviceSpeed(RUsbInterface::TDeviceSpeed& aDeviceSpeed)
431 return DoControl(EGetDeviceSpeed, &aDeviceSpeed);
444 TUint32 RUsbPipe::Handle() const
450 Close a pipe to a remote device.
452 void RUsbPipe::Close()
456 static_cast<void>(iInterface->DoControl(EClose, (TAny*)iHandle));
458 iHeadEndpointDescriptor = NULL;
464 Clear a stall on the remote endpoint.
466 @return System-wide error code.
468 TInt RUsbPipe::ClearRemoteStall()
470 __ASSERT_ALWAYS(iHandle, User::Panic(UsbdiPanics::KUsbdiPanicCat, UsbdiPanics::EPipeRequestMadeWhileClosed));
471 __ASSERT_DEBUG(iInterface, User::Panic(UsbdiFaults::KUsbdiFaultCat, UsbdiFaults::EUsbPipeHasHandleButNoInterface));
472 return iInterface->DoControl(EClearRemoteStall, (TAny*)iHandle);
476 Cancel all queued transfers
478 void RUsbPipe::CancelAllTransfers()
480 __ASSERT_ALWAYS(iHandle, User::Panic(UsbdiPanics::KUsbdiPanicCat, UsbdiPanics::EPipeRequestMadeWhileClosed));
481 __ASSERT_DEBUG(iInterface, User::Panic(UsbdiFaults::KUsbdiFaultCat, UsbdiFaults::EUsbPipeHasHandleButNoInterface));
482 static_cast<void>(iInterface->DoControl(EAbort, (TAny*)iHandle));
489 void RUsbPipe::IssueTransfer(TInt aTransferHandle, TRequestStatus& aRequest)
491 __ASSERT_ALWAYS(iHandle, User::Panic(UsbdiPanics::KUsbdiPanicCat, UsbdiPanics::EPipeRequestMadeWhileClosed));
492 __ASSERT_DEBUG(iInterface, User::Panic(UsbdiFaults::KUsbdiFaultCat, UsbdiFaults::EUsbPipeHasHandleButNoInterface));
493 iInterface->DoRequest(EIssueTransfer, aRequest, (TAny*)iHandle, (TAny*)aTransferHandle);
499 TInt RUsbPipe::GetEndpointId(TUsbEndpointId& aEndpointId)
501 __ASSERT_ALWAYS(iHandle, User::Panic(UsbdiPanics::KUsbdiPanicCat, UsbdiPanics::EPipeRequestMadeWhileClosed));
502 __ASSERT_DEBUG(iInterface, User::Panic(UsbdiFaults::KUsbdiFaultCat, UsbdiFaults::EUsbPipeHasHandleButNoInterface));
503 return iInterface->DoControl(EGetEndpointId, reinterpret_cast<TAny*>(iHandle), &aEndpointId);
509 TInt RUsbPipe::GetBusId(TUsbBusId& aBusId)
511 __ASSERT_ALWAYS(iHandle, User::Panic(UsbdiPanics::KUsbdiPanicCat, UsbdiPanics::EPipeRequestMadeWhileClosed));
512 __ASSERT_DEBUG(iInterface, User::Panic(UsbdiFaults::KUsbdiFaultCat, UsbdiFaults::EUsbPipeHasHandleButNoInterface));
513 return iInterface->GetBusId(aBusId);
517 Return the section of the USB Configuration Descriptor under the supplied endpoint.
519 @param [out] aDescriptor The descriptor tree for this endpoint.
520 @return System-wide error code.
522 TInt RUsbPipe::GetEndpointDescriptor(TUsbEndpointDescriptor& aDescriptor)
524 __ASSERT_ALWAYS(iHandle, User::Panic(UsbdiPanics::KUsbdiPanicCat, UsbdiPanics::EPipeRequestMadeWhileClosed));
525 __ASSERT_DEBUG(iInterface, User::Panic(UsbdiFaults::KUsbdiFaultCat, UsbdiFaults::EUsbPipeHasHandleButNoInterface));
527 if (iHeadEndpointDescriptor)
529 aDescriptor = *iHeadEndpointDescriptor;