FreeCalypso > hg > tcs211-fcmodem
comparison g23m/condat/ms/src/aci/cmh_mmf.c @ 0:509db1a7b7b8
initial import: leo2moko-r1
author | Space Falcon <falcon@ivan.Harhan.ORG> |
---|---|
date | Mon, 01 Jun 2015 03:24:05 +0000 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 0:509db1a7b7b8 |
---|---|
1 /* | |
2 +----------------------------------------------------------------------------- | |
3 | Project : GSM-PS (6147) | |
4 | Modul : CMH_MMF | |
5 +----------------------------------------------------------------------------- | |
6 | Copyright 2002 Texas Instruments Berlin, AG | |
7 | All rights reserved. | |
8 | | |
9 | This file is confidential and a trade secret of Texas | |
10 | Instruments Berlin, AG | |
11 | The receipt of or possession of this file does not convey | |
12 | any rights to reproduce or disclose its contents or to | |
13 | manufacture, use, or sell anything it may describe, in | |
14 | whole, or in part, without the specific written consent of | |
15 | Texas Instruments Berlin, AG. | |
16 +----------------------------------------------------------------------------- | |
17 | Purpose : This module defines the functions used by the command | |
18 | handler for mobility management. | |
19 +----------------------------------------------------------------------------- | |
20 */ | |
21 | |
22 #ifndef CMH_MMF_C | |
23 #define CMH_MMF_C | |
24 #endif | |
25 | |
26 #include "aci_all.h" | |
27 | |
28 /*==== INCLUDES ===================================================*/ | |
29 #include "aci_cmh.h" | |
30 #include "ati_cmd.h" | |
31 #include "aci_cmd.h" | |
32 #include "aci_mem.h" | |
33 #include "pcm.h" | |
34 | |
35 #ifdef FAX_AND_DATA | |
36 #include "aci_fd.h" | |
37 #endif /* of #ifdef FAX_AND_DATA */ | |
38 | |
39 #include "aci.h" | |
40 #include "psa.h" | |
41 #include "psa_cc.h" | |
42 #include "psa_mm.h" | |
43 #include "psa_sim.h" | |
44 #include "psa_sat.h" | |
45 #include "psa_util.h" | |
46 #include "cmh.h" | |
47 | |
48 #ifdef DTI | |
49 #include "dti.h" /* functionality of the dti library */ | |
50 #include "dti_conn_mng.h" | |
51 /* #include "dti_cntrl_mng.h" */ | |
52 #endif | |
53 | |
54 | |
55 #ifdef GPRS | |
56 #include "gaci.h" | |
57 #include "gaci_cmh.h" | |
58 #include "psa_gmm.h" | |
59 #include "cmh_gmm.h" | |
60 #endif /* GPRS */ | |
61 | |
62 #include "cmh_mm.h" | |
63 #include "cmh_sim.h" | |
64 | |
65 | |
66 #ifndef _SIMULATION_ | |
67 #include "ffs/ffs.h" | |
68 #include "ffs_coat.h" | |
69 #endif /* !_SIMULATION_ */ | |
70 | |
71 #ifdef FF_CPHS | |
72 #include "cphs.h" | |
73 #endif /* FF_CPHS */ | |
74 | |
75 /*==== CONSTANTS ==================================================*/ | |
76 /* Countries, for which the ECC should be ignored, i.e. done as normal call */ | |
77 /* contains MCC and ECC, last element should be always NULL*/ | |
78 GLOBAL const T_ECCIgnoreRec ECCIgnoreTable[] = | |
79 { | |
80 { 0x260, ALL_MNC, "999" }, /* Poland */ | |
81 { 0x454, 0x04F, "112" }, /* "HK ORANGE" */ | |
82 { 0x454, 0x10F, "112" }, /* "HK New World" */ | |
83 { -1, -1, NULL } /* End mark */ | |
84 }; | |
85 | |
86 #define ONS_EXT_DCS 0x80 | |
87 | |
88 /*==== TYPES ======================================================*/ | |
89 | |
90 /*==== EXPORT =====================================================*/ | |
91 | |
92 /*==== VARIABLES ==================================================*/ | |
93 LOCAL T_OPER_ENTRY ONSDesc; | |
94 GLOBAL T_ONS_READ_STATE ONSReadStatus = ONS_READ_NOT_DONE; | |
95 | |
96 /* This is required because FFS is not available in the Simulation Mode */ | |
97 #ifdef _SIMULATION_ | |
98 T_FFSPLMNIMSI RAMPlmnIMSI = {0}; | |
99 #endif | |
100 | |
101 #if defined GPRS AND defined (DTI) | |
102 EXTERN T_GMM_SHRD_PRM gmmShrdPrm; | |
103 #endif | |
104 /*==== FUNCTIONS ==================================================*/ | |
105 | |
106 /* | |
107 +-------------------------------------------------------------------+ | |
108 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
109 | ROUTINE : cmhMM_wild_compare | | |
110 +-------------------------------------------------------------------+ | |
111 | |
112 PURPOSE : | |
113 | |
114 */ | |
115 | |
116 LOCAL BOOL cmhMM_wild_compare( int sim, int bcch ) | |
117 { | |
118 if ((bcch & 0x00F) != (sim & 0x00F) AND (sim & 0x00F) != 0x00D) | |
119 return FALSE; | |
120 if ((bcch & 0x0F0) != (sim & 0x0F0) AND (sim & 0x0F0) != 0x0D0) | |
121 return FALSE; | |
122 if ((bcch & 0xF00) != (sim & 0xF00) AND (sim & 0xF00) != 0xD00) | |
123 return FALSE; | |
124 | |
125 return TRUE; | |
126 } | |
127 | |
128 | |
129 /* | |
130 +-------------------------------------------------------------------+ | |
131 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
132 | ROUTINE : cmhSIM_plmn_equal_sim_wc| | |
133 +-------------------------------------------------------------------+ | |
134 | |
135 PURPOSE : | |
136 | |
137 */ | |
138 | |
139 LOCAL BOOL cmhSIM_plmn_equal_sim_wc (int bcch_mcc, int bcch_mnc, | |
140 int sim_mcc, int sim_mnc) | |
141 { | |
142 /*TRACE_FUNCTION ("cmhSIM_plmn_equal_sim()");*/ | |
143 | |
144 /* Check MCC */ | |
145 if (!cmhMM_wild_compare(sim_mcc, bcch_mcc)) | |
146 return FALSE; | |
147 | |
148 /* Check first 2 MNC digits */ | |
149 if (!cmhMM_wild_compare(sim_mnc & 0xff0, bcch_mnc & 0xff0)) | |
150 return FALSE; | |
151 | |
152 /* Check for full match */ | |
153 if (cmhMM_wild_compare(sim_mnc & 0xf, bcch_mnc & 0xf)) | |
154 return TRUE; | |
155 | |
156 /* The 3rd digit of the MNC differs */ | |
157 if ((bcch_mcc >= 0x310) AND (bcch_mcc <= 0x316)) | |
158 { | |
159 /* | |
160 * The MCC is in the range 310..316, this means North America. | |
161 * The zero suffix rule applies. | |
162 */ | |
163 return ((((sim_mnc & 0xf) EQ 0xf) AND ((bcch_mnc & 0xf) EQ 0x0)) OR | |
164 (((sim_mnc & 0xf) EQ 0x0) AND ((bcch_mnc & 0xf) EQ 0xf))); | |
165 } | |
166 return ((bcch_mnc & 0xf) EQ 0xf); | |
167 } | |
168 | |
169 | |
170 /* | |
171 +-------------------------------------------------------------------+ | |
172 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
173 | ROUTINE : cmhMM_FindPLMN | | |
174 +-------------------------------------------------------------------+ | |
175 | |
176 PURPOSE : scan the list of operators to find the PLMN | |
177 representation. | |
178 | |
179 NOTE: This function looks as if it could be simplified and | |
180 shortened. However, it is expected that it works. | |
181 | |
182 */ | |
183 | |
184 GLOBAL BOOL cmhMM_FindPLMN( T_OPER_ENTRY * plmnDesc, | |
185 SHORT mcc, SHORT mnc, U16 lac, BOOL nw_search ) | |
186 { | |
187 int i; /* holds list idx */ | |
188 EF_PLMN plmn; /* holds PLMN identifier */ | |
189 SHORT sim_mcc; /* Integer representation MCC */ | |
190 SHORT sim_mnc; /* Integer representation MNC */ | |
191 USHORT maxRec; /* holds maximum records */ | |
192 USHORT recNr; /* holds record number */ | |
193 UBYTE retVal; /* holds return value */ | |
194 UBYTE ver; /* holds version */ | |
195 | |
196 /* | |
197 *------------------------------------------------------------------- | |
198 * search list of PNN records if SIM is EONS enabled | |
199 *------------------------------------------------------------------- | |
200 */ | |
201 | |
202 TRACE_FUNCTION ("cmhMM_FindPLMN()"); | |
203 TRACE_EVENT_P3 ("mcc: %x, mnc %x, lac: %x", mcc, mnc, lac); | |
204 | |
205 /* Initial conditions for Data Coding Scheme and PLMN name source */ | |
206 memset (plmnDesc, 0, sizeof (T_OPER_ENTRY)); | |
207 /* plmnDesc.dcs = 0; */ | |
208 plmnDesc->pnn = FALSE; | |
209 | |
210 if(psaSIM_ChkSIMSrvSup(SRV_PNN)) | |
211 { | |
212 if (!nw_search) /* this is not for network search list */ | |
213 { | |
214 if(psaSIM_ChkSIMSrvSup(SRV_OPL) || cmhSIM_plmn_is_hplmn (mcc, mnc)) | |
215 { | |
216 if (mmShrdPrm.PNNLst.plmn.v_plmn EQ VLD_PLMN) | |
217 { | |
218 plmnDesc->mcc = mcc; | |
219 plmnDesc->mnc = mnc; | |
220 plmnDesc->pnn = TRUE; | |
221 plmnDesc->long_len = mmShrdPrm.PNNLst.long_len; | |
222 plmnDesc->long_ext_dcs = mmShrdPrm.PNNLst.long_ext_dcs; | |
223 plmnDesc->shrt_len = mmShrdPrm.PNNLst.shrt_len; | |
224 plmnDesc->shrt_ext_dcs = mmShrdPrm.PNNLst.shrt_ext_dcs; | |
225 | |
226 if(mmShrdPrm.PNNLst.long_len) | |
227 { | |
228 memcpy (plmnDesc->longName, | |
229 mmShrdPrm.PNNLst.long_name, | |
230 MINIMUM(mmShrdPrm.PNNLst.long_len, MAX_LONG_OPER_LEN - 1)); | |
231 } | |
232 | |
233 if(mmShrdPrm.PNNLst.shrt_len > 0) | |
234 { | |
235 memcpy (plmnDesc->shrtName, | |
236 mmShrdPrm.PNNLst.shrt_name, | |
237 MINIMUM(mmShrdPrm.PNNLst.shrt_len, MAX_SHRT_OPER_LEN - 1)); | |
238 } | |
239 return TRUE; | |
240 } | |
241 else | |
242 { | |
243 U16 current_lac = lac; | |
244 if (current_lac EQ NOT_PRESENT_16BIT) | |
245 current_lac = mmShrdPrm.lac; | |
246 | |
247 for (i=0; i<simShrdPrm.opl_list.num_rcd; i++) | |
248 { | |
249 int pnn_rec_number = simShrdPrm.opl_list.opl_rcd[i].pnn_rec_num; | |
250 SHORT sim_mcc, sim_mnc; | |
251 | |
252 cmhSIM_getMncMccFrmPLMNsel (simShrdPrm.opl_list.opl_rcd[i].plmn, | |
253 &sim_mcc, &sim_mnc); | |
254 | |
255 if (pnn_rec_number EQ 0xFF) | |
256 continue; /* empty record, continue with next one */ | |
257 | |
258 if(cmhSIM_plmn_equal_sim_wc( mcc, mnc, sim_mcc, sim_mnc) AND | |
259 ( (simShrdPrm.opl_list.opl_rcd[i].lac1 EQ 0x0000 AND | |
260 simShrdPrm.opl_list.opl_rcd[i].lac2 EQ 0xFFFE) | |
261 OR | |
262 (simShrdPrm.opl_list.opl_rcd[i].lac1 <= current_lac AND | |
263 simShrdPrm.opl_list.opl_rcd[i].lac2 >= current_lac) ) ) | |
264 { | |
265 if (pnn_rec_number EQ 0) | |
266 break; /* 31.102: 4.2.59 EFOPL (Operator PLMN List): | |
267 A value of '00' indicates that the name is to be taken from other sources, see TS 22.101 [24] */ | |
268 | |
269 pnn_rec_number--; /* adjust to zero based array */ | |
270 | |
271 if (simShrdPrm.pnn_list.pnn_rcd[pnn_rec_number].v_plmn) | |
272 { | |
273 plmnDesc->mcc = mcc; | |
274 plmnDesc->mnc = mnc; | |
275 plmnDesc->pnn = TRUE; | |
276 plmnDesc->long_len = | |
277 MINIMUM (simShrdPrm.pnn_list.pnn_rcd[pnn_rec_number].long_len, MAX_LONG_OPER_LEN - 1); | |
278 memcpy (plmnDesc->longName, | |
279 simShrdPrm.pnn_list.pnn_rcd[pnn_rec_number].long_name, | |
280 plmnDesc->long_len); | |
281 plmnDesc->long_ext_dcs = simShrdPrm.pnn_list.pnn_rcd[pnn_rec_number].long_ext_dcs; | |
282 plmnDesc->shrt_len = | |
283 MINIMUM (simShrdPrm.pnn_list.pnn_rcd[pnn_rec_number].shrt_len, MAX_SHRT_OPER_LEN - 1); | |
284 memcpy (plmnDesc->shrtName, | |
285 simShrdPrm.pnn_list.pnn_rcd[pnn_rec_number].shrt_name, | |
286 plmnDesc->shrt_len); | |
287 plmnDesc->shrt_ext_dcs = simShrdPrm.pnn_list.pnn_rcd[pnn_rec_number].shrt_ext_dcs; | |
288 return TRUE; | |
289 } | |
290 } | |
291 } | |
292 } | |
293 } | |
294 } | |
295 | |
296 else /* the procedure is used for building the PLMN selection list */ | |
297 { | |
298 T_pnn_name *current; /* points to PNN element currently compared */ | |
299 current = (T_pnn_name *)mmShrdPrm.PNNLst.next; | |
300 while (current != NULL) | |
301 { | |
302 SHORT sim_mcc, sim_mnc; | |
303 cmhMM_CnvrtPLMN2INT (current->plmn.mcc, current->plmn.mnc, | |
304 &sim_mcc, &sim_mnc); | |
305 if (cmhSIM_plmn_equal_sim (mcc, mnc, sim_mcc, sim_mnc) AND | |
306 (current->lac EQ lac)) | |
307 { | |
308 if(current->plmn.v_plmn EQ VLD_PLMN) | |
309 { | |
310 plmnDesc->mcc = mcc; | |
311 plmnDesc->mnc = mnc; | |
312 plmnDesc->pnn = TRUE; | |
313 plmnDesc->long_len = | |
314 MINIMUM (current->long_len, MAX_LONG_OPER_LEN - 1); | |
315 plmnDesc->long_ext_dcs = current->long_ext_dcs; | |
316 plmnDesc->shrt_len = | |
317 MINIMUM (current->shrt_len, MAX_SHRT_OPER_LEN - 1); | |
318 plmnDesc->shrt_ext_dcs = current->shrt_ext_dcs; | |
319 if (current->long_len) | |
320 { | |
321 memcpy (plmnDesc->longName, | |
322 current->long_name, | |
323 plmnDesc->long_len); | |
324 } | |
325 | |
326 if (current->shrt_len) | |
327 { | |
328 memcpy (plmnDesc->shrtName, | |
329 current->shrt_name, | |
330 plmnDesc->shrt_len); | |
331 } | |
332 return TRUE; | |
333 } | |
334 } | |
335 else | |
336 current = (T_pnn_name *)current->next; | |
337 } | |
338 } | |
339 } | |
340 | |
341 if((ONSReadStatus EQ ONS_READ_OVER) AND cmhSIM_plmn_is_hplmn (mcc, mnc)) | |
342 { | |
343 | |
344 *plmnDesc = ONSDesc; /* Struct assignment */ | |
345 | |
346 plmnDesc->mcc = mcc; | |
347 plmnDesc->mnc = mnc; | |
348 return TRUE; | |
349 } | |
350 | |
351 /* | |
352 *------------------------------------------------------------------- | |
353 * search list of operators ( PCM table ) | |
354 *------------------------------------------------------------------- | |
355 */ | |
356 recNr = 1; | |
357 | |
358 do | |
359 { | |
360 retVal= pcm_ReadRecord( (UBYTE *) EF_PLMN_ID, recNr, SIZE_EF_PLMN, | |
361 (UBYTE *)&plmn, &ver, &maxRec ); | |
362 | |
363 if( retVal EQ PCM_INVALID_SIZE OR retVal EQ PCM_INVALID_RECORD ) | |
364 break; | |
365 | |
366 sim_mcc = (plmn.mcc[0] << 8) + plmn.mcc[1]; | |
367 sim_mnc = (plmn.mnc[0] << 8) + plmn.mnc[1]; | |
368 | |
369 if (cmhSIM_plmn_equal_sim (mcc, mnc, sim_mcc, sim_mnc)) | |
370 { | |
371 cmhMM_CnvrtTrmPCMOpNam( plmnDesc, &plmn ); | |
372 | |
373 plmnDesc->mcc = sim_mcc; | |
374 plmnDesc->mnc = sim_mnc; | |
375 return TRUE; | |
376 } | |
377 | |
378 recNr++; | |
379 } | |
380 while( recNr <= maxRec ); | |
381 | |
382 /* | |
383 *------------------------------------------------------------------- | |
384 * search list of operators ( fixed table ) | |
385 *------------------------------------------------------------------- | |
386 */ | |
387 for( i = 0; | |
388 operListFixed[i].mcc NEQ -1 AND operListFixed[i].mnc NEQ -1; | |
389 i++ ) | |
390 { | |
391 sim_mcc = operListFixed[i].mcc; | |
392 sim_mnc = operListFixed[i].mnc; | |
393 | |
394 if (cmhSIM_plmn_equal_sim (mcc, mnc, sim_mcc, sim_mnc)) | |
395 { | |
396 strncpy (plmnDesc->longName, | |
397 operListFixed[i].longName, | |
398 MAX_LONG_OPER_LEN - 1); | |
399 strncpy (plmnDesc->shrtName, | |
400 operListFixed[i].shrtName, | |
401 MAX_SHRT_OPER_LEN - 1); | |
402 plmnDesc->mcc = operListFixed[i].mcc; | |
403 plmnDesc->mnc = operListFixed[i].mnc; | |
404 return TRUE; | |
405 } | |
406 } | |
407 | |
408 /* | |
409 *------------------------------------------------------------------- | |
410 * only mnc and mcc description available | |
411 *------------------------------------------------------------------- | |
412 */ | |
413 plmnDesc->mcc = mcc; | |
414 plmnDesc->mnc = mnc; | |
415 /* | |
416 * The competitor has here a more sophisticated algorithm: | |
417 * If they know the country code, they don't display the | |
418 * MCC in numerical representation, but an abbreviation | |
419 * for the country. We are satisfied displaying the MCC | |
420 * and saving the ROM space for the country table. | |
421 */ | |
422 if ( ( mnc & 0xF) NEQ 0xf ) /* is 3rd digit mnc? */ | |
423 { | |
424 sprintf (plmnDesc->longName, "%03x %03x", mcc, mnc); | |
425 } | |
426 else | |
427 { /* only 2-digit-MNC */ | |
428 mnc = (mnc & 0x0FF0) >> 4; | |
429 sprintf (plmnDesc->longName, "%03x %02x", mcc, mnc); | |
430 } | |
431 strcpy (plmnDesc->shrtName, plmnDesc->longName); | |
432 | |
433 return TRUE; | |
434 } | |
435 | |
436 /* | |
437 +-------------------------------------------------------------------+ | |
438 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
439 | ROUTINE : cmhMM_isBandAllowed | | |
440 +-------------------------------------------------------------------+ | |
441 | |
442 PURPOSE : Check if band combination is part of bands allowed by manufacturer. | |
443 To be used by AT%BAND. | |
444 Per convention: if AllowedBands = 0, all bands are allowed. | |
445 band 0 is never allowed (that is not a combination !!) | |
446 */ | |
447 | |
448 GLOBAL BOOL cmhMM_isBandAllowed( UBYTE band, UBYTE AllowedBands ) | |
449 { | |
450 if( AllowedBands EQ 0x00 ) | |
451 { | |
452 /* no manufacturer constraint */ | |
453 return(TRUE); | |
454 } | |
455 if( band EQ 0x00) | |
456 { | |
457 return(FALSE); | |
458 } | |
459 | |
460 if( (band & AllowedBands) EQ band ) | |
461 { | |
462 return( TRUE ); | |
463 } | |
464 else | |
465 return( FALSE ); | |
466 } | |
467 | |
468 /* | |
469 +---------------------------------------------------------------------+ | |
470 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
471 | ROUTINE : cmhMM_writeSetBand | | |
472 +---------------------------------------------------------------------+ | |
473 | |
474 PURPOSE : | |
475 | |
476 */ | |
477 #define SIZE_OF_RFCAP (16*sizeof(UBYTE)) | |
478 | |
479 GLOBAL BOOL cmhMM_writeSetBand( UBYTE setband ) | |
480 { | |
481 #ifndef _SIMULATION_ | |
482 T_FFS_RET ret_ffs; | |
483 UBYTE RFCap[SIZE_OF_RFCAP]; /* RFCap has 16 bytes */ | |
484 | |
485 ret_ffs = FFS_fread("/gsm/com/rfcap", | |
486 RFCap, | |
487 SIZE_OF_RFCAP); | |
488 if(ret_ffs < 1) /* error */ | |
489 { | |
490 TRACE_EVENT_P1("RFCap: cannot read FFS: error n: %d", ret_ffs); | |
491 return(FALSE); | |
492 } | |
493 | |
494 /* write user bands into FFS */ | |
495 RFCap[0] = setband; | |
496 | |
497 ret_ffs = FFS_fwrite("/gsm/com/rfcap", | |
498 RFCap, | |
499 SIZE_OF_RFCAP); | |
500 | |
501 if( ret_ffs NEQ EFFS_OK ) | |
502 { | |
503 TRACE_EVENT_P1("Cannot write value on RFCap, error n: %d", ret_ffs); | |
504 return(FALSE); | |
505 } | |
506 TRACE_EVENT("cmhMM_writeSetBand: data writing in FFS successful"); | |
507 #endif /* _SIMULATION_ */ | |
508 | |
509 return( TRUE ); | |
510 } | |
511 | |
512 /* | |
513 +---------------------------------------------------------------------+ | |
514 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
515 | ROUTINE : cmhMM_getBandSettings | | |
516 +---------------------------------------------------------------------+ | |
517 | |
518 PURPOSE : | |
519 */ | |
520 | |
521 GLOBAL void cmhMM_getBandSettings( UBYTE *SetBands, UBYTE *AllowedBands ) | |
522 { | |
523 UBYTE intern_set_bands = 0x00, /* initialized to NOT SET */ | |
524 intern_allowed_bands = 0x00; /* avoid several checks against NULL */ | |
525 #ifndef _SIMULATION_ | |
526 UBYTE RFCap[SIZE_OF_RFCAP]; /* RFCap has 16 bytes */ | |
527 T_FFS_RET ret_ffs; | |
528 | |
529 TRACE_FUNCTION("cmhMM_getBandSettings()"); | |
530 | |
531 ret_ffs = FFS_fread("/gsm/com/rfcap", | |
532 RFCap, | |
533 SIZE_OF_RFCAP); | |
534 if(ret_ffs < 1) /* error */ | |
535 { | |
536 TRACE_EVENT_P1("cmhMM_getBandSettings: RFCap: cannot read FFS: error n: %d", ret_ffs); | |
537 } | |
538 else | |
539 { | |
540 TRACE_EVENT("cmhMM_getBandSettings: data reading from FFS successful"); | |
541 intern_set_bands = RFCap[0]; | |
542 intern_allowed_bands = RFCap[1]; | |
543 } | |
544 #endif /* #ifndef _SIMULATION_ */ | |
545 | |
546 if( SetBands NEQ NULL ) | |
547 { | |
548 *SetBands = intern_set_bands; | |
549 TRACE_EVENT_P1("User defined Band bitfield: %d", *SetBands); | |
550 } | |
551 if( AllowedBands NEQ NULL ) | |
552 { | |
553 *AllowedBands = intern_allowed_bands; | |
554 TRACE_EVENT_P1("Manufacturer defined Band bitfield: %d", *AllowedBands); | |
555 } | |
556 } | |
557 | |
558 /* | |
559 +-------------------------------------------------------------------+ | |
560 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
561 | ROUTINE : cmhMM_FindNumeric | | |
562 +-------------------------------------------------------------------+ | |
563 | |
564 PURPOSE : Convert operator representation from string into numeric. | |
565 | |
566 NOTE: The cmhMM_FindNumeric() function is not part of the public | |
567 declared interface to ACI. | |
568 However some guys from the MFW team are using this function | |
569 in mfw_nma.c, declaring the function prototype in the | |
570 nfw_nma.c file locally. Be careful with changes. | |
571 | |
572 */ | |
573 | |
574 GLOBAL BOOL cmhMM_FindNumeric( T_OPER_ENTRY * plmnDesc, const CHAR *numStr ) | |
575 { | |
576 USHORT idx; | |
577 BOOL ready; | |
578 | |
579 TRACE_FUNCTION ("cmhMM_FindNumeric()"); | |
580 | |
581 plmnDesc->mcc = 0; | |
582 plmnDesc->mnc = 0; | |
583 ready = FALSE; | |
584 | |
585 /* | |
586 * Convert string representation into internal represention | |
587 */ | |
588 for (idx = 0; idx < SIZE_MCC + SIZE_MNC; idx++) | |
589 { | |
590 if (idx < SIZE_MCC) | |
591 { | |
592 /* | |
593 * Converting MCC | |
594 */ | |
595 if (!ready AND | |
596 numStr[idx] >= '0' AND numStr[idx] <= '9') | |
597 { | |
598 plmnDesc->mcc = (plmnDesc->mcc << 4) + numStr[idx] - '0'; | |
599 } | |
600 else | |
601 { | |
602 ready = TRUE; | |
603 } | |
604 } | |
605 else | |
606 { | |
607 /* | |
608 * Converting MNC | |
609 */ | |
610 if (!ready AND | |
611 numStr[idx] >= '0' AND numStr[idx] <= '9') | |
612 { | |
613 plmnDesc->mnc = (plmnDesc->mnc << 4) + numStr[idx] - '0'; | |
614 } | |
615 else | |
616 { | |
617 ready = TRUE; | |
618 if ((plmnDesc->mnc & 0x00F) NEQ 0xF) | |
619 plmnDesc->mnc = (plmnDesc->mnc << 4) + 0xF; | |
620 } | |
621 } | |
622 } | |
623 | |
624 /* | |
625 *------------------------------------------------------------------- | |
626 * search list of operators | |
627 *------------------------------------------------------------------- | |
628 */ | |
629 | |
630 (void)cmhMM_FindPLMN (plmnDesc, plmnDesc->mcc, plmnDesc->mnc, NOT_PRESENT_16BIT, FALSE); | |
631 | |
632 /* decode 7 Bit default GSM for MFW */ | |
633 if (plmnDesc->pnn) | |
634 { | |
635 char longName[(MAX_ALPHA_OPER_LEN*8+6)/7]; | |
636 char shrtName[(MAX_ALPHA_OPER_LEN*8+6)/7]; | |
637 | |
638 longName[0]=shrtName[0]='\0'; | |
639 | |
640 if (plmnDesc->long_len) | |
641 { | |
642 switch (plmnDesc->long_ext_dcs>>4 & 0x07) | |
643 { | |
644 case 0x00: | |
645 utl_cvtPnn7To8((UBYTE *)plmnDesc->longName, | |
646 (UBYTE) plmnDesc->long_len, | |
647 plmnDesc->long_ext_dcs, | |
648 (UBYTE *)longName); | |
649 break; | |
650 case 0x01: | |
651 TRACE_ERROR ("ERROR: Unhandled UCS2"); | |
652 break; | |
653 default: | |
654 TRACE_ERROR ("ERROR: Unknown DCS"); | |
655 break; | |
656 } | |
657 strncpy (plmnDesc->longName, longName, MAX_LONG_OPER_LEN - 1); | |
658 plmnDesc->longName[MAX_LONG_OPER_LEN - 1] = '\0'; | |
659 plmnDesc->long_len = strlen(longName); | |
660 } | |
661 if (plmnDesc->shrt_len) | |
662 { | |
663 switch (plmnDesc->shrt_ext_dcs>>4 & 0x07) | |
664 { | |
665 case 0x00: | |
666 utl_cvtPnn7To8((UBYTE *)plmnDesc->shrtName, | |
667 (UBYTE) plmnDesc->shrt_len, | |
668 plmnDesc->shrt_ext_dcs, | |
669 (UBYTE *)shrtName); | |
670 break; | |
671 case 0x01: | |
672 TRACE_ERROR ("ERROR: Unhandled UCS2"); | |
673 break; | |
674 default: | |
675 TRACE_ERROR ("ERROR: Unknown DCS"); | |
676 break; | |
677 } | |
678 strncpy (plmnDesc->shrtName, shrtName, MAX_SHRT_OPER_LEN - 1); | |
679 plmnDesc->shrtName[MAX_SHRT_OPER_LEN - 1] = '\0'; | |
680 plmnDesc->shrt_len = strlen(shrtName); | |
681 } | |
682 } | |
683 return TRUE; | |
684 } | |
685 | |
686 /* | |
687 +-------------------------------------------------------------------+ | |
688 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
689 | ROUTINE : cmhMM_FindName | | |
690 +-------------------------------------------------------------------+ | |
691 | |
692 PURPOSE : scan the list of operators to find the short or long name | |
693 representation. | |
694 | |
695 NOTE: The cmhMM_FindName() function is not part of the public | |
696 declared interface to ACI. | |
697 However some guys from the MFW team are using this function | |
698 in mfw_nma.c, declaring the function prototype in the | |
699 nfw_nma.c file locally. Be careful with changes. | |
700 | |
701 | |
702 */ | |
703 | |
704 GLOBAL BOOL cmhMM_FindName( T_OPER_ENTRY * plmnDesc, | |
705 const CHAR *string, T_ACI_CPOL_FRMT format ) | |
706 { | |
707 USHORT idx; /* holds list idx */ | |
708 EF_PLMN plmn; /* holds PLMN identifier */ | |
709 USHORT maxRec; /* holds maximum records */ | |
710 USHORT recNr; /* holds record number */ | |
711 UBYTE retVal; /* holds return value */ | |
712 UBYTE ver; /* holds version */ | |
713 char longName[(MAX_ALPHA_OPER_LEN*8+6)/7]; | |
714 char shrtName[(MAX_ALPHA_OPER_LEN*8+6)/7]; | |
715 | |
716 /* | |
717 *------------------------------------------------------------------- | |
718 * search list of PNN records if SIM is EONS enabled | |
719 *------------------------------------------------------------------- | |
720 */ | |
721 if(psaSIM_ChkSIMSrvSup(SRV_PNN) && psaSIM_ChkSIMSrvSup(SRV_OPL)) | |
722 { | |
723 T_pnn_name *current; /* points to PNN element currently compared */ | |
724 | |
725 current = (T_pnn_name *)mmShrdPrm.PNNLst.next; | |
726 while (current != NULL) | |
727 { | |
728 longName[0]=shrtName[0]='\0'; | |
729 | |
730 if (current->long_len) | |
731 { | |
732 switch (current->long_ext_dcs>>4 & 0x07) | |
733 { | |
734 case 0x00: | |
735 utl_cvtPnn7To8((UBYTE *)current->long_name, | |
736 current->long_len, | |
737 current->long_ext_dcs, | |
738 (UBYTE *)longName); | |
739 break; | |
740 case 0x01: | |
741 TRACE_ERROR ("ERROR: Unhandled UCS2"); | |
742 break; | |
743 default: | |
744 TRACE_ERROR ("ERROR: Unknown DSC"); | |
745 break; | |
746 } | |
747 } | |
748 | |
749 if (current->shrt_len) | |
750 { | |
751 switch (current->shrt_ext_dcs>>4 & 0x07) | |
752 { | |
753 case 0x00: | |
754 utl_cvtPnn7To8((UBYTE *)current->shrt_name, | |
755 current->shrt_len, | |
756 current->shrt_ext_dcs, | |
757 (UBYTE *)shrtName); | |
758 break; | |
759 case 0x01: | |
760 TRACE_ERROR ("ERROR: Unhandled UCS2"); | |
761 break; | |
762 default: | |
763 TRACE_ERROR ("ERROR: Unknown DSC"); | |
764 break; | |
765 } | |
766 } | |
767 | |
768 if((format EQ CPOL_FRMT_Long) && (strcmp (string, (char *)longName) EQ 0) || | |
769 (format EQ CPOL_FRMT_Short) && (strcmp (string,(char *)shrtName) EQ 0)) | |
770 { | |
771 plmnDesc->pnn = 1; | |
772 /* decode Name since MFW expects 8-Bit string */ | |
773 strncpy (plmnDesc->longName, longName, MAX_LONG_OPER_LEN - 1); | |
774 plmnDesc->longName[MAX_LONG_OPER_LEN - 1] = '\0'; | |
775 plmnDesc->long_len = strlen (plmnDesc->longName); | |
776 | |
777 strncpy (plmnDesc->shrtName, shrtName, MAX_SHRT_OPER_LEN - 1); | |
778 plmnDesc->shrtName[MAX_SHRT_OPER_LEN -1] = '\0'; | |
779 plmnDesc->shrt_len = strlen (plmnDesc->shrtName); | |
780 | |
781 cmhMM_CnvrtPLMN2INT( current->plmn.mcc, | |
782 current->plmn.mnc, | |
783 &plmnDesc->mcc, &plmnDesc->mnc ); | |
784 | |
785 if (current->plmn.v_plmn EQ VLD_PLMN) | |
786 { | |
787 return TRUE; | |
788 } | |
789 } | |
790 else | |
791 current = (T_pnn_name *)current->next; | |
792 } | |
793 } | |
794 | |
795 /* | |
796 *------------------------------------------------------------------- | |
797 * search list of operators ( PCM table ) | |
798 *------------------------------------------------------------------- | |
799 */ | |
800 recNr = 1; | |
801 | |
802 do | |
803 { | |
804 retVal= pcm_ReadRecord( (UBYTE *) EF_PLMN_ID, recNr, SIZE_EF_PLMN, | |
805 (UBYTE *)&plmn, &ver, &maxRec ); | |
806 | |
807 if( retVal EQ PCM_INVALID_SIZE OR retVal EQ PCM_INVALID_RECORD ) | |
808 break; | |
809 | |
810 cmhMM_CnvrtTrmPCMOpNam( plmnDesc, &plmn ); | |
811 | |
812 if ((format EQ CPOL_FRMT_Short AND | |
813 strcmp(plmnDesc->shrtName, string) EQ 0) OR | |
814 (format EQ CPOL_FRMT_Long AND | |
815 strcmp(plmnDesc->longName, string) EQ 0)) | |
816 { | |
817 plmnDesc->mcc = (plmn.mcc[0] << 8) + plmn.mcc[1]; | |
818 plmnDesc->mnc = (plmn.mnc[0] << 8) + plmn.mnc[1]; | |
819 return TRUE; | |
820 } | |
821 | |
822 recNr++; | |
823 } | |
824 while( recNr <= maxRec ); | |
825 | |
826 /* | |
827 *------------------------------------------------------------------- | |
828 * search list of operators ( fixed table ) | |
829 *------------------------------------------------------------------- | |
830 */ | |
831 for( idx = 0; operListFixed[idx].shrtName NEQ NULL; idx++ ) | |
832 { | |
833 if ((format EQ CPOL_FRMT_Short AND | |
834 strcmp((char*)operListFixed[idx].shrtName, string) EQ 0) OR | |
835 (format EQ CPOL_FRMT_Long AND | |
836 strcmp((char*)operListFixed[idx].longName, string) EQ 0)) | |
837 { | |
838 memset (plmnDesc, 0, sizeof (T_OPER_ENTRY)); | |
839 strncpy (plmnDesc->longName, | |
840 operListFixed[idx].longName, | |
841 MAX_LONG_OPER_LEN - 1); | |
842 strncpy (plmnDesc->shrtName, | |
843 operListFixed[idx].shrtName, | |
844 MAX_SHRT_OPER_LEN - 1); | |
845 plmnDesc->mcc = operListFixed[idx].mcc; | |
846 plmnDesc->mnc = operListFixed[idx].mnc; | |
847 return TRUE; | |
848 } | |
849 } | |
850 | |
851 return FALSE; | |
852 } | |
853 | |
854 /* | |
855 +-------------------------------------------------------------------+ | |
856 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
857 | ROUTINE : cmhMM_CnvrtPLMN2INT | | |
858 +-------------------------------------------------------------------+ | |
859 | |
860 PURPOSE : convert the BCD PLMN notation for mcc and mnc into | |
861 integer values. | |
862 */ | |
863 | |
864 GLOBAL void cmhMM_CnvrtPLMN2INT( const UBYTE * BCDmcc, const UBYTE * BCDmnc, | |
865 SHORT * mccBuf, SHORT * mncBuf ) | |
866 { | |
867 SHORT idx; | |
868 | |
869 /* | |
870 * Convert MCC | |
871 */ | |
872 *mccBuf = 0; | |
873 | |
874 for (idx = 0; idx < SIZE_MCC; idx++) | |
875 { | |
876 *mccBuf = (*mccBuf << 4) + BCDmcc[idx]; | |
877 } | |
878 | |
879 /* | |
880 * Convert MNC | |
881 */ | |
882 *mncBuf = 0; | |
883 | |
884 for (idx = 0; idx < SIZE_MNC; idx++) | |
885 { | |
886 *mncBuf = (*mncBuf << 4) + BCDmnc[idx]; | |
887 } | |
888 } | |
889 | |
890 /* | |
891 +-------------------------------------------------------------------+ | |
892 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
893 | ROUTINE : cmhMM_CnvrtINT2PLMN | | |
894 +-------------------------------------------------------------------+ | |
895 | |
896 PURPOSE : convert the integer PLMN notation for mcc and mnc into | |
897 BCD representation. | |
898 */ | |
899 | |
900 GLOBAL void cmhMM_CnvrtINT2PLMN( SHORT INTmcc, SHORT INTmnc, | |
901 UBYTE * mccBuf, UBYTE * mncBuf ) | |
902 { | |
903 SHORT idx; | |
904 SHORT shift; | |
905 | |
906 /* | |
907 * Convert MCC | |
908 */ | |
909 shift = 0; | |
910 | |
911 for (idx = SIZE_MCC - 1; idx >= 0; idx--) | |
912 { /*lint -e{702} */ | |
913 mccBuf[idx] = (INTmcc >> shift) & 0xf; | |
914 shift += 4; | |
915 } | |
916 | |
917 /* | |
918 * Convert MNC | |
919 */ | |
920 shift = 0; | |
921 | |
922 for (idx = SIZE_MNC - 1; idx >= 0; idx--) | |
923 { /*lint -e{702} */ | |
924 mncBuf[idx] = (INTmnc >> shift) & 0xf; | |
925 shift += 4; | |
926 } | |
927 } | |
928 | |
929 /* | |
930 +-------------------------------------------------------------------+ | |
931 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
932 | ROUTINE : cmhMM_GetNregCREGStat | | |
933 +-------------------------------------------------------------------+ | |
934 | |
935 PURPOSE : return CREG status for a not registered state. | |
936 */ | |
937 | |
938 GLOBAL T_ACI_CREG_STAT cmhMM_GetNregCREGStat( void ) | |
939 { | |
940 switch( mmShrdPrm.regMode ) | |
941 { | |
942 case( MODE_AUTO ): | |
943 | |
944 #if defined GPRS AND defined (DTI) | |
945 if ( cmhGMM_isClassCG() ) | |
946 return(CREG_STAT_NoSearch); | |
947 else | |
948 #endif | |
949 { | |
950 /* Depending on the deregistration cause proper CREG state has been sent to the user */ | |
951 switch(mmShrdPrm.deregCs) | |
952 { | |
953 case( NREG_LIMITED_SERVICE ): | |
954 return(CREG_STAT_Denied); | |
955 case( NREG_NO_SERVICE ): | |
956 return(CREG_STAT_NoSearch); | |
957 default: | |
958 return(CREG_STAT_Unknown); | |
959 } | |
960 } | |
961 case( MODE_MAN ): | |
962 return(CREG_STAT_NoSearch); | |
963 | |
964 default: | |
965 return(CREG_STAT_Unknown); | |
966 } | |
967 } | |
968 | |
969 /* | |
970 +-------------------------------------------------------------------+ | |
971 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
972 | ROUTINE : cmhMM_GetNregCMEStat | | |
973 +-------------------------------------------------------------------+ | |
974 | |
975 PURPOSE : return CME error status for a not registered state. | |
976 */ | |
977 | |
978 GLOBAL T_ACI_CME_ERR cmhMM_GetNregCMEStat( void ) | |
979 { | |
980 switch( mmShrdPrm.regStat ) | |
981 { | |
982 case( NO_VLD_RS ): | |
983 case( RS_NO_SRV ): | |
984 | |
985 return( CME_ERR_NoServ ); | |
986 | |
987 case( RS_LMTD_SRV ): | |
988 | |
989 return( CME_ERR_LimServ ); | |
990 | |
991 default: | |
992 | |
993 return( CME_ERR_Unknown ); | |
994 } | |
995 } | |
996 | |
997 /* | |
998 +-------------------------------------------------------------------+ | |
999 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
1000 | ROUTINE : cmhMM_GetOperLstLen | | |
1001 +-------------------------------------------------------------------+ | |
1002 | |
1003 PURPOSE : return CME error status for a not registered state. | |
1004 */ | |
1005 | |
1006 GLOBAL USHORT cmhMM_GetOperLstLen( void ) | |
1007 { | |
1008 return((sizeof(operListFixed)/sizeof(T_OPER_NTRY_FIXED))-1); | |
1009 } | |
1010 | |
1011 /* | |
1012 +-------------------------------------------------------------------+ | |
1013 | PROJECT : GSM-PS (6147) MODULE : CMH_MM | | |
1014 | ROUTINE : cmhMM_CnvrtTrmPCMOpNam | | |
1015 +-------------------------------------------------------------------+ | |
1016 | |
1017 PURPOSE : Convert and terminate PCM long and short operator names. | |
1018 */ | |
1019 | |
1020 GLOBAL void cmhMM_CnvrtTrmPCMOpNam( T_OPER_ENTRY *plmnDesc, void *pPCMBuf ) | |
1021 { | |
1022 UBYTE len; | |
1023 EF_PLMN * pPlmn = (EF_PLMN *)pPCMBuf; | |
1024 | |
1025 /* convert and terminate long name */ | |
1026 for( len = 0; | |
1027 len < SIZE_EF_PLMN_LONG AND pPlmn->lngNam[len] NEQ 0xFF; | |
1028 len++ ) | |
1029 ; | |
1030 | |
1031 if (len > MAX_LONG_OPER_LEN - 1) | |
1032 len = MAX_LONG_OPER_LEN - 1; | |
1033 | |
1034 utl_cvtGsmIra( pPlmn->lngNam, len, | |
1035 (UBYTE *)plmnDesc->longName, len, | |
1036 CSCS_DIR_GsmToIra ); | |
1037 | |
1038 plmnDesc->longName[len] = '\0'; | |
1039 | |
1040 /* convert and terminate short name */ | |
1041 for( len = 0; | |
1042 len < SIZE_EF_PLMN_SHRT AND pPlmn->shrtNam[len] NEQ 0xFF; | |
1043 len++ ) | |
1044 ; | |
1045 | |
1046 if (len > MAX_SHRT_OPER_LEN - 1) | |
1047 len = MAX_SHRT_OPER_LEN - 1; | |
1048 | |
1049 utl_cvtGsmIra( pPlmn->shrtNam, len, | |
1050 (UBYTE *)plmnDesc->shrtName, len, | |
1051 CSCS_DIR_GsmToIra ); | |
1052 | |
1053 plmnDesc->shrtName[len] = '\0'; | |
1054 } | |
1055 | |
1056 /* | |
1057 +----------------------------------------------------------------------------+ | |
1058 | PROJECT : GSM-PS (6147) MODULE : CMH | | |
1059 | ROUTINE : cmhMM_Ntfy_NtwRegistrationStatus | | |
1060 +----------------------------------------------------------------------------+ | |
1061 | |
1062 PURPOSE : report function | |
1063 */ | |
1064 #ifdef FF_CPHS | |
1065 LOCAL BOOL has_roaming_state_changed(T_ACI_CREG_STAT creg) | |
1066 { | |
1067 static T_ACI_CREG_STAT last_creg_notified = CREG_STAT_NotPresent; | |
1068 UBYTE previous_creg; | |
1069 | |
1070 TRACE_EVENT_P2("has_roaming_state_changed() creg: %d, previous creg: %d", creg, last_creg_notified); | |
1071 previous_creg = last_creg_notified; | |
1072 | |
1073 /* update last_creg_notified */ | |
1074 last_creg_notified = creg; | |
1075 | |
1076 /* status has not changed at all */ | |
1077 if(creg EQ previous_creg) | |
1078 return(FALSE); | |
1079 | |
1080 /* Mobile is entering Roaming service: since creg is different from last_creg */ | |
1081 if(creg EQ CREG_STAT_Roam) | |
1082 return(TRUE); | |
1083 | |
1084 /* Mobile is not in Roaming service: Check whether this has already been told to the user */ | |
1085 if(previous_creg EQ CREG_STAT_Roam) | |
1086 return(TRUE); | |
1087 | |
1088 return(FALSE); | |
1089 } | |
1090 | |
1091 LOCAL void cmhMM_cphs_roaming(T_ACI_CREG_STAT creg) | |
1092 { | |
1093 if(has_roaming_state_changed(creg)) | |
1094 { | |
1095 /* Roaming state has changed */ | |
1096 if(creg EQ CREG_STAT_Roam) | |
1097 { | |
1098 cphs_roaming_ind(CPHS_ROAMING_ON); | |
1099 } | |
1100 else if(creg NEQ CREG_STAT_NotPresent) | |
1101 { | |
1102 cphs_roaming_ind(CPHS_ROAMING_OFF); | |
1103 } | |
1104 } | |
1105 } | |
1106 #endif /* FF_CPHS */ | |
1107 | |
1108 GLOBAL void cmhMM_Ntfy_NtwRegistrationStatus( T_ACI_CREG_STAT creg ) | |
1109 { | |
1110 SHORT src; | |
1111 T_ACI_P_CREG_GPRS_IND gprs_ind; | |
1112 | |
1113 TRACE_FUNCTION ("cmhMM_Ntfy_NtwRegistrationStatus()"); | |
1114 | |
1115 #ifdef FF_CPHS | |
1116 cmhMM_cphs_roaming(creg); | |
1117 #endif /* FF_CPHS */ | |
1118 | |
1119 #if defined (GPRS) AND defined (DTI) | |
1120 gprs_ind = gmmShrdPrm.gprs_indicator; | |
1121 #else | |
1122 gprs_ind = P_CREG_GPRS_Not_Supported; /* ACI-SPR-17218: use ACI type */ | |
1123 #endif | |
1124 | |
1125 /* notify GSM change in network status */ | |
1126 /* +CREG */ | |
1127 if( creg NEQ CREG_STAT_NotPresent ) | |
1128 { | |
1129 mmShrdPrm.creg_status = creg; | |
1130 | |
1131 for( src = 0; src < CMD_SRC_MAX; src++ ) | |
1132 { | |
1133 R_AT( RAT_CREG, src )( creg, mmShrdPrm.lac, mmShrdPrm.cid ); | |
1134 R_AT( RAT_P_CREG, src )( creg, mmShrdPrm.lac, mmShrdPrm.cid, gprs_ind ); | |
1135 } | |
1136 } | |
1137 } | |
1138 | |
1139 /* | |
1140 +-------------------------------------------------------------------------------------+ | |
1141 | PROJECT : GSM-PS MODULE : CMH | | |
1142 | ROUTINE : cmhMM_OpCheckName | | |
1143 +-------------------------------------------------------------------------------------+ | |
1144 | |
1145 PURPOSE : Function for EONS support. Checks if the operator name should be | |
1146 read from EF_PNN upon registration or location update. | |
1147 */ | |
1148 GLOBAL void cmhMM_OpCheckName() | |
1149 { | |
1150 SHORT mncCur, mccCur; /* holds converted mnc and mcc of current PLMN */ | |
1151 SHORT mncOpl, mccOpl; | |
1152 T_opl_field *ptrOpl; | |
1153 UBYTE i, pnn_rec_num=0; | |
1154 | |
1155 TRACE_FUNCTION ("cmhMM_OpCheckName()"); | |
1156 | |
1157 /* check if PNN is active and allocated on SIM */ | |
1158 if(psaSIM_ChkSIMSrvSup(SRV_PNN)) | |
1159 { | |
1160 /* check if OPL is active and allocated */ | |
1161 /* if OPL not activate the first record of PNN holds the operator name of the HPLMN */ | |
1162 if(!psaSIM_ChkSIMSrvSup(SRV_OPL)) | |
1163 { | |
1164 /* only read PNN if not already present */ | |
1165 if(mmShrdPrm.PNNLst.plmn.v_plmn != VLD_PLMN) | |
1166 { | |
1167 /* if HPLMN: this sequence is copied from cmhMM_Registered() */ | |
1168 cmhMM_CnvrtPLMN2INT( mmShrdPrm.usedPLMN.mcc, | |
1169 mmShrdPrm.usedPLMN.mnc, | |
1170 &mccCur, &mncCur ); | |
1171 if( cmhSIM_plmn_is_hplmn (mccCur, mncCur)) | |
1172 { | |
1173 cmhMM_OpReadName(1); | |
1174 } | |
1175 } | |
1176 return; | |
1177 } | |
1178 /* check if the registered PLMN or LAC has changed since last registration */ | |
1179 | |
1180 if (memcmp (mmShrdPrm.usedPLMN.mcc, mmShrdPrm.PNNLst.plmn.mcc, SIZE_MCC) != 0 || | |
1181 memcmp (mmShrdPrm.usedPLMN.mnc, mmShrdPrm.PNNLst.plmn.mnc, SIZE_MNC) != 0 || | |
1182 mmShrdPrm.lac != mmShrdPrm.PNNLst.lac) | |
1183 { | |
1184 ptrOpl = cmhSIM_GetOPL(); | |
1185 if (ptrOpl->opl_status) | |
1186 { | |
1187 /* search the OPL list for the new PNN record number */ | |
1188 i = 0; | |
1189 pnn_rec_num = 0; /* if PNN record number in OPL is 0, no PNN name is available. */ | |
1190 while (i < ptrOpl->num_rcd) | |
1191 { | |
1192 cmhMM_CnvrtPLMN2INT( mmShrdPrm.usedPLMN.mcc, | |
1193 mmShrdPrm.usedPLMN.mnc, | |
1194 &mccCur, &mncCur ); | |
1195 | |
1196 cmhSIM_getMncMccFrmPLMNsel (ptrOpl->opl_rcd[i].plmn, | |
1197 &mccOpl, &mncOpl); | |
1198 | |
1199 if(cmhSIM_plmn_equal_sim_wc (mccCur, mncCur, mccOpl, mncOpl) AND | |
1200 (ptrOpl->opl_rcd[i].lac1 <= mmShrdPrm.lac) AND | |
1201 (mmShrdPrm.lac <= ptrOpl->opl_rcd[i].lac2)) | |
1202 { | |
1203 pnn_rec_num = ptrOpl->opl_rcd[i].pnn_rec_num; | |
1204 break; | |
1205 } | |
1206 else | |
1207 i++; | |
1208 } | |
1209 if (pnn_rec_num != 0) | |
1210 { | |
1211 /* only read PNN if it is different from the last one saved */ | |
1212 if (mmShrdPrm.PNNLst.pnn_rec_num != pnn_rec_num) | |
1213 { | |
1214 cmhMM_OpReadName(pnn_rec_num); | |
1215 mmShrdPrm.PNNLst.plmn.v_plmn = INVLD_PLMN; | |
1216 } | |
1217 } | |
1218 } | |
1219 else | |
1220 TRACE_EVENT("OPL list unavailable");; /* no OPL list available */ | |
1221 } | |
1222 } | |
1223 | |
1224 /* if CPHS ONS file is not already read then read it */ | |
1225 if (!pnn_rec_num AND (ONSReadStatus EQ ONS_READ_NOT_DONE)) | |
1226 { | |
1227 #ifdef FF_CPHS | |
1228 cmhMM_ONSReadName(); | |
1229 #endif | |
1230 mmShrdPrm.PNNLst.plmn.v_plmn = INVLD_PLMN; | |
1231 } | |
1232 } | |
1233 /* | |
1234 +-------------------------------------------------------------------+ | |
1235 | PROJECT : GSM-PS (6147) MODULE : CMH_SMSF | | |
1236 | STATE : code ROUTINE : cmhMM_OpReadName | | |
1237 +-------------------------------------------------------------------+ | |
1238 | |
1239 PURPOSE : Sends a SIM read request | |
1240 | |
1241 RESULT: has an error occured ? | |
1242 Returns: FALSE if primitive has been sent and TRUE if not | |
1243 */ | |
1244 GLOBAL BOOL cmhMM_OpReadName(UBYTE rcd) | |
1245 { | |
1246 TRACE_FUNCTION ("cmhMM_OpReadName()"); | |
1247 | |
1248 if (cmhSIM_ReadRecordEF (CMD_SRC_NONE, | |
1249 AT_CMD_NONE, | |
1250 SIM_PNN, | |
1251 rcd, | |
1252 NOT_PRESENT_8BIT, | |
1253 NULL, | |
1254 cmhMM_OpReadNameCb) EQ AT_FAIL) | |
1255 { | |
1256 TRACE_EVENT("FATAL ERROR"); | |
1257 return(TRUE); | |
1258 } | |
1259 return(FALSE); | |
1260 } | |
1261 | |
1262 /* | |
1263 +---------------------------------------------------------------------------------+ | |
1264 | PROJECT : GSM-PS (6147) MODULE : CMH_SMSF | | |
1265 | STATE : code ROUTINE : cmhMM_OpReadNameCb | | |
1266 +---------------------------------------------------------------------------------+ | |
1267 | |
1268 PURPOSE : Call back for SIM retrieval of EF_PNN. | |
1269 */ | |
1270 GLOBAL void cmhMM_OpReadNameCb(SHORT table_id) | |
1271 { | |
1272 UBYTE *data; | |
1273 TRACE_FUNCTION ("cmhMM_OpReadNameCb()"); | |
1274 | |
1275 data = simShrdPrm.atb[table_id].exchData; | |
1276 | |
1277 /* Decode and copy PNN record data to PNN list*/ | |
1278 /*------------------------------------------*/ | |
1279 | |
1280 if (simShrdPrm.atb[table_id].errCode EQ SIM_NO_ERROR) /* Process only if file is read */ | |
1281 { | |
1282 | |
1283 mmShrdPrm.PNNLst.plmn = mmShrdPrm.usedPLMN; | |
1284 mmShrdPrm.PNNLst.lac = mmShrdPrm.lac; | |
1285 mmShrdPrm.PNNLst.pnn_rec_num = simShrdPrm.atb[table_id].recNr; | |
1286 | |
1287 if (*data++ EQ PNN_LONG_NAME_IEI) | |
1288 { | |
1289 mmShrdPrm.PNNLst.long_len = (*data++)-1; /* substract dcs byte from lem */ | |
1290 mmShrdPrm.PNNLst.long_ext_dcs = *data++; | |
1291 memcpy (mmShrdPrm.PNNLst.long_name, | |
1292 data, | |
1293 MINIMUM(mmShrdPrm.PNNLst.long_len, sizeof(mmShrdPrm.PNNLst.long_name))); | |
1294 data += mmShrdPrm.PNNLst.long_len; | |
1295 | |
1296 /* PNN_SHORT_NAME_IEI is an optional field */ | |
1297 if (*data++ EQ PNN_SHORT_NAME_IEI) | |
1298 { | |
1299 mmShrdPrm.PNNLst.shrt_len = (*data++)-1; /* substract dcs byte from len */ | |
1300 mmShrdPrm.PNNLst.shrt_ext_dcs = *data++; | |
1301 memcpy (mmShrdPrm.PNNLst.shrt_name, | |
1302 data, | |
1303 MINIMUM(mmShrdPrm.PNNLst.shrt_len, sizeof(mmShrdPrm.PNNLst.shrt_name))); | |
1304 } | |
1305 else | |
1306 { | |
1307 mmShrdPrm.PNNLst.shrt_len = 0; | |
1308 } | |
1309 | |
1310 mmShrdPrm.PNNLst.next = NULL; | |
1311 } | |
1312 else /* PNN_LONG_NAME_IEI is a mandatory field in PNN, if not present, PNN is invalid */ | |
1313 { | |
1314 /* PNN record number 0 indicates no PNN name is available */ | |
1315 mmShrdPrm.PNNLst.pnn_rec_num = 0; | |
1316 } | |
1317 } | |
1318 cmhMM_Registered (); | |
1319 simShrdPrm.atb[table_id].ntryUsdFlg = FALSE; | |
1320 } | |
1321 | |
1322 /* | |
1323 +---------------------------------------------------------------------------------------------+ | |
1324 | PROJECT : GSM-PS (6147) MODULE : CMH_SMSF | | |
1325 | STATE : code ROUTINE : cmhMM_OpSetPNNLst | | |
1326 +---------------------------------------------------------------------------------------------+ | |
1327 | |
1328 PURPOSE : starts reading of EF_PNN from SIM and creates PNN list for manual PLMN selection. | |
1329 */ | |
1330 GLOBAL void cmhMM_OpSetPNNLst() | |
1331 { | |
1332 T_opl_field *ptrOpl; | |
1333 UBYTE i,j, pnn_rec_num; | |
1334 | |
1335 TRACE_FUNCTION ("cmhMM_OpSetPNNLst()"); | |
1336 | |
1337 ptrOpl = cmhSIM_GetOPL(); | |
1338 if (ptrOpl->opl_status) | |
1339 { | |
1340 mmShrdPrm.pnn_read_cnt = 0; | |
1341 /* search the OPL list for the PNN record number | |
1342 for every entry in the network search lists */ | |
1343 | |
1344 /* OPL list is searched for every entry in the network search lists */ | |
1345 for (j=0; j < MAX_PLMN_ID; j++) | |
1346 { | |
1347 /* check if entry in PLMNLst is valid */ | |
1348 if( mmShrdPrm.PLMNLst[j].v_plmn != INVLD_PLMN ) | |
1349 { | |
1350 pnn_rec_num = 0; /* if PNN record number in OPL is 0, no PNN name is available. */ | |
1351 i = 0; | |
1352 ptrOpl = cmhSIM_GetOPL(); | |
1353 while (i < ptrOpl->num_rcd) /* search OPL list for PLMN nr. j */ | |
1354 { | |
1355 SHORT bcch_mcc, bcch_mnc; | |
1356 SHORT sim_mcc, sim_mnc; | |
1357 | |
1358 cmhMM_CnvrtPLMN2INT (mmShrdPrm.PLMNLst[j].mcc, mmShrdPrm.PLMNLst[j].mnc, | |
1359 &bcch_mcc, &bcch_mnc); | |
1360 cmhSIM_getMncMccFrmPLMNsel (ptrOpl->opl_rcd[i].plmn, | |
1361 &sim_mcc, &sim_mnc); | |
1362 if (cmhSIM_plmn_equal_sim (bcch_mcc, bcch_mnc, sim_mcc, sim_mnc) AND | |
1363 ((ptrOpl->opl_rcd[i].lac1 EQ 0x0000 AND | |
1364 ptrOpl->opl_rcd[i].lac2 EQ 0xFFFE) OR | |
1365 (ptrOpl->opl_rcd[i].lac1 <= mmShrdPrm.LACLst[j] AND | |
1366 mmShrdPrm.LACLst[j] <= ptrOpl->opl_rcd[i].lac2))) | |
1367 { | |
1368 pnn_rec_num = ptrOpl->opl_rcd[i].pnn_rec_num; | |
1369 break; | |
1370 } | |
1371 else | |
1372 i++; | |
1373 } | |
1374 if (pnn_rec_num != 0) | |
1375 { | |
1376 if (cmhSIM_ReadRecordEF (CMD_SRC_NONE, | |
1377 AT_CMD_NONE, | |
1378 SIM_PNN, | |
1379 pnn_rec_num, /* the PNN rec. number is read */ | |
1380 NOT_PRESENT_8BIT, | |
1381 NULL, | |
1382 cmhMM_OpExtractNameCB) EQ AT_FAIL) | |
1383 { | |
1384 TRACE_EVENT("FATAL ERROR"); | |
1385 break; /* try to continue with the next one instead of return | |
1386 since otherwise AT+COPS=? will hang if cmhMM_NetworkLst() is not called! */ | |
1387 } | |
1388 /* in cmhMM_OpExtractNameCB the latest received PNN is placed right after the PNN-list head */ | |
1389 /* pnn_nr array is used for finding the associated plmn and lac in cmhMM_OpExtractNameCB. */ | |
1390 mmShrdPrm.pnn_nr[j] = pnn_rec_num; | |
1391 mmShrdPrm.pnn_read_cnt++; | |
1392 } | |
1393 else | |
1394 { | |
1395 /* name is to be taken from other sources, 3G TS 22.101 */ | |
1396 } | |
1397 } | |
1398 else | |
1399 { | |
1400 break; /* and Invalid PLMN indicates the end of the list. */ | |
1401 } | |
1402 } | |
1403 if (mmShrdPrm.pnn_read_cnt EQ 0) /* nothing processed? */ | |
1404 { | |
1405 if(( mmEntStat.curCmd EQ AT_CMD_COPS) OR ( mmEntStat.curCmd EQ AT_CMD_P_COPS) ) | |
1406 cmhMM_NetworkLst(); /* then print list anyway, otherwise AT+COPS=? will fail. */ | |
1407 } | |
1408 } | |
1409 else | |
1410 { | |
1411 if(( mmEntStat.curCmd EQ AT_CMD_COPS) OR ( mmEntStat.curCmd EQ AT_CMD_P_COPS) ) | |
1412 cmhMM_NetworkLst(); /* then print list anyway, otherwise AT+COPS=? will fail. */ | |
1413 } | |
1414 } | |
1415 | |
1416 /* | |
1417 +-------------------------------------------------------------------+ | |
1418 | PROJECT : GSM-PS (6147) MODULE : CMH_SMSF | | |
1419 | STATE : code ROUTINE : cmhMM_OpExtractName | | |
1420 +-------------------------------------------------------------------+ | |
1421 | |
1422 PURPOSE : decodes EF_PNN record read from SIM. | |
1423 */ | |
1424 GLOBAL void cmhMM_OpExtractNameCB(SHORT table_id) | |
1425 { | |
1426 UBYTE *data; | |
1427 UBYTE i; | |
1428 T_pnn_name *newPNN; | |
1429 | |
1430 TRACE_FUNCTION ("cmhMM_OpExtractNameCB()"); | |
1431 | |
1432 data = simShrdPrm.atb[table_id].exchData; | |
1433 | |
1434 if (*data++ EQ PNN_LONG_NAME_IEI) | |
1435 { | |
1436 MALLOC(newPNN,sizeof(T_pnn_name)); | |
1437 newPNN->next = NULL; | |
1438 | |
1439 newPNN->pnn_rec_num = simShrdPrm.atb[table_id].recNr; | |
1440 | |
1441 /*find the associated PLMN and LAC for this PNN */ | |
1442 i = 0; | |
1443 while (i < MAX_PLMN_ID) | |
1444 { | |
1445 if (mmShrdPrm.pnn_nr[i] EQ newPNN->pnn_rec_num) | |
1446 { | |
1447 newPNN->plmn = mmShrdPrm.PLMNLst[i]; | |
1448 newPNN->lac = mmShrdPrm.LACLst[i]; | |
1449 break; | |
1450 } | |
1451 else | |
1452 i++; | |
1453 } | |
1454 | |
1455 newPNN->long_len = (*data++)-1; /* substract dcs byte */ | |
1456 newPNN->long_ext_dcs = *data++; | |
1457 memcpy (newPNN->long_name, | |
1458 data, | |
1459 MINIMUM(newPNN->long_len, sizeof(newPNN->long_name))); | |
1460 data += newPNN->long_len; | |
1461 | |
1462 /*----- IEI PNN short name ------*/ | |
1463 if (*data++ EQ PNN_SHORT_NAME_IEI) | |
1464 { | |
1465 newPNN->shrt_len = (*data++)-1; /* substract dcs byte */ | |
1466 newPNN->shrt_ext_dcs = *data++; | |
1467 memcpy (newPNN->shrt_name, | |
1468 data, | |
1469 MINIMUM(newPNN->shrt_len,sizeof(newPNN->shrt_name))); | |
1470 } | |
1471 else | |
1472 { | |
1473 newPNN->shrt_len = 0; | |
1474 } | |
1475 | |
1476 /* insert new element right after the static stored PNN name. */ | |
1477 newPNN->next = mmShrdPrm.PNNLst.next; | |
1478 mmShrdPrm.PNNLst.next = newPNN; | |
1479 } | |
1480 | |
1481 mmShrdPrm.pnn_read_cnt--; /* reading of PNN records completed, network list is prepared */ | |
1482 if (mmShrdPrm.pnn_read_cnt EQ 0) | |
1483 cmhMM_NetworkLst (); | |
1484 | |
1485 /* Deallocation of elements in linked list PNN */ | |
1486 // As you see, you see nothing. | |
1487 // ============================ | |
1488 | |
1489 // Immediate deallocation is not possible as someone could have | |
1490 // the idea to do a AT+COPS=? and immediately thereafter a | |
1491 // AT+COPS=1,0,"T-Mobile D" or AT+COPS=1,1,"TMO D" | |
1492 // A workaround could be to free everything not present in the actual | |
1493 // network list. Until now only a theoretical problem. For further study. | |
1494 | |
1495 // There is a second problem that this MM EONS extension list is not | |
1496 // affected by SAT changes of the respective files. Until now also | |
1497 // only theoretical and for further study. | |
1498 | |
1499 // And there is a third problem. This is, there exist 2 solutions | |
1500 // in the PS for EONS, the first one sitting in SIM related ACI around | |
1501 // the T_pnn data structure and the second one sitting in MM related ACI | |
1502 // around the T_pnn_name data structure. Two solutions for the same | |
1503 // problem. | |
1504 | |
1505 // => We should consider in the future to remove the implementation around | |
1506 // T_pnn_name and to enhance the T_pnn in SIM to end up in only one | |
1507 // implementation. This would improve the PS and save ROM. | |
1508 // Maybe the increasement of PNN_MAX_RECORDS to MAX_PLMN_ID + 1 or | |
1509 // something somewhat bigger will already do, otherwise an | |
1510 // intelligent entry replacement (aging) algorithm has to be introduced. | |
1511 // For further study. | |
1512 | |
1513 simShrdPrm.atb[table_id].ntryUsdFlg = FALSE; | |
1514 | |
1515 } | |
1516 | |
1517 /* | |
1518 +-------------------------------------------------------------------------------------+ | |
1519 | PROJECT : GSM-PS MODULE : CMH | | |
1520 | ROUTINE : cmhMM_OpUpdateName | | |
1521 +-------------------------------------------------------------------------------------+ | |
1522 | |
1523 PURPOSE : Function for EONS support. Updates the operator name by reading EF_PNN. | |
1524 RETURNS: false if PS busy with reading files (primitive has been sent) | |
1525 TRUE if PS can go on with the updating | |
1526 */ | |
1527 GLOBAL BOOL cmhMM_OpUpdateName() | |
1528 { | |
1529 SHORT mncCur, mccCur; /* holds converted mnc and mcc of current PLMN */ | |
1530 T_opl_field *ptrOpl; | |
1531 UBYTE i, pnn_rec_num; | |
1532 | |
1533 TRACE_FUNCTION ("cmhMM_OpUpdateName()"); | |
1534 | |
1535 /* check if OPL is active and allocated */ | |
1536 /* if OPL not activate the first record of PNN holds the operator name of the HPLMN */ | |
1537 if(!psaSIM_ChkSIMSrvSup(SRV_OPL)) | |
1538 { | |
1539 cmhMM_CnvrtPLMN2INT( mmShrdPrm.usedPLMN.mcc, | |
1540 mmShrdPrm.usedPLMN.mnc, | |
1541 &mccCur, &mncCur ); | |
1542 if( cmhSIM_plmn_is_hplmn (mccCur, mncCur)) | |
1543 { | |
1544 return(cmhMM_OpReadName(1)); | |
1545 } | |
1546 return(TRUE); /* no primitive has been sent to SIM */ | |
1547 } | |
1548 else /* OPL is active and allocated */ | |
1549 { | |
1550 ptrOpl = cmhSIM_GetOPL(); | |
1551 if (ptrOpl->opl_status) | |
1552 { | |
1553 /* search the OPL list for the new PNN record number */ | |
1554 i = 0; | |
1555 pnn_rec_num = 0; /* if PNN record number in OPL is 0, no PNN name is available. */ | |
1556 while (i < ptrOpl->num_rcd) | |
1557 { | |
1558 SHORT bcch_mcc, bcch_mnc; | |
1559 SHORT sim_mcc, sim_mnc; | |
1560 | |
1561 cmhMM_CnvrtPLMN2INT (mmShrdPrm.usedPLMN.mcc, mmShrdPrm.usedPLMN.mnc, | |
1562 &bcch_mcc, &bcch_mnc); | |
1563 cmhSIM_getMncMccFrmPLMNsel (ptrOpl->opl_rcd[i].plmn, | |
1564 &sim_mcc, &sim_mnc); | |
1565 if (cmhSIM_plmn_equal_sim (bcch_mcc, bcch_mnc, sim_mcc, sim_mnc) AND | |
1566 (ptrOpl->opl_rcd[i].lac1 <= mmShrdPrm.lac) AND | |
1567 (mmShrdPrm.lac <= ptrOpl->opl_rcd[i].lac2)) | |
1568 { | |
1569 pnn_rec_num = ptrOpl->opl_rcd[i].pnn_rec_num; | |
1570 break; | |
1571 } | |
1572 else | |
1573 i++; | |
1574 } | |
1575 if (pnn_rec_num != 0) | |
1576 { | |
1577 return(cmhMM_OpReadName(pnn_rec_num)); | |
1578 } | |
1579 } | |
1580 else | |
1581 TRACE_EVENT("OPL list unavailable");; /* no OPL list available */ | |
1582 } | |
1583 return(TRUE); /* no primitive has been sent to SIM */ | |
1584 } | |
1585 | |
1586 /* | |
1587 +-------------------------------------------------------------------------------------+ | |
1588 | PROJECT : GSM-PS MODULE : CMH | | |
1589 | ROUTINE : cmhMM_GetCmerSettings | | |
1590 +-------------------------------------------------------------------------------------+ | |
1591 */ | |
1592 GLOBAL void cmhMM_GetCmerSettings(T_ACI_CMD_SRC srcId, T_ACI_MM_CMER_VAL_TYPE *sCmerSettings) | |
1593 { | |
1594 sCmerSettings->sCmerModeParam = cmhPrm[srcId].mmCmdPrm.sIndicationParam.sMmCMERSettings.sCmerModeParam; | |
1595 sCmerSettings->sCmerIndParam = cmhPrm[srcId].mmCmdPrm.sIndicationParam.sMmCMERSettings.sCmerIndParam; | |
1596 sCmerSettings->sCmerBfrParam = cmhPrm[srcId].mmCmdPrm.sIndicationParam.sMmCMERSettings.sCmerBfrParam; | |
1597 } | |
1598 | |
1599 /* | |
1600 +--------------------------------------------------------------------+ | |
1601 | PROJECT : GSM-PS (6147) MODULE : CMH_MMF | | |
1602 | STATE : code ROUTINE : cmhMM_ChkIgnoreECC | | |
1603 +--------------------------------------------------------------------+ | |
1604 | |
1605 PURPOSE : This function is used to check if ECC should be set | |
1606 as a normal call. (Issue 17570) | |
1607 RETURN: TRUE - if further ECC checking should be ignored and call must be placed as normal | |
1608 FALSE - if ECC checking should be continued | |
1609 */ | |
1610 | |
1611 GLOBAL BOOL cmhMM_ChkIgnoreECC(CHAR *dialStr) | |
1612 { | |
1613 int i=0; | |
1614 SHORT mnc, mcc; | |
1615 | |
1616 /* remove any CLIR suppression/invocation prior checking for ECC */ | |
1617 if (!strncmp( dialStr, "*31#", 4) OR | |
1618 !strncmp( dialStr, "#31#", 4) ) | |
1619 { | |
1620 dialStr+=4; /* skip CLIR supression/invocation digits */ | |
1621 if ( *dialStr EQ '\0' ) | |
1622 { | |
1623 return( FALSE ); /* end already reached? */ | |
1624 } | |
1625 } | |
1626 | |
1627 if(mmShrdPrm.usedPLMN.v_plmn EQ VLD_PLMN) | |
1628 { | |
1629 cmhMM_CnvrtPLMN2INT( mmShrdPrm.usedPLMN.mcc, mmShrdPrm.usedPLMN.mnc, &mcc, &mnc ); | |
1630 | |
1631 while (ECCIgnoreTable[i].mcc NEQ -1 AND ECCIgnoreTable[i].mnc NEQ -1 ) | |
1632 { | |
1633 if ( ECCIgnoreTable[i].mcc EQ mcc AND | |
1634 (ECCIgnoreTable[i].mnc EQ mnc OR ECCIgnoreTable[i].mnc EQ ALL_MNC) ) | |
1635 { | |
1636 if(!strcmp(ECCIgnoreTable[i].ecc, dialStr)) | |
1637 { | |
1638 TRACE_EVENT("cmhCC_ChkIgnoreECC(): no ECC but normal call detected"); | |
1639 return (TRUE); | |
1640 } | |
1641 } | |
1642 i++; | |
1643 } | |
1644 } | |
1645 return (FALSE); | |
1646 } | |
1647 | |
1648 | |
1649 /* | |
1650 +--------------------------------------------------------------------+ | |
1651 | PROJECT : GSM-PS (6147) MODULE : CMH_MMF | | |
1652 | STATE : code ROUTINE : cmhMM_OperatorSelect | | |
1653 +--------------------------------------------------------------------+ | |
1654 | |
1655 PURPOSE : | |
1656 | |
1657 This function encapsulates the common CMH functionality of | |
1658 sAT_PlusCOPS(), sAT_PercentCOPS(), and sAT_PercentNRG(). | |
1659 | |
1660 The body of this function is based on the former sAT_PercentNRG(). | |
1661 | |
1662 */ | |
1663 | |
1664 GLOBAL T_ACI_RETURN cmhMM_OperatorSelect(T_ACI_CMD_SRC srcId, | |
1665 T_ACI_NRG_RGMD regMode, | |
1666 T_ACI_NRG_SVMD srvMode, | |
1667 T_ACI_NRG_FRMT oprFrmt, | |
1668 CHAR *opr) | |
1669 { | |
1670 T_MM_CMD_PRM * pMMCmdPrm; /* points to MM command parameters */ | |
1671 T_SIM_SET_PRM * pSIMSetPrm; /* points to SIM parameter set */ | |
1672 T_OPER_ENTRY plmnDesc; /* points to PLMN description */ | |
1673 T_ACI_RETURN retVal=AT_FAIL; /* holds return value */ | |
1674 BOOL found; /* result of cmhMM_FindXXX() */ | |
1675 UBYTE rm; /* holds converted registration mode */ | |
1676 | |
1677 TRACE_FUNCTION ("cmhMM_OperatorSelect()"); | |
1678 | |
1679 /* | |
1680 *------------------------------------------------------------------- | |
1681 * check command source | |
1682 *------------------------------------------------------------------- | |
1683 */ | |
1684 if(!cmh_IsVldCmdSrc (srcId)) | |
1685 { | |
1686 mmEntStat.curCmd = AT_CMD_NONE; /*Unset the Current Command */ | |
1687 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter ); | |
1688 return( AT_FAIL ); | |
1689 } | |
1690 | |
1691 pMMCmdPrm = &cmhPrm[srcId].mmCmdPrm; | |
1692 pSIMSetPrm = &simShrdPrm.setPrm[srcId]; | |
1693 mmShrdPrm.regModeBeforeAbort = mmShrdPrm.regMode; | |
1694 | |
1695 /* | |
1696 *------------------------------------------------------------------- | |
1697 * process the regMode parameter | |
1698 *------------------------------------------------------------------- | |
1699 */ | |
1700 | |
1701 switch( regMode ) | |
1702 { | |
1703 case( NRG_RGMD_Auto ): rm = MODE_AUTO; break; | |
1704 case( NRG_RGMD_Manual ): | |
1705 case( NRG_RGMD_Both ): rm = MODE_MAN; | |
1706 if(!cmhSIM_isplmnmodebit_set()) | |
1707 { | |
1708 mmEntStat.curCmd = AT_CMD_NONE; /*Unset the Current Command */ | |
1709 ACI_ERR_DESC( ACI_ERR_CLASS_Cme, CME_ERR_OpNotAllow ); | |
1710 return( AT_FAIL ); | |
1711 } | |
1712 break; | |
1713 case( NRG_RGMD_Dereg ): rm = MODE_AUTO; break; | |
1714 | |
1715 default: | |
1716 mmEntStat.curCmd = AT_CMD_NONE; /*Unset the Current Command */ | |
1717 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter ); | |
1718 return( AT_FAIL ); | |
1719 } | |
1720 | |
1721 | |
1722 /* | |
1723 *------------------------------------------------------------------- | |
1724 * process the srvMode parameter | |
1725 *------------------------------------------------------------------- | |
1726 */ | |
1727 | |
1728 switch( srvMode ) | |
1729 { | |
1730 /* | |
1731 *--------------------------------------------------------------- | |
1732 * setting of registration mode only. This is only used by %NRG. It is obsolete and | |
1733 * should be removed at some point in the future. | |
1734 *--------------------------------------------------------------- | |
1735 */ | |
1736 case( NRG_SVMD_SetRegModeOnly ): | |
1737 | |
1738 /* | |
1739 * Unset the Current Command to match exactly what sAT_PercentNRG() | |
1740 * used to do. | |
1741 */ | |
1742 mmEntStat.curCmd = AT_CMD_NONE; | |
1743 mmShrdPrm.regMode = rm; | |
1744 mmShrdPrm.regModeAutoBack = FALSE; | |
1745 | |
1746 /* | |
1747 * Documentation (8415.052.00.002) says: | |
1748 * | |
1749 * "<srvMode>=3 can be used to change the behaviour of registration | |
1750 * in case of a loss of coverage. If connection to the operator is | |
1751 * lost and <regMode> was set to manual, ME tries to register the previous | |
1752 * operator automatically.". This is not exactly what is done here. | |
1753 * What is done is that the registration mode is set to automatic in MM. | |
1754 * The main difference is that for phase 2+ the HPLMN search period | |
1755 * will be evaluated by RR and the HPLMN may be reselected after 30 minutes, | |
1756 * even without any loss of coverage. | |
1757 */ | |
1758 | |
1759 | |
1760 #ifdef GPRS | |
1761 if( psaG_MM_CMD_SET_REGMD ( rm ) < 0 ) /* restore former registration mode*/ | |
1762 #else | |
1763 if( psaMM_SetRegMode ( rm ) < 0 ) /* set registration mode */ | |
1764 #endif | |
1765 { | |
1766 TRACE_EVENT( "FATAL RETURN psaMM_SetRegMode in %%NRG" ); | |
1767 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Internal ); | |
1768 retVal = AT_FAIL; | |
1769 } | |
1770 else | |
1771 retVal = AT_CMPL; | |
1772 break; | |
1773 | |
1774 /* | |
1775 *--------------------------------------------------------------- | |
1776 * registration to full service | |
1777 *--------------------------------------------------------------- | |
1778 */ | |
1779 case( NRG_SVMD_Full ): | |
1780 | |
1781 /* For the %NRG case, only if SIM data is available we can start a registration attempt | |
1782 * But the COPS commands dont require that. | |
1783 */ | |
1784 | |
1785 if(( simShrdPrm.SIMStat EQ SS_OK AND | |
1786 simShrdPrm.PINStat EQ PS_RDY ) | |
1787 OR (mmEntStat.curCmd NEQ AT_CMD_NRG)) | |
1788 { | |
1789 switch( regMode ) | |
1790 { | |
1791 case( NRG_RGMD_Auto ): /* automatic registration */ | |
1792 | |
1793 mmShrdPrm.regModeAutoBack = FALSE; | |
1794 mmShrdPrm.regMode = MODE_AUTO; | |
1795 mmEntStat.entOwn = mmShrdPrm.owner = srcId; | |
1796 | |
1797 #ifdef GPRS | |
1798 /* Taken from the former sAT_PlusCOPS() | |
1799 * COPS will be request an only GSM registration. | |
1800 * If the ME is an only GPRS mobile, then it is impossible to request only GSM. | |
1801 * This patch will be eliminate an error in this situation. | |
1802 * | |
1803 * brz, 02.07.01 | |
1804 */ | |
1805 if ( cmhGMM_class_eq_CG() EQ TRUE ) | |
1806 { | |
1807 mmEntStat.curCmd = AT_CMD_NONE; /*Unset the Current Command */ | |
1808 return AT_CMPL; | |
1809 } | |
1810 #endif | |
1811 | |
1812 | |
1813 #ifdef GPRS | |
1814 if( psaG_MM_CMD_REG ( ) < 0 ) /* register to network */ | |
1815 #else | |
1816 if( psaMM_Registrate () < 0 ) /* register to network */ | |
1817 #endif | |
1818 { | |
1819 TRACE_EVENT( "FATAL RETURN psaMM_Registrate in cmhMM_OperatorSelect()" ); | |
1820 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Internal ); | |
1821 retVal = AT_FAIL; | |
1822 break; | |
1823 } | |
1824 | |
1825 cmhMM_Ntfy_NtwRegistrationStatus(CREG_STAT_Search); | |
1826 | |
1827 retVal = AT_EXCT; | |
1828 break; | |
1829 | |
1830 case( NRG_RGMD_Manual ): /* manual registration */ | |
1831 case( NRG_RGMD_Both ): /* manual followed by automatic reg. */ | |
1832 | |
1833 mmShrdPrm.regModeAutoBack = (regMode EQ NRG_RGMD_Both); | |
1834 | |
1835 /* | |
1836 * The following code is for %COPS=1 and %NRG=1. | |
1837 * It will get the IMSI and last registered PLMN from FFS. | |
1838 * If the IMSI has changed or no information is available in FFS, | |
1839 * it will tell MM to register to the operator stored in EF(LOCI). | |
1840 * If the IMSI hasn't changed, it will manually register to the Operator stored in FFS. | |
1841 * | |
1842 */ | |
1843 if( !opr OR !*opr) | |
1844 { | |
1845 UBYTE mcc[SIZE_MCC]; | |
1846 UBYTE mnc[SIZE_MNC]; | |
1847 UBYTE IMSI[MAX_IMSI-1]; | |
1848 char tempIMSI[MAX_IMSI] = {'\0'}; | |
1849 BOOL readFromLOCI =FALSE; | |
1850 | |
1851 | |
1852 mmShrdPrm.slctPLMN.v_plmn = VLD_PLMN; | |
1853 | |
1854 /* Get the IMSI and last registered PLMN from FFS. | |
1855 * Compare the IMSI stored in FFS to the current one. If the same, copy MCC and MNC | |
1856 * to pMMSetPrm -> slctPLMN. This will select this network manually. | |
1857 */ | |
1858 | |
1859 if (cmhMM_OperatorReadFromFFS(mcc,mnc,IMSI)) | |
1860 { | |
1861 TRACE_EVENT("Operator and IMSI succesfully read from FFS!"); | |
1862 if (!memcmp(IMSI,simShrdPrm.imsi.field,MAX_IMSI-1)) | |
1863 { | |
1864 memcpy(mmShrdPrm.slctPLMN.mcc, mcc,SIZE_MCC); | |
1865 memcpy(mmShrdPrm.slctPLMN.mnc, mnc,SIZE_MNC); | |
1866 } | |
1867 else | |
1868 { | |
1869 readFromLOCI = TRUE; | |
1870 } | |
1871 } | |
1872 else | |
1873 readFromLOCI = TRUE; | |
1874 /* If the SIM has changed or FFS cannot be read, read EF(LOCI) from the SIM. | |
1875 /* This wil now lead to a PLMN_RES with MNC,MCC=0xFFF: */ | |
1876 if (readFromLOCI) | |
1877 { | |
1878 psaSIM_cnvrtIMSI2ASCII(tempIMSI); | |
1879 psaSIM_decodeIMSI(IMSI,simShrdPrm.imsi.c_field,tempIMSI); | |
1880 | |
1881 cmhMM_CnvrtINT2PLMN( 0xFFF, | |
1882 0xFFF, | |
1883 mmShrdPrm.slctPLMN.mcc, | |
1884 mmShrdPrm.slctPLMN.mnc ); | |
1885 } | |
1886 } | |
1887 else | |
1888 { | |
1889 switch( oprFrmt ) | |
1890 { | |
1891 case( NRG_FRMT_Long ): | |
1892 found = cmhMM_FindName (&plmnDesc, opr, COPS_FRMT_Long); | |
1893 break; | |
1894 | |
1895 case( NRG_FRMT_Short ): | |
1896 found = cmhMM_FindName (&plmnDesc, opr, COPS_FRMT_Short); | |
1897 break; | |
1898 | |
1899 case( NRG_FRMT_Numeric ): | |
1900 found = cmhMM_FindNumeric(&plmnDesc, opr); | |
1901 break; | |
1902 | |
1903 default: | |
1904 found = FALSE; | |
1905 break; | |
1906 } | |
1907 | |
1908 if (!found) | |
1909 { | |
1910 mmEntStat.curCmd = AT_CMD_NONE; /* Unset the Current Command */ | |
1911 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter ); | |
1912 return( AT_FAIL ); | |
1913 } | |
1914 | |
1915 mmShrdPrm.slctPLMN.v_plmn = VLD_PLMN; | |
1916 cmhMM_CnvrtINT2PLMN( plmnDesc.mcc, /*lint -e644 plmnDesc may not have been initialized*/ | |
1917 plmnDesc.mnc, | |
1918 mmShrdPrm.slctPLMN.mcc, | |
1919 mmShrdPrm.slctPLMN.mnc ); | |
1920 } | |
1921 mmShrdPrm.regMode = MODE_MAN; | |
1922 | |
1923 mmEntStat.entOwn = mmShrdPrm.owner = srcId; | |
1924 | |
1925 #ifdef GPRS | |
1926 if( psaG_MM_CMD_NET_SEL ( ) < 0 ) /* register to network */ | |
1927 #else | |
1928 if( psaMM_NetSel () < 0 ) /* register to network */ | |
1929 #endif | |
1930 { | |
1931 TRACE_EVENT( "FATAL RETURN psaMM_NetSel in %%NRG" ); | |
1932 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Internal ); | |
1933 retVal = AT_FAIL; | |
1934 break; | |
1935 } | |
1936 | |
1937 cmhMM_Ntfy_NtwRegistrationStatus(CREG_STAT_Search); | |
1938 | |
1939 retVal = AT_EXCT; | |
1940 break; | |
1941 } | |
1942 } | |
1943 /* | |
1944 * No SIM data is available, activate SIM | |
1945 * The following block of code was only available in the former sAT_PercentNRG() | |
1946 * Therefore, it will only be executed by the %NRG command. | |
1947 */ | |
1948 else | |
1949 { | |
1950 /* check SIM entity status */ | |
1951 if( simEntStat.curCmd NEQ AT_CMD_NONE ) | |
1952 | |
1953 return( AT_BUSY ); | |
1954 | |
1955 /* prepare PLMN desc for later use */ | |
1956 if( regMode EQ NRG_RGMD_Manual ) | |
1957 { | |
1958 if( ! opr ) | |
1959 { | |
1960 mmEntStat.curCmd = AT_CMD_NONE; /*Unset the Current Command */ | |
1961 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter ); | |
1962 return( AT_FAIL ); | |
1963 } | |
1964 | |
1965 switch( oprFrmt ) | |
1966 { | |
1967 case( NRG_FRMT_Long ): | |
1968 found = cmhMM_FindName (&plmnDesc, opr, COPS_FRMT_Long); | |
1969 break; | |
1970 | |
1971 case( NRG_FRMT_Short ): | |
1972 found = cmhMM_FindName (&plmnDesc, opr, COPS_FRMT_Short); | |
1973 break; | |
1974 | |
1975 case( NRG_FRMT_Numeric): | |
1976 found = cmhMM_FindNumeric (&plmnDesc, opr); | |
1977 break; | |
1978 | |
1979 default: | |
1980 found = FALSE; | |
1981 break; | |
1982 } | |
1983 | |
1984 if (!found) | |
1985 { | |
1986 mmEntStat.curCmd = AT_CMD_NONE; /* Unset the Current Command */ | |
1987 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter ); | |
1988 return( AT_FAIL ); | |
1989 } | |
1990 | |
1991 mmShrdPrm.slctPLMN.v_plmn = VLD_PLMN; | |
1992 cmhMM_CnvrtINT2PLMN( plmnDesc.mcc, | |
1993 plmnDesc.mnc, | |
1994 mmShrdPrm.slctPLMN.mcc, | |
1995 mmShrdPrm.slctPLMN.mnc ); | |
1996 } | |
1997 | |
1998 pSIMSetPrm -> actProc = SIM_INITIALISATION; | |
1999 | |
2000 simEntStat.curCmd = AT_CMD_NRG; | |
2001 simEntStat.entOwn = simShrdPrm.owner = srcId; | |
2002 | |
2003 if( psaSIM_ActivateSIM() < 0 ) /* activate SIM card */ | |
2004 { | |
2005 TRACE_EVENT( "FATAL RETURN psaSIM in %%NRG" ); | |
2006 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Internal ); | |
2007 retVal = AT_FAIL; | |
2008 break; | |
2009 } | |
2010 | |
2011 retVal = AT_EXCT; | |
2012 } | |
2013 break; | |
2014 | |
2015 /* | |
2016 *--------------------------------------------------------------- | |
2017 * registration to limited service. | |
2018 * The COPS commands will use this for deregistration | |
2019 *--------------------------------------------------------------- | |
2020 */ | |
2021 | |
2022 case( NRG_SVMD_Limited ): | |
2023 | |
2024 mmShrdPrm.regModeAutoBack = FALSE; | |
2025 | |
2026 switch( mmShrdPrm.regStat ) | |
2027 { | |
2028 case( RS_LMTD_SRV ): | |
2029 | |
2030 mmEntStat.curCmd = AT_CMD_NONE; /*Unset the Current Command */ | |
2031 retVal = AT_CMPL; | |
2032 break; | |
2033 | |
2034 case( RS_NO_SRV ): | |
2035 case( NO_VLD_RS ): | |
2036 /* | |
2037 * if SIM data is available, do not destory it in ACI. | |
2038 * It may be needed later | |
2039 */ | |
2040 if(( simShrdPrm.SIMStat EQ SS_OK AND | |
2041 simShrdPrm.PINStat EQ PS_RDY ) OR (mmEntStat.curCmd NEQ AT_CMD_NRG)) | |
2042 { | |
2043 | |
2044 /* flag a pending request. It only applies to the %NRG case */ | |
2045 if (mmEntStat.curCmd EQ AT_CMD_NRG) | |
2046 regReqPnd = TRUE; | |
2047 | |
2048 mmEntStat.entOwn = mmShrdPrm.owner = srcId; | |
2049 | |
2050 #ifdef GPRS | |
2051 mmShrdPrm.nrgCs = GMMREG_DT_COMB; | |
2052 if( psaG_MM_CMD_DEREG ( mmShrdPrm.nrgCs ) < 0 ) /* deregister from network */ | |
2053 #else | |
2054 mmShrdPrm.nrgCs = CS_SIM_REM; | |
2055 if( psaMM_DeRegistrate () < 0 ) /* deregister from network */ | |
2056 #endif /* GPRS */ | |
2057 { | |
2058 TRACE_EVENT( "FATAL RETURN psaMM_DeRegistrate in %%NRG" ); | |
2059 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Internal ); | |
2060 retVal = AT_FAIL; | |
2061 } | |
2062 else | |
2063 { | |
2064 /* Next line commented out HM 28.08.00 */ | |
2065 /* simShrdPrm.SIMStat = NO_VLD_SS; */ /* no SIM data available */ | |
2066 retVal = AT_EXCT; | |
2067 } | |
2068 } | |
2069 /* | |
2070 * No SIM data is available, try a registration to limited service. | |
2071 * The following block of code was only available in the former sAT_PercentNRG() | |
2072 * Therefore, it will only be executed by the %NRG command. | |
2073 */ | |
2074 else | |
2075 { | |
2076 mmShrdPrm.regMode = MODE_AUTO; | |
2077 mmEntStat.entOwn = mmShrdPrm.owner = srcId; | |
2078 #ifdef GPRS | |
2079 if( psaG_MM_CMD_REG ( ) < 0 ) /* register to network */ | |
2080 #else | |
2081 if( psaMM_Registrate () < 0 ) /* register to network */ | |
2082 #endif | |
2083 { | |
2084 TRACE_EVENT( "FATAL RETURN psaMM_Registrate in %%NRG" ); | |
2085 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Internal ); | |
2086 retVal = AT_FAIL; | |
2087 } | |
2088 else | |
2089 { | |
2090 retVal = AT_EXCT; | |
2091 } | |
2092 } | |
2093 break; | |
2094 | |
2095 case( RS_FULL_SRV ): | |
2096 | |
2097 mmEntStat.entOwn = mmShrdPrm.owner = srcId; | |
2098 | |
2099 #ifdef GPRS | |
2100 mmShrdPrm.nrgCs = GMMREG_DT_COMB; | |
2101 if( psaG_MM_CMD_DEREG ( mmShrdPrm.nrgCs ) < 0 ) /* deregister from network */ | |
2102 #else | |
2103 mmShrdPrm.nrgCs = CS_SIM_REM; | |
2104 if( psaMM_DeRegistrate () < 0 ) /* deregister from network */ | |
2105 #endif | |
2106 { | |
2107 TRACE_EVENT( "FATAL RETURN psaMM_Deregistrate in %%NRG" ); | |
2108 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Internal ); | |
2109 retVal = AT_FAIL; | |
2110 break; | |
2111 } | |
2112 | |
2113 /* Next line commented out HM 28.08.00 */ | |
2114 /* simShrdPrm.SIMStat = NO_VLD_SS; */ /* no SIM data available */ | |
2115 retVal = AT_EXCT; | |
2116 break; | |
2117 } | |
2118 break; | |
2119 | |
2120 /* | |
2121 *--------------------------------------------------------------- | |
2122 * registration to no service. Only used by %NRG | |
2123 *--------------------------------------------------------------- | |
2124 */ | |
2125 | |
2126 case( NRG_SVMD_NoSrv ): | |
2127 | |
2128 mmShrdPrm.regModeAutoBack = FALSE; | |
2129 | |
2130 mmEntStat.entOwn = mmShrdPrm.owner = srcId; | |
2131 | |
2132 #ifdef GPRS | |
2133 mmShrdPrm.nrgCs = GMMREG_DT_COMB; | |
2134 if( psaG_MM_CMD_DEREG ( mmShrdPrm.nrgCs ) < 0 ) /* deregister from network */ | |
2135 #else | |
2136 mmShrdPrm.nrgCs = CS_POW_OFF; | |
2137 if( psaMM_DeRegistrate () < 0 ) /* deregister from network */ | |
2138 #endif | |
2139 { | |
2140 TRACE_EVENT( "FATAL RETURN psaMM_Deregistrate in %%NRG" ); | |
2141 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Internal ); | |
2142 retVal = AT_FAIL; | |
2143 break; | |
2144 } | |
2145 | |
2146 retVal = AT_EXCT; | |
2147 break; | |
2148 | |
2149 default: | |
2150 mmEntStat.curCmd = AT_CMD_NONE; /*Unset the Current Command */ | |
2151 ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter ); | |
2152 return( AT_FAIL ); | |
2153 } | |
2154 | |
2155 /* | |
2156 *------------------------------------------------------------------- | |
2157 * update NRG parameters | |
2158 *------------------------------------------------------------------- | |
2159 */ | |
2160 if (mmEntStat.curCmd EQ AT_CMD_NRG) | |
2161 { | |
2162 pMMCmdPrm -> NRGsrvMode = srvMode; | |
2163 pMMCmdPrm -> NRGregMode = regMode; | |
2164 pMMCmdPrm -> NRGoprFrmt = oprFrmt; | |
2165 } | |
2166 | |
2167 /* | |
2168 *------------------------------------------------------------------- | |
2169 * log command execution | |
2170 *------------------------------------------------------------------- | |
2171 */ | |
2172 #if defined SMI OR defined MFW OR defined FF_MMI_RIV | |
2173 { | |
2174 T_ACI_CLOG cmdLog; /* holds logging info */ | |
2175 | |
2176 cmdLog.atCmd = mmEntStat.curCmd; | |
2177 cmdLog.cmdType = CLOG_TYPE_Set; | |
2178 cmdLog.retCode = retVal; | |
2179 cmdLog.cId = ACI_NumParmNotPresent; | |
2180 cmdLog.sId = ACI_NumParmNotPresent; | |
2181 | |
2182 if (mmEntStat.curCmd EQ AT_CMD_NRG) | |
2183 { | |
2184 cmdLog.cmdPrm.sNRG.srcId = srcId; | |
2185 cmdLog.cmdPrm.sNRG.regMode = regMode; | |
2186 cmdLog.cmdPrm.sNRG.srvMode = srvMode; | |
2187 cmdLog.cmdPrm.sNRG.oprFrmt = oprFrmt; | |
2188 cmdLog.cmdPrm.sNRG.opr = opr; | |
2189 } | |
2190 else /*+COPS and %COPS. At the moment, the same message sent by both */ | |
2191 { | |
2192 cmdLog.cmdPrm.sCOPS.srcId = srcId; | |
2193 cmdLog.cmdPrm.sCOPS.mode = mmShrdPrm.COPSmode; | |
2194 cmdLog.cmdPrm.sCOPS.format = pMMCmdPrm -> COPSfrmt; | |
2195 cmdLog.cmdPrm.sCOPS.oper = opr; | |
2196 } | |
2197 | |
2198 rAT_PercentCLOG( &cmdLog ); | |
2199 } | |
2200 #endif | |
2201 | |
2202 return( retVal ); | |
2203 | |
2204 | |
2205 } | |
2206 | |
2207 /* | |
2208 +--------------------------------------------------------------------+ | |
2209 | PROJECT : GSM-PS (6147) MODULE : CMH_MMF | | |
2210 | STATE : code ROUTINE : cmhMM_OperatorQuery | | |
2211 +--------------------------------------------------------------------+ | |
2212 | |
2213 PURPOSE : | |
2214 | |
2215 This function encapsulates the common CMH functionality of | |
2216 qAT_PlusCOPS(), qAT_PercentCOPS(), and qAT_PercentNRG(). It basically gets the | |
2217 Network Operator name based on the format parameter. | |
2218 | |
2219 */ | |
2220 | |
2221 | |
2222 GLOBAL void cmhMM_OperatorQuery( T_ACI_CMD_SRC srcId, | |
2223 T_ACI_COPS_FRMT format, | |
2224 CHAR *oper) | |
2225 { | |
2226 SHORT mcc, mnc; /* holds coverted values for mcc and mnc */ | |
2227 T_OPER_ENTRY plmnDesc; /* entry of operator list */ | |
2228 BOOL found; | |
2229 | |
2230 TRACE_FUNCTION ("cmhMM_OperatorQuery()"); | |
2231 | |
2232 if( mmShrdPrm.regStat EQ RS_FULL_SRV AND | |
2233 mmShrdPrm.usedPLMN.v_plmn EQ VLD_PLMN) | |
2234 { | |
2235 cmhMM_CnvrtPLMN2INT( mmShrdPrm.usedPLMN.mcc, mmShrdPrm.usedPLMN.mnc, | |
2236 &mcc, &mnc ); | |
2237 | |
2238 found = cmhMM_FindPLMN (&plmnDesc, mcc, mnc, mmShrdPrm.lac, FALSE); | |
2239 | |
2240 if (!found) | |
2241 { | |
2242 TRACE_EVENT("!cmhMM_FindPLMN()"); | |
2243 if( format EQ COPS_FRMT_Numeric ) | |
2244 { | |
2245 if ((mnc & 0x00F) EQ 0xF) | |
2246 /*lint -e{702} */ | |
2247 sprintf (oper, "%03X%02X", mcc, mnc >> 4); | |
2248 else | |
2249 sprintf (oper, "%03X%03X", mcc, mnc); | |
2250 } | |
2251 } | |
2252 else | |
2253 { | |
2254 char longName[(MAX_ALPHA_OPER_LEN*8+6)/7]; | |
2255 char shrtName[(MAX_ALPHA_OPER_LEN*8+6)/7]; | |
2256 | |
2257 longName[0] = shrtName[0] = '\0'; | |
2258 | |
2259 switch( format ) | |
2260 { | |
2261 case( COPS_FRMT_Long ): | |
2262 TRACE_EVENT("COPS_FRMT_Long"); | |
2263 | |
2264 if (plmnDesc.pnn) | |
2265 { | |
2266 if (plmnDesc.long_len) | |
2267 { | |
2268 switch (plmnDesc.long_ext_dcs>>4 & 0x07) | |
2269 { | |
2270 case 0x00: | |
2271 utl_cvtPnn7To8((UBYTE *)plmnDesc.longName, | |
2272 plmnDesc.long_len, | |
2273 plmnDesc.long_ext_dcs, | |
2274 (UBYTE *)longName); | |
2275 break; | |
2276 case 0x01: | |
2277 TRACE_ERROR ("ERROR: Unhandled UCS2 DCS"); | |
2278 break; | |
2279 default: | |
2280 TRACE_ERROR ("ERROR: Unknown DCS"); | |
2281 break; | |
2282 } | |
2283 } | |
2284 strncpy( oper, longName, MAX_ALPHA_OPER_LEN-1 ); | |
2285 } | |
2286 else | |
2287 { | |
2288 strncpy( oper, plmnDesc.longName, MAX_ALPHA_OPER_LEN-1 ); | |
2289 } | |
2290 oper[MAX_ALPHA_OPER_LEN-1] = '\0'; | |
2291 break; | |
2292 | |
2293 case( COPS_FRMT_Short ): | |
2294 if (plmnDesc.pnn) | |
2295 { | |
2296 if (plmnDesc.shrt_len) | |
2297 { | |
2298 switch (plmnDesc.shrt_ext_dcs>>4 & 0x07) | |
2299 { | |
2300 case 0x00: | |
2301 utl_cvtPnn7To8((UBYTE *)plmnDesc.shrtName, | |
2302 plmnDesc.shrt_len, | |
2303 plmnDesc.shrt_ext_dcs, | |
2304 (UBYTE *)shrtName); | |
2305 break; | |
2306 case 0x01: | |
2307 TRACE_ERROR ("ERROR: Unhandled UCS2"); | |
2308 break; | |
2309 default: | |
2310 TRACE_ERROR ("ERROR: Unknown DCS"); | |
2311 break; | |
2312 } | |
2313 } | |
2314 strncpy( oper, shrtName, MAX_ALPHA_OPER_LEN-1 ); | |
2315 } | |
2316 else | |
2317 { | |
2318 strncpy( oper, plmnDesc.shrtName, MAX_ALPHA_OPER_LEN-1 ); | |
2319 } | |
2320 oper[MAX_ALPHA_OPER_LEN-1] = '\0'; | |
2321 break; | |
2322 | |
2323 case( COPS_FRMT_Numeric ): | |
2324 if ((mnc & 0x00F) EQ 0xF) | |
2325 /*lint -e{702} */ | |
2326 sprintf (oper, "%03X%02X", mcc, mnc >> 4); | |
2327 else | |
2328 sprintf (oper, "%03X%03X", mcc, mnc); | |
2329 break; | |
2330 } | |
2331 } | |
2332 } | |
2333 } | |
2334 | |
2335 | |
2336 /* | |
2337 +--------------------------------------------------------------------+ | |
2338 | PROJECT : GSM-PS (6147) MODULE : CMH_MMF | | |
2339 | STATE : code ROUTINE : cmhMM_OperatorStoreInFFS | | |
2340 +--------------------------------------------------------------------+ | |
2341 | |
2342 PURPOSE : | |
2343 | |
2344 This function stores the Operator given by MCC and MNC, and the IMSI number in the FFS file | |
2345 given by MM_FFS_OPER_FILE. If the directory in which the file is stored does not exist, it will create it. | |
2346 | |
2347 | |
2348 */ | |
2349 | |
2350 | |
2351 | |
2352 GLOBAL BOOL cmhMM_OperatorStoreInFFS(UBYTE* mcc, UBYTE* mnc, UBYTE* IMSI) | |
2353 { | |
2354 | |
2355 #ifndef _SIMULATION_ | |
2356 T_FFS_RET result; | |
2357 #endif | |
2358 T_FFSPLMNIMSI currentPlmnIMSI; | |
2359 | |
2360 TRACE_FUNCTION("cmhMM_OperatorStoreInFFS"); | |
2361 | |
2362 | |
2363 memcpy( currentPlmnIMSI.IMSI, IMSI, MAX_IMSI-1); | |
2364 memcpy(currentPlmnIMSI.mcc,mcc,SIZE_MCC); | |
2365 memcpy(currentPlmnIMSI.mnc,mnc,SIZE_MNC); | |
2366 | |
2367 #ifndef _SIMULATION_ | |
2368 | |
2369 switch(FFS_mkdir(MM_FFS_OPER_DIR)) | |
2370 {/* Check / Create ffs directory */ | |
2371 case EFFS_OK: | |
2372 TRACE_EVENT("cmhMM_OperatorStoreInFFS: Dir Created "); | |
2373 break; | |
2374 case EFFS_EXISTS: | |
2375 TRACE_EVENT("cmhMM_OperatorStoreInFFS:Dir Exists "); | |
2376 break; | |
2377 default: | |
2378 TRACE_EVENT("cmhMM_OperatorStoreInFFS:Dir Not Created"); | |
2379 return ( FALSE ); | |
2380 } | |
2381 | |
2382 result = FFS_fwrite(MM_FFS_OPER_FILE, ¤tPlmnIMSI, sizeof(T_FFSPLMNIMSI)); | |
2383 TRACE_EVENT_P1("cmhMM_OperatorStoreInFFS:FFS dir ok,FFS_fwrite res %x", result); | |
2384 | |
2385 if (result < EFFS_OK) | |
2386 return FALSE; | |
2387 | |
2388 #else /*Simulation is defined*/ | |
2389 | |
2390 memcpy(&RAMPlmnIMSI,¤tPlmnIMSI,sizeof(T_FFSPLMNIMSI)); | |
2391 | |
2392 #endif | |
2393 | |
2394 return TRUE; | |
2395 | |
2396 } | |
2397 | |
2398 /* | |
2399 +--------------------------------------------------------------------+ | |
2400 | PROJECT : GSM-PS (6147) MODULE : CMH_MMF | | |
2401 | STATE : code ROUTINE : cmhMM_OperatorReadFromFFS | | |
2402 +--------------------------------------------------------------------+ | |
2403 | |
2404 PURPOSE : | |
2405 | |
2406 This function reads the Operator given by MCC and MNC, and the IMSI number from the FFS file | |
2407 given by MM_FFS_OPER_FILE. | |
2408 | |
2409 | |
2410 */ | |
2411 | |
2412 | |
2413 GLOBAL BOOL cmhMM_OperatorReadFromFFS(UBYTE* mcc, UBYTE* mnc, UBYTE* IMSI) | |
2414 { | |
2415 #ifndef _SIMULATION_ | |
2416 T_FFS_RET result; | |
2417 #endif | |
2418 | |
2419 T_FFSPLMNIMSI currentPlmnIMSI; | |
2420 | |
2421 TRACE_FUNCTION("cmhMM_OperatorReadFromFFS()"); | |
2422 | |
2423 #ifndef _SIMULATION_ | |
2424 | |
2425 result = FFS_fread(MM_FFS_OPER_FILE, ¤tPlmnIMSI, sizeof(T_FFSPLMNIMSI)); | |
2426 | |
2427 TRACE_EVENT_P1("FFS result = %d",result); | |
2428 | |
2429 if (result < EFFS_OK) | |
2430 return FALSE; | |
2431 | |
2432 #else | |
2433 | |
2434 memcpy(¤tPlmnIMSI,&RAMPlmnIMSI,sizeof(T_FFSPLMNIMSI)); | |
2435 | |
2436 | |
2437 #endif | |
2438 | |
2439 memcpy( IMSI,currentPlmnIMSI.IMSI, MAX_IMSI-1); | |
2440 memcpy(mcc,currentPlmnIMSI.mcc,SIZE_MCC); | |
2441 memcpy(mnc,currentPlmnIMSI.mnc,SIZE_MNC); | |
2442 | |
2443 | |
2444 | |
2445 return TRUE; | |
2446 | |
2447 | |
2448 | |
2449 } | |
2450 | |
2451 /* | |
2452 +--------------------------------------------------------------------+ | |
2453 | PROJECT : GSM-PS (6147) MODULE : CMH_MMF | | |
2454 | STATE : code ROUTINE : cmhMM_GetActingHPLMN | | |
2455 +--------------------------------------------------------------------+ | |
2456 | |
2457 PURPOSE : fills the Acting HPLMN if present. | |
2458 */ | |
2459 GLOBAL BOOL cmhMM_GetActingHPLMN(SHORT * mccBuf, SHORT * mncBuf) | |
2460 { | |
2461 TRACE_FUNCTION("cmhMM_GetActingHPLMN"); | |
2462 if(mmShrdPrm.ActingHPLMN.v_plmn EQ VLD_PLMN) | |
2463 { | |
2464 cmhMM_CnvrtPLMN2INT( mmShrdPrm.ActingHPLMN.mcc, mmShrdPrm.ActingHPLMN.mnc, mccBuf, mncBuf ); | |
2465 TRACE_EVENT("Acting HPLMN Present"); | |
2466 return(TRUE); | |
2467 } | |
2468 TRACE_EVENT("Acting HPLMN not Present"); | |
2469 return (FALSE); | |
2470 } | |
2471 | |
2472 /* | |
2473 +-------------------------------------------------------------------+ | |
2474 | PROJECT : GSM-PS (6147) MODULE : CMH_MMF | | |
2475 | STATE : code ROUTINE : cmhMM_ONSReadName | | |
2476 +-------------------------------------------------------------------+ | |
2477 | |
2478 PURPOSE : Sends a SIM read request | |
2479 | |
2480 RESULT: has an error occured ? | |
2481 Returns: FALSE if primitive has been sent and TRUE if not | |
2482 */ | |
2483 GLOBAL BOOL cmhMM_ONSReadName() | |
2484 { | |
2485 TRACE_FUNCTION ("cmhMM_ONSReadName()"); | |
2486 | |
2487 if (cmhSIM_ReadTranspEF (CMD_SRC_NONE, | |
2488 AT_CMD_NONE, | |
2489 SIM_CPHS_ONSTR, | |
2490 0, | |
2491 NOT_PRESENT_8BIT, | |
2492 NULL, | |
2493 cmhMM_ONSReadNameCb) EQ AT_FAIL) | |
2494 { | |
2495 TRACE_EVENT("FATAL ERROR"); | |
2496 return(TRUE); | |
2497 } | |
2498 ONSReadStatus = ONS_READING; | |
2499 return(FALSE); | |
2500 } | |
2501 | |
2502 /* | |
2503 +-------------------------------------------------------------------+ | |
2504 | PROJECT : GSM-PS (6147) MODULE : CMH_MMF | | |
2505 | STATE : code ROUTINE : cmhMM_ONSReadNameCb | | |
2506 +-------------------------------------------------------------------+ | |
2507 | |
2508 PURPOSE : Call back for SIM retrieval of EF_ONS. | |
2509 */ | |
2510 | |
2511 GLOBAL void cmhMM_ONSReadNameCb(SHORT table_id) | |
2512 { | |
2513 int i =0; | |
2514 | |
2515 TRACE_FUNCTION ("cmhMM_ONSReadNameCb()"); | |
2516 | |
2517 if(!(simShrdPrm.atb[table_id].exchData NEQ NULL AND | |
2518 simShrdPrm.atb[table_id].exchData[0] NEQ 0xFF AND | |
2519 simShrdPrm.atb[table_id].errCode EQ SIM_NO_ERROR)) | |
2520 { | |
2521 ONSReadStatus = ONS_READ_FAIL; | |
2522 } | |
2523 else | |
2524 { | |
2525 ONSReadStatus = ONS_READ_OVER; | |
2526 memset (&ONSDesc, 0, sizeof (T_OPER_ENTRY)); | |
2527 ONSDesc.long_len = 0; | |
2528 while (simShrdPrm.atb[table_id].exchData[i] NEQ 0xFF AND | |
2529 ONSDesc.long_len < MAX_LONG_OPER_LEN - 1) | |
2530 { | |
2531 ONSDesc.long_len++; | |
2532 i++; | |
2533 } | |
2534 if (ONSDesc.long_len <= MAX_SHRT_OPER_LEN - 1) | |
2535 { | |
2536 ONSDesc.shrt_len = ONSDesc.long_len; | |
2537 } | |
2538 else | |
2539 { | |
2540 ONSDesc.shrt_len = MAX_SHRT_OPER_LEN - 1; | |
2541 } | |
2542 | |
2543 /* Issue OMAPS00058768 : EFons may have fewer bytes than MAX_LONG_OPER_LEN and if all of them are used bytes then | |
2544 we will not come across 0xFF (unused bytes), which causes ONSDesc.long_len = MAX_LONG_OPER_LEN - 1. | |
2545 So this leads to the function memcpy() to copy garbage characters in front of operator name. | |
2546 */ | |
2547 ONSDesc.long_len = MINIMUM(simShrdPrm.atb[table_id].dataLen, ONSDesc.long_len); | |
2548 ONSDesc.shrt_len = MINIMUM(simShrdPrm.atb[table_id].dataLen, ONSDesc.shrt_len); | |
2549 | |
2550 memcpy (ONSDesc.longName, simShrdPrm.atb[table_id].exchData, ONSDesc.long_len); | |
2551 memcpy (ONSDesc.shrtName, simShrdPrm.atb[table_id].exchData, ONSDesc.shrt_len); | |
2552 | |
2553 ONSDesc.long_ext_dcs = ONS_EXT_DCS; | |
2554 ONSDesc.shrt_ext_dcs = ONS_EXT_DCS; | |
2555 | |
2556 /* Issue OMAPS00058768 : For file update of ONS we no need to call cmhMM_Registered() */ | |
2557 #ifdef SIM_TOOLKIT | |
2558 if (simShrdPrm.fuRef < 0) | |
2559 #endif | |
2560 { | |
2561 cmhMM_Registered (); | |
2562 } | |
2563 #ifdef SIM_TOOLKIT | |
2564 if (simShrdPrm.fuRef >= 0) | |
2565 { | |
2566 psaSAT_FUConfirm (simShrdPrm.fuRef, | |
2567 (USHORT)((simShrdPrm.atb[table_id].errCode EQ SIM_NO_ERROR)? | |
2568 SIM_FU_SUCCESS: SIM_FU_ERROR)); | |
2569 } | |
2570 #endif | |
2571 } | |
2572 simShrdPrm.atb[table_id].ntryUsdFlg = FALSE; | |
2573 } | |
2574 | |
2575 /* | |
2576 +-------------------------------------------------------------- + | |
2577 | PROJECT : GSM-PS (6147) MODULE : CMH_MMF | | |
2578 | STATE : code ROUTINE : cmhMM_Reset_ONSDesc | | |
2579 +---------------------------------------------------------------+ | |
2580 | |
2581 PURPOSE :Issue OMAPS00058768: Reset ONSDesc. | |
2582 */ | |
2583 | |
2584 GLOBAL void cmhMM_Reset_ONSDesc() | |
2585 { | |
2586 memset(&ONSDesc,0x00,sizeof(ONSDesc)); | |
2587 ONSReadStatus = ONS_READ_NOT_DONE; | |
2588 } | |
2589 /*==== EOF ========================================================*/ |