FreeCalypso > hg > freecalypso-citrine
diff g23m-gsm/rr/rr_dats.c @ 0:75a11d740a02
initial import of gsm-fw from freecalypso-sw rev 1033:5ab737ac3ad7
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Thu, 09 Jun 2016 00:02:41 +0000 |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/g23m-gsm/rr/rr_dats.c Thu Jun 09 00:02:41 2016 +0000 @@ -0,0 +1,3084 @@ +/* ++----------------------------------------------------------------------------- +| Project : +| Modul : ++----------------------------------------------------------------------------- +| 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 the data transfer +| capability of the module Radio Resource. ++----------------------------------------------------------------------------- +*/ + +#ifndef RR_DATS_C +#define RR_DATS_C + +#include "config.h" +#include "fixedconf.h" +#include "condat-features.h" + +#define ENTITY_RR + +/*==== INCLUDES ===================================================*/ + +#include <string.h> +#include <stdlib.h> +#include <stddef.h> /* offsetof */ +#include "typedefs.h" +#include "pcm.h" +#include "pconst.cdg" +#include "mconst.cdg" +#include "message.h" +#include "ccdapi.h" +#include "vsi.h" +#include "custom.h" +#include "gsm.h" +#include "prim.h" +#include "cnf_rr.h" +#include "tok.h" +#include "rr.h" +#include "em.h" +#include "rr_em.h" + +/*==== EXPORT =====================================================*/ + +/*==== PRIVAT =====================================================*/ + +#define TDMA_FRAMES_PER_HYPERFRAME 2715648 +#define QUARTER_BITS_PER_FRAME 5000 + +/*==== VARIABLES ==================================================*/ + +/*==== FUNCTIONS ==================================================*/ + +LOCAL BOOL dat_for_handover_mob_alloc(UBYTE *mob_alloc, + T_LIST *hop_list_handover, + T_VOID_STRUCT *mob_alloc_handover, + T_DL_DATA_IND *dl_data_ind); + +LOCAL void dat_cr_data_multirate_conf(U8 v_multirate_conf, T_multirate_conf *multirate_conf); + +LOCAL void dat_dedicated_req_ch_type2(T_ch_type2 *ch_type2, T_chan_desc_before *chan_desc_before, + T_LIST *hop_list_before); +/* + * ------------------------------------------------------------------- + * SIGNAL Processing functions + * ------------------------------------------------------------------- + */ + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_att_cell_selected | ++--------------------------------------------------------------------+ + + PURPOSE : Attachment process indicates that it has camped on a cell. + Data transfer process changes the state and connections + are possible. + +*/ + +GLOBAL void dat_att_cell_selected (void) +{ + GET_INSTANCE_DATA; + TRACE_FUNCTION ("dat_att_cell_selected()"); + + SET_STATE (STATE_DAT, DAT_IDLE); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_att_null | ++--------------------------------------------------------------------+ + + PURPOSE : Attachment process indicates loss of coverage and connections + are not longer possible. + +*/ + +GLOBAL void dat_att_null (void) +{ + GET_INSTANCE_DATA; + + TRACE_FUNCTION ("dat_att_null()"); + + SET_STATE (STATE_DAT, DAT_NULL); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_assign_cmd | ++--------------------------------------------------------------------+ + + PURPOSE : The function handles the reception of a channel + assignment message. + +*/ + +GLOBAL void dat_for_assign_cmd (T_DL_DATA_IND *dl_data_ind, + T_D_ASSIGN_CMD *assign_cmd, + T_LIST *hop_list_after, + T_LIST *hop_list_before, + T_LIST *cell_chan_desc) +{ + GET_INSTANCE_DATA; + UBYTE mob_alloc[65]; + + PALLOC (dedicated_req, MPH_DEDICATED_REQ); + + TRACE_FUNCTION ("dat_for_assign_cmd()"); + + switch (GET_STATE (STATE_DAT)) + { + case DAT_DEDICATED: + if (rr_data->ms_data.error.cs) + { + TRACE_EVENT_P1 ("RRC cause = %02x", rr_data->ms_data.error.cs); + } + /* + * dynamic configuration command : IHO + * Lock the DUT to the cell it is already camping - + * Ignore the Channel Assignment command message and send an + * Assignment failure message to the network. + */ + + if(rr_data->dyn_config.iho AND (rr_data->sc_data.ch_mode EQ CM_AMR)) + { + TRACE_EVENT("D_ASSIGN_CMD : IHO"); + for_set_content_error (RRC_CHANNEL_MODE); + } + + switch (rr_data->ms_data.error.cs) + { + /* case RRC_INVALID_MAN_INFO: this value is currently never set */ + case RRC_COND_IE_ERROR: /* 0x64 */ + { + /* + * If an mandatory info element error or a + * conditional info element error has been detected, + * a RR STATUS message is returned on the existing + * connection before l2 is suspended + */ + dat_send_rr_status_msg(rr_data->ms_data.error.cs); + PFREE (dedicated_req); + PFREE (dl_data_ind); + break ; + } + + case RRC_INCORRECT_MSG: /* 0x5f */ + { + /* + * If a structurally correct message has been detected, + * containing erroneous data, an Assignment Failure message + * is sent back. + */ + + /* + * If the ASSIGNMENT COMMAND is erroneous, then the + * ASSIGNMENT FAILURE command is sent via a priority + * DL_RECONNECT_REQ. This ensures DL will halt processing + * anything in its buffer until it has sent this message + * onto the nw. + * + */ + for_suspend_layer_2 (); + dat_send_assign_fail_msg(rr_data->ms_data.error.val); + + RR_EM_SET_ASSIGN_FAIL_CAUSE(rr_data->ms_data.error.val); + + PFREE (dedicated_req); + PFREE (dl_data_ind); + break; + } + + default: + /* + * the initial check was successful and the + * message is processed. + * + */ + for_suspend_layer_2 (); + + /* + * use data of the old cell if no new data + * are inserted in the message + */ + rr_data->cr_data.ch_mode = rr_data->sc_data.ch_mode; + rr_data->cr_data.ciph_on = rr_data->sc_data.ciph_on; + rr_data->cr_data.algo = rr_data->sc_data.algo; + rr_data->cr_data.cd.v_cell_chan_desc = + rr_data->sc_data.cd.v_cell_chan_desc; + memcpy (&rr_data->cr_data.cd.cell_chan_desc, + &rr_data->sc_data.cd.cell_chan_desc, + sizeof (T_LIST)); + + /* + * if AMR is supported set the default values + * to the current serving cell values. + */ + if(rr_data->sc_data.ch_mode EQ CM_AMR) + { + memcpy(&rr_data->cr_data.amr_conf, + &rr_data->sc_data.amr_conf, + sizeof (T_multirate_conf)); + } + else { + /* + * AMR is not supported, therefore set some dummy values. This is necessary because + * the later Layer1 configuration must include an AMR configuration!! + */ + memset(&rr_data->cr_data.amr_conf, 0, sizeof (T_multirate_conf)); + } + + if (assign_cmd->v_cell_chan_desc) + { + /* + * If the message contains a cell channel description + * use the new one. + */ + memcpy (&rr_data->cr_data.cd.cell_chan_desc, + cell_chan_desc, + sizeof (T_LIST)); + rr_data->cr_data.cd.v_cell_chan_desc = WITH_CHANGED_CONTENT; + } + + if (assign_cmd->v_chan_mode) + /* + * if the message contains a channel mode, use the new one. + */ + rr_data->cr_data.ch_mode = assign_cmd->chan_mode; + + /* + * If AMR is signalled check if new multi-rate speech codec is part of the assignment cmd + * otherwise use default values set earlier. + */ + +/* Implements RR Clone findings #9 */ + dat_cr_data_multirate_conf(assign_cmd->v_multirate_conf,&assign_cmd->multirate_conf); + + + if (assign_cmd->v_ciph_mode_set) + { + /* + * If ciphering is defined in the message, handle it. + */ + rr_data->cr_data.ciph_on = assign_cmd->ciph_mode_set.sc; + rr_data->cr_data.algo = assign_cmd->ciph_mode_set.algo_ident; + + if (rr_data->cr_data.ciph_on EQ CIPH_ON AND + rr_data->sc_data.ciph_received EQ FALSE) + { + /* + * if ciphering is not active, but set in the message + * this is a failure and the configuration is aborted. + * Instead the reconnection on the old channel is started. + */ + dat_send_assign_fail_msg(RRC_PROT_UNSPECIFIED); + + RR_EM_SET_ASSIGN_FAIL_CAUSE(RRC_PROT_UNSPECIFIED); + + PFREE (dedicated_req); + PFREE (dl_data_ind); + + return; + } + } + + if (assign_cmd->chan_desc.hop EQ 1 AND + assign_cmd->v_mob_alloc_after) + { + if (rr_data->cr_data.cd.v_cell_chan_desc EQ NO_CONTENT) + { + /* + * If the new channel needs hopping, but there is no + * cell channel description available, the configuration + * is aborted due to a conditional error. + * Instead the reconnection on the old channel is started. + */ + dat_send_assign_fail_msg(RRC_NO_CELL_ALLOC); + + RR_EM_SET_ASSIGN_FAIL_CAUSE(RRC_NO_CELL_ALLOC); + + PFREE (dedicated_req); + PFREE (dl_data_ind); + return; + } + + /* + * if the message contains a mobile allocation, + * build a list of 1-bits from the bitmap. + */ + att_bits_to_byte (mob_alloc, + assign_cmd->mob_alloc_after.c_mac, + assign_cmd->mob_alloc_after.mac); + + /* + * create a hopping list from mobile allocation and cell channel + * description + */ + if(!srv_create_chan_mob_alloc (&rr_data->cr_data.cd.cell_chan_desc, + hop_list_after, + mob_alloc)) + { + dat_send_assign_fail_msg(RRC_FREQ_NOT_IMPL); + + RR_EM_SET_ASSIGN_FAIL_CAUSE(RRC_FREQ_NOT_IMPL); + + PFREE (dedicated_req); + PFREE (dl_data_ind); + return; + } + } + + /* + * clean primitive to layer 1 + */ + memset (dedicated_req, 0, sizeof (T_MPH_DEDICATED_REQ)); + + dedicated_req->mod = MODE_CHAN_ASSIGN; + + rr_data->cr_data.chan_desc = assign_cmd->chan_desc; + + /* + * Set Channel Type + */ + dedicated_req->ch_type.ch = assign_cmd->chan_desc.chan_type; + dedicated_req->ch_type.tn = assign_cmd->chan_desc.tn; + dedicated_req->ch_type.tsc = assign_cmd->chan_desc.tsc; + dedicated_req->ch_type.h = assign_cmd->chan_desc.hop; + if (assign_cmd->chan_desc.hop EQ H_NO) + { + dedicated_req->ch_type.arfcn = assign_cmd->chan_desc.arfcn; + } + else + { + dedicated_req->ch_type.maio = assign_cmd->chan_desc.maio; + dedicated_req->ch_type.hsn = assign_cmd->chan_desc.hsn; + + /* CSI-LLD section:4.1.1.11 + * This function Updates the black list with the MA list received + * inthe assignment command + */ + cs_remove_BA_MA_from_black_list(rr_data->cs_data.region,hop_list_after); + + srv_create_list (hop_list_after, dedicated_req->ch_type.ma, + MAX_MA_CHANNELS, TRUE, 0); + } + + /* + * set initial power + */ + dedicated_req->tr_para.power = assign_cmd->pow_cmd.pow; + + /* + * set starting time if available. + */ + if (assign_cmd->v_start_time) + { + dedicated_req->start.v_start = TRUE; + dedicated_req->start.t1 = assign_cmd->start_time.t1; + dedicated_req->start.t2 = assign_cmd->start_time.t2; + dedicated_req->start.t3 = assign_cmd->start_time.t3; + } + + /* + * Setting of before starting time elements ! + */ + if (assign_cmd->v_chan_desc_before EQ FALSE) + dedicated_req->ch_type2.ch = NOT_PRESENT_8BIT; + else + { + if (assign_cmd->v_mob_alloc_before) + { + att_bits_to_byte (mob_alloc, + assign_cmd->mob_alloc_before.c_mac, + assign_cmd->mob_alloc_before.mac); + if(!srv_create_chan_mob_alloc (&rr_data->cr_data.cd.cell_chan_desc, + hop_list_before, + mob_alloc)) + { + dat_send_assign_fail_msg(RRC_FREQ_NOT_IMPL); + + RR_EM_SET_ASSIGN_FAIL_CAUSE( RRC_FREQ_NOT_IMPL); + + PFREE (dedicated_req); + PFREE (dl_data_ind); + return; + } + } + + dat_dedicated_req_ch_type2(&dedicated_req->ch_type2, &assign_cmd->chan_desc_before, + hop_list_before ); + + } + + + /* + * set dtx depending on the channel type (halfrate or fullrate) + */ + + if (dedicated_req->ch_type.ch EQ 2 OR + dedicated_req->ch_type.ch EQ 3) + dedicated_req->tr_para.dtx = + rr_data->sc_data.cd.dtx = rr_data->sc_data.cd.dtx_half; + else + dedicated_req->tr_para.dtx = + rr_data->sc_data.cd.dtx = rr_data->sc_data.cd.dtx_full; + + dedicated_req->arfcn = rr_data->nc_data[SC_INDEX].arfcn; + dedicated_req->tr_para.rlt = rr_data->sc_data.cd.cell_options.rlt; + dedicated_req->tr_para.pwrc = rr_data->sc_data.cd.cell_options.pow_ctrl; + dedicated_req->tr_para.mode = rr_data->cr_data.ch_mode; + + /* + * Set multi-rate speech codec + */ + dedicated_req->amr_conf.nscb = rr_data->cr_data.amr_conf.nscb; + dedicated_req->amr_conf.icmi = rr_data->cr_data.amr_conf.icmi; + dedicated_req->amr_conf.st_mode = rr_data->cr_data.amr_conf.st_mode; + dedicated_req->amr_conf.acs = rr_data->cr_data.amr_conf.set_amr; + + /* + * valid flag for the threshold and hystersis values. amr_conf.c_cod_prop + * defines the number of threshold and hystersis values. + */ + dedicated_req->amr_conf.v_cod_prop = rr_data->cr_data.amr_conf.v_cod_prop; + + if(dedicated_req->amr_conf.v_cod_prop) + { + int i; + dedicated_req->amr_conf.c_cod_prop = rr_data->cr_data.amr_conf.c_cod_prop; + for(i=0; i<dedicated_req->amr_conf.c_cod_prop; i++) + memcpy(&dedicated_req->amr_conf.cod_prop[i], + &rr_data->cr_data.amr_conf.cod_prop[i], + sizeof(T_cod_prop)); + } + + if (rr_data->cr_data.ciph_on) + { + /* + * set cipher parameter if available. + */ + dedicated_req->ciph.stat = rr_data->cr_data.ciph_on; + dedicated_req->ciph.algo = rr_data->cr_data.algo; + memcpy (dedicated_req->ciph.kc, rr_data->ms_data.kc, KC_STRING_SIZE); + } + + RR_EM_GET_HOPPING_CHANNEL (dedicated_req->ch_type.ma, dedicated_req->ch_type2.ma, + dedicated_req->start.v_start,dedicated_req->ch_type2.maio); + + EM_ASSIGNMENT_RECEIVED; + + /* + * configure layer 1 + */ + SET_STATE (STATE_DAT, DAT_CHAN_ASS); + PSENDX (PL, dedicated_req); + PFREE (dl_data_ind); + return; + } + break; + + default: + PFREE (dedicated_req); + PFREE (dl_data_ind); + break; + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_chan_mod | ++--------------------------------------------------------------------+ + + PURPOSE : The function handles a received channel mode modify message. + +*/ + +GLOBAL void dat_for_chan_mod (T_DL_DATA_IND *dl_data_ind, + T_D_CHAN_MOD *chan_mod) +{ + GET_INSTANCE_DATA; + MCAST (chan_mod_ack, U_CHAN_MOD_ACK); + + TRACE_FUNCTION ("dat_for_chan_mod()"); + + if (rr_data->ms_data.error.cs EQ 0) + { + /* + * the check in the formatter indicates no problems + * store new channel mode. + */ + rr_data->sc_data.ch_mode = chan_mod->chan_mode; + + /* + * the channel mode modify message contains a multi-rate configuration IEI + */ + if( chan_mod->v_multirate_conf AND (chan_mod->chan_mode EQ CM_AMR) ) + { + int i; + rr_data->sc_data.amr_conf.mr_vers = chan_mod->multirate_conf.mr_vers; + rr_data->sc_data.amr_conf.nscb = chan_mod->multirate_conf.nscb; + rr_data->sc_data.amr_conf.icmi = chan_mod->multirate_conf.icmi; + rr_data->sc_data.amr_conf.st_mode = chan_mod->multirate_conf.st_mode; + rr_data->sc_data.amr_conf.set_amr = chan_mod->multirate_conf.set_amr; + + rr_data->sc_data.amr_conf.v_cod_prop = chan_mod->multirate_conf.v_cod_prop; + if(rr_data->sc_data.amr_conf.v_cod_prop) + { + rr_data->sc_data.amr_conf.c_cod_prop = chan_mod->multirate_conf.c_cod_prop; + for (i=0; i< rr_data->sc_data.amr_conf.c_cod_prop; i++) + memcpy(&rr_data->sc_data.amr_conf.cod_prop[i], &chan_mod->multirate_conf.cod_prop[i], sizeof(T_cod_prop)); + } + } + + /* + * configure layer 1 + */ + dat_code_mph_chan_mode_req (chan_mod); + + EM_CHANNEL_MODE_MODIFY; + + /* + * indicate new channel mode to MM + */ + dat_code_channel_mode_to_mm (); + } + + { + /* + * build the answer to the network + * (channel mode modify acknowledge message) + */ + PALLOC_MSG (dl_data_req, DL_DATA_REQ, U_CHAN_MOD_ACK); + + /* + * set channel type and SAPI + */ + dat_code_prr_channel (&dl_data_req->ch_type, + &dl_data_req->sapi, + rr_data->sc_data.chan_desc.chan_type); + + chan_mod_ack->msg_type = U_CHAN_MOD_ACK; + memcpy (&chan_mod_ack->chan_desc, + &rr_data->sc_data.chan_desc, + sizeof (T_chan_desc)); + + /* + * set the current channel mode. if the new + * channel mode is supported by the MS, the new + * one is returned, else it is the previous one + * and layer 1 was not re-configured. + */ + chan_mod_ack->chan_mode = rr_data->sc_data.ch_mode; + + for_dat_data_req (dl_data_req); + + EM_CHANNEL_MODE_MODIFY_ACK; + } + + PFREE(dl_data_ind); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_chan_rel | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of a channel release message. + +*/ + +GLOBAL void dat_for_chan_rel (T_DL_DATA_IND *dl_data_ind, + T_D_CHAN_REL *chan_rel) +{ + GET_INSTANCE_DATA; + TRACE_FUNCTION ("dat_for_chan_rel()"); + + if (GET_STATE (STATE_DAT) NEQ DAT_NULL) + { + /* + * disconnect layer 2 link + */ + dat_disconnect_link (CAUSE_MAKE (DEFBY_STD, + ORIGSIDE_NET, + RR_ORIGINATING_ENTITY, + chan_rel->rr_cause)); + +#ifdef GPRS + if (chan_rel->v_gprs_resum) + { + rr_data->gprs_data.gprs_resump = chan_rel->gprs_resum.res_ack; + } + /* + o if the element is not available but we have send a susp_req + a resumption failure has occured (gprs_resump was already set + on tx of the suspension request) + o if the element is not present and we have not send a suspension + request there is no resumption failure. + o For Ericsson we have to do a RAU after every CS call even if the + call started on a GSM-only cell and we did not send a suspension request */ + else + if(att_gprs_is_avail()) + rr_data->gprs_data.gprs_resump = GPRS_RESUMPTION_NOT_ACK; +#endif + + if (chan_rel->v_ba_range) + { + /* + * convert RR_BA_RANGE to BCCH-LIST and + * send it with RR SYNC IND to MM + */ + dat_code_prr_bcch_info (chan_rel->v_ba_range, + &chan_rel->ba_range); + } + + EM_CHANNEL_RELEASE; + } + PFREE (dl_data_ind); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_class_enq | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of a classmark enquiry message. + +*/ + +#ifdef REL99 +GLOBAL void dat_for_class_enq (T_DL_DATA_IND *dl_data_ind, + T_D_CLASS_ENQ *class_enq) +#else +GLOBAL void dat_for_class_enq (T_DL_DATA_IND *dl_data_ind) +#endif +{ + TRACE_FUNCTION ("dat_for_class_enq()"); + + if (dat_check_error_flag (SEND_RR_STATUS)) + { + /* + * The syntax check indicates no problems, then + * process the message. + * + * The MS returns a classmark change message. + */ +/* Implements RR Clone findings #15 */ +#ifdef REL99 + /*Perform checks on classmark enquiry mask IE, if present*/ + if ((class_enq->v_class_enq_mask EQ FALSE) OR + ((class_enq->v_class_enq_mask EQ TRUE) AND + (class_enq->class_enq_mask.class_req EQ CLASS_CHANGE_REQ) ) ) +#endif + dat_class_chng_data_req(); + } + + EM_CLASSMARK_ENQUIRY; + + PFREE (dl_data_ind); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : send_mph_tch_loop_req | ++--------------------------------------------------------------------+ + + PURPOSE : Send the L1 primitive for close TCH loop. + +*/ + +static void send_mph_tch_loop_req(T_DL_DATA_IND * dl_data_ind, + UBYTE loop_command) +{ + /* + * configure layer 1 + */ + PREUSE (dl_data_ind, loop_req, MPH_TCH_LOOP_REQ);/* T_MPH_TCH_LOOP_REQ */ + loop_req->tch_loop = loop_command; + PSENDX (PL, loop_req); +} + +/* ++-----------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : send_close_tch_loop_ack_to_nw | ++-----------------------------------------------------------------------+ + + PURPOSE : Send the CLOSE TCH LOOP ACK message to the network. + +*/ +static void send_close_tch_loop_ack_to_nw(void) +{ + GET_INSTANCE_DATA; + /* + * if the TCH loop is open and a TCH is assigned + */ + + PALLOC_SDU (data_req, DL_DATA_REQ, 2*BITS_PER_BYTE); + /* + * set channel type and sapi for the response to the network + */ + dat_code_prr_channel (&data_req->ch_type, + &data_req->sapi, + rr_data->sc_data.chan_desc.chan_type); + + /* + * code the message without CCD + */ + data_req->sdu.l_buf = 16; + data_req->sdu.o_buf = ENCODE_OFFSET; + data_req->sdu.buf [0] = 0; + /*lint -e415 -e416 Likely access of out-of-bounds pointer*/ + data_req->sdu.buf [1] = 0; + data_req->sdu.buf [2] = 0; + data_req->sdu.buf [3] = 0x0F; /* TI=0, PD = TST */ + data_req->sdu.buf [4] = 0x01; /* MT = Close TCH Ack */ + /*lint +e415 +e416 Likely access of out-of-bounds pointer*/ + TRACE_EVENT ("DL_DATA_REQ (RR message)"); + + EM_TCH_LOOP_CLOSED; + + PSENDX (DL, data_req); +} + + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_close_loop_cmd | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of a TCH Close Loop Command message. + +*/ + +static const UBYTE LOOP_TYPE [32] = + { /* C B A Z Y */ + 0x00, /* 0 0 0 0 0 -> Type A */ + 0x01, /* 0 0 0 0 1 -> Type B */ + 0x02, /* 0 0 0 1 0 -> Type C */ + 0x02, /* 0 0 0 1 1 -> Type C */ + 0x03, /* 0 0 1 0 0 -> Type D */ + 0x03, /* 0 0 1 0 1 -> Type D */ + 0x03, /* 0 0 1 1 0 -> Type D */ + 0x03, /* 0 0 1 1 1 -> Type D */ + 0x04, /* 0 1 0 0 0 -> Type E */ + 0x04, /* 0 1 0 0 1 -> Type E */ + 0x04, /* 0 1 0 1 0 -> Type E */ + 0x04, /* 0 1 0 1 1 -> Type E */ + 0x05, /* 0 1 1 0 0 -> Type F */ + 0x05, /* 0 1 1 0 1 -> Type F */ + 0x05, /* 0 1 1 1 0 -> Type F */ + 0x05, /* 0 1 1 1 1 -> Type F */ + 0xFF, /* 1 0 0 0 0 -> Not valid */ + 0xFF, /* 1 0 0 0 1 -> Not valid */ + 0xFF, /* 1 0 0 1 0 -> Not valid */ + 0xFF, /* 1 0 0 1 1 -> Not valid */ + 0xFF, /* 1 0 1 0 0 -> Not valid */ + 0xFF, /* 1 0 1 0 1 -> Not valid */ + 0xFF, /* 1 0 1 1 0 -> Not valid */ + 0xFF, /* 1 0 1 1 1 -> Not valid */ + 0xFF, /* 1 1 0 0 0 -> Not valid */ + 0xFF, /* 1 1 0 0 1 -> Not valid */ + 0xFF, /* 1 1 0 1 0 -> Not valid */ + 0xFF, /* 1 1 0 1 1 -> Not valid */ + 0x06, /* 1 1 1 0 0 -> Type I */ + 0x06, /* 1 1 1 0 1 -> Type I */ + 0x06, /* 1 1 1 1 0 -> Type I */ + 0x06 /* 1 1 1 1 1 -> Type I */ + }; + + +GLOBAL void dat_for_close_loop_cmd (T_DL_DATA_IND * dl_data_ind, + UBYTE subchannel) +{ + GET_INSTANCE_DATA; + UBYTE loop_command = NOT_PRESENT_8BIT; + + TRACE_FUNCTION ("dat_for_close_loop_cmd()"); + + if (dat_test_sim_available () OR !dat_check_sim_available () ) + { + /* + * only if a test SIM card is inserted + */ + if ((rr_data->tch_loop_subch EQ NOT_PRESENT_8BIT) AND + rr_data->sc_data.chan_desc.chan_type < CH_SDCCH_4_0) + { + switch ((loop_command = LOOP_TYPE [(subchannel>>1) & 0x1F])) + { + case TCH_LOOP_C: /* Loop C */ + /* first send ACK msg, then activate L1 */ + send_close_tch_loop_ack_to_nw(); + /* + * Delay to allow L1/HW to switch + */ + vsi_t_sleep (VSI_CALLER DELAY_CLOSE_TCH_LOOP_ACK); + send_mph_tch_loop_req(dl_data_ind, loop_command); + /* will be needed when TCH Open Loop Command will be received */ + rr_data->tch_loop_subch = loop_command; + break; + case TCH_LOOP_I: /* Loop I */ + if (rr_data->sc_data.ch_mode NEQ CM_AMR) + { + PFREE (dl_data_ind); + break; + } + case TCH_LOOP_A: + case TCH_LOOP_B: + case TCH_LOOP_D: + case TCH_LOOP_E: + case TCH_LOOP_F: + /* Loop A, B, D, E, F, I */ + send_mph_tch_loop_req(dl_data_ind, loop_command); + /* + * Delay to allow L1/HW to switch + */ + vsi_t_sleep (VSI_CALLER DELAY_CLOSE_TCH_LOOP_ACK); + send_close_tch_loop_ack_to_nw(); + /* will be needed when TCH Open Loop Command will be received */ + rr_data->tch_loop_subch = loop_command; + break; + default : + TRACE_EVENT_P1("TCH_LOOP_CLOSE_CMD : wrong subchannel (%x)", subchannel); + PFREE (dl_data_ind); + break; + } + } + else + { + PFREE (dl_data_ind); + } + } + else + { + PFREE (dl_data_ind); + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_test_interface | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of a Test-Interface message. + +*/ + +GLOBAL void dat_for_test_interface (T_DL_DATA_IND * dl_data_ind, + UBYTE tested_device) +{ + TRACE_FUNCTION ("dat_for_test_interface()"); + + if (dat_test_sim_available ()) + { + /* + * Only if a test SIM card is inserted + * + * then configure layer 1 + */ + PREUSE (dl_data_ind, dai_req, MPH_DAI_REQ); /* T_MPH_DAI_REQ */ + + dai_req->device = tested_device; + + EM_TEST_INTERFACE; + + PSENDX (PL, dai_req); + } + else + { + /* + * else ignore the message + */ + PFREE (dl_data_ind); + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_ciph_cmd | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of a cipher mode command message. + +*/ + +GLOBAL void dat_for_ciph_cmd (T_DL_DATA_IND *dl_data_ind, + T_D_CIPH_CMD *ciph_cmd) +{ + GET_INSTANCE_DATA; + TRACE_FUNCTION ("dat_for_cyph_cmd()"); + + if (dat_check_error_flag (SEND_RR_STATUS)) + { + /* + * the check in the formatter was successful + */ + if ( + ((rr_data->sc_data.ciph_on EQ CIPH_ON) AND + (ciph_cmd->ciph_mode_set.sc EQ START_CIPH_YES)) + OR + ((ciph_cmd->ciph_mode_set.sc EQ START_CIPH_YES) AND + (rr_data->dyn_config.nkc EQ 0 AND rr_data->ms_data.cksn > 6)) + ) + { + /* + * Respond with RR Status in 2 cases + * + * 1: if NW re-enables ciphering + * 2: if network has enabled ciphering "and" no valid ciphering key + * is available (and user specific handling of cksn is + * disabled (nck==0)). + * If network has not enabled ciphering, then ciphering key + * value is not checked + */ + dat_send_rr_status_msg(RRC_PROT_UNSPECIFIED); + } + else + { + MCAST (ciph_comp, U_CIPH_COMP); + PALLOC_MSG (dl_data_req, DL_DATA_REQ, U_CIPH_COMP); + + /* + * set channel type and SAPI for response to the network + */ + dat_code_prr_channel (&dl_data_req->ch_type, + &dl_data_req->sapi, + rr_data->sc_data.chan_desc.chan_type); + + /* + * store cipher parameter + */ + rr_data->sc_data.ciph_on = ciph_cmd->ciph_mode_set.sc; + + rr_data->sc_data.algo = ciph_cmd->ciph_mode_set.algo_ident; + rr_data->sc_data.ciph_received = TRUE; + memcpy (rr_data->ms_data.kc, rr_data->ms_data.new_kc, KC_STRING_SIZE); + + /* + * configure layer 1 + */ + + if ( rr_data->ms_data.cksn <= 6 ) + { + dat_code_mph_ciphering_req (rr_data->sc_data.ciph_on, + rr_data->sc_data.algo, + rr_data->ms_data.kc); + } + else + { + dat_code_mph_ciphering_req (CIPH_OFF, 0, NULL); + } + + if (ciph_cmd->ciph_res.cr EQ INC_IMEISV_YES) + { + /* + * if the response shall contain the IMEI, fill it in. + */ + ciph_comp->v_mob_ident = TRUE; + memcpy (&ciph_comp->mob_ident, &rr_data->ms_data.imei, + sizeof (T_mob_ident)); + } + else + { + ciph_comp->v_mob_ident = FALSE; + } + + ciph_comp->msg_type = U_CIPH_COMP; + + /* + * send response to the network + */ + for_dat_data_req (dl_data_req); + + /* + * Indicate changed ciphering mode to MM. + * Any supression of ciphering information to MMI/ACI will + * be done by the upper layers. + */ + dat_code_ciphering_to_mm (rr_data->sc_data.ciph_on); + + EM_CIPHERING_CMD; + } + } + + PFREE (dl_data_ind); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_freq_redef | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of a frequency redefinition message. + +*/ + +GLOBAL void dat_for_freq_redef (T_DL_DATA_IND *dl_data_ind, + T_D_FREQ_REDEF *freq_redef, + T_LIST *cell_chan_desc) +{ + GET_INSTANCE_DATA; + T_start start; + T_LIST hop_list; + UBYTE mob_alloc[65]; + + TRACE_FUNCTION ("dat_for_freq_redef()"); + + if (dat_check_error_flag (SEND_RR_STATUS)) + { + /* + * the check in the formatter has passed + */ + memcpy (&rr_data->sc_data.chan_desc, + &freq_redef->chan_desc, + sizeof (T_chan_desc)); + + /* + * convert the mobile allocation from the message format + * to a list of 1-bit positions to build the hopping list. + */ + att_bits_to_byte (mob_alloc, + freq_redef->mob_alloc.c_mac, + freq_redef->mob_alloc.mac); + + dat_set_last_used_channel (&rr_data->sc_data.chan_desc); + + if (freq_redef->v_cell_chan_desc) + { + /* + * if the message contains a new cell channel description + * copy the new one, else use the old one. + */ + srv_copy_list (&rr_data->sc_data.cd.cell_chan_desc, + cell_chan_desc, + sizeof (T_LIST)); + rr_data->sc_data.cd.v_cell_chan_desc = WITH_CONTENT; + } + + if (rr_data->sc_data.cd.v_cell_chan_desc NEQ NO_CONTENT) + { + /* + * create the hopping list from cell channel description and + * mobile allocation. + */ + if(! srv_create_chan_mob_alloc (&rr_data->sc_data.cd.cell_chan_desc, + &hop_list, + mob_alloc)) + { + + U32 st = 51*((26 + freq_redef->start_time.t3 - freq_redef->start_time.t2 )%26) + + freq_redef->start_time.t3 + 1326*freq_redef->start_time.t1; + U32 ct = dl_data_ind->fn%STARTING_TIME_INTERVAL; + +#if defined(_SIMULATION_) + TRACE_EVENT_WIN_P5 ("D_FREQ_REDEF: t1=%u t2=%u t3=%u, st=%u, ct=%u", + freq_redef->start_time.t1, freq_redef->start_time.t2, + freq_redef->start_time.t3, st, ct); + TRACE_EVENT_WIN_P2 ("D_FREQ_REDEF: (st-ct) %u <= %u ?", + ((STARTING_TIME_INTERVAL + st - ct)%STARTING_TIME_INTERVAL), STARTING_TIME_INTERVAL1); +#endif + + if(((STARTING_TIME_INTERVAL + st - ct)%STARTING_TIME_INTERVAL) <= STARTING_TIME_INTERVAL1) + { + /*XXX this should only be done if the starting time has not yet expired */ + dat_send_rr_status_msg(RRC_FREQ_NOT_IMPL); + } + else + { + /* + * 3GPP TS 04.18, section 3.4.5.1 + * Frequency redefinition procedure, abnormal cases: + * If the mobile station receives a FREQUENCY REDEFINITION message + * with a Mobile Allocation IE indexing frequencies that are not all + * in one band and a Starting Time IE indicating a time that has + * elapsed, then the mobile station shall locally abort the radio + * connection and, if permitted, attempt Call Re-establishment. + * + * Inform MM about a radio link failure and start cell reselection. + * It would be possible to create a new cause but RLF does exactly + * what is needed and this is really 'some kind of' RLF. + */ + rr_data->net_lost = TRUE; + att_code_rr_abort_ind (RRCS_ABORT_RAD_LNK_FAIL); + att_stop_dedicated(); + } + } + else + { + /* + * copy start time for the new hopping list + */ + start.v_start = TRUE; + start.t1 = freq_redef->start_time.t1; + start.t2 = freq_redef->start_time.t2; + start.t3 = freq_redef->start_time.t3; + + /* + * configure layer 1 with the new hopping list + */ + dat_code_mph_freq_redef_req (&start, + &hop_list); + } + } + } + + PFREE (dl_data_ind); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_handov_cmd | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of a handover command message. + +*/ + +GLOBAL void dat_for_handov_cmd (T_DL_DATA_IND *dl_data_ind, + T_D_HANDOV_CMD *handov_cmd, + T_LIST *cell_chan_desc, + T_LIST *hop_list_after, + T_LIST *hop_list_before) +{ + GET_INSTANCE_DATA; + UBYTE mob_alloc [65]; + + TRACE_FUNCTION ("dat_for_handov_cmd()"); + + rr_data->dyn_config.fho = 0; + + /* + * dynamic configuration command : IHO + * Lock the DUT to the cell it is already camping - + * Ignore the Handover command message and send an + * Handover failure message to the network. + */ + + if(rr_data->dyn_config.iho AND (rr_data->sc_data.ch_mode EQ CM_AMR)) + { + TRACE_EVENT("D_HANDOV_CMD : IHO"); + for_set_content_error (RRC_CHANNEL_MODE); + } + + switch (rr_data->ms_data.error.cs) + { + /* + * in case of mandatory info element error, + * the message is returned immediately. + */ + /* case RRC_INVALID_MAN_INFO: this value is currently never set */ + case RRC_COND_IE_ERROR: + { + /* + * build a RR status message. + */ + dat_send_rr_status_msg(rr_data->ms_data.error.cs); + PFREE (dl_data_ind); + break; + } + + case RRC_INCORRECT_MSG: + { + /* + * If a structurally correct message has been detected, + * containing erroneous data, an Assignment Failure message + * is sent back. + */ + + /* + * Even though it's not possible to go onto the new channel + * we still need to suspend the current link and send the + * HANDOVER FAILURE command via a priority DL_RECONNECT_REQ. + * This ensures DL will halt processing anything in its + * buffer until it has sent this message onto the nw + */ + for_suspend_layer_2 (); + dat_send_handov_fail_msg(rr_data->ms_data.error.val); + + RR_EM_SET_HANDOVER_FAIL_CAUSE(rr_data->ms_data.error.val); + + PFREE (dl_data_ind); + break; + } + + default: + /* + * the message check has passed. + * first of all suspend current layer 2 link + */ + + TRACE_EVENT_P1 ("HO:default (%02x)", rr_data->ms_data.error.cs); + + for_suspend_layer_2 (); + + /* + * set for the optional information elements + * of the handover message the default value + * to the current serving cell value. + */ + rr_data->cr_data.ch_mode = rr_data->sc_data.ch_mode; + rr_data->cr_data.ciph_on = rr_data->sc_data.ciph_on; + rr_data->cr_data.algo = rr_data->sc_data.algo; + rr_data->cr_data.cd.v_cell_chan_desc = + rr_data->sc_data.cd.v_cell_chan_desc; + memcpy (&rr_data->cr_data.cd.cell_chan_desc, + &rr_data->sc_data.cd.cell_chan_desc, + sizeof (T_LIST)); + + /* + * if AMR is supported set the default values + * to the current serving cell values. + */ + if(rr_data->sc_data.ch_mode EQ CM_AMR) + { + memcpy(&rr_data->cr_data.amr_conf, + &rr_data->sc_data.amr_conf, + sizeof (T_multirate_conf)); + } + else { + /* + * AMR is not supported, therefore set some dummy values. This is necessary because + * the later Layer1 configuration must include an AMR configuration!! + */ + memset(&rr_data->cr_data.amr_conf, 0, sizeof (T_multirate_conf)); + } + + /* + * set BSIC, BCCH channel number and channel description from + * the handover command. + */ + rr_data->nc_data[CR_INDEX].bsic = (handov_cmd->cell_desc.ncc << 3) + + handov_cmd->cell_desc.bcc; + rr_data->nc_data[CR_INDEX].arfcn = handov_cmd->cell_desc.bcch_arfcn_lo + + (handov_cmd->cell_desc.bcch_arfcn_hi << 8); + memcpy (&rr_data->cr_data.chan_desc, + &handov_cmd->chan_desc_after, + sizeof (T_chan_desc)); + + if (handov_cmd->v_synch_ind) + { + /* + * store the Handover synchronisation information if available. + */ + memcpy (&rr_data->ms_data.ho_type, &handov_cmd->synch_ind, + sizeof (T_synch_ind)); + } + else + { + /* + * else set the values to the default values. + */ + rr_data->ms_data.ho_type.rot = TIME_DIFF_NO; + rr_data->ms_data.ho_type.nci = TRUE; + rr_data->ms_data.ho_type.si = SYI_NON_SYNCH; + } + + if (rr_data->ms_data.ho_type.si EQ SYI_PSEUDO_SYNCH AND + ! rr_data->ms_data.classmark2.ps) + { + /* + * if the handover requests a pseudo synchronized handover + * and the mobile does not support this, a handover failure + * message is send and the procedure is aborted with + * reconnection to the old channel. + */ + dat_send_handov_fail_msg(RRC_INCORRECT_MSG); + + RR_EM_SET_HANDOVER_FAIL_CAUSE(RRC_INCORRECT_MSG); + + PFREE (dl_data_ind); + return; + } + else + { + if (handov_cmd->v_cell_chan_desc) + { + /* + * if the handover command contains a new cell channel description + * copy the new list. + */ + srv_copy_list (&rr_data->cr_data.cd.cell_chan_desc, + cell_chan_desc, + sizeof (T_LIST)); + rr_data->cr_data.cd.v_cell_chan_desc = WITH_CHANGED_CONTENT; + } + + if (handov_cmd->v_chan_mode) + { + /* + * store a new channel mode if available. + */ + rr_data->cr_data.ch_mode = handov_cmd->chan_mode; + } + + + /* + * If AMR is signalled check if new multi-rate speech codec is part of the handover cmd + * otherwise use default values set earlier. If AMR is not signalled use the dummy values + * instead either set earlier. + */ + +/* Implements RR Clone findings #9 */ + dat_cr_data_multirate_conf(handov_cmd->v_multirate_conf, &handov_cmd->multirate_conf); + + + if (handov_cmd->v_ciph_mode_set) + { + /* + * if the message contains cipher mode parameter + * copy the parameters + */ + rr_data->cr_data.ciph_on = handov_cmd->ciph_mode_set.sc; + rr_data->cr_data.algo = handov_cmd->ciph_mode_set.algo_ident; + + /* + * if ciphering is already enabled and the handover command + * requests ciphering again, the procedure is aborted with + * a handover failure message. + */ + if (rr_data->cr_data.ciph_on EQ CIPH_ON AND + rr_data->sc_data.ciph_received EQ FALSE) + { + dat_send_handov_fail_msg(RRC_PROT_UNSPECIFIED); + + RR_EM_SET_HANDOVER_FAIL_CAUSE(RRC_PROT_UNSPECIFIED); + + PFREE (dl_data_ind); + return; + } + } + + if(handov_cmd->v_mob_alloc_after) + { + if(dat_for_handover_mob_alloc(mob_alloc, hop_list_after, (T_VOID_STRUCT *) &handov_cmd->mob_alloc_after,dl_data_ind)) + return; + } + + if(handov_cmd->v_mob_alloc_before) + { + if(dat_for_handover_mob_alloc(mob_alloc, hop_list_before, (T_VOID_STRUCT *) &handov_cmd->mob_alloc_before,dl_data_ind)) + return; + } + + /* + * Handover resets a SAPI 3 connection for SMS + */ + SET_STATE (STATE_SAPI_3, SMS_IDLE); + PFREE (dl_data_ind); + { + /* + * All Parameters are checked + * Now the handover is started + */ + PALLOC (dedicated_req, MPH_DEDICATED_REQ); + + memset (dedicated_req, 0, sizeof (T_MPH_DEDICATED_REQ)); + + if (handov_cmd->v_start_time) + { + /* + * copy starting time if available. + */ + dedicated_req->start.v_start = TRUE; + dedicated_req->start.t1 = handov_cmd->start_time.t1; + dedicated_req->start.t2 = handov_cmd->start_time.t2; + dedicated_req->start.t3 = handov_cmd->start_time.t3; + } + + /* + * Calculate observed time difference + */ + { + UBYTE i1; + ULONG fn_offset; + + rr_data->sc_data.observed_ta = 0; + for (i1 = 0; i1< rr_data->ms_data.measurement_report.ncells.no_of_ncells; i1++) + { + /* + * find the handover cell inn the neighbourcells of the last measurement report + */ + if (rr_data->nc_data[CR_INDEX].arfcn EQ + rr_data->ms_data.measurement_report.ncells.arfcn[i1]) + { + /* + * According to 05.10 OTD is defined as the timing difference + * between BTS0 and BTS1 ( system time in BTS 0 minus that of + * BTS 1..), with BTS1 as the new cell (neighbour cell, HO + * Target cell) and BTS0 and the current cell (serving cell) + */ + fn_offset = (HYPERFRAME - + rr_data->ms_data.measurement_report.ncells.frame_offset[i1]) + % HYPERFRAME; + + /* + * calculate the observed time difference from the relative + * time difference of neighbourcell and serving cell + * (given by time_alignment and frame offset) and the observed + * time difference of the serving cell (coming from timing + * advance in layer 1 header of the downlink SACCH messages). + */ + /* + * A.1.3 of 3GPP TS 05.10 + * after successful handover, either synchronized, + * non-synchronized or pseudo-synchronized, the MS shall + * provide to BTS 1 the value of OTD + t0 in the + * "HANDOVER COMPLETE" message. + * + * NOTE : measurement_report.otd is the TA sent by the + * network in downlink SACCH. TA is roundtrip propogation + * delay in bit periods. + * t0 denotes the "one way" line of sight propagation delay + * between the MS and BTS 0, in "half bits". + * t0 = measurement_report.otd * 2 / 2. + */ + rr_data->sc_data.observed_ta = + ( (rr_data->ms_data.measurement_report.ncells.time_alignmt[i1] + + fn_offset* QUARTER_BITS_PER_FRAME)/2 + + rr_data->ms_data.measurement_report.otd ) % 2097152; + } + } + } + + dedicated_req->ho_param.ho_nci = rr_data->ms_data.ho_type.nci; + + /* + * Set the handover mode + */ + switch (rr_data->ms_data.ho_type.si) + { + case SYI_NON_SYNCH: + /* + * asynchronous handover + */ + dedicated_req->mod = MODE_ASYNC_HANDOVER; + break; + + case SYI_NORM_SYNCH: + /* + * synchronous handover + */ + rr_data->sc_data.new_ta = rr_data->ms_data.measurement_report.otd/2; + att_set_tim_advance_info(); + dedicated_req->mod = MODE_SYNC_HANDOVER; + break; + + case SYI_PRE_SYNCH: + /* + * pre-synchronized handover + */ + dedicated_req->mod = MODE_PRE_SYNC_HANDOVER; + + if (handov_cmd->v_time_advance) + /* + * if the handover command contains a new timing advance + */ + dedicated_req->tr_para.tav = handov_cmd->time_advance.ta; + else + /* + * else set the default value 1 + */ + dedicated_req->tr_para.tav = 1; + + rr_data->sc_data.new_ta = dedicated_req->tr_para.tav * 2; + att_set_tim_advance_info(); + break; + + case SYI_PSEUDO_SYNCH: + /* + * pseudo-synchronized handover + */ + dedicated_req->mod = MODE_PSEUDO_SYNC_HANDOVER; + dedicated_req->tr_para.tav = handov_cmd->time_diff; + rr_data->sc_data.new_ta = dedicated_req->tr_para.tav; + att_set_tim_advance_info(); + break; + } + + /* + * Set Channel Type + */ + dedicated_req->ch_type.ch = handov_cmd->chan_desc_after.chan_type; + dedicated_req->ch_type.tn = handov_cmd->chan_desc_after.tn; + dedicated_req->ch_type.tsc = handov_cmd->chan_desc_after.tsc; + dedicated_req->ch_type.h = handov_cmd->chan_desc_after.hop; + + if (handov_cmd->chan_desc_after.hop EQ H_NO) + dedicated_req->ch_type.arfcn = handov_cmd->chan_desc_after.arfcn; + else + { + dedicated_req->ch_type.maio = handov_cmd->chan_desc_after.maio; + dedicated_req->ch_type.hsn = handov_cmd->chan_desc_after.hsn; + + /* CSI-LLD section:4.1.1.11 + * This function Updates the black list with the MA list received + * in the handover command + */ + cs_remove_BA_MA_from_black_list(rr_data->cs_data.region , hop_list_after); + + srv_create_list (hop_list_after, dedicated_req->ch_type.ma, + MAX_MA_CHANNELS, TRUE, 0); + } + + dedicated_req->bsic = rr_data->nc_data[CR_INDEX].bsic & 0x3F; + dedicated_req->arfcn = rr_data->nc_data[CR_INDEX].arfcn; + dedicated_req->ho_param.ho_ref = handov_cmd->handov_ref; + dedicated_req->ho_param.ho_pow = handov_cmd->pow_cmd_access.pow; + dedicated_req->ho_param.ho_acc_type = handov_cmd->pow_cmd_access.atc; + dedicated_req->tr_para.mode = rr_data->cr_data.ch_mode; + + /* + * Set multi-rate speech codec + */ + dedicated_req->amr_conf.nscb = rr_data->cr_data.amr_conf.nscb; + dedicated_req->amr_conf.icmi = rr_data->cr_data.amr_conf.icmi; + dedicated_req->amr_conf.st_mode = rr_data->cr_data.amr_conf.st_mode; + dedicated_req->amr_conf.acs = rr_data->cr_data.amr_conf.set_amr; + + /* + * valid flag for the threshold and hystersis values. amr_conf.c_cod_prop + * defines the number of threshold and hystersis values. + */ + dedicated_req->amr_conf.v_cod_prop = rr_data->cr_data.amr_conf.v_cod_prop; + if(dedicated_req->amr_conf.v_cod_prop) + { + int i; + dedicated_req->amr_conf.c_cod_prop = rr_data->cr_data.amr_conf.c_cod_prop; + for(i=0; i<dedicated_req->amr_conf.c_cod_prop; i++) + memcpy(&dedicated_req->amr_conf.cod_prop[i], &rr_data->cr_data.amr_conf.cod_prop[i], sizeof(T_cod_prop)); + } + + /* + * Set Channel Type before starting time + */ + + if (handov_cmd->v_chan_desc_before EQ FALSE) + dedicated_req->ch_type2.ch = NOT_PRESENT_8BIT; + else + { + +/* Implements RR Clone findings #22 */ + dat_dedicated_req_ch_type2(&dedicated_req->ch_type2, &handov_cmd->chan_desc_before, + hop_list_before); + } + + if (rr_data->cr_data.ciph_on) + { + /* + * set cipher parameter + */ + dedicated_req->ciph.stat = rr_data->cr_data.ciph_on; + dedicated_req->ciph.algo = rr_data->cr_data.algo; + memcpy (dedicated_req->ciph.kc, rr_data->ms_data.kc, KC_STRING_SIZE); + } + + /* + * clear neighbourcell lists for the new cell. + */ + srv_clear_list (&rr_data->sc_data.cd.ncell_list); + srv_clear_list (&rr_data->sc_data.five_ter_list); + + att_clean_buf (IND_ALL_DEDI_SI); + + rr_data->sc_data.cd.sys_info_read &= ~ALL_DEDI_SYS_INFOS; + + SET_STATE (STATE_DAT, DAT_HANDOVER); + +#if defined (REL99) && defined (TI_PS_FF_EMR) + /*clear EMR parameters also, if present*/ + if (rr_data->sc_data.enh_para_status EQ ENH_PARA_DEDICATED ) + { + /*discard the enhanced para that were related to BA(SACCH) before HO*/ + rr_data->sc_data.enh_para_status = ENH_PARA_INVALID_STATE; + memset (rr_data->sc_data.rep_count, NOT_PRESENT_8BIT, MAX_NEIGHBOURCELLS); + for_set_default_emr_data(&rr_data->sc_data.emr_data_current); + /*Indicate to ALR that enhanced para are invalid*/ + attf_send_enh_para_to_alr(rr_data->sc_data.emr_data_current.rep_type, + &rr_data->sc_data.emr_data_current.enh_para); + } +#endif + + RR_EM_GET_HOPPING_CHANNEL (dedicated_req->ch_type.ma, dedicated_req->ch_type2.ma, + dedicated_req->start.v_start,dedicated_req->ch_type2.maio); + + EM_HANDOVER_CMD; + +#if defined FF_EOTD + if ( rr_data->eotd_req_id NEQ NOT_PRESENT_16BIT ) + { + PALLOC (rrlc_error_ind, RRLC_ERROR_IND); + rrlc_error_ind->cause = LCS_HANDOVER; + + rr_data->eotd_req_id = NOT_PRESENT_16BIT; + + PSENDX (LC, rrlc_error_ind); + } +#endif /* FF_EOTD */ + + /* + * configure layer 1. + */ + PSENDX (PL, dedicated_req); + } + } /* else */ + } /* switch */ +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : stop_rach_and_enter_idle | ++--------------------------------------------------------------------+ + + PURPOSE : Invalid frequency list received during Immediate Assignment + procedure. The sending of Channel Request messages is + stopped and Idle Mode entered. +*/ + +LOCAL void stop_rach_and_enter_idle(void) +{ + PALLOC (mph_random_access_req, MPH_RANDOM_ACCESS_REQ); + + TRACE_ERROR ("invalid frequencies (Frequency Hopping)"); + + TIMERSTOP (T3122); + TIMERSTOP (T3126); + + /* + * Stop sending Random Burst + */ + memset (&mph_random_access_req->send_mode, 0, sizeof (T_send_mode)); + PSENDX (PL, mph_random_access_req); + + dat_send_release_ind (RRCS_INVALID_HOP_FREQ); + +/*SET_STATE (STATE_DAT, DAT_IDLE); + att_build_idle_req (SC_INDEX, MODE_CELL_SELECTION);*/ + +#ifdef GPRS + att_start_cell_reselection_gprs (CELL_RESELECTION_RACH); +#else + att_start_cell_reselection (CELL_RESELECTION_RACH); +#endif +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_imm_assign | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of an immediate assignment message. + +*/ + +GLOBAL void dat_for_imm_assign (T_MPH_UNITDATA_IND *mph_unitdata_ind, + T_D_IMM_ASSIGN *imm_assign) +{ + GET_INSTANCE_DATA; + T_SC_DATA *rrd = &rr_data->sc_data; + T_start start; + UBYTE mob_alloc [65]; + T_LIST hop_list_bef; + T_LIST hop_list_after; + UBYTE maio; + T_IA_REST ia_rest; + UBYTE index = 0; + + TRACE_FUNCTION ("dat_for_imm_assign()"); + + switch (GET_STATE (STATE_DAT)) + { +#ifdef GPRS + case DAT_IDLE: + TRACE_EVENT("check dl idle"); + dat_check_imm_assign_pch (mph_unitdata_ind, imm_assign); + break; +#endif + + case DAT_IMM_ASS: + if (dat_check_error_flag (SEND_NO_RR_STATUS)) + { +#ifdef GPRS + TRACE_EVENT("check dl pa"); + if(GET_STATE(STATE_GPRS) EQ GPRS_PAM_BCCH AND + dat_check_imm_assign_pch(mph_unitdata_ind, imm_assign)) + return; +#endif + if (dat_compare_request_ref (&imm_assign->req_ref, &index)) + { + /* + * the request reference in the immediate assignment + * message matches to one of the last three channel + * request messages. + */ + TRACE_EVENT("matched"); + /* + * check the channel description + */ + if(imm_assign->v_chan_desc) + for_check_channel_descr (&imm_assign->chan_desc); + + /* was channel description ok? */ + if(!dat_check_error_flag (SEND_NO_RR_STATUS)) + return; +#ifdef GPRS + if(dat_check_gprs_imm_ass (mph_unitdata_ind, + imm_assign, + index)) + return; +#endif + if (imm_assign->v_chan_desc) + { + if (imm_assign->chan_desc.hop AND + imm_assign->mob_alloc.c_mac) + { + TRACE_EVENT ("create mob alloc (after st)"); + /* + * if the message contains a mobile allocation + * build a hopping list together with the cell + * channel description of system information + * type 1 message. + */ + att_bits_to_byte (mob_alloc, + imm_assign->mob_alloc.c_mac, + imm_assign->mob_alloc.mac); + if( rrd->cd.v_cell_chan_desc EQ NO_CONTENT OR + !srv_create_chan_mob_alloc (&rrd->cd.cell_chan_desc, + &hop_list_after, + mob_alloc)) + { + stop_rach_and_enter_idle(); + return; + } + } + else + { + /* + * else clear the hopping list + */ + srv_clear_list (&hop_list_after); + } + } + else + { + TRACE_EVENT("IMM ASS discarded: neither sent to GRR nor channel description found"); + return; /* for non-packet access we need a channel description */ + } + + TRACE_EVENT("now get started"); + if (imm_assign->v_start_time) + { + /* + * if the message contains a starting time, + * store the starting time. + */ + start.v_start = TRUE; + start.t1 = imm_assign->start_time.t1; + start.t2 = imm_assign->start_time.t2; + start.t3 = imm_assign->start_time.t3; + } + else + { + /* + * clear the starting time. + */ + memset (&start, 0, sizeof (T_start)); + } + + /* + * decode IA Rest Octet + */ + memset (&ia_rest, 0, sizeof (T_IA_REST)); + + ia_rest.ia_p = imm_assign->ia_rest_oct.flag_2bit; + ia_rest.ia_maio = imm_assign->ia_rest_oct.ia_freq_par.maio; + ia_rest.c_ia_mac = imm_assign->ia_rest_oct.ia_freq_par.c_mac; + if (ia_rest.c_ia_mac > 9) + ia_rest.c_ia_mac = 9; + + memcpy (ia_rest.ia_mac, + &imm_assign->ia_rest_oct.ia_freq_par.mac, + ia_rest.c_ia_mac); + + if (imm_assign->v_start_time AND + imm_assign->chan_desc.hop AND + ia_rest.ia_p EQ 2) + { + /* + * calculate frequency list before starting time + */ + TRACE_EVENT("create mob alloc (before st)"); + maio = ia_rest.ia_maio; + + att_bits_to_byte (mob_alloc, + ia_rest.c_ia_mac, + ia_rest.ia_mac); + if(rrd->cd.v_cell_chan_desc EQ NO_CONTENT OR + !srv_create_chan_mob_alloc (&rrd->cd.cell_chan_desc, + &hop_list_bef, + mob_alloc)) + { + stop_rach_and_enter_idle(); + return; + } + } + else + { + maio = 0; + srv_clear_list (&hop_list_bef); + } + + + /* + * stop T3122 and T3126 if they are running. + */ + TIMERSTOP (T3122); + TIMERSTOP (T3126); + SET_STATE (STATE_DAT, DAT_IMM_ASS_1); + + /* + * store channel description + */ + memcpy (&rrd->chan_desc, &imm_assign->chan_desc, + sizeof (T_chan_desc)); + + /* + * the initial channel mode is always signalling only + */ + rrd->ch_mode = MODE_SIG_ONLY; + + /* + * set the timing advance + */ + rrd->new_ta = imm_assign->time_advance.ta; + att_set_tim_advance_info(); + dat_set_last_used_channel (&rrd->chan_desc); + + + /* + * configure layer 1 + */ + dat_code_mph_imm_assign_req (&start, + rr_data->nc_data[SC_INDEX].select_para.ms_txpwr_max_cch, + maio, + &hop_list_after, + &hop_list_bef); + } + EM_IMMEDIATE_ASSIGNMENT; + } + break; + + default: + break; + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_imm_assign_ext | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of the immediate assignment extended message. + +*/ + + +GLOBAL void dat_for_imm_assign_ext (T_MPH_UNITDATA_IND *mph_unitdata_ind, + T_D_IMM_ASSIGN_EXT *imm_assign_ext) +{ + GET_INSTANCE_DATA; + USHORT i; + T_start start; + UBYTE mob_alloc [65]; + T_SC_DATA *rrd = &rr_data->sc_data; + T_LIST hop_list_after; + T_LIST hop_list_bef; + UBYTE index; + T_chan_desc *p_chan_desc; + + TRACE_FUNCTION ("dat_for_imm_assign_ext()"); + + switch (GET_STATE (STATE_DAT)) + { + case DAT_IMM_ASS: + if (dat_check_error_flag (SEND_NO_RR_STATUS)) + { + /* + * the message check in the formatter has passed + */ + for (i=0; i<2; i++) + { + /* + * the immediate assignment extended message contains + * two request references. + */ + if (dat_compare_request_ref ((i EQ 0) + ? &imm_assign_ext->req_ref + : (T_req_ref *)&imm_assign_ext->req_ref_2, + &index)) + { +#ifdef GPRS + dat_check_imm_ass_ext (mph_unitdata_ind,(UBYTE)(i+1)); +#endif + /* + * check channel description + */ + if(i EQ 0) + p_chan_desc = &imm_assign_ext->chan_desc; + else + p_chan_desc = (T_chan_desc *)&imm_assign_ext->chan_desc_2; + for_check_channel_descr (p_chan_desc); + + if (!dat_check_error_flag (SEND_NO_RR_STATUS)) + return; + /* + * the request reference in the immediate assignment + * extended message matches to one of the last three + * channel request messages. + */ + if (imm_assign_ext->mob_alloc.c_mac AND p_chan_desc->hop) + { + /* + * if the message contains a mobile allocation and + * the mobile shall hop + * build a frequency hopping list together with + * the cell channel description of system information + * type 1 message. + */ + att_bits_to_byte (mob_alloc, + imm_assign_ext->mob_alloc.c_mac, + imm_assign_ext->mob_alloc.mac); + if(rrd->cd.v_cell_chan_desc EQ NO_CONTENT OR + ! srv_create_chan_mob_alloc (&rr_data->sc_data.cd.cell_chan_desc, + &hop_list_after, + mob_alloc)) + { + stop_rach_and_enter_idle(); + return; + } + } + else + { + /* + * else clear frequency hopping list + */ + srv_clear_list (&hop_list_after); + } + + /* + * stop T3122 and T3126 if they are running. + */ + TIMERSTOP (T3122); + TIMERSTOP (T3126); + /* + * store channel description + */ + memcpy (&rrd->chan_desc, + p_chan_desc, + sizeof (T_chan_desc)); + + /* + * the initial channel mode is ever signalling only + */ + rrd->ch_mode = MODE_SIG_ONLY; + + /* + * store the new timing advance + */ + rrd->new_ta = (i EQ 0) + ? imm_assign_ext->time_advance.ta + : imm_assign_ext->time_advance_2.ta; + att_set_tim_advance_info(); + dat_set_last_used_channel (&rrd->chan_desc); + + + if (imm_assign_ext->v_start_time) + { + /* + * copy starting time if available + */ + start.v_start = TRUE; + start.t1 = imm_assign_ext->start_time.t1; + start.t2 = imm_assign_ext->start_time.t2; + start.t3 = imm_assign_ext->start_time.t3; + } + else + memset (&start, 0, sizeof (T_start)); + + srv_clear_list (&hop_list_bef); + SET_STATE (STATE_DAT, DAT_IMM_ASS_1); + + /* + * configure layer 1. + */ + dat_code_mph_imm_assign_req (&start, + rr_data->nc_data[SC_INDEX].select_para.ms_txpwr_max_cch, + 0, + &hop_list_after, + &hop_list_bef); + + EM_IMMEDIATE_ASSIGNMENT_EXT; + return; + } + } + } + break; + + default: + break; + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_imm_assign_rej | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of an immediate assignment reject message. + +*/ + +GLOBAL void dat_for_imm_assign_rej (T_D_IMM_ASSIGN_REJ *imm_assign_rej) +{ + GET_INSTANCE_DATA; + UBYTE index; + + TRACE_FUNCTION ("dat_for_imm_assign_rej()"); + + switch (GET_STATE (STATE_DAT)) + { + case DAT_IMM_ASS: + if (dat_check_error_flag (SEND_NO_RR_STATUS)) + { + /* + * the message has passed the checks in the formatter. + */ + + if (! IS_TIMER_ACTIVE(T3122)) + { + /* + * Only if T3122 is not running, that means there is no + * immediate assignment reject message taken in account + * before. + */ + BOOL result = FALSE; + UBYTE t3122 = 0; + + if (dat_compare_request_ref (&imm_assign_rej->req_ref, &index)) + { + /* + * if the request reference matches to one of the last + * three channel requests, set the result of TRUE and + * store the timer value. This is checked for the up to + * four request references in the message. + */ + result = TRUE; + t3122 = imm_assign_rej->t3122; + } + + else if (dat_compare_request_ref ((T_req_ref *)&imm_assign_rej->req_ref_2, &index)) + { + result = TRUE; + t3122 = imm_assign_rej->t3122_2; + } + + else if (dat_compare_request_ref ((T_req_ref *)&imm_assign_rej->req_ref_3, &index)) + { + result = TRUE; + t3122 = imm_assign_rej->t3122_3; + } + + else if (dat_compare_request_ref ((T_req_ref *)&imm_assign_rej->req_ref_4, &index)) + { + result = TRUE; + t3122 = imm_assign_rej->t3122_4; + } + + if (result) + { + /* + * a request reference has matched + */ +#ifdef GPRS + if (dat_check_imm_ass_rej (t3122) EQ FALSE) +#endif + if (t3122 NEQ 0) + { + /* + * start T3122 if a value is defined + */ + TIMERSTART (T3122, T3122_VALUE(t3122)); + } + TRACE_EVENT("set rej_rec"); + rr_data->imm_ass_rej_rec = TRUE; + rr_data->ms_data.all_conf_received = TRUE; + + /* + * Start T3126 if the timer is not running yet. + */ +/* Implements Measure#32: Row 217,218 */ + (IS_TIMER_ACTIVE(T3126)) ? + TRACE_TIMER ( "T3126 re-start") : TRACE_TIMER ( "T3126 start"); + + + if (! IS_TIMER_ACTIVE(T3126)) + { + TIMERSTART (T3126, T3126_VALUE); + /* + * Stop sending Random Burst + */ + { + PALLOC (mph_random_access_req, MPH_RANDOM_ACCESS_REQ); + + memset (&mph_random_access_req->send_mode, 0, sizeof (T_send_mode)); + PSENDX (PL, mph_random_access_req); + } + } + } + EM_IMMEDIATE_ASSIGNMENT_REJECT; + } + } + break; + + default: + break; + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_ext_meas_order | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of an extended measurement order message. + +*/ + +GLOBAL void dat_for_ext_meas_order (T_D_EXT_MEAS_ORDER *ext_meas_order) +{ + GET_INSTANCE_DATA; + UBYTE new_seq; + T_LIST *chan_list; + + PALLOC (mph_emo_req, MPH_EMO_REQ); + + TRACE_FUNCTION ("dat_for_ext_meas_order()"); + + /* + * Decode EMO: new_seq + */ + + ccd_decodeByte (ext_meas_order->ext_meas_freq.b_ext_meas_freq, + (USHORT)(ext_meas_order->ext_meas_freq.o_ext_meas_freq+3), + 1, &new_seq); + + /* + * Ignore EMO if EMO proc already running and new SEQ EQ current SEQ. + */ + + if ( rr_data->emo_arfcn NEQ NULL AND rr_data->emo_seq EQ new_seq ) + { + PFREE ( mph_emo_req ); + return; + } + + /* + * Decode and store EMO frequency list + */ + + if ( rr_data->emo_arfcn EQ NULL ) + { + MALLOC ( rr_data->emo_arfcn, MAX_EMO_CHANNELS * sizeof (rr_data->emo_arfcn[0]) ); + } + rr_data->emo_seq = new_seq; + MALLOC ( chan_list, sizeof ( T_LIST ) ); + for_create_channel_list ( (T_f_range*) &ext_meas_order->ext_meas_freq, chan_list); + + /* + * Function srv_create_list_dedicated ensures that the frequencies are sorted + * and the number of frequencies are limited to 21 frequencies + */ + + rr_data->c_emo_arfcn = srv_create_list (chan_list, + rr_data->emo_arfcn, + MAX_EMO_CHANNELS, + FALSE, + 0); + MFREE ( chan_list ); + + memcpy ( &mph_emo_req->arfcn[0], + &rr_data->emo_arfcn[0], + rr_data->c_emo_arfcn * sizeof (rr_data->emo_arfcn[0]) ); + + mph_emo_req->c_arfcn = + srv_remove_frequencies_in_array_gen ( &mph_emo_req->arfcn[0], rr_data->c_emo_arfcn ); + + /* + * Create newBA_ID, save as currentBA_ID. + */ + + rr_data->ba_id = RR_ALLOCATE_NEW_BA ( rr_data->ba_id ); + mph_emo_req->ba_id = rr_data->ba_id; + + /* + * EMOtime = 10 seconds and request PL to perform extended measurement. + */ + + TIMERSTART (TIM_EXT_MEAS, 10000); + + PSENDX (PL, mph_emo_req); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_mph_emo_meas_ind | ++--------------------------------------------------------------------+ + + PURPOSE : Measurement report for the Extended Measurment procedure + has been received. + +*/ + +GLOBAL void dat_mph_emo_meas_ind (T_MPH_EMO_MEAS_IND *mph_emo_meas_ind) +{ + GET_INSTANCE_DATA; + TRACE_FUNCTION ("att_mph_emo_meas_ind()"); + + switch (GET_STATE (STATE_DAT)) + { + case DAT_DEDICATED: + if ( rr_data->emo_arfcn NEQ NULL AND mph_emo_meas_ind->ba_id EQ rr_data->ba_id ) + { + dat_code_ext_meas_report (mph_emo_meas_ind); + dat_emo_stop ( TRUE ); + } + else + { + /* + * Build an invalid measurement reports + */ + MCAST (meas, U_MEAS_REP); + PALLOC_MSG (dl_unitdata_req, DL_UNITDATA_REQ, U_MEAS_REP); + memset (&dl_unitdata_req->sdu.buf[0], 0, dl_unitdata_req->sdu.o_buf / BITS_PER_BYTE); + + memset (meas, 0, sizeof (T_U_MEAS_REP)); + meas->msg_type = U_MEAS_REP; + meas->meas_result.meas_valid = 1; + for_dat_unitdata_req (dl_unitdata_req); + } + break; + + default: + break; + } + PFREE (mph_emo_meas_ind); +} + + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_emo_stop | ++--------------------------------------------------------------------+ + + PURPOSE : Stop the Extended Measurement Order procedure. + +*/ + +GLOBAL void dat_emo_stop (BOOL send_ncell_req ) +{ + GET_INSTANCE_DATA; + if ( rr_data->emo_arfcn NEQ NULL ) + { + MFREE (rr_data->emo_arfcn); + rr_data->emo_arfcn = NULL; + + /* restore the neighbour cell description which was used prior EMO */ + if ( send_ncell_req AND + (rr_data->sc_data.cd.sys_info_read & (SYS_INFO_5_READ | SYS_INFO_5BIS_READ | SYS_INFO_5TER_READ) ) ) + + att_code_mph_ncell_req_dedicated(); + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_l3_data_ind | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of a layer 3 message for upper layers. + +*/ + +GLOBAL void dat_for_l3_data_ind (T_DL_DATA_IND *dl_data_ind) +{ + GET_INSTANCE_DATA; + /* RR_DATA_IND is not the same as DL_DATA_IND anymore because of the new + * member fn (frame number) of T_DL_DATA_IND which is not contained in + * T_RR_DATA_IND. + */ + PALLOC_SDU(rr_data_ind, RR_DATA_IND, (USHORT)(dl_data_ind->sdu.l_buf+dl_data_ind->sdu.o_buf)); + + TRACE_FUNCTION ("dat_for_l3_data_ind()"); + + if (dl_data_ind->sapi EQ SAPI_3) + { + /* + * if it is a SMS message, this is implicitly an + * indication for an established SAPI 3 link + */ + SET_STATE (STATE_SAPI_3, SMS_ESTABLISHED); + } + + rr_data_ind->sdu.l_buf = dl_data_ind->sdu.l_buf; + rr_data_ind->sdu.o_buf = dl_data_ind->sdu.o_buf; + memcpy(rr_data_ind->sdu.buf, dl_data_ind->sdu.buf, + (dl_data_ind->sdu.l_buf+dl_data_ind->sdu.o_buf)>>3); + PFREE (dl_data_ind); + /* + * forward the message to MM for distribution + */ + PSENDX (MM, rr_data_ind); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_FOR | +| STATE : code ROUTINE : dat_for_open_loop_cmd | ++--------------------------------------------------------------------+ + + PURPOSE : Reception of a TCH OPEN LOOP COMMAND message. + +*/ + +GLOBAL void dat_for_open_loop_cmd (T_DL_DATA_IND *dl_data_ind) +{ + GET_INSTANCE_DATA; + TRACE_FUNCTION ("dat_for_open_loop_cmd()"); + + if (dat_test_sim_available () OR !dat_check_sim_available () ) + { + /* + * only if a test SIM card is inserted + */ + if (rr_data->tch_loop_subch NEQ NOT_PRESENT_8BIT) + { + /* + * A TCH Loop must be closed before, then + * open in layer 1. + */ + PREUSE (dl_data_ind, loop_req, MPH_TCH_LOOP_REQ); + loop_req->tch_loop = NOT_PRESENT_8BIT; + PSENDX (PL, loop_req); + + if(rr_data->tch_loop_subch EQ TCH_LOOP_C) + { + /* only TCH_LOOP_C is acknowledged */ + PALLOC_SDU (data_req, DL_DATA_REQ, 3*BITS_PER_BYTE); + + /* + * set channel type and SAPI for answer + */ + dat_code_prr_channel (&data_req->ch_type, + &data_req->sapi, + rr_data->sc_data.chan_desc.chan_type); + + TRACE_EVENT_P1 ( "Value of tch_loop_subch %x", rr_data->tch_loop_subch); + + /* + * do not use CCD for the response + */ + data_req->sdu.l_buf = 24; + data_req->sdu.o_buf = ENCODE_OFFSET; + /*lint -e415 -e416 Likely access of out-of-bounds pointer*/ + data_req->sdu.buf [0] = 0; + data_req->sdu.buf [1] = 0; + data_req->sdu.buf [2] = 0; + data_req->sdu.buf [3] = 0x0F; /* TI=0, PD = TST */ + data_req->sdu.buf [4] = 0x06; /* MT = Open Loop Cmd */ + data_req->sdu.buf [5] = 0x81; /* IE acknowledge */ + /*lint +e415 +e416 Likely access of out-of-bounds pointer*/ + TRACE_EVENT ("DL_DATA_REQ (RR message)"); + + EM_TCH_LOOP_OPEN; + + PSENDX (DL, data_req); + } + /* tch loop "open" */ + rr_data->tch_loop_subch = NOT_PRESENT_8BIT; + } + else + { + PFREE (dl_data_ind); + } + } + else + { + PFREE (dl_data_ind); + } +} + +#if defined FF_EOTD +/* ++------------------------------------------------------------------------------ +| Function : rr_applic_rx_init ++------------------------------------------------------------------------------ +| Description : Initialize the data structures related to +| Application Information Transfer +| Reference: 3GPP TS 04.18, 3.4.21.3 +| +| Parameters : The downlink (RX) part of the APDU structure. +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void rr_applic_rx_init ( T_APPLIC_RX *applic_rx) +{ + applic_rx->state = SAI_NULL; + if ( applic_rx->rrrrlp_data_ind NEQ NULL ) + { + PFREE ( applic_rx->rrrrlp_data_ind ) + } + applic_rx->rrrrlp_data_ind = NULL; +#ifdef REL99 + /* Send RRLP procedure stop indication to MM*/ + { + PALLOC (rr_rrlp_stop_ind, RR_RRLP_STOP_IND); + PSENDX (MM, rr_rrlp_stop_ind); + } +#endif + + TIMERSTOP ( TAPDU ); +} + +/* ++------------------------------------------------------------------------------ +| Function : rr_applic_rx_msg_store ++------------------------------------------------------------------------------ +| Description : Store the first segment of an APDU. +| Reference: 3GPP TS 04.18, 3.4.21.3.2 +| +| Parameters : The first part of the APDU. +| ++------------------------------------------------------------------------------ +*/ +LOCAL void rr_applic_rx_msg_store ( T_B_APPLIC_INFO *b_applic_info ) +{ + GET_INSTANCE_DATA; + T_APPLIC_RX *applic_rx = &rr_data->applic_rx; + T_apdu_data *apdu_data = &b_applic_info->apdu_data; + UBYTE size = apdu_data->c_apdu_info; + T_sdu *sdu; + PALLOC_SDU ( rrrrlp_data_ind, RRRRLP_DATA_IND, (USHORT)(size * BITS_PER_BYTE) ); + + if ( applic_rx->rrrrlp_data_ind NEQ NULL ) + { + TRACE_EVENT_P1 ( "APPLIC: non empty store message found", 0 ); + PFREE ( applic_rx->rrrrlp_data_ind ); + } + + applic_rx->rrrrlp_data_ind = rrrrlp_data_ind; + sdu = &applic_rx->rrrrlp_data_ind->sdu; + + memcpy ( &sdu->buf[0], apdu_data->apdu_info, size ); + sdu->l_buf = size * BITS_PER_BYTE; + sdu->o_buf = 0; +} + +/* ++------------------------------------------------------------------------------ +| Function : rr_applic_rx_msg_append ++------------------------------------------------------------------------------ +| Description : Append segments to the APDU. +| Reference: 3GPP TS 04.18, 3.4.21.3.2 +| +| Parameters : APDU segment to be appended. +| ++------------------------------------------------------------------------------ +*/ + +LOCAL int rr_applic_rx_msg_append ( T_B_APPLIC_INFO *b_applic_info ) +{ + GET_INSTANCE_DATA; + T_APPLIC_RX *applic_rx = &rr_data->applic_rx; + T_apdu_data *apdu_data = &b_applic_info->apdu_data; + T_sdu *sdu = &applic_rx->rrrrlp_data_ind->sdu; /* current APDU */ + T_sdu *sdu2; /* new APDU */ + USHORT size_cur = (USHORT)(sdu->l_buf/BITS_PER_BYTE); /* Current size of stored APDU */ + USHORT size_inf = (USHORT)apdu_data->c_apdu_info; /* size of new APDU INFOrmation */ + USHORT size_tot = (USHORT)(size_cur + size_inf); /* total APDU size after append */ + + if ( size_tot <= MAX_APDU_SIZE*BITS_PER_BYTE ) /*lint !e648 !e650/ Overflow caused by alternative defines, not applicable to target*/ + { + PALLOC_SDU ( rrrrlp_data_ind, RRRRLP_DATA_IND, (USHORT)(size_tot * BITS_PER_BYTE) ); + + sdu2 = &rrrrlp_data_ind->sdu; + + memcpy ( &sdu2->buf[ 0 ], &sdu->buf [0], size_cur ); + memcpy ( &sdu2->buf[size_cur], &apdu_data->apdu_info[0], size_inf ); + + sdu2->l_buf = (USHORT)(size_tot * BITS_PER_BYTE); + sdu2->o_buf = 0; + + PFREE ( applic_rx->rrrrlp_data_ind ); + applic_rx->rrrrlp_data_ind = rrrrlp_data_ind; + + return TRUE; + } + else + { + return FALSE; + } +} + +/* ++------------------------------------------------------------------------------ +| Function : rr_applic_rx_msg_send ++------------------------------------------------------------------------------ +| Description : Send the re-segmented APDU to RRRRLP. +| Reference: 3GPP TS 04.18, 3.4.21.3.2 +| +| Parameters : The C/R bit of the last APDU segment received. +| ++------------------------------------------------------------------------------ +*/ +LOCAL void rr_applic_rx_msg_send ( UBYTE cr ) +{ + GET_INSTANCE_DATA; + T_APPLIC_RX *applic_rx = &rr_data->applic_rx; + T_RRRRLP_DATA_IND *rrrrlp_data_ind = applic_rx->rrrrlp_data_ind; + + rrrrlp_data_ind->cr = cr; + PSENDX ( RRLP, rrrrlp_data_ind ); + applic_rx->rrrrlp_data_ind = NULL; + applic_rx->state = SAI_NULL; + TIMERSTOP ( TAPDU ); +} + +/* ++------------------------------------------------------------------------------ +| Function : dat_for_applic_info_rrlp_rx_null ++------------------------------------------------------------------------------ +| Description : Received the first segment of an APDU. +| Reference: 3GPP TS 04.18, 3.4.21.3.2 +| +| Parameters : b_applic_info: The first segment of an APDU. +| seg: the combinbed APDU control flags +| ++------------------------------------------------------------------------------ +*/ +LOCAL void dat_for_applic_info_rrlp_rx_null ( T_B_APPLIC_INFO * b_applic_info, UBYTE seg ) +{ + GET_INSTANCE_DATA; + T_APPLIC_RX *applic_rx = &rr_data->applic_rx; + + TRACE_ASSERT ( applic_rx->rrrrlp_data_ind EQ NULL ); + + switch ( seg ) + { + case FIRST_SEG | LAST_SEG : + /* Allowed, simple case. Forward segment and stay in state SAI_NULL */ + rr_applic_rx_msg_store ( b_applic_info ); +#ifdef REL99 + /* Send RRLP procedure start indication to MM*/ + { + PALLOC (rr_rrlp_start_ind, RR_RRLP_START_IND); + PSENDX (MM, rr_rrlp_start_ind); + } +#endif + rr_applic_rx_msg_send ( b_applic_info->apdu_flags.c_r ); + break; + + case FIRST_SEG | NOT_LAST_SEG: + /* Allowed, standard case of APDU segmentation. */ + /* Check length of this segment -> L2 frame must be 251 bytes, + otherwise protocol error as described in 3GPP 04.18, section 3.4.21.3.3 a) */ + + if ( b_applic_info->apdu_data.c_apdu_info EQ APDU_FULL_L2_FRAME ) + { + /* store this segment, start de-segmentation */ + rr_applic_rx_msg_store ( b_applic_info ); + TIMERSTART ( TAPDU, 2500 /* milli seconds */ ); /* 3GPP TS 04.18, 3.4.21.3.2 */ + applic_rx->state = SAI_SEGM; +#ifdef REL99 + /* Send RRLP procedure start indication to MM*/ + { + PALLOC (rr_rrlp_start_ind, RR_RRLP_START_IND); + PSENDX (MM, rr_rrlp_start_ind); + } +#endif + } + else + { + /* Protocol error occured, remain in state SAI_NULL, + discard segment (cf 3.4.21.3.3, last clause). */ + } + break; + + case NOT_FIRST_SEG | LAST_SEG : + case NOT_FIRST_SEG | NOT_LAST_SEG: + /* Not allowed. Protocol error as described in 3GPP 04.18, section 3.4.21.3.3 c), + discard segment as described in last sentence of 3.4.21.3.3 */ + break; + + default: + TRACE_EVENT_P1 ("unexpected 'default:' seg=%d", seg ); + break; + } +} + +/* ++------------------------------------------------------------------------------ +| Function : dat_for_applic_info_rrlp_rx_segm ++------------------------------------------------------------------------------ +| Description : Received second and subsequent segments of an APDU. +| Reference: 3GPP TS 04.18, 3.4.21.3.2 +| +| Parameters : b_applic_info: A segment of an APDU. +| seg: the combinbed APDU control flags +| ++------------------------------------------------------------------------------ +*/ +LOCAL void dat_for_applic_info_rrlp_rx_segm ( T_B_APPLIC_INFO * b_applic_info, UBYTE seg ) +{ + GET_INSTANCE_DATA; + T_APPLIC_RX *applic_rx = &rr_data->applic_rx; + + TRACE_ASSERT ( applic_rx->rrrrlp_data_ind NEQ NULL ); + + switch ( seg ) + { + case FIRST_SEG | LAST_SEG : + case FIRST_SEG | NOT_LAST_SEG: + + /* Abnormal case, refer to 3GPP TS 04.18, 3.4.21.3.3 b), clause 2 */ + /* Discard any partially reassembed APDU, enter state SAI_NULL */ + + rr_applic_rx_init ( applic_rx ); + + /* Now (re-)use the current segment, refer to 3GPP TS 04.18, 3.4.21.3.3: + "...reprocess any received APDU or APDU segment that caused the error... " */ + + dat_for_applic_info_rrlp_rx_null ( b_applic_info, seg ); + break; + + case NOT_FIRST_SEG | LAST_SEG : + + /* Normal case, end of re-segmentation, TAPDU stop, + send the message to the application entity. + Enter state SAI_NULL. */ + + if ( rr_applic_rx_msg_append ( b_applic_info ) EQ FALSE ) + rr_applic_rx_init ( applic_rx ); + else + rr_applic_rx_msg_send ( b_applic_info->apdu_flags.c_r ); + break; + + case NOT_FIRST_SEG | NOT_LAST_SEG: + + /* Normal case, re-segmetation is still ongoing. + If 'append' operation fails, then return to state SAI_NULL. */ + + if ( rr_applic_rx_msg_append ( b_applic_info ) EQ FALSE ) + rr_applic_rx_init ( applic_rx ); + break; + + default: + break; + } +} + + +/* ++------------------------------------------------------------------------------ +| Function : dat_for_applic_info_rrlp ++------------------------------------------------------------------------------ +| Description : Application Information Transfer (RX) for RRRLP +| Reference: 3GPP TS 04.18, 3.4.21.3 +| +| Parameters : b_applic_info: Segment of an APDU. +| ++------------------------------------------------------------------------------ +*/ +LOCAL void dat_for_applic_info_rrlp ( T_B_APPLIC_INFO * b_applic_info ) +{ + GET_INSTANCE_DATA; + T_APPLIC_RX *applic_rx = &rr_data->applic_rx; + + UBYTE seg = ((b_applic_info->apdu_flags.f_seg << 1) | + (b_applic_info->apdu_flags.l_seg ) ) & 0x03; + + switch ( applic_rx->state ) + { + case SAI_NULL : + dat_for_applic_info_rrlp_rx_null ( b_applic_info, seg ); + break; + + case SAI_SEGM : + dat_for_applic_info_rrlp_rx_segm ( b_applic_info, seg ); + break; + + default: + break; + } +} + +/* ++------------------------------------------------------------------------------ +| Function : dat_for_applic_info ++------------------------------------------------------------------------------ +| Description : Main entry point for Application Information Transfer (RX) +| Reference: 3GPP TS 04.18, 3.4.21.3 +| +| Parameters : b_applic_info: Segment of an APDU. +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void dat_for_applic_info ( T_B_APPLIC_INFO * b_applic_info ) +{ + /* + * handle RRLP, all other protocols are not supported + */ + if ( b_applic_info->apdu_id.protoc_ident EQ RRLP_LCS ) + { + dat_for_applic_info_rrlp ( b_applic_info ); + } + else + { + TRACE_EVENT_P1 ( "unsupported protocol %x", b_applic_info->apdu_id.protoc_ident ); + } +} +#endif /* FF_EOTD */ + + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_for_handover_mob_alloc | ++--------------------------------------------------------------------+ + + PURPOSE : This function generates a frequency hopping list for handover + +*/ + +LOCAL BOOL dat_for_handover_mob_alloc(UBYTE *mob_alloc, + T_LIST *hop_list_handover, + T_VOID_STRUCT *mob_alloc_handover, + T_DL_DATA_IND *dl_data_ind) +{ + GET_INSTANCE_DATA; + TRACE_FUNCTION("dat_for_handover_mob_alloc()"); + /* + * the handover command contains a mobile allocation. + * Convert the bitmap to a list of the 1-bits in the + * bitmap for generating a frequency hopping list. + */ + att_bits_to_byte (mob_alloc, + ((T_mob_alloc *)mob_alloc_handover)->c_mac, + ((T_mob_alloc *)mob_alloc_handover)->mac); + + /* + * Now create the frequency hopping list + */ + if(!srv_create_chan_mob_alloc (&rr_data->cr_data.cd.cell_chan_desc, + hop_list_handover, + mob_alloc)) + { + dat_send_handov_fail_msg(RRC_FREQ_NOT_IMPL); + + RR_EM_SET_HANDOVER_FAIL_CAUSE(RRC_FREQ_NOT_IMPL); + + PFREE (dl_data_ind); + return TRUE; + } + return FALSE; +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_cr_data_multirate_conf | ++--------------------------------------------------------------------+ + + PURPOSE : This function extracts the multirate configuration and stores + it in the rr_data. + +*/ +LOCAL void dat_cr_data_multirate_conf(U8 v_multirate_conf, T_multirate_conf *multirate_conf) + +{ + GET_INSTANCE_DATA; + TRACE_FUNCTION("dat_cr_data_multirate_conf()"); + + if(rr_data->cr_data.ch_mode EQ CM_AMR) + { + if (v_multirate_conf) + { + /* + * store a new multi-rate speech codec if available. + */ + UBYTE i; + + rr_data->cr_data.amr_conf.mr_vers = multirate_conf->mr_vers; + rr_data->cr_data.amr_conf.nscb = multirate_conf->nscb; + rr_data->cr_data.amr_conf.icmi = multirate_conf->icmi; + rr_data->cr_data.amr_conf.st_mode = multirate_conf->st_mode; + rr_data->cr_data.amr_conf.set_amr = multirate_conf->set_amr; + + /* + * valid flag for the threshold and hystersis values. + * multirate_conf.c_cod_prop + * defines the number of threshold and hystersis values. + */ + rr_data->cr_data.amr_conf.v_cod_prop = multirate_conf->v_cod_prop; + + if(rr_data->cr_data.amr_conf.v_cod_prop) + { + rr_data->cr_data.amr_conf.c_cod_prop = multirate_conf->c_cod_prop; + + for (i=0; i< multirate_conf->c_cod_prop; i++) + memcpy(&rr_data->cr_data.amr_conf.cod_prop[i], &multirate_conf->cod_prop[i], sizeof(T_cod_prop)); + } /* if(rr_data->cr_data.amr_conf.v_cod_prop) */ + } /* if (assign_cmd->v_multirate_conf) */ + } /* if (rr_data->cr_data.ch_mode EQ CM_AMR) */ +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_class_chng_data_req | ++--------------------------------------------------------------------+ + + PURPOSE : This function forms a peer CLASSMARK CHANGE request + +*/ +GLOBAL void dat_class_chng_data_req(void) +{ + GET_INSTANCE_DATA; + + TRACE_FUNCTION("dat_class_chng_data_req()"); + /* + * The syntax check indicates no problems, then + * process the message. + * + * The MS returns a classmark change message. + */ + { + MCAST (class_chng, U_CLASS_CHNG);/* T_U_CLASS_CHNG */ + PALLOC_MSG (dl_data_req, DL_DATA_REQ, U_CLASS_CHNG); + + /* + * set channel type and sapi + */ + dat_code_prr_channel (&dl_data_req->ch_type, + &dl_data_req->sapi, rr_data->sc_data.chan_desc.chan_type); + + class_chng->msg_type = U_CLASS_CHNG; + class_chng->mob_class_2 = rr_data->ms_data.classmark2; + class_chng->mob_class_2.rf_pow_cap = att_get_power (); + class_chng->mob_class_3 = rr_data->ms_data.classmark3; + class_chng->v_mob_class_3 = rr_data->ms_data.classmark2.class3; + + for_dat_data_req (dl_data_req); + } +} +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : RR_DAT | +| STATE : code ROUTINE : dat_dedicated_req_ch_type2 | ++--------------------------------------------------------------------+ + + PURPOSE : This function extracts the channle type from the channel + description IE . + +*/ +LOCAL void dat_dedicated_req_ch_type2(T_ch_type2 *ch_type2, T_chan_desc_before *chan_desc_before, + T_LIST *hop_list_before) +{ + GET_INSTANCE_DATA; + + TRACE_FUNCTION("dat_dedicated_req_ch_type2()"); + + ch_type2->ch = chan_desc_before->chan_type; + ch_type2->tn = chan_desc_before->tn; + ch_type2->tsc = chan_desc_before->tsc; + ch_type2->h = chan_desc_before->hop; + if(ch_type2->h EQ H_NO) + ch_type2->arfcn = chan_desc_before->arfcn; + else + { + ch_type2->maio = chan_desc_before->maio; + ch_type2->hsn = chan_desc_before->hsn; + + /* CSI-LLD section:4.1.1.11 + * This function Updates the black list with the MA list received + * in the assignment command + */ + cs_remove_BA_MA_from_black_list(rr_data->cs_data.region,hop_list_before); + + srv_create_list (hop_list_before, ch_type2->ma, + MAX_MA_CHANNELS, TRUE, 0); + } +} + +#if defined (REL99) && defined (TI_PS_FF_EMR) +/* ++------------------------------------------------------------------------------ +| Function : dat_for_meas_inf ++------------------------------------------------------------------------------ +| Description : Processing of measurement information message is done in this function. +| All possible errors, if present, are detected and an error free MI-message +| instance is decoded and data base updated with the enhanced measurement parameters. +| Parameters : MI-message pointer +| ++------------------------------------------------------------------------------ +*/ +GLOBAL BOOL dat_for_meas_inf (T_D_MEAS_INF *p_mi) +{ + GET_INSTANCE_DATA; + T_rr_enh_para *p_cur = &rr_data->sc_data.emr_data_current; + T_rr_enh_para *p_temp = &rr_data->sc_data.emr_data_temp; + BOOL send_enh_para = FALSE; + T_gprs_rep_prio *p_rep = NULL; + T_gprs_bsic *p_bl = NULL; + +#if defined (TI_PS_FF_RTD) AND defined (REL99) + UBYTE i,j; +#endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */ + + + /* Step 1: Check if we received right BA_IND */ + if( (rr_data->sc_data.ba_list_ded EQ TRUE) AND + (p_mi->ba_ind NEQ rr_data->sc_data.ba_index ) ) + { + rr_data->sc_data.ba_list_ded = FALSE; + rr_data->sc_data.ba_index = p_mi->ba_ind ; + srv_clear_list (&rr_data->sc_data.cd.ncell_list); +#ifdef TI_PS_FF_REL4 + srv_clear_list (&rr_data->sc_data.cd.multiband_ncell_list); +#else + srv_clear_list (&rr_data->sc_data.five_ter_list); +#endif + att_clean_buf (IND_ALL_DEDI_SI); + rr_data->sc_data.cd.sys_info_read &= ~ALL_DEDI_SYS_INFOS; + TRACE_EVENT("Flushed off the entire dedicated mode BA-LIST as the BA-IND got changed"); + } + + /* Step 2: Check report type. + IMPORTANT ASSUMPTION: We will not process the other parameters if report type is Normal*/ + if( p_mi->report_type NEQ ENHANCED_MEAS ) + { + /*check whether there are enhanced parameters and BA list, already. + If present then it means that report type is changing from + Enhanced to Normal*/ + if ( p_cur->is_data_valid EQ TRUE ) + { + for_set_default_emr_data(p_cur); + return TRUE; /*send enh para update to indicate change in report type*/ + } + else + return send_enh_para; + } + + /* Step 3: Check if we already have enh_para in current or temp + and if there is change in parameters or continuation of reception*/ + if(p_temp->is_data_valid EQ FALSE ) + { + /*This means we were not in the process of receiving. Check whether there + is already information in current and if so, is there change in mp_change_mark*/ + if( (p_cur->is_data_valid EQ TRUE ) AND + (p_cur->mp_change_mark EQ p_mi->mp_cm ) ) + { + TRACE_EVENT("No change in Enhanced measurement parameters -ignore "); + return send_enh_para; + } + /* This means there's either a change in MP change mark or receiving EMP for first time */ + /* Decode rest of the parameters*/ + p_temp->is_data_valid = TRUE; + rr_data->sc_data.enh_para_status = ENH_PARA_INVALID_STATE; + p_temp->enh_para.ncc_permitted = rr_data->sc_data.cd.ncc_permitted; + } + + /*Note :If different values occur for the same parameter in different instances of a message, + the instance with the highest index shall be used (sec.3.4.1.2.1, 4.18)*/ + if ( (p_mi->mi_idx > rr_data->sc_data.prev_highest_index ) OR + (rr_data->sc_data.prev_highest_index EQ NOT_PRESENT_8BIT) ) + { + p_temp->enh_para.rep_rate = p_mi->rep_rate; + p_temp->enh_para.inv_bsic_enabled = p_mi->inv_bsic_rep; + p_temp->mp_change_mark = p_mi->mp_cm; + p_temp->msg_count = p_mi->mi_c; + p_temp->rep_type = p_mi->report_type; + if (p_mi->v_emp EQ TRUE ) /* This is updation of parameters other than BSIC list*/ + { + dat_update_emr_rep_para(&p_mi->emp,&p_temp->enh_para); + } + rr_data->sc_data.prev_highest_index = p_mi->mi_idx; + } + +#if defined (TI_PS_FF_RTD) AND defined (REL99) + if(p_mi->v_rtdd) + dat_update_rtd_data(p_mi,p_temp); +#endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */ + + + /*Get relevant parameters to pass to BSIC and report priority list handler*/ + if(p_mi->v_gprs_rep_prio EQ TRUE) + p_rep = &p_mi->gprs_rep_prio; + + if( p_mi->v_gprs_bsic EQ TRUE) + p_bl = &p_mi->gprs_bsic; + + if (for_dat_process_common_emr_data(p_rep,p_bl,p_mi->mi_idx, + rr_data->sc_data.ba_list_ded) ) + { + rr_data->sc_data.enh_para_status = ENH_PARA_DEDICATED; + + if ( rr_data->sc_data.ba_list_ded EQ TRUE) + send_enh_para = TRUE; + } + +#if defined (TI_PS_FF_RTD) AND defined (REL99) + if(p_mi->v_rtdd) + { + /* reset the temporary storage to RTD value not available */ + for(j = 0;j < MAX_NR_OF_NCELL; j++ ) + { + p_temp->enh_para.enh_cell_list[j].v_rtd = FALSE; + for(i = 0;i < MAX_NUM_OF_RTD_VALUES; i++) + p_temp->enh_para.enh_cell_list[j].rtd[i]= RTD_NOT_AVAILABLE; + }/*for*/ + }/*if*/ +#endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */ + + return send_enh_para; +} +#endif + +#endif