diff src/g23m-gsm/cc/cc_rel.c @ 104:27a4235405c6

src/g23m-gsm: import from LoCosto source
author Mychaela Falconia <falcon@freecalypso.org>
date Tue, 04 Oct 2016 18:24:05 +0000
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/g23m-gsm/cc/cc_rel.c	Tue Oct 04 18:24:05 2016 +0000
@@ -0,0 +1,1224 @@
+/* 
++----------------------------------------------------------------------------- 
+|  Project :  GSM-PS (6147)
+|  Modul   :  CC_REL
++----------------------------------------------------------------------------- 
+|  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 defines the functions for call release
+|             of the component CC of the mobile station.
++----------------------------------------------------------------------------- 
+*/ 
+
+#ifndef CC_REL_C
+#define CC_REL_C
+
+#define ENTITY_CC
+/*==== INCLUDES ===================================================*/
+
+#include <string.h>
+#include "typedefs.h"
+#include "pcm.h"
+#include "vsi.h"
+#include "custom.h"
+#include "gsm.h"
+#include "message.h"
+#include "ccdapi.h"
+#include "prim.h"
+#include "cnf_cc.h"
+#include "mon_cc.h"
+#include "pei.h"
+#include "tok.h"
+#include "cc.h"
+#include "cc_em.h"
+
+/*==== EXPORT =====================================================*/
+
+/*==== PROTOTYPES =================================================*/
+/* Implements Measure#  8 */
+LOCAL void cc_mncc_release_ind  (T_PRIM * prim);
+/* Implements Measure#  9 */
+LOCAL void cc_mncc_release_cnf (T_PRIM * prim);
+/* Implements Measure#  41 */
+LOCAL void cc_mncc_sync_ind ( UBYTE ti, 
+                              UBYTE new_state);
+/*==== PRIVAT =====================================================*/
+
+/*==== VARIABLES ==================================================*/
+
+/*==== FUNCTIONS ==================================================*/
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_check_setup_reattempt   |
++--------------------------------------------------------------------+
+
+  PURPOSE : This function checks whether a the mobile originated setup
+            may be reattempted. This will be done for some certain
+            reasons which may indicate network failure.
+            This function is considered as a hack only
+            for networks which tend to deliver a poor quality
+            of service only (e.g. typically in China). 
+            This behaviour or the algorithm is not specified / implied 
+            by GSM but driven by customer requests and the desire
+            to have a good, working product.
+*/
+
+LOCAL void cc_check_setup_reattempt ( const T_M_CC_cc_cause *cc_cause)
+{
+  GET_INSTANCE_DATA;
+  EXTERN BOOL poor_quality_network;
+
+  TRACE_FUNCTION ("cc_check_setup_reattempt()");
+
+#ifdef  WIN32
+  /*
+   * Set poor_quality_network for test, pei_config() is the appropriate place,
+   * but this seems not to be working for some strange reasons
+   */
+  poor_quality_network = TRUE;
+#endif
+
+  /* reattempt already "running", see cc_setup_reattempt for further reasoning */
+  if (cc_data->setup_reattempt_ti NEQ NOT_PRESENT_8BIT)
+    return;
+
+  switch (cc_data->state[cc_data->index_ti])
+  {
+    case M_CC_CS_1:  /* call initiated                     */
+    case M_CC_CS_3:  /* mobile originating call proceeding */
+      cc_data->setup_attempts++;
+      if (poor_quality_network AND
+          (cc_data->setup_attempts < MAX_SETUP_ATTEMPTS) AND
+          (cc_cause->v_cause))
+      {
+        /*
+         * Mobile originated call got disconnect,
+         * max. number of setup attempts not reached.
+         * New dial attempt for certain disconnect reasons
+         */
+        switch (cc_cause->cause)
+        {
+          case M_CC_CAUSE_NO_CHAN_AVAIL:       /* #34  */
+          case M_CC_CAUSE_NETWORK_ORDER:       /* #38  */
+          case M_CC_CAUSE_TEMP_FAIL:           /* #41  */
+          case M_CC_CAUSE_SWITCH_CONGEST:      /* #42  */
+          case M_CC_CAUSE_INFO_DISCARD:        /* #43  */
+          case M_CC_CAUSE_REQ_CHAN_UNAVAIL:    /* #44  */
+          case M_CC_CAUSE_RESOURCE_UNAVAIL:    /* #47  */
+          case M_CC_CAUSE_QOS_UNAVAIL:         /* #49  */
+          case M_CC_CAUSE_BEARER_CAP_UNAVAIL:  /* #58  */
+          case M_CC_CAUSE_TIMER:               /* #102 */
+          case M_CC_CAUSE_INTERWORKING:        /* #127 */
+            /*
+             * Maybe some more causes to added here
+             * which indicate network failure
+             */
+
+            /* try setup reattempt, remember this by setting setup_reattempt_ti */
+            cc_data->setup_reattempt_ti = cc_data->ti;
+            break;
+
+          default: /* Don't try setup reattempt, no appropriate cause */
+            srv_free_stored_setup ();
+            break;
+        }
+      }
+      break;
+
+    default: /* Don't try setup reattempt in other states */
+      break;
+  }
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_setup_reattempt         |
++--------------------------------------------------------------------+
+
+  PURPOSE : This function reattempts the mobile originated setup
+            for some certain reasons which may indicate network
+            failure. This function is considered as a hack only
+            for networks which tend to deliver a poor quality
+            of service only.
+
+*/
+
+LOCAL void cc_setup_reattempt (void)
+{
+  GET_INSTANCE_DATA;
+  TRACE_FUNCTION ("cc_setup_reattempt()");
+
+  if (cc_data->ti EQ cc_data->setup_reattempt_ti)
+  {
+    PALLOC (est, MMCM_ESTABLISH_REQ); /* T_MMCM_ESTABLISH_REQ */
+    est->ti    = cc_data->ti;
+    est->estcs = cc_data->estcs;
+    for_est_req (est);
+
+    cc_data->setup_reattempt_ti = NOT_PRESENT_8BIT;
+    cc_data->progress_desc[cc_data->index_ti] = NOT_PRESENT_8BIT;
+    cc_set_state (M_CC_CS_01);
+    TIMERSTART (T303, T303_VALUE);
+  }
+}
+
+/*==== VARIABLES ==================================================*/
+
+/*==== FUNCTIONS ==================================================*/
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_mncc_disconnect_req     |
++--------------------------------------------------------------------+
+
+  PURPOSE : Disconnection of call by the mobile side.
+
+*/
+
+GLOBAL void cc_mncc_disconnect_req (T_MNCC_DISCONNECT_REQ * disc)
+{
+  GET_INSTANCE_DATA;
+  TRACE_FUNCTION ("cc_mncc_disconnect_req()");
+
+  if ((cc_data->index_ti = srv_convert_ti (disc->ti))
+       EQ NOT_PRESENT_8BIT)
+  {
+    PFREE (disc);
+    return;
+  }
+  if (cc_data->ti EQ cc_data->setup_reattempt_ti)
+  {
+    srv_free_stored_setup ();  
+  }
+  switch (cc_data->state[cc_data->index_ti])
+  {
+    case M_CC_CS_03: /* WAIT FOR NW. INFO */
+    case M_CC_CS_05: /* CC ESTABLISHMENT CONFIRMED */
+    case M_CC_CS_06: /* RECALL PRESENT */
+      /* Forget stored CCBS setup */
+      if (cc_data->stored_ccbs_setup NEQ NULL)
+      {
+        PFREE (cc_data->stored_ccbs_setup);
+        cc_data->stored_ccbs_setup = NULL;
+      }
+
+      /* Send a (faked) MNCC_RELEASE_IND to MMI */
+      {
+        PALLOC (rel, MNCC_RELEASE_IND);
+        rel->ti    = disc->ti;
+        rel->cause = disc->cause;
+        rel->c_raw_cause = 0;
+        PSENDX (MMI, rel);
+      }
+
+      /* Send RELEASE COMPLETE to network */
+      CCD_START;
+      {
+        MCAST (rel_com, U_RELEASE_COMP);
+        cc_build_release_complete (rel_com, disc->cause);
+        for_release_complete (rel_com);
+      }
+      CCD_END;
+
+      /* Release MM connection */
+      for_rel_req ();
+
+      PFREE (disc);
+
+      /* Stop all running timers */
+      TIMERSTOP (TIMER_CC);
+
+      /* Next state is NULL */
+      cc_set_state (M_CC_CS_0);
+      break;
+
+    case M_CC_CS_01:
+    case CS_101:
+    case CS_261:
+      {
+        TIMERSTOP (TIMER_CC);
+
+        cc_reset_dtmf ();
+
+        {
+          PALLOC (rel, MNCC_RELEASE_IND);
+          rel->ti    = disc->ti;
+          rel->cause = disc->cause;
+          rel->c_raw_cause = 0;
+          PSENDX (MMI, rel);
+        }
+
+        for_rel_req ();
+        cc_set_state (M_CC_CS_0);
+        srv_use_stored_prim ();
+        PFREE (disc);
+      }
+      break;
+
+    case M_CC_CS_1:      
+    case M_CC_CS_3:
+    case M_CC_CS_4:
+    case M_CC_CS_7:
+    case M_CC_CS_8:
+    case M_CC_CS_9:
+    case M_CC_CS_10:
+    case M_CC_CS_26:
+      TIMERSTOP (TIMER_CC);
+
+      CCD_START;
+      {
+        MCAST (disconnect, U_DISCONNECT);
+        
+        cc_build_disconnect (disconnect, disc->cause,
+                             &disc->fac_inf, disc->ss_version);
+        cc_set_state (M_CC_CS_11);
+        for_disconnect (disconnect);
+      }
+      CCD_END;
+      TIMERSTART (T305, T305_VALUE);
+      PFREE (disc);
+      break;
+
+    case M_CC_CS_12: /* DISCONNECT INDICATION, this is clear collision here */
+      CCD_START;
+      {
+        MCAST (release, U_RELEASE);
+
+        cc_build_release (release, disc->cause,
+                          NULL, MNCC_SS_VER_NOT_PRES);
+        cc_set_state (M_CC_CS_19);
+        for_release (release);
+
+      }
+      CCD_END;
+      TIMERSTART (T308, T308_VALUE);
+      PFREE (disc);
+      break;
+
+    default:
+      PFREE (disc);
+      break;
+
+  }
+}
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_mncc_reject_req         |
++--------------------------------------------------------------------+
+
+  PURPOSE : Release of call by the mobile side by sending
+            a RELEASE COMPLETE message.
+
+*/
+
+GLOBAL void cc_mncc_reject_req (T_MNCC_REJECT_REQ * rej)
+{
+  GET_INSTANCE_DATA;
+
+  TRACE_FUNCTION ("cc_mncc_reject_req");
+
+  if ((cc_data->index_ti = srv_convert_ti (rej->ti))
+      EQ NOT_PRESENT_8BIT)
+  {
+    PFREE (rej);
+    return;
+  }
+
+  if (cc_data->ti EQ cc_data->setup_reattempt_ti)
+  {
+    srv_free_stored_setup ();
+  }
+
+  switch (cc_data->state[cc_data->index_ti])
+  {
+    case M_CC_CS_03: /* WAIT FOR NW. INFO */
+    case M_CC_CS_05: /* CC ESTABLISHMENT CONFIRMED */
+    case M_CC_CS_06: /* RECALL PRESENT */
+      /* Forget stored CCBS setup */
+      if (cc_data->stored_ccbs_setup NEQ NULL)
+      {
+        PFREE (cc_data->stored_ccbs_setup);
+        cc_data->stored_ccbs_setup = NULL;
+      }
+
+      /* Send RELEASE COMPLETE */
+      CCD_START;
+      {
+        MCAST (rel_com, U_RELEASE_COMP);
+        cc_build_release_complete (rel_com, rej->cause);
+        for_release_complete (rel_com);
+      }
+      CCD_END;
+
+      /* Release MM connection */
+      for_rel_req ();
+
+      PFREE (rej);
+
+      /* Stop all running timers */
+      TIMERSTOP (TIMER_CC);
+
+      /* Next state is NULL */
+      cc_set_state (M_CC_CS_0);
+      break;
+    default:
+      PFREE (rej);
+      break;
+  }
+}
+
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_mncc_release_req        |
++--------------------------------------------------------------------+
+
+  PURPOSE : Release of call by the mobile side.
+
+*/
+GLOBAL void cc_mncc_release_req (T_MNCC_RELEASE_REQ * rel)
+{
+  GET_INSTANCE_DATA;
+
+  TRACE_FUNCTION ("cc_mncc_release_req()");
+
+  if ((cc_data->index_ti = srv_convert_ti (rel->ti))
+      EQ NOT_PRESENT_8BIT)
+  {
+    PFREE (rel);
+    return;
+  }
+
+  if (cc_data->ti EQ cc_data->setup_reattempt_ti)
+  {
+    srv_free_stored_setup ();
+  }
+
+  switch (cc_data->state[cc_data->index_ti])
+  {
+    case M_CC_CS_12: /* DISCONNECT INDICATION */
+
+      CCD_START;
+      {
+        MCAST (release, U_RELEASE);
+
+        cc_build_release (release, rel->cause, 
+                          &rel->fac_inf, rel->ss_version);
+        cc_set_state (M_CC_CS_19);
+        for_release (release);
+
+      }
+      CCD_END;
+      TIMERSTART (T308, T308_VALUE);
+      PFREE (rel);
+      break;
+
+    default:
+      PFREE (rel);
+      break;
+
+  }
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_disconnect              |
++--------------------------------------------------------------------+
+
+  PURPOSE : Processing an incoming disconnect message.
+
+*/
+
+GLOBAL void cc_disconnect (T_D_DISCONNECT * disconnect)
+{
+  GET_INSTANCE_DATA;
+
+  TRACE_FUNCTION ("cc_disconnect()");
+
+  /* The cause is in DISCONNECT a mandatory IE (given intact message) */
+  TRACE_EVENT_P1 ("DISCONNECT cause: %02x", disconnect->cc_cause.cause);
+
+  EM_CC_DISCONNECT_RECEIVED;
+
+  switch (cc_data->state[cc_data->index_ti])
+  {
+    case M_CC_CS_11: /* DISCONNECT REQUEST, this is clear collision here */
+      CCD_END;
+      CCD_START;
+      {
+        MCAST (release, U_RELEASE);
+
+        TIMERSTOP (TIMER_CC);
+        /* PATCH LE 10.04.00
+         * set disconnect collision flag
+         */ 
+        cc_data->disc_coll [cc_data->index_ti]    = TRUE;
+        /* END PATCH LE 10.04.00 */
+
+        switch (cc_data->error)
+        {
+          case M_CC_CAUSE_INVALID_MAND_INFO:
+          case M_CC_CAUSE_COND_INFO_ELEM:
+            cc_build_release (release, CAUSE_MAKE(DEFBY_STD, 
+                                                  ORIGSIDE_MS, 
+                                                  MNCC_CC_ORIGINATING_ENTITY,
+                                                  cc_data->error),
+                              NULL, MNCC_SS_VER_NOT_PRES);
+            cc_set_state (M_CC_CS_19);
+            for_release (release);
+            TIMERSTART (T308, T308_VALUE);
+            break;
+
+          default:
+            if (disconnect->v_progress) 
+            {
+              cc_data->progress_desc[cc_data->index_ti] = 
+                disconnect->progress.progress_desc;
+            }
+            cc_build_release (release, CAUSE_MAKE(DEFBY_CONDAT, 
+                                                  ORIGSIDE_MS, 
+                                                  MNCC_CC_ORIGINATING_ENTITY,
+                                                  NOT_PRESENT_8BIT),
+                              NULL, MNCC_SS_VER_NOT_PRES);
+            cc_set_state (M_CC_CS_19);
+            for_release (release);
+            TIMERSTART (T308, T308_VALUE);
+            break;
+        }
+      }
+      CCD_END;
+      break;
+
+    case M_CC_CS_12: /* DISCONNECT INDICATION */
+    case M_CC_CS_19: /* RELEASE REQUEST */
+      /* NULL (M_CC_CS_0) is handled by the formatter */
+      CCD_END;
+/* Implements Measure#  3 and streamline encoding */
+      cc_send_status (M_CC_CAUSE_MESSAGE_TYPE_INCOMPAT);
+      break;
+
+    default:
+      /* any "normal" state */
+      TIMERSTOP (TIMER_CC);
+
+      switch (cc_data->error)
+      {
+        case M_CC_CAUSE_INVALID_MAND_INFO:
+        case M_CC_CAUSE_COND_INFO_ELEM:
+          CCD_END;
+          CCD_START;
+          {
+            USHORT curr_cause; /* local variable to avoid repeated CAUSE_MAKEs for the same cause */
+
+            MCAST (release, U_RELEASE);
+            PALLOC (disc_ind, MNCC_DISCONNECT_IND);
+
+            disc_ind->ti    = cc_data->ti;
+            curr_cause = CAUSE_MAKE(DEFBY_STD, 
+                                    ORIGSIDE_MS, 
+                                    MNCC_CC_ORIGINATING_ENTITY,
+                                    cc_data->error);
+            disc_ind->cause = curr_cause; 
+            /* Setting raw_cause to empty as this is a local release
+             * of MM connection,CC is not receiving any cause value
+             * from Network
+             */
+            disc_ind->c_raw_cause = 0;
+
+            disc_ind->diagnostic = NOT_PRESENT_8BIT;
+            disc_ind->progress_desc = MNCC_PROG_NOT_PRES;
+            /* CQ 23619: get the ss diagnostic >> */
+            disc_ind->ss_diag = MNCC_SS_DIAG_NOT_PROVIDED;
+            /* CQ 23619 << */
+            PSENDX (MMI, disc_ind);
+            
+            /* Patch HM 29-Jan-02 >>> */
+            cc_build_facility_ind (MNCC_FAC_IN_DISCONNECT,
+                                   disconnect->v_facility, &disconnect->facility);
+            /* Patch HM 29-Jan-02 <<< */
+            cc_build_release (release, cc_data->error,
+                              NULL, MNCC_SS_VER_NOT_PRES);
+            cc_set_state (M_CC_CS_19);
+            for_release (release);
+            TIMERSTART (T308, T308_VALUE);
+          }
+          CCD_END;
+        break;
+
+        default:
+          if (disconnect->v_progress) 
+          {
+            cc_data->progress_desc[cc_data->index_ti] = 
+              disconnect->progress.progress_desc;
+          }
+          cc_check_setup_reattempt (&disconnect->cc_cause);
+
+          if (cc_data->ti NEQ cc_data->setup_reattempt_ti)
+          {
+            /*
+             * We don't perform setup reattempt, so send disconnect to MMI
+             */
+            USHORT curr_cause; /* local variable to avoid repeated CAUSE_MAKEs for the same cause */             
+            PALLOC (disc_ind, MNCC_DISCONNECT_IND); /* T_MNCC_DISCONNECT_IND */
+            disc_ind->ti    = cc_data->ti;
+            disc_ind->cause = curr_cause = CAUSE_MAKE(DEFBY_STD, 
+                                                      ORIGSIDE_NET, 
+                                                      MNCC_CC_ORIGINATING_ENTITY,
+                                                      disconnect->cc_cause.cause);
+            disc_ind->c_raw_cause = cc_build_cause (&disconnect->cc_cause,
+                                                     disc_ind->raw_cause);
+            disc_ind->diagnostic = NOT_PRESENT_8BIT;
+
+            if (disconnect->v_progress)
+              disc_ind->progress_desc = disconnect->progress.progress_desc;
+            else
+              disc_ind->progress_desc = MNCC_PROG_NOT_PRES;
+
+            /* Handle CCBS possible flag */
+            if ((cc_data->call_ctrl_cap.pcp NEQ 0) AND
+                (disconnect->v_allowed_actions NEQ 0) AND
+                (disconnect->allowed_actions.ccbs_act EQ M_CC_CCBS_YES))
+            {
+              /* CCBS indicated as possible by the network */
+              disc_ind->diagnostic = MNCC_DIAG_CCBS_POSSIBLE;
+            }
+            /* CQ 23619: get the ss diagnostic */
+            disc_ind->ss_diag = cc_get_ss_diag(curr_cause, disconnect);
+            /* CQ 23619 << */
+            PSENDX (MMI, disc_ind);
+            /* Patch HM 29-Jan-02 >>> */
+            cc_build_facility_ind (MNCC_FAC_IN_DISCONNECT,
+                                   disconnect->v_facility, &disconnect->facility);            
+            /* Patch HM 29-Jan-02 <<< */
+          }
+          if ((cc_data->ti NEQ cc_data->setup_reattempt_ti) AND
+              (((disconnect->v_progress AND
+                (disconnect->progress.progress_desc EQ M_CC_PROG_INBAND_AVAIL) AND
+                (cc_data->channel_mode != NAS_CHM_SIG_ONLY)))
+              OR
+              ((cc_data->call_ctrl_cap.pcp NEQ 0) AND
+               (disconnect->v_allowed_actions NEQ 0) AND
+               (disconnect->allowed_actions.ccbs_act EQ M_CC_CCBS_YES))))
+          {
+            /*
+             * Hook on tone is generated by the
+             * infrastructure and signalled via TCH OR CCBS is possible
+             */
+            CCD_END;
+            cc_set_state (M_CC_CS_12);
+          }
+          else
+          {
+            CCD_END;
+            CCD_START;
+            {
+              MCAST (release, U_RELEASE);
+
+              /*
+               * Hook on tone is generated internally by the mobile station
+               * and no CCBS possible, the connection is released
+               */
+              cc_build_release (release, CAUSE_MAKE(DEFBY_CONDAT, 
+                                                    ORIGSIDE_MS, 
+                                                    MNCC_CC_ORIGINATING_ENTITY,
+                                                    NOT_PRESENT_8BIT),
+                                NULL, MNCC_SS_VER_NOT_PRES);
+              cc_set_state (M_CC_CS_19);
+              for_release (release);
+              TIMERSTART (T308, T308_VALUE);
+            }
+            CCD_END;
+          }
+          break;
+      }
+      break;
+  }
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_release                 |
++--------------------------------------------------------------------+
+
+  PURPOSE : Processing an incoming release message.
+
+*/
+
+GLOBAL void cc_release (T_D_RELEASE * release)
+{
+  GET_INSTANCE_DATA;
+
+  TRACE_FUNCTION ("cc_release()");
+ 
+  if (release->cc_cause.v_cause)
+  {
+    /* The cause is in RELEASE an optional IE */
+    TRACE_EVENT_P1 ("RELEASE cause: %02x", release->cc_cause.cause);
+  }
+  else
+  {
+    TRACE_EVENT ("RELEASE cause: --");
+  }
+
+  switch (cc_data->state[cc_data->index_ti])
+  {
+    /*
+     * M_CC_CS_0 is handled by the formatter
+     */
+    case M_CC_CS_19: /* Release Request */
+    {
+      cc_reset_dtmf ();
+      /* PATCH LE 10.04.00
+       * in disconnect collision case send
+       * MNCC_RELEASE_IND instead of MNCC_RELEASE_CNF
+....   */
+      TIMERSTOP (TIMER_CC);
+      if (cc_data->disc_coll [cc_data->index_ti])
+      {
+/* Implements Measure#  8 */
+        cc_mncc_release_ind ((T_PRIM *)D2P(release));
+      }
+      else
+      {
+/* Implements Measure#  9 */
+        cc_mncc_release_cnf ((T_PRIM *)D2P(release));
+      }
+      // END PATCH LE 10.04.00     
+
+      for_rel_req ();
+      cc_set_state (M_CC_CS_0);
+      /* END PATCH LE 10.04.00 */
+      CCD_END;
+      break;
+    }
+
+    case M_CC_CS_1:  /* call initiated                     */
+    case M_CC_CS_3:  /* mobile originating call proceeding */ 
+      cc_check_setup_reattempt (&release->cc_cause);
+      /*FALLTHROUGH*/ /*lint -fallthrough*/
+    default:
+      cc_reset_dtmf (); /* China change HM 11.07.00 */
+      CCD_END;
+      CCD_START;
+      {
+        MCAST (rel_com, U_RELEASE_COMP);
+        
+        TIMERSTOP (TIMER_CC);
+        switch (cc_data->error)
+        {
+          case M_CC_CAUSE_INVALID_MAND_INFO:
+          case M_CC_CAUSE_COND_INFO_ELEM:
+            {
+              USHORT curr_cause; /* local variable to avoid repeated CAUSE_MAKEs for the same cause */
+              curr_cause = CAUSE_MAKE(DEFBY_STD, 
+                                      ORIGSIDE_MS, 
+                                      MNCC_CC_ORIGINATING_ENTITY,
+                                      cc_data->error);
+
+              if (cc_data->ti NEQ cc_data->setup_reattempt_ti)
+              {
+                PALLOC (rel_ind, MNCC_RELEASE_IND);
+                rel_ind->ti = cc_data->ti;
+                rel_ind->cause = curr_cause;
+                rel_ind->c_raw_cause = 0;
+                PSENDX (MMI, rel_ind);
+              }
+              cc_build_release_complete (rel_com, curr_cause);
+            }
+            break;
+
+          default:
+            if (cc_data->ti NEQ cc_data->setup_reattempt_ti)
+            {
+/* Implements Measure#  8 */
+              cc_mncc_release_ind ((T_PRIM *)D2P(release));
+            }       
+            cc_build_release_complete (rel_com, CAUSE_MAKE(DEFBY_CONDAT, 
+                                                           ORIGSIDE_MS, 
+                                                           MNCC_CC_ORIGINATING_ENTITY,
+                                                           NOT_PRESENT_8BIT));
+            break;
+        }
+
+        for_release_complete (rel_com);
+      }
+      CCD_END;
+      for_rel_req ();
+      cc_set_state (M_CC_CS_0);
+      cc_setup_reattempt ();
+      break;
+  }
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_release_complete        |
++--------------------------------------------------------------------+
+
+  PURPOSE : Processing an incoming release complete message.
+
+*/
+
+GLOBAL void cc_release_complete (T_D_RELEASE_COMP * rel_com)
+{
+  GET_INSTANCE_DATA;
+
+  TRACE_FUNCTION ("cc_release_complete()");
+
+  if (rel_com->cc_cause.v_cause)
+  {
+    /* The cause is in RELEASE COMPLETE an optional IE */
+    TRACE_EVENT_P1 ("RELEASE COMPLETE cause: %02x", rel_com->cc_cause.cause);
+  }
+  else
+  {
+    TRACE_EVENT ("RELEASE COMPLETE cause: --");
+  }
+
+  EM_CC_RELEASE_COMPLETE_RECEIVED;
+
+  switch (cc_data->state[cc_data->index_ti])
+  {
+    case M_CC_CS_0: /* NULL */
+      CCD_END;
+      for_rel_req ();
+      break;
+
+    case M_CC_CS_03: /* WAIT FOR NETWORK INFO*/
+      /* Stop all running CC timers */
+      TIMERSTOP (TIMER_CC);
+
+      /* Inform MMI */
+      {
+        PALLOC (release, MNCC_RELEASE_IND);
+        release->ti  = cc_data->ti;
+        if (rel_com->v_cc_cause AND rel_com->cc_cause.v_cause)
+        {
+          release->cause = CAUSE_MAKE(DEFBY_STD, 
+                                      ORIGSIDE_NET, 
+                                      MNCC_CC_ORIGINATING_ENTITY,
+                                      rel_com->cc_cause.cause);
+          release->c_raw_cause = cc_build_cause (&rel_com->cc_cause,
+                                                  release->raw_cause);
+        }
+        else
+        {
+          release->cause = CAUSE_MAKE(DEFBY_CONDAT, 
+                                      ORIGSIDE_NET, 
+                                      MNCC_CC_ORIGINATING_ENTITY,
+                                      NOT_PRESENT_8BIT);
+          release->c_raw_cause = 0;
+        }
+        PSENDX (MMI, release);
+      }
+
+      CCD_END;
+
+      /* Release MM connection */
+      for_rel_req ();
+
+      /* Next state is NULL */
+      cc_set_state (M_CC_CS_0);
+      break;
+
+    case M_CC_CS_05: /* CC ESTABLISHMENT CONFIRMED */
+    case M_CC_CS_06: /* RECALL PRESENT */
+      /* Forget stored CCBS setup */
+      PFREE (cc_data->stored_ccbs_setup);
+      cc_data->stored_ccbs_setup = NULL;
+
+      /* Stop all running CC timers */
+      TIMERSTOP (TIMER_CC);
+
+      /* Inform MMI */
+      {
+        PALLOC (release, MNCC_RELEASE_IND);
+        release->ti  = cc_data->ti;
+        if (rel_com->v_cc_cause AND rel_com->cc_cause.v_cause)
+        {
+          release->cause = CAUSE_MAKE(DEFBY_STD, 
+                                      ORIGSIDE_NET, 
+                                      MNCC_CC_ORIGINATING_ENTITY,
+                                      rel_com->cc_cause.cause);
+          release->c_raw_cause = cc_build_cause (&rel_com->cc_cause,
+                                                  release->raw_cause);
+        }
+        else
+        {
+          release->cause = CAUSE_MAKE(DEFBY_CONDAT, 
+                                      ORIGSIDE_NET, 
+                                      MNCC_CC_ORIGINATING_ENTITY,
+                                      NOT_PRESENT_8BIT);
+          release->c_raw_cause = 0;
+        }
+        PSENDX (MMI, release);
+      }
+
+      CCD_END;
+
+      /* Release MM connection */
+      for_rel_req ();
+
+      /* Next state is NULL */
+      cc_set_state (M_CC_CS_0);
+      break;
+
+    case M_CC_CS_19: /* RELEASE REQUEST */
+      cc_reset_dtmf ();
+      /* PATCH LE 10.04.00
+       * in disconnect collision case send
+       * MNCC_RELEASE_IND instead of MNCC_RELEASE_CNF
+       */
+      TIMERSTOP (TIMER_CC);
+
+      if (cc_data->ti NEQ cc_data->setup_reattempt_ti) 
+      {
+        /* Inform MMI only if no setup reattempt for this ti */
+        if (cc_data->disc_coll [cc_data->index_ti])
+        {
+/* Implements Measure#  9 */
+          cc_mncc_release_ind ((T_PRIM *)D2P(rel_com));
+        }
+        else
+        {
+/* Implements Measure#  9 */
+          cc_mncc_release_cnf ((T_PRIM *)D2P(rel_com));
+        }
+        /* END PATCH LE 10.04.00 */
+
+      }
+
+      for_rel_req ();
+      cc_set_state (M_CC_CS_0);
+      CCD_END;
+      cc_setup_reattempt ();
+      break;
+
+    case M_CC_CS_1:  /* call initiated                     */
+    case M_CC_CS_3:  /* mobile originating call proceeding */
+      cc_check_setup_reattempt (&rel_com->cc_cause);
+      /*FALLTHROUGH*/ /*lint -fallthrough*/
+    default:
+      cc_reset_dtmf (); /* China change HM 11.07.00 */
+      if (cc_data->ti NEQ cc_data->setup_reattempt_ti)
+      {
+        /* Implements Measure#  10 */
+        cc_mncc_release_ind ((T_PRIM *)D2P(rel_com));
+      }
+      for_rel_req ();
+      CCD_END;
+      cc_set_state (M_CC_CS_0);
+      cc_setup_reattempt ();
+      break;
+  }
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_rel_ind                 |
++--------------------------------------------------------------------+
+
+  PURPOSE : Processing of an incoming release indication from
+            mobility management.
+
+*/
+
+GLOBAL void cc_rel_ind (USHORT cause)
+{
+  GET_INSTANCE_DATA;
+
+  TRACE_FUNCTION ("cc_rel_ind()");
+
+  switch (cc_data->state[cc_data->index_ti])
+  {
+    case M_CC_CS_0:
+      break;
+
+    case M_CC_CS_05: /* CC ESTABLISHMENT CONFIRMED */
+    case M_CC_CS_06: /* RECALL PRESENT */
+      /* Stop all timers */
+      TIMERSTOP (TIMER_CC);
+
+      /* Forget stored CCBS setup message */
+      PFREE (cc_data->stored_ccbs_setup);
+      cc_data->stored_ccbs_setup = NULL;
+
+      /* Inform MMI */
+      {
+        PALLOC (release, MNCC_RELEASE_IND);
+        release->ti    = cc_data->ti;
+        release->cause = cause; /* provide cause from lower layers transparently to higher layers */
+        release->c_raw_cause = 0;
+        PSENDX (MMI, release);
+      }
+
+      cc_set_state (M_CC_CS_0);
+      srv_use_stored_prim ();
+      break;
+
+    case CS_101: /* Re-establishment request in state M_CC_CS_10 */\
+    case CS_261: /* Re-establishment request in state CS_26 */\
+      EM_CC_REESTABLISHED_FAILED;
+      /*FALLTHROUGH*/ /*lint -fallthrough*/
+
+    default:
+      cc_reset_dtmf (); /* China change HM 11.07.00 */
+      if (cc_data->ti NEQ cc_data->setup_reattempt_ti)
+      {
+        PALLOC (rej_ind, MNCC_REJECT_IND);
+        rej_ind->ti  = cc_data->ti;
+        rej_ind->cause = cause;
+        PSENDX (MMI, rej_ind);
+      }
+
+      cc_set_state (M_CC_CS_0);
+      if (cc_data->ti EQ cc_data->setup_reattempt_ti)
+      {
+        cc_setup_reattempt();
+      }
+      else
+      {
+        srv_use_stored_prim ();
+      }
+      break;
+  }
+}
+
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_err_ind                 |
++--------------------------------------------------------------------+
+
+  PURPOSE : Processing of an incoming error indication from
+            mobility management.
+
+*/
+
+GLOBAL void cc_err_ind (T_MMCM_ERROR_IND * err_ind)
+{
+  GET_INSTANCE_DATA;
+
+  TRACE_FUNCTION ("cc_err_ind()");
+
+  switch (cc_data->state[cc_data->index_ti])
+  {
+    case M_CC_CS_0:
+      break;
+
+    case M_CC_CS_10:
+      {
+/* Implements Measure#  41 */
+        cc_mncc_sync_ind(err_ind->ti, CS_101);
+
+        EM_CC_REESTABLISHED_STARTED;
+
+      }
+      break;
+
+    case M_CC_CS_26:
+      {
+/* Implements Measure#  41 */
+        cc_mncc_sync_ind(err_ind->ti, CS_261);
+
+        EM_CC_REESTABLISHED_STARTED;
+
+      }
+      break;
+
+    default:
+      cc_reset_dtmf ();
+      if (cc_data->ti NEQ cc_data->setup_reattempt_ti)
+      {
+        PALLOC (rel_ind, MNCC_RELEASE_IND);
+        rel_ind->cause = err_ind->cause;
+        rel_ind->ti    = cc_data->ti;
+        rel_ind->c_raw_cause = 0;
+        PSENDX (MMI, rel_ind);
+      }
+      cc_set_state (M_CC_CS_0);
+      for_rel_req ();
+      cc_setup_reattempt ();
+      break;
+  }
+  PFREE (err_ind);
+}
+/* Implements Measure#  8 */
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_mncc_release_ind        |
++--------------------------------------------------------------------+
+
+  PURPOSE : send mncc_release_ind
+
+*/
+
+LOCAL void cc_mncc_release_ind (T_PRIM * prim)
+{
+  GET_INSTANCE_DATA;
+  UBYTE                     v_cc_cause;
+  T_M_CC_cc_cause                *cc_cause;                             
+  
+  TRACE_FUNCTION ("cc_mncc_release_ind()");
+  
+  if (prim->custom.opc EQ D_RELEASE)
+  {
+    v_cc_cause = ((T_D_RELEASE *)P2D(prim))->v_cc_cause;
+    cc_cause   = &((T_D_RELEASE *)P2D(prim))->cc_cause;
+    cc_build_facility_ind (MNCC_FAC_IN_RELEASE, 
+                           ((T_D_RELEASE *)P2D(prim))->v_facility,
+                           &((T_D_RELEASE *)P2D(prim))->facility);
+  }
+  else
+  {
+    v_cc_cause = ((T_D_RELEASE_COMP *)P2D(prim))->v_cc_cause;
+    cc_cause   = &((T_D_RELEASE_COMP *)P2D(prim))->cc_cause;
+    cc_build_facility_ind (MNCC_FAC_IN_RELEASE_COMP,
+                           ((T_D_RELEASE_COMP *)P2D(prim))->v_facility, 
+                           &((T_D_RELEASE_COMP *)P2D(prim))->facility);
+  }
+
+  /* Patch HM 29-Jan-02 <<< */
+  {
+    PALLOC (rel_ind, MNCC_RELEASE_IND);
+    rel_ind->ti    = cc_data->ti;
+    if (v_cc_cause)
+    {
+      rel_ind->cause = CAUSE_MAKE(DEFBY_STD, 
+                                   ORIGSIDE_NET, 
+                                   MNCC_CC_ORIGINATING_ENTITY,
+                                   cc_cause->cause);
+      rel_ind->c_raw_cause = cc_build_cause (cc_cause,
+                                             rel_ind->raw_cause);
+    }
+    else
+    {
+      rel_ind->cause = CAUSE_MAKE(DEFBY_CONDAT, 
+                                   ORIGSIDE_NET, 
+                                   MNCC_CC_ORIGINATING_ENTITY,
+                                   NOT_PRESENT_8BIT);
+      /* Setting raw_cause to empty as this is a local release
+       * of MM connection,CC is not receiving any cause value
+       * from Network
+       */
+       rel_ind->c_raw_cause = 0;
+    }
+    PSENDX (MMI, rel_ind);
+  }
+}
+
+/* Implements Measure#  9 */
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_mncc_release_cnf        |
++--------------------------------------------------------------------+
+
+  PURPOSE : send mncc_release_cnf
+
+*/
+
+LOCAL void cc_mncc_release_cnf (T_PRIM * prim)
+{
+  GET_INSTANCE_DATA;
+  UBYTE                     v_cc_cause;
+  T_M_CC_cc_cause                *cc_cause;                             
+
+  TRACE_FUNCTION ("cc_mncc_release_cnf()");
+  
+  if (prim->custom.opc EQ D_RELEASE)
+  {
+    v_cc_cause = ((T_D_RELEASE *)P2D(prim))->v_cc_cause;
+    cc_cause   = &((T_D_RELEASE *)P2D(prim))->cc_cause;
+    cc_build_facility_ind (MNCC_FAC_IN_RELEASE_COMP, /* A little bit dirty */
+                           ((T_D_RELEASE *)P2D(prim))->v_facility,
+                           &((T_D_RELEASE *)P2D(prim))->facility);
+  }
+  else
+  {
+    v_cc_cause = ((T_D_RELEASE_COMP *)P2D(prim))->v_cc_cause;
+    cc_cause   = &((T_D_RELEASE_COMP *)P2D(prim))->cc_cause;
+    cc_build_facility_ind (MNCC_FAC_IN_RELEASE_COMP,
+                           ((T_D_RELEASE_COMP *)P2D(prim))->v_facility, 
+                           &((T_D_RELEASE_COMP *)P2D(prim))->facility);
+  }
+
+  /* Patch HM 29-Jan-02  <<< */
+  {
+    PALLOC (rel_cnf, MNCC_RELEASE_CNF);
+    rel_cnf->ti    = cc_data->ti;
+    if (v_cc_cause)
+    {
+      rel_cnf->cause = CAUSE_MAKE(DEFBY_STD, 
+                                  ORIGSIDE_NET, 
+                                  MNCC_CC_ORIGINATING_ENTITY,
+                                  cc_cause->cause);
+      rel_cnf->c_raw_cause = cc_build_cause (cc_cause,
+                                             rel_cnf->raw_cause);
+    }
+    else
+    {
+      rel_cnf->cause = CAUSE_MAKE(DEFBY_CONDAT, 
+                                  ORIGSIDE_NET, 
+                                  MNCC_CC_ORIGINATING_ENTITY,
+                                  NOT_PRESENT_8BIT);
+      /* Setting raw_cause to empty as this is a local release
+       * of MM connection,CC is not receiving any cause value
+       * from Network
+       */
+       rel_cnf->c_raw_cause = 0;
+    }
+    PSENDX (MMI, rel_cnf);
+  }
+}
+
+
+/* Implements Measure#  41 */
+/*
++--------------------------------------------------------------------+
+| PROJECT : GSM-PS (6147)       MODULE  : CC_REL                     |
+| STATE   : code                ROUTINE : cc_mncc_sync_ind           |
++--------------------------------------------------------------------+
+
+  PURPOSE : send cc_mncc_sync_ind
+
+*/
+
+LOCAL void cc_mncc_sync_ind (UBYTE ti, UBYTE new_state)
+{
+  TRACE_FUNCTION ("cc_mncc_sync_ind()");
+  {
+    PALLOC (sync_ind, MNCC_SYNC_IND); /* T_MNCC_SYNC_IND */
+
+    sync_ind->ti    = ti;
+    sync_ind->cause = MNCC_CAUSE_REEST_STARTED;
+    sync_ind->ch_info.ch_mode=NOT_PRESENT_8BIT;
+    sync_ind->ch_info.ch_type=NOT_PRESENT_8BIT;
+    
+    PSENDX (MMI, sync_ind);
+    
+    for_reest_req ();
+    cc_set_state (new_state);
+  }
+
+}
+#endif