SMS Scheduler that will automatically send and receive text messages using the Enfora 1308 GSM Modem. Please note that it uses a modified NetServices library and to set the baud rate for the GSM Modem to 19.2K.

Dependencies:   mbed

Committer:
mafischl
Date:
Thu Oct 13 18:01:31 2011 +0000
Revision:
1:5a7cce9994a3
Parent:
0:d9266031f832

        

Who changed what in which revision?

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