NetServices Stack source

Dependents:   HelloWorld ServoInterfaceBoardExample1 4180_Lab4

Committer:
donatien
Date:
Fri Jul 09 14:46:47 2010 +0000
Revision:
4:fd826cad83c0
Child:
5:dd63a1e02b1b

        

Who changed what in which revision?

UserRevisionLine numberNew contents of line
donatien 4:fd826cad83c0 1
donatien 4:fd826cad83c0 2 /*
donatien 4:fd826cad83c0 3 Copyright (c) 2010 Donatien Garnier (donatiengar [at] gmail [dot] com)
donatien 4:fd826cad83c0 4
donatien 4:fd826cad83c0 5 Permission is hereby granted, free of charge, to any person obtaining a copy
donatien 4:fd826cad83c0 6 of this software and associated documentation files (the "Software"), to deal
donatien 4:fd826cad83c0 7 in the Software without restriction, including without limitation the rights
donatien 4:fd826cad83c0 8 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
donatien 4:fd826cad83c0 9 copies of the Software, and to permit persons to whom the Software is
donatien 4:fd826cad83c0 10 furnished to do so, subject to the following conditions:
donatien 4:fd826cad83c0 11
donatien 4:fd826cad83c0 12 The above copyright notice and this permission notice shall be included in
donatien 4:fd826cad83c0 13 all copies or substantial portions of the Software.
donatien 4:fd826cad83c0 14
donatien 4:fd826cad83c0 15 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
donatien 4:fd826cad83c0 16 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
donatien 4:fd826cad83c0 17 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
donatien 4:fd826cad83c0 18 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
donatien 4:fd826cad83c0 19 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
donatien 4:fd826cad83c0 20 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
donatien 4:fd826cad83c0 21 THE SOFTWARE.
donatien 4:fd826cad83c0 22 */
donatien 4:fd826cad83c0 23
donatien 4:fd826cad83c0 24 #include "netCfg.h"
donatien 4:fd826cad83c0 25 #if NET_UMTS
donatien 4:fd826cad83c0 26
donatien 4:fd826cad83c0 27 #include "UMTSStick.h"
donatien 4:fd826cad83c0 28
donatien 4:fd826cad83c0 29 #define __DEBUG
donatien 4:fd826cad83c0 30 #include "dbg/dbg.h"
donatien 4:fd826cad83c0 31
donatien 4:fd826cad83c0 32 UMTSStick::UMTSStick() : m_host(), m_pDev(NULL)
donatien 4:fd826cad83c0 33 {
donatien 4:fd826cad83c0 34
donatien 4:fd826cad83c0 35 }
donatien 4:fd826cad83c0 36
donatien 4:fd826cad83c0 37 UMTSStick::~UMTSStick()
donatien 4:fd826cad83c0 38 {
donatien 4:fd826cad83c0 39
donatien 4:fd826cad83c0 40 }
donatien 4:fd826cad83c0 41
donatien 4:fd826cad83c0 42
donatien 4:fd826cad83c0 43 UMTSStickErr UMTSStick::getSerial(UsbSerial** ppUsbSerial)
donatien 4:fd826cad83c0 44 {
donatien 4:fd826cad83c0 45 m_host.init();
donatien 4:fd826cad83c0 46
donatien 4:fd826cad83c0 47 UMTSStickErr rc;
donatien 4:fd826cad83c0 48
donatien 4:fd826cad83c0 49 rc = waitForDevice();
donatien 4:fd826cad83c0 50 if(rc)
donatien 4:fd826cad83c0 51 return rc;
donatien 4:fd826cad83c0 52
donatien 4:fd826cad83c0 53 //Device is now enumerated, read table
donatien 4:fd826cad83c0 54
donatien 4:fd826cad83c0 55 uint16_t vid = m_pDev->getVid();
donatien 4:fd826cad83c0 56 uint16_t pid = m_pDev->getPid();
donatien 4:fd826cad83c0 57
donatien 4:fd826cad83c0 58 DBG("Configuration set: vid:%04x pid:%04x\n", vid, pid);
donatien 4:fd826cad83c0 59
donatien 4:fd826cad83c0 60 bool handled = false;
donatien 4:fd826cad83c0 61 bool cdfs = false;
donatien 4:fd826cad83c0 62 const UMTSSwitchingInfo* pInfo;
donatien 4:fd826cad83c0 63 for(int i = 0; i < UMTS_SWITCHING_COUNT; i++)
donatien 4:fd826cad83c0 64 {
donatien 4:fd826cad83c0 65 pInfo = &UMTSwitchingTable[i];
donatien 4:fd826cad83c0 66 if( !checkDeviceState(pInfo, &cdfs) )
donatien 4:fd826cad83c0 67 {
donatien 4:fd826cad83c0 68 handled = true;
donatien 4:fd826cad83c0 69 break;
donatien 4:fd826cad83c0 70 }
donatien 4:fd826cad83c0 71
donatien 4:fd826cad83c0 72 } //for(int i = 0; i < UMTS_SWITCHING_COUNT; i++)
donatien 4:fd826cad83c0 73
donatien 4:fd826cad83c0 74 if(!handled)
donatien 4:fd826cad83c0 75 {
donatien 4:fd826cad83c0 76 DBG("Don't know this device!\n");
donatien 4:fd826cad83c0 77 return UMTSERR_NOTIMPLEMENTED;
donatien 4:fd826cad83c0 78 }
donatien 4:fd826cad83c0 79
donatien 4:fd826cad83c0 80 //Check if the device is in CDFS mode, in this case switch
donatien 4:fd826cad83c0 81 if(cdfs)
donatien 4:fd826cad83c0 82 {
donatien 4:fd826cad83c0 83 DBG("Switching the device by sending a magic packet\n");
donatien 4:fd826cad83c0 84
donatien 4:fd826cad83c0 85 rc = switchMode(pInfo);
donatien 4:fd826cad83c0 86 if(rc)
donatien 4:fd826cad83c0 87 return rc;
donatien 4:fd826cad83c0 88
donatien 4:fd826cad83c0 89 DBG("Now wait for device to reconnect\n");
donatien 4:fd826cad83c0 90
donatien 4:fd826cad83c0 91 m_host.releaseDevice(m_pDev);
donatien 4:fd826cad83c0 92
donatien 4:fd826cad83c0 93 //Wait for device to reconnect
donatien 4:fd826cad83c0 94 wait(3);
donatien 4:fd826cad83c0 95 rc = waitForDevice();
donatien 4:fd826cad83c0 96 if(rc)
donatien 4:fd826cad83c0 97 return rc;
donatien 4:fd826cad83c0 98 }
donatien 4:fd826cad83c0 99
donatien 4:fd826cad83c0 100 rc = findSerial(ppUsbSerial);
donatien 4:fd826cad83c0 101 if(rc)
donatien 4:fd826cad83c0 102 return rc;
donatien 4:fd826cad83c0 103
donatien 4:fd826cad83c0 104 return UMTSERR_OK;
donatien 4:fd826cad83c0 105 }
donatien 4:fd826cad83c0 106
donatien 4:fd826cad83c0 107 UMTSStickErr UMTSStick::waitForDevice()
donatien 4:fd826cad83c0 108 {
donatien 4:fd826cad83c0 109 bool ready = false;
donatien 4:fd826cad83c0 110 while(!ready)
donatien 4:fd826cad83c0 111 {
donatien 4:fd826cad83c0 112 while(!m_host.devicesCount())
donatien 4:fd826cad83c0 113 {}
donatien 4:fd826cad83c0 114 wait(1);
donatien 4:fd826cad83c0 115 if(m_host.devicesCount())
donatien 4:fd826cad83c0 116 ready = true;
donatien 4:fd826cad83c0 117 }
donatien 4:fd826cad83c0 118
donatien 4:fd826cad83c0 119 m_pDev = m_host.getDevice(0);
donatien 4:fd826cad83c0 120
donatien 4:fd826cad83c0 121 while(!m_pDev->enumerated())
donatien 4:fd826cad83c0 122 {
donatien 4:fd826cad83c0 123 m_host.poll();
donatien 4:fd826cad83c0 124 if(!m_host.devicesCount())
donatien 4:fd826cad83c0 125 return UMTSERR_DISCONNECTED;
donatien 4:fd826cad83c0 126 }
donatien 4:fd826cad83c0 127
donatien 4:fd826cad83c0 128 wait(3);
donatien 4:fd826cad83c0 129
donatien 4:fd826cad83c0 130 return UMTSERR_OK;
donatien 4:fd826cad83c0 131 }
donatien 4:fd826cad83c0 132
donatien 4:fd826cad83c0 133 UMTSStickErr UMTSStick::checkDeviceState(const UMTSSwitchingInfo* pInfo, bool* pCdfs)
donatien 4:fd826cad83c0 134 {
donatien 4:fd826cad83c0 135 uint16_t vid = m_pDev->getVid();
donatien 4:fd826cad83c0 136 uint16_t pid = m_pDev->getPid();
donatien 4:fd826cad83c0 137 bool handled = false;
donatien 4:fd826cad83c0 138 if( (vid == pInfo->cdfsVid) && (pid == pInfo->cdfsPid) )
donatien 4:fd826cad83c0 139 {
donatien 4:fd826cad83c0 140 DBG("Match on dongles list\n");
donatien 4:fd826cad83c0 141 if( !pInfo->targetClass ) //No specific interface to check, vid/pid couple is specific to CDFS mode
donatien 4:fd826cad83c0 142 {
donatien 4:fd826cad83c0 143 DBG("Found device in CDFS mode\n");
donatien 4:fd826cad83c0 144 handled = true;
donatien 4:fd826cad83c0 145 *pCdfs = true;
donatien 4:fd826cad83c0 146 }
donatien 4:fd826cad83c0 147 else //if( !pInfo->targetClass )
donatien 4:fd826cad83c0 148 {
donatien 4:fd826cad83c0 149 //Has to check if there is an interface of class targetClass
donatien 4:fd826cad83c0 150 byte* desc = NULL;
donatien 4:fd826cad83c0 151 int c = 0;
donatien 4:fd826cad83c0 152
donatien 4:fd826cad83c0 153 while( !m_pDev->getInterfaceDescriptor(1, c++, &desc) )
donatien 4:fd826cad83c0 154 {
donatien 4:fd826cad83c0 155 if( desc[5] == pInfo->targetClass )
donatien 4:fd826cad83c0 156 {
donatien 4:fd826cad83c0 157 DBG("Found device in Serial mode\n");
donatien 4:fd826cad83c0 158 handled = true;
donatien 4:fd826cad83c0 159 *pCdfs = false;
donatien 4:fd826cad83c0 160 break;
donatien 4:fd826cad83c0 161 }
donatien 4:fd826cad83c0 162 }
donatien 4:fd826cad83c0 163
donatien 4:fd826cad83c0 164 if(!handled)
donatien 4:fd826cad83c0 165 {
donatien 4:fd826cad83c0 166 //All interfaces were tried, so we are in CDFS mode
donatien 4:fd826cad83c0 167 DBG("Found device in CDFS mode\n");
donatien 4:fd826cad83c0 168 handled = true;
donatien 4:fd826cad83c0 169 *pCdfs = true;
donatien 4:fd826cad83c0 170 }
donatien 4:fd826cad83c0 171 } //if( !pInfo->targetClass )
donatien 4:fd826cad83c0 172 } //if( (vid == pInfo->cdfsVid) && (pid == pInfo->cdfsPid) )
donatien 4:fd826cad83c0 173 else
donatien 4:fd826cad83c0 174 {
donatien 4:fd826cad83c0 175 //Try every vid/pid couple of the serial list
donatien 4:fd826cad83c0 176 for( int i = 0; i < 16 ; i++)
donatien 4:fd826cad83c0 177 {
donatien 4:fd826cad83c0 178 if(!pInfo->serialPidList[i])
donatien 4:fd826cad83c0 179 break;
donatien 4:fd826cad83c0 180 if( (pInfo->serialVid == vid) && (pInfo->serialPidList[i] == pid) )
donatien 4:fd826cad83c0 181 {
donatien 4:fd826cad83c0 182 DBG("Found device in Serial mode\n");
donatien 4:fd826cad83c0 183 handled = true;
donatien 4:fd826cad83c0 184 *pCdfs = false;
donatien 4:fd826cad83c0 185 break;
donatien 4:fd826cad83c0 186 }
donatien 4:fd826cad83c0 187 }
donatien 4:fd826cad83c0 188 } //if( (vid == pInfo->cdfsVid) && (pid == pInfo->cdfsPid) )
donatien 4:fd826cad83c0 189
donatien 4:fd826cad83c0 190 if(!handled)
donatien 4:fd826cad83c0 191 return UMTSERR_NOTFOUND;
donatien 4:fd826cad83c0 192
donatien 4:fd826cad83c0 193 return UMTSERR_OK;
donatien 4:fd826cad83c0 194 }
donatien 4:fd826cad83c0 195
donatien 4:fd826cad83c0 196 UMTSStickErr UMTSStick::switchMode(const UMTSSwitchingInfo* pInfo)
donatien 4:fd826cad83c0 197 {
donatien 4:fd826cad83c0 198 if(!pInfo->huaweiPacket) //Send SCSI packet on first bulk ep
donatien 4:fd826cad83c0 199 {
donatien 4:fd826cad83c0 200 //Find first bulk ep
donatien 4:fd826cad83c0 201 byte* desc = NULL;
donatien 4:fd826cad83c0 202 int c = 0;
donatien 4:fd826cad83c0 203
donatien 4:fd826cad83c0 204 UsbEndpoint *pEpOut = NULL;
donatien 4:fd826cad83c0 205
donatien 4:fd826cad83c0 206 while( !m_pDev->getInterfaceDescriptor(1, c++, &desc) )
donatien 4:fd826cad83c0 207 {
donatien 4:fd826cad83c0 208 byte* p = desc;
donatien 4:fd826cad83c0 209 int epNum = 0;
donatien 4:fd826cad83c0 210 p = p + p[0]; //Move to next descriptor (which should be an ep descriptor)
donatien 4:fd826cad83c0 211 while (epNum < desc[4]) //Eps count in this if
donatien 4:fd826cad83c0 212 {
donatien 4:fd826cad83c0 213 if (p[1] != USB_DESCRIPTOR_TYPE_ENDPOINT)
donatien 4:fd826cad83c0 214 break;
donatien 4:fd826cad83c0 215
donatien 4:fd826cad83c0 216 if( (p[3] == 0x02) && !(p[2] & 0x80) ) //Bulk endpoint, out
donatien 4:fd826cad83c0 217 {
donatien 4:fd826cad83c0 218 DBG("Found bulk ep %02x\n", p[2]);
donatien 4:fd826cad83c0 219 pEpOut = new UsbEndpoint( m_pDev, p[2], false, USB_BULK, *((uint16_t*)&p[4]) );
donatien 4:fd826cad83c0 220 break;
donatien 4:fd826cad83c0 221 }
donatien 4:fd826cad83c0 222
donatien 4:fd826cad83c0 223 p = p + p[0]; //Move to next ep desc
donatien 4:fd826cad83c0 224 epNum++;
donatien 4:fd826cad83c0 225 }
donatien 4:fd826cad83c0 226 if(pEpOut)
donatien 4:fd826cad83c0 227 break;
donatien 4:fd826cad83c0 228 }
donatien 4:fd826cad83c0 229
donatien 4:fd826cad83c0 230 if(!pEpOut)
donatien 4:fd826cad83c0 231 return UMTSERR_NOTFOUND;
donatien 4:fd826cad83c0 232
donatien 4:fd826cad83c0 233 //Send SCSI packet
donatien 4:fd826cad83c0 234
donatien 4:fd826cad83c0 235 DBG("Sending SCSI Packet to switch\n");
donatien 4:fd826cad83c0 236 byte ramCdfsBuf[31];
donatien 4:fd826cad83c0 237 memcpy(ramCdfsBuf, pInfo->cdfsPacket, 31);
donatien 4:fd826cad83c0 238 pEpOut->transfer((volatile byte*)ramCdfsBuf, 31);
donatien 4:fd826cad83c0 239 while(pEpOut->status() == USBERR_PROCESSING);
donatien 4:fd826cad83c0 240 int ret = pEpOut->status();
donatien 4:fd826cad83c0 241 if((ret < 0) && (ret !=USBERR_DISCONNECTED)) //Packet was not transfered
donatien 4:fd826cad83c0 242 {
donatien 4:fd826cad83c0 243 DBG("Usb error %d\n", ret);
donatien 4:fd826cad83c0 244 delete pEpOut;
donatien 4:fd826cad83c0 245 return UMTSERR_USBERR;
donatien 4:fd826cad83c0 246 }
donatien 4:fd826cad83c0 247
donatien 4:fd826cad83c0 248 delete pEpOut;
donatien 4:fd826cad83c0 249 }
donatien 4:fd826cad83c0 250 else
donatien 4:fd826cad83c0 251 {
donatien 4:fd826cad83c0 252 UsbErr usbErr;
donatien 4:fd826cad83c0 253 //Send the Huawei-specific control packet
donatien 4:fd826cad83c0 254 usbErr = m_pDev->controlSend(0, 0x03, 1, 0, NULL, 0);
donatien 4:fd826cad83c0 255 if(usbErr && (usbErr != USBERR_DISCONNECTED))
donatien 4:fd826cad83c0 256 return UMTSERR_USBERR;
donatien 4:fd826cad83c0 257 }
donatien 4:fd826cad83c0 258
donatien 4:fd826cad83c0 259 DBG("The stick should be switching in serial mode now\n");
donatien 4:fd826cad83c0 260
donatien 4:fd826cad83c0 261 return UMTSERR_OK;
donatien 4:fd826cad83c0 262 }
donatien 4:fd826cad83c0 263
donatien 4:fd826cad83c0 264 UMTSStickErr UMTSStick::findSerial(UsbSerial** ppUsbSerial)
donatien 4:fd826cad83c0 265 {
donatien 4:fd826cad83c0 266 byte* desc = NULL;
donatien 4:fd826cad83c0 267 int c = 0;
donatien 4:fd826cad83c0 268
donatien 4:fd826cad83c0 269 int epOut = 0;
donatien 4:fd826cad83c0 270 int epIn = 0;
donatien 4:fd826cad83c0 271
donatien 4:fd826cad83c0 272 while( !m_pDev->getInterfaceDescriptor(1, c++, &desc) )
donatien 4:fd826cad83c0 273 {
donatien 4:fd826cad83c0 274 byte* p = desc;
donatien 4:fd826cad83c0 275 int epNum = 0;
donatien 4:fd826cad83c0 276
donatien 4:fd826cad83c0 277 DBG("Interface of type %02x\n", desc[5]);
donatien 4:fd826cad83c0 278
donatien 4:fd826cad83c0 279 if(desc[5] != 0xFF) //Not a serial-like if
donatien 4:fd826cad83c0 280 continue;
donatien 4:fd826cad83c0 281
donatien 4:fd826cad83c0 282 p = p + p[0]; //Move to next descriptor (which should be an ep descriptor)
donatien 4:fd826cad83c0 283 while (epNum < desc[4]) //Eps count in this if
donatien 4:fd826cad83c0 284 {
donatien 4:fd826cad83c0 285 if (p[1] == USB_DESCRIPTOR_TYPE_ENDPOINT)
donatien 4:fd826cad83c0 286 {
donatien 4:fd826cad83c0 287 if( (p[3] == 0x02) && !(p[2] & 0x80) && !epOut ) //Bulk endpoint, out
donatien 4:fd826cad83c0 288 {
donatien 4:fd826cad83c0 289 DBG("Found bulk out ep %02x of payload size %04x\n", p[2], *((uint16_t*)&p[4]));
donatien 4:fd826cad83c0 290 epOut = p[2] & 0x7F;
donatien 4:fd826cad83c0 291 }
donatien 4:fd826cad83c0 292
donatien 4:fd826cad83c0 293 if( (p[3] == 0x02) && (p[2] & 0x80) && !epIn ) //Bulk endpoint, in
donatien 4:fd826cad83c0 294 {
donatien 4:fd826cad83c0 295 DBG("Found bulk in ep %02x of payload size %04x\n", p[2], *((uint16_t*)&p[4]));
donatien 4:fd826cad83c0 296 epIn = p[2] & 0x7F;
donatien 4:fd826cad83c0 297 }
donatien 4:fd826cad83c0 298
donatien 4:fd826cad83c0 299 if(epOut && epIn)
donatien 4:fd826cad83c0 300 break;
donatien 4:fd826cad83c0 301 }
donatien 4:fd826cad83c0 302
donatien 4:fd826cad83c0 303 p = p + p[0]; //Move to next ep desc
donatien 4:fd826cad83c0 304 epNum++;
donatien 4:fd826cad83c0 305 }
donatien 4:fd826cad83c0 306
donatien 4:fd826cad83c0 307 if(epOut && epIn)
donatien 4:fd826cad83c0 308 break;
donatien 4:fd826cad83c0 309 }
donatien 4:fd826cad83c0 310
donatien 4:fd826cad83c0 311 if(!epOut || !epIn)
donatien 4:fd826cad83c0 312 return UMTSERR_NOTFOUND;
donatien 4:fd826cad83c0 313
donatien 4:fd826cad83c0 314 DBG("Endpoints found, create serial object\n");
donatien 4:fd826cad83c0 315
donatien 4:fd826cad83c0 316 *ppUsbSerial = new UsbSerial(m_pDev, epIn, epOut);
donatien 4:fd826cad83c0 317
donatien 4:fd826cad83c0 318 DBG("UsbSerial object created\n");
donatien 4:fd826cad83c0 319
donatien 4:fd826cad83c0 320 return UMTSERR_OK;
donatien 4:fd826cad83c0 321 }
donatien 4:fd826cad83c0 322
donatien 4:fd826cad83c0 323 #endif