xref: /utopia/UTPA2-700.0.x/modules/usb/drv/usb_ecos/newhost/drvMsg.c (revision 53ee8cc121a030b8d368113ac3e966b4705770ef)
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