diff src/aci2/aci/cmh_ccf.c @ 3:93999a60b835

src/aci2, src/condat2: import of g23m/condat source pieces from TCS211
author Mychaela Falconia <falcon@freecalypso.org>
date Mon, 26 Sep 2016 00:29:36 +0000
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/aci2/aci/cmh_ccf.c	Mon Sep 26 00:29:36 2016 +0000
@@ -0,0 +1,2915 @@
+/*
++-----------------------------------------------------------------------------
+|  Project :  GSM-PS (6147)
+|  Modul   :  CMH_CCF
++-----------------------------------------------------------------------------
+|  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 module defines the functions used by the command
+|             handler for call control.
++-----------------------------------------------------------------------------
+*/
+
+#ifndef CMH_CCF_C
+#define CMH_CCF_C
+#endif
+
+#include "aci_all.h"
+
+/*==== INCLUDES ===================================================*/
+
+#include "aci_cmh.h"
+#include "ati_cmd.h"
+#include "aci_cmd.h"
+#include "aci_mem.h"
+
+#ifdef FAX_AND_DATA
+#include "aci_fd.h"
+#endif
+
+#ifdef FF_TTY
+#include "audio.h"
+#endif
+
+#include "aci_io.h"
+
+#include "phb.h"
+#include "ksd.h"
+#include "aoc.h"
+#include "aci.h"
+#include "psa.h"
+#include "psa_cc.h"
+#include "psa_ss.h"
+#include "psa_util.h"
+#include "cmh.h"
+#include "cmh_cc.h"
+#include "cmh_ss.h"
+#include "cmh_phb.h"
+
+#include "aci_lst.h"
+
+#ifdef DTI
+#include "dti.h"      /* functionality of the dti library */
+#include "dti_conn_mng.h"
+#include "dti_cntrl_mng.h"
+#endif
+
+#ifdef UART
+#include "psa_uart.h"
+#include "cmh_uart.h"
+#endif
+
+#ifdef FAX_AND_DATA
+
+#include "psa_ra.h"
+#include "cmh_ra.h"
+#include "psa_l2r.h"
+#include "cmh_l2r.h"
+
+#ifdef FF_FAX
+#include "psa_t30.h"
+#include "cmh_t30.h"
+#endif /* FF_FAX */
+
+#endif /* FAX_AND_DATA */
+
+#if defined (FF_WAP) || defined (FF_TCP_IP) || defined(FF_GPF_TCPIP) || defined (FF_SAT_E)
+#include "wap_aci.h"
+#include "psa_ppp_w.h"
+#include "cmh_ppp.h"
+#endif /* of FF_WAP or FF_TCP_IP  || (FF_GPF_TCPIP)|| (FF_SAT_E)*/
+
+#ifdef FF_GPF_TCPIP
+#include "dcm_utils.h"
+#include "dcm_state.h"
+#include "dcm_env.h"
+#endif
+
+#ifdef SIM_TOOLKIT
+#include "psa_sat.h"
+#include "cmh_sat.h"
+#include "psa_sim.h"
+#endif
+
+#ifdef FF_PSI
+#include "psa_psi.h"
+#include "cmh_psi.h"
+#endif /*FF_PSI*/
+#include "l4_tim.h"
+
+#if defined(_TARGET_)
+#include "ffs/ffs.h"
+#include "ffs_coat.h"
+#endif /* _TARGET_ */
+
+#include "dcm_f.h"
+
+/*==== CONSTANTS ==================================================*/
+#define   ACI_MAX_DIAL_SHORT_STRING_LEN     2
+EXTERN T_ACI_CC_REDIAL_BLACKL * cc_blacklist_ptr;
+/*==== TYPES ======================================================*/
+
+/*==== EXPORT =====================================================*/
+
+/*==== VARIABLES ==================================================*/
+GLOBAL T_ACI_CUSCFG_PARAMS cuscfgParams;
+
+/*==== FUNCTIONS ==================================================*/
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH                     |
+|                                 ROUTINE : cmhCC_PrepareCmdEnd     |
++-------------------------------------------------------------------+
+
+  PURPOSE : Prepares the end of a CC related AT command.
+            cId describes a valid (non-NULL) call table entry.
+
+*/
+
+GLOBAL void cmhCC_PrepareCmdEnd (SHORT cId, UBYTE *cmdBuf, UBYTE *srcBuf)
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+
+  if (cmdBuf NEQ NULL)
+    *cmdBuf = ctb->curCmd;
+  if (srcBuf NEQ NULL)
+    *srcBuf = ctb->curSrc;
+  ctb->curCmd = AT_CMD_NONE;
+  ctb->curSrc = CMD_SRC_NONE;
+}
+
+
+GLOBAL UBYTE cmhCC_set_speech_serv (T_CC_CMD_PRM  *pCCCmdPrm)
+{
+#ifdef FF_TTY
+  UBYTE ctmReq;
+
+  /* check temporary inversion of CTM service request */
+  if (!ccShrdPrm.ctmOvwr)
+  {
+    ctmReq = ccShrdPrm.ctmReq;
+  }
+  else
+  {
+    if (ccShrdPrm.ctmReq EQ CTM_ENABLED)
+    {
+      ctmReq = CTM_DISABLED;
+      ccShrdPrm.ttyCmd = (UBYTE)TTY_OFF;
+    }
+    else
+    {
+      ctmReq = CTM_ENABLED;
+      ccShrdPrm.ttyCmd = (UBYTE)TTY_ALL;
+    }
+    /* reset temporary inversion */
+    ccShrdPrm.ctmOvwr = FALSE;
+  }
+#endif /* FF_TTY */
+#ifdef FF_TTY
+  if (ctmReq EQ CTM_ENABLED)
+  {
+    if (ccShrdPrm.ctmState EQ TTY_STATE_NONE)
+    {
+      ccShrdPrm.ctmState = TTY_STATE_IDLE;
+    }
+    if (pCCCmdPrm->ALSmode EQ ALS_MOD_AUX_SPEECH)
+    {
+      return BEARER_SERV_AUX_SPEECH_CTM;
+    }
+    else
+    {
+      return BEARER_SERV_SPEECH_CTM;
+    }
+  }
+  else
+#endif /* FF_TTY */
+  {
+#ifdef FF_TTY
+    if (ccShrdPrm.ctmState EQ TTY_STATE_IDLE)
+    {
+      ccShrdPrm.ctmState = TTY_STATE_NONE;
+    }
+#endif /* FF_TTY */
+    if (pCCCmdPrm->ALSmode EQ ALS_MOD_AUX_SPEECH)
+    {
+      return BEARER_SERV_AUX_SPEECH;
+    }
+  }
+  return BEARER_SERV_SPEECH;
+}
+
+GLOBAL T_ACI_RETURN cmhCC_chkShortString (T_ACI_CMD_SRC srcId,
+                                          SHORT         cId,
+                                          T_CLPTY_PRM   *cldPty)
+{
+  USHORT              num_len;
+  T_ACI_KSD_USSD_PRM  ksdString;
+
+  num_len = strlen (cldPty->num);
+
+  /* check the length */
+  if (!num_len OR (num_len > ACI_MAX_DIAL_SHORT_STRING_LEN))
+  {
+    return (AT_EXCT);
+  }
+
+  ksdString.ussd = (UBYTE*)cldPty->num;
+
+  /* check if MS is in a call */
+  if (cId NEQ NO_ENTRY)
+  {
+    /* call is active */
+    return (cmhSS_ksdUSSD (srcId, &ksdString));
+  }
+
+  /* check if input string is 2 digit starting with a 1 */
+  if ((num_len EQ ACI_MAX_DIAL_SHORT_STRING_LEN) AND
+      (*(cldPty->num) EQ '1') OR cuscfgParams.Two_digit_MO_Call)
+  {
+    /* normal dial */
+    return (AT_EXCT);
+  }
+
+  return (cmhSS_ksdUSSD (srcId, &ksdString));
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH                     |
+|                                 ROUTINE : cmhCC_find_call_for_DTMF|
++-------------------------------------------------------------------+
+
+  PURPOSE :searches for a call where it is possible to
+             send DTMF tones.
+*/
+GLOBAL BOOL is_call_ok_for_dtmf( SHORT cId )
+{
+  T_CC_CALL_TYPE calltype = cmhCC_getcalltype(cId);
+
+  if(calltype NEQ VOICE_CALL)
+  {
+    TRACE_EVENT_P1("calltype NEQ VOICE_CALL: %d", calltype);
+    return(FALSE);
+  }
+
+  if (ccShrdPrm.ctb[cId] EQ NULL)
+  {
+    TRACE_EVENT("ccShrdPrm.ctb[cId] EQ NULL");
+    return(FALSE);
+  }
+
+  switch(psaCC_ctb(cId)->calStat)
+  {
+    case(CS_ACT):
+    case(CS_ACT_REQ):
+    case(CS_DSC_REQ):
+    case(CS_CPL_REQ):
+      return(TRUE);
+
+    default:
+      TRACE_EVENT_P1("unexpected call status: %d", psaCC_ctb(cId)->calStat);
+      return(FALSE);
+  }
+}
+
+GLOBAL SHORT cmhCC_find_call_for_DTMF( void )
+{
+  SHORT ctbIdx;        /* holds call table index */
+
+  TRACE_FUNCTION("cmhCC_find_call_for_DTMF");
+
+  for( ctbIdx = 0; ctbIdx < MAX_CALL_NR; ctbIdx++ )
+  {
+    if( is_call_ok_for_dtmf(ctbIdx) )
+    {
+      return(ctbIdx);
+    }
+  }
+  return( NO_ENTRY );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH                     |
+|                                 ROUTINE : cmhCC_SendDTMFdig       |
++-------------------------------------------------------------------+
+
+  PURPOSE : send a DTMF digit for that call.
+    return value: TRUE if digit has been correctly proceeded.
+                  FALSE if an error occured.
+*/
+
+GLOBAL BOOL cmhCC_SendDTMFdig( T_ACI_AT_CMD cmd, SHORT cId, CHAR digit, UBYTE mode)
+{
+  TRACE_FUNCTION("cmhCC_SendDTMFdig");
+
+  TRACE_EVENT_P1("Sending DTMF tone: %d", digit);
+
+  switch( digit )
+  {
+    case( '0' ):
+    case( '1' ):
+    case( '2' ):
+    case( '3' ):
+    case( '4' ):
+    case( '5' ):
+    case( '6' ):
+    case( '7' ):
+    case( '8' ):
+    case( '9' ):
+    case( '*' ):
+    case( '#' ):
+    case( 'A' ):
+    case( 'B' ):
+    case( 'C' ):
+    case( 'D' ):
+      psaCC_SendDTMF( cId, digit, mode );
+      return( TRUE );
+
+    case( 'W' ):   /* FIXME: w should prompt the user for starting sending the following DTMF tones */
+    case( 'w' ):   /* since BMI does not support 'w' we assume to be the same as 'p' */
+    case( 'P' ):
+    case( 'p' ):
+      if(cmd NEQ AT_CMD_VTS)  /* this is only valid within a number to dial */
+      {
+        /* p within a number string: this is a 3 seconds pause */
+        TRACE_EVENT("DTMF pause requested: 3 seconds");
+#if defined (NEW_FRAME)
+        TIMERSTART( TDTMF_VALUE, ACI_TDTMF );
+#else
+        TIMERSTART( TDTMF_VALUE, t_dtmf_handle );
+#endif
+        return( TRUE );
+      }
+      /* fallthrough for else */
+      /*lint -fallthrough*/
+    default:
+      TRACE_EVENT_P1("Invalid DTMF digit: %d", digit);
+  }
+  return( FALSE );
+}
+
+
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_SelRate           |
++-------------------------------------------------------------------+
+
+  PURPOSE : select the user rate out of the current speed setting
+
+*/
+
+GLOBAL UBYTE cmhCC_SelRate ( T_ACI_BS_SPEED speed )
+{
+  switch( speed )
+  {
+  case( BS_SPEED_300_V21      ):
+  case( BS_SPEED_300_V110     ): return( UR_0_3_KBIT );
+  case( BS_SPEED_1200_V22     ):
+  case( BS_SPEED_1200_V120    ):
+  case( BS_SPEED_1200_V110    ): return( UR_1_2_KBIT );
+  case( BS_SPEED_1200_75_V23  ): return( UR_1_2_KBIT_V23 );
+  case( BS_SPEED_2400_V22bis  ):
+  case( BS_SPEED_2400_V26ter  ):
+  case( BS_SPEED_2400_V120    ):
+  case( BS_SPEED_2400_V110    ): return( UR_2_4_KBIT );
+  case( BS_SPEED_4800_V32     ):
+  case( BS_SPEED_4800_V120    ):
+  case( BS_SPEED_4800_V110    ): return( UR_4_8_KBIT );
+  case( BS_SPEED_AUTO         ):
+  case( BS_SPEED_9600_V32     ):
+  case( BS_SPEED_9600_V34     ):
+  case( BS_SPEED_9600_V120    ):
+  case( BS_SPEED_9600_V110    ): return( UR_9_6_KBIT );
+  case( BS_SPEED_14400_V34    ):
+  case( BS_SPEED_14400_V120   ):
+  case( BS_SPEED_14400_V110   ): return( UR_14_4_KBIT );
+  default:                       return( UR_NOT_PRES  );
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_SelMT             |
++-------------------------------------------------------------------+
+
+  PURPOSE : select the modem type out of the current speed setting
+
+*/
+
+GLOBAL UBYTE cmhCC_SelMT ( T_ACI_BS_SPEED speed )
+{
+  switch( speed )
+  {
+  case( BS_SPEED_300_V21      ): return( MT_V21 );
+  case( BS_SPEED_1200_V22     ): return( MT_V22 );
+  case( BS_SPEED_2400_V22bis  ): return( MT_V22_BIS );
+  case( BS_SPEED_1200_75_V23  ): return( MT_V23 );
+  case( BS_SPEED_2400_V26ter  ): return( MT_V26_TER );
+  case( BS_SPEED_9600_V32     ):
+  case( BS_SPEED_4800_V32     ): return( MT_V32 );
+  case( BS_SPEED_1200_V120    ):
+  case( BS_SPEED_2400_V120    ):
+  case( BS_SPEED_4800_V120    ):
+  case( BS_SPEED_9600_V120    ):
+  case( BS_SPEED_14400_V120   ):
+  case( BS_SPEED_2400_V110    ):
+  case( BS_SPEED_4800_V110    ):
+  case( BS_SPEED_9600_V110    ):
+  case( BS_SPEED_14400_V110   ):
+  case( BS_SPEED_300_V110     ):
+  case( BS_SPEED_1200_V110    ): return( MT_NONE );
+  case( BS_SPEED_9600_V34     ):
+  case( BS_SPEED_14400_V34    ): return( MT_V34 );
+  default:                       return( MT_AUTOBAUD  );
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_SelServ           |
++-------------------------------------------------------------------+
+
+  PURPOSE : select the bearer service out of the current name setting
+
+*/
+
+GLOBAL UBYTE cmhCC_SelServ ( T_ACI_CBST_NAM name )
+{
+  switch( name )
+  {
+  case( CBST_NAM_Asynch ): return( BEARER_SERV_ASYNC );
+  case( CBST_NAM_Synch  ): return( BEARER_SERV_SYNC );
+  default:                 return( BEARER_SERV_NOT_PRES );
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_SelCE             |
++-------------------------------------------------------------------+
+
+  PURPOSE : select the connection element out of the current ce setting
+
+*/
+
+GLOBAL UBYTE cmhCC_SelCE ( T_ACI_CBST_CE ce )
+{
+  switch( ce )
+  {
+  case( CBST_CE_Transparent      ): return( CONN_ELEM_TRANS );
+  case( CBST_CE_NonTransparent   ): return( CONN_ELEM_NON_TRANS );
+  case( CBST_CE_BothTransPref    ): return( CONN_ELEM_TRANS_PREF );
+  case( CBST_CE_BothNonTransPref ): return( CONN_ELEM_NON_TRANS_PREF );
+  default:                          return( CONN_ELEM_NOT_PRES );
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_SelStopBit        |
++-------------------------------------------------------------------+
+
+  PURPOSE : select the stop bits out of the current character framing
+            setting
+
+*/
+
+GLOBAL UBYTE cmhCC_SelStopBit ( T_ACI_CMD_SRC srcId )
+{
+#ifdef UART
+  if (UART_IO_SB_2 EQ cmhUART_GetStopBitOverSrcID( (UBYTE) srcId ))
+  {
+    return( STOP_2_BIT );
+  }
+#endif /*UART*/
+#if defined (FF_PSI) AND defined (DTI)
+  if (UART_IO_SB_2 EQ cmhPSI_GetStopBitOverSrcID( (UBYTE) srcId ))
+  {
+    return( STOP_2_BIT );
+  }
+#endif /*FF_PSI*/
+
+  return( STOP_1_BIT );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_SelDataBit        |
++-------------------------------------------------------------------+
+
+  PURPOSE : select the data bits out of the current character framing
+            setting
+
+*/
+
+GLOBAL UBYTE cmhCC_SelDataBit ( T_ACI_CMD_SRC srcId )
+{
+#ifdef UART
+  if ( UART_IO_BPC_7 EQ cmhUART_GetDataBitOverSrcID( (UBYTE) srcId ))
+
+  {
+    return( DATA_7_BIT );
+  }
+#endif /*UART*/
+
+#if defined (FF_PSI) AND defined (DTI)
+  if ( UART_IO_BPC_7 EQ cmhPSI_GetDataBitOverSrcID( (UBYTE) srcId ) )
+  {
+    return( DATA_7_BIT );
+  }
+#endif /*FF_PSI*/
+
+  return( DATA_8_BIT );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_SelParity         |
++-------------------------------------------------------------------+
+
+  PURPOSE : select the data bits out of the current character framing
+            setting
+
+*/
+
+GLOBAL UBYTE cmhCC_SelParity ( T_ACI_CMD_SRC srcId )
+{
+  UBYTE parity=0;
+#if defined (FF_PSI) AND defined (DTI)
+  T_ACI_DTI_PRC_PSI  *src_infos = find_element (psi_src_params,
+                         (UBYTE)srcId, cmhPSItest_srcId);
+
+  if (src_infos EQ NULL)
+#endif /*FF_PSI*/
+
+#ifdef UART
+    parity =  cmhUART_GetParityOverSrcID( (UBYTE) srcId );
+#else /*UART*/
+    parity=0;
+#endif /*UART*/
+
+#if defined (FF_PSI) AND defined (DTI)
+  else
+    parity =  cmhPSI_GetParityOverSrcID( (UBYTE) srcId );
+#endif /*FF_PSI*/
+#ifdef UART
+  switch( parity )
+  {
+    case UART_IO_PA_NONE:                 return( PARITY_NONE );
+    case UART_IO_PA_EVEN:                 return( PARITY_EVEN );
+    case UART_IO_PA_ODD:                  return( PARITY_ODD );
+    case UART_IO_PA_SPACE:                return( PARITY_FORCED_TO_0 );
+    case NOT_SUPPORTED_UART_IO_PA_MARK:   return( PARITY_FORCED_TO_1 );
+    default:                              TRACE_EVENT( "UNEXP PARITY IN ICF" );
+  }
+#endif /*UART*/
+  return( PARITY_NONE );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_GetCallType       |
++-------------------------------------------------------------------+
+
+  PURPOSE : get call type out of bearer capabilities
+
+*/
+
+GLOBAL SHORT cmhCC_GetCallType_from_bearer ( void * bearCap )
+{
+  T_bcpara * pBC = (T_bcpara*) bearCap;
+
+  switch( pBC->bearer_serv )
+  {
+  case( BEARER_SERV_SYNC          ):
+    switch( pBC -> conn_elem )
+    {
+    case( CONN_ELEM_TRANS         ): return( CRING_TYP_Sync );
+    case( CONN_ELEM_NON_TRANS     ): return( CRING_TYP_RelSync );
+    default:                         TRACE_EVENT( "UNEXP CONN ELEMENT IN CTB" );
+    /*lint -fallthrough*/
+    case( CONN_ELEM_NOT_PRES      ): return( CRING_TYP_NotPresent );
+    }
+
+  case( BEARER_SERV_ASYNC         ):
+    switch( pBC -> conn_elem )
+    {
+    case( CONN_ELEM_TRANS         ): return( CRING_TYP_Async );
+    case( CONN_ELEM_NON_TRANS     ): return( CRING_TYP_RelAsync );
+    default:                         TRACE_EVENT( "UNEXP CONN ELEMENT IN CTB" );
+    /*lint -fallthrough*/
+    case( CONN_ELEM_NOT_PRES      ): return( CRING_TYP_NotPresent );
+    }
+
+  case( BEARER_SERV_FAX           ): return( CRING_TYP_Fax );
+
+  case( BEARER_SERV_SPEECH_CTM    ):
+  case( BEARER_SERV_SPEECH        ): return( CRING_TYP_Voice );
+
+  case( BEARER_SERV_AUX_SPEECH_CTM):
+  case( BEARER_SERV_AUX_SPEECH    ): return( CRING_TYP_AuxVoice );
+
+  default:                           TRACE_EVENT( "UNEXP BEARER SERVICE IN CTB" );
+  /*lint -fallthrough*/
+  case( BEARER_SERV_PAD_ACCESS    ):
+  case( BEARER_SERV_PACKET_ACCESS ):
+  case( BEARER_SERV_NOT_PRES      ): return( CRING_TYP_NotPresent );
+
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_GetSrvType        |
++-------------------------------------------------------------------+
+
+  PURPOSE : get service type out of bearer capabilities
+
+*/
+
+GLOBAL SHORT cmhCC_GetSrvType ( void * bearCap )
+{
+  T_bcpara * pBC = (T_bcpara*) bearCap;
+
+  switch( pBC->bearer_serv )
+  {
+  case( BEARER_SERV_SYNC          ):
+    switch( pBC -> conn_elem )
+    {
+    case( CONN_ELEM_TRANS         ): return( CR_SERV_Sync );
+    case( CONN_ELEM_NON_TRANS     ): return( CR_SERV_RelSync );
+    default:                         TRACE_EVENT( "UNEXP CONN ELEMENT IN CTB" );
+    /*lint -fallthrough*/
+    case( CONN_ELEM_NOT_PRES      ): return( CR_SERV_NotPresent );
+    }
+
+  case( BEARER_SERV_ASYNC         ):
+    switch( pBC -> conn_elem )
+    {
+    case( CONN_ELEM_TRANS         ): return( CR_SERV_Async );
+    case( CONN_ELEM_NON_TRANS     ): return( CR_SERV_RelAsync );
+    default:                         TRACE_EVENT( "UNEXP CONN ELEMENT IN CTB" );
+    /*lint -fallthrough*/
+    case( CONN_ELEM_NOT_PRES      ): return( CR_SERV_NotPresent );
+    }
+
+  default:                           TRACE_EVENT( "UNEXP BEARER SERVICE IN CTB" );
+  /*lint -fallthrough*/
+  case( BEARER_SERV_FAX           ):
+  case( BEARER_SERV_SPEECH_CTM    ):
+  case( BEARER_SERV_AUX_SPEECH_CTM):
+  case( BEARER_SERV_SPEECH        ):
+  case( BEARER_SERV_AUX_SPEECH    ):
+  case( BEARER_SERV_PAD_ACCESS    ):
+  case( BEARER_SERV_PACKET_ACCESS ):
+  case( BEARER_SERV_NOT_PRES      ): return( CR_SERV_NotPresent );
+
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_GetDataRate       |
++-------------------------------------------------------------------+
+
+  PURPOSE : get user data rate out of bearer capabilities
+
+*/
+
+GLOBAL SHORT cmhCC_GetDataRate ( void * bearCap )
+{
+  T_bcpara * pBC = (T_bcpara*) bearCap;
+
+  switch( pBC->rate )
+  {
+  case( UR_0_3_KBIT  ): return( BS_SPEED_300_V110   );
+  case( UR_1_2_KBIT  ): return( BS_SPEED_1200_V110  );
+  case( UR_2_4_KBIT  ): return( BS_SPEED_2400_V110  );
+  case( UR_4_8_KBIT  ): return( BS_SPEED_4800_V110  );
+  case( UR_9_6_KBIT  ): return( BS_SPEED_9600_V110  );
+  case( UR_14_4_KBIT ): return( BS_SPEED_14400_V110 );
+  default:              return( BS_SPEED_NotPresent );
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_GetFormat         |
++-------------------------------------------------------------------+
+
+  PURPOSE : get character format out of bearer capabilities
+
+*/
+
+GLOBAL SHORT cmhCC_GetFormat ( void * bearCap )
+{
+  T_bcpara * pBC = (T_bcpara*) bearCap;
+
+  if( pBC->data_bits EQ DATA_7_BIT)
+  {
+    if( pBC->stop_bits EQ STOP_1_BIT )
+    {
+      if( pBC->parity EQ PARITY_NONE ) return( BS_FRM_Dat7_Par0_St1 );
+      else                             return( BS_FRM_Dat7_Par1_St1 );
+    }
+    else return( BS_FRM_Dat7_Par0_St2 );
+  }
+  else
+  {
+    if( pBC->stop_bits EQ STOP_1_BIT )
+    {
+      if( pBC->parity EQ PARITY_NONE ) return( BS_FRM_Dat8_Par0_St1 );
+      else                             return( BS_FRM_Dat8_Par1_St1 );
+    }
+    else return( BS_FRM_Dat8_Par0_St2 );
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_GetParity         |
++-------------------------------------------------------------------+
+
+  PURPOSE : get parity out of bearer capabilities
+
+*/
+
+GLOBAL SHORT cmhCC_GetParity ( void * bearCap )
+{
+  T_bcpara * pBC = (T_bcpara*) bearCap;
+
+  switch( pBC->parity )
+  {
+  case( PARITY_ODD ):         return( BS_PAR_Odd );
+  case( PARITY_EVEN ):        return( BS_PAR_Even );
+  case( PARITY_FORCED_TO_0 ): return( BS_PAR_Space );
+  case( PARITY_FORCED_TO_1 ): return( BS_PAR_Mark );
+  case( PARITY_NONE ):
+  default:                    return( BS_PAR_NotPresent );
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_GetCallClass      |
++-------------------------------------------------------------------+
+
+  PURPOSE : get call class out of bearer service
+
+*/
+
+GLOBAL T_ACI_CLASS cmhCC_GetCallClass ( SHORT cId )
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+
+  switch( ctb->BC[ctb->curBC].bearer_serv )
+  {
+  case( BEARER_SERV_SPEECH_CTM ):
+  case( BEARER_SERV_SPEECH ):     return( CLASS_Vce );
+  case( BEARER_SERV_AUX_SPEECH_CTM ):
+  case( BEARER_SERV_AUX_SPEECH ): return( CLASS_AuxVce );
+  case( BEARER_SERV_FAX    ):     return( CLASS_Fax );
+  case( BEARER_SERV_SYNC   ):     return( CLASS_Dat );
+  case( BEARER_SERV_ASYNC  ):     return( CLASS_Dat );
+  default:                        return( CLASS_NotPresent );
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : PSA_CCF                 |
+|                                 ROUTINE : cmhCC_ctbGetCldNumTyp   |
++-------------------------------------------------------------------+
+
+  PURPOSE : this function builds the type of address information out
+            of the called party information.
+*/
+
+GLOBAL T_ACI_TOA* cmhCC_ctbGetCldNumTyp ( SHORT cId, T_ACI_TOA* pToaBuf )
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+/*
+ *-------------------------------------------------------------------
+ * build type of address information
+ *-------------------------------------------------------------------
+ */
+  if (ctb->cldPty.ton EQ TON_NOT_PRES )
+
+    return( NULL );
+
+  pToaBuf -> ton = ctb->cldPty.ton;
+  pToaBuf -> npi = ctb->cldPty.npi;
+
+  return( pToaBuf );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : PSA_CCF                 |
+|                                 ROUTINE : cmhCC_ctbGetCldSubTyp   |
++-------------------------------------------------------------------+
+
+  PURPOSE : this function builds the type of subaddress information
+            out of the called party information.
+*/
+
+GLOBAL T_ACI_TOS* cmhCC_ctbGetCldSubTyp ( SHORT cId, T_ACI_TOS * pTosBuf )
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+/*
+ *-------------------------------------------------------------------
+ * build type of subaddress information
+ *-------------------------------------------------------------------
+ */
+  if( ctb->cldPtySub.tos EQ TOS_NOT_PRES )
+
+    return( NULL );
+
+  pTosBuf -> tos = ctb->cldPtySub.tos;
+  pTosBuf -> oe  = ctb->cldPtySub.odd_even;
+
+  return( pTosBuf );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : PSA_CCF                 |
+|                                 ROUTINE : cmhCC_ctbGetClrNumTyp   |
++-------------------------------------------------------------------+
+
+  PURPOSE : this function builds the type of address information out
+            of the calling party information.
+*/
+
+GLOBAL T_ACI_TOA* cmhCC_ctbGetClrNumTyp ( SHORT cId, T_ACI_TOA* pToaBuf )
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+/*
+ *-------------------------------------------------------------------
+ * build type of address information
+ *-------------------------------------------------------------------
+ */
+  if (ctb->clgPty.ton EQ TON_NOT_PRES )
+
+    return( NULL );
+
+  pToaBuf -> ton = ctb->clgPty.ton;
+  pToaBuf -> npi = ctb->clgPty.npi;
+
+  return( pToaBuf );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : PSA_CCF                 |
+|                                 ROUTINE : cmhCC_ctbGetClrSubTyp   |
++-------------------------------------------------------------------+
+
+  PURPOSE : this function builds the type of subaddress information
+            out of the calling party information.
+*/
+
+GLOBAL T_ACI_TOS* cmhCC_ctbGetClrSubTyp ( SHORT cId, T_ACI_TOS * pTosBuf )
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+/*
+ *-------------------------------------------------------------------
+ * build type of subaddress information
+ *-------------------------------------------------------------------
+ */
+  if( ctb->clgPtySub.tos EQ TOS_NOT_PRES )
+
+    return( NULL );
+
+  pTosBuf -> tos = ctb->clgPtySub.tos;
+  pTosBuf -> oe  = ctb->clgPtySub.odd_even;
+
+  return( pTosBuf );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : PSA_CCF                 |
+|                                 ROUTINE : cmhCC_ctbGetRdrNumTyp   |
++-------------------------------------------------------------------+
+
+  PURPOSE : this function builds the type of address information out
+            of the redirecting party information.
+*/
+
+GLOBAL T_ACI_TOA* cmhCC_ctbGetRdrNumTyp ( SHORT cId, T_ACI_TOA* pToaBuf )
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+/*
+ *-------------------------------------------------------------------
+ * build type of address information
+ *-------------------------------------------------------------------
+ */
+  if( ctb->rdrPty.ton EQ TON_NOT_PRES )
+
+    return( NULL );
+
+  pToaBuf -> ton = ctb->rdrPty.ton;
+  pToaBuf -> npi = ctb->rdrPty.npi;
+
+  return( pToaBuf );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : PSA_CCF                 |
+|                                 ROUTINE : cmhCC_ctbGetRdrSubTyp   |
++-------------------------------------------------------------------+
+
+  PURPOSE : this function builds the type of subaddress information
+            out of the redirecting party information.
+*/
+
+GLOBAL T_ACI_TOS* cmhCC_ctbGetRdrSubTyp ( SHORT cId, T_ACI_TOS * pTosBuf )
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+/*
+ *-------------------------------------------------------------------
+ * build type of subaddress information
+ *-------------------------------------------------------------------
+ */
+  if( ctb->rdrPtySub.tos EQ TOS_NOT_PRES )
+
+    return( NULL );
+
+  pTosBuf -> tos = ctb->rdrPtySub.tos;
+  pTosBuf -> oe  = ctb->rdrPtySub.odd_even;
+
+  return( pTosBuf );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)    MODULE  : CMH_CCR                      |
+|                            ROUTINE : cmhrat_no_carrier            |
++-------------------------------------------------------------------+
+
+  PURPOSE : Send NO CARRIER to user (saves memory in comparaison to R_AT).
+            It also send a log to MFW, and sends NO CARRIER to the owner
+            of the call if it is necessary (owner different from call killer).
+
+            It also sends a DCD off to UART.
+
+*/
+
+GLOBAL void cmhrat_calldisc( RAT_ID response, UBYTE srcId, T_ACI_AT_CMD cmdId, SHORT cId )
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+
+  TRACE_FUNCTION("cmhrat_calldisc");
+
+  /* CHECK response: THIS ONLY WORKS BECAUSE R_AT has the same parameters
+  for RAT_NO_CARRIER, RAT_BUSY and RAT_NO_ANSWER */
+  switch( response )
+  {
+  case( RAT_NO_CARRIER ):
+  case( RAT_BUSY ):
+  case( RAT_NO_ANSWER ):
+    break;
+  default:
+    return;
+  }
+
+  if( srcId EQ (UBYTE)CMD_SRC_NONE       OR
+      srcId EQ ctb->calOwn )
+  {
+    /* inform only call owner */
+
+    R_AT( response, ctb->calOwn ) ( AT_CMD_NONE, cId+1 );
+  }
+  else
+  {
+    /* inform both call owner and call killer */
+    R_AT( response, ctb->calOwn ) ( AT_CMD_NONE, cId+1 );
+    R_AT( response, srcId ) ( cmdId, cId+1 );
+  }
+
+#ifdef FF_ATI
+  /* tlu:io_setDCD (ctb->calOwn, IO_DCD_OFF); */ /* V.24 DCD Line */
+#endif
+
+  cmh_logRslt ( (T_ACI_CMD_SRC)srcId, response, cmdId, (SHORT)(cId+1), -1, -1 );
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_SndDiscRsn        |
++-------------------------------------------------------------------+
+
+  PURPOSE : Find out the reason for disconnection and send it to the
+            caller.
+
+*/
+
+GLOBAL void cmhCC_SndDiscRsn ( SHORT cId )
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+  UBYTE cmdBuf;     /* buffers current command */
+  UBYTE srcBuf;     /* buffers current command source */
+
+  TRACE_FUNCTION ("cmhCC_SndDiscRsn()");
+
+  cmdBuf = ctb->curCmd;
+  srcBuf = ctb->curSrc;
+
+  if (rdlPrm.rdlcId EQ NO_ENTRY)
+  {
+    cmhCC_PrepareCmdEnd (cId, NULL, NULL);
+  }
+
+  if( cmdBuf EQ AT_CMD_NONE )
+  {
+    srcBuf = ctb->calOwn; /* call owner is the only one that
+                             could possibly be informed */
+  }
+
+  if( cmdBuf EQ AT_CMD_ABRT )
+  {
+    R_AT( RAT_OK, srcBuf )( cmdBuf );
+    /* log result */
+    cmh_logRslt ( srcBuf, RAT_OK,
+                  (T_ACI_AT_CMD)cmdBuf, (SHORT)(cId+1), -1, -1 );
+    return;
+  }
+
+/*
+ *-------------------------------------------------------------------
+ * check for reject cause
+ *-------------------------------------------------------------------
+ */
+  /* 
+   * Make use of ctb here or die. Problem with TI compiler 1.22e.
+   * Seen. Exactly here. GET_CAUSE_VALUE() expression seems to become too 
+   * complicated without the help of this variable. You may get a wrong 
+   * ARM instruction otherwise in the generated code.
+   */
+  if(GET_CAUSE_VALUE(ctb->rejCs) NEQ NOT_PRESENT_8BIT)
+  {
+    /* when there is no SIM error, send the NO CARRIER command,
+       otherwise cmhSIM_GetSIMError send the error command (CME) */
+    if (GET_CAUSE_ORIGIN_ENTITY(ctb->rejCs) NEQ SIM_ORIGINATING_ENTITY OR
+        cmhSIM_GetSIMError(srcBuf, cmdBuf) EQ AT_FAIL )
+    {
+      cmhrat_calldisc(RAT_NO_CARRIER, srcBuf, cmdBuf, cId);
+      return;
+    }
+    else
+    {
+      cmh_logRslt ( srcBuf, RAT_CME,
+                   (T_ACI_AT_CMD)cmdBuf, (SHORT)(cId+1), -1, -1 );
+    }
+    return;
+  }
+
+/*
+ *-------------------------------------------------------------------
+ * otherwise check for normal cause
+ *-------------------------------------------------------------------
+ */
+  switch( ctb->nrmCs )
+  {
+  case( MNCC_CAUSE_USER_BUSY ):
+    cmhrat_calldisc(RAT_BUSY, srcBuf, cmdBuf, cId);
+    break;
+
+  case( MNCC_CAUSE_NO_RESPONSE ):
+  case( MNCC_CAUSE_ALERT_NO_ANSWER ):
+    cmhrat_calldisc(RAT_NO_ANSWER, srcBuf, cmdBuf, cId);
+    break;
+
+  default:
+    cmhrat_calldisc(RAT_NO_CARRIER, srcBuf, cmdBuf, cId);
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_ChckInCallMdfy    |
++-------------------------------------------------------------------+
+
+  PURPOSE : check the case of in-call modification. The function
+            returns TRUE or FALSE, whether a modification is allowed
+            or not.
+
+*/
+
+GLOBAL BOOL cmhCC_ChckInCallMdfy ( SHORT cId, UBYTE cmd )
+{
+#ifdef FAX_AND_DATA
+  T_CC_CALL_TYPE calltype = cmhCC_getcalltype(cId);
+
+  TRACE_FUNCTION ("cmhCC_ChckInCallMdfy()");
+
+#ifdef CO_UDP_IP
+  if(calltype EQ UDPIP_CALL)
+  {
+    /* modifying a Wap Call should not be allowed */
+    return( FALSE);
+  }
+#endif /* CO_UDP_IP */
+
+  /* check call mode */
+  switch( psaCC_ctb(cId)->rptInd )
+  {
+    /* single call */
+    case( RI_NOT_PRES ):
+      return( FALSE );
+
+    /* voice followed by data */
+    case( RI_SEQUENTIAL ):
+      if(calltype EQ VOICE_CALL)
+      {
+        switch( cmd )
+        {
+          case( AT_CMD_D ):       /* modify commands */
+          case( AT_CMD_A ):
+
+            TRACE_EVENT( "MODIFY FROM SPEECH" );
+            return( TRUE );
+        }
+      }
+      return( FALSE );
+
+    /* alternating call */
+    case( RI_CIRCULAR ):
+      switch(calltype)
+      {
+        case(VOICE_CALL):
+          switch( cmd )
+          {
+            case( AT_CMD_D ):       /* modify commands */
+            case( AT_CMD_A ):
+
+              TRACE_EVENT( "MODIFY FROM SPEECH" );
+              return( TRUE );
+          }
+          return( FALSE );
+
+        case( TRANS_CALL ):
+        case( NON_TRANS_CALL ):
+        case( FAX_CALL ):
+          if(cmd EQ AT_CMD_H )       /* modify commands */
+          {
+            TRACE_EVENT( "MODIFY FROM DATA/FAX" );
+            return( TRUE );
+          }
+          break;
+      }
+      return( FALSE );
+  }
+#endif /* FAX_AND_DATA */
+  return(FALSE);
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_atdsendok         |
++-------------------------------------------------------------------+
+
+  PURPOSE : returns TRUE if OK is to be sent directly: in case of ATD,
+            if COLP is not set and source is neither LCL nor SAT nor CSSN ...
+*/
+
+GLOBAL BOOL cmhCC_atdsendok ( SHORT cId )
+{
+  T_CC_CALL_TYPE calltype = cmhCC_getcalltype(cId);
+  T_ACI_AT_CMD   cmd      = psaCC_ctb(cId)->curCmd;
+  SHORT          srcId    = psaCC_ctb(cId)->curSrc;
+  
+#ifdef FF_ATI 
+  TRACE_EVENT_P6("cmhCC_atdsendok(): cmd: %d, calltype: %d, srcId: %d, COLP_stat: %d, SATinv: %d, CSSI_stat: %d",
+                                     cmd,  calltype, srcId, at.flags.COLP_stat, psaCC_ctb(cId)->SATinv,
+                                     ati_user_output_cfg[srcId].CSSI_stat);
+#endif /* FF_ATI */ 
+
+#ifdef FF_ATI
+  if(cmd NEQ AT_CMD_D) /* can happen ONLY to ATD */
+    return(FALSE);
+
+  if (calltype NEQ VOICE_CALL      OR
+      srcId EQ CMD_SRC_LCL         OR
+      at.flags.COLP_stat EQ 1      OR
+      ati_user_output_cfg[srcId].CSSI_stat EQ 1      OR
+      ati_is_src_type((UBYTE)srcId, ATI_SRC_TYPE_SAT) OR   /* SAT run at cmd */
+      psaCC_ctb(cId)->SATinv )       /* call started by SAT */
+  {
+    TRACE_EVENT("send OK normally: at the end");
+    return(FALSE);   /* OK sent normally: after MNCC_SETUP_CNF has been received */
+  }
+  else
+  {
+    TRACE_EVENT("send OK directly after ATD");
+    return(TRUE);    /* OK sent as soon as MNCC_SETUP_REQ has been sent (see ITU. V25ter) */
+  }
+#else
+  return(FALSE);
+#endif /* AT INTERPRETER */
+
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_flagCall          |
++-------------------------------------------------------------------+
+
+  PURPOSE : flag a call for multy-call operation
+
+*/
+
+GLOBAL void cmhCC_flagCall ( SHORT cId, USHORT * flags )
+{
+  *flags |= (0x1 << cId);
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_tstAndUnflagCall  |
++-------------------------------------------------------------------+
+
+  PURPOSE : test and unflag a call for multy-call operation. Return
+            the test result.
+
+*/
+
+GLOBAL BOOL cmhCC_tstAndUnflagCall ( SHORT cId, USHORT * flags )
+{
+  if( *flags & (1u << cId))
+  {
+    *flags &= ~(1u << cId);
+    return( TRUE );
+  }
+  return( FALSE );
+}
+
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CC                  |
+|                                 ROUTINE : cmhCC_getcalltype       |
++-------------------------------------------------------------------+
+
+  PURPOSE : returns the type of call of cId (voice, data, fax...).
+
+*/
+
+GLOBAL T_CC_CALL_TYPE cmhCC_getcalltype(SHORT cId)
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+
+  /* Check whether the cId is valid or not, if not return NO_VLD_CC_CALL_TYPE */
+  if (!psaCC_ctbIsValid (cId))
+    return(NO_VLD_CC_CALL_TYPE);
+
+  if (ctb EQ NULL)
+    return (T_CC_CALL_TYPE)ccShrdPrm.callType[cId]; /* Call already gone */
+
+  /* determine type of call */
+  switch( ctb->BC[ctb->curBC].bearer_serv )
+  {
+    default:
+      return(NO_VLD_CC_CALL_TYPE);
+
+    case( BEARER_SERV_AUX_SPEECH ):
+    case( BEARER_SERV_SPEECH ):     /* voice call */
+    case( BEARER_SERV_AUX_SPEECH_CTM ):
+    case( BEARER_SERV_SPEECH_CTM ):     /* voice call with CTM */
+      return(VOICE_CALL);
+
+#ifdef FAX_AND_DATA
+    case( BEARER_SERV_ASYNC ):      /* asynchronous data call */
+      switch( ctb->BC[ctb->curBC].conn_elem )
+      {
+        case( CONN_ELEM_TRANS ):    /* transparent data call */
+        case( CONN_ELEM_NON_TRANS ):  /* non transparent data call */
+ /* check if a WAP call is in process... If yes then end call */
+#if defined(CO_UDP_IP) OR defined(FF_GPF_TCPIP)
+          if (Wap_Call)
+          {
+            if (cId EQ wapId)
+            {
+              if(is_gpf_tcpip_call()) {
+                GPF_TCPIP_STATEMENT(return(TCPIP_CALL));
+              }
+              else {
+                UDPIP_STATEMENT(return(UDPIP_CALL));
+              }
+            }
+          }
+#endif  /* CO_UDP_IP || FF_GPF_TCPIP */
+
+#ifdef FF_TCP_IP
+          if (pppShrdPrm.is_PPP_CALL EQ TRUE)
+          {
+            return (PPP_CALL);
+          }
+#endif  /* of #ifdef FF_TCP_IP */
+
+          if( ctb->BC[ctb->curBC].conn_elem EQ CONN_ELEM_TRANS )
+          {
+            return(TRANS_CALL);
+          }
+          else
+            return(NON_TRANS_CALL);
+      }
+      return(NO_VLD_CC_CALL_TYPE);
+
+    case( BEARER_SERV_FAX ):          /* fax call */
+      return(FAX_CALL);
+
+    case( BEARER_SERV_SYNC  ):
+    case( BEARER_SERV_PAD_ACCESS  ):
+    case( BEARER_SERV_PACKET_ACCESS ):
+      return(UNSUP_DATA_CALL);
+#endif /* of #ifdef FAX_AND_DATA */
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                 |
+|                                 ROUTINE : cmhCC_AcceptCall        |
++-------------------------------------------------------------------+
+
+  PURPOSE : accept a call.
+*/
+
+GLOBAL void cmhCC_AcceptCall(SHORT cId, T_ACI_CMD_SRC srcId, T_ACI_AT_CMD cmd)
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+  UBYTE idx;                  /* holds index value */
+
+  TRACE_FUNCTION("cmhCC_AcceptCall( )");
+
+  if (ctb EQ NULL OR ctb->calStat NEQ CS_ACT_REQ)
+  {
+    ACI_ERR_DESC( ACI_ERR_CLASS_Cme, CME_ERR_Unknown );
+    R_AT(RAT_CME, srcId)(cmd, CME_ERR_Unknown);
+    return;
+  }
+
+  CHLDaddInfo = NO_CHLD_ADD_INFO;
+  ctb->curCmd = cmd;
+  ctb->calOwn = srcId;
+  ctb->curSrc = srcId;
+
+  /*
+   * ETSI 11.10 26.12.4 requires "that for speech calls, the mobile station
+   * shall attach the user connection at latest when sending the connect
+   * message, except if there is no compatible radio resource available at
+   * this time."
+   */
+
+  psaCC_phbAddNtry ( LRN, cId, CT_MTC, NULL );  /* add call to LRN PB */
+
+#ifdef FF_ATI
+  io_setRngInd ( IO_RING_OFF, CRING_TYP_NotPresent, CRING_TYP_NotPresent ); /* V.24 Ring Indicator Line */
+#endif
+
+  for( idx = 0; idx < CMD_SRC_MAX; idx++ )
+  {
+    R_AT( RAT_CRING_OFF, idx )( cId+1 );
+  }
+
+  psaCC_AcceptCall(cId);
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                 |
+|                                 ROUTINE : cmhCC_RetrieveCall      |
++-------------------------------------------------------------------+
+
+  PURPOSE : Retrieve either a Multyparty or a Normal Call
+*/
+GLOBAL void cmhCC_RetrieveCall(SHORT cId, T_ACI_CMD_SRC srcId)
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+
+  TRACE_FUNCTION("cmhCC_RetrieveCall( )");
+
+  if (ctb EQ NULL OR ctb->calStat NEQ CS_HLD)
+  {
+    return;
+  }
+
+  CHLDaddInfo = NO_CHLD_ADD_INFO;
+  ctb->curCmd = AT_CMD_CHLD;
+  ctb->curSrc = srcId;
+
+  if (ctb->mptyStat EQ CS_ACT )
+  {
+    psaCC_RetrieveMPTY (cId);
+  }
+  else
+  {
+    psaCC_RetrieveCall (cId);
+  }
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                 |
+|                                 ROUTINE : cmhCC_NewCall           |
++-------------------------------------------------------------------+
+
+  PURPOSE : Initiate a new call.
+*/
+GLOBAL void cmhCC_NewCall(SHORT cId, T_ACI_CMD_SRC srcId, T_ACI_AT_CMD cmd)
+{
+  T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+
+  TRACE_FUNCTION("cmhCC_NewCall( )");
+
+  CHLDaddInfo = NO_CHLD_ADD_INFO;
+  ctb->curCmd = cmd;
+  ctb->calOwn = srcId;
+  ctb->curSrc = srcId;
+
+  psaCC_NewCall(cId);
+}
+
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                 |
+|                                 ROUTINE : cmhCC_HoldCall          |
++-------------------------------------------------------------------+
+
+  PURPOSE : Put a call on hold (multyparty or normal call)
+*/
+GLOBAL void cmhCC_HoldCall(SHORT cId, T_ACI_CMD_SRC srcId, T_ACI_AT_CMD cmd)
+{
+  T_CC_CALL_TBL  *ctb;         /* points to CC call table */
+  T_CC_CMD_PRM   *pCCCmdPrm;   /* points to CC command parameters */
+
+  TRACE_FUNCTION("cmhCC_HoldCall( )");
+
+  ctb = ccShrdPrm.ctb[cId];
+  pCCCmdPrm = &cmhPrm[srcId].ccCmdPrm;
+
+  cmhCC_flagCall( cId, &(pCCCmdPrm->mltyCncFlg));
+  ctb->curCmd = cmd;
+  ctb->curSrc = srcId;
+
+  if (ctb->mptyStat EQ CS_ACT)
+  {
+    psaCC_HoldMPTY(cId);
+  }
+  else
+  {
+    psaCC_HoldCall(cId);
+  }
+}
+/*
++-------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                 |
+|                                 ROUTINE : cmhCC_ClearCall         |
++-------------------------------------------------------------------+
+
+  PURPOSE : disconnect a call.
+*/
+
+GLOBAL void cmhCC_ClearCall ( SHORT cId,
+                              USHORT cs,
+                              T_ACI_CMD_SRC srcId,
+                              UBYTE cmd,
+                              SHORT *waitId )
+{
+  T_CC_CMD_PRM   *pCCCmdPrm = NULL;   /* points to CC command parameters */
+  T_CC_CALL_TYPE calltype = cmhCC_getcalltype(cId);
+  T_CC_CALL_TBL  *ctb = ccShrdPrm.ctb[cId];
+
+  TRACE_FUNCTION( "cmhCC_ClearCall()" );
+
+  if(srcId > CMD_SRC_NONE AND
+     srcId < CMD_SRC_MAX)
+  {
+    pCCCmdPrm = &cmhPrm[srcId].ccCmdPrm;
+  }
+
+  if (ccShrdPrm.ctb[cId] EQ NULL)
+  {
+    TRACE_EVENT("cmhCC_ClearCall(): Not a valid cId");
+    return;
+  }
+
+  TRACE_EVENT_P2("Clear Call ==> calStat: %d, calType: %d", ctb->calStat, ctb->calType NEQ CT_MOC);
+
+  if(cmd NEQ AT_CMD_NONE) /* for instance +CHUP / H / Z */
+  {
+    /* Check whether an incoming call is rejected by the user. */
+    if( ctb->calStat EQ CS_ACT_REQ AND
+        ctb->calType EQ CT_MTC)
+    {
+      psaCC_phbAddNtry ( LRN, cId, CT_MTC, NULL );  /* Add call to Last Received Number (LRN) PB */
+    }
+
+    /* Check whether call is a waiting call */
+    if(waitId NEQ NULL)
+    {
+      if( ctb->calStat EQ CS_ACT_REQ AND
+          ctb->calType EQ CT_MTC     AND
+          ctb->calOwn  EQ OWN_NONE )
+      {
+        /* tag a found waiting call */
+        TRACE_EVENT_P1("Waiting Call Found ! waitId=%d", cId);
+        *waitId = cId;
+        return;
+      }
+    }
+  }
+
+  ccShrdPrm.cIdFail = cId;
+  if ( cmd NEQ AT_CMD_NONE ) /* this will decide whether a result is sent to the user or not */
+  {
+    /* ATZ and ATH or AT+CHUP for instance */
+    if(pCCCmdPrm NEQ NULL)
+    {
+      cmhCC_flagCall( cId, &(pCCCmdPrm -> mltyDscFlg));
+    }
+    ctb->curCmd = cmd;
+    ctb->curSrc = srcId;
+  }
+  else
+  {
+    /* AOC for instance */
+    cmhCC_SndDiscRsn(cId);
+  }
+  ctb->nrmCs     = cs;
+
+  /* determine call status */
+  switch( ctb->calStat )
+  {
+    /* active calls */
+    case( CS_ACT ):
+      switch( calltype )
+      {
+        case( VOICE_CALL ):
+          psaCC_ClearCall (cId);
+          break;
+
+#if defined FAX_AND_DATA AND defined (DTI)
+
+        case( TRANS_CALL ):
+          ccShrdPrm.datStat = DS_DSC_REQ;
+          cmhTRA_Deactivate ();
+          break;
+
+        case( NON_TRANS_CALL ):
+          ccShrdPrm.datStat = DS_DSC_REQ;
+          cmhL2R_Deactivate ();
+          break;
+
+#ifdef FF_FAX
+        case( FAX_CALL ):
+          ccShrdPrm.datStat = DS_DSC_REQ;
+          cmhT30_Deactivate ();
+          break;
+#endif /* FF_FAX */
+
+#if defined(CO_UDP_IP) OR defined(FF_GPF_TCPIP)
+
+#ifdef CO_UDP_IP
+        case( UDPIP_CALL ):
+#endif
+
+#ifdef FF_GPF_TCPIP
+        case( TCPIP_CALL ):
+          {
+            T_DCM_STATUS_IND_MSG msg;
+            msg.hdr.msg_id = DCM_ERROR_IND_MSG;
+            msg.result     = DCM_PS_CONN_BROKEN;
+            dcm_send_message(msg, DCM_SUB_WAIT_SATDN_CNF);
+          }
+#endif
+          ccShrdPrm.datStat = DS_DSC_REQ;
+          /* also test whether a SAT E connection has to be shut down */
+          if ( (cmd NEQ AT_CMD_NONE)
+#ifdef FF_SAT_E
+              OR (satShrdPrm.opchAcptSrc EQ srcId)
+#endif
+             )
+          {
+            /* set the last possible state for the next "else if" statement */
+            T_ACI_WAP_STATES last_state = Wap_Not_Init;
+            if(is_gpf_tcpip_call()) {
+              GPF_TCPIP_STATEMENT(last_state = TCPIP_Deactivation);
+            }
+            else {
+              last_state = UDPA_Deactivation;
+            }
+
+            /* Terminate PPP if established. UP indicate that entities under PPP are still activ. */
+            if((pppShrdPrm.state EQ PPP_ESTABLISHED) OR
+               (pppShrdPrm.state EQ PPP_ESTABLISH))
+            {
+              cmhPPP_Terminate(UP);
+            }
+            else if (wap_state >  Wap_Not_Init  AND
+                     wap_state <= last_state)
+            {
+              cmhPPP_Terminated(); /* Trigger termination of SCPIP if PPP did not yet start (ACI-FIX-9317) */
+            }
+          }
+          break;
+#endif  /* defined(CO_UDP_IP) OR defined(FF_GPF_TCPIP) */
+
+#if defined (FF_TCP_IP) AND defined (DTI)
+        case( PPP_CALL ):
+          ccShrdPrm.datStat = DS_DSC_REQ;
+          if (pppShrdPrm.is_PPP_CALL EQ TRUE)
+          {
+            dti_cntrl_close_dpath_from_src_id((UBYTE)srcId);
+          }
+#endif  /* of #ifdef FF_TCP_IP */
+
+#endif    /* of #ifdef FAX_AND_DATA */
+        default:
+            TRACE_ERROR("Wrong call type");
+    }
+      break;
+
+      /* other present calls */
+    case( CS_HLD ):
+    case( CS_HLD_REQ ):
+    case( CS_MDF_REQ ):
+    case( CS_DSC_REQ ):
+    case( CS_ACT_REQ ):
+    case( CS_CPL_REQ ):
+      psaCC_ClearCall (cId);
+      break;
+  }
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                  |
+| STATE   : code                  ROUTINE : cmhCC_chkDTMFDig         |
++--------------------------------------------------------------------+
+
+  PURPOSE : This function is used extract DTMF digits from the dialled
+            number. The DTMF digits, including separators, are stored
+            in the DTMF parameters and will be send automatically after
+            the connection is established.
+            BOOL within_dial_string parameter to tell if DTMF tones are
+            within a dial string or directly given.
+            USHROT length: possibility to give the number of dtmf to be
+            sent (relevant only if not within a dial string).
+*/
+
+/* next function fills up dtmf structure with extracted prms */
+LOCAL void cmhCC_fillDTMFDig ( SHORT cId, CHAR *dtmf, USHORT dtmf_length )
+{
+  TRACE_FUNCTION("cmhCC_fillDTMFDig()");
+
+  if( dtmf NEQ NULL )
+  {
+    ccShrdPrm.dtmf.cnt = MINIMUM( dtmf_length, MAX_DTMF_DIG );
+
+    strncpy( (char *)ccShrdPrm.dtmf.dig, dtmf, ccShrdPrm.dtmf.cnt );
+    ccShrdPrm.dtmf.dig[dtmf_length] = '\0';
+
+    ccShrdPrm.dtmf.cur = 0;
+    ccShrdPrm.dtmf.cId = cId;
+    TRACE_EVENT_P2("ccShrdPrm.dtmf.cnt: %d, ccShrdPrm.dtmf.cId: %d", ccShrdPrm.dtmf.cnt, ccShrdPrm.dtmf.cId);
+  }
+  else
+  {
+    ccShrdPrm.dtmf.cnt = 0;
+    ccShrdPrm.dtmf.cur = 0;
+    ccShrdPrm.dtmf.cId = NO_ENTRY;
+  }
+}
+
+GLOBAL BOOL is_digit_dtmf_separator(CHAR digit)
+{
+  switch(digit)
+  {
+    case('P'):  /* PHB_DIAL_PAUSE, PHB_DTMF_SEP */
+    case('p'):
+    case('W'):  /* PHB_DIAL_WAIT */
+    case('w'):
+      TRACE_EVENT("DTMF Separator Found");
+      return(TRUE);
+  }
+  return(FALSE);
+}
+
+LOCAL CHAR * find_dtmf_separator(CHAR *number_string, USHORT length)
+{
+  int i;
+
+  TRACE_FUNCTION("find_dtmf_separator()");
+
+  for(i=0; i<length; i++)
+  {
+    if(is_digit_dtmf_separator(*number_string))
+    {
+      return(number_string);
+    }
+    number_string++;
+  }
+  return(NULL);
+}
+
+GLOBAL void cmhCC_chkDTMFDig ( CHAR*  num,
+                               SHORT  cId,
+                               USHORT length,
+                               BOOL   within_dial_string )
+{
+  CHAR   *dtmf = num;
+  USHORT len   = length;
+
+  TRACE_FUNCTION("cmhCC_chkDTMFDig()");
+
+  if( within_dial_string )
+  {
+    dtmf = find_dtmf_separator(num, (USHORT)strlen(num)); /*strchr( num, PHB_DTMF_SEP ); */
+
+    if(dtmf NEQ NULL)
+    {
+      len = strlen(dtmf);
+      TRACE_EVENT_P1("Call number string contains %d DTMF tones", len);
+    }
+  }
+  else
+  {
+    /* already inialized for the case "not within a phone call" (e.g SAT Send DTMF) */
+  }
+
+  cmhCC_fillDTMFDig ( cId, dtmf, len );
+
+  if(dtmf NEQ NULL)
+  {
+    *dtmf = '\0';  /* split dial string from DTMF tones */
+  }
+
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                  |
+| STATE   : code                  ROUTINE : cmhCC_chkKeySeq          |
++--------------------------------------------------------------------+
+
+  PURPOSE : This function is used to decode a keystroke sequence and
+            to initiate an appropriate action.
+*/
+GLOBAL T_ACI_RETURN cmhCC_chkKeySeq ( T_ACI_CMD_SRC srcId,
+                                      T_CLPTY_PRM *cldPty,
+                                      T_ACI_D_TOC *callType,
+                                      T_ACI_D_CLIR_OVRD *CLIRovrd,
+                                      T_CC_SIM_QLF ccSIMQlf )
+{
+  T_KSD_SEQGRP   seqGrp = SEQGRP_UNKNOWN;
+  T_KSD_SEQPARAM seqPrm;
+  CHAR          *restSeq = NULL;
+  T_ACI_RETURN   retVal;
+  T_PHB_RECORD   phbNtry;
+  T_PHB_TYPE     slctPHB;
+#ifdef SIM_TOOLKIT
+  T_sat_ussd     SATussd; /* to hold USSD string in case of SAT Control */
+#endif /* of SIM_TOOLKIT */
+
+  TRACE_FUNCTION ("cmhCC_chkKeySeq ()");
+
+  if( cldPty EQ NULL         OR
+    strlen (cldPty->num) EQ 0 )
+  {
+    return( AT_CMPL );
+  }
+
+  restSeq = cldPty->num;
+  seqGrp  = SEQGRP_UNKNOWN;
+
+#ifdef SIM_TOOLKIT
+  /* check for SAT SIM Call Control of SS strings */
+  if (ccSIMQlf EQ CC_SIM_YES)
+  {
+    if ( !ksd_isFDNCheckSeq ( cldPty-> num ) )
+      ;    /* bypass *#06# from SSCntrlBySIM */
+    else
+    {
+      retVal = AT_CMPL; /* init */
+      if( ksd_isSATSscs (cldPty->num) )
+      {
+        retVal = cmhSAT_SSCntrlBySIM( cldPty, (UBYTE)srcId );
+      }
+      else if( ksd_isUSSD (cldPty->num) )
+      {
+        if (psaSIM_ChkSIMSrvSup(SRV_USSDsupportInCC))
+        {
+          SATussd.dcs        = 0x0F; /* default alphabet see GSM 03.38 Cell Broadcast dcs */
+          SATussd.c_ussd_str = strlen(cldPty->num);
+          SATussd.ussd_str   = (UBYTE *)cldPty->num;
+
+          retVal = cmhSAT_USSDCntrlBySIM( &SATussd, (UBYTE)srcId );
+        }
+        else
+        {
+          retVal = cmhSAT_SSCntrlBySIM( cldPty, (UBYTE)srcId );
+        }
+      }
+
+      if( retVal NEQ AT_CMPL )
+        return( retVal );
+    }
+  }
+#endif
+
+/*
+ *-------------------------------------------------------------
+ * decode SS string
+ *-------------------------------------------------------------
+ */
+  if (ksd_decode (cldPty->num,
+                  psaCC_ctbAnyCallInUse(),  /* is there any call??? */
+                  &seqGrp,
+                  &restSeq,
+                  &seqPrm) EQ TRUE)
+  {
+    retVal  = AT_FAIL;
+
+  /*
+   *-------------------------------------------------------------
+   * perform desired action
+   *-------------------------------------------------------------
+   */
+    switch (seqGrp)
+    {
+      /*
+       *---------------------------------------------------------
+       * handling of sequence group SEQGRP_IMEI
+       *---------------------------------------------------------
+       */
+      case ( SEQGRP_PRSNT_IMEI ):
+        retVal = cmhSS_ksdIMEI( srcId );
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence group SEQGRP_DIAL
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_DIAL):
+        strncpy(cldPty->num, seqPrm.dial.number, sizeof (cldPty->num) - 1);
+        cldPty->num[sizeof(cldPty->num) - 1] = '\0';
+        return( AT_CONT );
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence group SEQGRP_CHLD
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_CHLD):
+        retVal = sAT_PlusCHLD( srcId, seqPrm.chld.mode, seqPrm.chld.call );
+        if (retVal NEQ AT_FAIL)
+          break; /* successfull CHLD */
+
+        /* else handle as USSD if this CHLD was unexpected/illegal */
+
+        seqPrm.ussd.ussd = (UBYTE *)cldPty->num;
+        seqGrp  = SEQGRP_USSD;
+        restSeq = cldPty->num + strlen (cldPty->num);
+        /* fallthrough */
+      /*lint -fallthrough*/
+      /*
+       *---------------------------------------------------------
+       * handling of sequence group SEQGRP_USSD
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_USSD):
+        retVal = cmhSS_ksdUSSD( srcId, &seqPrm.ussd );
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence group SEQGRP_DIAL_IDX
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_DIAL_IDX):
+
+        if( !cmhPHB_cvtPhbType(cmhPrm[srcId].phbCmdPrm.cmhStor,
+                               &slctPHB ))
+          return( AT_FAIL );
+
+#ifdef TI_PS_FFS_PHB
+        if (pb_read_record (slctPHB, 
+                            seqPrm.dialIdx.index, 
+                            &phbNtry) EQ PHB_OK)
+#else
+        if( pb_read_phys_record( slctPHB, seqPrm.dialIdx.index,
+                                 &phbNtry )
+            EQ PHB_OK )
+#endif
+        {
+          cmhPHB_getAdrStr( cldPty->num, sizeof (cldPty->num) - 1,
+                            phbNtry.number, phbNtry.len );
+
+          cmh_demergeTOA ( phbNtry.ton_npi, &cldPty->ton, &cldPty->npi );
+
+          /*
+           *---------------------------------------------------------
+           * as long as the bearer capabilities are not read from
+           * the phonebook only voice calls are supported
+           *---------------------------------------------------------
+           */
+          *callType = D_TOC_Voice;
+
+          return( AT_CONT );
+        }
+
+        return( AT_FAIL );
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence groups which relates
+       * to call forwarding parameter
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_CF):
+        /* type of address has already been set,
+        and a possible "+" already been removed... */
+        if(seqPrm.cf.num NEQ NULL AND
+           seqPrm.cf.num[0] NEQ 0)   /* only set if there is actually a number */
+        {
+          seqPrm.cf.ton = cldPty->ton;
+          seqPrm.cf.npi = cldPty->npi;
+        }
+        else
+        {
+          seqPrm.cf.ton = TON_NOT_PRES;
+          seqPrm.cf.npi = NPI_NOT_PRES;
+        }
+
+        /* subaddress has also already been split up if existing */
+        if(cldPty->sub[0] NEQ 0)
+        {
+          seqPrm.cf.sub = (UBYTE *)cldPty->sub;
+          seqPrm.cf.tos = cldPty->tos;
+          seqPrm.cf.oe  = cldPty->oe;
+        }
+        else
+          seqPrm.cf.sub = NULL;
+
+        retVal = cmhSS_ksdCF( srcId, &seqPrm.cf );
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence groups which relates to
+       * call barring parameter
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_CB):
+
+        retVal = cmhSS_ksdCB( srcId, &seqPrm.cb );
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence groups which results in a change
+       * of CLIR parameter
+       * in temporary mode a call is set up
+       *---------------------------------------------------------
+       */
+
+      case (SEQGRP_SUP_COLR):
+      case (SEQGRP_INV_COLR):
+      case (SEQGRP_SUP_COLP):
+      case (SEQGRP_INV_COLP):
+      case (SEQGRP_SUP_CLIP):
+      case (SEQGRP_INV_CLIP):
+        if (strlen (restSeq) EQ 0)
+        {
+          T_ACI_KSD_USSD_PRM ksdString;
+
+          ksdString.ussd = (UBYTE*)cldPty->num;
+          retVal = cmhSS_ksdUSSD( srcId, &ksdString );
+        }
+        else
+        {
+          /*
+           *-----------------------------------------------------
+           * temporary mode nonexistant on CLIR, COLR and COLP!!
+           * So ignore USSD and continue calling: "Number not assigned"
+           * will be indicated to the user by the network
+           * This is the common behaviour as found on other phones
+           *-----------------------------------------------------
+           */
+          return( AT_CONT );
+        }
+        break;
+
+      case (SEQGRP_SUP_CLIR):
+      case (SEQGRP_INV_CLIR):
+        if (strlen (restSeq) EQ 0)
+        {
+#if defined(MFW) OR defined(FF_MMI_RIV)
+          if (srcId EQ CMD_SRC_LCL) /* VO: temp solution because MFW uses sAT_Dn to set CLIR */
+            retVal = sAT_PlusCLIR(srcId, seqPrm.Clir.mode);
+          else
+#endif
+          {
+            T_ACI_KSD_USSD_PRM ksdString;
+
+            ksdString.ussd = (UBYTE*)cldPty->num;
+            retVal = cmhSS_ksdUSSD( srcId, &ksdString );
+          }
+        }
+        else
+        {
+          /*
+           *-----------------------------------------------------
+           * in temporary mode a call is set up
+           *-----------------------------------------------------
+           */
+          T_KSD_CLIR   Clir;
+          T_KSD_SEQGRP nextGrp  = SEQGRP_UNKNOWN;
+          CHAR*        remSeq   = NULL;
+
+          Clir.mode = seqPrm.Clir.mode;
+
+          if (ksd_decode (restSeq, FALSE,
+                          &nextGrp, &remSeq,
+                          &seqPrm)
+              AND
+              nextGrp EQ SEQGRP_DIAL)
+          {
+            *CLIRovrd = (Clir.mode EQ CLIR_MOD_Supp)?
+                        D_CLIR_OVRD_Supp: D_CLIR_OVRD_Invoc;
+            strncpy(cldPty->num, seqPrm.dial.number, sizeof(cldPty->num) - 1);
+            cldPty->num[sizeof(cldPty->num) - 1] = '\0';
+            return( AT_CONT );
+          }
+        }
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence group for TTY service
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_TTY_SERV):
+        if (strlen (restSeq) EQ 0)
+        {
+#ifdef FF_TTY
+          retVal = sAT_PercentCTTY (srcId, CTTY_MOD_NotPresent,
+                                    seqPrm.ctty.req);
+#else
+          retVal = AT_FAIL;
+#endif /* FF_TTY */
+        }
+        else
+        {
+          T_KSD_SEQGRP nextGrp  = SEQGRP_UNKNOWN;
+          CHAR*        remSeq   = NULL;
+#ifdef FF_TTY
+          if (seqPrm.ctty.req EQ CTTY_REQ_Off
+              AND ccShrdPrm.ctmReq EQ CTM_ENABLED)
+          {
+            ccShrdPrm.ctmOvwr = TRUE;
+          }
+          else if (seqPrm.ctty.req EQ CTTY_REQ_On
+                   AND ccShrdPrm.ctmReq EQ CTM_DISABLED)
+          {
+            ccShrdPrm.ctmOvwr = TRUE;
+          }
+          else
+          {
+            ccShrdPrm.ctmOvwr = FALSE;
+          }
+#endif /* FF_TTY */
+          if (ksd_decode (restSeq, FALSE,
+                          &nextGrp, &remSeq,
+                          &seqPrm)
+              AND
+              nextGrp EQ SEQGRP_DIAL)
+          {
+            strncpy(cldPty->num, seqPrm.dial.number, sizeof(cldPty->num) - 1);
+            cldPty->num[sizeof(cldPty->num) - 1] = '\0';
+            return( AT_CONT );
+          }
+        }
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence group which relates to
+       * calling line supplementary services
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_CL):
+        retVal = cmhSS_ksdCL( srcId, &seqPrm.cl );
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence groups which results in a
+       * registration of passwords
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_PWD):
+        retVal = cmhSS_ksdPW( srcId, &seqPrm.pwd );
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence groups which results in a
+       * registration of passwords
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_UBLK):
+        retVal = cmhSS_ksdUBLK( srcId, &seqPrm.ublk );
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence groups which relates to
+       * call waiting
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_CW):
+
+        retVal = cmhSS_ksdCW( srcId, &seqPrm.cw );
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence group SEQGRP_CCBS
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_CCBS):
+        retVal = cmhSS_ksdCCBS( srcId, &seqPrm.ccbs );
+        break;
+
+      /*
+       *---------------------------------------------------------
+       * handling of sequence group SEQGRP_UNKNOWN
+       *---------------------------------------------------------
+       */
+      case (SEQGRP_UNKNOWN):
+        TRACE_EVENT ("sequence group unknown");
+        break;
+    }
+  }
+  else
+  {
+    TRACE_EVENT ("ksd_decode failed");
+    return( AT_FAIL );
+  }
+
+  return( retVal );
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                  |
+| STATE   : code                  ROUTINE : cmhCC_fillSetupPrm       |
++--------------------------------------------------------------------+
+
+  PURPOSE : This function fills the call table entry with the
+            necessary setup parameters.
+*/
+
+GLOBAL T_ACI_RETURN cmhCC_fillSetupPrm ( SHORT              cId,
+                                         T_ACI_CMD_SRC      srcId,
+                                         T_CLPTY_PRM       *cldPty,
+                                         T_bcpara          *bc,
+                                         UBYTE              prio,
+                                         T_ACI_D_CLIR_OVRD  clirOvrd,
+                                         T_ACI_D_CUG_CTRL   cugCtrl,
+                                         T_ACI_D_TOC        callType )
+{
+  T_CC_CALL_TBL *pfilled_ctb;
+  T_CC_CMD_PRM  *pCCCmdPrm;  /* points to CC command parameters */
+#ifdef FF_FAX
+  T_T30_CMD_PRM *pT30CmdPrm; /* points to T30 command parameters */
+#endif
+
+  TRACE_FUNCTION ("cmhCC_fillSetupPrm()");
+
+  /* initializing local pointers: */
+  pCCCmdPrm  = &cmhPrm[srcId].ccCmdPrm;
+
+#ifdef FF_FAX
+  pT30CmdPrm = &fnd_cmhPrm[srcId].t30CmdPrm;
+#endif
+
+  pfilled_ctb = ccShrdPrm.ctb[cId];
+
+/*
+ *-----------------------------------------------------------------
+ * check data capabilities
+ *-----------------------------------------------------------------
+ */
+#ifndef FAX_AND_DATA
+  if( callType NEQ D_TOC_Voice )
+  {
+    TRACE_EVENT("No data calls supported");
+    ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_NoDataCallSup );
+    return( AT_FAIL );        /* no data calls allowed */
+  }
+#endif  /* of #ifndef FAX_AND_DATA */
+
+/*
+ *-----------------------------------------------------------------
+ * called address
+ *-----------------------------------------------------------------
+ */
+  if( cldPty EQ NULL )
+  {
+    ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter );
+    return( AT_FAIL );        /* no number */
+  }
+
+  /* priority */
+  pfilled_ctb->prio = prio;
+
+  if ((prio EQ PRIO_NORM_CALL) AND
+      (aoc_check_moc() EQ FALSE))
+    /*
+     * check ACM exceeds ACMmax
+     * for non-emergency calls
+     */
+  {
+    ACI_ERR_DESC( ACI_ERR_CLASS_Ceer, CAUSE_MAKE(DEFBY_STD,
+    ORIGSIDE_MS, ACI_ORIGINATING_ENTITY, CEER_ACM_Max ));
+    return( AT_FAIL );
+  }
+
+  /* subaddress */
+  pfilled_ctb->cldPtySub.tos      = cldPty->tos;
+  pfilled_ctb->cldPtySub.odd_even = cldPty->oe;
+  pfilled_ctb->cldPtySub.c_subaddr
+    = (UBYTE)utl_dialStr2BCD (cldPty->sub,
+                              pfilled_ctb->cldPtySub.subaddr,
+                              SUB_LENGTH);
+
+  /* address */
+  if( pCCCmdPrm -> CSTAdef NEQ TRUE )
+  {
+    pfilled_ctb->cldPty.npi = pCCCmdPrm -> CSTAtoa.npi;
+    pfilled_ctb->cldPty.ton = pCCCmdPrm -> CSTAtoa.ton;
+  }
+  else
+  {
+    pfilled_ctb->cldPty.npi = cldPty->npi;
+    pfilled_ctb->cldPty.ton = cldPty->ton;
+  }
+
+  if (pfilled_ctb->cldPty.called_num NEQ NULL)
+  {
+    ACI_MFREE (pfilled_ctb->cldPty.called_num);
+    pfilled_ctb->cldPty.called_num = NULL;
+  }
+  pfilled_ctb->cldPty.c_called_num =
+    (UBYTE)utl_dialStr2BCD (cldPty->num, NULL, MAX_CC_CALLED_NUMBER);
+  if (pfilled_ctb->cldPty.c_called_num NEQ 0)
+  {
+    ACI_MALLOC (pfilled_ctb->cldPty.called_num,
+                pfilled_ctb->cldPty.c_called_num);
+    (void)utl_dialStr2BCD (cldPty->num,
+                           pfilled_ctb->cldPty.called_num,
+                           pfilled_ctb->cldPty.c_called_num);
+  }
+
+  psaCC_phbSrchNum( cId, CT_MOC);    /* get alpha identifier */
+
+/*
+ *-----------------------------------------------------------------
+ * CLIR setting
+ *-----------------------------------------------------------------
+ */
+  switch( clirOvrd )
+  {
+    case( D_CLIR_OVRD_Supp ):
+      pfilled_ctb->CLIRsup = CLR_SUP;
+      break;
+
+    case( D_CLIR_OVRD_Invoc ):
+      pfilled_ctb->CLIRsup = CLR_SUP_NOT;
+      break;
+
+    case( D_CLIR_OVRD_Default ):
+
+      switch( pCCCmdPrm -> CLIRmode )
+      {
+        case( CLIR_MOD_Subscript ):
+          pfilled_ctb->CLIRsup = CLR_NOT_PRES;
+          break;
+        case( CLIR_MOD_Invoc ):
+          pfilled_ctb->CLIRsup = CLR_SUP_NOT;
+          break;
+        case( CLIR_MOD_Supp ):
+          pfilled_ctb->CLIRsup = CLR_SUP;
+          break;
+      }
+      break;
+
+    default:
+      ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter );
+      return( AT_FAIL );
+  }
+
+/*
+ *-----------------------------------------------------------------
+ * CUG setting
+ *-----------------------------------------------------------------
+ */
+  if ( cugCtrl               EQ D_CUG_CTRL_Present OR
+       pCCCmdPrm -> CCUGmode EQ CCUG_MOD_EnableTmp    )
+  {
+    pfilled_ctb->CUGidx = pCCCmdPrm -> CCUGidx;
+
+    switch( pCCCmdPrm -> CCUGinfo )
+    {
+      case( CCUG_INFO_No ):
+        break;
+      case( CCUG_INFO_SuppOa ):
+        pfilled_ctb->OAsup = TRUE;
+        break;
+      case( CCUG_INFO_SuppPrefCug ):
+        pfilled_ctb->CUGprf = TRUE;
+        break;
+      case( CCUG_INFO_SuppBoth ):
+        pfilled_ctb->OAsup = TRUE;
+        pfilled_ctb->CUGprf = TRUE;
+        break;
+    }
+  }
+
+#ifdef SIM_TOOLKIT
+/*
+ *-----------------------------------------------------------------
+ * SAT settings
+ *-----------------------------------------------------------------
+ */
+
+  /* decides if a disconnect event for a call has already been sent to SAT */
+  pfilled_ctb->SatDiscEvent = FALSE;
+#endif
+
+/*
+ *-----------------------------------------------------------------
+ * bearer services
+ *-----------------------------------------------------------------
+ */
+  switch( callType )
+  {
+    /*
+     *-----------------------------------------------------------------
+     * start with a voice call
+     *-----------------------------------------------------------------
+     */
+    case( D_TOC_Voice ):
+
+      pfilled_ctb->BC[0].rate        = UR_NOT_PRES;
+      pfilled_ctb->BC[0].bearer_serv = cmhCC_set_speech_serv (pCCCmdPrm);
+      pfilled_ctb->BC[0].conn_elem   = CONN_ELEM_NOT_PRES;
+
+      switch( ccShrdPrm.CMODmode )
+      {
+
+      case( CMOD_MOD_Single ):             /* single voice call */
+        pfilled_ctb->rptInd          = RI_NOT_PRES;
+
+        pfilled_ctb->BC[1].rate        = UR_NOT_PRES;
+        pfilled_ctb->BC[1].bearer_serv = BEARER_SERV_NOT_PRES;
+        pfilled_ctb->BC[1].conn_elem   = CONN_ELEM_NOT_PRES;
+        break;
+#ifdef UART
+      case( CMOD_MOD_VoiceDat ):           /* alternating voice/data call */
+        pfilled_ctb->rptInd          = RI_CIRCULAR;
+
+        pfilled_ctb->BC[1].rate        = cmhCC_SelRate( ccShrdPrm.CBSTspeed );
+        pfilled_ctb->BC[1].bearer_serv = cmhCC_SelServ( ccShrdPrm.CBSTname );
+        pfilled_ctb->BC[1].conn_elem   = cmhCC_SelCE  ( ccShrdPrm.CBSTce );
+        pfilled_ctb->BC[1].modem_type  = cmhCC_SelMT  ( ccShrdPrm.CBSTspeed );
+
+        pfilled_ctb->BC[1].stop_bits   = cmhCC_SelStopBit( srcId );
+        pfilled_ctb->BC[1].data_bits   = cmhCC_SelDataBit( srcId );
+        pfilled_ctb->BC[1].parity      = cmhCC_SelParity ( srcId );
+        break;
+
+      case( CMOD_MOD_VoiceFlwdDat ):       /* voice followed by data */
+        pfilled_ctb->rptInd          = RI_SEQUENTIAL;
+
+        pfilled_ctb->BC[1].rate        = cmhCC_SelRate( ccShrdPrm.CBSTspeed );
+        pfilled_ctb->BC[1].bearer_serv = cmhCC_SelServ( ccShrdPrm.CBSTname );
+        pfilled_ctb->BC[1].conn_elem   = cmhCC_SelCE(   ccShrdPrm.CBSTce );
+        pfilled_ctb->BC[1].modem_type  = cmhCC_SelMT  ( ccShrdPrm.CBSTspeed );
+
+        pfilled_ctb->BC[1].stop_bits   = cmhCC_SelStopBit( srcId );
+        pfilled_ctb->BC[1].data_bits   = cmhCC_SelDataBit( srcId );
+        pfilled_ctb->BC[1].parity      = cmhCC_SelParity ( srcId );
+        break;
+
+      case( CMOD_MOD_VoiceFax ):           /* alternating voice/fax call */
+        pfilled_ctb->rptInd          = RI_CIRCULAR;
+
+        pfilled_ctb->BC[1].rate        = cmhCC_SelRate( ccShrdPrm.CBSTspeed );
+        pfilled_ctb->BC[1].bearer_serv = BEARER_SERV_FAX;
+        pfilled_ctb->BC[1].conn_elem   = CONN_ELEM_TRANS;
+        pfilled_ctb->BC[1].modem_type  = MT_NONE;
+        break;
+#endif
+      default:                             /* unexpected call mode */
+        ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_DataCorrupt );
+        return( AT_FAIL );
+      }
+      break;
+
+    /*
+     *-----------------------------------------------------------------
+     * start with a data call
+     *-----------------------------------------------------------------
+     */
+#ifdef FAX_AND_DATA
+
+      case( D_TOC_Data ):
+
+      switch( ccShrdPrm.CMODmode )
+      {
+      case( CMOD_MOD_Single ):             /* single data call */
+        pfilled_ctb->rptInd          = RI_NOT_PRES;
+
+        pfilled_ctb->BC[0].rate        = cmhCC_SelRate( ccShrdPrm.CBSTspeed );
+
+#ifdef FF_FAX
+        if( pT30CmdPrm -> FCLASSclass EQ FCLASS_CLASS_Fax20 )
+        {
+          pfilled_ctb->BC[0].bearer_serv = BEARER_SERV_FAX;
+          pfilled_ctb->BC[0].conn_elem   = CONN_ELEM_TRANS;
+          pfilled_ctb->BC[0].modem_type  = MT_NONE;
+        }
+        else
+#endif /* FF_FAX */
+        {
+          pfilled_ctb->BC[0].bearer_serv = cmhCC_SelServ( ccShrdPrm.CBSTname );
+          pfilled_ctb->BC[0].conn_elem   = cmhCC_SelCE(   ccShrdPrm.CBSTce );
+          pfilled_ctb->BC[0].modem_type  = cmhCC_SelMT  ( ccShrdPrm.CBSTspeed );
+
+          pfilled_ctb->BC[0].stop_bits   = cmhCC_SelStopBit( srcId );
+          pfilled_ctb->BC[0].data_bits   = cmhCC_SelDataBit( srcId );
+          pfilled_ctb->BC[0].parity      = cmhCC_SelParity ( srcId );
+        }
+
+        pfilled_ctb->BC[1].rate        = UR_NOT_PRES;
+        pfilled_ctb->BC[1].bearer_serv = BEARER_SERV_NOT_PRES;
+        pfilled_ctb->BC[1].conn_elem   = CONN_ELEM_NOT_PRES;
+        break;
+
+      case( CMOD_MOD_VoiceDat ):           /* alternating data/voice call */
+        pfilled_ctb->rptInd          = RI_CIRCULAR;
+
+        pfilled_ctb->BC[0].rate        = cmhCC_SelRate( ccShrdPrm.CBSTspeed );
+        pfilled_ctb->BC[0].bearer_serv = cmhCC_SelServ( ccShrdPrm.CBSTname );
+        pfilled_ctb->BC[0].conn_elem   = cmhCC_SelCE(   ccShrdPrm.CBSTce );
+        pfilled_ctb->BC[0].modem_type  = cmhCC_SelMT  ( ccShrdPrm.CBSTspeed );
+
+        pfilled_ctb->BC[0].stop_bits   = cmhCC_SelStopBit( srcId );
+        pfilled_ctb->BC[0].data_bits   = cmhCC_SelDataBit( srcId );
+        pfilled_ctb->BC[0].parity      = cmhCC_SelParity ( srcId );
+
+        pfilled_ctb->BC[1].rate        = UR_NOT_PRES;
+        pfilled_ctb->BC[1].bearer_serv = cmhCC_set_speech_serv (pCCCmdPrm);
+        pfilled_ctb->BC[1].conn_elem   = CONN_ELEM_NOT_PRES;
+        break;
+
+      case( CMOD_MOD_VoiceFlwdDat ):       /* voice followed by data call */
+
+        ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter );
+        return( AT_FAIL );                 /* operation not allowed */
+
+      case( CMOD_MOD_VoiceFax ):           /* alternating fax/voice call */
+        pfilled_ctb->rptInd          = RI_CIRCULAR;
+
+        pfilled_ctb->BC[0].rate        = cmhCC_SelRate( ccShrdPrm.CBSTspeed );
+        pfilled_ctb->BC[0].bearer_serv = BEARER_SERV_FAX;
+        pfilled_ctb->BC[0].conn_elem   = CONN_ELEM_TRANS;
+        pfilled_ctb->BC[0].modem_type  = MT_NONE;
+
+        pfilled_ctb->BC[1].rate        = UR_NOT_PRES;
+        pfilled_ctb->BC[1].bearer_serv = cmhCC_set_speech_serv (pCCCmdPrm);
+        pfilled_ctb->BC[1].conn_elem   = CONN_ELEM_NOT_PRES;
+        break;
+
+      default:                             /* unexpected call mode */
+        ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_DataCorrupt );
+        return( AT_FAIL );
+      }
+      break;
+
+#endif    /* of #ifdef FAX_AND_DATA */
+
+    default:                               /* unexpected call type */
+      ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_DataCorrupt );
+      return( AT_FAIL );
+  }
+
+  return( AT_CMPL );
+}
+
+#ifdef FF_TTY
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                  |
+| STATE   : code                  ROUTINE : cmhCC_notifyTTY          |
++--------------------------------------------------------------------+
+
+  PURPOSE : This function converts the TTY activity to the trx state
+            to be provided by '%CTYI: ...'.
+*/
+
+GLOBAL T_ACI_CTTY_TRX cmhCC_getTTYtrx_state (int ttyTrxState)
+{
+  switch (ttyTrxState)
+  {
+  case TTY_OFF:
+    return CTTY_TRX_Off;
+  case TTY_HCO:
+    return CTTY_TRX_SendOn;
+  case TTY_VCO:
+    return CTTY_TRX_RcvOn;
+  case TTY_ALL:
+    return CTTY_TRX_RcvSendOn;
+  default:
+    break;
+  }
+  return CTTY_TRX_Unknown;
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                  |
+| STATE   : code                  ROUTINE : cmhCC_notifyTTY          |
++--------------------------------------------------------------------+
+
+  PURPOSE : This function generates the unsolicited message '%CTYI: ...'
+            for any logical channel having registered for.
+*/
+
+GLOBAL void cmhCC_notifyTTY (T_ACI_CTTY_NEG neg,
+                             T_ACI_CTTY_TRX trx)
+{
+  int i;
+
+  for (i = CMD_SRC_LCL; i < CMD_SRC_MAX; i++)
+  {
+    if (cmhPrm[i].ccCmdPrm.CTTYmode EQ CTTY_MOD_Enable)
+    {
+      R_AT (RAT_CTYI, i) (neg, trx);
+    }
+  }
+}
+#endif /* FF_TTY */
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                  |
+| STATE   : code                  ROUTINE : cmhCC_checkALS_Support   |
++--------------------------------------------------------------------+
+
+  PURPOSE : These function checks for ALS support in two steps
+*/
+
+GLOBAL void cmhCC_checkALS_Support()
+{
+  TRACE_FUNCTION ("cmhCC_checkALS_Support()");
+
+  /* Step #1: read CPHS info first */
+  cmhSIM_ReadTranspEF( CMD_SRC_NONE,
+                       AT_CMD_NONE,
+                       SIM_CPHS_CINF,
+                       0,
+                       ACI_CPHS_INFO_SIZE,
+                       NULL,
+                       cmhCC_checkALS_Support_cb );
+}
+
+GLOBAL void cmhCC_checkALS_Support_2()
+{
+  TRACE_FUNCTION ("cmhCC_checkALS_Support_2()");
+
+  /* Step #2: then read the actual CSP */
+  cmhSIM_ReadTranspEF( CMD_SRC_NONE,
+                       AT_CMD_NONE,
+                       SIM_CPHS_CSP,
+                       0,
+                       0,
+                       NULL,
+                       cmhCC_checkALS_Support_cb_2 );
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                  |
+| STATE   : code                  ROUTINE : cmhCC_redialCheck        |
++--------------------------------------------------------------------+
+
+  PURPOSE : This function evaluates conditions for redialling
+*/
+
+GLOBAL void cmhCC_redialCheck(SHORT cId)
+{
+  T_CC_CALL_TBL *ctbFail = ccShrdPrm.ctb[ccShrdPrm.cIdFail];
+  T_CC_CALL_TBL *ctb     = ccShrdPrm.ctb[cId];
+  SHORT error_indication;
+  int i;
+
+  TRACE_FUNCTION ("cmhCC_redialCheck()");
+
+  /* check conditions for error_indication check */
+  if((rdlPrm.rdlMod EQ AUTOM_REPEAT_ON) AND (rdlPrm.rdlcId EQ NO_ENTRY))
+  {/* if black list is full no further redial attempts; user has to delete the black list */
+    if(cmhCC_redialChkBlackl(cId) EQ AT_CMPL)
+    {
+      if(cc_blacklist_ptr->blCount NEQ MAX_NUM_ENTR_BLACK_LIST)
+      {
+        if( ccShrdPrm.cIdFail < 0 OR ccShrdPrm.cIdFail > MAX_CALL_NR )
+        {
+          error_indication = GET_CAUSE_VALUE(CAUSE_MAKE(DEFBY_STD,
+                                                        ORIGSIDE_MS, 
+                                                        ACI_ORIGINATING_ENTITY,
+                                                        NOT_PRESENT_8BIT));
+        }
+        else if( GET_CAUSE_VALUE(ctbFail->rejCs) NEQ NOT_PRESENT_8BIT )
+        {
+         error_indication = GET_CAUSE_VALUE(ctbFail->rejCs);
+        }
+        else if( GET_CAUSE_VALUE(ctbFail->nrmCs) NEQ NOT_PRESENT_8BIT )
+        {
+          error_indication = GET_CAUSE_VALUE(ctbFail->nrmCs);
+        }
+        else if( GET_CAUSE_VALUE(ctbFail->rslt) NEQ NOT_PRESENT_8BIT )
+        {
+          error_indication = GET_CAUSE_VALUE(ctbFail->rslt);
+        }
+        else
+        {/* in case network has sent no extended report */
+          error_indication = GET_CAUSE_VALUE(CAUSE_MAKE(DEFBY_STD,
+                                                        ORIGSIDE_MS,
+                                                        ACI_ORIGINATING_ENTITY,
+                                                        NOT_PRESENT_8BIT));
+        }
+
+        /* check error_indication (ETSI 2.07, Annex A) and redial counter */
+        switch (error_indication)
+        {
+          /*** indications of category 1 ***/
+          case 17:            /* user busy */
+          /*** indications of category 2 ***/
+          case 18:            /* no user responding */
+          case 19:            /* user alterting, no answer */
+          case 27:            /* destination out of order */
+          case 34:            /* no circuit/channel available */
+          case 41:            /* temporary failure */
+          case 42:            /* switching equipment congestion */
+          case 44:            /* requested circuit/channel available */
+          case 47:            /*esources unavailable, unspecified */
+
+            if(!ctb->rdlCnt)
+            {/* first redial attempt, 5+5 sec for first call attempt */
+              rdlPrm.rdlcId = cId;
+              TIMERSTART(ACI_REPEAT_1+ACI_DELAY_TIMER,ACI_REPEAT_HND);
+              for(i = 0; i < CMD_SRC_MAX; i++)
+              {
+                R_AT(RAT_RDL, i)(CALL_ATTEMPT_FAILED);
+                R_AT(RAT_RDL, i)(REDIAL_TIM_START);
+              }
+            }
+            else if((ctb->rdlCnt>=1) AND (ctb->rdlCnt<=3))
+            {/* from 2nd to 4th redial attempt, 1 min + 5 sec for the following call attempt */
+              rdlPrm.rdlcId = cId;
+   #ifdef _SIMULATION_
+              TIMERSTART(ACI_REPEAT_1,ACI_REPEAT_HND);
+   #else
+              TIMERSTART(ACI_REPEAT_2_4+ACI_DELAY_TIMER,ACI_REPEAT_HND);
+   #endif
+              for(i = 0; i < CMD_SRC_MAX; i++)
+              {
+                R_AT(RAT_RDL, i)(CALL_ATTEMPT_FAILED);
+                R_AT(RAT_RDL, i)(REDIAL_TIM_START);
+              }
+            }
+            else if((ctb->rdlCnt >= 4) AND (ctb->rdlCnt < MAX_NUM_REPEAT_ATT))
+            {/* from 5th to 10th redial attempt, 3 x 1 min + 5 sec for the following call attempt */
+              rdlPrm.rdlcId = cId;
+  #ifdef _SIMULATION_
+              TIMERSTART(ACI_REPEAT_1,ACI_REPEAT_HND);
+  #else
+              ctb->rdlTimIndex = 1;
+              TIMERSTART(ACI_REPEAT_2_4+ACI_DELAY_TIMER,ACI_REPEAT_HND);
+  #endif
+              for(i = 0; i < CMD_SRC_MAX; i++)
+              {
+                R_AT(RAT_RDL, i)(CALL_ATTEMPT_FAILED);
+                R_AT(RAT_RDL, i)(REDIAL_TIM_START);
+              }
+            }
+            /* else : ctb->rdlCnt = MAX_NUM_REPEAT_ATT -> entry is black list */
+            break;
+          /*** indications of category 3 -> only one redial attempt is allowed ***/
+          case 28:            /* invalid number format */
+            ctb->rdlCnt = 0;
+            break;            /* for incorrect number we do not start redialling */
+          case 1:             /* unassigned (unallocated) number */
+          case 3:             /* no route to destination */
+          case 22:            /* number changed */
+          case 38:            /* network out of order */
+            if(!ctb->rdlCnt)
+            {/* first redial attempt */
+              ctb->rdlCnt = MAX_NUM_REPEAT_ATT - 1;
+              rdlPrm.rdlcId = cId;
+              /* 5+5 sec for first (= only) call attempt */
+              TIMERSTART(ACI_REPEAT_1+ACI_DELAY_TIMER,ACI_REPEAT_HND);
+              for(i = 0; i < CMD_SRC_MAX; i++)
+              {
+                R_AT(RAT_RDL, i)(CALL_ATTEMPT_FAILED);
+                R_AT(RAT_RDL, i)(REDIAL_TIM_START);
+              }
+            }
+            else
+            {/* more than one attempt is not allowed --> entry in blacklist */
+              ctb->rdlCnt = MAX_NUM_REPEAT_ATT;
+              for(i = 0; i < CMD_SRC_MAX; i++)
+              {
+                R_AT(RAT_RDL, i)(CALL_ATTEMPT_FAILED);
+              }
+             }
+             break;
+          default:
+            ctb->rdlCnt = 0;
+            break;
+        }
+      }
+
+      /* check redial counter against black list: add entry if the list is not full yet */
+      if((ctb->rdlCnt EQ MAX_NUM_REPEAT_ATT) AND
+        (cc_blacklist_ptr->blCount< MAX_NUM_ENTR_BLACK_LIST))
+      {/* add entry in black list */
+        cc_blacklist_ptr->blCount++;
+        cc_blacklist_ptr->blNum[cc_blacklist_ptr->blCount-1].numb_len = 
+          ctb->cldPty.c_called_num;
+        memcpy(cc_blacklist_ptr->blNum[cc_blacklist_ptr->blCount-1].number, 
+               ctb->cldPty.called_num,
+               ctb->cldPty.c_called_num);
+        cc_blacklist_ptr->blNum[cc_blacklist_ptr->blCount-1].type.ton =
+          ctb->cldPty.ton;
+        cc_blacklist_ptr->blNum[cc_blacklist_ptr->blCount-1].type.npi =
+          ctb->cldPty.npi;
+
+        ctb->rdlCnt = 0; /* reset redial counter in call table */
+        ctb->rdlTimIndex = RDL_TIM_INDEX_NOT_PRESENT;
+        if(cc_blacklist_ptr->blCount EQ MAX_NUM_ENTR_BLACK_LIST)
+        {
+          for(i = 0; i < CMD_SRC_MAX; i++)
+          {
+            R_AT(RAT_RDLB, i)(BLACKLIST_FULL);
+          }
+        }
+        else
+        {
+          for(i = 0; i < CMD_SRC_MAX; i++)
+          {
+            R_AT(RAT_RDLB, i)(ENTRY_BLACKLISTED);
+          }
+        }
+      }
+    }
+  }
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                  |
+| STATE   : code                  ROUTINE : cmhCC_redialChkBlackl    |
++--------------------------------------------------------------------+
+
+  PURPOSE : This function checks the new cId against the black list
+*/
+
+GLOBAL T_ACI_RETURN cmhCC_redialChkBlackl(SHORT cId)
+{
+  UBYTE bl_index = 0;
+  int i;
+
+  TRACE_FUNCTION ("cmhCC_redialChkBlackl()");
+
+  if(cc_blacklist_ptr NEQ NULL)
+  {
+    for(bl_index=0; bl_index<cc_blacklist_ptr->blCount;bl_index++)
+    {
+      T_CC_CALL_TBL *ctb = ccShrdPrm.ctb[cId];
+
+      if (memcmp (cc_blacklist_ptr->blNum[bl_index].number,
+                  ctb->cldPty.called_num,
+                  ctb->cldPty.c_called_num) EQ 0)
+      {
+        for(i = 0; i < CMD_SRC_MAX; i++)
+        {
+          R_AT(RAT_RDLB, i)(ENTRY_IN_BLACKLIST);
+        }
+        return( AT_FAIL );
+      }
+    }
+  }
+  return( AT_CMPL );
+}
+#if defined(_TARGET_)
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)         MODULE  : CMH_CCF                  |
+| STATE   : code                  ROUTINE : cmhCC_rd_mode_FFS        |
++--------------------------------------------------------------------+
+
+  PURPOSE : This function writes redial_mode in FFS
+*/
+
+GLOBAL T_ACI_RETURN cmhCC_rd_mode_FFS(T_ACI_CC_REDIAL_MODE rdl_mode,
+                                      T_ACI_CC_RDL_FFS ffs_mode)
+{
+  T_ACI_CC_REDIAL_MODE redial_mode;
+
+  TRACE_FUNCTION ("cmhCC_rd_mode_FFS()");
+
+  switch(FFS_mkdir("/gsm/com"))
+  {/* create/check ffs directory */
+    case EFFS_OK:
+    case EFFS_EXISTS:
+      break;
+    default:
+    ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter );
+    return ( AT_FAIL );
+  }
+
+  if(ffs_mode EQ WRITE_RDLmode)
+  {
+    if(FFS_fwrite("/gsm/com/redialMode",&rdl_mode, sizeof(T_ACI_CC_REDIAL_MODE)) NEQ EFFS_OK)
+    {
+      ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter );
+      return ( AT_FAIL );
+    }
+    return AT_CMPL;
+  }
+  else if(ffs_mode EQ READ_RDLmode)
+  {
+    FFS_fread("/gsm/com/redialMode",&redial_mode, sizeof(T_ACI_CC_REDIAL_MODE));
+    switch(redial_mode)
+    {
+      case AUTOM_REP_NOT_PRESENT:
+        rdlPrm.rdlMod = AUTOM_REPEAT_OFF;
+        return AT_CMPL;;
+      case AUTOM_REPEAT_OFF:
+      case AUTOM_REPEAT_ON:
+        rdlPrm.rdlMod = redial_mode;
+        return AT_CMPL;;
+      default:
+       ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter );
+       return ( AT_FAIL );
+    }
+  }
+  else
+  {
+    ACI_ERR_DESC( ACI_ERR_CLASS_Ext, EXT_ERR_Parameter );
+    return ( AT_FAIL );
+  }
+}
+#endif /* _TARGET_ */
+/*==== EOF ========================================================*/
+