diff gsm-fw/g23m-aci/aci/phb_sim.c @ 775:eedbf248bac0

gsm-fw/g23m-aci subtree: initial import from LoCosto source
author Michael Spacefalcon <msokolov@ivan.Harhan.ORG>
date Sun, 12 Oct 2014 01:45:14 +0000
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/gsm-fw/g23m-aci/aci/phb_sim.c	Sun Oct 12 01:45:14 2014 +0000
@@ -0,0 +1,3117 @@
+/* 
++----------------------------------------------------------------------------- 
+|  Project :  MMI-Framework (8417)
+|  Modul   :  PHB
++----------------------------------------------------------------------------- 
+|  Copyright 2002 Texas Instruments Berlin, AG 
+|                 All rights reserved. 
+| 
+|                 This file is confidential and a trade secret of Texas 
+|                 Instruments Berlin, AG 
+|                 The receipt of or possession of this file does not convey 
+|                 any rights to reproduce or disclose its contents or to 
+|                 manufacture, use, or sell anything it may describe, in 
+|                 whole, or in part, without the specific written consent of 
+|                 Texas Instruments Berlin, AG. 
++----------------------------------------------------------------------------- 
+|  Purpose :  This modul contains the functions to establish the phone book.
++----------------------------------------------------------------------------- 
+*/
+#ifdef TI_PS_FFS_PHB
+
+#include "aci_all.h"
+#include "aci_cmh.h"
+#include "aci_mem.h"
+
+#include "phb_sim.h"
+#include "phb_aci.h"
+
+#include "ffs/ffs.h"
+
+#ifdef SIM_TOOLKIT
+#include "psa.h"
+#include "psa_sim.h"
+#include "psa_cc.h"
+#include "psa_sat.h"
+#endif /* #ifdef SIM_TOOLKIT */
+
+#include "cmh.h"
+#include "cmh_phb.h"
+#include "pcm.h"
+
+/*
+ * Constants and enumerations
+ */
+ 
+#define MAX_ECC_RCD                5
+#define FFS_IMSI_SIZE              8
+#define MAX_EXTNS_PER_RECORD       9
+#define MAX_ELEM_FILES            15
+#define SIM_MAX_RECORD_SIZE      256 /* Maximum size of a SIM record */
+#define RDM_DATA_FILE_ID      0xff04 /* File ID to store data related to LDN, LMN and LRN Phonebooks */
+#define SIZE_DATA_DATE_TIME       12
+#define MAX_EXT_RECORDS           10
+#define PHB_PHY_NUM_LENGTH        10
+
+
+/*
+ * Type definitions
+ */
+ 
+typedef struct
+{
+	/* Handle for the Database */ 
+  int db_handle;
+
+  /* Maximum number of records */
+  UBYTE max_record[MAX_PHONEBOOK];
+
+  /* Number of used records */
+  UBYTE used_record[MAX_PHONEBOOK];
+
+  /* Records sizes */
+  USHORT record_size[MAX_PHONEBOOK];
+ 
+} T_PHB_SIM_DATA;
+
+/* Maximum no of records for LRN, LDN and LMN Phonebooks. */
+#define RDM_PHB_DATA_SIZE      10
+
+/* Record numbers for LDN, LMN and LRN Phonebook data. */
+#define LDN_DATA_RECNO         1
+#define LMN_DATA_RECNO         2
+#define LRN_DATA_RECNO         3
+
+EXTERN UBYTE cphs_mb_ext_record_num[]; 
+/* ECC records */
+T_PHB_ECC_RECORD phb_ecc_element[MAX_ECC_RCD];
+
+LOCAL T_PHB_SIM_DATA pbs_data;
+
+/* Array to hold Reference count for Extension records. */
+LOCAL UBYTE ext_ref_count[MAX_PHB_EXT][MAX_EXT_RECORDS];
+
+/* Prototypes for search and compare functions */
+int pb_sim_search_alpha_func(ULONG flags, const UBYTE *key, int db_handle, USHORT field_id, USHORT rec_num);
+int pb_sim_search_num_func(ULONG flags, const UBYTE *key, int db_handle, USHORT field_id, USHORT rec_num);
+int pb_sim_alpha_cmp (int db_handle, USHORT field_id, USHORT recno_1, USHORT recno_2, ULONG flags);
+int pb_sim_number_cmp (int db_handle, USHORT field_id,USHORT recno_1,USHORT recno_2, ULONG flags);
+/* 
+ * Prototypes for local functions
+ */
+
+LOCAL BOOL pb_sim_record_empty (USHORT field_id, 
+                                USHORT entry_size, const UBYTE *buffer);
+LOCAL void pb_sim_read_eeprom_ecc (void);
+LOCAL int pb_sim_nibblecopy (UBYTE dest[], int destlen, UBYTE src[], int count);
+LOCAL void pb_sim_revString(char *);
+LOCAL void pb_sim_read_ext(UBYTE *buffer, T_PHB_RECORD *entry);
+LOCAL void pb_sim_prepare_ext_data(UBYTE *ext_data, int ext_count, UBYTE *number, UBYTE no_len, UBYTE *subaddr);
+LOCAL USHORT pb_sim_get_field_id (T_PHB_TYPE type);
+LOCAL USHORT pb_sim_get_ext_file (T_PHB_TYPE type);
+LOCAL USHORT pb_sim_get_ext_file_id (USHORT field_id);
+LOCAL T_EXT_TYPE pb_sim_get_ext_type (USHORT field_id);
+LOCAL USHORT pb_sim_get_size_except_tag (USHORT field_id);
+
+LOCAL int pb_sim_cmpString ( UBYTE* cur_tag, UBYTE* check_tag, UBYTE cmpLen );
+LOCAL void pb_sim_cvt_alpha_for_cmp ( UBYTE* entry_tag, UBYTE* cur_tag, UBYTE len );
+LOCAL int pb_sim_cmp2Bytes(UBYTE *s1, UBYTE *s2, UBYTE len, UBYTE flag);
+LOCAL T_PHB_RETURN pb_sim_update_extn_records(USHORT ext_field_id, USHORT rec_num, SHORT ref_type);
+LOCAL T_PHB_RETURN pb_sim_del_ext_records(T_PHB_TYPE type, USHORT field_id, USHORT db_recno);
+LOCAL USHORT pb_sim_retrieve_rdm_recno (T_PHB_TYPE type);
+LOCAL T_PHB_RETURN pb_sim_del_record_internal (T_PHB_TYPE type,
+                                               USHORT phy_recno,
+                                               T_DB_CHANGED *rec_affected,
+                                               BOOL replacement);
+LOCAL BOOL pb_sim_ext_records_used (T_PHB_TYPE type, 
+                                    USHORT field_id, 
+                                    USHORT db_recno);
+
+
+LOCAL T_PHB_RETURN pb_sim_update_index (T_PHB_TYPE    type, 
+                                        T_DB_CHANGED *rec_affected,
+                                        USHORT        field_id,
+                                        UBYTE         ext_rec_cnt);
+
+/*
++----------------------------------------------------------------------+
+| PROJECT: MMI-Framework                MODULE : PHB                   |
+| STATE  : code                         ROUTINE: pb_sim_init       |
++----------------------------------------------------------------------+
+
+  PURPOSE :   Initializes internal data structures for SIM Phonebook.
+*/
+
+void pb_sim_init (void)
+{
+  USHORT i;
+  
+  TRACE_FUNCTION ("pb_sim_init()");
+  db_init();
+
+  /* Initialise the data structures. */
+
+  /* Initialise ECC Phonebook to contain no records. */
+  for(i = 0; i < MAX_ECC_RCD; i++)
+  {
+    phb_ecc_element[i].phy_idx = 0;
+    memset(phb_ecc_element[i].number,0xFF,ECC_NUM_LEN);
+  }
+
+  return;
+}
+
+/*
++----------------------------------------------------------------------+
+| PROJECT: MMI-Framework                MODULE : PHB                   |
+| STATE  : code                         ROUTINE: pb_sim_exit       |
++----------------------------------------------------------------------+
+
+  PURPOSE :   This function is called by pb_exit() to inform the SIM part 
+              of the phonebook to shut down.
+*/
+void pb_sim_exit (void)
+{
+  TRACE_FUNCTION ("pb_sim_exit()");
+
+  db_exit();
+
+  return;
+}
+
+/*
++----------------------------------------------------------------------+
+| PROJECT: MMI-Framework                MODULE : PHB                   |
+| STATE  : code                         ROUTINE: pb_sim_record_empty   |
++----------------------------------------------------------------------+
+
+  PURPOSE :   This function is used to determine if a record is
+              considered as empty.
+*/
+LOCAL BOOL pb_sim_record_empty (USHORT field_id, 
+                                USHORT entry_size, const UBYTE *data)
+{
+  USHORT alpha_len;
+
+  /* TRACE_FUNCTION ("pb_sim_record_empty()"); */ /* Trace load */
+
+  /* If the entry contains an alpha identifier it is not empty */
+  if (data[0] NEQ 0xFF)
+    return FALSE;
+
+  alpha_len = entry_size - pb_sim_get_size_except_tag (field_id);
+
+//TISH for MMISIM sometime entry_size = 0, then alpha_len < 0;
+#ifdef WIN32
+  if (entry_size == 0)
+  {
+	  return TRUE;
+  }
+#endif
+  if ((data[alpha_len] NEQ 0x00) AND (data[alpha_len + 2] NEQ 0xff))
+    return FALSE;
+  return TRUE; /* No alpha identifier and no phone number => record empty */
+}
+
+
+/*
++----------------------------------------------------------------------+
+| PROJECT: MMI-Framework                MODULE : PHB                   |
+| STATE  : code                         ROUTINE: pb_sim_set_ecc       |
++----------------------------------------------------------------------+
+
+  PURPOSE :   The emergency call numbers are read from SIM card and 
+              written to FFS.
+*/
+T_PHB_RETURN pb_sim_set_ecc (UBYTE ecc_len, const UBYTE *sim_ecc)
+{
+  USHORT ecc_rec_num;
+  UBYTE  *data_ptr;
+  
+  TRACE_FUNCTION ("pb_sim_set_ecc()");
+
+  /* Initialise used records for ECC. */ 
+  pbs_data.used_record[ECC] = 0;
+  
+  /* if SIM ECC data is not empty, copy SIM ECC data to phonebook */
+  if ( ecc_len NEQ 0)
+  {
+    data_ptr         = ( UBYTE *) sim_ecc;
+    pbs_data.max_record[ECC] = (SHORT)((ecc_len/ECC_NUM_LEN) > MAX_ECC_RCD)? MAX_ECC_RCD: ecc_len/ECC_NUM_LEN;
+    pbs_data.record_size[ECC] = ECC_NUM_LEN;
+
+    /* Store ECC into RAM, since ECC records will be less in number. */
+    /*
+     * For the Issue OMAPS00081622
+     * Replaced MAX_ECC_RCD with pbs_data.max_record[ECC] in the for loop
+     */
+    for (ecc_rec_num = 0; ecc_rec_num < pbs_data.max_record[ECC]; ecc_rec_num++)
+    {
+      if(*data_ptr NEQ 0xff)
+      {
+        memset(&phb_ecc_element[ecc_rec_num],0xFF,sizeof(T_PHB_ECC_RECORD));
+        phb_ecc_element[ecc_rec_num].phy_idx = ecc_rec_num + 1;
+        memcpy(phb_ecc_element[ecc_rec_num].number, data_ptr, ECC_NUM_LEN);
+        data_ptr += ECC_NUM_LEN;
+        (pbs_data.used_record[ECC])++;
+      }
+    }
+  }
+  else
+  {
+    pb_sim_read_eeprom_ecc();
+  }
+
+  return PHB_OK;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_create_ef          |
++--------------------------------------------------------------------+
+
+    PURPOSE : Creates a SIM elementary file. 
+
+*/
+T_PHB_RETURN pb_sim_create_ef  (USHORT ef, USHORT record_size, USHORT records)
+{
+  T_DB_INFO_FIELD field_info;
+  T_DB_TYPE db_type;
+  int db_result;
+
+  TRACE_FUNCTION ("pb_sim_create_ef()");
+
+  TRACE_EVENT_P1("Elementary file ID = %x",ef);
+  db_result = db_info_field(pbs_data.db_handle, ef, &field_info);
+
+  /* Check whether file already exists. */
+  if(db_result EQ DB_OK)
+  {
+    /* Check for Record size and No. of records in the present field. */
+    if((field_info.record_size EQ record_size) AND (field_info.num_records EQ records))
+      return PHB_OK;  /* Preserve the existing field. */
+    else
+    {  
+      if(pb_sim_remove_ef(ef) EQ PHB_FAIL)  /* Remove the existing file and recreate the field. */
+				return PHB_FAIL;
+    }
+  }
+  
+	/* Set DB_TYPE depending on the Elementary file. */ 
+  switch(ef)
+  {
+    case SIM_ADN:
+    case SIM_FDN:          
+    case SIM_BDN:  
+    case SIM_SDN:
+    case SIM_EXT1:
+    case SIM_EXT2:
+    case SIM_EXT3:
+    case SIM_EXT4:
+    case SIM_LND:         
+    case SIM_OCI:
+    //case SIM_ICI:
+    case FFS_LRN:
+    case FFS_LMN:
+    case FFS_EXT_LRN:
+    case FFS_EXT_LMN:
+    case SIM_EXT5:
+      db_type = DB_FREELIST;
+      break;
+        
+    case SIM_MSISDN:          
+    case SIM_IMSI:
+      db_type = DB_UNMANAGED;
+      break;
+
+    default:
+      TRACE_ERROR("Invalid ef passed to pb_sim_create_ef()");
+      return PHB_FAIL;        
+  }
+
+  db_result = db_create_field(pbs_data.db_handle, db_type, ef, record_size, records);
+
+  if(db_result EQ DB_OK)
+    return PHB_OK;
+
+  /* Return PHB_FAIL since DB has failed to create File.  */
+  return PHB_FAIL;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_write_ef     |
++--------------------------------------------------------------------+
+
+    PURPOSE : Writes entry_size bytes content of buffer at index to the elementary file ef.
+
+*/
+T_PHB_RETURN pb_sim_write_ef (USHORT ef, USHORT phy_recno,
+                                         USHORT entry_size, const UBYTE *buffer,
+                                         BOOL *changed, USHORT *ext_record_ef, UBYTE  *ext_record_no)
+{
+  int i;
+  T_DB_CHANGED records_affected;
+
+  TRACE_FUNCTION ("pb_sim_write_ef()");
+
+  /* Initialise changed to FALSE. */
+  *changed = FALSE;
+
+  /* Default is no extension record */
+  *ext_record_ef = 0;
+  *ext_record_no = 0;
+
+  /* Check for extension records. */
+  if (!pb_sim_record_empty(ef, entry_size, buffer))
+  {
+    /* The record is not empty (deleted) */
+    switch (ef)
+    {
+      case SIM_ADN:
+      case SIM_MSISDN:
+      case SIM_LND:
+        *ext_record_ef = SIM_EXT1;
+        *ext_record_no = buffer[entry_size - 1];
+        break;
+
+      case SIM_FDN:
+        *ext_record_ef = SIM_EXT2;
+        *ext_record_no = buffer[entry_size - 1];
+        break;
+
+      case SIM_SDN:
+        *ext_record_ef = SIM_EXT3;
+        *ext_record_no = buffer[entry_size - 1];
+        break;
+
+      case SIM_BDN:
+        *ext_record_ef = SIM_EXT4;
+        *ext_record_no = buffer[entry_size - 1];
+        break;
+
+      case FFS_LRN:
+      case FFS_LMN:
+      case SIM_OCI: /* Release 1999 and above 31.102 clause 4.2.34 */
+        *ext_record_ef = SIM_EXT5;
+        *ext_record_no = buffer[entry_size - 15]; // Jirli, please check, 14 instead of 15?
+        break;
+
+      case SIM_EXT1:  /* Extension records can reference other extension records */
+      case SIM_EXT2:
+      case SIM_EXT3:
+      case SIM_EXT4:
+      case SIM_EXT5:
+      case FFS_EXT_LRN:
+      case FFS_EXT_LMN:
+//TISH, patch for OMAPS00124850
+#if 0
+        *ext_record_ef = ef;
+        *ext_record_no = buffer[entry_size - 1];
+#else
+	if (buffer[entry_size - 1] EQ 255 OR buffer[entry_size-1] EQ (UBYTE)phy_recno)
+	{
+		 *ext_record_ef =0;
+		 *ext_record_no=0;
+	}
+	else
+	{
+	        *ext_record_ef = ef;
+	        *ext_record_no = buffer[entry_size - 1];	
+	}
+#endif
+//end
+        break;
+
+      default: /* No extension record defined for this field */
+        break;
+    }
+
+    /* Record is not referring any extensions. So set ef and record_no to ZERO. */
+    if (*ext_record_no EQ 0xff)
+    {
+      *ext_record_ef = 0;
+      *ext_record_no = 0;
+    }
+
+    /* Handle usage counter of extension records */
+    pb_sim_update_extn_records (*ext_record_ef, *ext_record_no, 1);
+
+    /* Write record into FFS */
+    if(db_write_record(pbs_data.db_handle, ef, phy_recno, 0, entry_size, buffer) > DB_OK)
+    {
+      if(db_read_change_log(pbs_data.db_handle, &records_affected) EQ DB_OK)
+      {
+        for(i = 0; i < records_affected.entries; i++)
+        {
+          /* Checking whether Elementary file in the database is changed. */
+          if((records_affected.field_id[i] EQ ef) AND (records_affected.record[i] EQ  phy_recno))
+          {
+            *changed = TRUE;
+            return PHB_OK;
+          }
+        }
+
+        /* Write operation has not changed File in the database. So returning PHB_OK */
+        return PHB_OK;
+      }
+      else    /* Unable to read change log from DB. So returning PHB_FAIL. */
+        return PHB_FAIL;
+    }
+    else   /* Write failure in DB. So returning PHB_FAIL */
+      return PHB_FAIL;
+  }
+  else
+  {
+    /* Empty record */
+    if (db_delete_record (pbs_data.db_handle, ef, phy_recno) NEQ DB_OK)
+      return PHB_FAIL;
+
+    if(db_read_change_log (pbs_data.db_handle, &records_affected) NEQ DB_OK)
+      return PHB_FAIL;
+
+    *changed = (records_affected.entries NEQ 0);
+    return PHB_OK;
+  }
+}
+
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_open          |
++--------------------------------------------------------------------+
+
+    PURPOSE : Opens the SIM phonebook for the given SIM determined by the IMSI.
+*/
+T_PHB_RETURN pb_sim_open (const T_imsi_field *imsi_field, BOOL *changed)
+{
+  T_DB_INFO database_info;
+  UBYTE ffsIMSI[MAX_IMSI_LEN+1];
+  UBYTE simIMSI[MAX_IMSI_LEN+1];
+  UBYTE imsi[FFS_IMSI_SIZE];
+  int db_result,rec_num;
+  UBYTE buffer[RDM_PHB_DATA_SIZE];
+
+  TRACE_FUNCTION("pb_sim_open()");
+
+  /* Initially set SIM changed as TRUE. */
+  *changed = TRUE;
+   
+  /* Open the database. */
+  db_result = db_open(FFS_PHB_DIR);
+
+  TRACE_EVENT_P1("DB handle is %d",db_result);
+
+  if(db_result >= DB_OK)
+  {
+    pbs_data.db_handle = db_result;
+
+    /* Get database info. */
+    db_result = db_info(pbs_data.db_handle, &database_info);
+
+    /* Read IMSI from the FFS if Database is clean. */
+    if((db_result EQ DB_OK) AND (database_info.clean EQ TRUE))
+    {
+      db_result = db_read_record(pbs_data.db_handle, SIM_IMSI, 1, 0, FFS_IMSI_SIZE, imsi);
+
+      /* Compare IMSI read from FFS with IMSI got from SIM. */
+      if(db_result > DB_OK)
+      { 
+        psaSIM_decodeIMSI ((UBYTE*) imsi_field->field,(UBYTE)imsi_field->c_field, (char *)simIMSI);
+
+        psaSIM_decodeIMSI (imsi, FFS_IMSI_SIZE, (char *)ffsIMSI);
+
+        if (!strcmp((char *)simIMSI, (char *)ffsIMSI))
+        { 
+          *changed = FALSE;
+          return PHB_OK;
+        }
+      }
+      else
+      {
+        /* Unable to read IMSI, regenerate database */
+        *changed = TRUE;
+      }
+    }
+
+    /* Remove database whenever database is Inconsistent and SIM is changed. */
+    if(db_close(pbs_data.db_handle) NEQ DB_OK)
+      return PHB_FAIL;
+
+    if(db_remove(FFS_PHB_DIR) NEQ DB_OK)
+      return PHB_FAIL;
+  }/*   if(db_result >= DB_OK) */
+
+  /* Create database: For the first time, whenever SIM is changed 
+              and whenever database is Inconsistent. */
+  db_result = db_create(FFS_PHB_DIR, MAX_ELEM_FILES, TRUE);
+
+  TRACE_EVENT_P1("DB handle is %d",db_result);
+
+  /* Creating DB is successful and valid db_handle is returned */
+  if(db_result >= DB_OK)  
+  {
+    if(db_create_field(pbs_data.db_handle, DB_UNMANAGED, SIM_IMSI, imsi_field->c_field, 1) NEQ DB_OK)
+      return PHB_FAIL;  
+
+    if(db_write_record(pbs_data.db_handle, SIM_IMSI, 1, 0, imsi_field->c_field, imsi_field->field) < DB_OK)
+      return PHB_FAIL;
+
+    /* Create Elementary file to store RDM Phonebook data */
+    if(db_create_field(pbs_data.db_handle, DB_UNMANAGED, RDM_DATA_FILE_ID,RDM_PHB_DATA_SIZE,3) NEQ DB_OK)
+      return PHB_FAIL;
+    
+    /* Initialise data for Recent call lists */
+    for(rec_num = 1; rec_num <= 3; rec_num++)
+    {
+    memset(buffer,0x00, RDM_PHB_DATA_SIZE); 
+
+      if(db_write_record(pbs_data.db_handle, RDM_DATA_FILE_ID, rec_num, 0, RDM_PHB_DATA_SIZE, buffer) < DB_OK)
+      return PHB_FAIL;
+    }
+      
+    return PHB_OK;
+  }
+
+  /* Unable to create Database. So returning PHB_FAIL. */ 
+  return PHB_FAIL;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_read_ef          |
++--------------------------------------------------------------------+
+
+    PURPOSE : Reads *buffer from elementary file ef at index.
+*/
+T_PHB_RETURN pb_sim_read_ef (USHORT ef, USHORT recno, USHORT *entry_size, UBYTE *buffer)
+{
+  int db_result;
+  T_DB_INFO_FIELD field_info;
+
+  TRACE_FUNCTION("pb_sim_read_ef()");
+
+  if(db_info_field(pbs_data.db_handle, ef, &field_info) EQ DB_OK)
+  {
+    *entry_size = field_info.record_size;
+
+    db_result = db_read_record (pbs_data.db_handle,
+                                ef,
+                                recno,
+                                0,
+                                field_info.record_size,
+                                buffer);
+
+    if (db_result > DB_OK)
+      return PHB_OK;  /* Successfully read */
+
+    if (db_result EQ DB_EMPTY_RECORD)
+    {
+      /* Return a deleted record content */
+      memset (buffer, 0xff, *entry_size);
+      return PHB_OK;
+    }
+    return PHB_FAIL; /* Some problem reading record */
+  }
+
+  /* Returning PHB_FAIL, since DB has failed to give Info about the field. */ 
+  return PHB_FAIL;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_remove_ef          |
++--------------------------------------------------------------------+
+
+    PURPOSE : Removes elementary file.
+*/
+T_PHB_RETURN pb_sim_remove_ef (USHORT ef)
+{
+  T_EXT_TYPE ext_type;
+  USHORT rec_num;
+
+  TRACE_FUNCTION("pb_sim_remove_ef()");
+
+  if(db_remove_field(pbs_data.db_handle, ef) EQ DB_OK)
+  
+  {
+    /* Get EXT Type for elementary file */
+    ext_type = pb_sim_get_ext_type(ef);
+
+    /* Reset reference count for extension records */
+    if(ext_type NEQ INVALID_EXT)
+    {
+      for(rec_num = 0; rec_num < MAX_EXT_RECORDS; rec_num++)
+      {
+          ext_ref_count[ext_type][rec_num] = 0;
+      }
+    }
+    return PHB_OK;
+  }
+   
+  return PHB_FAIL;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_build_index          |
++--------------------------------------------------------------------+
+
+    PURPOSE : Builds index for the given phonebook.
+*/
+
+T_PHB_RETURN pb_sim_build_index (T_PHB_TYPE type)
+{
+  USHORT field_id;
+  
+  TRACE_FUNCTION("pb_sim_build_index()");
+
+  field_id = pb_sim_get_field_id(type);
+
+  if(db_update_index(pbs_data.db_handle, field_id, NAME_IDX, pb_sim_alpha_cmp, PHB_MATCH_PARTIAL) NEQ DB_OK)
+    return PHB_FAIL;
+
+  if(db_update_index(pbs_data.db_handle, field_id, NUMBER_IDX, pb_sim_number_cmp, PHB_MATCH_PARTIAL) NEQ DB_OK)
+    return PHB_FAIL;
+  
+  return PHB_OK;
+}
+  
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_flush_data          |
++--------------------------------------------------------------------+
+
+    PURPOSE : This function informs the SIM phonebook that SIM reading has become finished and we reached a consistent state.
+*/
+
+T_PHB_RETURN pb_sim_flush_data  (void)
+{
+
+  TRACE_FUNCTION("pb_sim_flush_data()");
+
+  if(db_flush(pbs_data.db_handle) EQ DB_OK)
+    return PHB_OK;
+
+  return PHB_FAIL;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_add_record   |
++--------------------------------------------------------------------+
+
+    PURPOSE : Add a given record at index to the given phonebook.
+*/
+
+T_PHB_RETURN pb_sim_add_record  (T_PHB_TYPE type, 
+                                 USHORT phy_recno, 
+                           const T_PHB_RECORD *entry,
+                                 T_DB_CHANGED *rec_affected)
+{
+  USHORT field_id, ext_file_id, rec_no;
+  UBYTE    tag_len, max_tag_len;
+  UBYTE   data[SIM_MAX_RECORD_SIZE];
+  UBYTE   buffer[RDM_PHB_DATA_SIZE];
+  T_DB_INFO_FIELD info_field, ext_info_field;
+  UBYTE ext_rec_cnt = 0;
+  UBYTE ext_rec_num1, ext_rec_num2;
+  BOOL  record_empty;
+  int db_result,i;
+
+  TRACE_FUNCTION("pb_sim_add_record()");
+
+  /* Handling of ECC Phonebook */
+  if(type EQ ECC)
+  {
+    if((phy_recno > 0 ) AND (phy_recno <= MAX_ECC_RCD))
+    {
+      phb_ecc_element[phy_recno - 1].phy_idx = phy_recno;
+      memcpy(phb_ecc_element[phy_recno - 1].number,entry->number, ECC_NUM_LEN);
+
+      return PHB_OK;
+    }
+    else
+      return PHB_FAIL;
+  }
+
+  /* Get Elementary file ID for the Phonebook type. */
+  field_id = pb_sim_get_field_id(type);
+  TRACE_EVENT_P1("pb_sim_get_field_id->Field_id: %x", field_id);
+
+  /* Get Extension file for the Phonebook type. */
+  ext_file_id = pb_sim_get_ext_file(type);
+
+  TRACE_EVENT_P1("Ext_Field_id: %x", ext_file_id);
+
+  db_result = db_info_field(pbs_data.db_handle, field_id, &info_field);
+
+  /* Get record size to calculate alpha_len for the entry */
+  if(db_result NEQ DB_OK)
+  {
+    TRACE_EVENT_P1("db_result = %d",db_result);
+    return PHB_FAIL;
+  }
+
+  /* Handling of LDN, LMN and LRN Phonebook records.  */
+  if((type EQ LDN) OR (type EQ LMN) OR (type EQ LRN))
+  {
+    if(phy_recno NEQ 1)
+      return PHB_FAIL;
+    
+    /* Find free record in the elementary file. */
+    db_result = db_find_free_record( pbs_data.db_handle, field_id);
+
+    /* Get record number for the type of Phonebook. */
+    rec_no = pb_sim_retrieve_rdm_recno(type);
+
+     /* Error handling. */
+    if(rec_no EQ 0)
+      return PHB_FAIL;
+
+    /* Read data related to LDN, LMN and LRN Phonebook from FFS */
+    if(db_read_record(pbs_data.db_handle, 
+      RDM_DATA_FILE_ID, rec_no, 0, RDM_PHB_DATA_SIZE, buffer) < DB_OK)
+      return PHB_FAIL;
+
+    /* Database is unable to find the free record. So overwrite the oldest one. */
+    if(db_result < 0)
+      phy_recno = buffer[RDM_PHB_DATA_SIZE - 1];
+    else
+      phy_recno = db_result;
+
+    if(info_field.used_records EQ RDM_PHB_DATA_SIZE)
+      i = RDM_PHB_DATA_SIZE - 1;
+    else
+      i = info_field.used_records;
+
+    /* Move the records as new record has to be added. */
+    for(; i > 0; i--)
+    {
+      buffer[i] = buffer[i-1];  
+    }
+
+    /* Update the record number of new entry. */
+    buffer[0] = (UBYTE) phy_recno;
+
+    /* Write back RDM data back into the database. */
+    if(db_write_record(pbs_data.db_handle, 
+      RDM_DATA_FILE_ID, rec_no, 0, RDM_PHB_DATA_SIZE, buffer) < DB_OK)
+      return PHB_FAIL;
+
+    /* Drop information about changed RDM records from the database. */
+    (void)db_read_change_log(pbs_data.db_handle, rec_affected);
+    rec_affected->entries = 0;
+  }
+  
+  /* Convert T_PHB_TYPE into data structures as described in ETSI 11.11                  */
+  /*  Bytes        Description                                      M/O   Length            |
+  ----------------------------------------------------------------------
+       1 to X          Alpha Identifier                                O       X bytes
+       X+1                Length of BCD number/SSC contents    M       1 byte
+       X+2                TON and NPI                                         M    1 byte
+       X+3 to X+12  Dialling Number/SSC String                   M     10 bytes
+       X+13        Capability/Configuration Identifier          M      1 byte
+       X+14        Extension Record Identifier                   M     1 byte
+       -------------------------------------------------------
+       Extra fields for LDN, LMN, and LMN Phonebook records (As per 31.102)
+       -------------------------------------------------------
+       X+15 to X+21 Outgoing call date and time                  M     7 bytes
+       X+22 to X+24 Outgoing call duration                          M      3 bytes 
+       X+26              Line of the call                                      M    1 byte (New field added to 
+                                                                                                         store line of call)
+  ----------------------------------------------------------------------*/
+
+  tag_len = pb_sim_get_entry_len(entry->tag, PHB_MAX_TAG_LEN);
+
+  max_tag_len = info_field.record_size - pb_sim_get_size_except_tag(field_id);
+
+  if(tag_len > max_tag_len)
+    return PHB_TAG_EXCEEDED;
+
+  memset(data, 0xFF, sizeof(data));
+  memcpy(data, entry->tag, tag_len);
+
+  if(entry->number[10] NEQ 0xFF)
+  {
+    data[max_tag_len] = 11; /* max. length */
+  }
+  else
+  {
+    data[max_tag_len] = entry->len + 1;
+  }
+
+  data[max_tag_len+1] = entry->ton_npi;
+  memcpy((char *)&data[max_tag_len+2], 
+           (char *)entry->number, 10);
+  data[max_tag_len+12] = entry->cc_id;
+
+  /* Copy data specific to records of Phonebook Types (LRN, LDN and LMN). */  
+  if((type EQ LDN) OR (type EQ LRN) OR (type EQ LMN))
+  {
+    if(entry->v_time)
+    {
+      data[max_tag_len+14] = entry->time.year;
+      data[max_tag_len+15] = entry->time.month;
+      data[max_tag_len+16] = entry->time.day;
+      data[max_tag_len+17] = entry->time.hour;
+      data[max_tag_len+18] = entry->time.minute;
+      data[max_tag_len+19] = entry->time.second;
+      data[max_tag_len+20] = entry->time.time_zone;
+
+      data[max_tag_len+21] = (UBYTE)((entry->time.duration >> 16) & 0xff);
+      data[max_tag_len+22] = (UBYTE)((entry->time.duration >> 8) & 0xff);
+      data[max_tag_len+23] = (UBYTE)((entry->time.duration) & 0xff);
+
+    }
+
+    if(entry->v_line)
+    {
+      data[max_tag_len+24] = entry->line;
+    }
+    
+  }
+
+  /* Check how many extension records are needed for number */ 
+  if(entry->number[10] NEQ 0xFF)
+  {
+    ext_rec_cnt = entry ->len - 10;
+
+    if(ext_rec_cnt % 10)
+      ext_rec_cnt = (ext_rec_cnt  / 10) + 1;
+    else
+      ext_rec_cnt = (ext_rec_cnt  / 10);
+  }
+
+  /* Check how many extension records are needed for subaddress */ 
+  if(entry->subaddr[0] NEQ 0xFF)
+  {
+    ext_rec_cnt++;
+  }
+
+  if(entry->subaddr[11] NEQ 0xFF)
+  {
+    ext_rec_cnt++;
+  } 
+
+  TRACE_EVENT_P1("phy_recno = %d",phy_recno);
+
+  TRACE_EVENT_P2("no_rec = %d, used_rec = %d",info_field.num_records,info_field.used_records);
+
+  /* If record number is not mentioned, search for the free record and add the entry. */
+  if(phy_recno EQ 0)
+  {
+    db_result = db_find_free_record( pbs_data.db_handle, field_id);
+
+    /* Database is unable to find the free record. So returning PHB_FULL. */
+    if(db_result EQ DB_RECORD_NOT_FOUND)
+      return PHB_FULL;
+
+    if(db_result < DB_OK)
+      return PHB_FAIL;
+
+    /* Database has returned the free record number. */
+    phy_recno = db_result;
+  }
+  else
+  {
+    /* 
+     * Delete the record, if present to get rid of associated 
+     * ext records. This has the effect that when we afterwards are unable to
+     * create the new record (e.g. lack of extension records) the original 
+     * record remains deleted. Probably this is not a problem, at least it 
+     * should not be a big one.
+     */
+    db_result = db_record_empty (pbs_data.db_handle, 
+                                 field_id, phy_recno, &record_empty);
+    if ((db_result EQ DB_OK) AND !record_empty)
+    {
+      /* 
+       * In case ext_rec is needed and the existing record is not using EXT, 
+       * check if there is ext_rec available before deleting the existing entry
+       * ext_info_field is used to get this info
+       */
+      if (ext_rec_cnt NEQ 0 AND !pb_sim_ext_records_used(type, field_id,phy_recno))
+      {
+        db_result = db_info_field(pbs_data.db_handle, ext_file_id, &ext_info_field);
+        TRACE_EVENT_P3("before delelet rec; db_result = %d, ext_info_field.num_records = %d,  ext_info_field.used_records = %d", db_result, ext_info_field.num_records, ext_info_field.used_records);
+
+ 	      if((ext_info_field.num_records - ext_info_field.used_records) < ext_rec_cnt)
+			   return PHB_EXT_FULL;
+      }
+
+      (void)pb_sim_del_record_internal (type, phy_recno, rec_affected, TRUE);
+    }
+  }
+
+  /* Write record if entry does not require extension records */
+  if(ext_rec_cnt EQ 0)              
+  {
+    /* Write record into FFS  */
+    if(db_write_record(pbs_data.db_handle, field_id, phy_recno, 0, info_field.record_size, data) < DB_OK)
+      return PHB_FAIL;
+
+    /* For LDN record also write to SIM_LDN (without date and time Details */
+    if(type EQ LDN)
+    {
+      memset(&data[max_tag_len + pb_sim_get_size_except_tag(field_id)], 0xFF, 
+        SIZE_DATA_DATE_TIME);
+
+     /* if(db_info_field(pbs_data.db_handle, SIM_LND, &info_field) NEQ DB_OK)
+        return PHB_FAIL;
+
+      if(db_write_record(pbs_data.db_handle, SIM_LND, phy_recno, 0, info_field.record_size, data) < DB_OK)
+        return PHB_FAIL;*/
+    }
+
+  }
+  else        /* We need to find free extension record. */
+  {
+    TRACE_EVENT_P1("Extension records = %d",ext_rec_cnt);
+    /* Search for free extension records */
+
+    /* Check whether extension file has required no. of free records. */
+    db_result = db_info_field(pbs_data.db_handle, ext_file_id, &info_field);
+
+    if((info_field.num_records - info_field.used_records) < ext_rec_cnt)
+    {
+      return PHB_EXT_FULL;
+    }
+
+    /* Required no. of free extension records are found */
+    db_result = db_find_free_record( pbs_data.db_handle,ext_file_id);
+
+    /* DB has returned non-zero positive number. 
+            (Valid record number which is free). */ 
+
+    if(db_result EQ DB_RECORD_NOT_FOUND)
+    {
+      return PHB_EXT_FULL;
+    }
+
+    if(db_result > DB_OK) 
+    {  /* Set Extension record Identifier */
+      ext_rec_num1 = db_result;
+
+      data[max_tag_len+13] = ext_rec_num1;    
+
+      db_result = db_info_field(pbs_data.db_handle, field_id, &info_field);
+
+      /* Write record into FFS */
+      if(db_write_record(pbs_data.db_handle, 
+        field_id, phy_recno, 0, info_field.record_size, data) < DB_OK)
+        return PHB_FAIL;
+        
+    }
+    else
+      return PHB_FAIL;
+
+    db_result = db_info_field(pbs_data.db_handle, ext_file_id, &info_field);
+
+    /* Prepare extension data and write into Extension file after finding the free records. */
+    for(i = 0; i < ext_rec_cnt; i++)
+    {
+//TISH, patch for ASTec34494, added by Jinshu Wang, 2007-08-09
+//start
+			memset(data,0xff,SIM_MAX_RECORD_SIZE);
+//end
+
+      pb_sim_prepare_ext_data(data, i, (UBYTE*)entry->number, entry->len, (UBYTE*)entry->subaddr);
+
+      /* Find next free record to chain the extension records. */
+      db_result = db_find_free_record( pbs_data.db_handle,ext_file_id);
+     
+      if(db_result EQ DB_RECORD_NOT_FOUND)
+      {
+        return PHB_EXT_FULL;
+      }
+
+      if(db_result > DB_OK)
+      {
+        ext_rec_num2 = db_result;
+      }
+      else
+      {
+        return PHB_FAIL;
+      }
+
+      /* Chain extension records and set last link to "FF" */
+      if(i NEQ (ext_rec_cnt - 1))
+        data[12] =  ext_rec_num2;
+          
+      /* Write extension record into FFS */
+      if(db_write_record(pbs_data.db_handle, ext_file_id, ext_rec_num1, 0, info_field.record_size, data) < DB_OK)
+        return PHB_FAIL;
+
+      if(pb_sim_update_extn_records(ext_file_id, ext_rec_num1, 1) EQ PHB_FAIL)
+        return PHB_FAIL;
+
+      /* Preserve the previous free record number. */ 
+      ext_rec_num1 = ext_rec_num2;
+    }
+    
+  }
+
+  /* Get information about changed records from the database. */
+  if(db_read_change_log(pbs_data.db_handle, rec_affected) NEQ DB_OK)
+    return PHB_FAIL;
+
+  /* Implements Measure #167 */
+  return pb_sim_update_index ( type, rec_affected,field_id, ext_rec_cnt);
+
+}
+
+
+/*
++----------------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                          |
+| STATE  : code                         ROUTINE: pb_sim_del_record_internal  |
++----------------------------------------------------------------------------+
+
+    PURPOSE : Delete a given record at phy_recno from the given phonebook.
+              If this function is called internally by a replacement operation
+              on the phonebook this is indicated by the replacement parameter,
+              in this case neither the change log are read nor the ring buffer
+              management is touched in case of circular phonebooks.
+*/
+LOCAL T_PHB_RETURN pb_sim_del_record_internal (T_PHB_TYPE type,
+                                               USHORT phy_recno,
+                                               T_DB_CHANGED *rec_affected,
+                                               BOOL replacement)
+{
+  USHORT field_id, db_recno, rec_no;
+  UBYTE   buffer[RDM_PHB_DATA_SIZE], rec_cnt;
+
+  TRACE_FUNCTION("pb_sim_del_record_internal()");
+
+  /* Handling of ECC Phonebook */
+  if(type EQ ECC)
+  {
+    if((phy_recno > 0 ) AND (phy_recno <= MAX_ECC_RCD))
+    {
+      phb_ecc_element[phy_recno - 1].phy_idx = 0;
+      memset(phb_ecc_element[phy_recno - 1].number, 0xFF, ECC_NUM_LEN);
+
+      return PHB_OK;
+    }
+    else
+      return PHB_FAIL;
+  }
+
+  TRACE_EVENT_P1("phy_recno = %d", phy_recno);
+
+  /* Get Elementary file ID for the Phonebook type. */
+  field_id = pb_sim_get_field_id(type);
+
+  /* Handling of LDN, LMN and LRN Phonebook records.  */
+  if((type EQ LDN) OR (type EQ LMN) OR (type EQ LRN))
+  {
+    /* Get record number for the type of Phonebook. */
+    rec_no = pb_sim_retrieve_rdm_recno(type);
+
+    /* Error handling. */
+    if(rec_no EQ 0)
+      return PHB_FAIL;
+
+    /* Read data related to LDN, LMN and LRN Phonebook from FFS */
+    if(db_read_record(pbs_data.db_handle, 
+        RDM_DATA_FILE_ID,rec_no, 0, RDM_PHB_DATA_SIZE, (UBYTE *) buffer) < DB_OK)
+        return PHB_FAIL;
+
+    /* Retrieve record number in Database from the buffer. */
+    if((buffer[phy_recno - 1] EQ 0) OR (buffer[phy_recno - 1] > RDM_PHB_DATA_SIZE))
+      return PHB_FAIL;
+    
+    db_recno = buffer[phy_recno - 1];
+
+    if (!replacement)
+    {
+      /* Move the records */
+      for(rec_cnt = phy_recno -1; rec_cnt < RDM_PHB_DATA_SIZE - 1; rec_cnt ++)
+      {
+        buffer[rec_cnt] = buffer[rec_cnt + 1]; 
+      }
+
+      /* Update the deleted entry record number. */
+      buffer[RDM_PHB_DATA_SIZE - 1] = 0;
+
+      /* Write back RDM data back into the database. */
+      if(db_write_record(pbs_data.db_handle, 
+        RDM_DATA_FILE_ID, rec_no, 0, RDM_PHB_DATA_SIZE, buffer) < DB_OK)
+        return PHB_FAIL;
+
+      /* Drop information about changed RDM records from the database. */
+      (void)db_read_change_log(pbs_data.db_handle, rec_affected);
+      rec_affected->entries = 0;
+    }
+  }
+  else
+  {
+    db_recno = phy_recno;
+  }
+
+  /* Delete extension records if not referenced. */
+  (void)pb_sim_del_ext_records(type, field_id, db_recno);
+
+  /* Delete record from FFS. */
+  if(db_delete_record(pbs_data.db_handle, field_id,db_recno) NEQ DB_OK)
+    return PHB_FAIL;
+
+  /* Get final information about changed records from the database. */
+  if (!replacement)
+  {
+    if(db_read_change_log(pbs_data.db_handle, rec_affected) NEQ DB_OK)
+      return PHB_FAIL;
+  }
+
+  /* Implements Measure #167 */
+  return pb_sim_update_index ( type, rec_affected,field_id, 0);
+  
+}
+
+
+/*
++----------------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                          |
+| STATE  : code                         ROUTINE: pb_sim_del_record           |
++----------------------------------------------------------------------------+
+
+    PURPOSE : Delete a given record at phy_recno from the given phonebook.
+              This function reads the change log.
+
+*/
+T_PHB_RETURN pb_sim_del_record (T_PHB_TYPE type, 
+                                USHORT phy_recno, 
+                                T_DB_CHANGED *rec_affected)
+{
+  TRACE_FUNCTION ("pb_sim_del_record()");
+  return (pb_sim_del_record_internal (type, phy_recno, rec_affected, FALSE));
+}
+
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_read_record  |
++--------------------------------------------------------------------+
+
+    PURPOSE : Read a given physical record from the flash based phonebook.
+*/
+
+T_PHB_RETURN pb_sim_read_record (T_PHB_TYPE type, USHORT phy_recno, T_PHB_RECORD *entry)
+{
+  USHORT field_id, ext_file_id, rec_no;
+  USHORT db_recno;
+  T_DB_INFO_FIELD info_field;
+  UBYTE   buffer[RDM_PHB_DATA_SIZE];
+  UBYTE *data=NULL, *ptr;
+  UBYTE max_tag_len;
+  UBYTE ext_rcd_num;
+    
+
+  TRACE_FUNCTION("pb_sim_read_record()");
+
+  /* Handling of ECC Phonebook */
+  if(type EQ ECC)
+  {
+    if((phy_recno > 0 ) AND (phy_recno <= MAX_ECC_RCD))
+    {
+      memset(entry,0xFF,sizeof(T_PHB_RECORD));
+      /* Return PHB_FAIL whenever ECC entry is empty */
+      if((phb_ecc_element[phy_recno - 1].number[0] & 0x0F) 
+        EQ 0x0F)
+      {
+        return PHB_FAIL;
+      }
+      else
+      {
+        entry->phy_recno = phy_recno;
+        entry->len = ECC_NUM_LEN;
+        memcpy(entry->number, phb_ecc_element[phy_recno - 1].number, ECC_NUM_LEN);
+        return PHB_OK;
+      }
+    }
+    else
+      return PHB_FAIL;
+  }
+
+  /* Get Elementary file ID for the Phonebook type. */
+  field_id = pb_sim_get_field_id(type);
+
+  /* Get Extension file for the Phonebook type. */
+  ext_file_id = pb_sim_get_ext_file(type);
+
+  /* Read record from the database. */
+  if(db_info_field(pbs_data.db_handle, field_id, &info_field) NEQ DB_OK)
+    return PHB_FAIL;
+  
+  if((phy_recno EQ 0 ) OR (phy_recno > info_field.num_records))
+    return PHB_FAIL; 
+  
+  /* Handling of LDN, LMN and LRN Phonebook records.  */
+  if((type EQ LDN) OR (type EQ LMN) OR (type EQ LRN))
+  {
+    /* Get record number for the type of Phonebook. */
+    rec_no = pb_sim_retrieve_rdm_recno(type);
+
+     /* Error handling. */
+    if(rec_no EQ 0)
+      return PHB_FAIL;
+
+    /* Read data related to LDN,LMN and LRN Phonebook from FFS */
+    if(db_read_record(pbs_data.db_handle, 
+      RDM_DATA_FILE_ID, rec_no, 0, RDM_PHB_DATA_SIZE, (UBYTE *) buffer) < DB_OK)
+      return PHB_FAIL;
+    
+    db_recno = buffer[phy_recno - 1]; 
+  }
+  else
+    db_recno = phy_recno;
+
+  ACI_MALLOC(data,SIM_MAX_RECORD_SIZE);
+
+  if(db_read_record(pbs_data.db_handle, field_id, db_recno, 0, info_field.record_size, data) < DB_OK)
+  { 
+    ACI_MFREE(data);
+    return PHB_FAIL;
+  }
+
+  /* Convert SIM data to the type T_PHB_RECORD. */
+  ptr = data;
+  max_tag_len = info_field.record_size - pb_sim_get_size_except_tag(field_id);
+  if (max_tag_len > PHB_MAX_TAG_LEN)
+    max_tag_len = PHB_MAX_TAG_LEN;
+
+  entry->phy_recno = phy_recno;
+  entry->tag_len = (UBYTE)pb_sim_get_entry_len(ptr, max_tag_len);
+
+  memset(entry->tag, 0xFF, PHB_MAX_TAG_LEN); /* init the tag value */
+  memcpy ( (char*)entry->tag, (char*)ptr, entry->tag_len );
+
+  ptr += (info_field.record_size) - pb_sim_get_size_except_tag(field_id);
+
+  max_tag_len = (info_field.record_size) - pb_sim_get_size_except_tag(field_id);
+  
+  entry->len     = *ptr - 1;
+  ++ptr;
+  entry->ton_npi = *ptr;
+  ++ptr;
+  
+  /* 
+   * This error handling is done to avoid the accidental incorrect 
+   * record length stored in the test SIMs 
+   */
+  if (entry->len > PHB_PACKED_NUM_LEN)
+  {
+     entry->len = PHB_PACKED_NUM_LEN;
+  }
+	  
+  memset(entry->number, 0xFF, PHB_PACKED_NUM_LEN);
+  memcpy( (char*)entry->number, (char*)ptr, entry->len );
+  ptr += 10;
+  entry->cc_id     = *ptr;
+  ++ptr;
+
+  /* Copy data specific to records of LDN, LMN, and LRN phonebook types. */ 
+  if((type EQ LDN) OR (type EQ LMN) OR (type EQ LRN))
+  {
+    entry->v_time = 1;
+    
+    entry->time.year=data[max_tag_len+14] ;
+    entry->time.month=data[max_tag_len+15];
+    entry->time.day=data[max_tag_len+16];
+    entry->time.hour=data[max_tag_len+17] ;
+    entry->time.minute= data[max_tag_len+18];
+    entry->time.second=data[max_tag_len+19] ;
+    entry->time.time_zone=data[max_tag_len+20];
+
+    entry->time.duration = (data[max_tag_len+21] << 16) +
+                           (data[max_tag_len+22] << 8) +
+                           (data[max_tag_len+23] );
+
+    entry->v_line = 1;
+    entry->line=data[max_tag_len+24];
+
+  }
+
+  if (*ptr NEQ 0xFF) /* check for extention records */
+  {
+    ext_rcd_num =(UBYTE)*ptr;
+
+    if(db_info_field(pbs_data.db_handle, ext_file_id, &info_field) NEQ DB_OK)
+    {
+      ACI_MFREE(data); 
+      return PHB_FAIL;
+    }
+
+    /* Reset pointer to start location. */
+    ptr = data;
+
+    memset(ptr, 0xFF, info_field.record_size);
+
+    if(db_read_record(pbs_data.db_handle, ext_file_id, ext_rcd_num, 0, info_field.record_size, ptr) < DB_OK) 
+         {
+               ACI_MFREE(data); 
+//TISH, patch for OMAPS00123396
+//start
+//      return PHB_FAIL;
+			   return PHB_OK;
+//end
+         }
+    pb_sim_read_ext(ptr, entry);
+
+    while(*(ptr + 12) NEQ 0xFF) /* check if a further EXT entry exists */
+    {
+      memset(ptr, 0xFF, info_field.record_size);
+
+      db_read_record(pbs_data.db_handle, ext_file_id, (USHORT)*(ptr+12), 0, info_field.record_size,ptr); 
+
+      pb_sim_read_ext(ptr, entry);     
+    }
+  }
+
+  ACI_MFREE(data);
+  
+  return PHB_OK;
+}
+
+/* Implements Measure #30 */
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_search_name        |
++--------------------------------------------------------------------+
+
+    PURPOSE : Finds the first entry matching the search tag.
+*/
+
+T_PHB_RETURN pb_sim_search_name (T_PHB_TYPE type, T_PHB_MATCH match, const T_ACI_PB_TEXT *search_tag, SHORT *order_num)
+{
+  T_ACI_PB_TEXT key;
+  int res;
+  USHORT field_id;
+  
+  TRACE_FUNCTION("pb_sim_search_name()");
+
+  /* Get Elementary file ID for the Phonebook type. */
+  field_id = pb_sim_get_field_id(type);
+
+
+
+  key.len = search_tag->len;
+  key.cs  = search_tag->cs;
+  pb_sim_cvt_alpha_for_cmp ((UBYTE*) search_tag->data, (UBYTE*) key.data, search_tag->len);
+
+  res = db_search(pbs_data.db_handle, field_id, NAME_IDX, order_num, pb_sim_search_alpha_func, match, (const UBYTE*)&key);
+
+  if(res > DB_OK)
+  {
+    return PHB_OK;
+  }
+
+  return PHB_FAIL;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_search_number         |
++--------------------------------------------------------------------+
+
+    PURPOSE : Finds the first entry matching the number.
+*/
+
+T_PHB_RETURN pb_sim_search_number   (T_PHB_TYPE type, const UBYTE *number, SHORT *order_num)
+{
+  int res, i;
+  CHAR  cur_number[MAX_PHB_NUM_LEN];
+  CHAR  rev_number[MAX_PHB_NUM_LEN];
+  int cmpLen, cmp_res;
+  USHORT field_id;
+   
+  TRACE_FUNCTION("pb_sim_search_number()");
+
+  /* Handling for ECC Phonebook. */
+  if(type EQ ECC)
+  {
+    for(i = 0; i < pbs_data.max_record[ECC]; i++)
+    {
+
+      cmhPHB_getAdrStr(cur_number, MAX_PHB_NUM_LEN - 1,phb_ecc_element[i].number,pbs_data.record_size[ECC]);
+  
+      pb_sim_revString(cur_number);
+
+      strcpy( rev_number,(const char*)number);
+
+      pb_sim_revString(rev_number);
+
+      /*cmpLen = MINIMUM(strlen((const char*) cur_number), strlen((const char*)number));*/
+      cmpLen = strlen((const char*)number);
+      if(cmpLen > 7)
+      {
+        cmpLen = 7;
+      cmp_res = pb_sim_cmpString((UBYTE*)cur_number, (UBYTE*)rev_number, cmpLen);
+      }
+      else
+      {
+        cmp_res = strcmp(cur_number,rev_number);
+      }
+
+	  /*patch for ASTec29800 EFECC by dongfeng*/
+      if(cmp_res EQ 0)
+         return PHB_OK;
+	  }
+
+	  return PHB_FAIL;
+  }
+
+  /* Get Elementary file ID for the Phonebook type. */
+  field_id = pb_sim_get_field_id(type);
+
+
+
+
+  res = db_search(pbs_data.db_handle, field_id, NUMBER_IDX, order_num, pb_sim_search_num_func, PHB_MATCH_PARTIAL, number);
+
+  if(res > DB_OK)
+  {
+    return PHB_OK;
+  }
+
+  return PHB_FAIL;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_read_sizes   |
++--------------------------------------------------------------------+
+
+    PURPOSE : Reads the sizes of a given phonebook.
+*/
+
+T_PHB_RETURN pb_sim_read_sizes (T_PHB_TYPE type,
+                                SHORT      *max_rcd,
+                                SHORT      *used_rcd,
+                                UBYTE      *tag_len,
+                                SHORT      *max_ext,
+                                SHORT      *used_ext)
+{
+  T_DB_INFO_FIELD info_field;
+  USHORT field_id;
+  T_DB_CODE db_result;
+
+  TRACE_FUNCTION("pb_sim_read_sizes()");
+
+  /* Set every output parameter to zero for (not there) */
+  *max_rcd = 0;
+  *used_rcd = 0;
+  *tag_len = 0;
+  *max_ext = 0;
+  *used_ext = 0;
+
+  /* Handling for ECC Phonebook */
+  if(type EQ ECC)
+  {
+    *max_rcd = pbs_data.max_record[ECC];
+    *used_rcd = pbs_data.used_record[ECC];
+    *tag_len = 0; /* To Do:Alpha tag will not be there for ECC. So assigning zero here */
+    return PHB_OK;
+  }
+
+  /* Get elementary file ID for the Phonebook type. */
+  field_id = pb_sim_get_field_id (type);
+
+  /* Query the database about the field */
+  db_result = db_info_field (pbs_data.db_handle, field_id, &info_field);
+
+  if (db_result EQ DB_OK)
+  {
+    /* We got results for the main phonebook entries. */
+    *max_rcd  = info_field.num_records;
+    *used_rcd = info_field.used_records;
+    *tag_len  = info_field.record_size - pb_sim_get_size_except_tag(field_id);
+
+    /* Get elementary file ID for the respective extension. */
+    field_id = pb_sim_get_ext_file (type);
+    if (field_id NEQ 0)
+    {
+      /* 
+       * Extension records may exist for this kind of phonebook. 
+       * Query the database about the ext records.
+       */
+      db_result = db_info_field (pbs_data.db_handle, field_id, &info_field);
+      if (db_result EQ DB_OK)
+      {
+        /* There are also extension records present in the database. */
+        *max_ext  = info_field.num_records;
+        *used_ext = info_field.used_records;
+      }
+      /* There is no else as it is not a problem if there are no extension
+       * records for the phonebook on the SIM as those are optional only. */
+    }
+    return PHB_OK;
+  }
+  return PHB_FAIL; /* The phonebook does not exist */
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_get_entry_len          |
++--------------------------------------------------------------------+
+
+    PURPOSE : Used to get the length in bytes of a given entry which is coded as given in 11.11 Annex B.
+*/
+
+int pb_sim_get_entry_len    (const UBYTE *pb_tag, UBYTE max_pb_len)
+{
+  
+  int   pb_len    = 0;
+  UBYTE inc_count = 1;
+  BOOL  ucs2      = FALSE;
+  UBYTE chars = 0;
+
+  TRACE_FUNCTION("pb_sim_get_entry_len()");
+
+  if (*pb_tag EQ 0x80)
+  {
+    ucs2 = TRUE;
+    inc_count = 2;              /* check two bytes */
+    pb_len = 1;                 /* skip the 0x80 */
+  }
+  else if (*pb_tag EQ 0x81 OR *pb_tag EQ 0x82)
+  {
+    if (*pb_tag EQ 0x81)
+      pb_len = 3;                 /* 0x80 + len + pointer */
+    else
+      pb_len = 4;                 /* 0x80 + len + 2xpointer */
+
+    chars = pb_tag[1];
+    pb_tag += pb_len;                  /* go to data */
+    while (chars)
+    {
+      if (*pb_tag++ & 0x80)
+        pb_len+=2;
+      else
+        pb_len+=1;
+
+      pb_tag++;
+      chars--;
+    }
+    return MINIMUM(pb_len,max_pb_len);
+  }
+
+  while (pb_len < max_pb_len)
+  {
+    if (ucs2 EQ TRUE)
+    {
+      if (!(pb_len+1 < max_pb_len)) /* Check also if we traverse the upper bound */
+        break;                      /* so only a "half" UCS2 element is remaining */
+    }
+    if (pb_tag[pb_len] EQ 0xFF)
+    { 
+      /* one 0xFF indicates the end of a non UCS2 string */
+      if (ucs2 EQ FALSE)
+      {
+        break;
+      }
+      /* two 0xFF indicates the end of a UCS2 string */
+      if (pb_tag[pb_len + 1] EQ 0xFF)
+      {
+        break;
+      }
+    }
+    pb_len += inc_count;
+  }
+
+  return (pb_len);
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_alpha_cmp           |
++--------------------------------------------------------------------+
+
+    PURPOSE : Compares alpha identifier of two records.
+*/
+int pb_sim_alpha_cmp (int db_handle, USHORT field_id, USHORT recno_1, USHORT recno_2, ULONG flags)
+{
+  T_PHB_RECORD entry_1,entry_2;
+  UBYTE cmpLen = 0;
+  UBYTE *buffer;
+  UBYTE max_tag_len;
+  UBYTE cur_tag[PHB_MAX_TAG_LEN], check_tag[PHB_MAX_TAG_LEN];
+  int   cmp_res;
+  T_DB_INFO_FIELD info_field;
+  
+  TRACE_FUNCTION("pb_sim_alpha_cmp()");
+
+  
+  db_info_field(db_handle, field_id, &info_field);
+
+  MALLOC(buffer, info_field.record_size);
+
+  /* Read the first record. */
+  db_read_record(db_handle, field_id, recno_1, 0, info_field.record_size, buffer);
+
+  /* Unpack record 1 to do a string comparison on the alpha identifier field */
+  max_tag_len = info_field.record_size - pb_sim_get_size_except_tag(field_id);
+  if (max_tag_len > PHB_MAX_TAG_LEN)
+    max_tag_len = PHB_MAX_TAG_LEN;
+
+  entry_1.tag_len = (UBYTE)pb_sim_get_entry_len(buffer, max_tag_len);
+  
+  memset(entry_1.tag, 0xFF, PHB_MAX_TAG_LEN); /* init the tag value */
+  memcpy ( (char*)entry_1.tag, (char*)buffer, entry_1.tag_len );
+
+  pb_sim_cvt_alpha_for_cmp ( entry_1.tag, cur_tag, entry_1.tag_len );
+
+  memset(buffer, 0, info_field.record_size);
+
+  /* Read the second record. */
+  db_read_record(db_handle, field_id, recno_2, 0, info_field.record_size, buffer);
+
+  /* Unpack record 2 to do a string comparison on the alpha identifier field */
+  max_tag_len = info_field.record_size - pb_sim_get_size_except_tag(field_id);
+  if (max_tag_len > PHB_MAX_TAG_LEN)
+    max_tag_len = PHB_MAX_TAG_LEN;
+
+  entry_2.tag_len = (UBYTE)pb_sim_get_entry_len(buffer, max_tag_len);
+  
+  memset(entry_2.tag, 0xFF, PHB_MAX_TAG_LEN); /* init the tag value */
+  memcpy ( (char*)entry_2.tag, (char*)buffer, entry_2.tag_len );
+
+  pb_sim_cvt_alpha_for_cmp ( entry_2.tag, check_tag, entry_2.tag_len );
+
+  cmpLen = MINIMUM ( entry_1.tag_len,
+                           entry_2.tag_len );
+
+  TRACE_EVENT_P1("%d", cmpLen);
+      
+  cmp_res = pb_sim_cmpString ( cur_tag, check_tag, cmpLen );
+
+	if (cmp_res EQ 0) 
+  {
+    /* Correct result when length was different, ACIPHB201 */
+    if (entry_1.tag_len < entry_2.tag_len)
+      cmp_res = -1;
+    else if (entry_1.tag_len > entry_2.tag_len)
+      cmp_res = 1;
+  }
+
+  MFREE(buffer);
+
+  return cmp_res;
+}
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_read_number  |
++--------------------------------------------------------------------+
+
+    PURPOSE : Reads a phine number from a given record
+*/
+LOCAL void pb_sim_read_number (int db_handle,
+                               USHORT field_id, 
+                               USHORT recno,
+                               CHAR *number)
+{
+  T_PHB_RECORD entry;
+  T_DB_INFO_FIELD info_field;
+  UBYTE *sim_buffer, *buffer;
+  UBYTE ext_rcd_num;
+  unsigned ext_rec_cnt;
+  USHORT ext_id;
+
+  TRACE_FUNCTION("pb_sim_read_number()");
+
+  ACI_MALLOC(sim_buffer, SIM_MAX_RECORD_SIZE);
+  buffer = sim_buffer;
+
+  db_info_field(db_handle, field_id, &info_field);
+
+  /* Read record recno_1 from the database using db_read_record() */
+  db_read_record (db_handle, field_id, recno, 0, info_field.record_size, buffer);
+
+  /* Read only number from the buffer. */
+  buffer += (info_field.record_size) - pb_sim_get_size_except_tag(field_id);
+  entry.len     = *(buffer++) - 1;
+  entry.ton_npi = *buffer++;
+
+  /* 
+   * This error handling is done to avoid the accidental incorrect 
+   * record length stored in the test SIMs 
+   */
+  if (entry.len > PHB_PHY_NUM_LENGTH)
+  {
+    entry.len = PHB_PHY_NUM_LENGTH;
+  }
+
+  memset (entry.number, 0xFF, PHB_PACKED_NUM_LEN);
+  memcpy (entry.number, buffer, entry.len);
+  buffer += 10; 
+  entry.cc_id = *buffer++;
+  ext_rcd_num = (UBYTE)*buffer;
+  ext_id = pb_sim_get_ext_file_id (field_id);
+  if ((ext_id NEQ 0) AND
+      (db_info_field (db_handle, ext_id, &info_field) EQ DB_OK))
+  {
+    /* Extension records exist and we can obtain information about it */
+    ext_rec_cnt = 0;
+    while ((ext_rcd_num NEQ 0xFF) AND
+           (ext_rcd_num NEQ 0) AND
+           (ext_rcd_num <= info_field.num_records) AND
+           (ext_rec_cnt < info_field.num_records))
+    {
+      /* 
+       * Record not empty, in range 1..max num of ext records
+       * Impossible to have read all records (avoid infinite loop)
+       */
+      ext_rec_cnt++;
+      (void)db_read_record (db_handle, ext_id, ext_rcd_num,
+                            0, info_field.record_size, 
+                            buffer);
+      pb_sim_read_ext (buffer, &entry);
+      ext_rcd_num = sim_buffer[12];
+    }
+  }
+
+  cmhPHB_getAdrStr(number,
+                   MAX_PHB_NUM_LEN - 1,
+                   entry.number,
+                   entry.len);
+
+  ACI_MFREE (sim_buffer);
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_number_cmp   |
++--------------------------------------------------------------------+
+
+    PURPOSE : Compares two numbers of two records.
+*/
+int pb_sim_number_cmp (int db_handle,
+                       USHORT field_id, 
+                       USHORT recno_1, 
+                       USHORT recno_2,
+                       ULONG  flags)
+{
+  CHAR cur_number[MAX_PHB_NUM_LEN];
+  CHAR ref_number[MAX_PHB_NUM_LEN];
+
+  TRACE_FUNCTION("pb_sim_number_cmp()");
+
+  /* Read the numbers */
+  pb_sim_read_number (db_handle, field_id, recno_1, cur_number);
+  pb_sim_read_number (db_handle, field_id, recno_2, ref_number);
+
+  /* Reverse the numbers to compare number from right. */
+  pb_sim_revString(cur_number);
+  pb_sim_revString(ref_number);
+  
+  return pb_sim_cmpWild ((char *)cur_number, (char *)ref_number, MAX_PHB_NUM_LEN);
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_read_ext     |
++--------------------------------------------------------------------+
+
+    PURPOSE : Reads extension records from the database and copies them to number.
+*/
+/* Extesion records will be stored as per 11.11.
+          Bytes         Description      M/O    Length
+          --------------------------------------
+              1        Record type         M    1 byte
+           2 to 12  Extension data  M   11 bytes
+             13        Identifier                 M 1 byte
+          --------------------------------------*/
+
+void pb_sim_read_ext(UBYTE *buffer, T_PHB_RECORD *entry)
+{
+  UBYTE data_len;
+  UBYTE data_type;
+  UBYTE *data;
+  
+  TRACE_FUNCTION("pb_sim_read_ext()");
+
+  /* If this extension record is not empty, it is written in phonebook. */
+  data = buffer;
+ 
+  data_type = *data;
+  data_len = *(data+1);
+
+  switch (data_type)
+  {
+   
+    case 1: /* Called Party Subaddress */
+    {
+			int sa_len = 0;
+      while (sa_len<PHB_PACKED_NUM_LEN)  /* get length of possible already stored subaddr if more than one EXT is used */
+      {
+        if (entry->subaddr[sa_len] EQ 0xFF)
+					break;
+        else if ((entry->subaddr[sa_len] & 0xF0) EQ 0xF0)
+        {
+          sa_len++;
+          break;
+        }
+        else
+          sa_len++;
+      }
+
+      pb_sim_nibblecopy (entry->subaddr,
+                         sa_len,
+                         data + 2,
+                         data_len);
+    }
+    break;
+        
+    case 2: /* Additional data */
+      entry->len =
+      pb_sim_nibblecopy (entry->number,
+                         entry->len,
+                         data + 2,
+                         data_len);
+      break;
+
+      default: /* unknown type */
+        break;
+  }
+
+  return;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_revString                      |
++--------------------------------------------------------------------+
+
+    PURPOSE : Reverses a string within the same variable.
+*/
+
+void pb_sim_revString(char *str)
+{
+  UBYTE i, j,str_len;
+  char ch;
+
+  TRACE_FUNCTION("pb_sim_revString()");
+
+  str_len = strlen(str);
+  
+  for(i = 0, j = str_len - 1;i < (str_len / 2); i++, j--)
+  {
+    ch = *(str + i);
+    *(str + i) = *(str + j);
+    *(str + j) = ch;
+  }
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_search_alpha_func                      |
++--------------------------------------------------------------------+
+
+    PURPOSE : Searches for a given alpha key in the database.
+*/
+
+int pb_sim_search_alpha_func(ULONG flags, const UBYTE *key, int db_handle, USHORT field_id, USHORT rec_num)
+{
+  T_ACI_PB_TEXT *search_key;
+  T_PHB_RECORD entry;
+  UBYTE cmpLen = 0;
+  T_PHB_TYPE phb_type;
+  UBYTE cur_tag[PHB_MAX_TAG_LEN];
+  int   cmp_res;
+
+  TRACE_FUNCTION("pb_sim_search_alpha_func()");
+
+  /* Cast search key to appropriate data structure */
+  search_key = (T_ACI_PB_TEXT *)key;
+
+  /* Get PHB type from field ID using PHB_ACI function. */ 
+  phb_type = pb_get_phb_type_from_ef(field_id);
+
+  /* Read record from the database. */
+  pb_sim_read_record(phb_type, rec_num, &entry);
+	
+  pb_sim_cvt_alpha_for_cmp ( entry.tag, cur_tag, entry.tag_len );
+
+  cmpLen = search_key->len;
+  
+  if(flags EQ PHB_MATCH_PARTIAL)
+    cmpLen = MINIMUM ( entry.tag_len, cmpLen);
+
+  TRACE_EVENT_P1( "cmpLen=%d", cmpLen );
+      
+  cmp_res = pb_sim_cmpString ( cur_tag, search_key->data, cmpLen );
+  
+  return cmp_res;
+}
+
+
+/*
++----------------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                          |
+| STATE  : code                         ROUTINE: pb_sim_search_num_func      |
++----------------------------------------------------------------------------+
+
+    PURPOSE : Searches for a given number key in the database.
+*/
+int pb_sim_search_num_func(ULONG flags, const UBYTE *key, int db_handle,
+                           USHORT field_id, USHORT rec_num)
+{
+  T_PHB_RECORD entry;
+  UBYTE cmpLen = 0;
+  T_PHB_TYPE phb_type;
+  CHAR  cur_number[MAX_PHB_NUM_LEN];
+  CHAR  rev_key[MAX_PHB_NUM_LEN];
+  int   cmp_res;
+  size_t strg_len;
+   
+  TRACE_FUNCTION("pb_sim_search_num_func()");
+
+  /* Get PHB type from field ID using PHB_ACI function. */ 
+  phb_type = pb_get_phb_type_from_ef(field_id);
+
+  /* Read record from the database. */
+  if(pb_sim_read_record(phb_type, rec_num, &entry) NEQ PHB_OK)
+    return -1;
+
+  cmhPHB_getAdrStr(cur_number,
+        MAX_PHB_NUM_LEN - 1,
+        entry.number,
+        entry.len);
+  
+  /* Reverse the first number to compare number from right. */
+  pb_sim_revString(cur_number);
+
+  /* Reverse the second number to compare number from right. */
+  
+  strcpy (rev_key, (const char*) key);
+  
+  pb_sim_revString(rev_key);
+
+  cmpLen = strlen(rev_key);
+
+  if(flags EQ PHB_MATCH_PARTIAL)
+  {
+    strg_len = strlen(cur_number);
+    cmpLen = MINIMUM(strg_len, cmpLen);
+  }
+
+  TRACE_EVENT_P1("Number to be compared: %s", cur_number);
+  TRACE_EVENT_P1("Number to be searched: %s", rev_key);
+  if(cmpLen > 7)
+  {
+    cmpLen = 7;
+  }
+  
+  cmp_res = pb_sim_cmpWild((char*)cur_number, (char*)rev_key, cmpLen);
+  
+
+  TRACE_EVENT_P1("Result of the comparison: %d", cmp_res);
+
+  return cmp_res;
+  
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417)         MODULE: PHB                  |
+| STATE  : code                         ROUTINE: pb_sim_nibblecopy                      |
++--------------------------------------------------------------------+
+
+    PURPOSE : Used to convert BCD nibbles to string.
+*/
+    
+LOCAL int pb_sim_nibblecopy (UBYTE dest[], int destlen, UBYTE src[], int count)
+{
+
+  int i;
+  int nibble;
+
+  int destnibble=destlen*2;
+
+  TRACE_FUNCTION("pb_sim_nibblecopy()");
+
+  if (destnibble)
+  {
+    if ((dest[destlen-1] & 0xF0) == 0xF0)    /* check if there is space in last nibble */
+      destnibble--;
+  }
+
+  for ( i=0; i<count*2; i++ )
+  {
+    /* check if we access out of bounds */
+    if (destnibble/2 >= PHB_PACKED_NUM_LEN)
+      return PHB_PACKED_NUM_LEN;
+
+    /* get nibble */
+    if (i%2 == 0)
+      nibble = src[i/2] & 0x0F;
+    else
+      nibble = (src[i/2] & 0xF0) >> 4;
+
+    if (nibble == 0xF)      /* end of number detected */
+      break;
+
+    /* put nibble */
+    if (destnibble%2 == 0)
+    {
+      dest[destnibble/2] &= 0xF0;
+      dest[destnibble/2] |= nibble;
+    }
+    else
+    {
+      dest[destnibble/2] &= 0x0F;
+      dest[destnibble/2] |= nibble << 4;
+    }
+
+    destnibble++;
+  }
+  return destnibble/2 + destnibble%2; /* round up */
+}
+
+/*
++------------------------------------------------------------------------+
+| PROJECT : MMI-Framework (8417)         MODULE: PHB                     |
+| STATE   : code                         ROUTINE : pb_sim_read_eeprom_ecc    |
++------------------------------------------------------------------------+
+
+  PURPOSE : Read EC number from EEPROM.
+
+*/
+LOCAL void pb_sim_read_eeprom_ecc (void)
+{
+  EF_ECC       efecc;
+  UBYTE        *data_ptr;
+  UBYTE        version;
+  int          rec_ctr;
+
+  TRACE_FUNCTION("pb_sim_read_eeprom_ecc()");
+
+  /* Initialise ECC Phonebook Info. */
+  pbs_data.max_record[ECC] = MAX_ECC_RCD;
+  pbs_data.used_record[ECC] = 0;
+
+  if (pcm_ReadFile((UBYTE *)EF_ECC_ID,
+                    SIZE_EF_ECC,
+                    (UBYTE *)&efecc,
+                    &version) EQ DRV_OK)
+  {   
+    { /* workaround when invalid data stored on PCM */
+      CHAR  ecc_number[MAX_PHB_NUM_LEN];
+      int   num_len;
+
+      data_ptr = efecc.ecc1;
+      
+      for (rec_ctr=0; rec_ctr < pbs_data.max_record[ECC]; rec_ctr++)
+      {
+        if (*data_ptr NEQ 0xFF)
+        {
+          cmhPHB_getAdrStr (ecc_number,
+                            MAX_PHB_NUM_LEN - 1,
+                            data_ptr,
+                            ECC_NUM_LEN);
+          for (num_len = 0; num_len < ECC_NUM_LEN; num_len++)
+          {
+            if (!isdigit (ecc_number[num_len]))
+            {
+              TRACE_EVENT_P2 ("[ERR] pb_read_eeprom_ecc(): invalid character found %c (%d)",
+                              ecc_number[num_len], rec_ctr);
+              return;
+            }
+          }
+        }
+        data_ptr += ECC_NUM_LEN;
+      }
+    } /* workaround end */  
+    
+    data_ptr = &efecc.ecc1[0];
+
+    memset( phb_ecc_element,0xFF, (pbs_data.max_record[ECC] * sizeof(T_PHB_ECC_RECORD)) );
+
+    for (rec_ctr=0; rec_ctr < pbs_data.max_record[ECC]; rec_ctr++)
+    {
+      if(*data_ptr NEQ 0xff)
+      {
+        phb_ecc_element[rec_ctr].phy_idx = rec_ctr + 1;
+        memcpy(phb_ecc_element[rec_ctr].number, data_ptr, ECC_NUM_LEN);
+        data_ptr += ECC_NUM_LEN;
+        (pbs_data.used_record[ECC])++;
+      }
+    }
+  }
+}
+
+/*
++----------------------------------------------------------------------+
+| PROJECT :                              MODULE  : PHB                 |
+| STATE   : code                         ROUTINE : pb_sim_prepare_ext_data |
++----------------------------------------------------------------------+
+
+
+    PURPOSE :   Prepare the data for the extention record.
+                If NULL pointer is given for number and subaddress
+                then the extention record will marked as unused
+
+*/
+/* Extesion records will be stored as per 11.11.
+          Bytes         Description      M/O    Length
+          --------------------------------------
+              1        Record type         M    1 byte
+           2 to 12  Extension data  M   11 bytes
+             13        Identifier                 M 1 byte
+          --------------------------------------*/
+
+LOCAL void pb_sim_prepare_ext_data(UBYTE *ext_data, int ext_count, UBYTE *number, UBYTE no_len, UBYTE *subaddr)
+{
+  UBYTE *data_num = NULL;
+  UBYTE *data_subadd = NULL;
+
+  TRACE_FUNCTION("pb_sim_prepare_ext_data()");
+
+  if(number[10] NEQ 0xFF)
+    data_num = number + ((ext_count + 1) * 10);
+  
+  data_subadd = subaddr + (ext_count * 11);
+  
+  memset(ext_data, 0xFF, sizeof(ext_data));
+
+  if ((data_num NEQ NULL) AND (*data_num NEQ 0xFF))
+  {
+    /* Set record type to 2 which corresponds to Additional data.  Record type as per 11.11 */
+    ext_data[0] = 2;
+    ext_data[1] = no_len - ((ext_count + 1) * 10);
+    memcpy (ext_data + 2, data_num, 10);
+  }
+  else if ((data_subadd NEQ NULL) AND (*data_subadd NEQ 0xFF))
+  {
+    TRACE_EVENT ("SUBADDRESS EXTENTION");
+	 /* Set record type to 1 which corresponds to Called Party Subaddress. Record type as per 11.11 */
+    ext_data[0] = 1;
+    memcpy (ext_data + 1, data_subadd, 11);
+  }
+}
+
+/*
++----------------------------------------------------------------------+
+| PROJECT :                              MODULE  : PHB                 |
+| STATE   : code                         ROUTINE : pb_sim_get_field_id |
++----------------------------------------------------------------------+
+
+
+    PURPOSE :   Returns field ID for the corresponding Phonebook type.
+
+*/
+LOCAL USHORT pb_sim_get_field_id (T_PHB_TYPE type)
+{
+  USHORT field_id;
+
+  TRACE_FUNCTION("pb_sim_get_field_id()");
+
+  /* Get Elementary file ID for the Phonebook type. */
+  switch(type)
+  {
+    case ADN:
+      field_id = SIM_ADN;
+      break;
+
+    case LDN:
+      field_id = SIM_OCI;
+      break;
+
+    case LRN:
+      field_id = FFS_LRN;
+      break;
+
+    case LMN:
+      field_id = FFS_LMN;
+      break;
+
+    case UPN:
+      field_id = SIM_MSISDN;
+      break;
+
+    case FDN:
+      field_id = SIM_FDN;
+      break;
+
+    case SDN:
+      field_id = SIM_SDN;
+      break;
+
+    case BDN:
+      field_id = SIM_BDN;
+      break;
+
+    default:
+      TRACE_ERROR ("No such field");
+      TRACE_EVENT_P1 ("No such field for type = %04X", type); 
+      field_id = 0;
+      break;
+  }
+
+  return field_id;
+}
+
+/*
++----------------------------------------------------------------------+
+| PROJECT :                              MODULE  : PHB                 |
+| STATE   : code                         ROUTINE : pb_sim_get_ext_file |
++----------------------------------------------------------------------+
+
+
+    PURPOSE :   Returns field ID for the corresponding Phonebook type.
+
+*/
+LOCAL USHORT pb_sim_get_ext_file (T_PHB_TYPE type)
+{
+  USHORT ext_file_id;
+
+  TRACE_FUNCTION("pb_sim_get_ext_file()");
+
+  /* Get Extension Elementary file ID for the Phonebook type. */
+  switch(type)
+  {
+    case ADN:
+    case UPN:
+      ext_file_id = SIM_EXT1;
+      break;
+
+    case FDN:
+      ext_file_id = SIM_EXT2;
+      break;
+
+    case SDN:
+      ext_file_id = SIM_EXT3;
+      break;
+
+    case BDN:
+      ext_file_id = SIM_EXT4;
+      break;
+
+    case LDN:
+      ext_file_id = SIM_EXT5;
+      break;
+
+    case LRN:
+      ext_file_id = FFS_EXT_LRN;
+      break;
+
+    case LMN:
+      ext_file_id = FFS_EXT_LMN;
+      break;
+
+    default:
+      ext_file_id = 0; /* No extension records available */
+      break;
+  }
+
+  return ext_file_id;
+}
+
+/*
++---------------------------------------------------------------------+
+| PROJECT : MMI-Framework (8417)       MODULE  : PHB                  |
+| STATE   : code                       ROUTINE : pb_sim_cvt_alpha_for_cmp |
++---------------------------------------------------------------------+
+
+  PURPOSE : convert alpha to lower case when not unicode
+
+*/
+LOCAL void pb_sim_cvt_alpha_for_cmp ( UBYTE *src,
+                                      UBYTE *dst,
+                                      UBYTE len )
+{
+  int i;
+
+  TRACE_FUNCTION("pb_sim_cvt_alpha_for_cmp()");
+
+  if (*src NEQ 0x80) /* 11.11 Annex B 0x80 is the UCS2 indicator */
+  {
+    for ( i = 0; i < len; i++ )
+      dst[i] = (UBYTE)tolower((int)src[i]);
+  }
+  else
+  {
+    memcpy (dst, src, len);
+  }
+}
+
+/*
++----------------------------------------------------------------------+
+| PROJECT :                              MODULE  : PHB                 |
+| STATE   : code                         ROUTINE : pb_sim_get_ext_file_id                           |
++----------------------------------------------------------------------+
+
+
+    PURPOSE :   Returns Extension field ID for the corresponding field ID of a Phonebook.
+
+*/
+LOCAL USHORT pb_sim_get_ext_file_id (USHORT field_id)
+{
+  USHORT ext_file_id;
+
+  TRACE_FUNCTION("pb_sim_get_ext_file_id()");
+
+  /* Get Extension Elementary file ID for the Phonebook type. */
+  switch(field_id)
+  {
+    case SIM_ADN:
+    case SIM_LND:
+    case SIM_MSISDN:
+      ext_file_id = SIM_EXT1;
+      break;
+
+    case SIM_FDN:
+      ext_file_id = SIM_EXT2;
+      break;
+
+    case SIM_SDN:
+      ext_file_id = SIM_EXT3;
+      break;
+
+    case SIM_BDN:
+      ext_file_id = SIM_EXT4;
+      break;
+
+    case SIM_OCI:
+      ext_file_id = SIM_EXT5;
+      break;
+
+    case FFS_LRN:
+      ext_file_id = FFS_EXT_LRN;
+      break;
+
+    case FFS_LMN:
+      ext_file_id = FFS_EXT_LMN;
+      break;
+
+    default:
+      ext_file_id = 0; 
+      break;
+  }
+
+  return ext_file_id;
+}
+
+/*
++----------------------------------------------------------------------+
+| PROJECT :                              MODULE  : PHB                 |
+| STATE   : code                         ROUTINE : pb_sim_get_ext_type |
++----------------------------------------------------------------------+
+
+
+    PURPOSE :   Returns Extension Type ID for the corresponding 
+                field ID of a Phonebook.
+
+*/
+LOCAL T_EXT_TYPE pb_sim_get_ext_type (USHORT field_id)
+{
+  T_EXT_TYPE ext_type;
+  
+  /* Get Extension Type for the Phonebook field ID. */
+  switch(field_id)
+  {
+    case SIM_ADN:
+    case SIM_LND:
+    case SIM_MSISDN:
+      ext_type = EXT1;
+      break;
+
+    case SIM_FDN:
+      ext_type = EXT2;
+      break;
+
+    case SIM_SDN:
+      ext_type = EXT3;
+      break;
+
+    case SIM_BDN:
+      ext_type = EXT4;
+      break;
+
+    case SIM_OCI:
+      ext_type = EXT5;
+      break;
+
+    case FFS_LRN:
+      ext_type = EXT_LRN;
+      break;
+
+    case FFS_LMN:
+      ext_type = EXT_LMN;
+      break;
+
+    default:
+      ext_type = INVALID_EXT; 
+      break;
+  }
+
+  return ext_type;
+}
+
+/*
++----------------------------------------------------------------------+
+| PROJECT :                              MODULE  : PHB                 |
+| STATE   : code                         ROUTINE : pb_sim_find_free_record                        |
++----------------------------------------------------------------------+
+
+
+    PURPOSE :   Returns free record number for the Phonebook type.
+
+*/
+GLOBAL int  pb_sim_find_free_record   (T_PHB_TYPE type)
+{
+  int db_result;
+  unsigned i;
+  USHORT field_id;
+
+  TRACE_FUNCTION("pb_sim_find_free_record()");
+
+  switch (type)
+  {
+    case ECC: /* ECC not stored in DB, special handling */
+      for (i = 0; i < pbs_data.max_record[ECC]; i++)
+      {
+        if (phb_ecc_element[i].phy_idx EQ 0)
+          return i + 1;
+      }
+      return 0; /* No free record found */
+
+    case LDN:
+    case LRN:
+    case LMN:
+      return 1; /* For cyclic entries always the first */
+
+    default:
+      /* Get Elementary file ID for the Phonebook type. */
+      field_id = pb_sim_get_field_id(type);
+
+      db_result = db_find_free_record(pbs_data.db_handle, field_id);
+      
+			if (db_result <= 0)
+        return 0; /* No free record */
+      
+			return db_result;
+  }
+}
+
+GLOBAL int  pb_sim_find_free_ext_record   ()
+{
+  int db_result;
+  unsigned i;
+  USHORT field_id;
+
+  TRACE_FUNCTION("pb_find_ext_free_record()");
+
+      field_id =SIM_EXT1;
+
+      db_result = db_find_free_record(pbs_data.db_handle, field_id);
+      
+      if (db_result <= 0)
+         return 0; /* No free record */
+      
+      return db_result;
+}
+
+
+
+
+
+/*
++-------------------------------------------------------------------------------+
+| PROJECT : MMI-Framework (8417)        MODULE  : PHB                           |
+| STATE   : code                        ROUTINE : pb_sim_get_size_except_tag    |
++-------------------------------------------------------------------------------+
+
+  PURPOSE : Returns size of data excluding length of tag (alpha identifier)
+*/
+USHORT pb_sim_get_size_except_tag (USHORT field_id)
+{
+
+  TRACE_FUNCTION("pb_sim_get_size_except_tag()");
+  switch(field_id)
+  {
+    case SIM_ADN:
+    case SIM_FDN:
+    case SIM_BDN:
+    case SIM_MSISDN:
+    case SIM_SDN:
+      return 14; /* 11.11 */
+
+    case FFS_LRN:
+    case FFS_LMN:
+    case SIM_OCI:
+      return 27; /* Using EF_OCI, 31.102 4.2.34 */
+
+    //case SIM_ICI:
+    //  return 28; /* Using EF_ICI, 31.102 4.2.33 */
+
+    default:
+      TRACE_ERROR("Invalid field ID passed !");
+      return 0;
+  }
+}
+/*
++--------------------------------------------------------------------+
+| PROJECT : MMI-Framework (8417)        MODULE  : PHB                |
+| STATE   : code                        ROUTINE : pb_sim_cmpString          |
++--------------------------------------------------------------------+
+
+  PURPOSE : compare two strings.
+            if s1 < s2 return value < 0
+            if s1 = s2 return value = 0
+            if s1 > s2 return value > 0
+*/
+
+static int pb_sim_cmpString ( UBYTE *s1, UBYTE *s2, UBYTE len )
+{
+  int n = 0;
+
+  /* TRACE_FUNCTION("pb_sim_cmpString()"); */ /* Called too often to trace */
+
+  if ((*s1 EQ 0x80) AND
+      (*s2 NEQ 0x80)    )
+  {
+    s1++;
+    len--;
+    return pb_sim_cmp2Bytes(s1, s2, len, 1);
+  }
+  else if ((*s1 NEQ 0x80) AND
+           (*s2 EQ 0x80)     )
+  {
+    s2++;
+    len--;
+    return pb_sim_cmp2Bytes(s1, s2, len, 2);
+  }
+  else
+  {
+    while (*s1 EQ *s2)
+    {
+      if (*s1 EQ 0xff)
+        return 0;
+      s1++;
+      s2++;
+      n++;
+      if (n EQ len)
+        return 0;
+    }
+
+    if ((*s1 > *s2) AND (*s1 NEQ 0xff))
+      return 1;
+
+    return -1;
+  }
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : MMI-Framework (8417)        MODULE  : PHB                |
+| STATE   : code                        ROUTINE : pb_sim_cmp2Bytes       |
++--------------------------------------------------------------------+
+
+  PURPOSE : compare two strings.
+            if s1 < s2 return value < 0
+            if s1 = s2 return value = 0
+            if s1 > s2 return value > 0
+
+            flag = 1, s1 is unicode
+            flag = 2, s2 is unicode
+*/
+
+LOCAL int pb_sim_cmp2Bytes(UBYTE *s1, UBYTE *s2, UBYTE len, UBYTE flag)
+{
+  int n = 0;
+
+  /*  TRACE_FUNCTION("pb_sim_cmp2Bytes()"); */
+
+  if (flag EQ 1)
+  {
+    while ( (*s1 EQ 0x00 OR *s1 EQ 0xFF ) AND ( *(s1+1) EQ *s2) )
+    {
+      if (*s1 EQ 0xff AND *(s1+1) EQ 0xFF)
+        return 0;
+      
+			s1 += 2;
+      s2++;
+      n += 2;
+
+      if (n >= len)
+        return 0;
+    }
+
+    if ( ( *s1 > 0 AND *s1 NEQ 0xff ) OR
+         ( !*s1 AND ( *(s1+1) > *s2 ) ) )
+      return 1;
+
+    return -1;
+  }
+
+  if (flag EQ 2)
+  {
+    while ((*s2 EQ 0x00 OR *s2 EQ 0xFF) AND (*s1 EQ *(s2+1)))
+    {
+      if (*s2 EQ 0xff AND *(s2+1) EQ 0xFF)
+        return 0;
+      
+			s1++;
+      s2 += 2;
+      n += 2;
+
+      if (n >= len)
+        return 0;
+    }
+
+    if ((*s2 > 0 AND *s2 NEQ 0xff) OR
+        (!*s2 AND (*(s2+1) > *s1))    )
+      return -1;
+
+    return 1;
+  }
+  return -1;
+}
+
+
+/*
++----------------------------------------------------------------------------+
+| PROJECT : MMI-Framework (8417)        MODULE  : PHB                        |
+| STATE   : code                        ROUTINE : pb_sim_update_extn_records |
++----------------------------------------------------------------------------+
+
+  PURPOSE : Update reference count for extension record and delete if record 
+            is not referenced at all.
+            ETSI 11.11 clause 11.5.1 states that reference counts not only 
+            have to be maintained for EXT1 but also for other EXT records.
+*/
+LOCAL T_PHB_RETURN pb_sim_update_extn_records(USHORT ext_field_id, 
+                                              USHORT rec_num, 
+                                              SHORT ref_type)
+{
+  T_EXT_TYPE ext_type;
+  UBYTE *refptr;
+  int db_result;
+  UBYTE dummy_ref;
+
+  switch (ext_field_id)
+  {
+    case SIM_EXT1:    ext_type = EXT1;        break;
+    case SIM_EXT2:    ext_type = EXT2;        break;
+    case SIM_EXT3:    ext_type = EXT3;        break;
+    case SIM_EXT4:    ext_type = EXT4;        break;
+    case SIM_EXT5:    ext_type = EXT5;        break;
+    case FFS_EXT_LRN: ext_type = EXT_LRN;     break;
+    case FFS_EXT_LMN: ext_type = EXT_LMN;     break;
+    default:
+      return PHB_FAIL;
+  }
+
+  if (rec_num <= MAX_EXT_RECORDS)
+  {
+    refptr = &ext_ref_count[ext_type][rec_num - 1];
+  }
+  else
+  {
+    /* Could become improved by using dynamic memory but better as using an
+     * illegal array index. Only ref counters will have a problem here. */
+    TRACE_ERROR ("SIM exceeds MAX_EXT_RECORDS"); 
+    dummy_ref = 1;
+    refptr = &dummy_ref;
+  }
+
+  if (ref_type EQ -1)
+  {
+    /* Decrement usage counter */
+    if (*refptr > 0)
+    {
+      *refptr += ref_type;
+    }
+    else
+    {
+      TRACE_ERROR ("EXT usage counter below zero");
+    }
+
+    if (*refptr EQ 0)
+    {
+      db_result = db_delete_record(pbs_data.db_handle, ext_field_id, rec_num);
+      if(db_result < DB_OK)
+        return PHB_FAIL;
+    }
+  }
+  else if (ref_type EQ 1)
+  {
+    *refptr += ref_type;
+  }
+  else
+    return PHB_FAIL; /* add_val not in -1, +1 */
+
+  TRACE_EVENT_P3 ("Usage count of EXT %04X, record %d = %d", 
+                  ext_field_id, rec_num, *refptr);
+
+  return PHB_OK;
+}
+
+/*
++----------------------------------------------------------------------------+
+| PROJECT : MMI-Framework (8417)        MODULE  : PHB                        |
+| STATE   : code                        ROUTINE : pb_sim_del_ext_records     |
++----------------------------------------------------------------------------+
+
+  PURPOSE : Read extension records and update the records.
+  
+*/
+#define RETURN(x) { retVal = x; goto cleanup_exit; }
+/*lint -e{801} Use of goto*/
+LOCAL T_PHB_RETURN pb_sim_del_ext_records (T_PHB_TYPE type, 
+                                           USHORT field_id, 
+                                           USHORT db_recno)
+{
+  int db_result;
+  T_PHB_RETURN retVal;
+  USHORT ext_file_id;
+  T_DB_INFO_FIELD info_field;
+  UBYTE *data;
+  UBYTE ext_rcd_num;
+
+  TRACE_FUNCTION("pb_sim_del_ext_records()");
+
+  ACI_MALLOC (data, SIM_MAX_RECORD_SIZE);
+
+  /* Get Extension file for the Phonebook type. */
+  ext_file_id = pb_sim_get_ext_file(type);
+
+  /* Read record from the database. */
+  if(db_info_field(pbs_data.db_handle, field_id, &info_field) NEQ DB_OK)
+    RETURN (PHB_FAIL)
+
+  db_result = db_read_record(pbs_data.db_handle, field_id, db_recno, 0, info_field.record_size, data);
+  if (db_result EQ DB_EMPTY_RECORD)
+    RETURN (PHB_OK) /* Nothing to do */
+
+  if (db_result < DB_OK)
+    RETURN (PHB_FAIL) /* Some other problem */
+
+  /* Get extension record identifier */
+  ext_rcd_num = data[info_field.record_size - 1];
+
+  while (ext_rcd_num NEQ 0xFF) /* check for extension records */
+  {
+    if(db_info_field (pbs_data.db_handle, ext_file_id, &info_field) NEQ DB_OK)
+      RETURN (PHB_FAIL)
+
+    if ((ext_rcd_num EQ 0) OR (ext_rcd_num > info_field.num_records))
+    {
+      TRACE_EVENT_P1 ("illegal ext record number %d ignored", ext_rcd_num);
+      RETURN (PHB_OK)
+    }
+
+    if(db_read_record(pbs_data.db_handle, ext_file_id, ext_rcd_num, 0, info_field.record_size, data) < DB_OK) 
+      RETURN (PHB_FAIL)
+
+    if (pb_sim_update_extn_records(ext_file_id, ext_rcd_num, -1) EQ PHB_FAIL)
+      RETURN (PHB_FAIL)
+    
+    ext_rcd_num = data[12];
+  }
+  RETURN (PHB_OK)
+
+cleanup_exit:
+  ACI_MFREE (data);
+  return retVal;
+}
+#undef RETURN
+
+/*
++----------------------------------------------------------------------------+
+| PROJECT : MMI-Framework (8417)        MODULE  : PHB                        |
+| STATE   : code                        ROUTINE : pb_sim_ext_records_used     |
++----------------------------------------------------------------------------+
+
+  PURPOSE : check if extension records is used by one phone book record (ADN & FDN).
+  
+*/
+#define RETURN(x) { retVal = x; goto cleanup_exit; }
+/*lint -e{801} Use of goto*/
+LOCAL BOOL pb_sim_ext_records_used (T_PHB_TYPE type, 
+                                    USHORT field_id, 
+                                    USHORT db_recno)
+{
+  int db_result;
+  BOOL retVal;
+  USHORT ext_file_id;
+  T_DB_INFO_FIELD info_field;
+  UBYTE *data;
+  UBYTE ext_rcd_num;
+
+  TRACE_FUNCTION("pb_sim_ext_records_used()");
+
+  ACI_MALLOC (data, SIM_MAX_RECORD_SIZE);
+
+  /* Get Extension file for the Phonebook type. */
+  ext_file_id = pb_sim_get_ext_file(type);
+
+  /* Read record from the database. */
+  if(db_info_field(pbs_data.db_handle, field_id, &info_field) NEQ DB_OK)
+    RETURN (FALSE)
+
+  db_result = db_read_record(pbs_data.db_handle, field_id, db_recno, 0, info_field.record_size, data);
+  if (db_result < DB_OK)
+    RETURN (FALSE) /* record can not be read - so no EXTn used  */
+
+  /* Get extension record identifier */
+  ext_rcd_num = data[info_field.record_size - 1];
+
+  TRACE_EVENT_P1("pb_sim_ext_records_used() --1-- ext_rcd_num = %d", ext_rcd_num);
+
+  if (ext_rcd_num NEQ 0xFF) /* check for extension records */
+  {
+    if(db_info_field (pbs_data.db_handle, ext_file_id, &info_field) NEQ DB_OK)
+      RETURN (FALSE)
+
+    if ((ext_rcd_num EQ 0) OR (ext_rcd_num > info_field.num_records))
+    {
+      TRACE_EVENT_P1 ("illegal ext record number %d ignored", ext_rcd_num);
+      RETURN (FALSE)
+    }
+
+    if(db_read_record(pbs_data.db_handle, ext_file_id, ext_rcd_num, 0, info_field.record_size, data) < DB_OK) 
+      RETURN (FALSE)
+    
+    ext_rcd_num = data[12];
+  TRACE_EVENT_P1("pb_sim_ext_records_used() --2-- ext_rcd_num = %d", ext_rcd_num);
+
+	 RETURN (TRUE) //return TRUE here since ext_rcd_num is valid
+  }
+  else
+  {
+  	RETURN (FALSE)
+  }
+cleanup_exit:
+  ACI_MFREE (data);
+  return retVal;
+}
+#undef RETURN
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : MMI-Framework (8417)        MODULE  : PHB                                       |
+| STATE   : code                       ROUTINE : pb_sim_retrieve_rdm_recno       |
++--------------------------------------------------------------------+
+
+  PURPOSE : Retrieving record number for LDN, LMN and LRN Phonebook.
+  
+*/
+
+LOCAL USHORT pb_sim_retrieve_rdm_recno (T_PHB_TYPE type)
+{
+  USHORT rec_no;
+
+  TRACE_FUNCTION("pb_sim_retrieve_rdm_recno()");
+
+  switch(type)
+  {
+    case LDN:
+      rec_no = LDN_DATA_RECNO;
+      break;
+
+    case LMN:
+      rec_no = LMN_DATA_RECNO;
+      break;
+  
+    case LRN:
+      rec_no = LRN_DATA_RECNO;
+      break;
+
+    default:
+      rec_no = 0;
+      break;
+  }
+
+  return rec_no;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : MMI-Framework (8417)        MODULE  : PHB                |
+| STATE   : code                        ROUTINE : pb_sim_cmpWild     |
++--------------------------------------------------------------------+
+
+  PURPOSE : compare two strings considering wildcard.
+            if s1 < s2 return value < 0
+            if s1 = s2 return value = 0
+            if s1 > s2 return value > 0
+*/
+
+GLOBAL int pb_sim_cmpWild ( const char *s1, const char *s2, size_t cmp_len )
+{
+  TRACE_FUNCTION("pb_sim_cmpWild()");
+
+  if (cmp_len EQ 0)
+    return 0; /* Nothing to compare */
+
+  while ((*s1 EQ *s2) OR (*s1 EQ PHB_WILD_CRD) OR (*s2 EQ PHB_WILD_CRD))
+  {
+    /* Character matches */
+    cmp_len--;
+
+    if (cmp_len EQ 0)
+      return 0;
+
+    if ((*s1 EQ '\0') AND (*s2 EQ '\0'))
+      return 0;
+
+    if (*s2 EQ '\0') /* Means *s1 > '\0' */
+      return 1;
+
+    if (*s1 EQ '\0') /* Means *s2 > '\0' */
+      return -1;
+
+    s1++;
+    s2++;
+  }
+  /* Character does not match */
+  if (*s1 > *s2)
+    return 1;
+  else
+    return -1;
+
+}
+
+
+/* Implements Measure #30 */
+/*
++------------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417) MODULE: PHB                              |
+| STATE  : code                 ROUTINE: pb_sim_read_alpha_num_record    |
++------------------------------------------------------------------------+
+
+    PURPOSE : Read a given physical record from the flash based phonebook
+              in alphabetical order Or in number sorted order
+*/
+
+GLOBAL T_PHB_RETURN pb_sim_read_alpha_num_record (T_PHB_TYPE    type, 
+                                                  USHORT        order_num,
+                                                  T_PHB_RECORD *entry,
+                                                  UBYTE         sort_index)
+{
+  int db_result;
+  USHORT field_id;
+  
+  TRACE_FUNCTION("pb_sim_read_alpha_num_record()");
+
+  /* Get Elementary file ID for the Phonebook type. */
+  field_id = pb_sim_get_field_id(type);
+
+  /* Read record from the FFS. */
+  db_result = db_get_phy_from_idx(pbs_data.db_handle, field_id, 
+                                  sort_index, order_num);
+
+  if(db_result > DB_OK)
+  {
+    return pb_sim_read_record(type, (USHORT)db_result, entry) ;
+  }
+
+  /* Check whether index is vaild. */
+  if(db_result EQ DB_INVALID_INDEX)
+  {
+    return PHB_INVALID_IDX;
+  }
+
+  /* 
+   * Unable to get record from the database. 
+   * Hence returning PHB_FAIL.
+   */
+  return PHB_FAIL;
+}
+
+/* Implements Measure #167 */
+/*
++------------------------------------------------------------------------+
+| PROJECT: MMI-Framework (8417) MODULE: PHB                              |
+| STATE  : code                 ROUTINE: pb_sim_read_alpha_num_record    |
++------------------------------------------------------------------------+
+
+    PURPOSE : Function check for extention records
+*/
+LOCAL T_PHB_RETURN pb_sim_update_index (T_PHB_TYPE    type, 
+                                        T_DB_CHANGED *rec_affected,
+                                        USHORT        field_id,
+                                        UBYTE         ext_rec_cnt)
+{
+  TRACE_FUNCTION("pb_sim_update_index()");
+
+  if((type NEQ LDN) AND (type NEQ LMN) AND (type NEQ LRN))
+  {
+    /* Update the sorted indexes. */
+    if(db_update_index(pbs_data.db_handle, field_id, NAME_IDX, 
+                       pb_sim_alpha_cmp, PHB_MATCH_PARTIAL) NEQ DB_OK)
+    {
+      return PHB_FAIL;
+    }
+
+    if(db_update_index(pbs_data.db_handle, field_id, NUMBER_IDX, 
+                       pb_sim_alpha_cmp, PHB_MATCH_PARTIAL) NEQ DB_OK)
+    {
+      return PHB_FAIL;
+    }
+  }
+  return PHB_OK;
+}
+
+void      pb_update_cphs_mb_ext_record(void)
+{
+   UBYTE i; 
+   USHORT ext_file_id; 
+
+/* Get Extension file for the Phonebook type. */
+  ext_file_id = SIM_EXT1; 
+
+  for(i =0; i< 4; i++)
+  {
+     if(cphs_mb_ext_record_num[i] NEQ 0xff)
+     {
+        db_update_ext_bitmap(pbs_data.db_handle,ext_file_id,cphs_mb_ext_record_num[i],TRUE); 
+        pb_sim_update_extn_records(ext_file_id, cphs_mb_ext_record_num[i], 1); 
+     }
+  }
+}
+GLOBAL void  pb_sim_update_ext_bitmap(UBYTE rec_num, BOOL flag)
+ {
+  USHORT ext_file_id; 
+
+/* Get Extension file for the Phonebook type. */
+  ext_file_id = SIM_EXT1;
+
+    db_update_ext_bitmap(pbs_data.db_handle,ext_file_id,rec_num, flag); 
+    if(flag EQ TRUE)
+     {
+        pb_sim_update_extn_records(ext_file_id, rec_num, 1);
+     }
+    else
+     {
+       pb_sim_update_extn_records(ext_file_id, rec_num,  -1);
+     }
+ }
+
+
+#endif /* #ifdef TI_PS_FFS_PHB */
+
+