comparison src/aci2/aci/phb_aci.c @ 3:93999a60b835

src/aci2, src/condat2: import of g23m/condat source pieces from TCS211
author Mychaela Falconia <falcon@freecalypso.org>
date Mon, 26 Sep 2016 00:29:36 +0000
parents
children
comparison
equal deleted inserted replaced
2:c41a534f33c6 3:93999a60b835
1 /*
2 +-----------------------------------------------------------------------------
3 | Project : MMI-Framework (8417)
4 | Modul : PHB
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 modul contains the functions to establish the phone book.
18 +-----------------------------------------------------------------------------
19 */
20
21 #ifdef TI_PS_FFS_PHB
22
23 #include "aci_all.h"
24 #include "aci_cmh.h"
25 #include "aci_mem.h"
26
27 #include "phb_aci.h"
28
29 #ifdef _SIMULATION_
30 /*
31 * Unfortunately we have a double definition of some preprocessor definitions
32 * within windows.h and within the protocol stack, so we undefine here.
33 */
34 #undef PARITY_NONE
35 #undef PARITY_ODD
36 #undef PARITY_EVEN
37 #undef DRV_OK
38 #include "ffs_pc_api.h"
39 #undef PARITY_NONE
40 #undef PARITY_ODD
41 #undef PARITY_EVEN
42 #undef DRV_OK
43 #else
44 #include "ffs/ffs.h"
45 #endif
46
47 #ifdef SIM_TOOLKIT
48 #include "psa.h"
49 #include "psa_sim.h"
50 #include "psa_cc.h"
51 #include "psa_sat.h"
52 #endif /* #ifdef SIM_TOOLKIT */
53
54 #include "cmh.h"
55 #include "cmh_phb.h"
56
57 #ifdef DTI
58 #include "dti_conn_mng.h"
59 #endif
60 #include "cmh_sim.h"
61
62
63 /*
64 * Constants and enumerations
65 */
66 //#define NR_EF_ICI 20 /* LRN, LMN */
67 //#define SIZE_EF_ICI (28+16) /* alpha id 16 bytes, 31.102 clause 4.2.33 */
68
69 #define NR_EF_OCI 10 /* Arbitrary, LDN */
70 #define SIZE_EF_OCI (27+16) /* alpha id 16 bytes, 31.102 clause 4.2.34 */
71
72 #define NR_EF_EXT5 10 /* Arbitrary */
73 #define SIZE_EF_EXT5 13 /* 31.102 clause 4.2.37 */
74
75 #define NR_EF_LRN 10 /* Arbitrary, LRN */
76 #define SIZE_EF_LRN SIZE_EF_OCI
77
78 #define NR_EF_LMN 10 /* Arbitrary, LMN */
79 #define SIZE_EF_LMN SIZE_EF_OCI
80
81 #define NR_EF_EXT_LRN 10 /* Arbitrary */
82 #define SIZE_EF_EXT_LRN SIZE_EF_EXT5
83
84 #define NR_EF_EXT_LMN 10 /* Arbitrary */
85 #define SIZE_EF_EXT_LMN SIZE_EF_EXT5
86
87 #define PHB_MAX_QUEUE 4 /* Queued LDN entries */
88 #define SIM_MAX_RECORD_SIZE 256 /* Maximum size of a SIM record */
89
90 #define PHB_STATE_NULL 0 /* NULL state before reading phonebook */
91 #define PHB_STATE_IDLE 1 /* IDLE state, phonebook has been read */
92 #define PHB_STATE_VERIFY 2 /* Verify SIM (Initialization, SAT) */
93 #define PHB_STATE_READ 3 /* Reading from SIM (Initialization, SAT) */
94 #define PHB_STATE_WRITE 4 /* Write/Delete single record to SIM */
95 #define PHB_STATE_DELETE_BOOK 5 /* Delete a whole SIM phonebook */
96
97 #define SET_PHB_STATE(n) pba_data.state = n;
98 #define GET_PHB_STATE() pba_data.state
99
100 typedef enum
101 {
102 INVALID_EXT = -1,
103 EXT1, /* ADN, LDN phonebook Extention */
104 EXT2, /* FDN phonebook Extention */
105 EXT3,
106 EXT4,
107 EXT5, /* Release 1999+ for EF_ICI, EF_OCI */
108 EXT_LRN, /* Extension for LRN */
109 EXT_LMN, /* Extension for LMN */
110 MAX_PHB_EXT
111 } T_EXT_TYPE;
112
113 typedef struct
114 {
115 USHORT index;
116 T_PHB_RECORD entry;
117 } T_PHB_QUEUE;
118
119 /*
120 * Type definitions
121 */
122 typedef struct
123 {
124 /* Fixed Dialling Numbers mode */
125 T_PHB_FDN_MODE fdn_mode;
126
127 /* Fixed Dialling Number class type */
128 T_ACI_CLASS fdn_classtype;
129
130 /* T_PHB_STATE of the phonebook */
131 T_PHB_STAT phb_stat;
132
133 /* Current state of the phonebook PHB_STATE_XXX */
134 UBYTE state;
135
136 /* SIM data */
137 UBYTE *data;
138
139 /* SIM service table */
140 UBYTE sim_service_table[MAX_SRV_TBL];
141
142 /* Database has to be recreated (IMSI changed or DB unclean) */
143 BOOL db_recreate;
144
145 /* Paused elementary file if reading extension record, otherwise 0 */
146 USHORT paused_ef;
147
148 /* Paused record number if reading extension record, otherwise 0 */
149 UBYTE paused_no;
150
151 /* Delete all: Book to delete, Read: Current book reading */
152 T_PHB_TYPE current_book;
153
154 /* Delete all: current record */
155 UBYTE del_record;
156
157 /* Book created */
158 BOOL book_created[MAX_PHONEBOOK];
159
160 /* Extension created */
161 BOOL ext_created[MAX_PHB_EXT];
162
163 /* We are reading the 1st ext record only to get the number of records */
164 BOOL ext_dummy_read;
165
166 /* Read flags for SIM phonebooks. Set to TRUE when reading has started */
167 BOOL book_read[MAX_PHONEBOOK];
168
169 /* Maximum number of phonebook records */
170 UBYTE phb_record_max[MAX_PHONEBOOK];
171 /* Maximum number of extension records */
172 UBYTE ext_record_max[MAX_PHB_EXT];
173
174 /* Record sizes of phonebook records */
175 UBYTE phb_record_len[MAX_PHONEBOOK];
176 /* Record sizes of extension records */
177 UBYTE ext_record_len[MAX_PHB_EXT];
178
179 /* Queued LDN entries during startup */
180 UBYTE c_queued;
181 T_PHB_QUEUE *queued[PHB_MAX_QUEUE];
182
183 /* Records to be synchronized to the SIM */
184 T_DB_CHANGED records_changed;
185 } T_PHB_ACI_DATA;
186
187 /* We pack all internal data into one data structure */
188 LOCAL T_PHB_ACI_DATA pba_data;
189
190
191 /*
192 * Prototypes for local functions
193 */
194 LOCAL UBYTE pb_get_max_num_len (T_PHB_TYPE type);
195 LOCAL BOOL pb_sim_service (UBYTE nr);
196 LOCAL BOOL pb_read_sim_record (USHORT data_id,
197 UBYTE rcd_num,
198 UBYTE len);
199 LOCAL BOOL pb_read_sim (USHORT data_id, UBYTE len, UBYTE max_length);
200 LOCAL void pb_read_next_sim_book (void);
201 LOCAL void pb_read_sim_record_cb (SHORT table_id);
202 LOCAL void pb_read_sim_cb (SHORT table_id);
203 LOCAL void pb_sat_update_reset (USHORT ef);
204 LOCAL T_PHB_RETURN pb_sync_next_sim_record (BOOL first);
205 LOCAL T_PHB_RETURN pb_write_sim_record (USHORT ef,
206 UBYTE phy_recno,
207 UBYTE entry_size,
208 const UBYTE *buffer);
209 LOCAL void pb_write_sim_record_cb (SHORT table_id);
210 T_PHB_TYPE pb_get_phb_type_from_ef (USHORT ef);
211 T_EXT_TYPE pb_get_ext_type_from_ef (USHORT ef);
212 LOCAL void pb_status_ind (T_PHB_STAT phb_stat,
213 SHORT cmeError);
214 LOCAL T_PHB_RETURN pb_write_queue (const T_PHB_RECORD *entry);
215 LOCAL void pb_clear_queue (void);
216 LOCAL T_PHB_RETURN pb_read_queue (void);
217
218 /*
219 * Functions - Interface functions to external modules
220 */
221
222
223 /*
224 +----------------------------------------------------------------------------+
225 | PROJECT : GSM-PS (6147) MODULE : PHB |
226 | STATE : code ROUTINE : pb_init |
227 +----------------------------------------------------------------------------+
228
229 PURPOSE : Power-on initialization of the phonebook module. This function
230 is called at a point in time by pei_init() when it cannot be
231 expected that the protocol stack has been booted up completely,
232 so it is undesirable to do anything here which might rely on
233 a module still not started up like the PCM or the FFS.
234
235 */
236 GLOBAL void pb_init (void)
237 {
238 TRACE_FUNCTION ("pb_init()");
239
240 memset (&pba_data, 0, sizeof (T_PHB_ACI_DATA));
241
242 pba_data.fdn_mode = NO_OPERATION;
243 pba_data.fdn_classtype = CLASS_VceDatFaxSms;
244
245 pb_sim_init();
246
247 #ifdef SIM_TOOLKIT
248 simShrdPrm.fuRef= - 1;
249 psaSAT_FURegister (pb_update);
250 #endif /* #ifdef SIM_TOOLKIT */
251
252 SET_PHB_STATE (PHB_STATE_NULL);
253 }
254
255
256 /*
257 +----------------------------------------------------------------------------+
258 | PROJECT : GSM-PS (6147) MODULE : PHB |
259 | STATE : code ROUTINE : pb_exit_phb |
260 +----------------------------------------------------------------------------+
261
262 PURPOSE : This function synchronizes, if necessary, all unwritten data,
263 shuts down the SIM phonebook layer and deallocates all
264 resources. The SIM itself is not deactivated here.
265
266
267 */
268 LOCAL void pb_exit_phb (void)
269 {
270 TRACE_FUNCTION ("pb_exit_phb()");
271
272 /*
273 * Flush everything which has not yet been written.
274 * Note: It's a good idea if everything is written immediately
275 * so that we don't need to flush anything here. Competitor phones
276 * do the job in this way, this has the advantage that even after
277 * a phone crash immediately after dialling a number or missing
278 * a call nothing is lost.
279 */
280 (void)pb_sim_flush_data ();
281
282 /* Shutdown the lower layers */
283 pb_sim_exit ();
284
285 /*
286 * Free all allocated resources
287 */
288 if (pba_data.data NEQ NULL)
289 {
290 ACI_MFREE (pba_data.data);
291 pba_data.data = NULL;
292 }
293
294 pb_clear_queue ();
295
296 /*
297 * Set phonebook to no operation
298 */
299 pba_data.fdn_mode = NO_OPERATION;
300 pba_data.phb_stat = PHB_UNKNOWN;
301 SET_PHB_STATE (PHB_STATE_NULL);
302 }
303
304
305 /*
306 +----------------------------------------------------------------------------+
307 | PROJECT : GSM-PS (6147) MODULE : PHB |
308 | STATE : code ROUTINE : pb_exit |
309 +----------------------------------------------------------------------------+
310
311 PURPOSE : Shutdown phonebook and the SIM itself also.
312
313 */
314 GLOBAL void pb_exit (void)
315 {
316 TRACE_FUNCTION ("pb_exit()");
317
318 /* Shutdown SIM phonebook */
319 pb_exit_phb ();
320
321 /* Shutdown the SIM itself */
322 simShrdPrm.synCs = SYNC_DEACTIVATE;
323 psaSIM_SyncSIM();
324 }
325
326
327 /*
328 +----------------------------------------------------------------------------+
329 | PROJECT : GSM-PS (6147) MODULE : PHB |
330 | STATE : code ROUTINE : pb_reset |
331 +----------------------------------------------------------------------------+
332
333 PURPOSE : Resets the phonebook, basically pb_exit() / pb_init().
334
335 */
336 GLOBAL void pb_reset (void)
337 {
338 TRACE_FUNCTION ("pb_reset()");
339
340 if (GET_PHB_STATE() EQ PHB_STATE_NULL)
341 return;
342
343 pb_exit_phb();
344 pb_init();
345 }
346
347 /*
348 +----------------------------------------------------------------------------+
349 | PROJECT : GSM-PS (6147) MODULE : PHB |
350 | STATE : code ROUTINE : pb_set_sim_ecc |
351 +----------------------------------------------------------------------------+
352
353 PURPOSE : This function set's the emergency call numbers.
354 This can only happen at startup time when the SIM when
355 a SIM_ACTIVATE_CNF / SIM_ACTIVATE_IND is received.
356 At this point in time the entering of PIN may be outstanding,
357 the only information we may have gotten are the emergency call
358 numbers.
359
360 */
361 GLOBAL void pb_set_sim_ecc (USHORT cause,
362 UBYTE ecc_len,
363 const UBYTE *sim_ecc)
364 {
365 TRACE_FUNCTION ("pb_set_sim_ecc()");
366
367 if (pba_data.fdn_mode NEQ NO_OPERATION)
368 {
369 /*
370 * Setting of the emergency call numbers after having gotten the
371 * IMSI of the SIM is ignored.
372 */
373 return;
374 }
375
376 if ((cause EQ SIM_CAUSE_OTHER_ERROR) OR
377 (cause EQ SIM_CAUSE_CARD_REMOVED) OR
378 (cause >= SIM_CAUSE_PARAM_WRONG AND cause <= SIM_CAUSE_DRV_TEMPFAIL))
379 {
380 /*
381 * Some error from the SIM. Indicate a fallback to the PCM/FFS.
382 */
383 (void)pb_sim_set_ecc (0, NULL);
384 }
385 else
386 {
387 (void)pb_sim_set_ecc (ecc_len, sim_ecc);
388 }
389
390
391
392 }
393
394
395 /*
396 +----------------------------------------------------------------------------+
397 | PROJECT : GSM-PS (6147) MODULE : PHB |
398 | STATE : code ROUTINE : pb_inserted_sim |
399 +----------------------------------------------------------------------------+
400
401 PURPOSE : This function is called when the PIN has been entered and we have
402 the IMSI and full access on the SIM phonebook. Reading/Verifying
403 of phonebook data is not started at this point except for the
404 ECC phonebook.
405
406 */
407 GLOBAL void pb_inserted_sim (UBYTE c_sim_serv,
408 UBYTE *sim_serv,
409 const T_imsi_field *imsi_field,
410 UBYTE adn_bdn_fdn_func,
411 UBYTE phase) // For further study
412 {
413 T_FFS_SIZE ffs_size; /* FFS result code */
414 T_ACI_CLASS classFDN; /* FDN class */
415 UBYTE ub_class = (UBYTE)CLASS_None; /* T_ACI_CLASS */
416
417 TRACE_FUNCTION ("pb_inserted_sim()");
418
419 /* Store SIM service table */
420 if (c_sim_serv > MAX_SRV_TBL)
421 c_sim_serv = MAX_SRV_TBL; /* Garbage protection */
422 memcpy (pba_data.sim_service_table, sim_serv, c_sim_serv);
423
424 /*
425 * Open SIM phonebook. Remember whether the database has to be recreated
426 * for some reason (SIM changed, database unclean).
427 */
428 if (pb_sim_open (imsi_field, &pba_data.db_recreate) NEQ PHB_OK)
429 {
430 /* We're here in really big trouble and can do nothing about it */
431 TRACE_ERROR ("Fatal: pb_sim_open() NEQ PHB_OK");
432 }
433
434 /*
435 * Update ECC phonebook
436 */
437 pba_data.book_read[ECC] = TRUE;
438 if (!pb_read_sim (SIM_ECC, NOT_PRESENT_8BIT, (UBYTE)SIM_MAX_RECORD_SIZE))
439 {
440 /* Unexpected problem. Live with previously read values */
441 TRACE_ERROR ("Unexpected: No free SIM slot");
442 }
443
444 switch (adn_bdn_fdn_func)
445 {
446 case SIM_ADN_ENABLED:
447 case SIM_ADN_BDN_ENABLED:
448 pba_data.fdn_mode = FDN_DISABLE;
449 break;
450
451 case SIM_FDN_ENABLED:
452 case SIM_FDN_BDN_ENABLED:
453 pba_data.fdn_mode = FDN_ENABLE;
454
455 /* read last fdn_classtype from FFS */
456 ffs_size = ffs_file_read ("/mmi/fdnClassType", &ub_class, sizeof(ub_class));
457
458 if (ffs_size EQ sizeof(ub_class)) /* successful read */
459 {
460 classFDN = (T_ACI_CLASS)ub_class;
461
462 /* only these two classes are currently supported */
463 if (classFDN EQ CLASS_VceDatFax OR
464 classFDN EQ CLASS_VceDatFaxSms)
465 {
466 pba_data.fdn_classtype = classFDN;
467 }
468 }
469 break;
470
471 default: /* SIM_NO_OPERATION or trash */
472 pba_data.fdn_mode = NO_OPERATION;
473 break;
474 }
475 }
476
477
478 /*
479 +----------------------------------------------------------------------------+
480 | PROJECT : GSM-PS (6147) MODULE : PHB |
481 | STATE : code ROUTINE : pb_start_build |
482 +----------------------------------------------------------------------------+
483
484 PURPOSE : This function is called when the flash phonebook has to be
485 read/verified from/against the phonebook stored on the SIM card.
486 This is triggered by the reception of the
487 MNSMS_REPORT_IND (SMS_STATE_READY) primitive.
488
489 */
490 GLOBAL void pb_start_build (BOOL unchanged)
491 {
492 TRACE_FUNCTION ("pb_start_build()");
493
494 if (pba_data.db_recreate)
495 {
496 /*
497 * Indicate a busy phonebook. We have to re-read all from scratch
498 */
499 pb_status_ind (PHB_BUSY, CME_ERR_NotPresent);
500 SET_PHB_STATE (PHB_STATE_READ);
501 }
502 else
503 {
504 /*
505 * SIM (IMSI) not changed and database clean,
506 * allow reading from the SIM phonebook by the MMI.
507 */
508 pb_status_ind (PHB_READY, CME_ERR_NotPresent);
509 SET_PHB_STATE (PHB_STATE_VERIFY);
510 }
511
512 pba_data.ext_dummy_read = FALSE;
513
514 #ifdef _SIMULATION_
515 TRACE_EVENT_P1 ("SRV_ADN: %s", pb_sim_service (SRV_ADN) ? "YES" : "NO");
516 TRACE_EVENT_P1 ("SRV_FDN: %s", pb_sim_service (SRV_FDN) ? "YES" : "NO");
517 TRACE_EVENT_P1 ("SRV_CCP: %s", pb_sim_service (SRV_CCP) ? "YES" : "NO");
518 TRACE_EVENT_P1 ("SRV_MSISDN: %s",pb_sim_service (SRV_MSISDN) ? "YES" : "NO");
519 TRACE_EVENT_P1 ("SRV_EXT1: %s", pb_sim_service (SRV_EXT1) ? "YES" : "NO");
520 TRACE_EVENT_P1 ("SRV_EXT2: %s", pb_sim_service (SRV_EXT2) ? "YES" : "NO");
521 TRACE_EVENT_P1 ("SRV_LDN: %s", pb_sim_service (SRV_LDN) ? "YES" : "NO");
522 TRACE_EVENT_P1 ("SRV_SDN: %s", pb_sim_service (SRV_SDN) ? "YES" : "NO");
523 TRACE_EVENT_P1 ("SRV_EXT3: %s", pb_sim_service (SRV_EXT3) ? "YES" : "NO");
524 TRACE_EVENT_P1 ("SRV_BDN: %s", pb_sim_service (SRV_BDN) ? "YES" : "NO");
525 TRACE_EVENT_P1 ("SRV_EXT4: %s", pb_sim_service (SRV_EXT4) ? "YES" : "NO");
526 #endif
527
528 pb_read_next_sim_book();
529 }
530
531
532 /*
533 +----------------------------------------------------------------------------+
534 | PROJECT : GSM-PS (6147) MODULE : PHB |
535 | STATE : code ROUTINE : pb_update |
536 +----------------------------------------------------------------------------+
537
538 PURPOSE : This function is called when the SIM wants to inform the
539 phonebook that some elementary files have been changed.
540 Returns TRUE if no files aere changed affecting the phonebook,
541 otherwise FALSE.
542
543 */
544 GLOBAL BOOL pb_update (int ref,
545 T_SIM_FILE_UPDATE_IND *fu)
546 {
547 unsigned i; /* Index variable */
548 T_PHB_TYPE phb_type; /* Phonebook type */
549 T_EXT_TYPE ext_type; /* Extension type */
550 USHORT ef; /* Elementary file number */
551
552 TRACE_FUNCTION ("pb_update()");
553
554 simShrdPrm.fuRef = -1; /* Assume phonebook is not affected */
555
556 for (i = 0; i < fu->val_nr; i++)
557 {
558 ef = fu->file_id[i];
559
560 if (ef EQ SIM_SST)
561 {
562 /*
563 * When SIM service table is changed, then all SIM phonebooks
564 * will be updated.
565 */
566
567 /* Mark all phonebooks as unread */
568 pb_sat_update_reset (SIM_ECC);
569 pb_sat_update_reset (SIM_ADN);
570 pb_sat_update_reset (SIM_FDN);
571 pb_sat_update_reset (SIM_BDN);
572 #ifdef SIM_LND_SUPPORT
573 pb_sat_update_reset (SIM_LND);
574 #endif
575 pb_sat_update_reset (SIM_MSISDN);
576 pb_sat_update_reset (SIM_SDN);
577
578 /* Mark all extensions as unread */
579 pb_sat_update_reset (EXT1);
580 pb_sat_update_reset (EXT2);
581 pb_sat_update_reset (EXT3);
582 pb_sat_update_reset (EXT4);
583
584 if (!pb_read_sim (SIM_SST, NOT_PRESENT_8BIT, (UBYTE)SIM_MAX_RECORD_SIZE))
585 {
586 /*
587 * We could not get an empty slot and are in trouble now as there is
588 * no good error handling possible here. On the other hand, it is
589 * not expected ever that an error occurs here.
590 */
591 TRACE_ERROR ("Internal problem getting a SIM slot");
592 return TRUE; /* No update in progress */
593 }
594
595 simShrdPrm.fuRef = ref; /* Something to do */
596
597 SET_PHB_STATE (PHB_STATE_READ);
598 return FALSE; /* Update in progress */
599 }
600 }
601
602 /*
603 * At least the SIM service table has not been changed, check now for
604 * single phonebooks and extension records.
605 */
606
607 for (i = 0; i < fu->val_nr; i++)
608 {
609 ef = fu->file_id[i];
610 phb_type = pb_get_phb_type_from_ef (ef);
611 ext_type = pb_get_ext_type_from_ef (ef);
612
613 if (phb_type NEQ INVALID_PHB)
614 {
615 /* Phonebook affected by change */
616 simShrdPrm.fuRef = ref;
617
618 /* Reset respective field */
619 pb_sat_update_reset (ef);
620
621 switch (phb_type)
622 {
623 case ADN:
624 pb_sat_update_reset (SIM_FDN);
625 break;
626
627 case FDN:
628 pb_sat_update_reset (SIM_ADN);
629 break;
630
631 default:
632 break;
633 }
634 }
635 else if (ext_type NEQ INVALID_EXT)
636 {
637 /* Phonebook affected by change */
638 simShrdPrm.fuRef = ref;
639
640 /* Reset respective field */
641 pb_sat_update_reset (ef);
642
643 /* Mark also the appropriate book itself as unread */
644 switch (ext_type)
645 {
646 case EXT1:
647 pb_sat_update_reset (SIM_ADN);
648 pb_sat_update_reset (SIM_MSISDN);
649 #ifdef SIM_LND_SUPPORT
650 pb_sat_update_reset (SIM_LND);
651 #endif
652 break;
653
654 case EXT2:
655 pb_sat_update_reset (SIM_FDN);
656 break;
657
658 case EXT3:
659 pb_sat_update_reset (SIM_SDN);
660 break;
661
662 case EXT4:
663 pb_sat_update_reset (SIM_EXT4);
664 break;
665
666 default:
667 TRACE_ERROR ("Unexpected default"); /* Everything caught */
668 break;
669 }
670 }
671 }
672
673 if (simShrdPrm.fuRef NEQ -1)
674 {
675 pb_start_build (FALSE);
676 return FALSE; /* FALSE means update in progress */
677 }
678
679 return TRUE; /* TRUE means nothing to do */
680 }
681
682
683 /*
684 +----------------------------------------------------------------------------+
685 | PROJECT : GSM-PS (6147) MODULE : PHB |
686 | STATE : code ROUTINE : pb_flush_data |
687 +----------------------------------------------------------------------------+
688
689 PURPOSE : This function is called when someone wants to flush some pending
690 data to persistent memory. Probably a bad concept, it's always
691 better to flush always internally if some operation is finished
692 and to synchronize also to the SIM immediately. This has the
693 advantage that even after a crash nothing is lost.
694
695 */
696 T_PHB_RETURN pb_flush_data (void)
697 {
698 TRACE_FUNCTION ("pb_flush_data()");
699
700 /* This function is empty, and this should remain this way if possible */
701 return PHB_OK;
702 }
703
704
705 /*
706 +----------------------------------------------------------------------------+
707 | PROJECT : GSM-PS (6147) MODULE : PHB |
708 | STATE : code ROUTINE : pb_add_record |
709 +----------------------------------------------------------------------------+
710
711 PURPOSE : This function adds or replaces a phonebook entry.
712
713 */
714 GLOBAL T_PHB_RETURN pb_add_record (T_PHB_TYPE type,
715 USHORT phy_recno,
716 const T_PHB_RECORD *entry)
717 {
718 T_PHB_RETURN phb_result;
719 T_DB_CHANGED dummy_changed;
720
721 TRACE_FUNCTION ("pb_add_record()");
722
723 #ifdef _SIMULATION_
724 {
725 char name [PHB_MAX_TAG_LEN + 1];
726 char number[MAX_PHB_NUM_LEN];
727
728 memset (name, 0, PHB_MAX_TAG_LEN + 1);
729 memcpy (name, entry->tag, entry->tag_len);
730 cmhPHB_getAdrStr (number,
731 MAX_PHB_NUM_LEN - 1,
732 entry->number,
733 entry->len);
734 if ((UBYTE)(name[0]) >= 0x80)
735 strcpy (name, "<Some UCS2 coding>");
736
737 TRACE_EVENT_P4 ("Adding number %s with name '%s' to book %d record %d",
738 number, name, type, phy_recno);
739 }
740 #endif
741
742 /* Fix phy_recno parameter for circular books */
743 if ((type EQ LDN) OR (type EQ LMN) OR (type EQ LRN))
744 {
745 if (phy_recno EQ 0)
746 phy_recno = 1;
747 }
748
749 switch (GET_PHB_STATE())
750 {
751 case PHB_STATE_IDLE:
752
753 /* Add the data in the database */
754 phb_result = pb_sim_add_record (type,
755 phy_recno,
756 entry,
757 &pba_data.records_changed);
758 if (phb_result NEQ PHB_OK)
759 return phb_result; /* Record could not be added to the database */
760
761 if (pba_data.records_changed.entries NEQ 0)
762 {
763 SET_PHB_STATE (PHB_STATE_WRITE);
764 }
765
766 /* Start synchronizing record(s) to the SIM */
767 return pb_sync_next_sim_record (TRUE);
768
769 case PHB_STATE_VERIFY:
770 case PHB_STATE_READ:
771 if ((type EQ LRN) OR (type EQ LMN))
772 {
773 /*
774 * In Release 1998- there is no counterpart on the SIM here,
775 * we can write immediately as we don't have to synch to the SIM.
776 */
777 return (pb_sim_add_record (type,
778 phy_recno,
779 entry,
780 &dummy_changed));
781 }
782 else if (type EQ LDN)
783 {
784 /* We have to queue the LDN entry and will write later */
785 return pb_write_queue (entry);
786 }
787 return PHB_FAIL;
788
789 default:
790 return PHB_FAIL;
791 }
792 }
793
794
795 /*
796 +----------------------------------------------------------------------------+
797 | PROJECT : GSM-PS (6147) MODULE : PHB |
798 | STATE : code ROUTINE : pb_del_record |
799 +----------------------------------------------------------------------------+
800
801 PURPOSE : This function deletes a phonebook entry.
802
803 */
804 GLOBAL T_PHB_RETURN pb_del_record (T_PHB_TYPE type,
805 USHORT phy_recno)
806 {
807 T_PHB_RETURN phb_result;
808
809 TRACE_FUNCTION ("pb_del_record()");
810
811 if (GET_PHB_STATE() NEQ PHB_STATE_IDLE)
812 return PHB_FAIL;
813
814 /* Delete the data in the database */
815 phb_result = pb_sim_del_record (type,
816 phy_recno,
817 &pba_data.records_changed);
818 if (phb_result NEQ PHB_OK)
819 return phb_result; /* Record could not be deleted from the database */
820
821 if (pba_data.records_changed.entries NEQ 0)
822 {
823 SET_PHB_STATE (PHB_STATE_WRITE);
824 }
825
826 /* Start synchronizing record(s) to the SIM */
827 return pb_sync_next_sim_record (TRUE);
828 }
829
830
831 /*
832 +----------------------------------------------------------------------------+
833 | PROJECT : GSM-PS (6147) MODULE : PHB |
834 | STATE : code ROUTINE : pb_check_fdn |
835 +----------------------------------------------------------------------------+
836
837 PURPOSE : This function checks whether a phone number is in FDN phonebook.
838
839 */
840 #define RETURN(x) { retVal = x; goto cleanup_exit; }
841 /*lint -e{801} Use of goto*/
842 GLOBAL T_PHB_RETURN pb_check_fdn (UBYTE toa,
843 const UBYTE *number)
844 {
845 T_PHB_RETURN retVal;
846 T_PHB_RECORD *phb_record;
847 size_t cmp_len;
848 CHAR cur_number[MAX_PHB_NUM_LEN];
849 USHORT phy_recno;
850 SHORT max_rcd;
851 SHORT dummy_used_rcd;
852 UBYTE dummy_tag_len;
853
854 ACI_MALLOC (phb_record, sizeof (T_PHB_RECORD));
855
856 TRACE_FUNCTION ("pb_check_fdn()");
857
858 retVal = pb_sim_read_sizes (FDN,
859 &max_rcd,
860 &dummy_used_rcd,
861 &dummy_tag_len);
862 if (retVal NEQ PHB_OK)
863 {
864 TRACE_ERROR ("Unexpected return from pb_sim_read_sizes()");
865 RETURN (retVal)
866 }
867
868 /*
869 * Do a linear search through all the records applying the number comparision
870 * rules which make us pass 51.010 test case 27.18.1.1.4.2.
871 * Those rules are different from those which are to be applied if a matching
872 * record has to be found for an incoming call, therefore we cannot use
873 * pb_sim_search_number() here.
874 * As normally FDN phonebooks have not more than 10 entries (max_rcd) a
875 * simple linear search will fulfill here all timing requirements.
876 */
877
878 for (phy_recno = 1; phy_recno <= max_rcd; phy_recno++)
879 {
880 retVal = pb_sim_read_record (FDN, phy_recno, phb_record);
881 if (retVal EQ PHB_OK)
882 {
883 cmhPHB_getAdrStr (cur_number,
884 MAX_PHB_NUM_LEN - 1,
885 phb_record->number,
886 phb_record->len);
887 cmp_len = strlen (cur_number);
888 if (strncmp ((char *)number, cur_number, cmp_len) EQ 0)
889 {
890 /* Number matches. ACI-SPR-11927: check type of address */
891 if ((toa EQ 0) OR
892 (toa EQ phb_record->ton_npi) OR
893 (phb_record->ton_npi EQ 0xFF))
894 {
895 RETURN (PHB_OK)
896 }
897 }
898 }
899 }
900 RETURN (PHB_FAIL) /* Not found */
901
902 cleanup_exit:
903 ACI_MFREE (phb_record);
904 return retVal;
905 }
906
907
908 /*
909 +----------------------------------------------------------------------------+
910 | PROJECT : GSM-PS (6147) MODULE : PHB |
911 | STATE : code ROUTINE : pb_switch_adn_fdn |
912 +----------------------------------------------------------------------------+
913
914 PURPOSE : This function switches FDN and ADN phonebooks (sAT_PlusCLCK()).
915
916 */
917 GLOBAL T_PHB_RETURN pb_switch_adn_fdn (T_PHB_FDN_MODE mode,
918 T_ACI_CLASS classFDN)
919 {
920 TRACE_FUNCTION ("pb_switch_adn_fdn()");
921
922 if (GET_PHB_STATE() NEQ PHB_STATE_IDLE)
923 return PHB_FAIL;
924
925 /* Check parameters */
926 switch (mode)
927 {
928 case FDN_DISABLE:
929 case FDN_ENABLE:
930 break;
931
932 default:
933 return PHB_FAIL;
934 }
935
936 if (classFDN NEQ pba_data.fdn_classtype)
937 {
938 UBYTE ub_class;
939
940 ub_class = (UBYTE)classFDN;
941 pba_data.fdn_classtype = classFDN;
942
943 #ifndef _SIMULATION_
944 if (ffs_fwrite("/mmi/fdnClassType", &classFDN, sizeof(ub_class)) < 1)
945 {
946 TRACE_ERROR ("sAT_PlusCLCK: failed to write /mmi/fdnClassType");
947 }
948 #endif /* #ifndef _SIMULATION_ */
949 }
950
951 #if 1
952
953 pba_data.fdn_mode = mode;
954
955 /* Remove ADN from database */
956 (void)pb_sim_remove_ef (SIM_ADN);
957 pba_data.book_created[ADN] = FALSE;
958 pba_data.book_read[ADN] = FALSE;
959
960 /* Remove FDN from database */
961 (void)pb_sim_remove_ef (SIM_FDN);
962 pba_data.book_created[FDN] = FALSE;
963 pba_data.book_read[FDN] = FALSE;
964
965 /* Remove EXT2 from database */
966 (void)pb_sim_remove_ef (SIM_EXT2);
967 pba_data.ext_created[EXT2] = FALSE;
968
969 /* Force cmhPHB_StatIndication (PHB_BUSY, CME_ERR_NotPresent) */
970 pba_data.db_recreate = TRUE;
971
972 SET_PHB_STATE (PHB_STATE_READ);
973
974 /* Start reading changed books */
975 pb_start_build (FALSE);
976 #else
977 // This optimization does not comply with current test cases,
978 // not understood perfectly whether the test cases can be
979 // changed, so kept old behaviour.
980 if (pba_data.fdn_mode NEQ mode)
981 {
982 pba_data.fdn_mode = mode;
983
984 if (mode EQ FDN_ENABLE)
985 {
986 /* Remove ADN from database */
987 (void)pb_sim_remove_ef (SIM_ADN);
988 pba_data.book_created[ADN] = FALSE;
989 pba_data.book_read[ADN] = FALSE;
990 }
991 else
992 {
993 /* Remove FDN from database */
994 (void)pb_sim_remove_ef (SIM_FDN);
995 pba_data.book_created[FDN] = FALSE;
996 pba_data.book_read[FDN] = FALSE;
997
998 /* Remove EXT2 from database */
999 (void)pb_sim_remove_ef (SIM_EXT2);
1000 pba_data.ext_created[EXT2] = FALSE;
1001 }
1002
1003 /* Force cmhPHB_StatIndication (PHB_BUSY, CME_ERR_NotPresent) */
1004 pba_data.db_recreate = TRUE;
1005
1006 SET_PHB_STATE (PHB_STATE_READ);
1007
1008 /* Start reading changed books */
1009 pb_start_build (FALSE);
1010 }
1011 #endif
1012 return PHB_OK;
1013 }
1014
1015
1016 /*
1017 +----------------------------------------------------------------------------+
1018 | PROJECT : GSM-PS (6147) MODULE : PHB |
1019 | STATE : code ROUTINE : pb_del_book |
1020 +----------------------------------------------------------------------------+
1021
1022 PURPOSE : This function deletes a whole SIM phonebook
1023
1024 */
1025 GLOBAL T_PHB_RETURN pb_del_book (T_PHB_TYPE book)
1026 {
1027 T_PHB_RETURN phb_result;
1028 UBYTE max_record;
1029
1030 TRACE_FUNCTION ("pb_del_book()");
1031
1032 if (GET_PHB_STATE() NEQ PHB_STATE_IDLE)
1033 return PHB_FAIL;
1034
1035 /* Remember book to delete, initialize record to delete */
1036 pba_data.current_book = book;
1037 pba_data.del_record = 1;
1038 max_record = pba_data.phb_record_max[pba_data.current_book];
1039
1040 do
1041 {
1042 /* Delete the record in the database */
1043 phb_result = pb_sim_del_record (pba_data.current_book,
1044 pba_data.del_record,
1045 &pba_data.records_changed);
1046 if (phb_result NEQ PHB_OK)
1047 return phb_result;
1048 pba_data.del_record++;
1049 }
1050 while ((pba_data.del_record <= max_record) AND
1051 (pba_data.records_changed.entries EQ 0));
1052
1053 if (pba_data.records_changed.entries NEQ 0)
1054 {
1055 /* Synchronize to SIM */
1056 SET_PHB_STATE (PHB_STATE_DELETE_BOOK);
1057
1058 /* Start synchronizing record(s) to the SIM */
1059 return pb_sync_next_sim_record (TRUE);
1060 }
1061
1062 return PHB_OK;
1063 }
1064
1065
1066 /*
1067 +----------------------------------------------------------------------------+
1068 | PROJECT : GSM-PS (6147) MODULE : PHB |
1069 | STATE : code ROUTINE : pb_get_fdn_classtype |
1070 +----------------------------------------------------------------------------+
1071
1072 PURPOSE : This function gets the FDN class type
1073
1074 */
1075 GLOBAL T_ACI_CLASS pb_get_fdn_classtype (void)
1076 {
1077 TRACE_FUNCTION ("pb_get_fdn_classtype()");
1078
1079 return pba_data.fdn_classtype;
1080
1081 }
1082
1083
1084 /*
1085 +----------------------------------------------------------------------------+
1086 | PROJECT : GSM-PS (6147) MODULE : PHB |
1087 | STATE : code ROUTINE : pb_get_fdn_mode |
1088 +----------------------------------------------------------------------------+
1089
1090 PURPOSE : This function gets the FDN mode
1091
1092 */
1093 GLOBAL T_PHB_FDN_MODE pb_get_fdn_mode (void)
1094 {
1095 TRACE_FUNCTION ("pb_get_fdn_mode()");
1096
1097 return pba_data.fdn_mode;
1098
1099 }
1100
1101
1102 /*
1103 * Wrapper functions. Wrapper functions don't define own functionality
1104 * except of routing to a different block which is the SIM part of
1105 * the phonebook here. For a description of these functions have
1106 * a look into the SIM part of the phonebook.
1107 */
1108
1109 GLOBAL T_PHB_RETURN pb_read_record (T_PHB_TYPE type,
1110 SHORT phy_recno,
1111 T_PHB_RECORD *entry)
1112 {
1113 T_PHB_RETURN phb_result;
1114
1115 TRACE_FUNCTION ("pb_read_record()");
1116
1117 if (phy_recno EQ 0)
1118 return PHB_FAIL;
1119
1120 switch (GET_PHB_STATE())
1121 {
1122 case PHB_STATE_VERIFY:
1123 /*
1124 * For LDN records, first have a look into the queue
1125 */
1126 if ((type EQ LDN) AND (phy_recno - 1 < pba_data.c_queued))
1127 {
1128 /* Entry freshly dialled and still in the queue */
1129 *entry = pba_data.queued[phy_recno - 1]->entry;
1130 return PHB_OK;
1131 }
1132 phb_result = pb_sim_read_record (type,
1133 (USHORT)(phy_recno - pba_data.c_queued),
1134 entry);
1135 TRACE_EVENT_P3 ("type = %d, phy_recno = %d, phb_result = %d",
1136 type, phy_recno, phb_result);
1137 return phb_result;
1138
1139 case PHB_STATE_READ:
1140 /*
1141 * For LDN records, first have a look into the queue
1142 */
1143 if ((type EQ LDN) AND (phy_recno - 1 < pba_data.c_queued))
1144 {
1145 /* Entry freshly dialled and still in the queue */
1146 *entry = pba_data.queued[phy_recno - 1]->entry;
1147 return PHB_OK;
1148 }
1149 return PHB_FAIL; /* We already know the SIM was changed */
1150
1151 case PHB_STATE_IDLE:
1152 phb_result = pb_sim_read_record (type, phy_recno, entry);
1153 TRACE_EVENT_P3 ("type = %d, phy_recno = %d, phb_result = %d",
1154 type, phy_recno, phb_result);
1155 return phb_result;
1156
1157 default:
1158 return PHB_FAIL;
1159 }
1160 }
1161
1162
1163 GLOBAL T_PHB_RETURN pb_read_alpha_record (T_PHB_TYPE type,
1164 SHORT order_num,
1165 T_PHB_RECORD *entry)
1166 {
1167 TRACE_FUNCTION ("pb_read_alpha_record()");
1168
1169 switch (GET_PHB_STATE())
1170 {
1171 case PHB_STATE_IDLE:
1172 case PHB_STATE_VERIFY:
1173 return pb_sim_read_alpha_record (type, order_num, entry);
1174 default:
1175 return PHB_FAIL;
1176 }
1177 }
1178
1179
1180 GLOBAL T_PHB_RETURN pb_read_number_record (T_PHB_TYPE type,
1181 SHORT order_num,
1182 T_PHB_RECORD *entry)
1183 {
1184 TRACE_FUNCTION ("pb_read_number_record()");
1185
1186 switch (GET_PHB_STATE())
1187 {
1188 case PHB_STATE_IDLE:
1189 case PHB_STATE_VERIFY:
1190 return pb_sim_read_number_record (type, order_num, entry);
1191 default:
1192 return PHB_FAIL;
1193 }
1194 }
1195
1196
1197 GLOBAL T_PHB_RETURN pb_search_name (T_PHB_TYPE type,
1198 T_PHB_MATCH match,
1199 const T_ACI_PB_TEXT *search_tag,
1200 SHORT *order_num)
1201 {
1202 T_PHB_RETURN phb_result;
1203
1204 TRACE_FUNCTION ("pb_search_name()");
1205
1206 switch (GET_PHB_STATE())
1207 {
1208 case PHB_STATE_IDLE:
1209 case PHB_STATE_VERIFY:
1210 TRACE_EVENT_P1 ("order_num = %d", *order_num);
1211 phb_result = pb_sim_search_name (type, match, search_tag, order_num);
1212 if (phb_result EQ PHB_OK) // #HM# Remove before production
1213 {
1214 TRACE_EVENT ("Entry found");
1215 }
1216 else
1217 {
1218 TRACE_EVENT ("Entry not found");
1219 }
1220 return phb_result;
1221
1222 default:
1223 return PHB_FAIL;
1224 }
1225 }
1226
1227
1228 GLOBAL T_PHB_RETURN pb_search_number (T_PHB_TYPE type,
1229 const UBYTE *number,
1230 SHORT *order_num)
1231 {
1232 TRACE_FUNCTION ("pb_search_number()");
1233
1234 switch (GET_PHB_STATE())
1235 {
1236 case PHB_STATE_IDLE:
1237 case PHB_STATE_VERIFY:
1238 return pb_sim_search_number (type, number, order_num);
1239 default:
1240 return PHB_FAIL;
1241 }
1242 }
1243
1244
1245 /*
1246 +----------------------------------------------------------------------------+
1247 | PROJECT : GSM-PS (6147) MODULE : PHB |
1248 | STATE : code ROUTINE : pb_get_max_num_len |
1249 +----------------------------------------------------------------------------+
1250
1251 PURPOSE : This function gets the maximum length of a number within a
1252 given phonebook. The length depends on the given phonebook and
1253 the availability of extension records (SIM service table).
1254
1255 Question: From an architectural point of view, belongs this into
1256 the ACI part of the phonebook or more into the SIM part of the
1257 phonebook? SIM part until now has no information about SIM service
1258 table.
1259
1260 */
1261 LOCAL UBYTE pb_get_max_num_len (T_PHB_TYPE type)
1262 {
1263 TRACE_FUNCTION ("pb_get_max_num_len()");
1264
1265 switch (type)
1266 {
1267 case ECC:
1268 return 3;
1269
1270 case ADN:
1271 case UPN:
1272 case ADN_ME:
1273 if (pb_sim_service (SRV_EXT1))
1274 return 2 * PHB_PACKED_NUM_LEN;
1275 break;
1276
1277 case FDN:
1278 if (pb_sim_service (SRV_EXT2))
1279 return 2 * PHB_PACKED_NUM_LEN;
1280 break;
1281
1282 case SDN:
1283 if (pb_sim_service (SRV_EXT3))
1284 return 2 * PHB_PACKED_NUM_LEN;
1285 break;
1286
1287 case BDN:
1288 if (pb_sim_service (SRV_EXT4))
1289 return 2 * PHB_PACKED_NUM_LEN;
1290 break;
1291
1292 case LDN:
1293 case LRN:
1294 case LMN:
1295 case ME:
1296 return 2 * PHB_PACKED_NUM_LEN;
1297
1298 default:
1299 TRACE_ERROR ("Invalid phonebook type");
1300 return 0;
1301 }
1302 return 20; /* Default according to 11.11 if no EXT records and not ECC */
1303 }
1304
1305
1306 GLOBAL T_PHB_RETURN pb_read_sizes (T_PHB_TYPE type, /* IN */
1307 SHORT *max_rcd, /* OUT */
1308 SHORT *used_rcd, /* OUT */
1309 UBYTE *num_len, /* OUT */
1310 UBYTE *tag_len) /* OUT */
1311 {
1312 T_PHB_RETURN phb_result;
1313
1314 TRACE_FUNCTION ("pb_read_sizes()");
1315
1316 switch (GET_PHB_STATE())
1317 {
1318 case PHB_STATE_IDLE:
1319 case PHB_STATE_VERIFY:
1320
1321 /*
1322 * For a valid phonebook type, set all parameters to zero instead of
1323 * reporting an error.
1324 */
1325 switch (type)
1326 {
1327 case ECC:
1328 case ADN:
1329 case FDN:
1330 case BDN:
1331 case LDN:
1332 case LRN:
1333 case SDN:
1334 case LMN:
1335 case UPN:
1336 /*case ME:
1337 case ADN_ME:*/
1338 *num_len = pb_get_max_num_len (type);
1339 phb_result = pb_sim_read_sizes (type, max_rcd, used_rcd, tag_len);
1340 if (phb_result NEQ PHB_OK)
1341 {
1342 *max_rcd = 0;
1343 *used_rcd = 0;
1344 *num_len = 0;
1345 *tag_len = 0;
1346 }
1347 return PHB_OK;
1348
1349 default:
1350 break;
1351 }
1352 return PHB_FAIL; /* Invalid phonebook */
1353
1354 default:
1355 return PHB_FAIL; /* Invalid state */
1356 }
1357 }
1358
1359
1360 GLOBAL int pb_get_entry_len (const UBYTE *pb_tag,
1361 UBYTE max_pb_len)
1362 {
1363 TRACE_FUNCTION ("pb_get_entry_len()");
1364
1365 return pb_sim_get_entry_len (pb_tag, max_pb_len);
1366 }
1367
1368
1369 GLOBAL int pb_find_free_record (T_PHB_TYPE type)
1370 {
1371 /*
1372 * This function finds the number of the first free record in
1373 * a given phonebook. This is new functionality as from S621.
1374 */
1375 TRACE_FUNCTION ("pb_find_free_record()");
1376
1377 switch (GET_PHB_STATE())
1378 {
1379 case PHB_STATE_IDLE:
1380 case PHB_STATE_VERIFY:
1381 return pb_sim_find_free_record (type);
1382 default:
1383 return PHB_FAIL;
1384 }
1385 }
1386
1387
1388 /*
1389 * Internal functions, not part of the interface
1390 */
1391
1392
1393
1394 /*
1395 +----------------------------------------------------------------------------+
1396 | PROJECT : GSM-PS (6147) MODULE : PHB |
1397 | STATE : code ROUTINE : pb_sim_service |
1398 +----------------------------------------------------------------------------+
1399
1400 PURPOSE : This function checks whether a given service is
1401 "allocated and activated" on the SIM, if so TRUE is returned,
1402 otherwise the function returns FALSE.
1403
1404 No, using psaSIM_ChkSIMSrvSup() is not a good alternative here
1405 as we may have to handle the situation where the SIM service
1406 table itself has been changed, in this case we are not signalled
1407 by the ACI when we have again a valid SIM service table, therefore
1408 this function exist here within the phonebook.
1409
1410 */
1411 LOCAL BOOL pb_sim_service (UBYTE nr)
1412 {
1413 UBYTE byte_offset;
1414 UBYTE bit_offset;
1415 UBYTE service;
1416
1417 TRACE_FUNCTION ("pb_sim_service()");
1418
1419 if (nr > MAX_SRV_TBL * 4)
1420 return FALSE; /* Table overflow */
1421
1422 byte_offset = (nr - 1) / 4;
1423 bit_offset = ((nr - 1) & 0x03) * 2;
1424 service = (pba_data.sim_service_table[byte_offset] >> bit_offset) & 0x03;
1425 return (service EQ ALLOCATED_AND_ACTIVATED);
1426 }
1427
1428
1429 /*
1430 +----------------------------------------------------------------------------+
1431 | PROJECT : GSM-PS (6147) MODULE : PHB |
1432 | STATE : code ROUTINE : pb_read_sim_record |
1433 +----------------------------------------------------------------------------+
1434
1435 PURPOSE : This function sends a SIM_READ_RECORD_REQ to the SIM card.
1436 On success this function returns TRUE, otherwise FALSE.
1437
1438 */
1439 LOCAL BOOL pb_read_sim_record (USHORT data_id, UBYTE rcd_num, UBYTE len)
1440 {
1441 SHORT table_id;
1442
1443 TRACE_FUNCTION ("pb_read_sim_record()");
1444
1445 table_id = psaSIM_atbNewEntry();
1446
1447 if (table_id NEQ NO_ENTRY)
1448 {
1449 /* Allocate room for the SIM data */
1450 if (pba_data.data EQ NULL)
1451 {
1452 ACI_MALLOC (pba_data.data, SIM_MAX_RECORD_SIZE);
1453 }
1454 simShrdPrm.atb[table_id].ntryUsdFlg = TRUE;
1455 simShrdPrm.atb[table_id].accType = ACT_RD_REC;
1456 simShrdPrm.atb[table_id].reqDataFld = data_id;
1457 simShrdPrm.atb[table_id].recNr = rcd_num;
1458 if (rcd_num EQ 1)
1459 simShrdPrm.atb[table_id].dataLen = NOT_PRESENT_8BIT;
1460 else
1461 simShrdPrm.atb[table_id].dataLen = len;
1462 simShrdPrm.atb[table_id].exchData = pba_data.data;
1463 simShrdPrm.atb[table_id].rplyCB = pb_read_sim_record_cb;
1464
1465 simShrdPrm.aId = table_id;
1466
1467 if (psaSIM_AccessSIMData() < 0)
1468 {
1469 TRACE_EVENT("FATAL ERROR");
1470 return FALSE;
1471 }
1472 return TRUE;
1473 }
1474
1475 return FALSE;
1476 }
1477
1478
1479 /*
1480 +----------------------------------------------------------------------------+
1481 | PROJECT : GSM-PS (6147) MODULE : PHB |
1482 | STATE : code ROUTINE : pb_read_sim |
1483 +----------------------------------------------------------------------------+
1484
1485 PURPOSE : This function sends a SIM_READ_REQ to the SIM card.
1486 On success this function returns TRUE, otherwise FALSE.
1487
1488 */
1489 LOCAL BOOL pb_read_sim (USHORT data_id, UBYTE len, UBYTE max_length)
1490 {
1491 SHORT table_id;
1492
1493 TRACE_FUNCTION ("pb_read_sim()");
1494
1495 table_id = psaSIM_atbNewEntry();
1496
1497 if (table_id NEQ NO_ENTRY)
1498 {
1499 /* Allocate room for the SIM data */
1500 if (pba_data.data EQ NULL)
1501 {
1502 ACI_MALLOC (pba_data.data, SIM_MAX_RECORD_SIZE);
1503 }
1504
1505 simShrdPrm.atb[table_id].ntryUsdFlg = TRUE;
1506 simShrdPrm.atb[table_id].accType = ACT_RD_DAT;
1507 simShrdPrm.atb[table_id].reqDataFld = data_id;
1508 simShrdPrm.atb[table_id].dataOff = 0;
1509 simShrdPrm.atb[table_id].dataLen = len;
1510 simShrdPrm.atb[table_id].recMax = max_length;
1511 simShrdPrm.atb[table_id].exchData = NULL;
1512 simShrdPrm.atb[table_id].rplyCB = pb_read_sim_cb;
1513
1514 simShrdPrm.aId = table_id;
1515
1516 if (psaSIM_AccessSIMData() < 0)
1517 {
1518 TRACE_EVENT("FATAL ERROR");
1519 return FALSE;
1520 }
1521 return TRUE;
1522 }
1523 return FALSE;
1524 }
1525
1526
1527 /*
1528 +----------------------------------------------------------------------------+
1529 | PROJECT : GSM-PS (6147) MODULE : PHB |
1530 | STATE : code ROUTINE : pb_read_next_sim_book |
1531 +----------------------------------------------------------------------------+
1532
1533 PURPOSE : This function starts reading the phonebooks based on the
1534 information whether the phonebook is activated and allocated
1535 and still has to be read at all.
1536
1537 */
1538 LOCAL void pb_read_next_sim_book (void)
1539 {
1540 TRACE_FUNCTION("pb_read_next_sim_book()");
1541
1542 /*
1543 * The next three phonebooks ADN, MSISDN and LND share EXT1 records
1544 */
1545 if ((pba_data.fdn_mode NEQ FDN_ENABLE) AND
1546 pb_sim_service (SRV_ADN) AND !pba_data.book_read[ADN])
1547 {
1548 /* Read ADN from SIM card */
1549 pba_data.current_book = ADN;
1550 pba_data.book_read[ADN] = TRUE;
1551 pb_read_sim_record (SIM_ADN, 1, NOT_PRESENT_8BIT);
1552 return;
1553 }
1554
1555 if (pb_sim_service (SRV_MSISDN) AND !pba_data.book_read[UPN])
1556 {
1557 /* Read MSISDN from SIM card */
1558 pba_data.current_book = UPN;
1559 pba_data.book_read[UPN] = TRUE;
1560 pb_read_sim_record (SIM_MSISDN, 1, NOT_PRESENT_8BIT);
1561 return;
1562 }
1563
1564 #ifdef SIM_LND_SUPPORT
1565 if (pb_sim_service (SRV_LDN) AND !pba_data.book_read[LDN])
1566 {
1567 /* Read LND from SIM card */
1568 pba_data.current_book = LDN;
1569 pba_data.book_read[LDN] = TRUE;
1570 pb_read_sim_record (SIM_LND, 1, NOT_PRESENT_8BIT);
1571 return;
1572 }
1573 #endif
1574
1575 #ifdef SIM_LND_SUPPORT
1576 if (!pb_sim_service (SRV_LDN) AND !pba_data.book_read[LDN])
1577 #else
1578 if (!pba_data.book_read[LDN])
1579 #endif
1580 {
1581 /*
1582 * SIM card without LDN service. Create SIM_ICI and SIM_OCI here.
1583 * Don't create obsolete SIM_LND
1584 */
1585 pba_data.book_read[LDN] = TRUE;
1586
1587 /* Create now also some helper elementary files as of R99+ */
1588 //(void)pb_sim_create_ef (SIM_ICI, SIZE_EF_ICI, NR_EF_ICI);
1589 (void)pb_sim_create_ef (FFS_LRN, SIZE_EF_LRN, NR_EF_LRN);
1590 (void)pb_sim_create_ef (FFS_EXT_LRN, SIZE_EF_EXT_LRN, NR_EF_EXT_LRN);
1591 (void)pb_sim_create_ef (FFS_LMN, SIZE_EF_LMN, NR_EF_LMN);
1592 (void)pb_sim_create_ef (FFS_EXT_LMN, SIZE_EF_EXT_LMN, NR_EF_EXT_LMN);
1593
1594 (void)pb_sim_create_ef (SIM_OCI, SIZE_EF_OCI, NR_EF_OCI);
1595 (void)pb_sim_create_ef (SIM_EXT5, SIZE_EF_EXT5, NR_EF_EXT5);
1596
1597 pba_data.phb_record_max[LDN] = NR_EF_OCI;
1598 pba_data.phb_record_len[LDN] = SIZE_EF_OCI;
1599 pba_data.book_created[LDN] = TRUE;
1600
1601 //pba_data.phb_record_max[LRN] = NR_EF_ICI; /* Shared! */
1602 //pba_data.phb_record_len[LRN] = SIZE_EF_ICI;
1603 pba_data.phb_record_max[LRN] = NR_EF_LRN;
1604 pba_data.phb_record_len[LRN] = SIZE_EF_LRN;
1605 pba_data.book_created[LRN] = TRUE;
1606
1607 //pba_data.phb_record_max[LMN] = NR_EF_ICI; /* Shared! */
1608 //pba_data.phb_record_len[LMN] = SIZE_EF_ICI;
1609 pba_data.phb_record_max[LMN] = NR_EF_LMN;
1610 pba_data.phb_record_len[LMN] = SIZE_EF_LMN;
1611 pba_data.book_created[LMN] = TRUE;
1612
1613 pba_data.ext_record_max[EXT5] = NR_EF_EXT5;
1614 pba_data.ext_record_max[EXT_LRN] = NR_EF_EXT_LRN;
1615 pba_data.ext_record_max[EXT_LMN] = NR_EF_EXT_LMN;
1616
1617 pba_data.ext_record_max[EXT5] = SIZE_EF_EXT5;
1618 pba_data.ext_record_max[EXT_LRN] = SIZE_EF_EXT_LRN;
1619 pba_data.ext_record_max[EXT_LMN] = SIZE_EF_EXT_LMN;
1620 }
1621
1622 #ifdef SIM_LND_SUPPORT
1623 if (pba_data.book_read[ADN] OR
1624 pba_data.book_read[UPN] OR
1625 pba_data.book_read[LDN])
1626 #else
1627 if (pba_data.book_read[ADN] OR
1628 pba_data.book_read[UPN])
1629 #endif
1630 {
1631 /* At least one of the books referencing EXT1 has been read */
1632 if (!pba_data.ext_created[EXT1] AND
1633 pb_sim_service (SRV_EXT1))
1634 {
1635 /* Reading to get the number of records only */
1636 pba_data.ext_dummy_read = TRUE;
1637 pb_read_sim_record (SIM_EXT1, 1, NOT_PRESENT_8BIT);
1638 return;
1639 }
1640 }
1641
1642 /*
1643 * FDN phonebook only to be read if ADN is blocked
1644 */
1645 if ((pba_data.fdn_mode EQ FDN_ENABLE) AND
1646 pb_sim_service (SRV_FDN) AND !pba_data.book_read[FDN])
1647 {
1648 /* Read FDN from SIM card */
1649 pba_data.current_book = FDN;
1650 pba_data.book_read[FDN] = TRUE;
1651 pb_read_sim_record (SIM_FDN, 1, NOT_PRESENT_8BIT);
1652 return;
1653 }
1654
1655 if (pba_data.book_read[FDN] AND
1656 !pba_data.ext_created[EXT2] AND
1657 pb_sim_service (SRV_EXT2))
1658 {
1659 /* Reading to get the number of records only */
1660 pba_data.ext_dummy_read = TRUE;
1661 pb_read_sim_record (SIM_EXT2, 1, NOT_PRESENT_8BIT);
1662 return;
1663 }
1664
1665 if (pb_sim_service (SRV_BDN) AND !pba_data.book_read[BDN])
1666 {
1667 /* Read BDN from SIM card */
1668 pba_data.current_book = BDN;
1669 pba_data.book_read[BDN] = TRUE;
1670 pb_read_sim_record (SIM_BDN, 1, NOT_PRESENT_8BIT);
1671 return;
1672 }
1673
1674 if (pba_data.book_read[BDN] AND
1675 !pba_data.ext_created[EXT4] AND
1676 pb_sim_service (SRV_EXT4))
1677 {
1678 /* Reading to get the number of records only */
1679 pba_data.ext_dummy_read = TRUE;
1680 pb_read_sim_record (SIM_EXT4, 1, NOT_PRESENT_8BIT);
1681 return;
1682 }
1683
1684 if (pb_sim_service (SRV_SDN) AND !pba_data.book_read[SDN])
1685 {
1686 /* Read SDN from SIM card */
1687 pba_data.current_book = SDN;
1688 pba_data.book_read[SDN] = TRUE;
1689 pb_read_sim_record (SIM_SDN, 1, NOT_PRESENT_8BIT);
1690 return;
1691 }
1692
1693 if (pba_data.book_read[SDN] AND
1694 !pba_data.ext_created[EXT3] AND
1695 pb_sim_service (SRV_EXT3))
1696 {
1697 /* Reading to get the number of records only */
1698 pba_data.ext_dummy_read = TRUE;
1699 pb_read_sim_record (SIM_EXT3, 1, NOT_PRESENT_8BIT);
1700 return;
1701 }
1702
1703 /*
1704 * For the phonebook entries which reside on SIM but don't have
1705 * timestamps some checks could be done here.
1706 * For example, we could read based on EF_LDN, subsequently we
1707 * could read the informations solely stored in the flash and
1708 * then we could decide whether we through away the last dialled
1709 * numbers or don't.
1710 * Probably this is overkill, old Nokia 5110 and more recent Nokia 6330i
1711 * simply does not support EF_LND at all but stores the last dialled
1712 * numbers in the flash.
1713 */
1714
1715 #ifdef SIM_TOOLKIT
1716 if (simShrdPrm.fuRef >= 0)
1717 {
1718 psaSAT_FUConfirm (simShrdPrm.fuRef, SIM_FU_SUCC_ADD);
1719 }
1720 #endif
1721
1722 /* Free SIM exchange data */
1723 if (pba_data.data NEQ NULL)
1724 {
1725 ACI_MFREE (pba_data.data);
1726 pba_data.data = NULL;
1727 }
1728
1729 SET_PHB_STATE (PHB_STATE_IDLE);
1730
1731 if (pb_read_queue() EQ PHB_EXCT)
1732 return; /* Queued LDN entry is written */
1733
1734 /* Finished reading phonebooks. Flush data. */
1735 (void)pb_sim_flush_data ();
1736
1737 /* Inform ACI that we're ready */
1738 pb_status_ind (PHB_READY, CME_ERR_NotPresent);
1739 }
1740
1741
1742 /*
1743 +----------------------------------------------------------------------------+
1744 | PROJECT : GSM-PS (6147) MODULE : PHB |
1745 | STATE : code ROUTINE : pb_read_sim_record_cb |
1746 +----------------------------------------------------------------------------+
1747
1748 PURPOSE : This function is the callback when a SIM record has been read
1749 (SIM_READ_RECORD_CNF).
1750
1751 */
1752
1753 LOCAL void pb_read_sim_record_cb (SHORT table_id)
1754 {
1755 T_PHB_RETURN phb_result; /* Phonebook result code */
1756 T_PHB_TYPE phb_type; /* Phonebook type */
1757 T_EXT_TYPE ext_type; /* Extension record type */
1758 T_PHB_TYPE next_phb_type; /* Phonebook type of record to read */
1759 T_EXT_TYPE next_ext_type; /* Extension record type of record to read */
1760 USHORT reqDataFld; /* requested datafield identifier */
1761 UBYTE recNr; /* record number */
1762 UBYTE dataLen; /* Length of data */
1763 UBYTE recMax; /* Maximum number of records */
1764 USHORT errCode; /* Error code from the SIM */
1765 USHORT ext_record_ef; /* Extension record elementary file id */
1766 UBYTE ext_record_no; /* Extension record number */
1767 BOOL changed; /* Record changed by pb_sim_write_ef() */
1768 BOOL write_record; /* Write the record */
1769
1770 TRACE_FUNCTION ("pb_read_sim_record_cb()");
1771
1772 /*
1773 * Currently we have a limitation of 254 records in the code.
1774 * This is because there is a note in 11.11 version 8.3.0 clause 6.4.2
1775 * that there is a limit of 254 records for linear fixed.
1776 * Same for USIM looking into 31.102 Annex G.
1777 * However this constraint is not valid for cyclic, and also this
1778 * note has not been present in 11.11 version 7.4.0 clause 6.2.4,
1779 * there is a limit of 255 records mentioned.
1780 * Not supporting the 255th record is a minor issue for now.
1781 * To avoid crashing, we have to limit here for now to 254.
1782 * This is not a theoretical issue only as there exists SIMs
1783 * which have 255 ADN entries.
1784 */
1785 if (simShrdPrm.atb[table_id].recMax EQ 255)
1786 simShrdPrm.atb[table_id].recMax = 254;
1787
1788 /* Get the important information from the read record from T_SIM_ACC_PRM */
1789 reqDataFld = simShrdPrm.atb[table_id].reqDataFld;
1790 recNr = simShrdPrm.atb[table_id].recNr;
1791 dataLen = simShrdPrm.atb[table_id].dataLen;
1792 recMax = simShrdPrm.atb[table_id].recMax;
1793 errCode = simShrdPrm.atb[table_id].errCode;
1794
1795 phb_type = pb_get_phb_type_from_ef (reqDataFld);
1796 ext_type = pb_get_ext_type_from_ef (reqDataFld);
1797
1798 if (phb_type EQ INVALID_PHB AND
1799 ext_type EQ INVALID_EXT)
1800 {
1801 TRACE_ERROR ("Invalid reqDataFld!");
1802 simShrdPrm.atb[table_id].ntryUsdFlg = FALSE;
1803 return;
1804 }
1805
1806 /* We assume we write the record until we have certain errors */
1807 write_record = TRUE;
1808
1809 if (errCode NEQ SIM_NO_ERROR)
1810 {
1811 TRACE_EVENT_P1 ("SIM indicated problem code %04x", errCode);
1812 /*
1813 * At least use cleanly deleted data instead of trash.
1814 */
1815 memset (pba_data.data, 0xff, SIM_MAX_RECORD_SIZE);
1816 }
1817
1818 if (ext_type NEQ INVALID_EXT)
1819 {
1820 /*
1821 * Got an extension record.
1822 */
1823 if (!pba_data.ext_created[ext_type])
1824 {
1825 if (errCode NEQ SIM_NO_ERROR)
1826 {
1827 /* We won't create a field on invalid data and won't write */
1828 write_record = FALSE;
1829 pba_data.ext_dummy_read = FALSE;
1830 }
1831 else
1832 {
1833 pba_data.ext_record_max[ext_type] = recMax;
1834 pba_data.ext_record_len[ext_type] = dataLen;
1835
1836 /* Shadow field for extension record not yet created */
1837 (void)pb_sim_create_ef (reqDataFld, dataLen, recMax);
1838 pba_data.ext_created[ext_type] = TRUE;
1839 }
1840
1841 if (pba_data.ext_dummy_read)
1842 {
1843 /*
1844 * We are reading the first extension record which is assumed
1845 * to be not used just to get the number of respective extension
1846 * records.
1847 */
1848 write_record = FALSE;
1849 pba_data.ext_dummy_read = FALSE;
1850 }
1851 }
1852 }
1853 else
1854 {
1855 /*
1856 * Got a normal phonebook record
1857 */
1858 if (!pba_data.book_created[phb_type])
1859 {
1860 if (errCode NEQ SIM_NO_ERROR)
1861 {
1862 /* We won't create a field on invalid data */
1863 write_record = FALSE;
1864 }
1865 else
1866 {
1867 pba_data.phb_record_max[phb_type] = recMax;
1868 pba_data.phb_record_len[phb_type] = dataLen;
1869
1870 /* Shadow field for normal phonebook record not yet created */
1871 (void)pb_sim_create_ef (reqDataFld, dataLen, recMax);
1872 pba_data.book_created[phb_type] = TRUE;
1873
1874 #ifdef SIM_LND_SUPPORT
1875 if (phb_type EQ LDN)
1876 {
1877 /* Create now also some helper elementary files as of R99+ */
1878 (void)pb_sim_create_ef (SIM_OCI, (USHORT)(dataLen + 14), recMax);
1879 pba_data.phb_record_max[LDN] = recMax;
1880 pba_data.phb_record_len[LDN] = dataLen;
1881 pba_data.book_created[LDN] = TRUE;
1882
1883 (void)pb_sim_create_ef (SIM_ICI, (USHORT)(dataLen + 14), recMax);
1884 pba_data.phb_record_max[LRN] = recMax;
1885 pba_data.phb_record_len[LRN] = dataLen;
1886 pba_data.book_created[LRN] = TRUE;
1887
1888 pba_data.phb_record_max[LMN] = recMax;
1889 pba_data.phb_record_len[LMN] = dataLen;
1890 pba_data.book_created[LMN] = TRUE;
1891 }
1892 #endif
1893 }
1894 }
1895 }
1896
1897 /*
1898 * Write raw record. The SIM layer is aware about the data structures
1899 * and gives us the information whether we have to read an/another
1900 * extension record now or whether we can continue reading normal
1901 * phonebook entries.
1902 */
1903 if (write_record)
1904 {
1905 phb_result = pb_sim_write_ef(reqDataFld,
1906 recNr,
1907 dataLen,
1908 pba_data.data,
1909 &changed,
1910 &ext_record_ef,
1911 &ext_record_no);
1912 }
1913 else
1914 {
1915 /*
1916 * New field, record was trash, don't write it, continue reading
1917 * phonebook entries.
1918 */
1919 TRACE_EVENT ("Record not written (trash/dummy ext)");
1920
1921 phb_result = PHB_OK;
1922 changed = FALSE;
1923 ext_record_ef = 0;
1924 ext_record_no = 0;
1925 }
1926
1927 /* Mark SIM table as free now */
1928 simShrdPrm.atb[table_id].ntryUsdFlg = FALSE;
1929
1930 if ((GET_PHB_STATE() EQ PHB_STATE_VERIFY) AND
1931 (phb_result EQ PHB_OK) AND changed)
1932 {
1933 /* We have a change detected. Prevent further reading until all read */
1934 SET_PHB_STATE (PHB_STATE_READ);
1935 }
1936
1937 /*
1938 * Here an optimization is possible. This is, if an extension record
1939 * and already has been read we could skip reading the extension record
1940 * here. The optimization is not worth the effort, so we simply don't do
1941 * it.
1942 */
1943
1944 /*
1945 * Do some protection against garbage
1946 */
1947 if (ext_record_ef NEQ 0)
1948 {
1949 /* Current record references an extension record */
1950 next_ext_type = pb_get_ext_type_from_ef (ext_record_ef);
1951
1952 if ((next_ext_type EQ INVALID_EXT) OR
1953 (ext_record_no < 1) OR
1954 (ext_record_no > 245))
1955 {
1956 TRACE_ERROR ("pb_sim_write_ef() for ext delivered garbage EXT record");
1957 ext_record_ef = 0; /* Try to proceed without this garbage */
1958 }
1959 }
1960
1961 if (ext_record_ef NEQ 0)
1962 {
1963 /* Current record references an extension record */
1964 next_ext_type = pb_get_ext_type_from_ef (ext_record_ef);
1965
1966 /*
1967 * We have to read an extension record.
1968 */
1969 if (ext_type NEQ INVALID_EXT)
1970 {
1971 /*
1972 * It's the first extension record. Remember the paused phonebook.
1973 */
1974 TRACE_EVENT_P4 ("PHB EF %04x record %d references EXT EF %04x record %d",
1975 reqDataFld,
1976 recNr,
1977 ext_record_ef,
1978 ext_record_ef);
1979 pba_data.paused_ef = reqDataFld;
1980 pba_data.paused_no = recNr;
1981 }
1982
1983 /* Read the extension record. */
1984 if (pba_data.ext_created[next_ext_type])
1985 {
1986 (void)pb_read_sim_record (ext_record_ef,
1987 ext_record_no,
1988 pba_data.ext_record_len[next_ext_type]);
1989 }
1990 else
1991 {
1992 (void)pb_read_sim_record (ext_record_ef,
1993 ext_record_no,
1994 NOT_PRESENT_8BIT);
1995 }
1996 }
1997 else if (ext_type NEQ INVALID_EXT)
1998 {
1999 /*
2000 * The record read was the last extension record in the chain.
2001 * Come back reading normal phonebook entries.
2002 */
2003
2004 next_phb_type = pb_get_phb_type_from_ef (pba_data.paused_ef);
2005 //////(void)pb_sim_build_index (next_phb_type);
2006 if (pba_data.paused_no < pba_data.phb_record_max[next_phb_type])
2007 {
2008 /* Continue reading the current phonebook */
2009 (void)pb_read_sim_record (pba_data.paused_ef,
2010 (UBYTE)(pba_data.paused_no + 1),
2011 pba_data.phb_record_len[next_phb_type]);
2012 }
2013 else
2014 {
2015 ////// Update index - for now here
2016 (void)pb_sim_build_index (phb_type);
2017
2018 /* Next book */
2019 pb_read_next_sim_book ();
2020 }
2021 pba_data.paused_ef = 0;
2022 pba_data.paused_no = 0;
2023
2024 }
2025 else if (simShrdPrm.atb[table_id].recNr < simShrdPrm.atb[table_id].recMax)
2026 {
2027 /*
2028 * Normal phonebook record read, no extension referenced and not
2029 * finished. Continue reading the current phonebook
2030 */
2031 //////(void)pb_sim_build_index (phb_type);
2032
2033 (void)pb_read_sim_record (simShrdPrm.atb[table_id].reqDataFld,
2034 (UBYTE)(simShrdPrm.atb[table_id].recNr + 1),
2035 simShrdPrm.atb[table_id].dataLen);
2036 }
2037 else
2038 {
2039 /*
2040 * Normal phonebook record read, no extension referenced and finished
2041 * reading this book. Continue reading the next book.
2042 */
2043 (void)pb_sim_build_index (phb_type);
2044
2045 /* Next book */
2046 pb_read_next_sim_book ();
2047 }
2048 }
2049
2050
2051 /*
2052 +----------------------------------------------------------------------------+
2053 | PROJECT : GSM-PS (6147) MODULE : PHB |
2054 | STATE : code ROUTINE : pb_sat_update_reset |
2055 +----------------------------------------------------------------------------+
2056
2057 PURPOSE : This function marks the given elementary file as unread.
2058
2059 */
2060 LOCAL void pb_sat_update_reset (USHORT ef)
2061 {
2062 T_PHB_TYPE phb_type;
2063
2064 TRACE_FUNCTION ("pb_sat_update_reset()");
2065
2066 phb_type = pb_get_phb_type_from_ef (ef);
2067
2068 if (phb_type NEQ INVALID_PHB)
2069 {
2070 /* Mark book as unread */
2071 pba_data.book_read[phb_type] = FALSE;
2072 }
2073 }
2074
2075
2076 /*
2077 +----------------------------------------------------------------------------+
2078 | PROJECT : GSM-PS (6147) MODULE : PHB |
2079 | STATE : code ROUTINE : pb_read_sim_cb |
2080 +----------------------------------------------------------------------------+
2081
2082 PURPOSE : This function is the callback when a SIM elementary file has
2083 been read (SIM_READ_CNF).
2084 There are two fields which can have been read here, those are
2085 the emergency call numbers and the SIM service table.
2086
2087 */
2088
2089
2090 LOCAL void pb_read_sim_cb (SHORT table_id)
2091 {
2092 TRACE_FUNCTION ("pb_read_sim_cb()");
2093
2094 switch (simShrdPrm.atb[table_id].reqDataFld)
2095 {
2096 case SIM_ECC:
2097 if (simShrdPrm.atb[table_id].errCode EQ SIM_NO_ERROR)
2098 {
2099 (void)pb_sim_set_ecc (simShrdPrm.atb[table_id].dataLen,
2100 simShrdPrm.atb[table_id].exchData);
2101
2102 #ifdef SIM_TOOLKIT
2103 if (simShrdPrm.fuRef >= 0)
2104 {
2105 psaSAT_FUConfirm (simShrdPrm.fuRef, SIM_FU_SUCCESS);
2106 }
2107 #endif /* #ifdef SIM_TOOLKIT */
2108
2109 }
2110 else
2111 {
2112 /*
2113 * Some error from the SIM. Indicate a fallback to the PCM/FFS.
2114 */
2115 (void)pb_sim_set_ecc (0, NULL);
2116
2117 #ifdef SIM_TOOLKIT
2118 if (simShrdPrm.fuRef >= 0)
2119 {
2120 psaSAT_FUConfirm (simShrdPrm.fuRef, SIM_FU_ERROR);
2121 }
2122 #endif
2123
2124 }
2125
2126 if (pba_data.data NEQ NULL)
2127 {
2128 ACI_MFREE (pba_data.data);
2129 pba_data.data = NULL;
2130 }
2131 break;
2132
2133 case SIM_SST:
2134 if (simShrdPrm.atb[table_id].errCode EQ SIM_NO_ERROR)
2135 {
2136 /* copy SIM service table */
2137 memset(pba_data.sim_service_table, 0, MAX_SRV_TBL);
2138 memcpy(pba_data.sim_service_table,
2139 simShrdPrm.atb[table_id].exchData,
2140 MINIMUM(MAX_SRV_TBL, simShrdPrm.atb[table_id].dataLen));
2141
2142 /* start reading SIM phonebook */
2143 simShrdPrm.atb[table_id].ntryUsdFlg = FALSE;
2144 pb_start_build (FALSE);
2145 }
2146 else
2147 {
2148 TRACE_ERROR ("Cannot read SIM service table, using old table");
2149
2150 #ifdef SIM_TOOLKIT
2151 if (simShrdPrm.fuRef >= 0)
2152 {
2153 psaSAT_FUConfirm (simShrdPrm.fuRef, SIM_FU_ERROR);
2154 }
2155 #endif
2156
2157 /* start reading SIM phonebook (fallback action, best effort) */
2158 simShrdPrm.atb[table_id].ntryUsdFlg = FALSE;
2159 pb_start_build (FALSE);
2160 }
2161 break;
2162
2163 default:
2164 TRACE_ERROR ("Unexpected SIM_READ_CNF");
2165 break;
2166 }
2167 simShrdPrm.atb[table_id].ntryUsdFlg = FALSE;
2168 }
2169
2170
2171 /*
2172 +----------------------------------------------------------------------------+
2173 | PROJECT : GSM-PS (6147) MODULE : PHB |
2174 | STATE : code ROUTINE : pb_is_sim_field |
2175 +----------------------------------------------------------------------------+
2176
2177 PURPOSE : This function determines wheter an elementary file has to
2178 be synchronized to the SIM or only exists locally in the FFS.
2179
2180 */
2181 LOCAL BOOL pb_is_sim_field (USHORT ef)
2182 {
2183 TRACE_FUNCTION ("pb_is_sim_field()");
2184
2185 switch (ef)
2186 {
2187 //case SIM_ICI:
2188 case FFS_LRN:
2189 case FFS_LMN:
2190 case FFS_EXT_LRN:
2191 case FFS_EXT_LMN:
2192 case SIM_OCI:
2193 case SIM_EXT5:
2194 return FALSE; /* Only in FFS */
2195
2196 default:
2197 return TRUE; /* Counterpart in SIM exists */
2198 }
2199 }
2200
2201
2202 /*
2203 +----------------------------------------------------------------------------+
2204 | PROJECT : GSM-PS (6147) MODULE : PHB |
2205 | STATE : code ROUTINE : pb_sync_next_sim_record |
2206 +----------------------------------------------------------------------------+
2207
2208 PURPOSE : This function synchronizes a changed record to the SIM.
2209
2210 The first parameter is given to determine whether it's the first
2211 record to be synchronized, in this case ACI has to be informed.
2212
2213 If there was no record to synchronize, PHB_OK is returned.
2214 If there is a record to synchronize, PHB_EXCT is returned.
2215 If there was an internal failure we get PHB_FAIL
2216 (which is unexpected).
2217
2218 */
2219
2220 LOCAL T_PHB_RETURN pb_sync_next_sim_record (BOOL first)
2221 {
2222 T_PHB_RETURN phb_result;
2223 USHORT field_id;
2224 USHORT record;
2225 USHORT entry_size;
2226
2227 TRACE_FUNCTION ("pb_sync_next_sim_record()");
2228
2229 do
2230 {
2231 if (pba_data.records_changed.entries EQ 0)
2232 {
2233 SET_PHB_STATE (PHB_STATE_IDLE);
2234
2235 if (pb_read_queue() EQ PHB_EXCT)
2236 return PHB_EXCT; /* Queued LDN entry is written */
2237
2238 /* Inform SIM layer we're done synchronizing */
2239 (void)pb_sim_flush_data ();
2240
2241 /* Inform ACI that we're ready */
2242 pb_status_ind (PHB_READY, CME_ERR_NotPresent);
2243
2244 return PHB_OK; /* No record to synchronize to the SIM */
2245 }
2246
2247 /* Get the next record from the list */
2248 pba_data.records_changed.entries--;
2249 field_id = pba_data.records_changed.field_id[0];
2250 record = pba_data.records_changed.record[0];
2251
2252 /* Remove the record from the list */
2253 memmove (&pba_data.records_changed.field_id[0],
2254 &pba_data.records_changed.field_id[1],
2255 sizeof (USHORT) * (DB_MAX_AFFECTED - 1));
2256 memmove (&pba_data.records_changed.record[0],
2257 &pba_data.records_changed.record[1],
2258 sizeof (USHORT) * (DB_MAX_AFFECTED - 1));
2259 }
2260 while (!pb_is_sim_field(field_id));
2261
2262 TRACE_EVENT_P3 ("Synch ef %04X record %d, still %d entries to do",
2263 field_id, record, pba_data.records_changed.entries);
2264
2265 /* Allocate room for the SIM data */ // #HM# Check for free
2266 if (pba_data.data EQ NULL)
2267 {
2268 ACI_MALLOC (pba_data.data, SIM_MAX_RECORD_SIZE);
2269 }
2270
2271 /* Fetch record in raw form from the database */
2272 phb_result = pb_sim_read_ef (field_id,
2273 record,
2274 &entry_size,
2275 pba_data.data);
2276
2277 if (phb_result EQ PHB_OK)
2278 {
2279 /* And write it to the SIM */
2280 phb_result = pb_write_sim_record (field_id,
2281 (UBYTE)record,
2282 (UBYTE)entry_size,
2283 pba_data.data);
2284
2285 if (phb_result NEQ PHB_EXCT)
2286 {
2287 /* Inform ACI that we've had a problem */
2288 pb_status_ind ( PHB_WRITE_FAIL, CME_ERR_NotPresent );
2289 return phb_result; /* Some unexpected internal failure */
2290 }
2291
2292 if (first)
2293 {
2294 /* Inform ACI that we're busy synchronizing SIM entries */
2295 pb_status_ind ( PHB_BUSY, CME_ERR_NotPresent );
2296 }
2297
2298 return PHB_EXCT;
2299 }
2300 return phb_result;
2301 }
2302
2303
2304 /*
2305 +----------------------------------------------------------------------------+
2306 | PROJECT : GSM-PS (6147) MODULE : PHB |
2307 | STATE : code ROUTINE : pb_write_sim_record |
2308 +----------------------------------------------------------------------------+
2309
2310 PURPOSE : This function writes a SIM record to the SIM.
2311
2312 */
2313
2314 LOCAL T_PHB_RETURN pb_write_sim_record (USHORT ef,
2315 UBYTE phy_recno,
2316 UBYTE entry_size,
2317 const UBYTE *buffer)
2318 {
2319 SHORT table_id;
2320
2321
2322 TRACE_FUNCTION("pb_write_sim_record()");
2323
2324 table_id = psaSIM_atbNewEntry();
2325 if (table_id EQ NO_ENTRY)
2326 {
2327 TRACE_ERROR ("pb_write_sim_record(): no more table entries");
2328 return (PHB_FAIL);
2329 }
2330
2331 simShrdPrm.atb[table_id].ntryUsdFlg = TRUE;
2332 simShrdPrm.atb[table_id].accType = ACT_WR_REC;
2333 simShrdPrm.atb[table_id].reqDataFld = ef;
2334 simShrdPrm.atb[table_id].recNr = phy_recno;
2335 simShrdPrm.atb[table_id].dataLen = entry_size;
2336 simShrdPrm.atb[table_id].exchData = (UBYTE *)buffer;
2337 simShrdPrm.atb[table_id].rplyCB = pb_write_sim_record_cb;
2338
2339 simShrdPrm.aId = table_id;
2340
2341 if (psaSIM_AccessSIMData() < 0)
2342 {
2343 TRACE_EVENT("pb_write_sim_record(): psaSIM_AccessSIMData() failed");
2344 return (PHB_FAIL);
2345 }
2346
2347 return (PHB_EXCT);
2348 }
2349
2350
2351 /*
2352 +----------------------------------------------------------------------------+
2353 | PROJECT : GSM-PS (6147) MODULE : PHB |
2354 | STATE : code ROUTINE : pb_write_sim_record_cb |
2355 +----------------------------------------------------------------------------+
2356
2357 PURPOSE : This function is the callback when a SIM record has been
2358 written to the SIM.
2359
2360 */
2361
2362 LOCAL void pb_write_sim_record_cb (SHORT table_id)
2363 {
2364 UBYTE max_record;
2365
2366 TRACE_FUNCTION("pb_write_sim_record_cb()");
2367
2368 simShrdPrm.atb[table_id].ntryUsdFlg = FALSE;
2369
2370 if (simShrdPrm.atb[table_id].errCode NEQ SIM_NO_ERROR)
2371 {
2372 /*
2373 * Data in the database and on the SIM probably now differ,
2374 * we can do not much about this.
2375 */
2376 TRACE_ERROR("pb_write_cb(): error for writing");
2377 pb_status_ind (PHB_WRITE_FAIL,
2378 (SHORT)cmhSIM_GetCmeFromSim (simShrdPrm.atb[table_id].errCode));
2379 return;
2380 }
2381
2382 switch (GET_PHB_STATE())
2383 {
2384 case PHB_STATE_WRITE:
2385 /* Synchronize next SIM record */
2386 switch (pb_sync_next_sim_record (FALSE))
2387 {
2388 case PHB_OK: /* All done */
2389 /* Inform the ACI that we're done */
2390 pb_status_ind (PHB_READY, CME_ERR_NotPresent);
2391 break;
2392
2393 case PHB_EXCT:
2394 break; /* Next record => SIM */
2395
2396 case PHB_FAIL:
2397 default:
2398 TRACE_ERROR ("Unexpected phb_result");
2399 break;
2400 }
2401 break;
2402
2403 case PHB_STATE_DELETE_BOOK:
2404 /* Synchronize next SIM record */
2405 switch (pb_sync_next_sim_record (FALSE))
2406 {
2407 case PHB_OK:
2408 /*
2409 * Record synched. Delete next record of phonebook, if available
2410 */
2411 max_record = pba_data.phb_record_max[pba_data.current_book];
2412 while ((pba_data.del_record <= max_record) AND
2413 (pba_data.records_changed.entries EQ 0))
2414 {
2415 /* Delete the record in the database */
2416 if (pb_sim_del_record (pba_data.current_book,
2417 pba_data.del_record,
2418 &pba_data.records_changed) NEQ PHB_OK)
2419 {
2420 TRACE_ERROR ("Unexpected problem from pb_sim_del_record()");
2421
2422 /* Inform SIM layer we're done synchronizing.
2423 * Should we indicate also there was a problem? */
2424 (void)pb_sim_flush_data ();
2425
2426 pb_status_ind (PHB_WRITE_FAIL, CME_ERR_NotPresent);
2427
2428 SET_PHB_STATE (PHB_STATE_IDLE);
2429 return;
2430 }
2431 pba_data.del_record++;
2432 }
2433
2434 if (pba_data.records_changed.entries NEQ 0)
2435 {
2436 /* Synchronizing record(s) to the SIM */
2437 if (pb_sync_next_sim_record (FALSE) NEQ PHB_EXCT)
2438 {
2439 TRACE_ERROR ("Unexpected problem from pb_sync_next_sim_record()");
2440
2441 /* Inform SIM layer we're done synchronizing.
2442 * Should we indicate also there was a problem? */
2443 (void)pb_sim_flush_data ();
2444
2445 pb_status_ind (PHB_WRITE_FAIL, CME_ERR_NotPresent);
2446
2447 SET_PHB_STATE (PHB_STATE_IDLE);
2448 return;
2449 }
2450 }
2451
2452 /*
2453 * We're through
2454 */
2455
2456 (void)pb_sim_flush_data ();
2457
2458 /* Inform the ACI that we're done */
2459 pb_status_ind (PHB_READY, CME_ERR_NotPresent);
2460
2461 SET_PHB_STATE (PHB_STATE_IDLE);
2462 break;
2463
2464 case PHB_EXCT:
2465 break; /* Next record => SIM */
2466
2467 case PHB_FAIL:
2468 default:
2469 TRACE_ERROR ("Unexpected problem synching SIM");
2470
2471 (void)pb_sim_flush_data ();
2472
2473 pb_status_ind (PHB_WRITE_FAIL, CME_ERR_NotPresent);
2474 /* Inform SIM layer we're done synchronizing.
2475 * Should we indicate also there was a problem? */
2476
2477 SET_PHB_STATE (PHB_STATE_IDLE);
2478 break;
2479 }
2480 break;
2481
2482 default:
2483 TRACE_ERROR ("Unexpected default");
2484 break;
2485 }
2486 }
2487
2488
2489 /*
2490 +----------------------------------------------------------------------------+
2491 | PROJECT : GSM-PS (6147) MODULE : PHB |
2492 | STATE : code ROUTINE : pb_get_phb_type_from_ef |
2493 +----------------------------------------------------------------------------+
2494
2495 PURPOSE : This function converts a SIM elementary file number to the
2496 appropriate constant of type T_PHB_TYPE, if applicable.
2497 Non applicable => INVALID_PHB
2498
2499 */
2500 T_PHB_TYPE pb_get_phb_type_from_ef (USHORT ef)
2501 {
2502 TRACE_FUNCTION ("pb_get_phb_type_from_ef()");
2503
2504 switch (ef)
2505 {
2506 case SIM_ECC:
2507 return ECC;
2508
2509 case SIM_ADN:
2510 return ADN;
2511
2512 case SIM_FDN:
2513 return FDN;
2514
2515 case SIM_BDN:
2516 return BDN;
2517
2518 case SIM_LND:
2519 return LDN;
2520
2521 case SIM_MSISDN:
2522 return UPN;
2523
2524 case SIM_SDN:
2525 return SDN;
2526
2527 default:
2528 return INVALID_PHB;
2529 }
2530 }
2531
2532
2533 /*
2534 +----------------------------------------------------------------------------+
2535 | PROJECT : GSM-PS (6147) MODULE : PHB |
2536 | STATE : code ROUTINE : pb_get_ext_type_from_ef |
2537 +----------------------------------------------------------------------------+
2538
2539 PURPOSE : This function converts a SIM elementary file number to the
2540 appropriate constant of T_PHB_EXT_TYPE, if applicable.
2541 Non applicable => INVALID_EXT
2542
2543 */
2544 T_EXT_TYPE pb_get_ext_type_from_ef (USHORT ef)
2545 {
2546 TRACE_FUNCTION ("pb_get_ext_type_from_ef()");
2547
2548 switch (ef)
2549 {
2550 case SIM_EXT1:
2551 return EXT1;
2552
2553 case SIM_EXT2:
2554 return EXT2;
2555
2556 case SIM_EXT3:
2557 return EXT3;
2558
2559 case SIM_EXT4:
2560 return EXT4;
2561
2562 case SIM_EXT5:
2563 return EXT5;
2564
2565 default:
2566 return INVALID_EXT;
2567 }
2568 }
2569
2570
2571 /* ------------------------------------------------------------------------ */
2572
2573 /*
2574 +----------------------------------------------------------------------------+
2575 | PROJECT : GSM-PS (6147) MODULE : PHB |
2576 | STATE : code ROUTINE : pb_search_number_ex |
2577 +----------------------------------------------------------------------------+
2578
2579 PURPOSE : At some places a more sohisticated number searching function
2580 is desirable which can be based on existing phonebook
2581 functionality.
2582
2583 */
2584 GLOBAL T_PHB_RETURN pb_search_number_ex (T_PHB_TYPE type,
2585 const UBYTE *number,
2586 SHORT *order_num,
2587 SHORT *found,
2588 T_PHB_RECORD *entry)
2589 {
2590 T_PHB_RETURN phb_result;
2591 SHORT index;
2592
2593 TRACE_FUNCTION ("pb_search_number_ex()");
2594
2595 /* Search for first entry */
2596 phb_result = pb_search_number (type,
2597 number,
2598 order_num);
2599 if (phb_result NEQ PHB_OK)
2600 return phb_result;
2601
2602 /*
2603 * Determine number of found entries. We are doing a linear algorithm,
2604 * a binary algorithm is possible here to speed up things.
2605 */
2606 if (found NEQ NULL)
2607 {
2608 *found = 1;
2609 index = *order_num;
2610
2611 while (pb_search_number (type, number, &index) EQ PHB_OK)
2612 {
2613 ++(*found);
2614 /* index should not be incremented as lower layer (DB) will take
2615 care of it */
2616 }
2617 }
2618
2619 /* Read the first found entry, if desired */
2620 if (entry NEQ NULL)
2621 return (pb_read_number_record (type, *order_num, entry));
2622 return PHB_OK;
2623 }
2624
2625
2626 /*
2627 +----------------------------------------------------------------------------+
2628 | PROJECT : GSM-PS (6147) MODULE : PHB |
2629 | STATE : code ROUTINE : pb_search_name_ex |
2630 +----------------------------------------------------------------------------+
2631
2632 PURPOSE : At some places a more sohisticated name searching function
2633 is desirable which can be based on existing phonebook
2634 functionality.
2635
2636 */
2637 GLOBAL T_PHB_RETURN pb_search_name_ex (T_PHB_TYPE type,
2638 T_PHB_MATCH match,
2639 const T_ACI_PB_TEXT *search_tag,
2640 SHORT *order_num,
2641 SHORT *found,
2642 T_PHB_RECORD *entry)
2643 {
2644 T_PHB_RETURN phb_result;
2645 SHORT index;
2646
2647 TRACE_FUNCTION ("pb_search_name_ex()");
2648
2649 /* Search for first entry */
2650 phb_result = pb_search_name (type,
2651 match,
2652 search_tag,
2653 order_num);
2654 if (phb_result NEQ PHB_OK)
2655 return phb_result;
2656
2657 /*
2658 * Determine number of found entries. We are doing a linear algorithm,
2659 * a binary algorithm is possible here to speed up things.
2660 */
2661 if (found NEQ NULL)
2662 {
2663 *found = 1;
2664 index = *order_num;
2665
2666 while (pb_search_name (type, match, search_tag, &index) EQ PHB_OK)
2667 {
2668 ++(*found);
2669 /* index should not be incremented as lower layer (DB) will take
2670 care of it */
2671 }
2672
2673 }
2674
2675 /* Read the first found entry, if desired */
2676 if (entry NEQ NULL)
2677 return (pb_read_alpha_record (type, *order_num, entry));
2678 return PHB_OK;
2679 }
2680
2681
2682 /*
2683 +----------------------------------------------------------------------------+
2684 | PROJECT : GSM-PS (6147) MODULE : PHB |
2685 | STATE : code ROUTINE : pb_status_ind |
2686 +----------------------------------------------------------------------------+
2687
2688 PURPOSE : This function indicates the new phonebook status to the ACI
2689 preventing we're indication twice PHB_READY or PHB_BUSY.
2690
2691 */
2692 LOCAL void pb_status_ind (T_PHB_STAT phb_stat, SHORT cmeError)
2693 {
2694 TRACE_FUNCTION ("pb_status_ind()");
2695
2696 /* Avoid to indicate twice busy or ready */
2697 if ((pba_data.phb_stat NEQ phb_stat) OR
2698 ((phb_stat NEQ PHB_READY) AND
2699 (phb_stat NEQ PHB_BUSY)))
2700 {
2701 cmhPHB_StatIndication ((UBYTE)phb_stat, cmeError,TRUE);
2702 pba_data.phb_stat = phb_stat;
2703 }
2704 }
2705
2706
2707 /*
2708 +----------------------------------------------------------------------------+
2709 | PROJECT : GSM-PS (6147) MODULE : PHB |
2710 | STATE : code ROUTINE : pb_write_queue |
2711 +----------------------------------------------------------------------------+
2712
2713 PURPOSE : This function writes a LDN entry to the queue if the SIM is busy.
2714
2715 */
2716 LOCAL T_PHB_RETURN pb_write_queue (const T_PHB_RECORD *entry)
2717 {
2718 TRACE_FUNCTION ("pb_write_queue()");
2719
2720 if (pba_data.c_queued < PHB_MAX_QUEUE)
2721 {
2722 memmove (&pba_data.queued[1],
2723 &pba_data.queued[0],
2724 pba_data.c_queued * sizeof (T_PHB_QUEUE *));
2725 ACI_MALLOC (pba_data.queued[pba_data.c_queued],
2726 sizeof (T_PHB_QUEUE));
2727 pba_data.queued[0]->entry = *entry;
2728 pba_data.c_queued++;
2729 return PHB_EXCT;
2730 }
2731
2732 return PHB_FAIL; /* Queue full */
2733 }
2734
2735
2736 /*
2737 +----------------------------------------------------------------------------+
2738 | PROJECT : GSM-PS (6147) MODULE : PHB |
2739 | STATE : code ROUTINE : pb_clear_queue |
2740 +----------------------------------------------------------------------------+
2741
2742 PURPOSE : This function clears the LDN queue.
2743
2744 */
2745 LOCAL void pb_clear_queue (void)
2746 {
2747 TRACE_FUNCTION ("pb_clear_queue()");
2748
2749 while (pba_data.c_queued NEQ 0)
2750 {
2751 pba_data.c_queued--;
2752 ACI_MFREE (pba_data.queued[pba_data.c_queued]);
2753 pba_data.queued[pba_data.c_queued] = NULL;
2754 }
2755 }
2756
2757
2758 /*
2759 +----------------------------------------------------------------------------+
2760 | PROJECT : GSM-PS (6147) MODULE : PHB |
2761 | STATE : code ROUTINE : pb_read_queue |
2762 +----------------------------------------------------------------------------+
2763
2764 PURPOSE : This function reads the queue and adds the queued phonebook
2765 entries.
2766
2767 */
2768 LOCAL T_PHB_RETURN pb_read_queue (void)
2769 {
2770 T_PHB_RETURN phb_result;
2771
2772 TRACE_FUNCTION ("pb_write_queue()");
2773
2774 while (pba_data.c_queued NEQ 0)
2775 {
2776 /*
2777 * LDN entries have been queued. Write them now.
2778 */
2779 pba_data.c_queued--;
2780 phb_result = pb_add_record (LDN,
2781 pba_data.queued[pba_data.c_queued]->index,
2782 &pba_data.queued[pba_data.c_queued]->entry);
2783 ACI_MFREE (pba_data.queued[pba_data.c_queued]);
2784 pba_data.queued[pba_data.c_queued] = NULL;
2785 switch (phb_result)
2786 {
2787 case PHB_EXCT:
2788 return phb_result;
2789 case PHB_OK:
2790 break;
2791 default:
2792 pb_clear_queue();
2793 return phb_result;
2794 }
2795 }
2796 return PHB_OK;
2797 }
2798
2799 #endif /* #ifdef TI_PS_FFS_PHB */