Update contrib.
1 // Copyright (c) 2000-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.
14 // e32\include\drivers\usbcsc.h
15 // Kernel side definitions for the USB Device driver stack (PIL + LDD).
27 #include <kernel/kernel.h>
28 #include <kernel/kern_priv.h>
29 #include <kernel/kpower.h>
32 #include <d32usbcsc.h>
34 #include <drivers/usbcshared.h>
36 /** LDD Major version, This should agree with the information in RDevUsbcClient::TVer.
38 const TInt KUsbcScMajorVersion = 0;
40 /** LDD Minor version, This should agree with the information in RDevUsbcClient::TVer.
42 const TInt KUsbcScMinorVersion = 1;
44 /** LDD Build version, This should agree with the information in RDevUsbcClient::TVer.
46 const TInt KUsbcScBuildVersion = KE32BuildVersionNumber;
48 /** Must correspond to the max enum of TRequest + 1;
49 currently this is ERequestOtgFeaturesNotify = 10.
51 const TInt KUsbcScMaxRequests = 11;
53 // Request queue sizes need to be power of 2.
55 /** The number of requests that can be queued on any IN endpoint */
56 const TInt KUsbcScInRequests = 4;
57 /** The number of requests that can be queued on any OUT endpoint */
58 const TInt KUsbcScOutRequests = 2;
60 /** In TUsbcScBuffer.iDirection, this indicates that the endpoint is an IN endpoint */
61 const TInt KUsbcScIn = 0;
62 /** In TUsbcScBuffer.iDirection, this indicates that the endpoint is an OUT endpoint */
63 const TInt KUsbcScOut = 1;
66 /** In TUsbcScBuffer.iDirection, this indicates that the endpoint is an Bidirectional endpoint
67 currently operating as an IN endpoint */
68 const TInt KUsbcScBiIn = 2;
69 /** In TUsbcScBuffer.iDirection, this indicates that the endpoint is an Bidirectional endpoint
70 currently operating as an OUT endpoint */
71 const TInt KUsbcScBiOut = 3;
73 /** The number of directions supported for endpoints, other then EP0. Currently 2, IN and OUT. */
74 const TInt KUsbcScDirections = 2;
76 /** In TUsbcScBuffer::iDirection, this indicates that the endpoint direction is Unknown. */
77 const TInt KUsbcScUnknown = 4;
79 const TInt KPageSize = 0x1000;
81 /** The default buffer size requested for a endpoint, if the user app does not specify a size.*/
82 const TInt KUsbcScDefaultBufferSize = 0x10000; // 64k
84 /** The size of the unmapped region of memory between endpoint buffers.
85 This serves as a guard region, making memory over/under runs more obviose.*/
86 const TInt KGuardSize = KPageSize;
88 /** The size put aside for the chunk header structre.*/
89 const TInt KHeaderSize = KPageSize;
91 /** For buffers of size >= KUsbScBigBuffIs, The smallest unit of continiouse memory that will be used.
92 No read will be set up smaller then this, to avoid overly fragmenting the data.
94 const TInt KUsbSc_BigBuff_MinimumRamRun = KPageSize;
96 /** For buffers of size < KUsbScBigBuffIs, The smallest unit of continiouse memory that will be used.
97 No read will be set up smaller then this, to avoid overly fragmenting the data.*/
98 const TInt KUsbSc_SmallBuff_MinimumRamRun = 1024;
100 /** The size a buffer request has to be to switch to using KUsbSc_BigBuff_MinimumRamRun.
101 If the requested buffer is smaller, then the smallest memory allocated to a buffer is KPageSize*/
102 const TInt KUsbScBigBuffIs = KPageSize*6;
106 // EP0 is mapped manually, unlike the other endpoints.
108 /** The position, within the chunk, that the EP0 IN buffer appears*/
109 const TInt KUsbScEP0InBufPos = 0x800;
110 /** The position, within the chunk, that the EP0 IN buffer ends*/
111 const TInt KUsbScEP0InBufEnd = KUsbScEP0InBufPos + 0x400;
113 // Its better for Out to go 2nd, so gaurd page after it.
114 /** The position, within the chunk, that the EP0 OUT buffer appears*/
115 const TInt KUsbScEP0OutBufPos = 0xc00;
116 /** The position, within the chunk, that the EP0 OUT buffer ends*/
117 const TInt KUsbScEP0OutBufEnd = KUsbScEP0OutBufPos + 0x400;
119 /** The number of the entry within the chunk BufferRecord table, for the OUT ep0 buffer.*/
120 const TInt KUsbcScEp0OutBuff = 0;
121 /** The number of the entry within the chunk BufferRecord table, for the IN ep0 buffer.*/
122 const TInt KUsbcScEp0InBuff = 1;
126 //########################### Logical Device Driver (LDD) #############################
129 /** USB LDD factory class.
131 class DUsbcScLogDevice : public DLogicalDevice
135 virtual TInt Install();
136 virtual void GetCaps(TDes8& aDes) const;
137 virtual TInt Create(DLogicalChannelBase*& aChannel);
141 class DLddUsbcScChannel;
143 /** Endpoint tracking for the LDD buffering etc.
145 class TUsbcScEndpoint
148 TUsbcScEndpoint(DLddUsbcScChannel* aLDD, DUsbClientController* aController,
149 const TUsbcScEndpointInfo* aEndpointInfo, TInt aEndpointNum
153 void CancelTransfer(DThread* aThread);
154 void AbortTransfer();
155 inline TUsbcScEndpointInfo* EndpointInfo();
156 inline TInt RxBytesAvailable() const;
157 inline void ResetTransferInfo();
158 inline void SetClientReadPending(TBool aVal);
159 inline void SetClientWritePending(TBool aVal);
160 inline TBool ClientWritePending();
161 inline TBool ClientReadPending();
162 inline void SetRealEpNumber(TInt aRealEpNumber);
163 inline TInt RealEpNumber() const;
164 inline TInt EpNumber() const;
165 inline void StartBuffer();
166 inline void SetBuffer(TUsbcScBuffer* aBuffer);
167 inline TUsbcScBuffer* GetBuffer();
171 static void RequestCallback(TAny* aTUsbcScEndpoint);
173 TInt RxComplete(TBool aReEntrant);
174 void RxCompleteNow();
179 TUsbcRequestCallback* iRequestCallbackInfo;
182 DUsbClientController* iController;
183 TUsbcScEndpointInfo iEndpointInfo;
184 TBool iClientReadPending;
185 TBool iClientWritePending;
186 TInt iEndpointNumber;
188 DLddUsbcScChannel* iLdd;
190 TUint32 iBytesTransferred;
191 TInt iBandwidthPriority;
192 TUsbcScBuffer* iBuffer;
197 /** Linked list of 'alternate setting' info for use by the LDD.
199 class TUsbcScAlternateSetting
202 TUsbcScAlternateSetting();
203 ~TUsbcScAlternateSetting();
206 TUsbcScAlternateSetting* iNext;
207 TUsbcScAlternateSetting* iPrevious;
208 TInt iNumberOfEndpoints;
210 TUsbcScEndpoint* iEndpoint[KMaxEndpointsPerClient + 1];
213 class TUsbcScAlternateSettingList
216 TUsbcScAlternateSettingList();
217 ~TUsbcScAlternateSettingList();
220 TUsbcScAlternateSetting* iHead;
221 TUsbcScAlternateSetting* iTail;
224 class TUsbcScChunkInfo
227 TUsbcScChunkInfo(DLogicalDevice* aLdd);
228 TInt CreateChunk(TInt aTotalSize);
230 TInt ChunkAlloc(TInt aOffset, TInt aSize);
232 TInt GetPhysical(TInt aOffset, TPhysAddr* aPhysical);
233 static TInt New(TUsbcScChunkInfo*& aChunk, TInt aSize, DLogicalDevice* aLdd);
240 TInt8 iPageNtz; // Number of trailing zeros for a page. (Eg 4k page has 12 t.z.)
243 TUint32 iChunkMapAttr;
244 DLogicalDevice* iLdd;
248 // Used to represent a matrix of endpoints with a column of sizes.
249 // Used by TRealizeInfo
251 class TEndpointSortBufs
254 TUsbcScEndpoint** iEp;
259 // This is used to calculate the layout of the shared chunk
260 // based on a list of alternative settings / endpoints provided.
265 void Init(TUsbcScAlternateSettingList* aAlternateSettingList);
266 TInt CopyAndSortEndpoints();
267 void CalcBuffSizes();
270 void LayoutChunkHeader(TUsbcScChunkInfo* aChunkInfo);
276 TEndpointSortBufs iBufs[KUsbcScDirections];
277 TUsbcScAlternateSettingList* iAlternateSettingList;
279 // Chunk layout info.
280 TUsbcScChunkBuffersHeader* iChunkStuct;
281 TUsbcScChunkAltSettingHeader* iAltSettingsTbl;
286 /** The channel class - the actual USB LDD.
288 class DLddUsbcScChannel : public DLogicalChannel
292 ~DLddUsbcScChannel();
293 virtual void HandleMsg(TMessageBase* aMsg);
294 virtual TInt DoCreate(TInt aUnit, const TDesC8* aInfo, const TVersion& aVer);
295 virtual TInt RequestUserHandle(DThread* aThread, TOwnerType aType);
296 inline DThread* Client() const {return iClient;}
297 inline TBool ChannelClosing() const {return iChannelClosing;}
298 inline TUint AlternateSetting() const {return iAlternateSetting;}
300 static void RequestCallbackEp0(TAny* aTUsbcScChannel);
304 TInt DoCancel(TInt aReqNo, TUint aMask, TUint a1);
305 TInt DoRequest(TInt aReqNo, TRequestStatus* aStatus, TAny* a1, TAny* a2);
306 TInt DoControl(TInt aFunction, TAny* a1, TAny* a2);
307 TInt DoReadDataNotify(TRequestStatus* aStatus, TInt aBufferNum, TInt aLength);
308 void StartDataRead(TInt aBufferNum);
309 TInt DoWriteData(TRequestStatus* aStatus,TInt aBufferNum, TUint aStart, TUint aLength, TUint aFlags);
310 TBool AlternateDeviceStateTestComplete();
311 TInt SetInterface(TInt aInterfaceNum, TUsbcScIfcInfo* aUserInterfaceInfoBuf);
313 void DestroyAllInterfaces();
314 void DestroyInterface(TUint aInterface);
316 inline TBool ValidEndpoint(TInt aEndpoint);
317 TInt GetRealEpForEpResource(TInt aEndpoint, TInt& aRealEp);
318 inline TBool Configured();
319 TInt DoEmergencyComplete();
320 void ReadDes8(const TAny* aPtr, TDes8& aDes);
322 void CancelNotifyEndpointStatus();
323 void CancelNotifyOtgFeatures();
324 static void StatusChangeCallback(TAny* aDLddUsbcChannel);
325 static void EndpointStatusChangeCallback(TAny* aDLddUsbcChannel);
326 static void OtgFeatureChangeCallback(TAny* aDLddUsbcChannel);
327 static void EmergencyCompleteDfc(TAny* aDLddUsbcChannel);
328 void DeConfigure(TInt aErrorCode);
329 TInt SelectAlternateSetting(TUint aAlternateSetting);
330 TInt EpFromAlternateSetting(TUint aAlternateSetting, TInt aEndpoint);
331 TInt ProcessAlternateSetting(TUint aAlternateSetting);
332 TInt32 StartNextInAlternateSetting();
333 TInt ProcessDeviceState(TUsbcDeviceState aDeviceState);
334 void ResetInterface(TInt aErrorCode);
335 void PanicClientThread(TInt aReason);
337 TInt RealizeInterface(void);
340 DUsbClientController* iController;
342 TBool iValidInterface;
344 TUsbcScAlternateSettingList* iAlternateSettingList;
345 TUsbcScEndpoint** iEndpoint; // Pointer to the current endpoint set.
347 static const TInt KUsbcMaxRequests = RDevUsbcScClient::ERequestMaxRequests;
348 TRequestStatus* iRequestStatus[KUsbcMaxRequests];
349 TUsbcClientCallback iCompleteAllCallbackInfo;
350 TAny* iStatusChangePtr;
351 TUsbcStatusCallback iStatusCallbackInfo;
352 TAny* iEndpointStatusChangePtr;
353 TUsbcEndpointStatusCallback iEndpointStatusCallbackInfo;
354 TAny* iOtgFeatureChangePtr;
355 TUsbcOtgFeatureCallback iOtgFeatureCallbackInfo;
356 TUint8* iBufferBaseEp0;
358 TInt iNumberOfEndpoints;
359 TUsbcDeviceState iDeviceState;
360 TUsbcDeviceState iOldDeviceState;
361 TBool iOwnsDeviceControl;
362 TUint16 iAlternateSetting;
365 TUsbcDeviceStatusQueue* iStatusFifo;
366 TBool iUserKnowsAltSetting;
367 TBool iDeviceStatusNeeded;
368 TBool iChannelClosing;
369 TBool iRealizeCalled;
371 TUsbcScChunkInfo* iChunkInfo;
373 TUsbcScBuffer *iBuffers;
375 TUsbcScEndpoint* iEp0Endpoint;
379 friend class TUsbcScBuffer;
380 friend void TUsbcScEndpoint::AbortTransfer();
384 This class is used by TUsbcScStatusList to form a queue of status requests.
385 These requests are on a buffer basis, so that all buffers can have at least two requests
386 pending, at the same time. (i.e. buffer 1 could have two requests outstanding, as well as 2 on buffer 2.)
389 class TUsbcScStatusElement
392 TRequestStatus* iStatus;
398 enum TUsbcScStatusState
403 EFramgementInProgress
406 class TUsbcScStatusList
409 TInt Construct(TInt aSize, DThread* aThread);
412 TUsbcScStatusElement* Next();
414 TInt Add(TRequestStatus* aStatus, TInt aLength, TUint aStart, TUint aFlags);
415 void CancelQueued(TInt aErrorCode=KErrCancel);
416 TInt Complete(TInt aError);
419 TUsbcScStatusState iState;
423 TInt iHead; // The element at the head of the queue, ie, the earliest added, and next to be removed.
424 TInt iLength; // Length of queue, ie number of elements within
425 TInt iSize; // size of array, ie, max # of requests at a time.
426 TUsbcScStatusElement* iElements;
433 This class holds the kernel's copy of all the details related to a shared endpoint buffer,
434 and provides methods for the LDD to manipulate it.
439 static const TInt8 KNoEpAssigned=0;
440 static const TInt8 KEpIsEnding=1;
441 static const TInt8 KEpIsStarting=2;
444 TInt Construct(TInt aDirection, DLddUsbcScChannel* aLdd, TInt aBufferOffset, TInt aBufferEndOffset, TInt aMinReadSize, TInt aMaxPacketSize, TInt aMaxReadSize);
445 void CreateChunkBufferHeader();
446 void StartEndpoint(TUsbcRequestCallback* iRequestInfo, TUint iFlags);
450 TInt StartDataRead();
451 void CompleteRead(TBool aStartNextRead=ETrue);
453 void StartDataWrite();
454 void CompleteWrite();
455 void Cancel(TInt aErrorCode);
457 void UpdateBufferList(TInt aByteCount,TUint aFlags, TBool aStartNextRead=ETrue);
458 void Ep0CancelLddRead();
459 void SendEp0StatusPacket(TInt aState);
467 DLddUsbcScChannel* iLdd;
469 SUsbcScBufferHeader* iBufferStart;
470 TUint iBufferEnd; // One word on from the last word in the buffer.
472 TUsbcScStatusList iStatusList;
473 TUsbcRequestCallback* iCallback;
476 TInt iHead; // Out endpoints only;
477 TUint iSent; // In endpoints only
479 TUsbcScChunkInfo* iChunkInfo;
482 TInt iMaxPacketSize; // 0 indicates unconfiured.
486 // needed for backwards compatibility
487 TUsbcPacketArray iIndexArray[KUsbcDmaBufNumMax]; // Has 2 elements
488 TUsbcPacketArray iSizeArray[KUsbcDmaBufNumMax]; // Has 2 elements
497 #include <drivers/usbcsc.inl>
499 #endif // __USBCSC_H__