1*53ee8cc1Swenshuai.xi //<MStar Software>
2*53ee8cc1Swenshuai.xi //******************************************************************************
3*53ee8cc1Swenshuai.xi // MStar Software
4*53ee8cc1Swenshuai.xi // Copyright (c) 2010 - 2012 MStar Semiconductor, Inc. All rights reserved.
5*53ee8cc1Swenshuai.xi // All software, firmware and related documentation herein ("MStar Software") are
6*53ee8cc1Swenshuai.xi // intellectual property of MStar Semiconductor, Inc. ("MStar") and protected by
7*53ee8cc1Swenshuai.xi // law, including, but not limited to, copyright law and international treaties.
8*53ee8cc1Swenshuai.xi // Any use, modification, reproduction, retransmission, or republication of all
9*53ee8cc1Swenshuai.xi // or part of MStar Software is expressly prohibited, unless prior written
10*53ee8cc1Swenshuai.xi // permission has been granted by MStar.
11*53ee8cc1Swenshuai.xi //
12*53ee8cc1Swenshuai.xi // By accessing, browsing and/or using MStar Software, you acknowledge that you
13*53ee8cc1Swenshuai.xi // have read, understood, and agree, to be bound by below terms ("Terms") and to
14*53ee8cc1Swenshuai.xi // comply with all applicable laws and regulations:
15*53ee8cc1Swenshuai.xi //
16*53ee8cc1Swenshuai.xi // 1. MStar shall retain any and all right, ownership and interest to MStar
17*53ee8cc1Swenshuai.xi // Software and any modification/derivatives thereof.
18*53ee8cc1Swenshuai.xi // No right, ownership, or interest to MStar Software and any
19*53ee8cc1Swenshuai.xi // modification/derivatives thereof is transferred to you under Terms.
20*53ee8cc1Swenshuai.xi //
21*53ee8cc1Swenshuai.xi // 2. You understand that MStar Software might include, incorporate or be
22*53ee8cc1Swenshuai.xi // supplied together with third party`s software and the use of MStar
23*53ee8cc1Swenshuai.xi // Software may require additional licenses from third parties.
24*53ee8cc1Swenshuai.xi // Therefore, you hereby agree it is your sole responsibility to separately
25*53ee8cc1Swenshuai.xi // obtain any and all third party right and license necessary for your use of
26*53ee8cc1Swenshuai.xi // such third party`s software.
27*53ee8cc1Swenshuai.xi //
28*53ee8cc1Swenshuai.xi // 3. MStar Software and any modification/derivatives thereof shall be deemed as
29*53ee8cc1Swenshuai.xi // MStar`s confidential information and you agree to keep MStar`s
30*53ee8cc1Swenshuai.xi // confidential information in strictest confidence and not disclose to any
31*53ee8cc1Swenshuai.xi // third party.
32*53ee8cc1Swenshuai.xi //
33*53ee8cc1Swenshuai.xi // 4. MStar Software is provided on an "AS IS" basis without warranties of any
34*53ee8cc1Swenshuai.xi // kind. Any warranties are hereby expressly disclaimed by MStar, including
35*53ee8cc1Swenshuai.xi // without limitation, any warranties of merchantability, non-infringement of
36*53ee8cc1Swenshuai.xi // intellectual property rights, fitness for a particular purpose, error free
37*53ee8cc1Swenshuai.xi // and in conformity with any international standard. You agree to waive any
38*53ee8cc1Swenshuai.xi // claim against MStar for any loss, damage, cost or expense that you may
39*53ee8cc1Swenshuai.xi // incur related to your use of MStar Software.
40*53ee8cc1Swenshuai.xi // In no event shall MStar be liable for any direct, indirect, incidental or
41*53ee8cc1Swenshuai.xi // consequential damages, including without limitation, lost of profit or
42*53ee8cc1Swenshuai.xi // revenues, lost or damage of data, and unauthorized system use.
43*53ee8cc1Swenshuai.xi // You agree that this Section 4 shall still apply without being affected
44*53ee8cc1Swenshuai.xi // even if MStar Software has been modified by MStar in accordance with your
45*53ee8cc1Swenshuai.xi // request or instruction for your use, except otherwise agreed by both
46*53ee8cc1Swenshuai.xi // parties in writing.
47*53ee8cc1Swenshuai.xi //
48*53ee8cc1Swenshuai.xi // 5. If requested, MStar may from time to time provide technical supports or
49*53ee8cc1Swenshuai.xi // services in relation with MStar Software to you for your use of
50*53ee8cc1Swenshuai.xi // MStar Software in conjunction with your or your customer`s product
51*53ee8cc1Swenshuai.xi // ("Services").
52*53ee8cc1Swenshuai.xi // You understand and agree that, except otherwise agreed by both parties in
53*53ee8cc1Swenshuai.xi // writing, Services are provided on an "AS IS" basis and the warranty
54*53ee8cc1Swenshuai.xi // disclaimer set forth in Section 4 above shall apply.
55*53ee8cc1Swenshuai.xi //
56*53ee8cc1Swenshuai.xi // 6. Nothing contained herein shall be construed as by implication, estoppels
57*53ee8cc1Swenshuai.xi // or otherwise:
58*53ee8cc1Swenshuai.xi // (a) conferring any license or right to use MStar name, trademark, service
59*53ee8cc1Swenshuai.xi // mark, symbol or any other identification;
60*53ee8cc1Swenshuai.xi // (b) obligating MStar or any of its affiliates to furnish any person,
61*53ee8cc1Swenshuai.xi // including without limitation, you and your customers, any assistance
62*53ee8cc1Swenshuai.xi // of any kind whatsoever, or any information; or
63*53ee8cc1Swenshuai.xi // (c) conferring any license or right under any intellectual property right.
64*53ee8cc1Swenshuai.xi //
65*53ee8cc1Swenshuai.xi // 7. These terms shall be governed by and construed in accordance with the laws
66*53ee8cc1Swenshuai.xi // of Taiwan, R.O.C., excluding its conflict of law rules.
67*53ee8cc1Swenshuai.xi // Any and all dispute arising out hereof or related hereto shall be finally
68*53ee8cc1Swenshuai.xi // settled by arbitration referred to the Chinese Arbitration Association,
69*53ee8cc1Swenshuai.xi // Taipei in accordance with the ROC Arbitration Law and the Arbitration
70*53ee8cc1Swenshuai.xi // Rules of the Association by three (3) arbitrators appointed in accordance
71*53ee8cc1Swenshuai.xi // with the said Rules.
72*53ee8cc1Swenshuai.xi // The place of arbitration shall be in Taipei, Taiwan and the language shall
73*53ee8cc1Swenshuai.xi // be English.
74*53ee8cc1Swenshuai.xi // The arbitration award shall be final and binding to both parties.
75*53ee8cc1Swenshuai.xi //
76*53ee8cc1Swenshuai.xi //******************************************************************************
77*53ee8cc1Swenshuai.xi //<MStar Software>
78*53ee8cc1Swenshuai.xi
79*53ee8cc1Swenshuai.xi //#include <MsCommon.h> // NUSED
80*53ee8cc1Swenshuai.xi //#include <cyg/hal/hal_if.h> // NUSED
81*53ee8cc1Swenshuai.xi
82*53ee8cc1Swenshuai.xi #include "include/drvConfig.h"
83*53ee8cc1Swenshuai.xi
84*53ee8cc1Swenshuai.xi #ifdef CONFIG_DEBUG
85*53ee8cc1Swenshuai.xi #define DEBUG
86*53ee8cc1Swenshuai.xi #else
87*53ee8cc1Swenshuai.xi #undef DEBUG
88*53ee8cc1Swenshuai.xi #endif
89*53ee8cc1Swenshuai.xi
90*53ee8cc1Swenshuai.xi //#include "include/drvCompiler.h" // NUSED
91*53ee8cc1Swenshuai.xi //#include "include/drvErrno.h" // NUSED
92*53ee8cc1Swenshuai.xi //#include "include/drvPorts.h" // NUSED
93*53ee8cc1Swenshuai.xi //#include "include/drvPCIMEM.h" // NUSED
94*53ee8cc1Swenshuai.xi //#include "include/drvTimer.h" // NUSED
95*53ee8cc1Swenshuai.xi //#include "include/drvList.h" // NUSED
96*53ee8cc1Swenshuai.xi
97*53ee8cc1Swenshuai.xi //#include "include/drvUSBHost.h" // NUSED
98*53ee8cc1Swenshuai.xi //#include "drvUsbd.h" // NUSED
99*53ee8cc1Swenshuai.xi //#include "include/drvCPE_AMBA.h" // NUSED
100*53ee8cc1Swenshuai.xi #include "drvEHCI.h"
101*53ee8cc1Swenshuai.xi
102*53ee8cc1Swenshuai.xi /**
103*53ee8cc1Swenshuai.xi * @brief complete function of URB
104*53ee8cc1Swenshuai.xi *
105*53ee8cc1Swenshuai.xi * @param urb
106*53ee8cc1Swenshuai.xi * @param regs
107*53ee8cc1Swenshuai.xi *
108*53ee8cc1Swenshuai.xi * @return none
109*53ee8cc1Swenshuai.xi */
ms_usb_api_completion(struct urb * urb,struct stPtRegs * regs)110*53ee8cc1Swenshuai.xi static void ms_usb_api_completion(struct urb *urb, struct stPtRegs *regs)
111*53ee8cc1Swenshuai.xi {
112*53ee8cc1Swenshuai.xi struct usb_api_data *pAwd = (struct usb_api_data *)urb->pContext;
113*53ee8cc1Swenshuai.xi
114*53ee8cc1Swenshuai.xi /* move to ms_usb_hcd_giveback_urb() to support 3rd driver */
115*53ee8cc1Swenshuai.xi //if (urb->SetDMABuf != NULL)
116*53ee8cc1Swenshuai.xi //{
117*53ee8cc1Swenshuai.xi // //MsOS_FreeMemory(urb->SetDMABuf, gs32NonCachedPoolID_MIU0 );
118*53ee8cc1Swenshuai.xi // Usb_FreeNonCachedMemory(urb->SetDMABuf);
119*53ee8cc1Swenshuai.xi //
120*53ee8cc1Swenshuai.xi // urb->SetDMABuf = NULL;
121*53ee8cc1Swenshuai.xi //}
122*53ee8cc1Swenshuai.xi //
123*53ee8cc1Swenshuai.xi //if (urb->TxDMABuf != NULL)
124*53ee8cc1Swenshuai.xi //{
125*53ee8cc1Swenshuai.xi // if (usb_pipein(urb->u32Pipe))
126*53ee8cc1Swenshuai.xi // {
127*53ee8cc1Swenshuai.xi // #ifdef DEBUG_PERFORMANCE
128*53ee8cc1Swenshuai.xi // if (urb->u32TransferBufferLength >= 512)
129*53ee8cc1Swenshuai.xi // diag_printf("ms_usb_api_completion: copy length: 0x%lx \n", urb->u32TransferBufferLength);
130*53ee8cc1Swenshuai.xi // #endif
131*53ee8cc1Swenshuai.xi // memcpy(urb->pTransferBuffer, urb->TxDMABuf, urb->u32TransferBufferLength);
132*53ee8cc1Swenshuai.xi // }
133*53ee8cc1Swenshuai.xi // //MsOS_FreeMemory(urb->TxDMABuf,gs32NonCachedPoolID_MIU0 );
134*53ee8cc1Swenshuai.xi // Usb_FreeNonCachedMemory(urb->TxDMABuf);
135*53ee8cc1Swenshuai.xi //
136*53ee8cc1Swenshuai.xi // urb->TxDMABuf = NULL;
137*53ee8cc1Swenshuai.xi //}
138*53ee8cc1Swenshuai.xi
139*53ee8cc1Swenshuai.xi pAwd->done = 1;
140*53ee8cc1Swenshuai.xi wmb();
141*53ee8cc1Swenshuai.xi
142*53ee8cc1Swenshuai.xi wake_up(&pAwd->wqh);
143*53ee8cc1Swenshuai.xi }
144*53ee8cc1Swenshuai.xi #ifdef DEBUG
145*53ee8cc1Swenshuai.xi extern BOOL ms_isHcdRunning(struct usb_hcd *hcd); // debug
146*53ee8cc1Swenshuai.xi extern void ms_dumpHcdRegister(struct usb_hcd *hcd); // debug
147*53ee8cc1Swenshuai.xi #endif
148*53ee8cc1Swenshuai.xi /**
149*53ee8cc1Swenshuai.xi * @brief Start the URB and wait until it complete
150*53ee8cc1Swenshuai.xi *
151*53ee8cc1Swenshuai.xi * @param urb
152*53ee8cc1Swenshuai.xi * @param timeout
153*53ee8cc1Swenshuai.xi * @param act_len
154*53ee8cc1Swenshuai.xi *
155*53ee8cc1Swenshuai.xi * @return function status
156*53ee8cc1Swenshuai.xi */
ms_usb_start_wait_urb(struct urb * urb,int timeout,int * act_len)157*53ee8cc1Swenshuai.xi static int ms_usb_start_wait_urb(struct urb *urb, int timeout, int* act_len)
158*53ee8cc1Swenshuai.xi {
159*53ee8cc1Swenshuai.xi struct usb_api_data stAwd;
160*53ee8cc1Swenshuai.xi struct usb_hcd *hcd = urb->dev->bus->hcpriv;
161*53ee8cc1Swenshuai.xi struct ehci_hcd *ehci = hcd_to_ehci (hcd);
162*53ee8cc1Swenshuai.xi int status;
163*53ee8cc1Swenshuai.xi int rh_csc_shutdown = 0;
164*53ee8cc1Swenshuai.xi U32 uDelay=0;
165*53ee8cc1Swenshuai.xi
166*53ee8cc1Swenshuai.xi stAwd.done = 0;
167*53ee8cc1Swenshuai.xi
168*53ee8cc1Swenshuai.xi urb->pContext = &stAwd;
169*53ee8cc1Swenshuai.xi status = ms_usb_submit_urb(urb, GFP_ATOMIC);
170*53ee8cc1Swenshuai.xi if (status)
171*53ee8cc1Swenshuai.xi {
172*53ee8cc1Swenshuai.xi ms_usb_free_urb(urb);
173*53ee8cc1Swenshuai.xi return status;
174*53ee8cc1Swenshuai.xi }
175*53ee8cc1Swenshuai.xi
176*53ee8cc1Swenshuai.xi while ( timeout && (stAwd.done == 0) )
177*53ee8cc1Swenshuai.xi {
178*53ee8cc1Swenshuai.xi /* to speed up one WIFI dongle with internal HUB */
179*53ee8cc1Swenshuai.xi //if((urb->dev->parent != NULL) && (urb->dev->parent->parent == NULL) // device on roothub
180*53ee8cc1Swenshuai.xi if ((hcd_reg_readb((U32)&ehci->op_regs->portsc[0]) & PORTSC_CSC) // CSC happen on ehci port
181*53ee8cc1Swenshuai.xi || !(hcd_reg_readb((U32)&ehci->op_regs->portsc[0]) & PORTSC_PE)) // PE cleared on ehci port
182*53ee8cc1Swenshuai.xi {
183*53ee8cc1Swenshuai.xi timeout = 0;
184*53ee8cc1Swenshuai.xi rh_csc_shutdown = 1;
185*53ee8cc1Swenshuai.xi diag_printf("[USB] CSC happend/PE clr while wait CMP PSC [0x%x]\n", hcd_reg_readb((U32)&ehci->op_regs->portsc[0]));
186*53ee8cc1Swenshuai.xi break;
187*53ee8cc1Swenshuai.xi }
188*53ee8cc1Swenshuai.xi
189*53ee8cc1Swenshuai.xi if ( uDelay > (U32) timeout)
190*53ee8cc1Swenshuai.xi {
191*53ee8cc1Swenshuai.xi timeout =0;
192*53ee8cc1Swenshuai.xi break;
193*53ee8cc1Swenshuai.xi }
194*53ee8cc1Swenshuai.xi else
195*53ee8cc1Swenshuai.xi {
196*53ee8cc1Swenshuai.xi mdelay(1);//tick base is 1 ms
197*53ee8cc1Swenshuai.xi //HAL_DELAY_US(1000); // NUSED
198*53ee8cc1Swenshuai.xi uDelay+=1;
199*53ee8cc1Swenshuai.xi }
200*53ee8cc1Swenshuai.xi rmb();
201*53ee8cc1Swenshuai.xi }
202*53ee8cc1Swenshuai.xi
203*53ee8cc1Swenshuai.xi if (!timeout && !stAwd.done)
204*53ee8cc1Swenshuai.xi {
205*53ee8cc1Swenshuai.xi diag_printf("usb_control/bulk_msg: timeout... (status=%d)\n", (int)urb->s32Status);
206*53ee8cc1Swenshuai.xi if (urb->s32Status != -EINPROGRESS)
207*53ee8cc1Swenshuai.xi {
208*53ee8cc1Swenshuai.xi status = urb->s32Status;
209*53ee8cc1Swenshuai.xi }
210*53ee8cc1Swenshuai.xi else
211*53ee8cc1Swenshuai.xi {
212*53ee8cc1Swenshuai.xi //int retv;
213*53ee8cc1Swenshuai.xi
214*53ee8cc1Swenshuai.xi //diag_printf("<%s> set Bad Device............\n", __FUNCTION__);
215*53ee8cc1Swenshuai.xi #ifdef DEBUG
216*53ee8cc1Swenshuai.xi ms_dumpHcdRegister(hcd);
217*53ee8cc1Swenshuai.xi if (ms_isHcdRunning(hcd)) // just for debugging
218*53ee8cc1Swenshuai.xi diag_printf("HCD RUN!!!\n");
219*53ee8cc1Swenshuai.xi #endif
220*53ee8cc1Swenshuai.xi hcd->isBadDevice = TRUE;
221*53ee8cc1Swenshuai.xi hcd->badDeviceCnt++;
222*53ee8cc1Swenshuai.xi
223*53ee8cc1Swenshuai.xi /* Linux kernel put 5 seconds wait time for control commands, we put 3 seconds here */
224*53ee8cc1Swenshuai.xi diag_printf("timeout unlink(kill) urb(qh:%p)\n", urb->hcpriv);
225*53ee8cc1Swenshuai.xi ms_hcd_unlink_urb(urb, -ENOENT);
226*53ee8cc1Swenshuai.xi //if ((retv = ms_usb_unlink_urb(urb)) == -ETIMEDOUT)
227*53ee8cc1Swenshuai.xi //{
228*53ee8cc1Swenshuai.xi // diag_printf("timeout retry %d\n", retv);
229*53ee8cc1Swenshuai.xi // retv = ms_usb_unlink_urb(urb);
230*53ee8cc1Swenshuai.xi //}
231*53ee8cc1Swenshuai.xi //diag_printf("timeout unlink urb(%p) error %d\n", urb, retv);
232*53ee8cc1Swenshuai.xi
233*53ee8cc1Swenshuai.xi status = rh_csc_shutdown ? -ESHUTDOWN : -ETIMEDOUT;
234*53ee8cc1Swenshuai.xi }
235*53ee8cc1Swenshuai.xi }
236*53ee8cc1Swenshuai.xi else
237*53ee8cc1Swenshuai.xi status = urb->s32Status;
238*53ee8cc1Swenshuai.xi
239*53ee8cc1Swenshuai.xi if (act_len)
240*53ee8cc1Swenshuai.xi *act_len = urb->u32ActualLength;
241*53ee8cc1Swenshuai.xi
242*53ee8cc1Swenshuai.xi ms_usb_free_urb(urb);
243*53ee8cc1Swenshuai.xi return status;
244*53ee8cc1Swenshuai.xi }
245*53ee8cc1Swenshuai.xi
246*53ee8cc1Swenshuai.xi /**
247*53ee8cc1Swenshuai.xi * @brief Send a control setup command
248*53ee8cc1Swenshuai.xi *
249*53ee8cc1Swenshuai.xi * @param pDev
250*53ee8cc1Swenshuai.xi * @param pipe
251*53ee8cc1Swenshuai.xi * @param u8Reg
252*53ee8cc1Swenshuai.xi * @param urReg_type
253*53ee8cc1Swenshuai.xi * @param u16Val
254*53ee8cc1Swenshuai.xi * @param u16Index
255*53ee8cc1Swenshuai.xi * @param data
256*53ee8cc1Swenshuai.xi * @param u16Size
257*53ee8cc1Swenshuai.xi * @param timeout
258*53ee8cc1Swenshuai.xi *
259*53ee8cc1Swenshuai.xi * @return function status or actual length
260*53ee8cc1Swenshuai.xi */
ms_usb_control_cmd(struct usb_device * pDev,unsigned int pipe,unsigned char u8Req,unsigned char u8Req_type,unsigned short u16Val,unsigned short u16Index,void * data,unsigned short u16Size,int timeout)261*53ee8cc1Swenshuai.xi int ms_usb_control_cmd(
262*53ee8cc1Swenshuai.xi struct usb_device *pDev,
263*53ee8cc1Swenshuai.xi unsigned int pipe,
264*53ee8cc1Swenshuai.xi unsigned char u8Req,
265*53ee8cc1Swenshuai.xi unsigned char u8Req_type,
266*53ee8cc1Swenshuai.xi unsigned short u16Val,
267*53ee8cc1Swenshuai.xi unsigned short u16Index,
268*53ee8cc1Swenshuai.xi void *data,
269*53ee8cc1Swenshuai.xi unsigned short u16Size,
270*53ee8cc1Swenshuai.xi int timeout)
271*53ee8cc1Swenshuai.xi {
272*53ee8cc1Swenshuai.xi struct usb_ctrlrequest *dr = (struct usb_ctrlrequest*) kmalloc(sizeof(struct usb_ctrlrequest), GFP_NOIO);
273*53ee8cc1Swenshuai.xi struct urb *pUrb;
274*53ee8cc1Swenshuai.xi int ret;
275*53ee8cc1Swenshuai.xi int length = 0;
276*53ee8cc1Swenshuai.xi void *addr_t;
277*53ee8cc1Swenshuai.xi
278*53ee8cc1Swenshuai.xi if (!dr)
279*53ee8cc1Swenshuai.xi return -ENOMEM;
280*53ee8cc1Swenshuai.xi
281*53ee8cc1Swenshuai.xi addr_t = (void *)dr;
282*53ee8cc1Swenshuai.xi dr->bRequestType= u8Req_type;
283*53ee8cc1Swenshuai.xi dr->bRequest = u8Req;
284*53ee8cc1Swenshuai.xi dr->wValue = u16Val;
285*53ee8cc1Swenshuai.xi dr->wIndex = u16Index;
286*53ee8cc1Swenshuai.xi dr->wLength = u16Size;
287*53ee8cc1Swenshuai.xi
288*53ee8cc1Swenshuai.xi pUrb = ms_usb_alloc_urb(GFP_NOIO);
289*53ee8cc1Swenshuai.xi if (!pUrb)
290*53ee8cc1Swenshuai.xi {
291*53ee8cc1Swenshuai.xi ret = -ENOMEM;
292*53ee8cc1Swenshuai.xi goto Func_done;
293*53ee8cc1Swenshuai.xi }
294*53ee8cc1Swenshuai.xi
295*53ee8cc1Swenshuai.xi ms_usb_stuff_control_urb(pUrb, pDev, pipe, (unsigned char*)dr, data, u16Size,
296*53ee8cc1Swenshuai.xi ms_usb_api_completion, 0);
297*53ee8cc1Swenshuai.xi ret = ms_usb_start_wait_urb(pUrb, timeout, &length);
298*53ee8cc1Swenshuai.xi if (ret < 0)
299*53ee8cc1Swenshuai.xi {
300*53ee8cc1Swenshuai.xi ms_debug_err("%s retv:%d, RT:%x,R:%x,V:%x,I:%x,L:%x\n", __FUNCTION__,
301*53ee8cc1Swenshuai.xi ret, dr->bRequestType, dr->bRequest, dr->wValue, dr->wIndex, dr->wLength);
302*53ee8cc1Swenshuai.xi }
303*53ee8cc1Swenshuai.xi else
304*53ee8cc1Swenshuai.xi ret = length;
305*53ee8cc1Swenshuai.xi
306*53ee8cc1Swenshuai.xi Func_done:
307*53ee8cc1Swenshuai.xi kfree(addr_t);
308*53ee8cc1Swenshuai.xi
309*53ee8cc1Swenshuai.xi return ret;
310*53ee8cc1Swenshuai.xi }
311*53ee8cc1Swenshuai.xi
312*53ee8cc1Swenshuai.xi /**
313*53ee8cc1Swenshuai.xi * @brief Send a bulk stansaction
314*53ee8cc1Swenshuai.xi *
315*53ee8cc1Swenshuai.xi * @param pUsb_dev
316*53ee8cc1Swenshuai.xi * @param pipe
317*53ee8cc1Swenshuai.xi * @param data
318*53ee8cc1Swenshuai.xi * @param len
319*53ee8cc1Swenshuai.xi * @param actual_length
320*53ee8cc1Swenshuai.xi * @param timeout
321*53ee8cc1Swenshuai.xi *
322*53ee8cc1Swenshuai.xi * @return function status
323*53ee8cc1Swenshuai.xi */
324*53ee8cc1Swenshuai.xi #if 0 // NUSED
325*53ee8cc1Swenshuai.xi int ms_usb_bulk_msg(
326*53ee8cc1Swenshuai.xi struct usb_device *pUsb_dev,
327*53ee8cc1Swenshuai.xi unsigned int pipe,
328*53ee8cc1Swenshuai.xi void *data,
329*53ee8cc1Swenshuai.xi int len,
330*53ee8cc1Swenshuai.xi int *actual_length,
331*53ee8cc1Swenshuai.xi int timeout)
332*53ee8cc1Swenshuai.xi {
333*53ee8cc1Swenshuai.xi struct urb *urb;
334*53ee8cc1Swenshuai.xi
335*53ee8cc1Swenshuai.xi if (len < 0)
336*53ee8cc1Swenshuai.xi return -EINVAL;
337*53ee8cc1Swenshuai.xi
338*53ee8cc1Swenshuai.xi urb=ms_usb_alloc_urb(GFP_KERNEL);
339*53ee8cc1Swenshuai.xi if (!urb)
340*53ee8cc1Swenshuai.xi return -ENOMEM;
341*53ee8cc1Swenshuai.xi
342*53ee8cc1Swenshuai.xi ms_usb_stuff_bulk_urb(urb, pUsb_dev, pipe, data, len,
343*53ee8cc1Swenshuai.xi ms_usb_api_completion, 0);
344*53ee8cc1Swenshuai.xi
345*53ee8cc1Swenshuai.xi return ms_usb_start_wait_urb(urb,timeout,actual_length);
346*53ee8cc1Swenshuai.xi }
347*53ee8cc1Swenshuai.xi #endif
348*53ee8cc1Swenshuai.xi
349*53ee8cc1Swenshuai.xi /**
350*53ee8cc1Swenshuai.xi * @brief get descriptor from the device
351*53ee8cc1Swenshuai.xi *
352*53ee8cc1Swenshuai.xi * @param pDev
353*53ee8cc1Swenshuai.xi * @param u8Type
354*53ee8cc1Swenshuai.xi * @param u8Index
355*53ee8cc1Swenshuai.xi * @param pBuf
356*53ee8cc1Swenshuai.xi * @param size
357*53ee8cc1Swenshuai.xi *
358*53ee8cc1Swenshuai.xi * @return function status
359*53ee8cc1Swenshuai.xi */
ms_usb_get_descriptor(struct usb_device * pDev,unsigned char u8Type,unsigned char u8Index,void * pBuf,int size)360*53ee8cc1Swenshuai.xi int ms_usb_get_descriptor(
361*53ee8cc1Swenshuai.xi struct usb_device *pDev,
362*53ee8cc1Swenshuai.xi unsigned char u8Type,
363*53ee8cc1Swenshuai.xi unsigned char u8Index,
364*53ee8cc1Swenshuai.xi void *pBuf,
365*53ee8cc1Swenshuai.xi int size)
366*53ee8cc1Swenshuai.xi {
367*53ee8cc1Swenshuai.xi int retry = 3;
368*53ee8cc1Swenshuai.xi int relt = 0;
369*53ee8cc1Swenshuai.xi
370*53ee8cc1Swenshuai.xi memset(pBuf,0,size);
371*53ee8cc1Swenshuai.xi
372*53ee8cc1Swenshuai.xi while (retry--)
373*53ee8cc1Swenshuai.xi {
374*53ee8cc1Swenshuai.xi relt = ms_usb_control_cmd(pDev, usb_rcvctrlpipe(pDev, 0),
375*53ee8cc1Swenshuai.xi USB_REQ_GET_DESCRIPTOR, USB_DIR_IN,
376*53ee8cc1Swenshuai.xi (u8Type << 8) + u8Index, 0, pBuf, size,
377*53ee8cc1Swenshuai.xi HZ * USB_CTRL_GET_TIMEOUT);
378*53ee8cc1Swenshuai.xi
379*53ee8cc1Swenshuai.xi if ( (relt > 0) || (relt == -EPIPE) )
380*53ee8cc1Swenshuai.xi break;
381*53ee8cc1Swenshuai.xi }
382*53ee8cc1Swenshuai.xi
383*53ee8cc1Swenshuai.xi return relt;
384*53ee8cc1Swenshuai.xi }
385*53ee8cc1Swenshuai.xi
386*53ee8cc1Swenshuai.xi /**
387*53ee8cc1Swenshuai.xi * @brief get the device descriptor from the device
388*53ee8cc1Swenshuai.xi *
389*53ee8cc1Swenshuai.xi * @param pDev
390*53ee8cc1Swenshuai.xi *
391*53ee8cc1Swenshuai.xi * @return function status
392*53ee8cc1Swenshuai.xi */
ms_usb_get_dev_descriptor(struct usb_device * pDev)393*53ee8cc1Swenshuai.xi int ms_usb_get_dev_descriptor(struct usb_device *pDev)
394*53ee8cc1Swenshuai.xi {
395*53ee8cc1Swenshuai.xi
396*53ee8cc1Swenshuai.xi return ms_usb_get_descriptor(pDev, USB_DT_DEVICE,
397*53ee8cc1Swenshuai.xi 0, &pDev->descriptor,
398*53ee8cc1Swenshuai.xi sizeof(pDev->descriptor));
399*53ee8cc1Swenshuai.xi }
400*53ee8cc1Swenshuai.xi
401*53ee8cc1Swenshuai.xi /**
402*53ee8cc1Swenshuai.xi * @brief Clear halt status of the endpoint
403*53ee8cc1Swenshuai.xi *
404*53ee8cc1Swenshuai.xi * @param pDev
405*53ee8cc1Swenshuai.xi * @param pipe
406*53ee8cc1Swenshuai.xi *
407*53ee8cc1Swenshuai.xi * @return function status
408*53ee8cc1Swenshuai.xi */
ms_usb_clear_halt(struct usb_device * pDev,int pipe)409*53ee8cc1Swenshuai.xi int ms_usb_clear_halt(struct usb_device *pDev, int pipe)
410*53ee8cc1Swenshuai.xi {
411*53ee8cc1Swenshuai.xi int retval;
412*53ee8cc1Swenshuai.xi int endp;
413*53ee8cc1Swenshuai.xi
414*53ee8cc1Swenshuai.xi if (usb_pipein (pipe))
415*53ee8cc1Swenshuai.xi endp = usb_pipeendpoint(pipe) | USB_DIR_IN;
416*53ee8cc1Swenshuai.xi else
417*53ee8cc1Swenshuai.xi endp = usb_pipeendpoint(pipe);
418*53ee8cc1Swenshuai.xi
419*53ee8cc1Swenshuai.xi retval = ms_usb_control_cmd(pDev, usb_sndctrlpipe(pDev, 0),
420*53ee8cc1Swenshuai.xi USB_REQ_CLEAR_FEATURE, USB_RECIP_ENDPOINT, 0, endp, NULL, 0,
421*53ee8cc1Swenshuai.xi HZ * USB_CTRL_SET_TIMEOUT);
422*53ee8cc1Swenshuai.xi
423*53ee8cc1Swenshuai.xi if (retval < 0)
424*53ee8cc1Swenshuai.xi return retval;
425*53ee8cc1Swenshuai.xi else
426*53ee8cc1Swenshuai.xi {
427*53ee8cc1Swenshuai.xi usb_settoggle(pDev, usb_pipeendpoint(pipe), usb_pipeout(pipe), 0);
428*53ee8cc1Swenshuai.xi //usb_endpoint_running(pDev, usb_pipeendpoint(pipe), usb_pipeout(pipe));
429*53ee8cc1Swenshuai.xi
430*53ee8cc1Swenshuai.xi return 0;
431*53ee8cc1Swenshuai.xi }
432*53ee8cc1Swenshuai.xi
433*53ee8cc1Swenshuai.xi }
434*53ee8cc1Swenshuai.xi
435*53ee8cc1Swenshuai.xi /**
436*53ee8cc1Swenshuai.xi * @brief disable the endpoint
437*53ee8cc1Swenshuai.xi *
438*53ee8cc1Swenshuai.xi * @param pDev
439*53ee8cc1Swenshuai.xi * @param epaddr
440*53ee8cc1Swenshuai.xi *
441*53ee8cc1Swenshuai.xi * @return none
442*53ee8cc1Swenshuai.xi */
ms_usb_disable_endpoint(struct usb_device * pDev,U32 epaddr)443*53ee8cc1Swenshuai.xi void ms_usb_disable_endpoint(struct usb_device *pDev, U32 epaddr)
444*53ee8cc1Swenshuai.xi {
445*53ee8cc1Swenshuai.xi U32 u32Ept = epaddr & USB_ENDPOINT_NUMBER_MASK;
446*53ee8cc1Swenshuai.xi
447*53ee8cc1Swenshuai.xi if (!pDev)
448*53ee8cc1Swenshuai.xi return;
449*53ee8cc1Swenshuai.xi //if (pDev && pDev->bus && pDev->bus->bus_ops && pDev->bus->bus_ops->ms_disable_ep)
450*53ee8cc1Swenshuai.xi {
451*53ee8cc1Swenshuai.xi if (pDev->pEpMaxPacketIn[u32Ept] || pDev->pEpMaxPacketOut[u32Ept])
452*53ee8cc1Swenshuai.xi {
453*53ee8cc1Swenshuai.xi ms_usb_hcd_flush_endpoint(pDev, epaddr);
454*53ee8cc1Swenshuai.xi /* unbind interface considered not to disable endpoint */
455*53ee8cc1Swenshuai.xi //pDev->bus->bus_ops->ms_disable_ep(pDev, epaddr);
456*53ee8cc1Swenshuai.xi ms_hcd_endpoint_disable(pDev, epaddr);
457*53ee8cc1Swenshuai.xi ms_hcd_check_dev_urb(pDev, epaddr); // check urb
458*53ee8cc1Swenshuai.xi }
459*53ee8cc1Swenshuai.xi }
460*53ee8cc1Swenshuai.xi }
461*53ee8cc1Swenshuai.xi
462*53ee8cc1Swenshuai.xi /**
463*53ee8cc1Swenshuai.xi * @brief disable the interface
464*53ee8cc1Swenshuai.xi *
465*53ee8cc1Swenshuai.xi * @param pDev
466*53ee8cc1Swenshuai.xi * @param pIntf
467*53ee8cc1Swenshuai.xi *
468*53ee8cc1Swenshuai.xi * @return none
469*53ee8cc1Swenshuai.xi */
ms_usb_disable_interface(struct usb_device * pDev,struct usb_interface * pIntf)470*53ee8cc1Swenshuai.xi void ms_usb_disable_interface(struct usb_device *pDev, struct usb_interface *pIntf)
471*53ee8cc1Swenshuai.xi {
472*53ee8cc1Swenshuai.xi int EptIdx;
473*53ee8cc1Swenshuai.xi struct usb_host_interface *hintf = &pIntf->altsetting[pIntf->act_altsetting];
474*53ee8cc1Swenshuai.xi
475*53ee8cc1Swenshuai.xi for (EptIdx = 0; EptIdx < hintf->desc.bNumEndpoints; ++EptIdx)
476*53ee8cc1Swenshuai.xi ms_usb_disable_endpoint(pDev, hintf->endpoint[EptIdx].desc.bEndpointAddress);
477*53ee8cc1Swenshuai.xi }
478*53ee8cc1Swenshuai.xi
479*53ee8cc1Swenshuai.xi /**
480*53ee8cc1Swenshuai.xi * @brief disable the device
481*53ee8cc1Swenshuai.xi *
482*53ee8cc1Swenshuai.xi * @param pDev
483*53ee8cc1Swenshuai.xi * @param skip_ep0
484*53ee8cc1Swenshuai.xi *
485*53ee8cc1Swenshuai.xi * @return none
486*53ee8cc1Swenshuai.xi */
ms_usb_disable_device(struct usb_device * pDev,int skip_ep0)487*53ee8cc1Swenshuai.xi void ms_usb_disable_device(struct usb_device *pDev, int skip_ep0)
488*53ee8cc1Swenshuai.xi {
489*53ee8cc1Swenshuai.xi int EptIdx, i;
490*53ee8cc1Swenshuai.xi
491*53ee8cc1Swenshuai.xi if (pDev->actconfig)
492*53ee8cc1Swenshuai.xi {
493*53ee8cc1Swenshuai.xi for (i = 0; i < pDev->actconfig->desc.bNumInterfaces; i++)
494*53ee8cc1Swenshuai.xi {
495*53ee8cc1Swenshuai.xi struct usb_interface *pUiface;
496*53ee8cc1Swenshuai.xi
497*53ee8cc1Swenshuai.xi /* remove this interface */
498*53ee8cc1Swenshuai.xi pUiface = pDev->actconfig->interface[i];
499*53ee8cc1Swenshuai.xi ms_usbhost_debug("%s - unregistering interface %s\n",
500*53ee8cc1Swenshuai.xi pUdev->dev.dev_name, pUdev->dev.bus_id);
501*53ee8cc1Swenshuai.xi ms_device_unregister(&pUiface->dev);
502*53ee8cc1Swenshuai.xi }
503*53ee8cc1Swenshuai.xi /* patched from Linux */
504*53ee8cc1Swenshuai.xi pDev->actconfig = NULL;
505*53ee8cc1Swenshuai.xi if (pDev->eState == USB_STATE_CONFIGURED)
506*53ee8cc1Swenshuai.xi ms_usb_set_device_state(pDev, USB_STATE_ADDRESS);
507*53ee8cc1Swenshuai.xi }
508*53ee8cc1Swenshuai.xi
509*53ee8cc1Swenshuai.xi /* if check_bandwidth, that needs two phases of disable endpoint */
510*53ee8cc1Swenshuai.xi for (EptIdx = skip_ep0; EptIdx < 16; ++EptIdx)
511*53ee8cc1Swenshuai.xi {
512*53ee8cc1Swenshuai.xi ms_usb_disable_endpoint(pDev, EptIdx);
513*53ee8cc1Swenshuai.xi ms_usb_disable_endpoint(pDev, EptIdx + USB_DIR_IN);
514*53ee8cc1Swenshuai.xi }
515*53ee8cc1Swenshuai.xi }
516*53ee8cc1Swenshuai.xi
517*53ee8cc1Swenshuai.xi /**
518*53ee8cc1Swenshuai.xi * @brief enable the endpoint
519*53ee8cc1Swenshuai.xi *
520*53ee8cc1Swenshuai.xi * @param pDev
521*53ee8cc1Swenshuai.xi * @param pEpt_desc
522*53ee8cc1Swenshuai.xi *
523*53ee8cc1Swenshuai.xi * @return none
524*53ee8cc1Swenshuai.xi */
ms_usb_enable_endpoint(struct usb_device * pDev,struct usb_endpoint_descriptor * pEpt_desc)525*53ee8cc1Swenshuai.xi void ms_usb_enable_endpoint(struct usb_device *pDev,
526*53ee8cc1Swenshuai.xi struct usb_endpoint_descriptor *pEpt_desc)
527*53ee8cc1Swenshuai.xi {
528*53ee8cc1Swenshuai.xi unsigned int eptaddr = pEpt_desc->bEndpointAddress;
529*53ee8cc1Swenshuai.xi unsigned int eptnum = eptaddr & USB_ENDPOINT_NUMBER_MASK;
530*53ee8cc1Swenshuai.xi int is_control = ((pEpt_desc->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) ==
531*53ee8cc1Swenshuai.xi USB_ENDPOINT_XFER_CONTROL);
532*53ee8cc1Swenshuai.xi
533*53ee8cc1Swenshuai.xi if (usb_endpoint_dir(eptaddr) || is_control)
534*53ee8cc1Swenshuai.xi {
535*53ee8cc1Swenshuai.xi //usb_endpoint_running(pDev, eptnum, 1);
536*53ee8cc1Swenshuai.xi usb_settoggle(pDev, eptnum, 1, 0);
537*53ee8cc1Swenshuai.xi pDev->pEpMaxPacketOut[eptnum] = pEpt_desc->wMaxPacketSize;
538*53ee8cc1Swenshuai.xi }
539*53ee8cc1Swenshuai.xi
540*53ee8cc1Swenshuai.xi if (!usb_endpoint_dir(eptaddr) || is_control)
541*53ee8cc1Swenshuai.xi {
542*53ee8cc1Swenshuai.xi //usb_endpoint_running(pDev, eptnum, 0);
543*53ee8cc1Swenshuai.xi usb_settoggle(pDev, eptnum, 0, 0);
544*53ee8cc1Swenshuai.xi pDev->pEpMaxPacketIn[eptnum] = pEpt_desc->wMaxPacketSize;
545*53ee8cc1Swenshuai.xi }
546*53ee8cc1Swenshuai.xi }
547*53ee8cc1Swenshuai.xi
548*53ee8cc1Swenshuai.xi /**
549*53ee8cc1Swenshuai.xi * @brief enable the interface
550*53ee8cc1Swenshuai.xi *
551*53ee8cc1Swenshuai.xi * @param pDev
552*53ee8cc1Swenshuai.xi * @param pIntf
553*53ee8cc1Swenshuai.xi *
554*53ee8cc1Swenshuai.xi * @return none
555*53ee8cc1Swenshuai.xi */
ms_usb_enable_interface(struct usb_device * pDev,struct usb_interface * pIntf)556*53ee8cc1Swenshuai.xi void ms_usb_enable_interface(struct usb_device *pDev,
557*53ee8cc1Swenshuai.xi struct usb_interface *pIntf)
558*53ee8cc1Swenshuai.xi {
559*53ee8cc1Swenshuai.xi struct usb_host_interface *pAlt_set =
560*53ee8cc1Swenshuai.xi &pIntf->altsetting[pIntf->act_altsetting];
561*53ee8cc1Swenshuai.xi int i;
562*53ee8cc1Swenshuai.xi
563*53ee8cc1Swenshuai.xi for (i = 0; i < pAlt_set->desc.bNumEndpoints; ++i)
564*53ee8cc1Swenshuai.xi ms_usb_enable_endpoint(pDev, &pAlt_set->endpoint[i].desc);
565*53ee8cc1Swenshuai.xi }
566*53ee8cc1Swenshuai.xi
567*53ee8cc1Swenshuai.xi /**
568*53ee8cc1Swenshuai.xi * @brief set specific interface to the device
569*53ee8cc1Swenshuai.xi *
570*53ee8cc1Swenshuai.xi * @param pDev
571*53ee8cc1Swenshuai.xi * @param interface
572*53ee8cc1Swenshuai.xi * @param alternate
573*53ee8cc1Swenshuai.xi *
574*53ee8cc1Swenshuai.xi * @return function status
575*53ee8cc1Swenshuai.xi */
ms_usb_set_interface(struct usb_device * pDev,int interface,int alternate)576*53ee8cc1Swenshuai.xi int ms_usb_set_interface(struct usb_device *pDev, int interface, int alternate)
577*53ee8cc1Swenshuai.xi {
578*53ee8cc1Swenshuai.xi struct usb_interface *pIntf;
579*53ee8cc1Swenshuai.xi int ret;
580*53ee8cc1Swenshuai.xi int manual = 0;
581*53ee8cc1Swenshuai.xi
582*53ee8cc1Swenshuai.xi pIntf = ms_usb_ifnum_to_if(pDev, interface);
583*53ee8cc1Swenshuai.xi if (!pIntf)
584*53ee8cc1Swenshuai.xi {
585*53ee8cc1Swenshuai.xi ms_usbhost_err("Invalid interface %d", interface);
586*53ee8cc1Swenshuai.xi return -EINVAL;
587*53ee8cc1Swenshuai.xi }
588*53ee8cc1Swenshuai.xi
589*53ee8cc1Swenshuai.xi if (alternate < 0 || alternate >= pIntf->num_altsetting)
590*53ee8cc1Swenshuai.xi return -EINVAL;
591*53ee8cc1Swenshuai.xi
592*53ee8cc1Swenshuai.xi ret = ms_usb_control_cmd(pDev, usb_sndctrlpipe(pDev, 0),
593*53ee8cc1Swenshuai.xi USB_REQ_SET_INTERFACE, USB_RECIP_INTERFACE,
594*53ee8cc1Swenshuai.xi pIntf->altsetting[alternate]
595*53ee8cc1Swenshuai.xi .desc.bAlternateSetting,
596*53ee8cc1Swenshuai.xi interface, NULL, 0, HZ * 5);
597*53ee8cc1Swenshuai.xi
598*53ee8cc1Swenshuai.xi if (ret == -EPIPE && pIntf->num_altsetting == 1)
599*53ee8cc1Swenshuai.xi {
600*53ee8cc1Swenshuai.xi ms_debug_msg("manual set interface for pIntf %d, alt %d",
601*53ee8cc1Swenshuai.xi interface, alternate);
602*53ee8cc1Swenshuai.xi manual = 1;
603*53ee8cc1Swenshuai.xi }
604*53ee8cc1Swenshuai.xi else if (ret < 0)
605*53ee8cc1Swenshuai.xi return ret;
606*53ee8cc1Swenshuai.xi
607*53ee8cc1Swenshuai.xi ms_usb_disable_interface(pDev, pIntf);
608*53ee8cc1Swenshuai.xi
609*53ee8cc1Swenshuai.xi pIntf->act_altsetting = alternate;
610*53ee8cc1Swenshuai.xi
611*53ee8cc1Swenshuai.xi if (manual)
612*53ee8cc1Swenshuai.xi {
613*53ee8cc1Swenshuai.xi struct usb_host_interface *iface_as = &pIntf->altsetting[alternate];
614*53ee8cc1Swenshuai.xi int ii;
615*53ee8cc1Swenshuai.xi
616*53ee8cc1Swenshuai.xi for (ii = 0; ii < iface_as->desc.bNumEndpoints; ii++)
617*53ee8cc1Swenshuai.xi {
618*53ee8cc1Swenshuai.xi unsigned int epaddr = iface_as->endpoint[ii].desc.bEndpointAddress;
619*53ee8cc1Swenshuai.xi unsigned int pipe =
620*53ee8cc1Swenshuai.xi __pack_pipe(pDev, USB_ENDPOINT_NUMBER_MASK & epaddr)
621*53ee8cc1Swenshuai.xi | (usb_endpoint_dir(epaddr) ? USB_DIR_OUT : USB_DIR_IN);
622*53ee8cc1Swenshuai.xi
623*53ee8cc1Swenshuai.xi ms_usb_clear_halt(pDev, pipe);
624*53ee8cc1Swenshuai.xi }
625*53ee8cc1Swenshuai.xi }
626*53ee8cc1Swenshuai.xi
627*53ee8cc1Swenshuai.xi ms_usb_enable_interface(pDev, pIntf);
628*53ee8cc1Swenshuai.xi
629*53ee8cc1Swenshuai.xi return 0;
630*53ee8cc1Swenshuai.xi }
631*53ee8cc1Swenshuai.xi
632*53ee8cc1Swenshuai.xi /**
633*53ee8cc1Swenshuai.xi * @brief set specific configuration to the device
634*53ee8cc1Swenshuai.xi *
635*53ee8cc1Swenshuai.xi * @param pDev
636*53ee8cc1Swenshuai.xi * @param config_val
637*53ee8cc1Swenshuai.xi *
638*53ee8cc1Swenshuai.xi * @return function status
639*53ee8cc1Swenshuai.xi */
ms_usb_set_config(struct usb_device * pDev,int config_val)640*53ee8cc1Swenshuai.xi int ms_usb_set_config(struct usb_device *pDev, int config_val)
641*53ee8cc1Swenshuai.xi {
642*53ee8cc1Swenshuai.xi int i, ret;
643*53ee8cc1Swenshuai.xi struct usb_host_config *cp = NULL;
644*53ee8cc1Swenshuai.xi
645*53ee8cc1Swenshuai.xi for (i=0; i<pDev->descriptor.bNumConfigurations; i++)
646*53ee8cc1Swenshuai.xi {
647*53ee8cc1Swenshuai.xi if (pDev->config[i].desc.bConfigurationValue == config_val)
648*53ee8cc1Swenshuai.xi {
649*53ee8cc1Swenshuai.xi cp = &pDev->config[i];
650*53ee8cc1Swenshuai.xi break;
651*53ee8cc1Swenshuai.xi }
652*53ee8cc1Swenshuai.xi }
653*53ee8cc1Swenshuai.xi if ((!cp && config_val != 0) || (cp && config_val == 0))
654*53ee8cc1Swenshuai.xi {
655*53ee8cc1Swenshuai.xi ms_usbhost_err("selecting invalid config_val %d", config_val);
656*53ee8cc1Swenshuai.xi return -EINVAL;
657*53ee8cc1Swenshuai.xi }
658*53ee8cc1Swenshuai.xi
659*53ee8cc1Swenshuai.xi if (pDev->eState != USB_STATE_ADDRESS)
660*53ee8cc1Swenshuai.xi ms_usb_disable_device (pDev, 1); // Skip ep0
661*53ee8cc1Swenshuai.xi pDev->toggle[0] = pDev->toggle[1] = 0;
662*53ee8cc1Swenshuai.xi //pDev->halted[0] = pDev->halted[1] = 0; // Obsolete
663*53ee8cc1Swenshuai.xi //pDev->eState = USB_STATE_ADDRESS;
664*53ee8cc1Swenshuai.xi ms_usb_set_device_state(pDev, USB_STATE_ADDRESS);
665*53ee8cc1Swenshuai.xi
666*53ee8cc1Swenshuai.xi if ((ret = ms_usb_control_cmd(pDev, usb_sndctrlpipe(pDev, 0),
667*53ee8cc1Swenshuai.xi USB_REQ_SET_CONFIGURATION, 0, config_val, 0,
668*53ee8cc1Swenshuai.xi NULL, 0, HZ * USB_CTRL_SET_TIMEOUT)) < 0)
669*53ee8cc1Swenshuai.xi return ret;
670*53ee8cc1Swenshuai.xi if (config_val)
671*53ee8cc1Swenshuai.xi //pDev->eState = USB_STATE_CONFIGURED;
672*53ee8cc1Swenshuai.xi ms_usb_set_device_state(pDev, USB_STATE_CONFIGURED);
673*53ee8cc1Swenshuai.xi pDev->actconfig = cp;
674*53ee8cc1Swenshuai.xi
675*53ee8cc1Swenshuai.xi for (i = 0; i < cp->desc.bNumInterfaces; ++i)
676*53ee8cc1Swenshuai.xi {
677*53ee8cc1Swenshuai.xi struct usb_interface *intf = cp->interface[i];
678*53ee8cc1Swenshuai.xi
679*53ee8cc1Swenshuai.xi intf->act_altsetting = 0;
680*53ee8cc1Swenshuai.xi ms_usb_enable_interface(pDev, intf);
681*53ee8cc1Swenshuai.xi }
682*53ee8cc1Swenshuai.xi
683*53ee8cc1Swenshuai.xi return 0;
684*53ee8cc1Swenshuai.xi }
685*53ee8cc1Swenshuai.xi
686*53ee8cc1Swenshuai.xi
687*53ee8cc1Swenshuai.xi /**
688*53ee8cc1Swenshuai.xi * @brief get the string from the device
689*53ee8cc1Swenshuai.xi *
690*53ee8cc1Swenshuai.xi * @param pDev
691*53ee8cc1Swenshuai.xi * @param index
692*53ee8cc1Swenshuai.xi * @param pBuf
693*53ee8cc1Swenshuai.xi * @param size
694*53ee8cc1Swenshuai.xi *
695*53ee8cc1Swenshuai.xi * @return function status
696*53ee8cc1Swenshuai.xi */
ms_usb_string(struct usb_device * pDev,int index,char * pBuf,size_t size)697*53ee8cc1Swenshuai.xi int ms_usb_string(struct usb_device *pDev, int index, char *pBuf, size_t size)
698*53ee8cc1Swenshuai.xi {
699*53ee8cc1Swenshuai.xi unsigned char *pTmp_buf;
700*53ee8cc1Swenshuai.xi int err, len;
701*53ee8cc1Swenshuai.xi U32 u, idx;
702*53ee8cc1Swenshuai.xi void *addr_t;
703*53ee8cc1Swenshuai.xi
704*53ee8cc1Swenshuai.xi if ( (int)size <= 0 || !pBuf || !index)
705*53ee8cc1Swenshuai.xi return -EINVAL;
706*53ee8cc1Swenshuai.xi
707*53ee8cc1Swenshuai.xi pBuf[0] = 0;
708*53ee8cc1Swenshuai.xi pTmp_buf = (unsigned char*) kmalloc(256, GFP_KERNEL);
709*53ee8cc1Swenshuai.xi
710*53ee8cc1Swenshuai.xi if (!pTmp_buf)
711*53ee8cc1Swenshuai.xi return -ENOMEM;
712*53ee8cc1Swenshuai.xi
713*53ee8cc1Swenshuai.xi addr_t = (void *)pTmp_buf;
714*53ee8cc1Swenshuai.xi /* get langid for strings if it's not yet known */
715*53ee8cc1Swenshuai.xi if (!pDev->u32HaveLangId)
716*53ee8cc1Swenshuai.xi {
717*53ee8cc1Swenshuai.xi err = ms_usb_control_cmd(pDev, usb_rcvctrlpipe(pDev, 0),
718*53ee8cc1Swenshuai.xi USB_REQ_GET_DESCRIPTOR, USB_DIR_IN,
719*53ee8cc1Swenshuai.xi (USB_DT_STRING << 8), 0, pTmp_buf, 4,
720*53ee8cc1Swenshuai.xi HZ * USB_CTRL_GET_TIMEOUT);
721*53ee8cc1Swenshuai.xi
722*53ee8cc1Swenshuai.xi if (err < 0)
723*53ee8cc1Swenshuai.xi {
724*53ee8cc1Swenshuai.xi ms_debug_err("error getting string descriptor 0 (error=%d)\n", err);
725*53ee8cc1Swenshuai.xi goto errout;
726*53ee8cc1Swenshuai.xi }
727*53ee8cc1Swenshuai.xi else if (err < 4 || pTmp_buf[0] < 4)
728*53ee8cc1Swenshuai.xi {
729*53ee8cc1Swenshuai.xi ms_debug_msg("string descriptor 0 too short!\n");
730*53ee8cc1Swenshuai.xi err = -EINVAL;
731*53ee8cc1Swenshuai.xi goto errout;
732*53ee8cc1Swenshuai.xi }
733*53ee8cc1Swenshuai.xi else
734*53ee8cc1Swenshuai.xi {
735*53ee8cc1Swenshuai.xi pDev->u32HaveLangId = -1;
736*53ee8cc1Swenshuai.xi pDev->u32StringLangId = pTmp_buf[2] | (pTmp_buf[3]<< 8);
737*53ee8cc1Swenshuai.xi /* always use the first langid listed */
738*53ee8cc1Swenshuai.xi ms_debug_debug("USB device number %d default language ID 0x%x\n",
739*53ee8cc1Swenshuai.xi pDev->u32DevNum, pDev->u32StringLangId);
740*53ee8cc1Swenshuai.xi }
741*53ee8cc1Swenshuai.xi }
742*53ee8cc1Swenshuai.xi
743*53ee8cc1Swenshuai.xi err = ms_usb_control_cmd(pDev, usb_rcvctrlpipe(pDev, 0),
744*53ee8cc1Swenshuai.xi USB_REQ_GET_DESCRIPTOR, USB_DIR_IN,
745*53ee8cc1Swenshuai.xi (USB_DT_STRING << 8) + index, pDev->u32StringLangId, pTmp_buf, 2,
746*53ee8cc1Swenshuai.xi HZ * USB_CTRL_GET_TIMEOUT);
747*53ee8cc1Swenshuai.xi
748*53ee8cc1Swenshuai.xi if(err<2)
749*53ee8cc1Swenshuai.xi goto errout;
750*53ee8cc1Swenshuai.xi else if (pTmp_buf[1]!=3)
751*53ee8cc1Swenshuai.xi {
752*53ee8cc1Swenshuai.xi err = -EINVAL;
753*53ee8cc1Swenshuai.xi goto errout;
754*53ee8cc1Swenshuai.xi }
755*53ee8cc1Swenshuai.xi len=pTmp_buf[0];
756*53ee8cc1Swenshuai.xi
757*53ee8cc1Swenshuai.xi err = ms_usb_control_cmd(pDev, usb_rcvctrlpipe(pDev, 0),
758*53ee8cc1Swenshuai.xi USB_REQ_GET_DESCRIPTOR, USB_DIR_IN,
759*53ee8cc1Swenshuai.xi (USB_DT_STRING << 8) + index, pDev->u32StringLangId, pTmp_buf, len,
760*53ee8cc1Swenshuai.xi HZ * USB_CTRL_GET_TIMEOUT);
761*53ee8cc1Swenshuai.xi
762*53ee8cc1Swenshuai.xi if (err < 0)
763*53ee8cc1Swenshuai.xi goto errout;
764*53ee8cc1Swenshuai.xi
765*53ee8cc1Swenshuai.xi size--;
766*53ee8cc1Swenshuai.xi for (idx = 0, u = 2; u < (U32) err; u += 2)
767*53ee8cc1Swenshuai.xi {
768*53ee8cc1Swenshuai.xi if (idx >= size)
769*53ee8cc1Swenshuai.xi break;
770*53ee8cc1Swenshuai.xi if (pTmp_buf[u+1])
771*53ee8cc1Swenshuai.xi pBuf[idx++] = '?';
772*53ee8cc1Swenshuai.xi else
773*53ee8cc1Swenshuai.xi pBuf[idx++] = pTmp_buf[u];
774*53ee8cc1Swenshuai.xi }
775*53ee8cc1Swenshuai.xi pBuf[idx] = 0;
776*53ee8cc1Swenshuai.xi err = idx;
777*53ee8cc1Swenshuai.xi
778*53ee8cc1Swenshuai.xi errout:
779*53ee8cc1Swenshuai.xi kfree(addr_t);
780*53ee8cc1Swenshuai.xi return err;
781*53ee8cc1Swenshuai.xi }
782