line source
+ − /*
+ − +-----------------------------------------------------------------------------
+ − | 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 module defines the functions for the attachment
+ − | capability of the module Radio Resource.
+ − +-----------------------------------------------------------------------------
+ − */
+ −
+ − #ifndef RR_ATTF_C
+ − #define RR_ATTF_C
+ −
+ − #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 "rr_em.h"
+ − #include "cl_shrd.h"
+ −
+ − #if defined (_SIMULATION_)
+ − #include <stdio.h>
+ − #endif
+ −
+ −
+ − /*==== EXPORT =====================================================*/
+ −
+ − /*==== PRIVATE =====================================================*/
+ − LOCAL void rr_unpack_plmn (T_plmn *plmn, const UBYTE *packed, USHORT index);
+ − /* OMAPS00085607 - N950 Memory Optimization */
+ − LOCAL void start_treg_timer(void);
+ − #define TWO 2
+ −
+ −
+ − static void att_build_back_from_dedicated (void);
+ − static void att_calculate_c2 (UBYTE index);
+ − static SHORT att_calculate_c2_diff (UBYTE index);
+ − static UBYTE att_calculate_digits (UBYTE *digits);
+ −
+ −
+ − static BOOL att_cell_in_data_structures (T_MPH_MEASUREMENT_IND *report,
+ − UBYTE index);
+ − static UBYTE att_cell_in_measure_report (T_MPH_MEASUREMENT_IND *report,
+ − UBYTE index);
+ − static void att_check_2ter_read (UBYTE index);
+ − static BOOL att_check_cell (void);
+ − LOCAL UBYTE att_get_txpwr_max_cch (UBYTE index);
+ − static void att_clear_forb_list (int list_type);
+ − static void att_clear_parallel_search (void);
+ − static void att_code_prr_mm_info (T_mm_info *mm_info);
+ − static void att_copy_c2_parameter (T_C2_PARAMETER *c2_par,
+ − T_si3_rest_oct *rest_oct );
+ − static void att_copy_c2_parameter_si3 (UBYTE index,
+ − T_si3_rest_oct *rest_oct);
+ − static void att_copy_found_plmn (T_FOUND_LIST * list,
+ − USHORT n_in_source_85_dBm,
+ − USHORT i_in_copy);
+ − static void att_copy_rach_parameter (UBYTE index,
+ − T_rach_ctrl *rach,
+ − UBYTE indicate_changes);
+ − static UBYTE att_get_highest_c2_index (void);
+ − static UBYTE att_get_next_highest_c2_idx(UBYTE old_index);
+ − static UBYTE att_get_next_highest_c2_val(void);
+ − static UBYTE att_get_func (void);
+ − static UBYTE att_get_next_highest_rx (void);
+ − static void att_insert_cell_in_data (T_MPH_MEASUREMENT_IND *report,
+ − UBYTE index);
+ − static void att_order_plmns (void);
+ − static USHORT att_number_of_plmns_greater_85_dBm (void);
+ − static void att_copy_plmns_lower_or_equal_85_dBm (T_FOUND_LIST * list, USHORT i_in_copy);
+ − static BOOL att_priority_check (void);
+ − static void att_reorder_mph_ncell_req (T_MPH_NEIGHBOURCELL_REQ*
+ − mph_ncell_req);
+ − static void att_search_cell (void);
+ − static void att_select_cell_dedicated (void);
+ − static void att_try_old_cell (void);
+ − static void att_print_selection_type (UBYTE selection_type);
+ − /* Implements Measure#32: Row 36, 39 and 40 */
+ − static void att_print_op (T_op *op,
+ − T_S2I_STRING titel);
+ − static void att_begin_cs (UBYTE req_mm_service);
+ −
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − static void att_check_for_si5ter_and_enhpara (UBYTE old_index);
+ − #endif
+ −
+ −
+ −
+ −
+ − LOCAL void att_copy_sys_info_2bis_2ter_par(UBYTE index, T_SI_TYPE si_type,
+ − T_LIST *new_2_bis_ter_list,
+ − BUF_neigh_cell_desc *neigh_cell_desc,
+ − UBYTE indicate_changes);
+ −
+ − /*==== VARIABLES ==================================================*/
+ − GLOBAL UBYTE test_house = FALSE;
+ −
+ − /*==== MACROS =====================================================*/
+ −
+ − #if !defined (NTRACE)
+ −
+ − #define TRACE_C1(index)\
+ − TRACE_EVENT_P4 ("[%u]i%u C1=%-2d rxlev=%u",\
+ − rr_data->nc_data[index].arfcn,index,\
+ − rr_data->nc_data[index].c1,\
+ − rr_data->nc_data[index].rxlev)
+ −
+ − #define TRACE_C2(index)\
+ − TRACE_EVENT_P3 ("[%u]i%u C2=%-2d AT=%d",\
+ − rr_data->nc_data[index].arfcn,index,\
+ − rr_data->nc_data[index].c2,\
+ − rr_data->nc_data[index].avail_time)
+ −
+ − #define TRACE_C1_C2(index)\
+ − TRACE_EVENT_P6 ("[%u]i%u C1=%-2d C2=%-2d rxlev=%-2u AT=%d",\
+ − rr_data->nc_data[index].arfcn,index,\
+ − rr_data->nc_data[index].c1,\
+ − rr_data->nc_data[index].c2,\
+ − rr_data->nc_data[index].rxlev,\
+ − rr_data->nc_data[index].avail_time)
+ −
+ − #define TRACE_SELECTION_TYPE(type) att_print_selection_type(type)
+ −
+ − /* Implements Measure#32: Row 36, 39 and 40 */
+ − #define TRACE_OP_TYPE(op,titel) att_print_op (op, S2I_STRING(titel))
+ −
+ − #else /* !NTRACE */
+ −
+ − #define TRACE_C1(i)
+ − #define TRACE_C2(i)
+ − #define TRACE_C1_C2(i)
+ − #define TRACE_SELECTION_TYPE(type)
+ − #define TRACE_OP_TYPE(op,titel)
+ −
+ − #endif /* !NTRACE */
+ −
+ − /*==== FUNCTIONS ==================================================*/
+ −
+ − /*
+ − * -------------------------------------------------------------------
+ − * Procedures
+ − * -------------------------------------------------------------------
+ − */
+ − /* Implements Measure#32: Row 60 */
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_print_forb_list
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After a successful location updating request MM informs
+ − RR about this. If the location area information is stored
+ − inside of one of the forbidden location area lists this
+ − information must be updated. In this function, the entry
+ − is deleted from the given forbidden list.
+ −
+ − */
+ −
+ − LOCAL void att_print_forb_list ( int list_type,
+ − const T_plmn * plmn,
+ − USHORT lac,
+ − unsigned int index,
+ − BOOL to_delete)
+ − {
+ − if(list_type EQ FORBIDDEN_LIST_NORMAL)
+ − {
+ − if(to_delete EQ FALSE)
+ − {
+ − TRACE_EVENT_P1 ( "ADD FORB LIST %u:", index);
+ − }
+ − else
+ − {
+ − TRACE_EVENT_P1 ( "DEL FORB LIST %u:", index);
+ − }
+ − }
+ − else
+ − {
+ − if(to_delete EQ FALSE)
+ − {
+ − TRACE_EVENT_P1 ( "ADD FORB LIST %u:", index);
+ − }
+ − else
+ − {
+ − TRACE_EVENT_P1 ( "DEL ROAM FORB LIST %u:", index);
+ − }
+ − }
+ − TRACE_EVENT_P7 ( " MCC/MNC=%x%x%x/%x%x%x LAC=%x",
+ − plmn->mcc[0],
+ − plmn->mcc[1],
+ − plmn->mcc[2],
+ − plmn->mnc[0],
+ − plmn->mnc[1],
+ − plmn->mnc[2],
+ − lac);
+ − }
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_add_to_forb_list |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : RR stores a list of forbidden location area codes. After
+ − a specific location updating reject cause this information
+ − will be given by MM with a RR_SYNC_REQ. The check against
+ − forbidden location areas is part of the cell selection and
+ − cell reselection process.
+ −
+ − */
+ −
+ − GLOBAL void att_add_to_forb_list (int list_type,
+ − const T_plmn *plmn,
+ − USHORT lac)
+ − {
+ − GET_INSTANCE_DATA;
+ − int i;
+ − T_loc_area_ident *forb_list;
+ −
+ − if (list_type EQ FORBIDDEN_LIST_NORMAL)
+ − forb_list = &rr_data->ms_data.forb_lac_list[0];
+ − else
+ − forb_list = &rr_data->ms_data.roam_forb_lac_list[0];
+ −
+ − TRACE_FUNCTION ("att_add_to_forb_list()");
+ −
+ − for (i = 0; i < MAX_LAI; i++)
+ − {
+ − /*
+ − * Check whether the location area code is already stored. In
+ − * this case it is not necessary to store it again
+ − */
+ − if ((lac EQ forb_list[i].lac) AND
+ − dat_plmn_equal_req (plmn->mcc, plmn->mnc,
+ − forb_list[i].mcc, forb_list[i].mnc))
+ − break;
+ − }
+ −
+ − if (i EQ MAX_LAI)
+ − {
+ − /*
+ − * Location Area Code is not stored yet. Look for a free entry
+ − * and store the location area code.
+ − */
+ − for (i = 0; i < MAX_LAI; i++)
+ − {
+ − if (forb_list[i].lac EQ NOT_PRESENT_16BIT)
+ − {
+ − memcpy (forb_list[i].mcc, plmn->mcc, SIZE_MCC);
+ − memcpy (forb_list[i].mnc, plmn->mnc, SIZE_MNC);
+ − forb_list[i].lac = lac;
+ − break;
+ − }
+ − }
+ −
+ − if (i EQ MAX_LAI)
+ − {
+ − /*
+ − * all entries are used. Then delete the first (and oldest) one,
+ − * move each entry and store the new one in the last position.
+ − */
+ − memmove (&forb_list[0], &forb_list[1],
+ − sizeof (T_loc_area_ident) * (MAX_LAI-1));
+ − memcpy (forb_list[MAX_LAI-1].mcc, plmn->mcc, SIZE_MCC);
+ − memcpy (forb_list[MAX_LAI-1].mnc, plmn->mnc, SIZE_MNC);
+ − forb_list[MAX_LAI-1].lac = lac;
+ − i--;
+ − }
+ − }
+ −
+ − /* Implements Measure#32: Row 4 */
+ − att_print_forb_list ( list_type, plmn, lac, i, FALSE);
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_analyze_measure_report |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After reception of a MPH_MEASUREMENT_IND from layer 1 the
+ − channels are analyzed for a cell reselection decision.
+ −
+ − */
+ −
+ − GLOBAL void att_analyze_measure_report (T_MPH_MEASUREMENT_IND *report)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE i = 0;
+ −
+ − TRACE_FUNCTION ("att_analyze_measure_report()");
+ −
+ − /*
+ − * The variable no_reselect controls the condition not to perform a
+ − * new cell reselection due to C2(NC) > C2(SC) after a cell reselection
+ − * for 15 seconds
+ − */
+ −
+ − /*
+ − * The fieldtest has shown, that sometimes the serving cell was also reported
+ − * as a neighbourcell. This function removes double entries
+ − */
+ − att_remove_multiple_channels ();
+ −
+ − /*
+ − * All possible neighbourcells plus the serving cell are checked.
+ − */
+ − for (i = 0; i < 6; i++)
+ − {
+ − /*
+ − * The tnnn variable controls the condition that a cell is barred for five seconds
+ − * for cell reselection purposes after a random access failure.
+ − */
+ − /*
+ − * check the status of a cell using the data stored in RR for neighbour- and serving
+ − * cell.
+ − */
+ − switch (rr_data->nc_data[i].bcch_status)
+ − {
+ − case EMPTY:
+ − /*
+ − * RR has nothing stored in this area. So there is nothing to do.
+ − */
+ − break;
+ −
+ − case DECODED:
+ − /*
+ − * In this area RR has stored Data for a neighbourcell. The status is decoded,
+ − * that means RR has received a system information 3,4,7 or 8 to calculate
+ − * the cell reselection criterion C2.
+ − *
+ − * Now it is checked against the incoming measurement report.
+ − */
+ − switch (att_cell_in_measure_report (report, i))
+ − {
+ − case CELL_IS_INSERTED:
+ − /*
+ − * The cell is still included in the measurement report, that means
+ − * it is one of the six strongest carrier and layer 1 is synchronized
+ − * to the cell.
+ − *
+ − * The time how long the cell is already synchronized is stored up to
+ − * a period of 700 seconds. This informatin is used for the calculation
+ − * of the cell reselection criterion C2 which may has a time dependent
+ − * factor. In fact the frames of layer 1 are counted.
+ − */
+ − if (rr_data->nc_data[i].avail_time < PERIOD_700_SEC)
+ − rr_data->nc_data[i].avail_time += rr_data->ms_data.fn_offset;
+ − break;
+ −
+ − case BSIC_HAS_CHANGED:
+ − /*
+ − * The cell is still included in the measurement report, but layer1
+ − * has detected a change of the BSIC during re-synchronisation on the
+ − * SB channel. So the stored information for calculation of C2 is useless
+ − * and the status and the avail time is cleared.
+ − */
+ −
+ − rr_data->nc_data[i].bcch_status = NON_DECODED;
+ − rr_data->nc_data[i].avail_time = 0;
+ − break;
+ −
+ − case CELL_IS_NOT_INSERTED:
+ − /*
+ − * The cell is not longer included. The area in RR is released.
+ − */
+ − rr_data->nc_data[i].bcch_status = EMPTY;
+ − break;
+ − }
+ − break;
+ −
+ − case NON_DECODED:
+ − /*
+ − * RR has already stored this cell in its data, but has not received
+ − * the needed information for calculation of cell reselection criterion C2.
+ − *
+ − * Now it is checked against the incoming measurement report.
+ − */
+ − switch (att_cell_in_measure_report (report, i))
+ − {
+ − case CELL_IS_INSERTED:
+ − /*
+ − * The cell is still included. For later calculation of the cell
+ − * reselection criterion C2 the time is counted for which the cell
+ − * is synchronized.
+ − */
+ − if (rr_data->nc_data[i].avail_time < PERIOD_700_SEC)
+ − rr_data->nc_data[i].avail_time += rr_data->ms_data.fn_offset;
+ − break;
+ −
+ − case BSIC_HAS_CHANGED:
+ − /*
+ − * The layer 1 indicated that the BSIC has changed during re-synchronisation
+ − * to the SB channel. The status will not change, but the avail time is
+ − * resetted.
+ − */
+ − rr_data->nc_data[i].avail_time = 0;
+ − break;
+ −
+ − case CELL_IS_NOT_INSERTED:
+ − /*
+ − * Layer 1 indicates that the cell is not longer one of the six strongest or
+ − * that the synchronisation has been lost. The storage area in RR is released.
+ − */
+ − rr_data->nc_data[i].bcch_status = EMPTY;
+ − rr_data->nc_data[i].avail_time = 0;
+ − break;
+ − }
+ − break;
+ − }
+ − }
+ −
+ − /*
+ − * store the Fieldstrength of the serving cell and update the information
+ − * of the power campaign.
+ − */
+ − rr_data->nc_data[SC_INDEX].rxlev = report->rx_lev_full;
+ − cs_set_rxlev (report->rx_lev_full, rr_data->nc_data[SC_INDEX].arfcn);
+ −
+ − if (rr_data->nc_data[SC_INDEX].rxlev >
+ − rr_data->lup_rxlev + 10)
+ − {
+ − /*
+ − * A fieldstrength jump more then 10 dBm has been detected. This can be the
+ − * trigger for MM to restart a location updating request if needed. It indicates
+ − * for example that the mobile has left a tunnel area without coverage.
+ − *
+ − * This is not used by MM if a test SIM is inserted. It is an improvement for
+ − * the field beside the GSM specifications.
+ − */
+ − PALLOC (sync, RR_SYNC_IND);
+ −
+ − sync->ciph = NOT_PRESENT_8BIT;
+ − sync->chm.ch_mode = NOT_PRESENT_8BIT;
+ − sync->synccs = SYNCCS_LUP_RETRY;
+ − sync->mm_info.valid = FALSE;
+ − sync->bcch_info.v_bcch = FALSE;
+ −
+ − PSENDX (MM, sync);
+ −
+ − /*
+ − * Store the new value to avoid multiple signalling of the fieldstrength jump
+ − */
+ − rr_data->lup_rxlev = rr_data->nc_data[SC_INDEX].rxlev;
+ − }
+ −
+ − /*
+ − * Now look for all neighbourcells in the measurement report if they are new.
+ − */
+ − for (i = 0; i < report->ncells.no_of_ncells; i++)
+ − {
+ − cs_set_rxlev (report->ncells.rx_lev[i], report->ncells.arfcn[i]);
+ − /*
+ − * If a channel is not inside the RR data structure, insert the cell
+ − */
+ − if (! att_cell_in_data_structures (report, i))
+ − att_insert_cell_in_data (report, i);
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_bcch_status_to_decoded |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After reception of the system information type 3,4,7 or 8
+ − the status of a cell changes from NON_DECODED to DECODED
+ − and it is possible to calculate the cell reselection
+ − criterion C2.
+ −
+ − */
+ −
+ − GLOBAL void att_bcch_status_to_decoded (UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_EVENT_P2 ("[%u]%u att_bcch_status_to_decoded()", rr_data->nc_data[index].arfcn, index);
+ −
+ − rr_data->nc_data[index].bcch_status = DECODED;
+ − rr_data->nc_data[index].bcch_counter = 0;
+ − rr_data->nc_data[index].c1_counter = 0;
+ − #ifdef GPRS
+ − if( rr_data->gprs_data.use_c31 NEQ TRUE )
+ − {
+ − #endif
+ −
+ − if ((rr_data->c_ncell_bcch > 0) AND (rr_data->c_ncell_bcch NEQ NOT_INITIALISED))
+ − rr_data->c_ncell_bcch--;
+ −
+ − if (rr_data->resel_pending)
+ − {
+ − if (rr_data->c_ncell_bcch EQ 0)
+ − {
+ − SET_STATE (STATE_ATT, ATT_CON_EST);
+ − #ifdef GPRS
+ − att_start_cell_reselection_gprs(CELL_RESELECTION_RACH);
+ − #else
+ − att_start_cell_reselection(CELL_RESELECTION_RACH);
+ − #endif
+ − rr_data->resel_pending = FALSE;
+ − rr_data->c_ncell_bcch = NOT_INITIALISED;
+ − }
+ − }
+ − else
+ − {
+ − att_calculate_c2 (index);
+ − }
+ − #ifdef GPRS
+ − }
+ − else
+ − {
+ − if( rr_data->nc_data[index].v_cr_par EQ CR_PAR_INVALID )
+ − {
+ − att_convert_idle_c31_cr(index);
+ − }
+ − else
+ − {
+ − TRACE_EVENT_P2(" arfcn %d, index = %d has valid CR_ PAR", rr_data->nc_data[index].arfcn, index);
+ − }
+ − }
+ − #endif
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_bits_to_byte |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function is used to calculate the positions of 1-bits
+ − into a byte array. This is needed for calculation of a
+ − frequency hopping list for a mobile allocation. The input
+ − parameter are the bitmap and the size of the bitmap in bytes.
+ − The result is a byte array containing the positions of 1-bits
+ − followed by an end identifier NOT_PRESENT_8BIT (0xFF). The
+ − maximum size of the incoming array is 32 bytes.
+ −
+ − Example: a mobile allocation looks like this:
+ −
+ − 0x42 = 0b01000010 -> bit 15 14 13 12 11 10 9 8
+ − 0x81 = 0b10000001 -> bit 7 6 5 4 3 2 1 0
+ −
+ − The size of the input array is 2 and the output is:
+ −
+ − 1,8,10,15,0xFF (starting with position 1 in bit 0)
+ −
+ − */
+ −
+ − GLOBAL void att_bits_to_byte (UBYTE * num,
+ − UBYTE size,
+ − UBYTE * bits)
+ − {
+ − SHORT ix;
+ − UBYTE bit;
+ − UBYTE buf_ix;
+ −
+ − TRACE_FUNCTION ("att_bits_to_byte()");
+ −
+ − /*
+ − * For all bytes in the input field
+ − */
+ − for (ix = 0; ix < size; ix++)
+ − {
+ − buf_ix = bits[size-1-ix];
+ −
+ − /*
+ − * for each bit inside this byte
+ − */
+ − bit = 0;
+ − while (8 > bit)
+ − {
+ − if (buf_ix & (1 << bit))
+ − {
+ − /*
+ − * If bit is set, store the position in the output field
+ − */
+ − *num++ = ((USHORT)ix << 3) + bit + 1;
+ − }
+ − bit++;
+ − }
+ − }
+ − /*
+ − * add the end identifier
+ − */
+ − *num = NOT_PRESENT_8BIT;
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_build_cbch |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : configure at the end of the cell selection the CBCH channel
+ − in layer 1. The CBCH channel is optional defined in system
+ − information type 4.
+ − When the cell has PBCCH and it is a Release-99 cell,
+ − the CBCH configuration should be taken from gprs database.
+ − This information is provided by GRR through RRGRR_CBCH_INFO_IND.
+ − CBCH channel configuration is received on PBCCH on PSI8.
+ − */
+ −
+ − #ifdef REL99
+ − LOCAL void att_build_cbch (void)
+ − #else
+ − GLOBAL void att_build_cbch (void)
+ − #endif
+ − {/*lint -e813 supress info of length of T_LIST*/
+ − GET_INSTANCE_DATA;
+ − T_LIST hop_list;
+ − PALLOC (cbch_req, MPH_CBCH_REQ);
+ −
+ − TRACE_FUNCTION ("att_build_cbch()");
+ −
+ − /*
+ − * Initialize the primitive to layer 1
+ − */
+ − memset (cbch_req, 0, sizeof (T_MPH_CBCH_REQ));
+ − if (rr_data->ms_data.rr_service EQ FULL_SERVICE)
+ − {
+ − /*
+ − * CBCH is activated only if RR is in full service
+ − */
+ − if (rr_data->sc_data.cd.cbch_chan_desc_avail NEQ NO_CONTENT )
+ − {
+ − /*
+ − * If a CBCH channel description has been received with
+ − * system information tpye 4, activate the channel in layer 1.
+ − */
+ − cbch_req->cbch.stat = STAT_ACT;
+ − cbch_req->cbch.ch = rr_data->sc_data.cd.cbch_chan_desc.chan_type;
+ − cbch_req->cbch.tn = rr_data->sc_data.cd.cbch_chan_desc.tn;
+ − cbch_req->cbch.tsc = rr_data->sc_data.cd.cbch_chan_desc.tsc;
+ − cbch_req->cbch.h = rr_data->sc_data.cd.cbch_chan_desc.hop;
+ −
+ − switch (cbch_req->cbch.h)
+ − {
+ − case H_NO:
+ − /*
+ − * CBCH does not use frequency hopping, then configure simply the
+ − * channel number
+ − */
+ − cbch_req->cbch.arfcn = rr_data->sc_data.cd.cbch_chan_desc.arfcn;
+ − break;
+ −
+ − case H_FREQ:
+ − /*
+ − * CBCH uses frequency hopping, then configure MAIO and HSN
+ − * and create a frequency hopping list from the cell channel
+ − * description in system information 1 and the mobile allocation
+ − * in system information 4.
+ − */
+ − cbch_req->cbch.maio = rr_data->sc_data.cd.cbch_chan_desc.maio;
+ − cbch_req->cbch.hsn = rr_data->sc_data.cd.cbch_chan_desc.hsn;
+ − srv_create_chan_mob_alloc (&rr_data->sc_data.cd.cell_chan_desc,
+ − &hop_list,
+ − rr_data->sc_data.cd.cbch_mob_alloc);
+ −
+ − /* CSI-LLD section:4.1.1.11
+ − * This function Updates the black list with the MA list received
+ − * in the CBCH allocation
+ − */
+ − cs_remove_BA_MA_from_black_list(rr_data->cs_data.region,&hop_list);
+ −
+ − srv_create_list (&hop_list, cbch_req->cbch.ma, MAX_MA_CHANNELS, TRUE,
+ − 0);
+ − break;
+ − }
+ − }
+ − else
+ − /*
+ − * In all other cases configure layer 1 without CBCH.
+ − */
+ − cbch_req->cbch.stat = STAT_INACT;
+ − }
+ −
+ − PSENDX (PL, cbch_req);
+ − }
+ −
+ − #ifdef REL99
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_config_cbch |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The following conditions have to be satisfied for
+ − configuring CBCH.
+ − 1. Full service should have been enabled.
+ − 2. Mobile should be in idle/packet mode.
+ − This function checks these conditions before configuring
+ − CBCH. If function is called during packet transfer mode,
+ − a flag in GPRS data is set signalling that this function
+ − should be called again after entering idle mode.
+ − */
+ − GLOBAL void att_config_cbch (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_config_cbch()");
+ − /*
+ − * CBCH is activated only if RR is in full service
+ − */
+ − if ((rr_data->ms_data.rr_service EQ FULL_SERVICE) )
+ − {
+ − #ifdef GPRS
+ − /* check if mobile is in PTM */
+ − if(!dat_gprs_cell_in_ptm())
+ − {
+ − /* mobile is NOT in packet transfer mode */
+ −
+ − if(!att_gprs_cell_has_pbcch() OR !att_gprs_get_nw_release())
+ − {
+ − /* This cell does not have pbcch,
+ − * OR this is a R - 98 or lower cell.
+ − * hence take info stored in SI 4
+ − */
+ − #endif /* GPRS */
+ − att_build_cbch();
+ − #ifdef GPRS
+ − }
+ − else
+ − {
+ − /* This Cell has PBCCH and also this is Release-99 cell.
+ − * Configure ALR if CBCH info is available form GRR.
+ − */
+ − if(rr_data->gprs_data.cbch_psi_valid)
+ − {
+ − PALLOC (cbch_req, MPH_CBCH_REQ);
+ − memcpy(&(cbch_req->cbch),
+ − &(rr_data->gprs_data.cbch_psi8),
+ − sizeof(T_cbch));
+ − PSENDX (PL, cbch_req);
+ − }
+ − /*else wait till CBCH info is received from GRR */
+ − }
+ − rr_data->gprs_data.cbch_info_rxvd_in_ptm = FALSE;
+ − }
+ − else
+ − {
+ − rr_data->gprs_data.cbch_info_rxvd_in_ptm = TRUE;
+ − }
+ − #endif /* GPRS */
+ − }
+ − else
+ − {
+ − #ifdef GPRS
+ − if(!dat_gprs_cell_in_ptm())
+ − #endif /* GPRS */
+ − {
+ − PALLOC (cbch_req, MPH_CBCH_REQ);
+ − memset (cbch_req, 0, sizeof (T_MPH_CBCH_REQ));
+ − cbch_req->cbch.stat = STAT_INACT;
+ − PSENDX (PL, cbch_req);
+ − }
+ − }
+ − }
+ − #endif
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_build_classmark_req |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Configure at the end of the cell selection the classmark of
+ − the mobile to layer 1. This information is used by layer 1
+ − for random access.
+ −
+ − */
+ −
+ − GLOBAL void att_build_classmark_req (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − PALLOC (classmark_req, MPH_CLASSMARK_REQ);
+ −
+ − TRACE_FUNCTION ("att_build_classmark_req()");
+ −
+ − #if defined (_SIMULATION_)
+ − /*
+ − * In the windows simulation it is necessary to read the data
+ − * from the non-volantile memory (PCM) again.
+ − */
+ − rr_csf_check_rfcap (FALSE);
+ − #endif
+ −
+ − if((std EQ STD_DUAL) OR (std EQ STD_DUAL_EGSM))
+ − {
+ − classmark_req->classmark.pclass = rr_data->ms_data.rf_cap.rf_power.pow_class4[IDX_PWRCLASS_900].pow_class-1;
+ − classmark_req->classmark.pclass2 = rr_data->ms_data.rf_cap.rf_power.pow_class4[IDX_PWRCLASS_1800].pow_class-1;
+ − }
+ − else if(std EQ STD_DUAL_US)
+ − {
+ − classmark_req->classmark.pclass = rr_data->ms_data.rf_cap.rf_power.pow_class4[IDX_PWRCLASS_850].pow_class-1;
+ − classmark_req->classmark.pclass2 = rr_data->ms_data.rf_cap.rf_power.pow_class4[IDX_PWRCLASS_1900].pow_class-1;
+ − }
+ − else
+ − {
+ − /*
+ − * In single bands only one power class is forwarded to layer 1.
+ − */
+ − #ifdef GPRS
+ −
+ − #ifdef REL99
+ − classmark_req->classmark.pclass = rr_data->ms_data.ra_cap.ra_cap_values.acc_cap.pow_class-1;
+ − #else
+ − classmark_req->classmark.pclass = rr_data->ms_data.ra_cap.acc_cap.pow_class-1;
+ − #endif
+ −
+ − #else
+ − if (std EQ STD_1900)
+ − {
+ − classmark_req->classmark.pclass = rr_data->ms_data.classmark3.pcs1900_cap-1;
+ − }
+ − else if (std EQ STD_850)
+ − {
+ − classmark_req->classmark.pclass = rr_data->ms_data.classmark3.gsm850_cap-1;
+ − }
+ − else
+ − {
+ − classmark_req->classmark.pclass = rr_data->ms_data.classmark3.radio_cap_1-1;
+ − }
+ − #endif
+ − classmark_req->classmark.pclass2= 0;
+ − }
+ − PSENDX (PL, classmark_req );
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_build_idle_req |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function is used to configure the idle mode in layer 1.
+ − Two modes are possible: cell selection and cell reselection.
+ − In cell selection mode RR has all information to perform
+ − a complete layer 1 configuration. In cell reselection mode
+ − layer 1 is configured only as much as needed and it is
+ − expected that layer 1 goes in paging reorganization mode
+ − to catch all pagings and BCCH messages.
+ − */
+ −
+ − GLOBAL void att_build_idle_req (UBYTE index,
+ − UBYTE mode)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_CELL_DATA *cd;
+ − PALLOC (idle_req, MPH_IDLE_REQ);
+ −
+ − TRACE_FUNCTION ("att_build_idle_req()");
+ −
+ − /*
+ − * configure the mode of layer 1 (cell selection or cell reselection
+ − */
+ − idle_req->mod = mode;
+ −
+ − /*
+ − * set the pointer to the correct data
+ − * serving cell or one of the neighbourcells.
+ − */
+ − if (index EQ SC_INDEX)
+ − {
+ − cd = &rr_data->sc_data.cd;
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − idle_req->si2quater_status = cd->si2quater_status;
+ − idle_req->si2quater_pos = cd->si2quater_pos;
+ −
+ − if ( (mode EQ MODE_CELL_SELECTION ) AND
+ − (cd->si2quater_status EQ SI2QUATER_CONFIGURE) AND
+ − ((rr_data->sc_data.cd.sys_info_read & (ALL_SYS_INFO_READ | SYS_INFO_2QUATER_READ)) EQ ALL_SYS_INFO_READ))
+ − rr_data->sc_data.cd.si2quater_status = SI2QUATER_ACQ_PENDING;
+ − #endif
+ − }
+ − else
+ − {
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − idle_req->si2quater_status = SI2QUATER_ABSENT;
+ − #endif
+ − cd = &rr_data->cr_data.cd;
+ − }
+ −
+ − /*
+ − * look to the reached service of RR
+ − */
+ − switch (rr_data->ms_data.rr_service)
+ − {
+ − case LIMITED_SERVICE:
+ − /*
+ − * In limited mode no pagings are possible. To save
+ − * power the minimum number of measurements (equal
+ − * to a long paging period) is configured.
+ − * The other parameters are independent from BCCH or
+ − * SIM card, because paging is not possible.
+ − */
+ − idle_req->pg = 0;
+ − idle_req->tn = 0;
+ − idle_req->ncc_permitted = NOT_PRESENT_8BIT;
+ − idle_req->dlt = 10; /* for bs_pa_mfmrs = 9 */
+ − idle_req->bs_pa_mfrms = 7; /* for bs_pa_mfmrs = 9 */
+ − break;
+ −
+ − case FULL_SERVICE:
+ − /*
+ − * The MS is in full service. In this case the paging
+ − * block must be correctly from the SIM and the BCCH
+ − * data.
+ − */
+ − idle_req->pg = dat_calc_paging_group (index);
+ − idle_req->tn = dat_calc_tn (index);
+ −
+ − /*
+ − * The NCC permitted check in layer 1 is used to
+ − * avoid synchronisation to cells of other networks.
+ − */
+ − #ifdef GPRS
+ − if(mode NEQ MODE_CELL_RESELECTION AND
+ − mode NEQ MODE_CELL_RESELECTION_SYNC_ONLY)
+ − #else
+ − if (mode NEQ MODE_CELL_RESELECTION)
+ − #endif
+ − idle_req->ncc_permitted = cd->ncc_permitted;
+ − else
+ − idle_req->ncc_permitted = NOT_PRESENT_8BIT;
+ −
+ − idle_req->dlt = dat_calc_downlink_timeout (index);
+ − idle_req->bs_pa_mfrms = rr_data->nc_data[index].control_descr.bs_pa_mfrms;
+ − break;
+ − }
+ − idle_req->arfcn = rr_data->nc_data[index].arfcn;
+ −
+ − /* XXX
+ − * use ext_bcch for forwarding bsic, this solves
+ − * fieldtest problem of mismatch of BSICs.
+ − */
+ − idle_req->ext_bcch = rr_data->nc_data[index].bsic;
+ −
+ − idle_req->comb_ccch = rr_data->nc_data[index].control_descr.ccch_conf EQ COMB_CCCH_COMB;
+ − idle_req->bs_ag_blocks_res =
+ − rr_data->nc_data[index].control_descr.bs_ag_blks_res;
+ −
+ − idle_req->power = att_get_txpwr_max_cch (index);
+ −
+ − idle_req->reorg_only = NORMAL_PGM;
+ − idle_req->gprs_support = NOT_PRESENT_8BIT;
+ − #if defined FF_EOTD
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − idle_req->eotd_avail = (std EQ STD_1900 OR std EQ STD_850 OR std EQ STD_DUAL_US OR std EQ STD_850_1800 OR std EQ STD_850_900_1800 OR std EQ STD_850_900_1900);
+ − #else
+ − idle_req->eotd_avail = (std EQ STD_1900 OR std EQ STD_850 OR std EQ STD_DUAL_US);
+ − #endif
+ − #else
+ − idle_req->eotd_avail = 0;
+ − #endif
+ −
+ − #ifdef GPRS
+ − att_gprs_idle_req(idle_req);
+ − #endif
+ −
+ − EM_IDLE_MODE;
+ −
+ − #if defined(_SIMULATION_)
+ − switch (idle_req->mod)
+ − {
+ − case MODE_CELL_SELECTION : /* 0x0 cell selection */
+ − TRACE_EVENT_WIN ("MODE_CELL_SELECTION");
+ − break;
+ − case MODE_CELL_RESELECTION : /* 0x1 cell reselection */
+ − TRACE_EVENT_WIN ("MODE_CELL_RESELECTION");
+ − break;
+ − #if defined(MODE_BACK_FROM_DEDICATED)
+ − case MODE_BACK_FROM_DEDICATED : /* 0x2 back from dedicated */
+ − TRACE_EVENT_WIN ("MODE_BACK_FROM_DEDICATED");
+ − break;
+ − #endif /* MODE_BACK_FROM_DEDICATED */
+ − case MODE_IMM_ASSIGN : /* 0x3 immediate assignment */
+ − TRACE_EVENT_WIN ("MODE_IMM_ASSIGN");
+ − break;
+ − case MODE_CHAN_ASSIGN : /* 0x4 channel assignment */
+ − TRACE_EVENT_WIN ("MODE_CHAN_ASSIGN");
+ − break;
+ − case MODE_ASYNC_HANDOVER : /* 0x5 asynchronous handover */
+ − TRACE_EVENT_WIN ("MODE_ASYNC_HANDOVER");
+ − break;
+ − case MODE_SYNC_HANDOVER : /* 0x6 synchronous handover */
+ − TRACE_EVENT_WIN ("MODE_SYNC_HANDOVER");
+ − break;
+ − case MODE_PRE_SYNC_HANDOVER : /* 0x7 pre synchronous handover */
+ − TRACE_EVENT_WIN ("MODE_PRE_SYNC_HANDOVER");
+ − break;
+ − case MODE_PSEUDO_SYNC_HANDOVER: /* 0x8 pseudo synchronous handover */
+ − TRACE_EVENT_WIN ("MODE_PSEUDO_SYNC_HANDOVER");
+ − break;
+ − case MODE_SYS_INFO_CHANGE : /* 0x9 sys info has changed */
+ − TRACE_EVENT_WIN ("MODE_SYS_INFO_CHANGE");
+ − break;
+ − case MODE_PACKET_TRANSFER : /* 0xa enter packet transfer mode */
+ − TRACE_EVENT_WIN ("MODE_PACKET_TRANSFER");
+ − break;
+ − case MODE_PDCH_ASSIGN : /* 0xb PDCH assignment */
+ − TRACE_EVENT_WIN ("MODE_PDCH_ASSIGN");
+ − break;
+ − case MODE_CELL_CHANGE_ORDER : /* 0xc Network controlled Cell Change */
+ − TRACE_EVENT_WIN ("MODE_CELL_CHANGE_ORDER");
+ − break;
+ − case MODE_CELL_RESELECTION_SYNC_ONLY: /* 0xc Network controlled Cell Change */
+ − TRACE_EVENT_WIN ("MODE_CR_SYNC_ONLY");
+ − break;
+ − case MODE_CONFIG_PL: /* PBCCH */
+ − TRACE_EVENT_WIN ("MODE_CONFIG_PL");
+ − break;
+ − default:
+ − TRACE_EVENT_WIN_P1 ("idle_req mode %d unknown!", idle_req->mod);
+ − break;
+ − }
+ − #endif /* _SIMULATION_ */
+ − if ((mode EQ MODE_CELL_SELECTION) AND
+ − (rr_data->ms_data.rr_service EQ FULL_SERVICE) AND
+ − (rr_data->cs_data.act_index NEQ NOT_PRESENT_8BIT))
+ − {
+ − rr_data->cs_data.act_index = NOT_PRESENT_8BIT;
+ − rr_csf_fit_capability ();
+ − #if !defined(NTRACE)
+ − rr_csf_trace_power ();
+ − #endif /* !NTRACE */
+ − }
+ − PSENDX (PL, idle_req);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_calculate_c1 |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function calculates the path loss criterion C1 for
+ − cell selection and cell reselection purposes.
+ −
+ − REFERENCE: GSM 5.08, chapter 6.4 Criteria for cell selection and reselection
+ −
+ − */
+ −
+ − /*
+ − * The following tables define conversion for the power in dBm depending
+ − * on the frequency band (GSM, DCS or PCS) and the power class (1 to 5).
+ − */
+ −
+ − /*lint -esym(765,p_dcs) | external could be made static | used by GRR */
+ − /*lint -esym(765,p_pcs) | external could be made static | used by GRR */
+ − /*lint -esym(765,p_gsm) | external could be made static | used by GRR */
+ − /*lint -esym(765,p_control_dcs) | external could be made static | used by GRR */
+ − /*lint -esym(765,p_control_pcs) | external could be made static | used by GRR */
+ − /*lint -esym(765,p_control_gsm) | external could be made static | used by GRR */
+ − const SHORT p_dcs [MAX_CLASSES] = { P_CLASS_1_1800,
+ − P_CLASS_2_1800,
+ − P_CLASS_3_1800,
+ − P_CLASS_3_1800,
+ − P_CLASS_3_1800 };
+ −
+ −
+ − const SHORT p_pcs [MAX_CLASSES] = { P_CLASS_1_1900,
+ − P_CLASS_2_1900,
+ − P_CLASS_3_1900,
+ − P_CLASS_3_1900,
+ − P_CLASS_3_1900 };
+ −
+ − const SHORT p_gsm [MAX_CLASSES] = { P_CLASS_2_900,
+ − P_CLASS_2_900,
+ − P_CLASS_3_900,
+ − P_CLASS_4_900,
+ − P_CLASS_5_900 };
+ −
+ −
+ − /*
+ − * The following tables converts the air-interface coding of
+ − * the system information parameter MAX_TXPWR_CCCH to a value
+ − * in dBm depending on the frequency standard (GSM, DCS or PCS).
+ − */
+ − const UBYTE p_control_gsm [32] =
+ − { 39, /* 0 -> 39 dBm */
+ − 39, /* 1 -> 39 dBm */
+ − 39, /* 2 -> 39 dBm */
+ − 37, /* 3 -> 37 dBm */
+ − 35, /* 4 -> 35 dBm */
+ − 33, /* 5 -> 33 dBm */
+ − 31, /* 6 -> 31 dBm */
+ − 29, /* 7 -> 29 dBm */
+ − 27, /* 8 -> 27 dBm */
+ − 25, /* 9 -> 25 dBm */
+ − 23, /* 10 -> 23 dBm */
+ − 21, /* 11 -> 21 dBm */
+ − 19, /* 12 -> 19 dBm */
+ − 17, /* 13 -> 17 dBm */
+ − 15, /* 14 -> 15 dBm */
+ − 13, /* 15 -> 13 dBm */
+ − 11, /* 16 -> 11 dBm */
+ − 9, /* 17 -> 9 dBm */
+ − 7, /* 18 -> 7 dBm */
+ − 5, /* 19 -> 5 dBm */
+ − 5, /* 20 -> 5 dBm */
+ − 5, /* 21 -> 5 dBm */
+ − 5, /* 22 -> 5 dBm */
+ − 5, /* 23 -> 5 dBm */
+ − 5, /* 24 -> 5 dBm */
+ − 5, /* 25 -> 5 dBm */
+ − 5, /* 26 -> 5 dBm */
+ − 5, /* 27 -> 5 dBm */
+ − 5, /* 28 -> 5 dBm */
+ − 5, /* 29 -> 5 dBm */
+ − 5, /* 30 -> 5 dBm */
+ − 5 /* 31 -> 5 dBm */
+ − };
+ −
+ − const UBYTE p_control_dcs [32] =
+ − { 30, /* 0 -> 30 dBm */
+ − 28, /* 1 -> 28 dBm */
+ − 26, /* 2 -> 26 dBm */
+ − 24, /* 3 -> 24 dBm */
+ − 22, /* 4 -> 22 dBm */
+ − 20, /* 5 -> 20 dBm */
+ − 18, /* 6 -> 18 dBm */
+ − 16, /* 7 -> 16 dBm */
+ − 14, /* 8 -> 14 dBm */
+ − 12, /* 9 -> 12 dBm */
+ − 10, /* 10 -> 10 dBm */
+ − 8, /* 11 -> 8 dBm */
+ − 6, /* 12 -> 6 dBm */
+ − 4, /* 13 -> 4 dBm */
+ − 2, /* 14 -> 2 dBm */
+ − 0, /* 15 -> 0 dBm */
+ − 0, /* 16 -> 0 dBm */
+ − 0, /* 17 -> 0 dBm */
+ − 0, /* 18 -> 0 dBm */
+ − 0, /* 19 -> 0 dBm */
+ − 0, /* 20 -> 0 dBm */
+ − 0, /* 21 -> 0 dBm */
+ − 0, /* 22 -> 0 dBm */
+ − 0, /* 23 -> 0 dBm */
+ − 0, /* 24 -> 0 dBm */
+ − 0, /* 25 -> 0 dBm */
+ − 0, /* 26 -> 0 dBm */
+ − 0, /* 27 -> 0 dBm */
+ − 0, /* 28 -> 0 dBm */
+ − 36, /* 29 -> 36 dBm */
+ − 34, /* 30 -> 34 dBm */
+ − 32 /* 31 -> 32 dBm */
+ − };
+ −
+ − const UBYTE p_control_pcs [32] =
+ − { 30, /* 0 -> 30 dBm */
+ − 28, /* 1 -> 28 dBm */
+ − 26, /* 2 -> 26 dBm */
+ − 24, /* 3 -> 24 dBm */
+ − 22, /* 4 -> 22 dBm */
+ − 20, /* 5 -> 20 dBm */
+ − 18, /* 6 -> 18 dBm */
+ − 16, /* 7 -> 16 dBm */
+ − 14, /* 8 -> 14 dBm */
+ − 12, /* 9 -> 12 dBm */
+ − 10, /* 10 -> 10 dBm */
+ − 8, /* 11 -> 8 dBm */
+ − 6, /* 12 -> 6 dBm */
+ − 4, /* 13 -> 4 dBm */
+ − 2, /* 14 -> 2 dBm */
+ − 0, /* 15 -> 0 dBm */
+ − 0, /* 16 -> 0 dBm */
+ − 0, /* 17 -> 0 dBm */
+ − 0, /* 18 -> 0 dBm */
+ − 0, /* 19 -> 0 dBm */
+ − 0, /* 20 -> 0 dBm */
+ − 0, /* 21 -> 0 dBm */
+ − 33, /* 22 -> 33 dBm */
+ − 33, /* 23 -> 33 dBm */
+ − 33, /* 24 -> 33 dBm */
+ − 33, /* 25 -> 33 dBm */
+ − 33, /* 26 -> 33 dBm */
+ − 33, /* 27 -> 33 dBm */
+ − 33, /* 28 -> 33 dBm */
+ − 33, /* 29 -> 33 dBm */
+ − 33, /* 30 -> 33 dBm */
+ − 32 /* 31 -> 32 dBm */
+ − };
+ − /*lint +e765 / used by GRR */
+ −
+ − GLOBAL void att_calculate_c1 (UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − SHORT a;
+ − SHORT b;
+ − const SHORT *p;
+ − const UBYTE *p_control;
+ − SHORT power_offset = 0;
+ − UBYTE ms_power;
+ − SHORT offset;
+ −
+ − TRACE_FUNCTION ("att_calculate_c1()");
+ −
+ − ms_power = att_get_power ();
+ − switch (std)
+ − {
+ − case STD_900:
+ − case STD_EGSM:
+ − case STD_850:
+ − /*
+ − * The table for the power class conversion is GSM 900, E-GSM or GSM 850
+ − */
+ − p = p_gsm;
+ − /*
+ − * The table for the MAX_TXPWR_CCCH conversion is GSM 900, E-GSM or GSM 850
+ − */
+ − p_control = p_control_gsm;
+ − /*
+ − * The gsm_offset parameter can be set by a dynamic configuration command
+ − * and is not used.
+ − */
+ − offset = rr_data->dyn_config.gsm_offset;
+ − break;
+ −
+ − case STD_1900:
+ − p = p_pcs;
+ − p_control = p_control_pcs;
+ −
+ − /*
+ − * The dcs_offset parameter can be set by a dynamic configuration command
+ − * and is not used.
+ − */
+ − offset = rr_data->dyn_config.dcs_offset;
+ − break;
+ −
+ − case STD_1800:
+ − p = p_dcs;
+ − p_control = p_control_dcs;
+ −
+ − /* Ref 04.18 Section 10.5.2.35
+ − * Rest Octets IE includes parameters which are used by the mobile station
+ − * for cell selection and reselection purposes. It may also include the
+ − * POWER OFFSET parameter used by DCS 1800 Class 3 MS.
+ − */
+ −
+ − if (ms_power EQ 0x02 AND
+ − rr_data->nc_data[index].c2_par.power_off_ind)
+ − {
+ − power_offset = rr_data->nc_data[index].c2_par.power_off << 1;
+ − }
+ −
+ − offset = rr_data->dyn_config.dcs_offset;
+ − break;
+ −
+ − case STD_DUAL:
+ − case STD_DUAL_EGSM:
+ − /*
+ − * For dualband mobiles the calculation depends on the channel number
+ − */
+ − if (INRANGE(LOW_CHANNEL_1800,rr_data->nc_data[index].arfcn,HIGH_CHANNEL_1800))
+ − {
+ − /*
+ − * All DCS 1800 channels
+ − */
+ − p = p_dcs;
+ − p_control = p_control_dcs;
+ −
+ − /* Ref 04.18 Section 10.5.2.35 */
+ − if (ms_power EQ 0x02 AND
+ − rr_data->nc_data[index].c2_par.power_off_ind)
+ − {
+ − power_offset = rr_data->nc_data[index].c2_par.power_off << 1;
+ − }
+ −
+ − offset = rr_data->dyn_config.dcs_offset;
+ − }
+ − else
+ − {
+ − /*
+ − * All GSM 900 and E-GSM channels
+ − */
+ − p = p_gsm;
+ − p_control = p_control_gsm;
+ − offset = rr_data->dyn_config.gsm_offset;
+ − }
+ − break;
+ −
+ − case STD_DUAL_US:
+ − if (rr_data->nc_data[index].arfcn < LOW_CHANNEL_1900)
+ − {
+ − /*
+ − * Then it is a GSM 850 channel
+ − */
+ − p = p_gsm;
+ − p_control = p_control_gsm;
+ − offset = rr_data->dyn_config.gsm_offset;
+ − }
+ − else
+ − {
+ − /*
+ − * else it is a PCS 1900 channel
+ − */
+ − p = p_pcs;
+ − p_control = p_control_pcs;
+ −
+ − offset = rr_data->dyn_config.dcs_offset;
+ − }
+ − break;
+ −
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − case STD_850_1800:
+ − case STD_850_900_1800:
+ − if (INRANGE(LOW_CHANNEL_1800,rr_data->nc_data[index].arfcn,HIGH_CHANNEL_1800))
+ − {
+ − /*
+ − * Then it is a DCS 1800 channel
+ − */
+ − p = p_dcs;
+ − p_control = p_control_dcs;
+ −
+ − if (ms_power EQ 0x02 AND
+ − rr_data->nc_data[index].c2_par.power_off_ind)
+ − {
+ − power_offset = rr_data->nc_data[index].c2_par.power_off << 1;
+ − }
+ −
+ − offset = rr_data->dyn_config.dcs_offset;
+ − }
+ − else
+ − {
+ − /*
+ − * Else it is a GSM 900 and E-GSM or GSM 850 channel
+ − */
+ − p = p_gsm;
+ − p_control = p_control_gsm;
+ − offset = rr_data->dyn_config.gsm_offset;
+ − }
+ − break;
+ −
+ − case STD_900_1900:
+ − case STD_850_900_1900:
+ − /*
+ − * For dualband mobiles the calculation depends on the channel number
+ − */
+ − if (INRANGE(LOW_CHANNEL_1900,rr_data->nc_data[index].arfcn,HIGH_CHANNEL_1900))
+ − {
+ − /*
+ − * All PCS 1900 channels
+ − */
+ − p = p_pcs;
+ − p_control = p_control_pcs;
+ −
+ − offset = rr_data->dyn_config.dcs_offset;
+ − }
+ − else
+ − {
+ − /*
+ − * All GSM 900 and E-GSM or GSM 850 channels
+ − */
+ − p = p_gsm;
+ − p_control = p_control_gsm;
+ − offset = rr_data->dyn_config.gsm_offset;
+ − }
+ − break;
+ − #endif
+ −
+ − default:
+ − /*
+ − * Just for LINT
+ − */
+ − rr_data->nc_data[index].c1 = 0;
+ − return;
+ − }
+ −
+ − /*
+ − * Parameter A indicates the difference between current fieldstrength and
+ − * the minimum expected from the infrastructure. It indicates how good is the
+ − * receiving part.
+ − */
+ − #ifdef GPRS
+ − if(rr_data->gprs_data.use_c31)
+ − {
+ − a = (SHORT) rr_data->nc_data[index].rxlev -
+ − (SHORT) rr_data->nc_data[index].cr_par.gprs_rxlev_access_min;
+ − }
+ − else
+ − {
+ − #endif
+ − a = (SHORT) rr_data->nc_data[index].rxlev -
+ − (SHORT) rr_data->nc_data[index].select_para.rxlev_access_min;
+ − #ifdef GPRS
+ − }
+ − #endif
+ −
+ − /*
+ − * Parameter B indicates the difference between the maximum sending power
+ − * expected by the infrastructure and the maximum power which is possible by
+ − * the mobile. In the formula the maximum of B and zero is taking in account.
+ − * If B is negative, the MS has enough power to reach the base station,
+ − * because it has more power than expected and it will not have any affect
+ − * in the C1 calculation. If B is positive, the MS will send with less power
+ − * than expected by the infrastructure. Then this difference is subtracted
+ − * from A. The maximum power is expected at the boarder of the cell. If the
+ − * mobile is nearer to the base station, a lower power is needed. On the
+ − * other hand will the mobile receive the base station with a higher
+ − * fieldstrength as on the boarder. So this surplus compensates the lower
+ − * sending capabilities.
+ − */
+ −
+ − #ifdef GPRS
+ − if(rr_data->gprs_data.use_c31)
+ − {
+ − b = p_control [rr_data->nc_data[index].cr_par.gprs_ms_txpwr_max_cch] - p[ms_power];
+ − }
+ − else
+ − {
+ − #endif
+ − b = p_control [rr_data->nc_data[index].select_para.ms_txpwr_max_cch] +
+ − power_offset - p[ms_power];
+ − #ifdef GPRS
+ − }
+ − #endif
+ −
+ − /*
+ − * The resulting path loss criterion is C1 = A - MAX (B,0)
+ − *
+ − * An additional C1_offset and offset is not really used. This values can
+ − * be configured and may compensate RF problems.
+ − */
+ − rr_data->nc_data[index].c1 = a - att_max (b, 0) + rr_data->c1_offset +
+ − offset;
+ −
+ − /*
+ − * To avoid too many reselection the C1 of a neighbourcell is decreased,
+ − * that means beside the GSM specifications a threshold value is introduced.
+ − * If a test SIM card is inserted, this is not used.
+ − */
+ − if (index NEQ SC_INDEX AND index NEQ CR_INDEX AND !dat_test_sim_available())
+ − rr_data->nc_data[index].c1 -= TWO;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_calculate_c2 |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The reselection criterion C2 is used for cell reselection
+ − only and is defined by:
+ −
+ − C2 = C1 + CELL_RESELECT_OFFSET -
+ − TEMPORARY OFFSET * H(PENALTY_TIME - T)
+ −
+ − for PENALTY_TIME <> 11111
+ −
+ − C2 = C1 - CELL_RESELECT_OFFSET
+ −
+ − for PENALTY_TIME = 11111
+ −
+ − where
+ −
+ − For non-serving cells:
+ −
+ − H(x) = 0 for x < 0
+ −
+ − = 1 for x = 0
+ −
+ − For serving cells:
+ −
+ − H(x) = 0
+ −
+ − T is a timer implemented for each cell in the list of strongest carriers.
+ − T shall be started from zero at the time the cell is placed by the MS on the
+ − list of strongest carriers, except when the previous serving cell is placed
+ − on the list of strongest carriers at cell reselection. In this, case,
+ − T shall be set to the value of PENALTY_TIME (i.e. expired).
+ − CELL_RESELECT_OFFSET applies an offset to the C2 reselection criterion for that cell.
+ −
+ − NOTE: CELL_RESELECT_OFFSET may be used to give different priorities to different
+ − bands when multiband operation is used.
+ −
+ − TEMPORARY_OFFSET applies a negative offset to C2 for the duration of PENALTY_TIME
+ − after the timer T has started for that cell.
+ −
+ − PENALTY_TIME is the duration for which TEMPORARY_OFFSET applies The all ones bit
+ − pattern on the PENALTY_TIME parameter is reserved to change the sign of
+ − CELL_RESELECT_OFFSET and the value of TEMPORARY_OFFSET is ignored as indicated
+ − by the equation defining C2.
+ −
+ − CELL_RESELECT_OFFSET, TEMPORARY_OFFSET, PENALTY_TIME and CELL_BAR_QUALIFY are
+ − optionally broadcast on the BCCH of the cell. If not broadcast, the default values
+ − are CELL_BAR_QUALIFY = 0, and C2 = C1. The use of C2 is described in GSM 03.22.
+ −
+ − These parameters are used to ensure that the MS is camped on the cell with which it
+ − has the highest probability of successful communication on uplink and downlink.
+ −
+ − REFERENCE: GSM 5.08, chapter 6.4 Criteria for cell selection and reselection
+ −
+ − */
+ −
+ − static void att_calculate_c2 (UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − USHORT h;
+ − T_NC_DATA *rrd;
+ −
+ − TRACE_FUNCTION ("att_calculate_c2()");
+ −
+ − /*
+ − * get a pointer to the data of the cell
+ − */
+ − rrd = &rr_data->nc_data[index];
+ −
+ − /*
+ − * Calculate C1 as a base if no C2 parameters are available
+ − */
+ − att_calculate_c1 (index);
+ −
+ − if ( rrd->c2_par.param_ind EQ 0 )
+ − {
+ − /*
+ − * C2 is equal C1 if no C2 parameters are available.
+ − */
+ − rrd->c2 = rrd->c1;
+ − TRACE_C1_C2(index);
+ − return;
+ − }
+ −
+ − /*
+ − * Special case penalty time = 0b11111
+ − */
+ − if (rrd->c2_par.penalty_time EQ 31)
+ − {
+ − rrd->c2 = rrd->c1 - 2 * rrd->c2_par.cell_reselect_offset;
+ − TRACE_C1_C2(index);
+ − return;
+ − }
+ −
+ − /*
+ − * Calculate x for H(x) and set H(x)
+ − */
+ − if ((USHORT)((rrd->c2_par.penalty_time + 1) * 20) >=
+ − (USHORT)(rrd->avail_time/PERIOD_1_SEC))
+ − h = 1;
+ − else
+ − h = 0;
+ −
+ − /*
+ − * for the serving cells H(x) us ever zero.
+ − */
+ − if (index EQ SC_INDEX)
+ − h = 0;
+ −
+ − /*
+ − * Calculate C2 using the formula
+ − */
+ − rrd->c2 = rrd->c1 + 2 * rrd->c2_par.cell_reselect_offset -
+ − 10 * rrd->c2_par.temp_offset * h;
+ − if (h EQ 1 AND rrd->c2_par.temp_offset EQ 7)
+ − {
+ − /*
+ − * temp offset = 7 means C2 = - infinite
+ − */
+ − rrd->c2 = -127;
+ − }
+ −
+ − TRACE_C1_C2(index);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_calculate_c2_diff |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : For cell reselection decision the C2 difference between
+ − a neighbourcell and the serving cell is calculated. This
+ − function shall be called only if both cells are in a
+ − different location area.
+ −
+ − If no SIM card is inserted (that means limited service)
+ − the threshold value is not taken in account. This is
+ − beside the GSM specifications and shall ensure a better
+ − coverage of the mobile. It leads not to more signalling
+ − on the air-interface.
+ −
+ − REFERENCE: GSM 5.08, chapter 6.6.2 Path loss criteria and timings for
+ − cell re-selection
+ −
+ − */
+ −
+ − static SHORT att_calculate_c2_diff (UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − SHORT delta;
+ −
+ − TRACE_FUNCTION ("att_calculate_c2_diff()");
+ −
+ − /*
+ − * calculate delta
+ − */
+ − delta = rr_data->nc_data[index].c2 - rr_data->nc_data[SC_INDEX].c2;
+ −
+ − /* According to 3GPP 05.08 Sec 6.8 MS is supposed to use ZERO for
+ − * CELL_RESELECT_HYSTERESIS when in Limited service.
+ − * c) The MS shall perform cell reselection at least among the cells of
+ − * the PLMN of the cell on which the MS has camped, according to the
+ − * algorithm of 3GPP TS 03.22, except that a zero value of
+ − * CELL_RESELECT_HYSTERESIS shall be used.
+ − */
+ −
+ − if ((rr_data->ms_data.imsi_available) AND
+ − (rr_data->ms_data.rr_service NEQ LIMITED_SERVICE))
+ − delta -= 2*rr_data->nc_data[SC_INDEX].select_para.cell_resel_hyst;
+ −
+ − /*
+ − * return the difference
+ − */
+ − return delta;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_calculate_digits |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function calculates the number of the digits in a
+ − mobile identity infoelement. The input parameter contains
+ − one BCD digit in each field entry.
+ −
+ − */
+ −
+ − static UBYTE att_calculate_digits (UBYTE *digits)
+ − {
+ − UBYTE i = 0;
+ −
+ − TRACE_FUNCTION ("att_calculate_digits()");
+ −
+ − /*
+ − * Check whether the end is detected
+ − */
+ − while (digits[i] < 0x0A AND i < 16)
+ − i++;
+ −
+ − return i;
+ − }
+ −
+ −
+ −
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_cell_barred_status_cr_no_cr |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Checks whether a cell is barred or not for cell reselection
+ − purposes.
+ −
+ − CELL_BAR QUALIFY CELL_BAR ACCESS CS priority Status for CR
+ − 0 0 normal normal
+ − 0 1 barred barred
+ − 1 0 low normal (see note 2)
+ − 1 1 low normal (see note 2)
+ −
+ − If all the following conditions are met, then the "Cell selection priority"
+ − and the "Status for cell reselection" shall be set to normal:
+ −
+ − - the cell belongs to the MS HPLMN;
+ − - the MS is in cell test operation mode;
+ − - the CELL_BAR_ACCESS is set to "1";
+ − - the CELL_BAR_QUALIFY is set to "0";
+ − - the Access Control class 15 is barred.
+ −
+ − NOTE 1: A low priority cell is only selected if there are no
+ − suitable cells
+ − of normal priority (see GSM 03.22).
+ − NOTE 2: Two identical semantics are used for cross phase
+ − compatibility reasons.
+ − This allows an operator to declare a cell always as a low
+ − priority one for a phase 2 MS, but keeps the opportunity
+ − for an operator to decide wether a phase 1 MS is permitted
+ − to camp on such a cell or not.
+ −
+ − REFERENCE: GSM 5.08, chapter 9, table 1a
+ −
+ −
+ − */
+ − GLOBAL BOOL att_cell_barred_status_cr_no_cr (UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ −
+ − TRACE_FUNCTION ("att_cell_barred_status_cr_no_cr()");
+ −
+ − TRACE_EVENT_P5 ("[%u]%u cbq=%u cba=%u access class=%04x",
+ − rr_data->nc_data[index].arfcn,
+ − index,
+ − rr_data->nc_data[index].c2_par.cbq,
+ − rr_data->nc_data[index].rach.cell_bar_access,
+ − rr_data->nc_data[index].rach.ac);
+ −
+ − /*
+ − * check first the special case
+ − */
+ − if (dat_hplmn (rr_data->nc_data[index].lai.mcc,
+ − rr_data->nc_data[index].lai.mnc) AND
+ − rr_data->cell_test_operation AND
+ − rr_data->nc_data[index].c2_par.cbq EQ 0 AND
+ − rr_data->nc_data[index].rach.cell_bar_access EQ 1)
+ − {
+ − /*
+ − * access class 15 is set
+ − */
+ − if (rr_data->nc_data[index].rach.ac & 0x8000)
+ − return FALSE;
+ − }
+ −
+ − /*
+ − * the cell reselection status is normal. According
+ − * to the table above the cell is only barred if
+ − * the cell bar qualifier is set to 0 and the
+ − * cell bar access is set to 1.
+ − */
+ − if((rr_data->nc_data[index].rach.cell_bar_access EQ BARRED_YES) AND
+ − rr_data->nc_data[index].c2_par.cbq EQ CBQ_NO)
+ − return TRUE;
+ − else
+ − return FALSE;
+ −
+ −
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_cell_in_data_structures|
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After reception of new measurement values from layer 1
+ − this function checks whether a neighbourcell indicated
+ − with the parameter index is stored inside the RR
+ − storage area.
+ −
+ − */
+ −
+ − static BOOL att_cell_in_data_structures (T_MPH_MEASUREMENT_IND *report,
+ − UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − USHORT i;
+ −
+ − TRACE_FUNCTION ("att_cell_in_data_structures()");
+ −
+ − /*
+ − * check for all areas
+ − */
+ − for (i = 0; i < 6; i++)
+ − {
+ − /*
+ − * If a cell is stored in the data area
+ − */
+ − if (rr_data->nc_data[i].bcch_status NEQ EMPTY)
+ − {
+ − /*
+ − * check the channel number
+ − */
+ − if (rr_data->nc_data[i].arfcn EQ report->ncells.arfcn[index])
+ − return TRUE;
+ − }
+ − }
+ − return FALSE;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_remove_multiple_channels |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : During fieldtests it had happens that the measurements from
+ − layer 1 may contain the serving cell as a neighbourcell
+ − (which is possible during connection and can be forwarded
+ − to RR after connection). This leads to double stored data
+ − in RR.
+ −
+ − */
+ −
+ − GLOBAL void att_remove_multiple_channels (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − USHORT i;
+ −
+ − TRACE_FUNCTION ("att_remove_multiple_channels()");
+ −
+ − /*
+ − * For all neighbourcells in the measurement report
+ − */
+ − for (i = 0; i < 6; i++)
+ − {
+ − if (rr_data->nc_data[i].arfcn EQ rr_data->nc_data[SC_INDEX].arfcn)
+ − {
+ − /*
+ − * If the neighbourcell channel number is equal to the serving
+ − * cell, clear the whole storage area.
+ − */
+ − rr_data->nc_data[i].arfcn = 0;
+ − rr_data->nc_data[i].bcch_status = EMPTY;
+ − }
+ − }
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_cell_in_measure_report |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function checks whether a cell in the RR storage
+ − area is still in a new measurement report and returns
+ − one of the three possibilities:
+ −
+ − CELL_IS_INSERTED the cell is inside with the same BSIC
+ − BSIC_HAS_CHANGED the cell is inside, but the BSIC has changed
+ − CELL_IS_NOT_INSERTED the cell is not longer inside.
+ −
+ − */
+ −
+ − static UBYTE att_cell_in_measure_report (T_MPH_MEASUREMENT_IND *report,
+ − UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE i;
+ − UBYTE bsic;
+ −
+ − TRACE_FUNCTION ("att_cell_in_measure_report()");
+ −
+ − /*
+ − * check all neighbourcells in the measurent report
+ − */
+ − for (i = 0; i < report->ncells.no_of_ncells; i++)
+ − {
+ − /*
+ − * Check first the channel number
+ − */
+ − if (rr_data->nc_data[index].arfcn EQ report->ncells.arfcn[i])
+ − {
+ − /*
+ − * Update fieldstrength value in the RR storage area
+ − */
+ − rr_data->nc_data[index].rxlev = report->ncells.rx_lev[i];
+ −
+ − /*
+ − * Check whether the BSIC has been changed. Layer 1 sends an
+ − * 8 bit value, but only the lower 6 bits are significant.
+ − */
+ − bsic = rr_data->nc_data[index].bsic & 0x3F;
+ − if (bsic EQ (report->ncells.bsic[i] & 0x3F))
+ − return CELL_IS_INSERTED;
+ − else
+ − {
+ − /*
+ − * If it is a changed BSIC, update the value in the RR storage area.
+ − */
+ − rr_data->nc_data[index].bsic = report->ncells.bsic[i] & 0x3F;
+ − return BSIC_HAS_CHANGED;
+ − }
+ − }
+ − }
+ − /*
+ − * Not found
+ − */
+ − return CELL_IS_NOT_INSERTED;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_bcch_carrier_cco |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : During a Cell Change Order procedure:
+ − After reception of all needed system information messages
+ − a carrier is checked for cell selection purposes.
+ −
+ − */
+ −
+ − #ifdef GPRS
+ − LOCAL void att_check_bcch_carrier_cco (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_check_bcch_carrier_cco()");
+ −
+ − dat_rrgrr_data_ind (rr_data->gprs_data.dl_data_ind);
+ − rr_data->gprs_data.dl_data_ind = NULL;
+ − att_build_idle_req (CR_INDEX, MODE_CELL_SELECTION);
+ − SET_STATE (STATE_ATT, ATT_IDLE);
+ − dat_att_cell_selected();
+ − if (att_gprs_is_avail())
+ − {
+ − att_signal_gprs_support();
+ − }
+ − att_start_registration_timer ();
+ − }
+ − #endif
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_bcch_carrier_si |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After reception of all needed system information messages
+ − a carrier is checked for cell selection purposes.
+ − Perform the check on the presence of the SIs.
+ −
+ − */
+ −
+ − GLOBAL void att_check_bcch_carrier_si (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − if (rr_data->cr_data.cd.sys_info_read EQ ALL_SYS_INFO_READ)
+ − {
+ − att_check_bcch_carrier ();
+ − }
+ − }
+ −
+ − GLOBAL void att_store_plmn_in_found_list (T_loc_area_ident *lai)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_FOUND_ELEMENT *found;
+ −
+ − /* Implements RR Clone findings #8 */
+ − BOOL cell_ok = !att_cell_barred_status_cr_no_cr(CR_INDEX) AND
+ − (rr_data->nc_data[CR_INDEX].c1 > 0) AND
+ − att_priority_check ();
+ −
+ −
+ −
+ − if ( (found = att_plmn_in_found_list (lai->mcc, lai->mnc)) NEQ NULL )
+ − {
+ − if ( found->cell_ok EQ 0 AND /* old cell did not allow access */
+ − cell_ok NEQ 0 ) /* new cell does allow access */
+ − {
+ − ; /* a better cell has been found */
+ − }
+ − else
+ − {
+ − found = NULL;
+ − }
+ − }
+ − else
+ − if ( rr_data->sc_data.found_entries < MAX_PLMN AND att_check_network(lai) )
+ − {
+ − found = &rr_data->sc_data.found[rr_data->sc_data.found_entries];
+ − rr_data->sc_data.found_entries++;
+ − }
+ − else
+ − {
+ − TRACE_EVENT ("cannot store PLMN" );
+ − }
+ −
+ − if ( found )
+ − {
+ − att_save_found_plmn (
+ − found,
+ − lai->mcc, lai->mnc,
+ − rr_data->nc_data[CR_INDEX].arfcn,
+ − rr_data->nc_data[CR_INDEX].rxlev, lai->lac, (UBYTE)cell_ok);
+ − }
+ − }
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_bcch_carrier |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After reception of all needed system information messages
+ − a carrier is checked for cell selection purposes.
+ −
+ − */
+ −
+ − GLOBAL void att_check_bcch_carrier (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_loc_area_ident * lai;
+ − BOOL plmn_ok = FALSE;
+ − BOOL cell_ok,prio_ok;
+ − UBYTE tried_to_reach_full_try_limited = (rr_data->cs_data.scan_mode EQ CS_SECOND_SCAN);
+ −
+ − TRACE_FUNCTION ("att_check_bcch_carrier()");
+ −
+ − /*
+ − * stop the timer which controls the reception of
+ − * all system information messages during cell selection.
+ − */
+ − TIMERSTOP (T_RESELECT);
+ −
+ − /*
+ − * If procedure RR Network Controlled Cell Change Order
+ − * is running then call a different function and return.
+ − */
+ − #ifdef GPRS
+ − if ( rr_data->gprs_data.tbf_est EQ TBF_EST_CCO )
+ − {
+ − att_check_bcch_carrier_cco();
+ − return;
+ − }
+ − #endif
+ −
+ − lai = &rr_data->nc_data[CR_INDEX].lai;
+ −
+ − /*
+ − * calculate the path loss criterion
+ − */
+ − att_calculate_c1 (CR_INDEX);
+ − TRACE_C1(CR_INDEX);
+ −
+ − /*
+ − * the check depends on the service which has been requested
+ − * by RR. This can be the same as the service requested by MM,
+ − */
+ − switch (rr_data->ms_data.req_mm_service)
+ − {
+ − case FUNC_PLMN_SRCH:
+ − /*
+ − * For full service the mobile country code and the mobile network code
+ − * must match with the PLMN identification requested by MM.
+ − */
+ − plmn_ok = dat_plmn_equal_req (lai->mcc,
+ − lai->mnc,
+ − rr_data->ms_data.plmn.mcc,
+ − rr_data->ms_data.plmn.mnc);
+ −
+ − TRACE_EVENT_P6 ( "MCC/MNC r=%x%x%x/%x%x%x",
+ − rr_data->ms_data.plmn.mcc[0],
+ − rr_data->ms_data.plmn.mcc[1],
+ − rr_data->ms_data.plmn.mcc[2],
+ − rr_data->ms_data.plmn.mnc[0],
+ − rr_data->ms_data.plmn.mnc[1],
+ − rr_data->ms_data.plmn.mnc[2]);
+ − TRACE_EVENT_P7 ( "MCC/MNC i=%x%x%x/%x%x%x %s",
+ − lai->mcc[0],
+ − lai->mcc[1],
+ − lai->mcc[2],
+ − lai->mnc[0],
+ − lai->mnc[1],
+ − lai->mnc[2],
+ − plmn_ok?"OK":"");
+ −
+ −
+ − if (!plmn_ok AND
+ − (GET_STATE(STATE_ATT) EQ ATT_CS2) AND
+ − (rr_data->ms_data.rr_service EQ LIMITED_SERVICE OR
+ − rr_data->ms_data.rr_service EQ NO_SERVICE) AND
+ − (rr_data->cs_data.check_hplmn EQ TRUE) AND
+ − (((rr_data->ms_data.operation_mode >> SHIFT_FOR_SEARCH_OFFSET) & 1) EQ M_AUTO))
+ − {
+ − /* Compare with HPLMN */
+ − if (rr_data->ms_data.imsi_available)
+ − {
+ − plmn_ok = dat_plmn_equal_req (lai->mcc,
+ − lai->mnc,
+ − &rr_data->ms_data.imsi.ident_dig[0],
+ − &rr_data->ms_data.imsi.ident_dig[3]);
+ − if (plmn_ok)
+ − {
+ − TRACE_EVENT ("HPLMN Matched");
+ − }
+ − }
+ − /* compare with AHPLMN */
+ − if(!plmn_ok AND rr_data->ms_data.ahplmn.v_plmn EQ TRUE)
+ − {
+ − plmn_ok = dat_plmn_equal_req (lai->mcc,
+ − lai->mnc,
+ − rr_data->ms_data.ahplmn.mcc,
+ − rr_data->ms_data.ahplmn.mnc);
+ − if (plmn_ok)
+ − {
+ − TRACE_EVENT ("AHPLMN Matched");
+ − }
+ − }
+ − /* Compare with LRPLMN */
+ − if (!plmn_ok AND rr_data->cs_data.last_reg_plmn.v_plmn EQ V_PLMN_PRES)
+ − {
+ − plmn_ok = dat_plmn_equal_req (lai->mcc,
+ − lai->mnc,
+ − rr_data->cs_data.last_reg_plmn.mcc,
+ − rr_data->cs_data.last_reg_plmn.mnc);
+ − if (plmn_ok)
+ − {
+ − TRACE_EVENT ("LRPLMN Matched");
+ − }
+ − }
+ − /* Compare with Last Serving Cell PLMN from White List */
+ − if ((!plmn_ok) AND (rr_data->sc_data.mm_started EQ RR_ORIGINATED) AND
+ − (rr_data->ms_data.plmn.v_plmn EQ V_PLMN_PRES))
+ − {
+ − plmn_ok = dat_plmn_equal_req (lai->mcc,
+ − lai->mnc,
+ − rr_data->cs_data.white_list.last_sc_lac.mcc,
+ − rr_data->cs_data.white_list.last_sc_lac.mnc);
+ − if (plmn_ok)
+ − {
+ − TRACE_EVENT ("White List PLMN Matched");
+ − }
+ − }
+ − }
+ −
+ − if (plmn_ok)
+ − tried_to_reach_full_try_limited = FALSE;
+ − /*lint -fallthrough*/
+ −
+ − case FUNC_LIM_SERV_ST_SRCH:
+ − if((rr_data->ms_data.req_mm_service EQ FUNC_LIM_SERV_ST_SRCH) OR
+ − tried_to_reach_full_try_limited)
+ − {
+ − /*
+ − * The mobile shall reach limited service. This is the case
+ − * if no full service is possible or no SIM card is inserted.
+ − *
+ − * The network check is in principle ever true. The function
+ − * att_check_network realises a SW shielding to reject public
+ − * networks in final type approvals.
+ − *
+ − * The criterion for the cell checking is a non-barred cell and
+ − * a positive path loss criterion.
+ − */
+ − plmn_ok = att_check_network (lai);
+ − /* Implements RR Clone findings #8 */
+ − cell_ok = !att_cell_barred_status_cr_no_cr (CR_INDEX) AND
+ − (rr_data->nc_data[CR_INDEX].c1 > 0);
+ −
+ − }
+ − else
+ − {
+ − /*
+ − * Full service is requested.
+ − *
+ − * First store PLMN of the cell in the PLMN found list
+ − * if the list is not stored yet and not full. SW shielding
+ − * is taking into account for FTA purposes.
+ − *
+ − * If the cell selection fails and no carrier was suitable,
+ − * at least a list of the found channels is available.
+ − */
+ − if(dat_forb_lai_check (CR_INDEX))
+ − {
+ − att_store_plmn_in_found_list (lai);
+ − }
+ − else
+ − {
+ − TRACE_EVENT ("Do not store Forbidden LAI PLMN in the found PLMN list");
+ − }
+ −
+ − /*
+ − * The requirements for the cell are: it must be a non-barred cell,
+ − * it must have a positive path loss criterion. The cell shall not
+ − * be inside of a forbidden location area and it must be a high priority
+ − * cell if it is the first scan first attempt.
+ − */
+ − /* Implements RR Clone findings #8 */
+ − cell_ok = !att_cell_barred_status_cr_no_cr (CR_INDEX) AND
+ − (rr_data->nc_data[CR_INDEX].c1 > 0);
+ −
+ −
+ − prio_ok = att_priority_check ();
+ −
+ − TRACE_EVENT_P1 ( "cell_ok = %d", cell_ok);
+ −
+ − TRACE_EVENT_P1 ( "prio_ok = %d", prio_ok);
+ − /*
+ − * if the network and cell are okay, but the priority check fails,
+ − * this cell is a low priority cell and can be checked in the
+ − * first scan second attempt.
+ − */
+ − if (plmn_ok AND cell_ok AND !prio_ok)
+ − cs_set_attributes (LOW_PRIORITY_CELL,
+ − rr_data->nc_data[CR_INDEX].arfcn);
+ −
+ − /*
+ − * if the bcch carrier is not suitable for full service,
+ − * but suitable for limited service, try it
+ − * for emergency services in the second attempt.
+ − * priority check is not required for emergency cells
+ − */
+ − if (cell_ok AND !plmn_ok)
+ − cs_set_attributes (EMERGENCY_CELL,
+ − rr_data->nc_data[CR_INDEX].arfcn);
+ −
+ − /* Update the cell_ok flag with with priority check flag */
+ − if(cell_ok)
+ − cell_ok = prio_ok;
+ −
+ − /* Checking access class and the list for cause #12 for */
+ − /* selection of the cell to camp on may not meet the */
+ − /* requirements of GSM 03.22 subclause 3.5.3, 3.5.4. */
+ − /* However, it is perfectly correct to consider the */
+ − /* forbidden location areas for roaming list here. */
+ −
+ −
+ − /*
+ − * If cell selection has been launched by MM in manual mode
+ − * we don't check against the forbidden location area lists.
+ − * This is to allow the user to select a network manually
+ − * which is a member of a forbidden list.
+ − * Maybe the distinction manual/automatic is not worth the
+ − * effort, as in contradiction to the forbidden list for
+ − * cause #11 the user can get rid of the forbidden lists for
+ − * cause #12 and #13 either by a power cycle or by simply
+ − * waiting some hours.
+ − */
+ − if (!rr_data->sc_data.mm_started OR
+ − ((rr_data->ms_data.operation_mode >> SHIFT_FOR_SEARCH_OFFSET) & 1) EQ M_AUTO)
+ − {
+ − /*
+ − * RR started cell selection, or MM started but in automatic
+ − * so ensure no barred LA is entered.
+ − */
+ − if (cell_ok)
+ − {
+ − cell_ok = (dat_forb_lai_check (CR_INDEX) AND
+ − dat_roam_forb_lai_check (CR_INDEX));
+ −
+ − /* Candidate for Limited service */
+ − if(!cell_ok)
+ − cs_set_attributes (EMERGENCY_CELL,
+ − rr_data->nc_data[CR_INDEX].arfcn);
+ − }
+ − }
+ −
+ − #if defined (_SIMULATION_)
+ − {
+ − /*
+ − * Trace output for cell selection analysis
+ − */
+ − /* Implements RR Clone findings #8 */
+ − BOOL cb_stat = att_cell_barred_status_cr_no_cr (CR_INDEX);
+ −
+ − BOOL for_lai = dat_forb_lai_check (CR_INDEX);
+ − BOOL prio = att_priority_check();
+ − TRACE_EVENT_WIN_P1 ( "CB_STAT = %d", cb_stat);
+ − TRACE_EVENT_WIN_P1 ( "C1 = %d", rr_data->nc_data[CR_INDEX].c1);
+ − TRACE_EVENT_WIN_P1 ( "FOR_LAI = %d", for_lai);
+ − TRACE_EVENT_WIN_P1 ( "PRIO = %d", prio);
+ − }
+ − #endif /* _SIMULATION_ */
+ − }
+ − break;
+ −
+ − default:
+ − /*
+ − * MM has requested the list of the available PLMNs. The algorithm
+ − * for this is to indicate all carriers as "not suitable" for cell
+ − * selection. As a result the list of the available PLMNs is build
+ − * after scanning all carriers.
+ − *
+ − * First step is to store PLMN in plmn found list if the PLMN is not
+ − * stored yet and the list is not full. SW shielding is used for
+ − * FTA purposes.
+ − */
+ − att_store_plmn_in_found_list (lai);
+ −
+ − /*
+ − * Define the channel as a candidate for limited service.
+ − */
+ − cs_set_attributes (EMERGENCY_CELL,
+ − rr_data->nc_data[CR_INDEX].arfcn);
+ −
+ − /*
+ − * Remove all channels of the neighbourcell list from the power campaign
+ − * list. If the configuration of the network is sensible, this cell will
+ − * contain also channels of the same network. So it is not necessary to
+ − * scan this channels. The available PLMN list is the same, it consumes
+ − * only time.
+ − */
+ − cs_del_list (&rr_data->cr_data.cd.ncell_list);
+ − plmn_ok = FALSE;
+ − cell_ok = FALSE;
+ −
+ − /*
+ − * start synchronisation attempt to the next cell
+ − */
+ − memset (&rr_data->nc_data[CR_INDEX], 0, sizeof (T_NC_DATA));
+ −
+ − if (rr_data->ms_data.rr_service EQ NO_SERVICE)
+ − {
+ − SET_STATE (STATE_ATT, ATT_CS1);
+ − }
+ − srv_clear_list (&rr_data->cr_data.cd.ncell_list);
+ − cs_sync_next ();
+ − return;
+ − }
+ −
+ − if( (!tried_to_reach_full_try_limited) AND (rr_data->cs_data.scan_mode EQ CS_SECOND_SCAN))
+ − {
+ − if(!cell_ok)
+ − {
+ − /*
+ − * plmn is OK, cell is not OK for full service
+ − * however cell may be OK for limited service
+ − */
+ − tried_to_reach_full_try_limited = TRUE;
+ − /* Implements RR Clone findings #8 */
+ − cell_ok = !att_cell_barred_status_cr_no_cr (CR_INDEX) AND
+ − (rr_data->nc_data[CR_INDEX].c1 > 0);
+ −
+ − }
+ − }
+ −
+ − /*
+ − * Trace output for cell selection analysis
+ − */
+ − TRACE_EVENT_P1 ( "req.service = %s", _rr_str_FUNC[rr_data->ms_data.req_mm_service]);
+ − TRACE_EVENT_P1 ( "plmn_ok = %d", plmn_ok );
+ − TRACE_EVENT_P1 ( "cell_ok = %d", cell_ok);
+ −
+ − /*
+ − * checking of the requirements has been done. The result
+ − * is defined by plmn_ok and cell_ok.
+ − */
+ − if (plmn_ok AND cell_ok)
+ − {
+ − /*
+ − * Network and Cell are okay. Just use the cell
+ − * set RR service according to the checked service
+ − */
+ − /* If the MS reselects back onto the same SC and T3122 is still running, it should */
+ − /* be left running. If a new cell is selected, this timer should be stopped. */
+ −
+ − if( rr_data->nc_data[SC_INDEX].arfcn NEQ rr_data->nc_data[CR_INDEX].arfcn)
+ − {
+ − TIMERSTOP(T3122);
+ − }
+ −
+ − if (rr_data->ms_data.req_mm_service EQ FUNC_LIM_SERV_ST_SRCH OR
+ − (tried_to_reach_full_try_limited))
+ − {
+ − rr_data->ms_data.rr_service = LIMITED_SERVICE;
+ − }
+ − else
+ − {
+ − rr_data->ms_data.rr_service = FULL_SERVICE;
+ − }
+ −
+ − TRACE_EVENT_P4 ("NEW SC [%d]->[%d] (rMM=%s RRs=%s)",
+ − rr_data->nc_data[SC_INDEX].arfcn,
+ − rr_data->nc_data[CR_INDEX].arfcn,
+ − _rr_str_FUNC[rr_data->ms_data.req_mm_service],
+ − _rr_str_SERVICE[rr_data->ms_data.rr_service]);
+ −
+ − /*
+ − * set the result in the CR_INDEX area and copy then all
+ − * stuff to the SC_INDEX area.
+ − */
+ − rr_data->nc_data[CR_INDEX].bcch_status = DECODED;
+ − rr_data->nc_data[CR_INDEX].bcch_counter = 0;
+ − rr_data->nc_data[CR_INDEX].c1_counter = 0;
+ − att_copy_cr_data ();
+ − memcpy (&rr_data->sc_data.cd, &rr_data->cr_data.cd,
+ − sizeof (T_CELL_DATA));
+ − rr_data->nc_data[CR_INDEX].arfcn = NOT_PRESENT_16BIT;
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − rr_data->sc_data.ba_index = rr_data->cr_data.ba_index;
+ − rr_data->cr_data.ba_index = NOT_PRESENT_8BIT;
+ − #endif
+ −
+ − /* CSI-LLD section:4.1.1.10
+ − * This function updates the black list after the first successful
+ − * FUNC_PLMN_SEARCH
+ − */
+ − if(rr_data->cs_data.initial_plmn_search EQ INITIAL_PLMN_SEARCH_ACTIVE)
+ − {
+ − cs_update_black_list();
+ − rr_data->cs_data.initial_plmn_search = INITIAL_PLMN_SEARCH_DONE;
+ − }
+ −
+ − /* Full service reached. Perform all necessary tasks */
+ − if(rr_data->ms_data.rr_service EQ FULL_SERVICE)
+ − {
+ − att_full_service_found();
+ − }
+ −
+ − /*
+ − * Indicate the result of cell selection to MM
+ − */
+ − if (tried_to_reach_full_try_limited)
+ − {
+ − /*
+ − * The result is not what MM has requested.
+ − * E.g. MM has requested full service, but RR has
+ − * reached only limited service. In this case
+ − * MM gets the list of the available PLMNs and may
+ − * select on of the networks for a new attempt in
+ − * case of automatic registration.
+ − */
+ − att_copy_old_lai_rac(CR_INDEX);
+ − #if 0
+ − memcpy (&rr_data->old_lai, &rr_data->nc_data[CR_INDEX].lai,
+ − sizeof(T_loc_area_ident));
+ − rr_data->old_cell_id = rr_data->nc_data[CR_INDEX].cell_id;
+ − #endif
+ −
+ − #ifdef GPRS
+ − if(att_gprs_is_avail())
+ − { /* we have GPRS and have reached limited service */
+ − cs_set_stop_active();
+ − if(att_gprs_cell_has_pbcch())
+ − {
+ − att_gprs_stop_pl();
+ − }
+ − else
+ − {
+ − att_set_pl_in_idle_mode ();
+ − }
+ − att_signal_gprs_support();
+ − /* and wait for CR_REQ(CR_COMPLETE) from GRR */
+ − }
+ − else
+ − {
+ − #endif
+ − /*
+ − * Configure layer 1
+ − */
+ − att_set_pl_in_idle_mode ();
+ −
+ − /*
+ − * Indicated the data transfer process that the MS is attached.
+ − */
+ − dat_att_cell_selected ();
+ −
+ − SET_STATE (STATE_ATT, ATT_IDLE);
+ − cs_set_stop_active();
+ −
+ − att_code_rr_abort_ind (RRCS_ABORT_CEL_SEL_FAIL);
+ − att_start_registration_timer ();
+ − att_mph_identity_req ();
+ − srv_use_stored_prim ();
+ − #ifdef GPRS
+ − }
+ − #endif
+ −
+ − }
+ − else
+ − {
+ − /*
+ − * the cell selection was successful. The counter
+ − * for failed registration attempts is resetted and
+ − * MM is informed.
+ − */
+ −
+ − rr_data->ms_data.reg_counter = 0;
+ − #ifdef GPRS
+ − if(! att_gprs_is_avail())
+ − {
+ − #endif
+ −
+ − /*
+ − * Configure layer 1
+ − */
+ − att_set_pl_in_idle_mode ();
+ − /*
+ − * Indicated the data transfer process that the MS is attached.
+ − */
+ − dat_att_cell_selected ();
+ − SET_STATE (STATE_ATT, ATT_IDLE);
+ − cs_set_stop_active();
+ −
+ − /* inform GRR if we have GPRS support */
+ − #ifdef GPRS
+ − if(att_gprs_is_avail())
+ − {
+ − att_signal_gprs_support();
+ − }
+ − #endif
+ − /* Inform MM */
+ − if (rr_data->sc_data.mm_started)
+ − att_code_rr_act_cnf ();
+ − else
+ − att_code_rr_act_ind ();
+ −
+ − srv_use_stored_prim ();
+ − att_start_registration_timer ();
+ − att_mph_identity_req ();
+ −
+ − #ifdef GPRS
+ − }
+ − else
+ − { /* we have GPRS and have reached full service */
+ − cs_set_stop_active();
+ − if(att_gprs_cell_has_pbcch())
+ − {
+ − att_gprs_stop_pl();
+ − }
+ − else
+ − {
+ − att_set_pl_in_idle_mode ();
+ − }
+ − att_signal_gprs_support();
+ − /* and wait for CR_REQ(CR_COMPLETE) from GRR */
+ − }
+ − #endif
+ − /* This flag will help to identify whether cell reselection following
+ − * TRESELECT expiry waiting for SI2TER is successfull or not
+ − */
+ − rr_data->cr_treselect_exp = TRUE;
+ − }
+ − return;
+ − }
+ −
+ − /*
+ − * BCCH carrier check has failed
+ − * Either cell_ok or plmn_ok is FALSE or both.
+ − */
+ −
+ − if(plmn_ok EQ FALSE)
+ − {
+ − /*
+ − * PLMN is not okay, but the cell
+ − */
+ − if (rr_data->ms_data.imsi_available AND
+ − (dat_test_sim_available () EQ FALSE))
+ − {
+ − /*
+ − * with SIM card, that means full service.
+ − * "delete neighbourcells", that means give all
+ − * neighbourcells a low priority in scanning.
+ − *
+ − * It is assumed that this cell will fail also
+ − * because they are also member of the wrong network.
+ − *
+ − * This is not done during FTA campaign.
+ − */
+ − cs_del_list (&rr_data->cr_data.cd.ncell_list);
+ − }
+ − }
+ −
+ − if(plmn_ok AND
+ − cell_ok EQ FALSE)
+ − {
+ − /*
+ − * plmn okay but not the cell
+ − */
+ − if (rr_data->ms_data.imsi_available AND
+ − dat_test_sim_available() EQ FALSE)
+ − {
+ − /*
+ − * with SIM card, that means full service.
+ − * "set neighbourcells", that means give all
+ − * neighbourcells a higher priority in scanning.
+ − * Not done during FTA campaign
+ − */
+ − cs_set_list (&rr_data->cr_data.cd.ncell_list);
+ − }
+ − }
+ −
+ − /*
+ − * start synchronisation attempt to the next cell
+ − */
+ − memset (&rr_data->nc_data[CR_INDEX], 0, sizeof (T_NC_DATA));
+ −
+ − /*
+ − * After BCCH Reading if the requested service is not found,
+ − * before sending BSIC_REQ RR has to be moved to ATT_CS1.
+ − */
+ −
+ − if(rr_data->ms_data.rr_service EQ NO_SERVICE OR
+ − GET_STATE(STATE_ATT) EQ ATT_CS2)
+ − {
+ − SET_STATE (STATE_ATT, ATT_CS1);
+ − }
+ −
+ − srv_clear_list (&rr_data->cr_data.cd.ncell_list);
+ − #ifdef GPRS
+ − if(GET_STATE(STATE_ATT) NEQ ATT_IDLE)
+ − gprs_init_data_cr();
+ − #endif
+ − cs_sync_next ();
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_fplmn_cell |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : If the mobile is in limited service, but MM has requested
+ − full service, RR starts after timeout of the registration
+ − timer new attempts to reach full service. After reading
+ − system information message 3 or 4 of the candidate, this
+ − function is called, to check whether the cell fullfills
+ − the requirements for a full service cell. In this case
+ − a cell reselection is started to read the whole BCCH to
+ − configure layer 1.
+ −
+ − */
+ −
+ − GLOBAL void att_check_fplmn_cell (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE cell_ok;
+ − UBYTE plmn_ok;
+ − T_loc_area_ident * lai = &rr_data->nc_data[CR_INDEX].lai;
+ −
+ − TRACE_FUNCTION ("att_check_fplmn_cell()");
+ −
+ − /*
+ − * Calculate the path loss criterion
+ − */
+ − att_calculate_c1 (CR_INDEX);
+ − TRACE_C1(CR_INDEX);
+ −
+ − /*
+ − * check that the cell is not barred and has a positive path loss
+ − * criterion.
+ − */
+ − /* Implements RR Clone findings #8 */
+ − cell_ok = (!att_cell_barred_status_cr_no_cr (CR_INDEX) AND
+ − (rr_data->nc_data[CR_INDEX].c1 > 0)AND dat_roam_forb_lai_check (CR_INDEX));
+ − ;
+ −
+ − /*
+ − * check whether the PLMN identification is equal to the PLMN which has
+ − * been requested by MM.
+ − */
+ − plmn_ok = dat_plmn_equal_req (lai->mcc,
+ − lai->mnc,
+ − rr_data->ms_data.plmn.mcc,
+ − rr_data->ms_data.plmn.mnc);
+ −
+ − /*
+ − * stop of T_RESELECT. This timer controls the reception of the initial
+ − * system information message.
+ − */
+ − TIMERSTOP (T_RESELECT);
+ −
+ − /*
+ − * store PLMN in plmn found list to provide this information to MM
+ − * at the end of a failed registration attempt.
+ − */
+ − if(dat_forb_lai_check (CR_INDEX))
+ − {
+ − att_store_plmn_in_found_list (lai);
+ − }
+ − else
+ − {
+ − TRACE_EVENT ("Do not store Forbidden LAI PLMN in the found PLMN list");
+ − }
+ −
+ − if (plmn_ok)
+ − {
+ − if (cell_ok)
+ − {
+ − /*
+ − * Enable nc monitoring to resume in lower layer
+ − * when recovering full service
+ − */
+ − att_notify_stop_plmn_search (FALSE);
+ −
+ − /*
+ − * cell and PLMN are okay, then start cell reselection to read
+ − * the complete BCCH to configure layer 1.
+ − */
+ − /*XY:n inform GRR, and wait for CR_RSP */
+ − #ifdef GPRS
+ − att_start_cell_reselection_gprs (CELL_RESELECTION_CR);
+ − #else
+ − att_start_cell_reselection (CELL_RESELECTION_CR);
+ − #endif
+ − return;
+ − }
+ − }
+ −
+ − if (cell_ok)
+ − {
+ − /*
+ − * PLMN is not okay, but the cell
+ − *
+ − * "delete neighbourcells", that means give all
+ − * neighbourcells a low priority in scanning.
+ − */
+ − if (dat_test_sim_available () EQ FALSE)
+ − cs_del_list (&rr_data->cr_data.cd.ncell_list);
+ − }
+ − else
+ − {
+ − /*
+ − * PLMN okay but not the cell
+ − *
+ − * "set neighbourcells", that means give all
+ − * neighbourcells a higher priority in scanning.
+ − */
+ − if (dat_test_sim_available () EQ FALSE)
+ − {
+ − if (plmn_ok)
+ − /*
+ − * with SIM card, that means full service.
+ − * "set neighbourcells", that means give all
+ − * neighbourcells a higher priority in scanning.
+ − */
+ − cs_set_list (&rr_data->cr_data.cd.ncell_list);
+ − else
+ − /*
+ − * with SIM card, that means full service.
+ − * "delete neighbourcells", that means give all
+ − * neighbourcells a low priority in scanning.
+ − *
+ − * It is assumed that this cell will fail also
+ − * because they are also member of the wrong network.
+ − *
+ − * This is not done during FTA campaign.
+ − */
+ − cs_del_list (&rr_data->cr_data.cd.ncell_list);
+ − }
+ − }
+ −
+ − /*
+ − * start synchronisation attempt to the next cell
+ − */
+ − memset (&rr_data->nc_data[CR_INDEX], 0, sizeof (T_NC_DATA));
+ − cs_sync_next ();
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_cell |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : if a cell reselection decision has been taken, this function
+ − looks for a candidate for cell reselection taking in account
+ − several requirements. The reselect_index stores the index
+ − of the RR storage area for the neighbourcell. The return
+ − value indicates whether a cell has been found or not.
+ −
+ − */
+ −
+ − static BOOL att_check_cell (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − BOOL cell_ok = FALSE;
+ −
+ − TRACE_FUNCTION ("att_check_cell()");
+ −
+ − /*
+ − * while no suitable candidate has been found,
+ − * but still possible cells available.
+ − */
+ − while (rr_data->reselect_index NEQ NO_AVAILABLE AND ! cell_ok)
+ − {
+ − /*
+ − * calculate the cell reselection criterion C2 for the candidate
+ − */
+ − att_calculate_c2 (rr_data->reselect_index);
+ −
+ − /*
+ − * check the cell requirements:
+ − * 1. check the cell barred status (if shall be non-barred)
+ − * 2. the cell shall be not temporarily excluded (tnnn e.g. after random access
+ − * failure).
+ − * 3. the cell must have a positive cell reselection criterion C2.
+ − */
+ −
+ − /* Implements RR Clone findings #8 */
+ − cell_ok = (! att_cell_barred_status_cr_no_cr (rr_data->reselect_index) AND
+ − ! is_tnnn (rr_data->reselect_index) AND
+ − rr_data->nc_data[rr_data->reselect_index].c2 > 0);
+ −
+ − /*
+ − * Forbidden LA for roaming shall be tested there in case a C2-base cell
+ − * reselection has been triggered on a cell candidate belonging to a
+ − * valid LA but unsuccesfull. Then RR will continue cell reselection
+ − * with the next candidate that might be part of a forbidden LAC.
+ − * Should be trapped by att_check_neighbourcell() at the end of the
+ − * CS3 session... but it skips one CR and secures the process.
+ − * Question : should this test be bypassed if MM requested limited
+ − * service ??
+ − */
+ −
+ − if (rr_data->ms_data.rr_service EQ FULL_SERVICE)
+ − {
+ − if (cell_ok)
+ − cell_ok = dat_roam_forb_lai_check (rr_data->reselect_index);
+ − }
+ −
+ − if (cell_ok)
+ − {
+ − /*
+ − * suitable cell has been found
+ − */
+ − TRACE_EVENT ("use neighbour cell");
+ −
+ − /*
+ − * after random access failure we found another cell,
+ − * that means, we will process a normal cell reselection
+ − * and will not come back directly to idle mode if the
+ − * cell reselection fails for the new cell.
+ − */
+ − if (rr_data->sc_data.selection_type EQ CELL_RESELECTION_RACH)
+ − {
+ − rr_data->sc_data.selection_type = CELL_RESELECTION;
+ − #ifdef GPRS
+ − rr_data->gprs_data.use_c31 = FALSE;
+ − #endif
+ − }
+ −
+ − /*
+ − * configure layer 1 for cell reselection
+ − */
+ − TRACE_EVENT_P2 ("config L1 for CR [%d]->[%d]",
+ − rr_data->nc_data[SC_INDEX].arfcn,
+ − rr_data->nc_data[rr_data->reselect_index].arfcn);
+ −
+ − #ifdef GPRS
+ − rr_data->gprs_data.ready_state = FALSE; /* Standby */
+ − #endif
+ −
+ − att_start_cr_in_pl(rr_data->reselect_index);
+ − }
+ − else
+ − {
+ − /*
+ − * cell is not suitable. Look for another cell.
+ − */
+ − rr_data->reselect_index =
+ − att_get_next_highest_c2_idx (rr_data->reselect_index);
+ − }
+ − }
+ −
+ − /*
+ − * return whether a cell has been found or not.
+ − */
+ − return cell_ok;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_clean_buf |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : trigger cleaning of the SI buffers indicated by si_to_clean
+ −
+ − */
+ − void att_clean_buf (USHORT si_to_clean)
+ − {
+ − PALLOC(clean_buf, MPH_CLEAN_BUF_REQ);
+ − clean_buf->si_to_clean = si_to_clean;
+ − PSENDX(PL, clean_buf);
+ − }
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_start_cr_in_pl |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : trigger CR in PL and init some necessary parameters
+ −
+ − */
+ − void att_start_cr_in_pl(UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − att_init_pl_status ();
+ − att_build_idle_req (index, MODE_CELL_RESELECTION);
+ −
+ − /*
+ − * Clean the CR_INDEX storage area in RR to get the complete
+ − * BCCH of the new cell
+ − */
+ − att_init_cr_data();
+ − memcpy (&rr_data->nc_data[CR_INDEX],
+ − &rr_data->nc_data[index],
+ − sizeof (T_NC_DATA));
+ − srv_clear_list (&rr_data->cr_data.cd.ncell_list);
+ − SET_STATE (STATE_ATT, ATT_CS3);
+ − rr_data->bcch_error = 0;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_stop_dedicated |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Send primitive MPH_STOP_DEDICATED_REQ
+ −
+ − */
+ − GLOBAL void att_stop_dedicated(void)
+ − {
+ − PALLOC(mph_stop_dedi, MPH_STOP_DEDICATED_REQ);
+ − PSENDX(PL, mph_stop_dedi);
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_leave_dedicated |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Decide if we do a CR or return to idle directly
+ −
+ − */
+ − GLOBAL void att_leave_dedicated(void)
+ − {
+ − GET_INSTANCE_DATA;
+ − if (IS_TIMER_ACTIVE(T_DEDICATED_MODE))
+ − { /* spent less than 30 seconds in dedicated mode */
+ − TIMERSTOP(T_DEDICATED_MODE);
+ − }
+ − else
+ − { /* spent at least 30 seconds in dedicated mode */
+ − rr_data->mode_after_dedi = MODE_CELL_RESELECTION;
+ − }
+ −
+ − if((rr_data->mode_after_dedi EQ MODE_CELL_SELECTION) AND
+ − (rr_data->sc_data.selection_type NEQ BACK_FROM_DEDICATED_RLF)) /*does not seem correct!!!*/
+ − {
+ − /*
+ − * go back directly to the old cell after dedicated
+ − * mode if we were in dedicated mode shorter than 30 seconds
+ − * and have not made a handover in dedicated mode
+ − */
+ − #ifdef GPRS
+ − if(GPRS_SUSPENDED_BCCH EQ GET_STATE(STATE_GPRS))
+ − {
+ − SET_STATE(STATE_GPRS,GPRS_PIM_BCCH);
+ − }
+ − else if(GPRS_SUSPENDED_PBCCH EQ GET_STATE(STATE_GPRS))
+ − {
+ − SET_STATE(STATE_GPRS,GPRS_PIM_PBCCH);
+ − }
+ − #endif
+ − att_return_to_idle();
+ −
+ − /*
+ − * restore the neighbour cell list which was valid
+ − * in idle mode (derived from SI 2x)
+ − * Note: SI 5x may contain other neighbour cells than SI 2x
+ − */
+ − memcpy(&rr_data->sc_data.cd.ncell_list,
+ − &rr_data->sc_data.ncell_list_idle,
+ − sizeof(T_LIST));
+ −
+ − /*
+ − * restore the sys_info_read bitmap if we are going
+ − * directly to idle
+ − */
+ − rr_data->sc_data.cd.sys_info_read = ALL_SYS_INFO_READ;
+ −
+ − if (rr_data->repeat_est)
+ − {
+ − rr_data->repeat_est = FALSE;
+ − dat_start_immediate_assign (rr_data->ms_data.establish_cause);
+ − }
+ − else
+ − {
+ − #ifdef GPRS
+ − if(!att_gprs_cell_has_pbcch())
+ − #endif
+ − att_code_mph_ncell_req(SC_INDEX);
+ − }
+ −
+ − /* After Dedicated mode lass than 30 seconds for a Location Area Update or
+ − * routing Area update.
+ − * start black list search to look for inactive carriers
+ − * Cell-Selection Improvement LLD - 4.1.4.3
+ − */
+ − if((rr_data->ms_data.establish_cause EQ ESTCS_SERV_REQ_BY_MM) AND
+ − (rr_data->cs_data.black_list_search_pending EQ TRUE))
+ − {
+ −
+ − TRACE_EVENT("black_list_search_pending : 1");
+ −
+ − rr_data->cs_data.black_list_search_pending = FALSE;
+ −
+ − if(cs_check_region(rr_data->cs_data.region))
+ − {
+ − /* Do not start Black list search,when Black list database is empty */
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − if (rr_data->cs_data.region EQ BOTH_REGIONS)
+ − {
+ − if(srv_is_list_set(&rr_data->cs_data.black_list.list[EUROPEAN_REGION]) OR
+ − srv_is_list_set(&rr_data->cs_data.black_list.list[AMERICAN_REGION]))
+ − {
+ − att_start_cell_selection(RR_ORIGINATED,CS_PARALLEL,BLACK_LIST_SEARCH_MODE);
+ − }
+ − else
+ − {
+ − TRACE_EVENT("Black List empty");
+ − }
+ − }
+ − else
+ − {
+ − #endif
+ − if(srv_is_list_set(
+ − &rr_data->cs_data.black_list.list[rr_data->cs_data.region]))
+ − {
+ − att_start_cell_selection(RR_ORIGINATED,CS_PARALLEL,BLACK_LIST_SEARCH_MODE);
+ − }
+ − else
+ − {
+ − TRACE_EVENT("Black List empty");
+ − }
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − }
+ − #endif
+ − }
+ − }
+ − }
+ − else
+ − {
+ − #ifdef GPRS
+ − att_start_cell_reselection_gprs(BACK_FROM_DEDICATED);
+ − #else
+ − att_start_cell_reselection(BACK_FROM_DEDICATED);
+ − #endif
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_leave_dat_imm_ass |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : return to idle mode directly and clean up
+ − immediate assignment procedure
+ −
+ − */
+ − GLOBAL void att_leave_dat_imm_ass(void)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_leave_dat_imm_ass()");
+ −
+ − #ifdef GPRS
+ − if(GPRS_SUSPENDED_BCCH EQ GET_STATE(STATE_GPRS))
+ − {
+ − SET_STATE(STATE_GPRS,GPRS_PIM_BCCH);
+ − }
+ − else if(GPRS_SUSPENDED_PBCCH EQ GET_STATE(STATE_GPRS))
+ − {
+ − SET_STATE(STATE_GPRS,GPRS_PIM_PBCCH);
+ − }
+ − #endif
+ −
+ − att_return_to_idle();
+ −
+ − rr_data->repeat_est = FALSE;
+ −
+ − #ifdef GPRS
+ − if(!att_gprs_cell_has_pbcch())
+ − #endif
+ − att_code_mph_ncell_req(SC_INDEX);
+ −
+ − /*
+ − * start black list search to look for inactive carriers
+ − * Cell-Selection Improvement LLD - 4.1.4.3
+ − */
+ − if((rr_data->ms_data.establish_cause EQ ESTCS_SERV_REQ_BY_MM) AND
+ − (rr_data->cs_data.black_list_search_pending EQ TRUE))
+ − {
+ −
+ − TRACE_EVENT("black_list_search_pending : 1");
+ −
+ − rr_data->cs_data.black_list_search_pending = FALSE;
+ −
+ − if(cs_check_region(rr_data->cs_data.region))
+ − {
+ − /* Do not start Black list search,when Black list database is empty */
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − if (rr_data->cs_data.region EQ BOTH_REGIONS)
+ − {
+ − if(srv_is_list_set(&rr_data->cs_data.black_list.list[EUROPEAN_REGION]) OR
+ − srv_is_list_set(&rr_data->cs_data.black_list.list[AMERICAN_REGION]))
+ − {
+ − att_start_cell_selection(RR_ORIGINATED,CS_PARALLEL,BLACK_LIST_SEARCH_MODE);
+ − }
+ − else
+ − {
+ − TRACE_EVENT("Black List empty");
+ − }
+ − }
+ − else
+ − {
+ − #endif
+ − if(srv_is_list_set(&rr_data->cs_data.black_list.list[rr_data->cs_data.region]))
+ − {
+ − att_start_cell_selection(RR_ORIGINATED,CS_PARALLEL,BLACK_LIST_SEARCH_MODE);
+ − }
+ − else
+ − {
+ − TRACE_EVENT("Black List empty");
+ − }
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − }
+ − #endif
+ − }
+ − }
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_select_cell_dedicated |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function selects a cell for cell reselection after
+ − coming back from a dedicated connected (normal release
+ − or radio link failure.
+ −
+ − */
+ −
+ − static void att_select_cell_dedicated (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − U8 search_mode;
+ −
+ − TRACE_FUNCTION ("att_select_cell_dedicated()");
+ −
+ − if (rr_data->reselect_index NEQ NO_AVAILABLE)
+ − {
+ − TRACE_EVENT_P3 ("[%u] new CS/CR (mode=%u, sel.type=%d)",
+ − rr_data->nc_data[rr_data->reselect_index].arfcn,
+ − rr_data->mode_after_dedi, rr_data->sc_data.selection_type);
+ −
+ − /*
+ − * A cell has been found for coming back from dedicated mode.
+ − * Configure layer 1 for cell reselection.
+ − */
+ − att_start_cr_in_pl(rr_data->reselect_index);
+ − }
+ − else
+ − {
+ − /*
+ − * no suitable cell is available
+ − */
+ − if (rr_data->sc_data.selection_type EQ BACK_FROM_DEDICATED_RLF)
+ − {
+ − /*
+ − * In case of coming back after radio link failure
+ − * the old cell shall be used only if no other cell
+ − * is suitable. So the selection type is changed to
+ − * "normal" comeback from dedicated mode. MM is informed
+ − * that no reestablishment is possible and a cell is
+ − * searched again inclusive the old serving cell.
+ − */
+ − rr_data->sc_data.selection_type = BACK_FROM_DEDICATED;
+ − dat_code_reestablishment_fail ();
+ − TIMERSTART (T_RESELECT, TRESELECT_VALUE);
+ − att_search_cell ();
+ − }
+ − else
+ − {
+ − /* Cell Selection Improvements-LLD section:4.1.3.9 */
+ − if((rr_data->net_lost) AND (rr_data->ms_data.req_mm_service EQ FUNC_PLMN_SRCH))
+ − {
+ − /* Radio Link failure in dedicated state and cell reselection failed.
+ − * Start Fast search
+ − */
+ − search_mode = FAST_SEARCH_MODE;
+ − }
+ − else
+ − {
+ − /* Normal return from dedicated mode, cell reselection failed
+ − * Start Normal search
+ − */
+ − search_mode = NORMAL_SEARCH_MODE;
+ − }
+ −
+ − /*
+ − * there is no candidate for coming back. So a cell
+ − * selection is started from RR.
+ − */
+ − /* XY:n inform GRR, and wait for CR_RSP */
+ − #ifdef GPRS
+ − att_start_cell_selection_gprs (RR_ORIGINATED,search_mode);
+ − #else
+ − att_start_cell_selection (RR_ORIGINATED, CS_NOT_PARALLEL,search_mode);
+ − #endif
+ − }
+ − }
+ − //rr_data->mode_after_dedi = MODE_CELL_SELECTION; /* reset mode */
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_neighbourcell |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function checks a cell after cell reselection.
+ −
+ − */
+ −
+ − GLOBAL void att_check_neighbourcell (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − BOOL cell_ok;
+ − T_loc_area_ident * lai = &rr_data->nc_data[CR_INDEX].lai;
+ −
+ − TRACE_FUNCTION ("att_check_neighbourcell()");
+ −
+ − #ifdef GPRS
+ − if(att_gprs_check_ncell())
+ − return;
+ − #endif
+ − /*
+ − * Calculate the cell reselection criterion C2 or C31/C32.
+ − */
+ − #ifdef GPRS
+ − if( rr_data->gprs_data.use_c31)
+ − {
+ − att_calculate_c31_c32(CR_INDEX);
+ −
+ − /* Implements RR Clone findings #8 */
+ − cell_ok = (!att_cell_barred_status_cr_no_cr (CR_INDEX) AND
+ − att_check_c31_criterion(CR_INDEX) );
+ − }
+ − else
+ − {
+ − #endif
+ − att_calculate_c2 (CR_INDEX);
+ − /*
+ − * Check the cell. It shall not be barred and must have
+ − * a positive cell reselection criterion C2.
+ − */
+ −
+ − /* Implements RR Clone findings #8 */
+ − cell_ok = ! att_cell_barred_status_cr_no_cr (CR_INDEX) AND
+ − rr_data->nc_data[CR_INDEX].c2 > 0;
+ −
+ −
+ − #ifdef GPRS
+ − }
+ − #endif
+ −
+ − /*
+ − * the cell can be rejected by the SW shielding function.
+ − * This has only an affect during type approval.
+ − */
+ − if (cell_ok)
+ − cell_ok = att_check_network (lai);
+ −
+ − /*
+ − * Check against the list of forbidden location areas for roaming,
+ − * but only if MM requested full service and we are still searching
+ − * for full service. Do not prevent limited service reselection.
+ − */
+ − if (cell_ok AND
+ − (rr_data->cs_data.scan_mode NEQ CS_SECOND_SCAN) AND
+ − (rr_data->ms_data.req_mm_service EQ FUNC_PLMN_SRCH))
+ − {
+ − cell_ok = dat_roam_forb_lai_check (CR_INDEX);
+ − }
+ −
+ − /*
+ − * Additional tests are performed depending on the cell
+ − * reselection type.
+ − */
+ − TRACE_SELECTION_TYPE (rr_data->sc_data.selection_type);
+ − switch (rr_data->sc_data.selection_type)
+ − {
+ − case CELL_RESELECTION_CR:
+ − /*
+ − * The MS was in idle mode and the check is
+ − * 1. from limited service to full service or
+ − * 2. from full service (VPLMN) to full service (HPLMN)
+ − *
+ − * The resulting RR service is set only if the cell is okay.
+ − */
+ − if (cell_ok)
+ − {
+ − if (rr_data->ms_data.req_mm_service EQ FUNC_LIM_SERV_ST_SRCH OR
+ − rr_data->cs_data.scan_mode EQ CS_SECOND_SCAN)
+ − {
+ − rr_data->ms_data.rr_service = LIMITED_SERVICE;
+ − }
+ − else
+ − {
+ − cell_ok = dat_plmn_equal_req(lai->mcc, lai->mnc,
+ − rr_data->ms_data.plmn.mcc,
+ − rr_data->ms_data.plmn.mnc);
+ − if (cell_ok)
+ − rr_data->ms_data.rr_service = FULL_SERVICE;
+ − else
+ − rr_data->ms_data.rr_service = LIMITED_SERVICE;
+ − }
+ − }
+ − break;
+ −
+ − case CELL_SELECTION:
+ − case CELL_RESELECTION:
+ − case CELL_RESELECTION_NC:
+ − case CELL_RESELECTION_RACH:
+ − if(cell_ok)
+ − {
+ − /*
+ − * 2 checks to be made
+ − *
+ − * - In case of FULL SERVICE additionally the PLMN is checked
+ − *
+ − * - We are in LIMITED service.
+ − * If the req_mm_service is FUNC_LIM_SERV_ST_SRCH we
+ − * will stay in LIMITED.
+ − * If we are in limited because of forbidden LAI we enter FULL SERVICE.
+ − */
+ − if(rr_data->ms_data.rr_service EQ FULL_SERVICE)
+ − {
+ − cell_ok = dat_plmn_equal_req (lai->mcc,
+ − lai->mnc,
+ − rr_data->ms_data.plmn.mcc,
+ − rr_data->ms_data.plmn.mnc);
+ − }
+ − else if((rr_data->ms_data.rr_service EQ LIMITED_SERVICE) AND
+ − (rr_data->ms_data.req_mm_service NEQ FUNC_LIM_SERV_ST_SRCH) AND
+ − dat_plmn_equal_req (lai->mcc,
+ − lai->mnc,
+ − rr_data->ms_data.plmn.mcc,
+ − rr_data->ms_data.plmn.mnc) )
+ − {
+ − rr_data->ms_data.rr_service = FULL_SERVICE;
+ − }
+ −
+ − }
+ − break;
+ −
+ − case BACK_FROM_DEDICATED:
+ − case BACK_FROM_DEDICATED_RLF:
+ − /* The mobile is trying to select a cell after coming back from dedicated
+ − * state. The cell is selected only if it belongs to the previously selected
+ − * PLMN
+ − */
+ − if(cell_ok)
+ − {
+ − /* While selecting a cell after an emergency call, we need not check the
+ − * PLMN ID of the cell */
+ − if(rr_data->ms_data.rr_service EQ FULL_SERVICE)
+ − {
+ − cell_ok = dat_plmn_equal_req (lai->mcc,
+ − lai->mnc,
+ − rr_data->ms_data.plmn.mcc,
+ − rr_data->ms_data.plmn.mnc);
+ − }
+ −
+ − TRACE_EVENT_P6 ("B_F_D New PLMN : MCC=%X%X%X MNC=%X%X%X",
+ − lai->mcc[0],
+ − lai->mcc[1],
+ − lai->mcc[2],
+ − lai->mnc[0],
+ − lai->mnc[1],
+ − lai->mnc[2]);
+ − TRACE_EVENT_P6 ("B_F_D Old PLMN : MCC=%X%X%X MNC=%X%X%X",
+ − rr_data->ms_data.plmn.mcc[0],
+ − rr_data->ms_data.plmn.mcc[1],
+ − rr_data->ms_data.plmn.mcc[2],
+ − rr_data->ms_data.plmn.mnc[0],
+ − rr_data->ms_data.plmn.mnc[1],
+ − rr_data->ms_data.plmn.mnc[2]);
+ − }
+ − if(rr_data->sc_data.selection_type EQ BACK_FROM_DEDICATED)
+ − break;
+ −
+ − /*
+ − * in case of call reestablishment check also call reestablishment flag
+ − */
+ − if (cell_ok)
+ − {
+ − if (rr_data->nc_data[CR_INDEX].rach.re NEQ REESTAB_YES)
+ − cell_ok = FALSE;
+ − }
+ − break;
+ − }
+ −
+ − /*
+ − * the requirement check is finished
+ − */
+ − if (cell_ok)
+ − {
+ − TRACE_EVENT_P2 ("NEW SC [%d]->[%d]",
+ − rr_data->nc_data[SC_INDEX].arfcn,
+ − rr_data->nc_data[CR_INDEX].arfcn);
+ −
+ − /*
+ − * Stop T_RESELECT, which controls the reception of the complete BCCH
+ − * during Cell Reselection.
+ − */
+ − TIMERSTOP (T_RESELECT);
+ −
+ −
+ − /* T3122 controls after how long the MS is able to attempt access back to the */
+ − /* cell from which it received a IMM ASSGN REJ. This no longer applies after a */
+ − /* cell reselection onto a new cell. */
+ −
+ − TIMERSTOP(T3122);
+ −
+ −
+ − /*
+ − * copy all data from the SC_INDEX storage area to the neighbourcell area.
+ − * copy all data from the CR_INDEX storage area to the SC_INDEX area.
+ − * old serving cell data index is stored in old_serving_cell.
+ − */
+ − att_copy_cr_data ();
+ −
+ − memcpy (&rr_data->sc_data.cd, &rr_data->cr_data.cd,
+ − sizeof (T_CELL_DATA));
+ −
+ − if(rr_data->ms_data.rr_service EQ FULL_SERVICE)
+ − {
+ − att_full_service_found();
+ − }
+ −
+ − rr_data->nc_data[CR_INDEX].arfcn = NOT_PRESENT_16BIT;
+ − /*
+ − * GSM 5.08, chapter 6.4:
+ − *
+ − * T is a timer implemented for each cell in the list of strongest
+ − * carriers.
+ − * T shall be started from zero at the time the cell is placed by the MS on
+ − * the list of strongest carriers, except when the previous serving cell is
+ − * placed on the list of strongest carriers at cell reselection. In this
+ − * case, T shall be set to the value of PENALTY_TIME (i.e. expired).
+ − */
+ − switch (rr_data->sc_data.selection_type)
+ − {
+ − case CELL_RESELECTION:
+ − case CELL_RESELECTION_NC:
+ − case CELL_RESELECTION_RACH:
+ − /*
+ − * set the avail time of the old serving cell to PENALTY_TIME.
+ − */
+ − if (rr_data->old_serving_cell < SC_INDEX)
+ − rr_data->nc_data[rr_data->old_serving_cell].avail_time =
+ − rr_data->nc_data[rr_data->old_serving_cell].c2_par.penalty_time;
+ − break;
+ −
+ − default:
+ − break;
+ − }
+ −
+ −
+ − /*
+ − * set a barrier of 15 seconds for the next cell reselection
+ − * if it was a cell reselection due to C2(NC) > C2(SC)
+ − */
+ − if (rr_data->sc_data.selection_type EQ CELL_RESELECTION_NC)
+ − TIMERSTART (T_NO_RESELECT, THIRTY_SEC/2);
+ −
+ − #ifdef GPRS
+ − if(! att_gprs_is_avail())
+ − {
+ − #endif
+ −
+ − /*
+ − * configure layer 1
+ − */
+ − att_remove_bad_rr_data_ncells();
+ − att_set_pl_in_idle_mode ();
+ −
+ − /*
+ − * inform data transfer process that the cell is attached.
+ − */
+ − dat_att_cell_selected ();
+ −
+ − /*
+ − * inform MM about new cell
+ − */
+ − if (rr_data->sc_data.mm_started)
+ − att_code_rr_act_cnf ();
+ − else
+ − att_code_rr_act_ind ();
+ −
+ − SET_STATE (STATE_ATT, ATT_IDLE);
+ − srv_use_stored_prim ();
+ − /*
+ − * Start registration timer if needed
+ − */
+ − att_start_registration_timer ();
+ − /*
+ − * if the cell reselection has been started after a dedicated connection
+ − * attempt with failed SABM / UA content a second attempt is started
+ − * immediately.
+ − */
+ − if (rr_data->repeat_est)
+ − {
+ − rr_data->repeat_est = FALSE;
+ − dat_start_immediate_assign (rr_data->ms_data.establish_cause);
+ − }
+ − #ifdef GPRS
+ − }
+ − else
+ − {
+ − if(rr_data->ms_data.rr_service NEQ FULL_SERVICE)
+ − {
+ − if(att_gprs_cell_has_pbcch())
+ − {
+ − /* set to PIM_BCCH if the cell hs a PBCCH but we
+ − * only have limited service
+ − */
+ − SET_STATE(STATE_GPRS, GPRS_PIM_BCCH);
+ − }
+ − }
+ −
+ − if(att_gprs_cell_has_pbcch())
+ − {
+ − att_gprs_stop_pl();
+ − }
+ − else
+ − {
+ − att_remove_bad_rr_data_ncells();
+ − att_set_pl_in_idle_mode ();
+ − }
+ − /* inform GRR if we have GPRS support */
+ − att_signal_gprs_support();
+ − /* and wait for CR_REQ(CR_COMPLETE) from GRR */
+ − }
+ − #endif
+ − }
+ − else
+ − {
+ − /*
+ − * the cell reselection failed for this channel and is continued
+ − * with the next candidate.
+ − */
+ − TRACE_EVENT_P1 ("[%u] CR failed", rr_data->nc_data[CR_INDEX].arfcn);
+ − rr_data->pag_rec = FALSE;
+ − srv_clear_stored_prim (MPH_PAGING_IND);
+ − att_continue_cell_reselect ();
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_neighbourcell_si_reestab |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function checks if revelant sys info is present is ATT_CS3 state.
+ − Special case : Sys Info 2Ter is not needed if we have a pending call re-establishment
+ −
+ − */
+ −
+ − GLOBAL void att_check_neighbourcell_si_reestab (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − /*
+ − * Sometimes the network indicates SI2Ter but never sends it.
+ − * If we have a call re-establishment pending, then we must not wait
+ − * for SI2Ter.
+ − */
+ − if( (rr_data->cr_data.cd.sys_info_read EQ ALL_SYS_INFO_READ)
+ − OR
+ − ( (rr_data->cr_data.cd.sys_info_read EQ SYS_INFO_EXCEPT_2TER) AND
+ − (rr_data->sc_data.selection_type EQ BACK_FROM_DEDICATED_RLF))
+ − )
+ − {
+ − att_check_neighbourcell ();
+ − }
+ −
+ − /*
+ − * In case of Radio Link Failure, check cell for call
+ − * re-establishment capability.
+ − */
+ − if (rr_data->sc_data.selection_type EQ BACK_FROM_DEDICATED_RLF)
+ − att_check_reestablishment();
+ −
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_reestablishment |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function checks whether a cell has no call re-establishment
+ − capability. In this case the cell reselection is resumed
+ − with the next cell.
+ −
+ − */
+ −
+ − GLOBAL void att_check_reestablishment (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_check_reestablishment()");
+ −
+ − if (rr_data->nc_data[CR_INDEX].rach.re NEQ REESTAB_YES)
+ − {
+ − /*
+ − * the cell reselection failed for this channel and is continued
+ − * with the next candidate due to no call re-establishment
+ − * capability.
+ − */
+ − TRACE_EVENT_P2 ("[%u]7 (BCCH-St=%u) have no call re-establishment capability",
+ − rr_data->nc_data[CR_INDEX].arfcn, rr_data->nc_data[CR_INDEX].bcch_status);
+ −
+ − rr_data->pag_rec = FALSE;
+ − srv_clear_stored_prim (MPH_PAGING_IND);
+ − att_continue_cell_reselect ();
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_reselect_decision|
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After reception of measurement values from layer 1 in
+ − idle mode the cell reselection decision is done by this
+ − function. It checks the measurement related criterions
+ − for cell reselection. The flag start_now indicates
+ − whether only the decision is taken or the cell reselection
+ − is started immediately. It is senseful to delay a cell
+ − reselection during a PLMN available search triggered by MM.
+ −
+ − */
+ −
+ − GLOBAL void att_check_reselect_decision (UBYTE start_now)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE i;
+ − SHORT delta;
+ −
+ − TRACE_FUNCTION ("att_check_reselect_decision()");
+ − #ifdef GPRS
+ − TRACE_EVENT_P2("check_reselect: start_cell_reselection=%u calc %s",
+ − rr_data->start_cell_reselection, rr_data->gprs_data.use_c31 ? "C31,C32":"C2");
+ − #endif
+ − /*
+ − * Calculate for all synchronized neighbourcells, which have
+ − * decoded the system information messages for calculation C2,
+ − * and for the serving cell the cell reselection criterion C2.
+ − */
+ − for (i = 0; i < 7; i++)
+ − {
+ − if (rr_data->nc_data[i].bcch_status EQ DECODED)
+ − {
+ − #ifdef GPRS
+ − if( rr_data->gprs_data.use_c31)
+ − att_calculate_c31_c32(i);
+ − else
+ − #endif
+ − att_calculate_c2 (i);
+ − }
+ − else if (rr_data->nc_data[i].bcch_status NEQ EMPTY AND
+ − rr_data->nc_data[i].arfcn NEQ NOT_PRESENT_16BIT )
+ − {
+ − TRACE_EVENT_P2("[%u]i%u no C2 (SIs missing)",
+ − rr_data->nc_data[i].arfcn,i);
+ − }
+ − }
+ −
+ − /*
+ − * Check C1 serving cell < 0 for more then 5 seconds
+ − * The c1_counter counts the number of reports where
+ − * this criterion has passed.
+ − */
+ − if (rr_data->nc_data[SC_INDEX].c1 < 0)
+ − {
+ − /*
+ − * if c1_counter has a value of at least 1 it had happened
+ − * for at least two successive reports, that means more
+ − * then five seconds.
+ − */
+ − if ( (rr_data->nc_data[SC_INDEX].c1_counter > 0)
+ − #ifdef GPRS
+ − OR
+ − (rr_data->gprs_data.use_c31)
+ − #endif
+ − )
+ − {
+ − /*
+ − * reset the counter and start !
+ − */
+ − rr_data->nc_data[SC_INDEX].c1_counter = 0;
+ − #ifdef GPRS
+ − /* check if we can do a cell reselection */
+ − if(!is_nc2_used())
+ − {
+ − #endif
+ − if (start_now )
+ − {
+ − /*XY:n inform GRR, and wait for CR_RSP */
+ − #ifdef GPRS
+ − att_start_cell_reselection_gprs (CELL_RESELECTION);
+ − #else
+ − att_start_cell_reselection (CELL_RESELECTION);
+ − #endif
+ − }
+ − #ifdef GPRS
+ − }
+ − #endif
+ − return;
+ − }
+ − else
+ − {
+ − /*
+ − * the criterion has passed the first time, so
+ − * increase the counter and wait for the next measurement value.
+ − */
+ − rr_data->nc_data[SC_INDEX].c1_counter++;
+ − #if defined(_SIMULATION_)
+ − TRACE_EVENT_WIN_P1 ("CR SC delayed (%d)", rr_data->nc_data[SC_INDEX].c1_counter);
+ − #endif
+ − }
+ − }
+ − else
+ − {
+ − /*
+ − * The C1 of the serving cell is positive, so clear the c1_counter.
+ − */
+ − rr_data->nc_data[SC_INDEX].c1_counter = 0;
+ − #if defined(_SIMULATION_)
+ − TRACE_EVENT_WIN_P1 ("CR SC c1_counter=%d", rr_data->nc_data[SC_INDEX].c1_counter);
+ − #endif
+ − }
+ − /*
+ − * check if C31\C32 based reselection is required
+ − */
+ − #ifdef GPRS
+ − if( rr_data->gprs_data.use_c31 )
+ − {
+ − att_check_c31_reselect_decision(start_now);
+ − return;
+ − }
+ − #endif
+ − /*
+ − * Check C2(NC) > C2(SC) for all neighbourcells
+ − */
+ − for (i = 0; i < 6; i++)
+ − {
+ − /*
+ − * If data is stored for this neighbourcell
+ − */
+ − if (rr_data->nc_data[i].bcch_status EQ DECODED)
+ − {
+ − /*
+ − TRACE_EVENT_P2 ("COMPARE NC[%d]-SC[%d]",
+ − rr_data->nc_data[i].arfcn,
+ − rr_data->nc_data[SC_INDEX].arfcn);
+ − */
+ − #ifdef GPRS
+ − /*
+ − * check if the location area has changed
+ − * or the routing area if the scell supports GPRS
+ − */
+ − {
+ − BOOL la_same, ra_same;
+ −
+ − if (dat_plmn_equal_req (rr_data->nc_data[i].lai.mcc,
+ − rr_data->nc_data[i].lai.mnc,
+ − rr_data->nc_data[SC_INDEX].lai.mcc,
+ − rr_data->nc_data[SC_INDEX].lai.mnc)
+ − AND
+ − rr_data->nc_data[i].lai.lac EQ
+ − rr_data->nc_data[SC_INDEX].lai.lac)
+ − la_same = TRUE;
+ − else
+ − la_same = FALSE;
+ − /*
+ − * If gprs is not available do not care about the routing area
+ − *
+ − */
+ − if(att_gprs_is_avail())
+ − {
+ − if(rr_data->nc_data[i].rac EQ NOT_PRESENT_8BIT OR
+ − rr_data->nc_data[i].rac EQ rr_data->nc_data[SC_INDEX].rac)
+ − {
+ − ra_same = TRUE;
+ − }
+ − else
+ − {
+ − ra_same = FALSE;
+ − }
+ − }
+ − else
+ − {
+ − ra_same = TRUE;
+ − }
+ −
+ − if(la_same AND ra_same AND !rr_data->gprs_data.ready_state)
+ − #else /* GPRS */
+ − if (dat_plmn_equal_req (rr_data->nc_data[i].lai.mcc,
+ − rr_data->nc_data[i].lai.mnc,
+ − rr_data->nc_data[SC_INDEX].lai.mcc,
+ − rr_data->nc_data[SC_INDEX].lai.mnc)
+ − AND
+ − rr_data->nc_data[i].lai.lac EQ
+ − rr_data->nc_data[SC_INDEX].lai.lac)
+ − #endif /* GPRS */
+ − {
+ − /*
+ − * If it is in the same location area as the serving cell,
+ − * the delta is simply the difference of both cell reselection
+ − * criterion values C2.
+ − */
+ − TRACE_EVENT_P3 ("NC[%d]i%u-SC[%d] same LAI",
+ − rr_data->nc_data[i].arfcn,i,
+ − rr_data->nc_data[SC_INDEX].arfcn);
+ −
+ − delta = rr_data->nc_data[i].c2 -
+ − rr_data->nc_data[SC_INDEX].c2;
+ −
+ − /*
+ − * 3GPP TS 05.08, section 6.6.2:
+ − * in case of a cell reselection occurring within the previous 15 seconds
+ − * in which case the C2 value for the new cell shall exceed the C2 value
+ − * of the serving cell by at least 5 dB for a period of 5 seconds
+ − */
+ −
+ − if ( IS_TIMER_ACTIVE (T_NO_RESELECT) )
+ − {
+ − delta -= 5;
+ − }
+ − else
+ − {
+ − /* When C2(NC)==C2(SC) i.e. delta=0, an invalid cell reselection gets triggered below.
+ − * 'delta' needs to be adjusted to prevent this happening. The '=' in the '>='
+ − * condition below only applies when T_NO_RESELECT timer is active and
+ − * c2_corr has been decremented by 5 (above statement)
+ − */
+ − if(delta EQ 0 )
+ − {
+ − delta = -1;
+ − }
+ − }
+ − }
+ − else
+ − {
+ − /*
+ − * If both are member of different location areas, a threshold
+ − * value CELL_RESELECTION_HYSTERESE is taken in account.
+ − */
+ − TRACE_EVENT_P3 ("NC[%d]i%u-SC[%d] different LAI",
+ − rr_data->nc_data[i].arfcn,i,
+ − rr_data->nc_data[SC_INDEX].arfcn);
+ −
+ − delta = att_calculate_c2_diff (i);
+ − }
+ − #ifdef GPRS
+ − }
+ − #endif
+ − /*
+ − * Check cell barred status before going
+ − * into Cell Reselection state.
+ − * This will actually also be checked again
+ − * in att_check_cell but the CRH value will
+ − * not be used for subsequent ncells leading
+ − * to false CR's. We check it here
+ − * to use CRH for the next cell if needed.
+ − */
+ − /* Implements RR Clone findings #8 */
+ − if(att_cell_barred_status_cr_no_cr(i))
+ − delta = -1;
+ −
+ − /*
+ − * Do not trigger a cell reselection decision on meas. report
+ − * receipt if the cell belong to a LA not allowed for roaming.
+ − * According to 3.22 chapter 3.5.4 cell reselection on a
+ − * forbidden LA for regional provision of service is allowed.
+ − */
+ − if (rr_data->ms_data.rr_service EQ FULL_SERVICE)
+ − {
+ − if (!dat_roam_forb_lai_check(i))
+ − delta = -1;
+ − }
+ −
+ − #if defined(_SIMULATION_)
+ − TRACE_EVENT_WIN_P1 ("delta=%d", delta);
+ − #endif
+ − if(delta == 0) TRACE_EVENT("No reselection delta == 0");
+ − if (delta > 0)
+ − {
+ − /*
+ − * The condition C2(NC) > C2(SC) is fulfilled.
+ − */
+ − if (rr_data->nc_data[i].c1_counter > 0)
+ − {
+ − /*
+ − * the conditions is fulfilled at least two times ( > 5 seconds)
+ − * and there is no barrier to start the cell
+ − * reselection.
+ − */
+ − #ifdef GPRS
+ − /* check if we can do a cell reselection */
+ − if(!is_nc2_used())
+ − {
+ − #endif
+ − if (start_now)
+ − {
+ − rr_data->nc_data[i].c1_counter = 0;
+ − /*XY:n inform GRR, and wait for CR_RSP */
+ − #ifdef GPRS
+ − att_start_cell_reselection_gprs(CELL_RESELECTION_NC);
+ − #else
+ − att_start_cell_reselection (CELL_RESELECTION_NC);
+ − #endif
+ − }
+ − #ifdef GPRS
+ − }
+ − #endif
+ − return;
+ − }
+ − else
+ − {
+ − /*
+ − * The condition has passed the first time or is delayed
+ − * by the barrier to perform a cell reselection due to this reasons
+ − * not faster then in periods of 15 seconds.
+ − */
+ − rr_data->nc_data[i].c1_counter++;
+ − TRACE_EVENT_P3 ("NC[%u]i%u CR delayed (%d)",
+ − rr_data->nc_data[i].arfcn,i,rr_data->nc_data[i].c1_counter);
+ − }
+ − }
+ − else
+ − {
+ − /*
+ − * the condition is not fullfilled and the counter is resetted.
+ − */
+ − rr_data->nc_data[i].c1_counter = 0;
+ − }
+ − }
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_barred_status_in_idle|
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE :
+ −
+ − */
+ −
+ − GLOBAL void att_check_barred_status_in_idle (void)
+ − {
+ − TRACE_FUNCTION ("att_check_barred_status_in_idle()");
+ −
+ − /*
+ − * check whether serving cell is barred
+ − */
+ − /*XY:n inform GRR, and wait for CR_RSP */
+ −
+ − /* Implements RR Clone findings #8 */
+ − if (att_cell_barred_status_cr_no_cr (SC_INDEX) EQ TRUE)
+ − {
+ −
+ − #ifdef GPRS
+ − att_start_cell_reselection_gprs (CELL_RESELECTION);
+ − #else
+ − att_start_cell_reselection (CELL_RESELECTION);
+ − #endif
+ − }
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_clear_forb_list |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : RR stores some forbidden lists. This function clears
+ − the given forbidden list.
+ −
+ − */
+ −
+ − static void att_clear_forb_list (int list_type)
+ − {
+ − GET_INSTANCE_DATA;
+ − int i;
+ − T_loc_area_ident *forb_list;
+ −
+ − TRACE_FUNCTION ("att_clear_forb_list()");
+ −
+ − if (list_type EQ FORBIDDEN_LIST_NORMAL)
+ − forb_list = &rr_data->ms_data.forb_lac_list[0];
+ − else
+ − forb_list = &rr_data->ms_data.roam_forb_lac_list[0];
+ −
+ − /* Implements Measure#32: Row 29 */
+ − if(list_type EQ FORBIDDEN_LIST_NORMAL)
+ − {
+ − TRACE_EVENT ( "CLEAR FORB LIST");
+ − }
+ − else
+ − {
+ − TRACE_EVENT ( "CLEAR ROAM FORB LIST");
+ − }
+ −
+ −
+ − for (i = 0; i < MAX_LAI; i++)
+ − {
+ − memset (forb_list[i].mcc, NOT_PRESENT_8BIT, SIZE_MCC);
+ − memset (forb_list[i].mnc, NOT_PRESENT_8BIT, SIZE_MNC);
+ − forb_list[i].lac = NOT_PRESENT_16BIT;
+ − }
+ − }
+ −
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_clear_registration_data|
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function is called after reception of RR_SYNC_REQ,
+ − which is sent by MM after some Location Updating Reject
+ − causes which indicate that the IMSI is invalid, paging
+ − shall be suppressed and RR shall enter only a limited mode.
+ −
+ − */
+ −
+ − GLOBAL void att_clear_registration_data (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_clear_registration_data()");
+ −
+ − TRACE_EVENT ("CLEAR registration data (IMSI, MCC, MNC)");
+ −
+ − /*
+ − * Clear the IMSI related data and enter the
+ − * limited mode.
+ − */
+ − rr_data->ms_data.operation_mode = 0;
+ − att_clear_parallel_search ();
+ − rr_data->ms_data.current_plmn_search_type = FUNC_LIM_SERV_ST_SRCH;
+ − rr_data->ms_data.imsi_available = FALSE;
+ − rr_data->ms_data.plmn.mcc[0] = NOT_PRESENT_8BIT;
+ − rr_data->ms_data.plmn.mnc[0] = NOT_PRESENT_8BIT;
+ − rr_data->ms_data.v_eq_plmn = FALSE;
+ −
+ − /*we only have limited service*/
+ − rr_data->ms_data.rr_service = LIMITED_SERVICE;
+ − rr_data->ms_data.req_mm_service = FUNC_LIM_SERV_ST_SRCH;
+ −
+ − TRACE_EVENT_P3 ("att_clear_registration_data: current=%s, service:rMM=%s RRs=%s",
+ − _rr_str_FUNC[rr_data->ms_data.current_plmn_search_type],
+ − _rr_str_FUNC[rr_data->ms_data.req_mm_service],
+ − _rr_str_SERVICE[rr_data->ms_data.rr_service]);
+ −
+ − /*
+ − * clear TMSI and ciphering data
+ − */
+ − att_clear_reg_without_imsi ();
+ − att_set_rr_service_info();
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_clear_reg_without_imsi |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After some location updating reject causes MM informs RR
+ − that the TMSI and the ciphering data are invalid.
+ −
+ − */
+ −
+ − GLOBAL void att_clear_reg_without_imsi (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_clear_reg_without_imsi()");
+ −
+ − /*
+ − * clear TMSI
+ − */
+ − rr_data->ms_data.tmsi_available = FALSE;
+ − rr_data->ms_data.tmsi_binary = 0;
+ −
+ − /*
+ − * clear ciphering data
+ − */
+ − rr_data->ms_data.cksn = CKSN_NOT_PRES;
+ − memset (rr_data->ms_data.kc, NOT_PRESENT_8BIT, KC_STRING_SIZE);
+ − memset (rr_data->ms_data.new_kc, NOT_PRESENT_8BIT, KC_STRING_SIZE);
+ −
+ − /*
+ − * update the identities in layer 1.
+ − */
+ − att_mph_identity_req ();
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_reorder_mph_ncell_req |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : reorders a neighbourcell list for layer 1 to make sure
+ − that the strongest neighbour cells are the first ones
+ − in the list and are therefore measured first by layer 1
+ − */
+ −
+ − static void att_reorder_mph_ncell_req (T_MPH_NEIGHBOURCELL_REQ * mph_ncell_req)
+ − {
+ − GET_INSTANCE_DATA;
+ − #define MAX_STRONGEST 6 /* MAX_STRONGEST must be at least 6 but up to 8 makes sense */
+ − UBYTE i_nc,
+ − i_nc1,
+ − c_str_nc = 0,
+ − c_str_cs = 0,
+ − is_old = !IS_TIMER_ACTIVE(TCSVALID),
+ − is_inserted, i, j;
+ − SHORT arfcn,
+ − rxlev,
+ − str_arfcn[MAX_STRONGEST],
+ − str_rxlev[MAX_STRONGEST];
+ −
+ − #if defined(_SIMULATION_)
+ − TRACE_EVENT_WIN_P6("ncell_req arfcns:%4d %4d %4d %4d %4d %4d",
+ − mph_ncell_req->arfcn[0] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[0] : -1,
+ − mph_ncell_req->arfcn[1] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[1] : -1,
+ − mph_ncell_req->arfcn[2] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[2] : -1,
+ − mph_ncell_req->arfcn[3] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[3] : -1,
+ − mph_ncell_req->arfcn[4] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[4] : -1,
+ − mph_ncell_req->arfcn[5] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[5] : -1);
+ − #endif
+ −
+ − memset (str_arfcn, -1, sizeof str_arfcn); /* for easier debugging */
+ − memset (str_rxlev, -1, sizeof str_rxlev);
+ −
+ − /*
+ − * search for the strongest neighbourcells and collect them in
+ − * the arrays str_arfcn/str_rxlev. An arfcn found in the nc_data
+ − * is always stored before an arfcn found in the cs_data. The reason
+ − * is that the nc_data are almost younger and probably more important be
+ − * and there may big differences between the rxlev values for the
+ − * same arfcn delevered by MPHC_RXLEV_IND and by MPHC_RXLEV_PERIODIC_IND
+ − */
+ − for (i_nc = 0; i_nc < MAX_NEIGHBOURCELLS AND
+ − mph_ncell_req->arfcn[i_nc] NEQ NOT_PRESENT_16BIT; i_nc++)
+ − {
+ − arfcn = mph_ncell_req->arfcn[i_nc];
+ −
+ − if (arfcn EQ rr_data->nc_data[SC_INDEX].arfcn)
+ − continue; /* processing neighbour cell list in *mph_ncell_req */
+ −
+ − /*
+ − * for all channels of stored neighbour cell data
+ − */
+ − is_inserted = FALSE;
+ − for (i_nc1 = 0; i_nc1 < 6; i_nc1++)
+ − {
+ − if (arfcn EQ rr_data->nc_data[i_nc1].arfcn)
+ − {
+ − if (rr_data->nc_data[i_nc1].bcch_status NEQ EMPTY)
+ − {
+ − rxlev = rr_data->nc_data[i_nc1].rxlev;
+ − is_inserted = TRUE; /* avoid double inserting */
+ − for (i = c_str_nc; i > 0 AND rxlev > str_rxlev[i-1]; i--)
+ − ; /* search for the place to insert */
+ − if (i EQ c_str_nc AND c_str_nc >= MAX_STRONGEST)
+ − break; /* not strong enough */
+ − j = (c_str_cs >= MAX_STRONGEST) ? MAX_STRONGEST-1 : c_str_cs;
+ − for ( ; j > i; j--)
+ − { /* make place for current entry */
+ − str_rxlev[j] = str_rxlev[j-1];
+ − str_arfcn[j] = str_arfcn[j-1];
+ − }
+ − str_arfcn[i] = arfcn;
+ − str_rxlev[i] = rxlev;
+ − /* if (c_str_nc < MAX_STRONGEST) needed if MAX_STRONGEST < 6 */
+ − c_str_nc++;
+ − if (c_str_cs < MAX_STRONGEST)
+ − c_str_cs++;
+ − }
+ − break; /* scanning stored neighbour cell data */
+ − }
+ − } /* for: stored neighbour cell data */
+ −
+ − if (is_old OR is_inserted)
+ − continue; /* CS measurements are too old */
+ −
+ − /*
+ − * for all channels of the power campaign list
+ − */
+ − for (i_nc1 = 0; i_nc1 < rr_data->cs_data.max_arfcn; i_nc1++)
+ − {
+ − if (arfcn EQ rr_data->cs_data.arfcn[i_nc1])
+ − {
+ − rxlev = rr_data->cs_data.rxlev[i_nc1];
+ − for (i = c_str_cs; i > c_str_nc AND rxlev > str_rxlev[i-1]; i--)
+ − ; /* search for the place to insert */
+ − if (i EQ c_str_cs AND c_str_cs >= MAX_STRONGEST)
+ − break; /* not strong enough */
+ − j = (c_str_cs >= MAX_STRONGEST) ? MAX_STRONGEST-1 : c_str_cs;
+ − for (; j > i; j--)
+ − { /* make place for current entry */
+ − str_rxlev[j] = str_rxlev[j-1];
+ − str_arfcn[j] = str_arfcn[j-1];
+ − }
+ − str_arfcn[i] = arfcn;
+ − str_rxlev[i] = rxlev;
+ − if (c_str_cs < MAX_STRONGEST)
+ − c_str_cs++;
+ − break; /* scanning power campaign list */
+ − }
+ − } /* for: power campaign list */
+ − } /* for: mph_ncell_req->arfcn */
+ −
+ − TRACE_EVENT_P7("c_str_nc=%u arfcn:%4d %4d %4d %4d %4d %4d",
+ − c_str_nc, str_arfcn[0], str_arfcn[1], str_arfcn[2],
+ − str_arfcn[3], str_arfcn[4], str_arfcn[5]);
+ − TRACE_EVENT_P7("c_str_cs=%u rxlev:%4d %4d %4d %4d %4d %4d",
+ − c_str_cs, str_rxlev[0], str_rxlev[1], str_rxlev[2],
+ − str_rxlev[3], str_rxlev[4], str_rxlev[5]);
+ − /* now reorder the arfcn's in the primitive */
+ − for (i_nc1 = 0; i_nc1 < c_str_cs; i_nc1++)
+ − {
+ − for (i_nc = 0; i_nc < MAX_NEIGHBOURCELLS AND
+ − mph_ncell_req->arfcn[i_nc] NEQ NOT_PRESENT_16BIT; i_nc++)
+ − {
+ − if (str_arfcn[i_nc1] EQ mph_ncell_req->arfcn[i_nc])
+ − { /* swap */
+ − arfcn = mph_ncell_req->arfcn[i_nc1];
+ − mph_ncell_req->arfcn[i_nc1] = mph_ncell_req->arfcn[i_nc];
+ − mph_ncell_req->arfcn[i_nc] = arfcn;
+ − break;
+ − }
+ − } /* for: mph_ncell_req->arfcn */
+ − } /* for: str_arfcn */
+ − TRACE_EVENT_P6("strongest arfcns:%4d %4d %4d %4d %4d %4d",
+ − mph_ncell_req->arfcn[0] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[0] : -1,
+ − mph_ncell_req->arfcn[1] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[1] : -1,
+ − mph_ncell_req->arfcn[2] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[2] : -1,
+ − mph_ncell_req->arfcn[3] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[3] : -1,
+ − mph_ncell_req->arfcn[4] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[4] : -1,
+ − mph_ncell_req->arfcn[5] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[5] : -1);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_remove_bad_ncell |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : removes outdated ncells from RR store.
+ −
+ − */
+ −
+ − GLOBAL void att_remove_bad_rr_data_ncells()
+ − {
+ − GET_INSTANCE_DATA;
+ − T_CELL_DATA * cd;
+ − UBYTE i_nc, index;
+ − UBYTE found;
+ − USHORT act_ncell_list [MAX_NEIGHBOURCELLS];
+ − /*
+ − * get the pointer to the correct list.
+ − */
+ −
+ − cd = &rr_data->sc_data.cd;
+ − memset (act_ncell_list, 0xFF, 2*MAX_NEIGHBOURCELLS);
+ − srv_create_list (&cd->ncell_list, act_ncell_list, MAX_NEIGHBOURCELLS, FALSE, 0);
+ −
+ − for( index = 0; index < SC_INDEX; index++)
+ − {
+ − TRACE_EVENT_P3("[%u]i%u bcch_status=%u", rr_data->nc_data[index].arfcn,
+ − index, rr_data->nc_data[index].bcch_status);
+ − if( rr_data->nc_data[index].bcch_status NEQ EMPTY )
+ − {
+ − found = FALSE;
+ − for( i_nc = 0; act_ncell_list[i_nc] NEQ NOT_PRESENT_16BIT ; i_nc++)
+ − {
+ − if( rr_data->nc_data[index].arfcn EQ act_ncell_list[i_nc] )
+ − {
+ − found = TRUE;
+ − break;
+ − }
+ − }
+ − if( ! found )
+ − {
+ − TRACE_EVENT_P1(" Removing bad ncell [%u]", rr_data->nc_data[index].arfcn );
+ − rr_data->nc_data[index].bcch_status = EMPTY;
+ − rr_data->nc_data[index].arfcn = 0;
+ − rr_data->nc_data[index].avail_time = 0;
+ − }
+ − }
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_code_mph_ncell_req |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : build a neighbourcell list for layer 1 in idle mode.
+ −
+ − */
+ −
+ − GLOBAL void att_code_mph_ncell_req (UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_CELL_DATA * cd;
+ −
+ − PALLOC (mph_ncell_req, MPH_NEIGHBOURCELL_REQ);
+ −
+ − TRACE_FUNCTION ("att_code_mph_ncell_req()");
+ −
+ − /*
+ − * get the pointer to the correct list.
+ − */
+ − if (index EQ SC_INDEX)
+ − cd = &rr_data->sc_data.cd;
+ − else
+ − cd = &rr_data->cr_data.cd;
+ −
+ −
+ − cs_remove_BA_MA_from_black_list(rr_data->cs_data.region,&cd->ncell_list);
+ −
+ − /*
+ − * use neighbour cell list from system info 2/2bis/2ter
+ − */
+ − memset (mph_ncell_req->arfcn, 0xFF, 2*MAX_NEIGHBOURCELLS);
+ − srv_create_list (&cd->ncell_list,
+ − mph_ncell_req->arfcn, MAX_NEIGHBOURCELLS, FALSE,
+ − 0);
+ − memcpy (rr_data->act_ncell_list, mph_ncell_req->arfcn,
+ − sizeof (rr_data->act_ncell_list));
+ −
+ − #if defined(_SIMULATION_)
+ − TRACE_EVENT_WIN_P6("ncell_req arfcns:%4d %4d %4d %4d %4d %4d",
+ − mph_ncell_req->arfcn[0] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[0] : -1,
+ − mph_ncell_req->arfcn[1] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[1] : -1,
+ − mph_ncell_req->arfcn[2] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[2] : -1,
+ − mph_ncell_req->arfcn[3] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[3] : -1,
+ − mph_ncell_req->arfcn[4] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[4] : -1,
+ − mph_ncell_req->arfcn[5] NEQ NOT_PRESENT_16BIT ?
+ − mph_ncell_req->arfcn[5] : -1);
+ − #endif
+ −
+ − /*
+ − * set the multiband parameter
+ − */
+ − mph_ncell_req->multi_band = rr_data->ncell_mb;
+ − #ifdef GPRS
+ − mph_ncell_req->sync_only = NORMAL_BA;
+ − #endif
+ − /*
+ − * check the list against band restrictions.
+ − */
+ − srv_remove_frequencies_in_array (&mph_ncell_req->arfcn[0]);
+ −
+ − att_reorder_mph_ncell_req(mph_ncell_req);
+ − /*
+ − * send the list to layer 1
+ − */
+ − PSENDX (PL, mph_ncell_req);
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − for_check_and_configure_si2quater(index);
+ − #endif
+ −
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_code_mph_ncell_req_dedicated |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : builds a neighbourcell list for layer 1 in dedicated mode.
+ −
+ − */
+ −
+ − GLOBAL void att_code_mph_ncell_req_dedicated (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE start_index = 0;
+ − UBYTE old_index;
+ −
+ − if ( rr_data->emo_arfcn EQ NULL )
+ − {
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − USHORT si_read = (USHORT) (SYS_INFO_5_READ | SYS_INFO_5BIS_READ);
+ −
+ − rr_data->sc_data.ba_list_ded = FALSE;
+ − if ( (rr_data->sc_data.cd.sys_info_read & si_read ) EQ si_read )
+ − {
+ − rr_data->sc_data.ba_list_ded = TRUE;
+ − rr_data->sc_data.ba_list_idle = FALSE;
+ − old_index = rr_data->sc_data.ba_index;
+ − rr_data->sc_data.ba_index = rr_data->sc_data.new_ba_index;
+ − }
+ − else if (rr_data->sc_data.ba_list_idle EQ TRUE)
+ − return;
+ − #endif
+ − {
+ − PALLOC (mph_ncell_req, MPH_NEIGHBOURCELL_REQ);
+ −
+ − /*
+ − * initialise data structures,
+ − * clear ncell list to layer 1,
+ − * set multiband parameter.
+ − */
+ − memset (mph_ncell_req->arfcn, 0xFF, 2*MAX_NEIGHBOURCELLS);
+ − mph_ncell_req->multi_band = rr_data->ncell_mb;
+ −
+ − /*
+ − * fill system info 5 and 5bis
+ − */
+ − if (rr_data->sc_data.cd.sys_info_read & (SYS_INFO_5_READ | SYS_INFO_5BIS_READ))
+ − start_index = srv_create_list (&rr_data->sc_data.cd.ncell_list,
+ − mph_ncell_req->arfcn,
+ − MAX_NEIGHBOURCELLS,
+ − FALSE,
+ − 0);
+ − /*
+ − * fill system info 5ter
+ − */
+ − if (rr_data->sc_data.cd.sys_info_read & SYS_INFO_5TER_READ)
+ − srv_create_list (&rr_data->sc_data.five_ter_list,
+ − mph_ncell_req->arfcn,
+ − MAX_NEIGHBOURCELLS,
+ − FALSE,
+ − start_index);
+ − /*
+ − * Copy complete list for position detection for uplink
+ − * measurement reports.
+ − */
+ − memcpy (rr_data->act_ncell_list, mph_ncell_req->arfcn,
+ − sizeof (rr_data->act_ncell_list));
+ −
+ − /*
+ − * check against band restrictions
+ − */
+ − srv_remove_frequencies_in_array (&mph_ncell_req->arfcn[0]);
+ −
+ − /*
+ − * send list to layer 1.
+ − */
+ − PSENDX (PL, mph_ncell_req);
+ − }
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − if (rr_data->sc_data.ba_list_ded NEQ TRUE )
+ − return;
+ − att_check_for_si5ter_and_enhpara(old_index);
+ − #endif
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_code_prr_mm_info |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : provide MM at the end of cell selection or cell reselection
+ − with MM relevant data from the BCCH channel. These
+ − informations are:
+ −
+ − IMSI attach / detach
+ − Call Reestablishment capability
+ − Base Station Identification Code
+ − Periodic Location Updating Time T3212
+ − and whether the MS camps in a forbidden location area.
+ −
+ − */
+ −
+ − static void att_code_prr_mm_info (T_mm_info *mm_info)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_NC_DATA *rrd = &rr_data->nc_data[SC_INDEX];
+ −
+ − TRACE_FUNCTION ("att_code_prr_mm_info()");
+ −
+ − mm_info->valid = TRUE;
+ − mm_info->att = rrd->control_descr.att;
+ − mm_info->re = rrd->rach.re;
+ − mm_info->ncc = (rrd->bsic >> 3) & 7;
+ − mm_info->bcc = rrd->bsic & 7;
+ − mm_info->t3212 = rrd->control_descr.t3212;
+ − mm_info->la = (!(dat_forb_lai_check (SC_INDEX) AND
+ − dat_roam_forb_lai_check (SC_INDEX)));
+ −
+ − /*
+ − * Parameter band needs a dummy value, otherwise component tests of RR
+ − * fail due to lack of ability of TAP
+ − */
+ − mm_info->band = BND_DMY_VAL;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_code_rr_abort_ind |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Indication of a failed cell selection or cell reselection
+ − to MM.
+ −
+ − */
+ − GLOBAL void att_code_rr_abort_ind_original (T_RR_DATA *rr_dat, USHORT cause)
+ − {
+ − GET_INSTANCE_DATA;
+ − int i;
+ − PALLOC (abort_ind, RR_ABORT_IND);/* T_RR_ABORT_IND */
+ −
+ − TRACE_FUNCTION ("att_code_rr_abort_ind()");
+ −
+ − memset (abort_ind, 0, sizeof (*abort_ind));
+ −
+ − /*
+ − * Indicates the end of a plmn available search
+ − */
+ − if (rr_dat->ms_data.req_mm_service EQ FUNC_NET_SRCH_BY_MMI)
+ − {
+ − /* this sets the state of CS */
+ − cs_set_stop_active();
+ − }
+ −
+ − TIMERSTOP(TABORT);
+ − /*
+ − * Set the parameters for MM, especially the requested service
+ − * from MM and the real reached service of RR.
+ − */
+ − abort_ind->op.v_op = 1;
+ − abort_ind->op.ts = (rr_dat->ms_data.operation_mode >> SHIFT_FOR_SIM_TYPE) & 1;
+ − abort_ind->op.m = (rr_dat->ms_data.operation_mode >> SHIFT_FOR_SEARCH_OFFSET) & 1;
+ − abort_ind->op.sim_ins = (rr_dat->ms_data.operation_mode >> SHIFT_FOR_SIM_INSERTED) & 1;
+ − abort_ind->op.func = rr_dat->ms_data.req_mm_service;
+ − abort_ind->op.service = rr_dat->ms_data.rr_service;
+ − abort_ind->cause = cause;
+ −
+ − if(rr_dat->ms_data.rr_service EQ NO_SERVICE)
+ − att_notify_stop_plmn_search (TRUE);
+ − else
+ − att_notify_stop_plmn_search (FALSE);
+ −
+ − /*
+ − * resume normal RR operation and update the previous
+ − * requested MM and RR services
+ − */
+ − att_clear_parallel_search ();
+ −
+ − rr_dat->ms_data.req_mm_service = att_get_func ();
+ − rr_data->cs_data.scan_mode = CS_NO_SCAN;
+ − TRACE_EVENT_P1 ("cause=%x", cause);
+ −
+ − if (cause EQ RRCS_ABORT_CEL_SEL_FAIL)
+ − {
+ − #ifdef GPRS
+ − if (abort_ind->op.func NEQ FUNC_NET_SRCH_BY_MMI AND
+ − abort_ind->op.service EQ NO_SERVICE AND
+ − att_gprs_is_avail())
+ − {
+ − att_signal_gprs_support();
+ − }
+ − #endif
+ −
+ − /*
+ − * If the abort cause is cell selection failed
+ − * add the list of available PLMNs.
+ − */
+ − att_order_plmns ();
+ − abort_ind->plmn_avail = rr_dat->sc_data.found_entries;
+ − for (i=0; i < rr_dat->sc_data.found_entries; i++)
+ − {
+ − T_FOUND_ELEMENT *pfound = &rr_dat->sc_data.found[i];
+ −
+ − abort_ind->plmn[i] = pfound->plmn;
+ − abort_ind->rxlevel[i] = pfound->rxlev;
+ − abort_ind->lac_list[i] =pfound->lac; //LOL 02.01.2003: added for EONS support
+ −
+ − TRACE_EVENT_P9 ("RR_ABORT_IND: [%u] MCC/MNC=%x%x%x/%x%x%x lac=%04x rx=%u",
+ − pfound->arfcn,
+ − pfound->plmn.mcc[0],
+ − pfound->plmn.mcc[1],
+ − pfound->plmn.mcc[2],
+ − pfound->plmn.mnc[0],
+ − pfound->plmn.mnc[1],
+ − pfound->plmn.mnc[2],
+ − pfound->lac,
+ − pfound->rxlev);
+ − }
+ −
+ − EM_NET_SEARCH_PASSED;
+ − }
+ − else
+ − {
+ − /*
+ − * else clear this area of the primitive.
+ − */
+ − abort_ind->plmn_avail = 0;
+ − memset (abort_ind->plmn, 0, sizeof (abort_ind->plmn));
+ − memset (abort_ind->rxlevel, 0, sizeof (abort_ind->rxlevel));
+ −
+ − EM_NET_SEARCH_FAILED;
+ − }
+ −
+ − /*
+ − * Set mobile RF capability for MM messages
+ − */
+ − if (rr_dat->ms_data.rr_service EQ NO_SERVICE)
+ − abort_ind->power = 0;
+ − else
+ − abort_ind->power = att_get_power();
+ −
+ − if ((rr_data->ms_data.req_mm_service NEQ FUNC_NET_SRCH_BY_MMI) AND
+ − (cause EQ RRCS_ABORT_CEL_SEL_FAIL))
+ − att_set_rr_service_info();
+ −
+ − TRACE_OP_TYPE (&abort_ind->op, "RR_ABORT_IND");
+ − PSENDX (MM, abort_ind);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_order_plmns |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function orders the found PLMNs after the following
+ − criterions:
+ −
+ − all PLMNs with a fieldstrength greater then -85 dBm are
+ − ordered randomly. This is 50 percent of the requirement
+ − regarding GSM 3.22, chapter 4.4.3.1. MM checks then
+ − the HPLMN condition and the preferred PLMN condition.
+ −
+ − */
+ −
+ − static void att_order_plmns (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − USHORT nr;
+ − USHORT i;
+ − /*
+ − * allocate dynamic memory to build a copy
+ − * of the list which is ordered randomly
+ − */
+ − T_FOUND_LIST * list;
+ −
+ − TRACE_FUNCTION ("att_order_plmns()");
+ −
+ − /*
+ − * initialize the copy
+ − */
+ − MALLOC (list, sizeof (T_FOUND_LIST));
+ − memset (list, 0, sizeof (T_FOUND_LIST));
+ −
+ − /*
+ − * get the number of PLMNs which have a fieldstrength
+ − * greater then -85 dBm.
+ − */
+ − i=0;
+ − while ((nr = att_number_of_plmns_greater_85_dBm()) NEQ 0 )
+ − {
+ − /*
+ − * get a value in the range 0..nr-1
+ − * i.e. select a PLMN greater -85 randomly
+ − */
+ − nr = dat_random (nr);
+ −
+ − /*
+ − * copy the nr-th plmn and indicate it in the source
+ − * as copied.
+ − */
+ − att_copy_found_plmn (list, nr, i);
+ −
+ − /*
+ − * increment the number of copied networks.
+ − */
+ − i++;
+ − }
+ −
+ − /*
+ − * copy all networks with a fieldstrength lower or equal -85 dBm.
+ − */
+ − att_copy_plmns_lower_or_equal_85_dBm (list, i);
+ −
+ − /*
+ − * copy back the randomly sorted list
+ − */
+ − for (i=0; i < MAX_PLMN; i++)
+ − rr_data->sc_data.found[i] = list->element[i];
+ −
+ − /*
+ − * de-allocate the dynamic memory
+ − */
+ − MFREE (list);
+ − }
+ −
+ − /*
+ − +-----------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_number_of_plmns_greater_85_dBm |
+ − +-----------------------------------------------------------------------+
+ −
+ − PURPOSE : The function calculates the number of available PLMNs with a
+ − fieldstrength higher then -85 dBm.
+ −
+ − */
+ −
+ − static USHORT att_number_of_plmns_greater_85_dBm (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − USHORT nr = 0;
+ − USHORT i;
+ −
+ − for (i=0; i< rr_data->sc_data.found_entries;i++)
+ − {
+ − if (rr_data->sc_data.found[i].plmn.v_plmn EQ V_PLMN_PRES)
+ − {
+ − if (rr_data->sc_data.found[i].rxlev > 25)
+ − {
+ − /*
+ − * -85 dBm = -110 + 25 !
+ − *
+ − * increment the number of found plmns.
+ − */
+ − nr++;
+ − }
+ − }
+ − }
+ −
+ − /*
+ − * return the number of PLMNs with a fieldstrength greater than -85 dBm
+ − */
+ − return nr;
+ − }
+ −
+ − /*
+ − +-----------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_found_plmn |
+ − +-----------------------------------------------------------------------+
+ −
+ − PURPOSE : The function copies the n-th plmn with a fieldstrength
+ − higher then -85 dBm in the found plmn copy.
+ −
+ − */
+ −
+ − static void att_copy_found_plmn (T_FOUND_LIST * list,
+ − USHORT n_in_source_85_dBm,
+ − USHORT i_in_copy)
+ − {
+ − GET_INSTANCE_DATA;
+ − USHORT x_in_source_85_dBm = 0;
+ − USHORT i_in_source = 0;
+ − T_FOUND_ELEMENT *in_source;
+ −
+ − in_source = &rr_data->sc_data.found[0];
+ − for (i_in_source=0; i_in_source< rr_data->sc_data.found_entries;
+ − i_in_source++, in_source++)
+ − {
+ − if ((in_source->plmn.v_plmn EQ V_PLMN_PRES) AND (in_source->rxlev > 25))
+ − {
+ − /*
+ − * -85 dBm = -110 + 25 !
+ − *
+ − * increment the number of found plmns.
+ − */
+ − if (x_in_source_85_dBm EQ n_in_source_85_dBm)
+ − {
+ − /*
+ − * n-th PLMN with fieldstrength greater -85 dBm found
+ − */
+ − list->element[i_in_copy] = *in_source;
+ − i_in_copy ++;
+ − in_source->plmn.v_plmn = V_PLMN_NOT_PRES;
+ − return;
+ − }
+ − else
+ − x_in_source_85_dBm++;
+ − }
+ − }
+ − }
+ −
+ − /*
+ − +-------------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_plmns_lower_or_equal_85_dBm |
+ − +-------------------------------------------------------------------------+
+ −
+ − PURPOSE : The function copies the remaining networks with a fieldstrength
+ − lower or equal -85 dBm in the found plmn copy.
+ −
+ − */
+ −
+ − static void att_copy_plmns_lower_or_equal_85_dBm (T_FOUND_LIST * list,
+ − USHORT i_in_copy)
+ − {
+ − GET_INSTANCE_DATA;
+ − USHORT i_in_source;
+ − T_FOUND_ELEMENT * in_source = &rr_data->sc_data.found[0];
+ −
+ − for (i_in_source=0; i_in_source< rr_data->sc_data.found_entries;
+ − i_in_source++, in_source++)
+ − {
+ − if (in_source->plmn.v_plmn EQ V_PLMN_PRES)
+ − {
+ − /*
+ − * PLMN with fieldstrength less or equal -85 dBm found
+ − */
+ − list->element[i_in_copy] = *in_source;
+ − i_in_copy ++;
+ − }
+ − }
+ − }
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_code_net_lost |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE :
+ −
+ − */
+ −
+ − GLOBAL void att_code_net_lost (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − PALLOC (abort_ind, RR_ABORT_IND);
+ −
+ − TRACE_FUNCTION ("att_code_net_lost()");
+ −
+ − srv_clear_stored_prim (RR_ESTABLISH_REQ);
+ −
+ − abort_ind->op.v_op = 1;
+ − abort_ind->op.ts = (rr_data->ms_data.operation_mode >> SHIFT_FOR_SIM_TYPE) & 1;
+ − abort_ind->op.m = (rr_data->ms_data.operation_mode >> SHIFT_FOR_SEARCH_OFFSET) & 1;
+ − abort_ind->op.sim_ins = (rr_data->ms_data.operation_mode >> SHIFT_FOR_SIM_INSERTED) & 1;
+ − abort_ind->op.func = rr_data->ms_data.req_mm_service;
+ − abort_ind->op.service = NO_SERVICE;
+ −
+ − abort_ind->cause = RRCS_ABORT_CEL_SEL_FAIL;
+ −
+ − abort_ind->plmn_avail = 0;
+ − memset (abort_ind->plmn, 0, sizeof(abort_ind->plmn));
+ − memset (abort_ind->rxlevel, 0, sizeof (abort_ind->rxlevel));
+ −
+ − TRACE_OP_TYPE (&abort_ind->op, "RR_ABORT_IND");
+ − PSENDX (MM, abort_ind);
+ −
+ − rr_data->old_lai.lac = 0xFFFF;
+ − #ifdef GPRS
+ − rr_data->old_rac = NOT_PRESENT_8BIT;
+ − #endif
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_code_rr_act_cnf |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function indicates a successful end of cell selection.
+ − The cell selection was initiated by MM.
+ −
+ − */
+ −
+ − GLOBAL void att_code_rr_act_cnf (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_NC_DATA * rrd = &rr_data->nc_data[SC_INDEX];
+ −
+ − PALLOC (rr_activate_cnf, RR_ACTIVATE_CNF);
+ −
+ − TRACE_FUNCTION ("att_code_rr_act_cnf()");
+ −
+ − TRACE_EVENT ("cell selection");
+ −
+ −
+ − rr_data->net_lost = FALSE;
+ −
+ − TIMERSTOP(TABORT);
+ −
+ − /*
+ − * indicate the requested MM service and the reached RR service
+ − * to MM.
+ − */
+ − rr_activate_cnf->op.v_op = 1;
+ − rr_activate_cnf->op.ts = (rr_data->ms_data.operation_mode >> SHIFT_FOR_SIM_TYPE) & 1;
+ − rr_activate_cnf->op.m = (rr_data->ms_data.operation_mode >> SHIFT_FOR_SEARCH_OFFSET) & 1;
+ − rr_activate_cnf->op.sim_ins = (rr_data->ms_data.operation_mode >> SHIFT_FOR_SIM_INSERTED) & 1;
+ − rr_activate_cnf->op.func = rr_data->ms_data.req_mm_service;
+ − rr_activate_cnf->op.service = rr_data->ms_data.rr_service;
+ −
+ − TRACE_EVENT_P9 ("RR_ACTIVATE_CNF[%u]: rMM=%s RRs=%s MCC/MNC=%x%x%x/%x%x%x",
+ − rrd->arfcn,
+ − _rr_str_FUNC[rr_data->ms_data.req_mm_service],
+ − _rr_str_SERVICE[rr_data->ms_data.rr_service],
+ − rrd->lai.mcc[0],
+ − rrd->lai.mcc[1],
+ − rrd->lai.mcc[2],
+ − rrd->lai.mnc[0],
+ − rrd->lai.mnc[1],
+ − rrd->lai.mnc[2]);
+ −
+ − if(GET_STATE(STATE_ATT) EQ ATT_CS_INIT )
+ − { /*Boot Time: Indicate only the start of power scan to MM*/
+ − memset( &rr_activate_cnf->mm_info, 0, sizeof(rr_activate_cnf->mm_info));
+ − memset( &rr_activate_cnf->plmn, 0, sizeof(rr_activate_cnf->plmn));
+ − rr_activate_cnf->lac = NOT_PRESENT_16BIT;
+ − rr_activate_cnf->cid = NOT_PRESENT_16BIT;
+ − rr_activate_cnf->gprs_indication = NOT_PRESENT_8BIT;
+ − rr_activate_cnf->power = NOT_PRESENT_8BIT;
+ − }
+ − else
+ − {
+ − /*
+ − * stop possibly parallel PLMN search
+ − */
+ − att_notify_stop_plmn_search (FALSE);
+ −
+ − /*
+ − * resume normal RR operation and update the previous
+ − * requested MM and RR services
+ − */
+ − att_clear_parallel_search ();
+ − rr_data->ms_data.req_mm_service = att_get_func ();
+ − rr_data->cs_data.scan_mode = CS_NO_SCAN;
+ −
+ −
+ − if (rr_data->ms_data.req_mm_service NEQ rr_activate_cnf->op.func)
+ − {
+ − TRACE_EVENT_P2 ("rMM=%s RRs=%s resume normal RR operation",
+ − _rr_str_FUNC[rr_data->ms_data.req_mm_service],
+ − _rr_str_SERVICE[rr_data->ms_data.rr_service]);
+ − }
+ − TRACE_OP_TYPE (&rr_activate_cnf->op, "RR_ACTIVATE_CNF");
+ −
+ − /*
+ − * indicate the relevant BCCH parameter to MM
+ − */
+ − att_code_prr_mm_info (&rr_activate_cnf->mm_info);
+ −
+ − /*
+ − * indicate the PLMN, Location Area and Cell Identity to NN
+ − */
+ − rr_activate_cnf->plmn.v_plmn = V_PLMN_PRES;
+ − memcpy (rr_activate_cnf->plmn.mcc, rrd->lai.mcc, SIZE_MCC);
+ − memcpy (rr_activate_cnf->plmn.mnc, rrd->lai.mnc, SIZE_MNC);
+ −
+ − #if defined(_SIMULATION_)
+ − /* Implements Measure#32: Row 40 */
+ − att_print_mcc_mnc(rrd->arfcn, rrd->lai.mcc, rrd->lai.mnc, S2I_STRING("RR_ACTIVATE_CNF"));
+ − #endif
+ −
+ − rr_activate_cnf->lac = rrd->lai.lac;
+ − rr_activate_cnf->cid = rrd->cell_id;
+ −
+ − EM_PLMN_SRCH_PASSED;
+ −
+ − /*
+ − * store the current location area and cell id to avoid
+ − * double signalling to MM
+ − */
+ − att_copy_old_lai_rac(SC_INDEX);
+ − #if 0
+ − memcpy (&rr_data->old_lai, &rr_data->nc_data[SC_INDEX].lai,
+ − sizeof(T_loc_area_ident));
+ − rr_data->old_cell_id = rr_data->nc_data[SC_INDEX].cell_id;
+ − #endif
+ −
+ − #ifdef GPRS
+ − /*
+ − * indicates whether gprs operation is provided by the cell
+ − */
+ − /* MS Patch: CQ 24473 */
+ − rr_activate_cnf->gprs_indication = (rr_data->nc_data[SC_INDEX].rac NEQ
+ − NOT_PRESENT_8BIT);
+ −
+ − #else
+ − rr_activate_cnf->gprs_indication = FALSE;
+ − #endif
+ −
+ − /*
+ − * set the power class needed for some MM messages depending
+ − * on the frequency standard and the channel number
+ − */
+ − rr_activate_cnf->power = att_get_power();
+ −
+ − /*
+ − * Display the Signal bar after camping on a cell with Full service
+ − */
+ − #ifdef FF_PS_RSSI
+ − RX_SetValue (rr_data->nc_data[SC_INDEX].rxlev,
+ − RX_QUAL_UNAVAILABLE,
+ − rr_data->nc_data[SC_INDEX].select_para.rxlev_access_min);
+ − #else
+ − RX_SetValue (rr_data->nc_data[SC_INDEX].rxlev);
+ − #endif
+ − att_set_rr_service_info();
+ − }
+ −
+ − PSENDX (MM, rr_activate_cnf);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_code_rr_act_ind |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function indicates a successful end of RR originated
+ − cell selection or a successful end of cell reselection.
+ −
+ − */
+ −
+ − GLOBAL void att_code_rr_act_ind (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_NC_DATA * rrd = &rr_data->nc_data[SC_INDEX];
+ −
+ − TRACE_FUNCTION ("att_code_rr_act_ind()");
+ −
+ − TRACE_EVENT ("cell reselection");
+ −
+ − rr_data->net_lost = FALSE;
+ −
+ − TIMERSTOP (TABORT);
+ −
+ − /* It is expected that this works, but this means that */
+ − /* the la-field in T_mm_info is superflous. This is also */
+ − /* true with some other variables in T_mm_info. */
+ −
+ − /*
+ − * Special case if we have reselected on a cell belonging to a forbidden
+ − * location area in FS : it should not be possible on a LAC barred with a
+ − * LU_REJ(#13) ie roaming not allowed but can happen with a LAC barred
+ − * for regional provision of service ie LU_REJ(#12).
+ − * Then a RR_ABORT_IND is sent to MM to switch the entity into limited
+ − * service even if RR is happy with full service. An empty list of
+ − * available PLMN is reported so that no PLMN selection is triggered
+ − * by MM if MS is in automatic mode.
+ − */
+ − if (rr_data->ms_data.rr_service EQ FULL_SERVICE)
+ − {
+ − if (!(dat_forb_lai_check (SC_INDEX) AND
+ − dat_roam_forb_lai_check (SC_INDEX)))
+ − {
+ − /*
+ − * store the current location area and cell id to notify MM
+ − * when we are back in service
+ − */
+ − att_copy_old_lai_rac(SC_INDEX);
+ − #if 0
+ − memcpy (&rr_data->old_lai, &rr_data->nc_data[SC_INDEX].lai,
+ − sizeof(T_loc_area_ident));
+ − rr_data->old_cell_id = rr_data->nc_data[SC_INDEX].cell_id;
+ − #endif
+ −
+ − /*
+ − * Indicate limited service, no PLMN available to MM
+ − */
+ − rr_data->sc_data.found_entries = 0;
+ − rr_data->ms_data.rr_service = LIMITED_SERVICE;
+ − att_code_rr_abort_ind (RRCS_ABORT_CEL_SEL_FAIL);
+ − rr_data->ms_data.rr_service = FULL_SERVICE;
+ −
+ − return;
+ − }
+ − }
+ −
+ − /* Set the black list search flag, whenever the location area or
+ − * routing area changes, so that black list search is started
+ − * after location/routing area update
+ − */
+ − if(rr_data->dyn_config.bl_cs_en)
+ − {
+ − if((!dat_plmn_equal_req (rrd->lai.mcc, rrd->lai.mnc,
+ − rr_data->old_lai.mcc, rr_data->old_lai.mnc))
+ − OR
+ − (rrd->lai.lac NEQ rr_data->old_lai.lac ))
+ − {
+ − /* Location area has changed. This flag shall be used to start black
+ − * list search after location area update
+ − */
+ − rr_data->cs_data.black_list_search_pending = TRUE;
+ − }
+ − #ifdef GPRS
+ − if(att_gprs_is_avail() AND (GET_STATE(STATE_GPRS) EQ GPRS_PIM_BCCH))
+ − {
+ − if((rrd->rac NEQ NOT_PRESENT_8BIT) AND (rrd->rac NEQ rr_data->old_rac))
+ − {
+ − /* Routing area has changed. This flag shall be used to start black
+ − * list search after routing area update in PIM_BCCH state
+ − */
+ − rr_data->cs_data.black_list_search_pending = TRUE;
+ − }
+ − }
+ − #endif
+ −
+ − if(rr_data->cs_data.black_list_search_pending)
+ − TRACE_EVENT("black_list_search_pending : 1");
+ − }
+ −
+ − /*
+ − * check the current location area identification and cell identity
+ − * against the last signalled one to MM. If it is not the same
+ − * a signalling with RR_ACTIVATE_IND is needed
+ − */
+ − if (!dat_plmn_equal_req (rrd->lai.mcc, rrd->lai.mnc,
+ − rr_data->old_lai.mcc, rr_data->old_lai.mnc)
+ − OR
+ − rrd->lai.lac NEQ rr_data->old_lai.lac
+ − OR
+ − rrd->cell_id NEQ rr_data->old_cell_id)
+ − {
+ − PALLOC (rr_activate_ind, RR_ACTIVATE_IND);
+ −
+ − /*
+ − * A cell change has occurred. So a running Immediate assignment reject
+ − * timer is stopped.
+ − */
+ − /* If the MS reselects back onto the same SC (ARFCN) and T3122 is still
+ − running, it should be left running. If a new cell is selected, this timer
+ − should is stopped. MM is not informed, because it have already received a
+ − RR_ABORT_IND and left its state.*/
+ −
+ − TIMERSTOP (T3122);
+ −
+ −
+ −
+ − /*
+ − * indicate the requested MM service and the reached RR service
+ − * to MM.
+ − */
+ − rr_activate_ind->op.v_op = 1;
+ − rr_activate_ind->op.ts = (rr_data->ms_data.operation_mode >> SHIFT_FOR_SIM_TYPE) & 1;
+ − rr_activate_ind->op.m = (rr_data->ms_data.operation_mode >> SHIFT_FOR_SEARCH_OFFSET) & 1;
+ − rr_activate_ind->op.sim_ins = (rr_data->ms_data.operation_mode >> SHIFT_FOR_SIM_INSERTED) & 1;
+ − if (rr_data->ms_data.req_mm_service EQ FUNC_NET_SRCH_BY_MMI)
+ − rr_activate_ind->op.func = FUNC_PLMN_SRCH;
+ − else
+ − rr_activate_ind->op.func = rr_data->ms_data.req_mm_service;
+ − rr_activate_ind->op.service = rr_data->ms_data.rr_service;
+ −
+ − TRACE_OP_TYPE (&rr_activate_ind->op, "RR_ACTIVATE_IND");
+ −
+ − /*
+ − * stop possibly parallel PLMN search
+ − */
+ − att_notify_stop_plmn_search (FALSE);
+ −
+ − /*
+ − * resume normal RR operation and update the previous
+ − * requested MM and RR services
+ − */
+ − att_clear_parallel_search();
+ − rr_data->ms_data.req_mm_service = att_get_func ();
+ − rr_data->cs_data.scan_mode = CS_NO_SCAN;
+ −
+ −
+ − /*
+ − * indicate the relevant BCCH parameter to MM
+ − */
+ − att_code_prr_mm_info (&rr_activate_ind->mm_info);
+ −
+ − /*
+ − * indicate the PLMN, Location Area and Cell Identity to NN
+ − */
+ − rr_activate_ind->plmn.v_plmn = V_PLMN_PRES;
+ − memcpy (rr_activate_ind->plmn.mcc, rrd->lai.mcc, SIZE_MCC);
+ − memcpy (rr_activate_ind->plmn.mnc, rrd->lai.mnc, SIZE_MNC);
+ − #if defined(_SIMULATION_)
+ − /* Implements Measure#32: Row 40 */
+ − att_print_mcc_mnc(rrd->arfcn, rrd->lai.mcc, rrd->lai.mnc, S2I_STRING("RR_ACTIVATE_IND"));
+ − #endif
+ − rr_activate_ind->lac = rrd->lai.lac;
+ − rr_activate_ind->cid = rrd->cell_id;
+ −
+ − /*
+ − * store the current location area and cell id to avoid
+ − * double signalling to MM
+ − */
+ − att_copy_old_lai_rac(SC_INDEX);
+ − #if 0
+ − memcpy (&rr_data->old_lai, &rr_data->nc_data[SC_INDEX].lai,
+ − sizeof(T_loc_area_ident));
+ − rr_data->old_cell_id = rr_data->nc_data[SC_INDEX].cell_id;
+ − #endif
+ −
+ − #ifdef GPRS
+ − /*
+ − * indicates whether gprs operation is provided by the cell
+ − */
+ − /* MS Patch: CQ 24473 */
+ − rr_activate_ind->gprs_indication = (rr_data->nc_data[SC_INDEX].rac NEQ
+ − NOT_PRESENT_8BIT);
+ −
+ − #else
+ − rr_activate_ind->gprs_indication = FALSE;
+ − #endif
+ −
+ − /*
+ − * set the power class needed for some MM messages depending
+ − * on the frequency standard and the channel number
+ − */
+ − rr_activate_ind->power = att_get_power();
+ − /*
+ − * Limited to full service reselection
+ − * TMSI/IMSI data to be refreshed in TIL otherwise no paging possible
+ − */
+ − att_mph_identity_req();
+ −
+ − EM_CELL_RESEL_FINISHED;
+ −
+ − EM_FMM_RESEL_END_IND;
+ −
+ − PSENDX (MM, rr_activate_ind);
+ − }
+ − else
+ − {
+ − /*
+ − * indicate changed imsi attach status or
+ − * changed periodic location updating time to MM
+ − * if it has changed, else it is only signalled to MM.
+ − */
+ − PALLOC (sync, RR_SYNC_IND);
+ −
+ − att_code_prr_mm_info (&sync->mm_info);
+ − sync->ciph = NOT_PRESENT_8BIT;
+ − sync->chm.ch_mode = NOT_PRESENT_8BIT;
+ −
+ − switch (rr_data->sc_data.selection_type)
+ − {
+ − case CELL_SELECTION:
+ − case CELL_RESELECTION:
+ − case CELL_RESELECTION_NC:
+ − case CELL_RESELECTION_RACH:
+ − case CELL_RESELECTION_CR:
+ − /*
+ − * cell reselection in idle mode
+ − */
+ − sync->synccs = SYNCCS_IDLE_SELECTION;
+ − break;
+ − default:
+ − /*
+ − * cell reselection after dedicated mode
+ − */
+ − sync->synccs = SYNCCS_BACK_FROM_DEDICATED;
+ − break;
+ − }
+ −
+ − sync->bcch_info.v_bcch = FALSE;
+ − PSENDX (MM, sync);
+ − EM_FMM_RESEL_END_IND;
+ − }
+ − att_set_rr_service_info();
+ − }
+ −
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_get_power |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : get the power class needed for some MM messages depending
+ − on the frequency standard and the channel number of the
+ − serving cell.
+ −
+ − */
+ −
+ − GLOBAL UBYTE att_get_power (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE power = 0;
+ − UBYTE power_idx = (UBYTE)-1;
+ −
+ − TRACE_FUNCTION ("att_get_power()");
+ −
+ − /* new method, use directly the rf capabilities */
+ − /* coding of power class according to classmark 3 [1..5] */
+ − switch (std)
+ − {
+ − case STD_900:
+ − case STD_EGSM:
+ − power_idx = IDX_PWRCLASS_900;
+ − break;
+ − case STD_1800:
+ − power_idx = IDX_PWRCLASS_1800;
+ − break;
+ −
+ − case STD_DUAL:
+ − case STD_DUAL_EGSM:
+ − if (INRANGE(LOW_CHANNEL_1800,rr_data->nc_data[SC_INDEX].arfcn,HIGH_CHANNEL_1800))
+ − power_idx = IDX_PWRCLASS_1800;
+ − else
+ − power_idx = IDX_PWRCLASS_900;
+ − break;
+ −
+ − case STD_1900:
+ − power_idx = IDX_PWRCLASS_1900;
+ − break;
+ −
+ − case STD_850:
+ − power_idx = IDX_PWRCLASS_850;
+ − break;
+ −
+ − case STD_DUAL_US:
+ − if (rr_data->nc_data[SC_INDEX].arfcn < LOW_CHANNEL_1900)
+ − power_idx = IDX_PWRCLASS_850;
+ − else
+ − power_idx = IDX_PWRCLASS_1900;
+ − break;
+ −
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − case STD_850_1800:
+ − if (INRANGE(LOW_CHANNEL_1800,rr_data->nc_data[SC_INDEX].arfcn,HIGH_CHANNEL_1800))
+ − power_idx = IDX_PWRCLASS_1800;
+ − else
+ − power_idx = IDX_PWRCLASS_850;
+ − break;
+ −
+ − case STD_900_1900:
+ − if (INRANGE(LOW_CHANNEL_1900,rr_data->nc_data[SC_INDEX].arfcn,HIGH_CHANNEL_1900))
+ − power_idx = IDX_PWRCLASS_1900;
+ − else
+ − power_idx = IDX_PWRCLASS_900;
+ − break;
+ −
+ − case STD_850_900_1800:
+ − if (INRANGE(LOW_CHANNEL_1800,rr_data->nc_data[SC_INDEX].arfcn,HIGH_CHANNEL_1800))
+ − power_idx = IDX_PWRCLASS_1800;
+ − else if (INRANGE(LOW_CHANNEL_850,rr_data->nc_data[SC_INDEX].arfcn,HIGH_CHANNEL_850))
+ − power_idx = IDX_PWRCLASS_850;
+ − else
+ − power_idx = IDX_PWRCLASS_900;
+ − break;
+ −
+ − case STD_850_900_1900:
+ − if (INRANGE(LOW_CHANNEL_1900,rr_data->nc_data[SC_INDEX].arfcn,HIGH_CHANNEL_1900))
+ − power_idx = IDX_PWRCLASS_1900;
+ − else if (INRANGE(LOW_CHANNEL_850,rr_data->nc_data[SC_INDEX].arfcn,HIGH_CHANNEL_850))
+ − power_idx = IDX_PWRCLASS_850;
+ − else
+ − power_idx = IDX_PWRCLASS_900;
+ − break;
+ − #endif
+ − }
+ −
+ − if (power_idx NEQ (UBYTE)-1)
+ − power = rr_data->ms_data.rf_cap.rf_power.pow_class4[power_idx].pow_class;
+ − else
+ − power = 0;
+ −
+ − /*
+ − TRACE_EVENT_P3 ("power=%u(new), idx=%d, std=%u", power, power_idx, std);
+ − */
+ −
+ − if (power)
+ − power--; /* coding of power class according to classmark 1 and 2 [0..4] */
+ −
+ − return power;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_continue_cell_reselect |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : A cell reselection candidate has failed. Inside this function
+ − the next one is searched.
+ −
+ − */
+ −
+ − GLOBAL void att_continue_cell_reselect (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_continue_cell_reselect()");
+ −
+ − switch (rr_data->sc_data.selection_type)
+ − {
+ − case BACK_FROM_DEDICATED:
+ − case BACK_FROM_DEDICATED_RLF:
+ − /*
+ − * After a dedicated connection
+ − */
+ − rr_data->reselect_index =
+ − att_get_next_highest_c2_idx (rr_data->reselect_index);
+ − att_select_cell_dedicated ();
+ − break;
+ −
+ − default:
+ − /*
+ − * In idle mode
+ − */
+ − #ifdef GPRS
+ − if( rr_data->gprs_data.use_c31)
+ − {
+ − rr_data->reselect_index = att_get_next_best_c32_index(TRUE);
+ − if (! att_check_cell_c31())
+ − att_try_old_cell();
+ − }
+ − else
+ − {
+ − #endif
+ − rr_data->reselect_index =
+ − att_get_next_highest_c2_idx (rr_data->reselect_index);
+ − /*
+ − * If no other cell is available, try the old one
+ − */
+ − if (! att_check_cell ())
+ − {
+ − TRACE_EVENT ("no further cell found -> try old one");
+ − att_try_old_cell ();
+ − }
+ − else
+ − {
+ − TRACE_EVENT_P2 ("try next cell [%u]i%u",
+ − rr_data->nc_data[rr_data->reselect_index].arfcn,
+ − rr_data->reselect_index);
+ − }
+ −
+ −
+ −
+ − #ifdef GPRS
+ − }
+ − #endif
+ − break;
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_rach_parameter |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After reception of system information message 1 to 4 in
+ − idle mode or cell selection / reselection the RACH parameters
+ − are stored. The flag indicate_changes defines whether
+ − changes in the RACH parameter shall be forwarded to MM
+ − (in idle mode for serving cell) or not (any other cases).
+ −
+ − */
+ −
+ − static void att_copy_rach_parameter (UBYTE index,
+ − T_rach_ctrl *rach,
+ − UBYTE indicate_changes)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_NC_DATA *rrd = &rr_data->nc_data[index];
+ −
+ − TRACE_FUNCTION ("att_copy_rach_parameter()");
+ −
+ − if (index EQ SC_INDEX AND
+ − indicate_changes)
+ − {
+ − /*
+ − * compare only for serving cell
+ − */
+ − if (rrd->rach.re NEQ rach->re)
+ − {
+ − /*
+ − * change of reestablishment flag indicate to MM
+ − */
+ − PALLOC (sync, RR_SYNC_IND);
+ − rrd->rach.re = rach->re;
+ −
+ − att_code_prr_mm_info (&sync->mm_info);
+ − sync->ciph = NOT_PRESENT_8BIT;
+ − sync->chm.ch_mode = NOT_PRESENT_8BIT;
+ − sync->synccs = SYNCCS_SYS_INFO_CHANGE;
+ − memset(&sync->bcch_info, 0, sizeof(T_bcch_info));
+ − sync->bcch_info.v_bcch = FALSE;
+ − PSENDX (MM, sync);
+ − }
+ −
+ − if (rrd->rach.ac NEQ rach->ac)
+ − {
+ − /*
+ − * change of access classes indicate to MM
+ − */
+ − PALLOC (sync, RR_SYNC_IND);
+ −
+ − sync->ciph = NOT_PRESENT_8BIT;
+ − sync->chm.ch_mode = NOT_PRESENT_8BIT;
+ − sync->synccs = SYNCCS_ACC_CLS_CHA;
+ − memset(&sync->mm_info, 0, sizeof(T_mm_info));
+ − sync->mm_info.valid = FALSE;
+ − memset(&sync->bcch_info, 0, sizeof(T_bcch_info));
+ − sync->bcch_info.v_bcch = FALSE;
+ − PSENDX (MM, sync);
+ − }
+ − }
+ −
+ − rrd->rach = *rach;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_sys_info_1_par |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : copy the relevant system information type 1 parameter after
+ − reception in idle mode or during cell selection or cell
+ − reselection.
+ −
+ − */
+ −
+ − GLOBAL void att_copy_sys_info_1_par (UBYTE index,
+ − T_D_SYS_INFO_1 *sys_info_1,
+ − T_LIST *cell_chan_desc)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_CELL_DATA * cd;
+ − UBYTE indicate_changes = index EQ SC_INDEX;
+ −
+ − switch (index)
+ − {
+ − case SC_INDEX:
+ − case CR_INDEX:
+ − /*
+ − * set pointer to data
+ − */
+ − if (index EQ SC_INDEX)
+ − cd = &rr_data->sc_data.cd;
+ − else
+ − cd = &rr_data->cr_data.cd;
+ −
+ − /*
+ − * If a cell channel description is not stored for
+ − * this cell or the new cell channel description is
+ − * different, store the new one.
+ − */
+ − if (cd->v_cell_chan_desc EQ NO_CONTENT OR
+ − srv_compare_list (&cd->cell_chan_desc,
+ − cell_chan_desc) EQ FALSE)
+ − {
+ − srv_copy_list (&cd->cell_chan_desc,
+ − cell_chan_desc,
+ − sizeof (T_LIST));
+ −
+ − cd->v_cell_chan_desc = WITH_CHANGED_CONTENT;
+ − if((indicate_changes) AND (GET_STATE(STATE_ATT) EQ ATT_IDLE)
+ − AND ( rr_data->sc_data.cd.cbch_chan_desc.hop EQ H_FREQ) )
+ − {
+ − #ifdef REL99
+ − att_config_cbch();
+ − #else
+ − att_build_cbch();
+ − #endif
+ − }
+ − }
+ − /* Get the band indicator value from SI1 message */
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − cd->band_indicator = sys_info_1->si1_rest_oct.band_indicator;
+ − if (index EQ SC_INDEX)
+ − {
+ − att_update_std_band_indicator (cd->band_indicator);
+ − }
+ − #endif
+ − break;
+ − }
+ −
+ − att_copy_rach_parameter (index, &sys_info_1->rach_ctrl, indicate_changes);
+ −
+ − /*
+ − * Indicate that the system information type 1 message has been read.
+ − */
+ − att_set_sys_info_read (SYS_INFO_1_READ, index);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_sys_info_2bis_par |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : copy the relevant system information type 2bis parameter after
+ − reception in idle mode (in this case changes are to detect)
+ − or during cell selection or cell reselection.
+ −
+ − */
+ −
+ − GLOBAL void att_copy_sys_info_2bis_par (UBYTE index,
+ − T_D_SYS_INFO_2BIS *sys_info_2bis,
+ − T_LIST *new_2bis_list,
+ − UBYTE ncell_ext,
+ − UBYTE indicate_changes)
+ − {
+ − /* Implements RR Clone findings #3 */
+ −
+ − TRACE_FUNCTION ("att_copy_sys_info_2bis_par()");
+ −
+ −
+ − /* Implements RR Clone findings #3 */
+ − att_copy_sys_info_2bis_2ter_par(index, SI_TYPE_2BIS, new_2bis_list,
+ − &sys_info_2bis->neigh_cell_desc,
+ − indicate_changes);
+ −
+ −
+ −
+ − att_copy_rach_parameter (index, &sys_info_2bis->rach_ctrl, indicate_changes);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_sys_info_2_par |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : copy the relevant system information type 2 parameter after
+ − reception in idle mode (in this case changes are to detect)
+ − or during cell selection or cell reselection.
+ −
+ − */
+ −
+ − GLOBAL void att_copy_sys_info_2_par (UBYTE index,
+ − T_D_SYS_INFO_2 *sys_info_2,
+ − T_LIST *new_2_list,
+ − UBYTE ncell_ext,
+ − UBYTE indicate_changes)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_LIST new_list;
+ − T_CELL_DATA *cd;
+ − BOOL modified = FALSE;
+ −
+ − TRACE_FUNCTION ("att_copy_sys_info_2_par()");
+ −
+ − switch (index)
+ − {
+ − case SC_INDEX:
+ − case CR_INDEX:
+ − if (index EQ SC_INDEX)
+ − {
+ − cd = &rr_data->sc_data.cd;
+ − if ((cd->sys_info_read & SYS_INFO_2_READ) EQ SYS_INFO_2_READ)
+ − {
+ − srv_copy_list (&cd->ncell_list, new_2_list, sizeof (T_LIST));
+ − att_clean_buf ((USHORT)(IND_SI_2BIS | IND_SI_2TER));
+ − cd->sys_info_read &= ~(SYS_INFO_2BIS_READ | SYS_INFO_2TER_READ);
+ − att_check_2ter_read (index);
+ − }
+ − else
+ − srv_merge_list (&cd->ncell_list, new_2_list);
+ −
+ − if( cd->ncell_ext NEQ ncell_ext )
+ − {
+ − PALLOC (mph_sync_req, MPH_SYNC_REQ);
+ − mph_sync_req->cs = CS_SYS_INFO_2BIS_SUPPORT_CHANGED;
+ − PSENDX (PL, mph_sync_req);
+ − cd->ncell_ext = ncell_ext;
+ − }
+ − modified = TRUE;
+ − #ifdef GPRS
+ − rr_data->gprs_data.ba_bcch_modified= FALSE;
+ − #endif
+ − }
+ − else
+ − {
+ − cd = &rr_data->cr_data.cd;
+ − /*
+ − * merge 2 list with the old neighbour cell list
+ − */
+ − srv_copy_list (&new_list, &cd->ncell_list,
+ − sizeof (T_LIST));
+ − srv_merge_list (&new_list, new_2_list);
+ −
+ − if (srv_compare_list (&cd->ncell_list, &new_list) EQ FALSE)
+ − {
+ − /*
+ − * both lists are different
+ − */
+ − srv_copy_list (&cd->ncell_list, &new_list, sizeof (T_LIST));
+ − modified = TRUE;
+ − }
+ − cd->ncell_ext = ncell_ext;
+ − }
+ −
+ − /* CSI-LLD section:4.1.1.11
+ − * This function Updates the black list with the BA list received in si2
+ − */
+ − cs_remove_BA_MA_from_black_list(rr_data->cs_data.region,&cd->ncell_list);
+ −
+ − dat_store_neigh_cell_desc (SYS_INFO_2_MSG, index,
+ − &sys_info_2->neigh_cell_desc,&cd->ncell_list);
+ − /*
+ − * Indicate that the system information type 2 message has been read
+ − * and if applicable the system information type 2bis is not needed to read.
+ − */
+ − att_set_sys_info_read (SYS_INFO_2_READ, index);
+ − if (ncell_ext EQ 0)
+ − att_set_sys_info_read (SYS_INFO_2BIS_READ, index);
+ − /*
+ − * forward new neighbour cell list to layer 1 if
+ − * changes shall be indicated
+ − */
+ − if (modified AND indicate_changes)
+ − {
+ − att_code_mph_ncell_req (index);
+ − }
+ −
+ − /*
+ − * copy ncc permitted field
+ − */
+ − cd->ncc_permitted = sys_info_2->ncc_permit;
+ − break;
+ − }
+ − att_copy_rach_parameter (index, &sys_info_2->rach_ctrl, indicate_changes);
+ −
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_sys_info_2ter_par |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : copy the relevant system information type 2ter parameter after
+ − reception in idle mode or during cell selection or cell
+ − reselection.
+ −
+ − */
+ − GLOBAL void att_copy_sys_info_2ter_par (UBYTE index,
+ − T_D_SYS_INFO_2TER *sys_info_2ter,
+ − T_LIST *new_2ter_list,
+ − UBYTE indicate_changes)
+ − {
+ −
+ − /* Implements RR Clone findings #3 */
+ −
+ − TRACE_FUNCTION ("att_copy_sys_info_2ter_par()");
+ −
+ − /* Implements RR Clone findings #8 */
+ − att_copy_sys_info_2bis_2ter_par(index, SI_TYPE_2TER, new_2ter_list,
+ − &sys_info_2ter->neigh_cell_desc,
+ − indicate_changes);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_2ter_read |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : checks whether to wait for SI_2ter.
+ − */
+ −
+ − static void att_check_2ter_read (UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − if (rr_data->nc_data[index].c2_par.two_ter EQ FALSE)
+ − att_set_sys_info_read (SYS_INFO_2TER_READ, index);
+ −
+ − /*
+ − * only in dualband standards 2ter is taken in account
+ − * in single bands it is ignored.
+ − */
+ − switch (std)
+ − {
+ − case STD_900:
+ − case STD_EGSM:
+ − case STD_1800:
+ − case STD_850:
+ − case STD_1900:
+ − att_set_sys_info_read (SYS_INFO_2TER_READ, index);
+ − break;
+ − default:
+ − break;
+ − }
+ − }
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_sys_info_3_par |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : copy the relevant system information type 3 parameter after
+ − reception in idle mode or during cell selection or cell
+ − reselection.
+ − */
+ −
+ − /*
+ − * conversion from air interface coding for DTX usage to internal
+ − * notation
+ − */
+ − static const UBYTE dtx_bcch [3] = { DTX_USED, DTX_USED, DTX_NOT_USED };
+ −
+ −
+ − GLOBAL void att_copy_sys_info_3_par (UBYTE index,
+ − T_D_SYS_INFO_3 *sys_info_3,
+ − UBYTE indicate_changes)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_NC_DATA * rrd = &rr_data->nc_data[index];
+ − UBYTE lai_changed = FALSE;
+ − UBYTE control_changed_mm = FALSE;
+ − UBYTE control_changed_l1 = FALSE;
+ − TRACE_FUNCTION ("att_copy_sys_info_3_par()");
+ −
+ − #if defined(_SIMULATION_)
+ − TRACE_EVENT_WIN_P7 ( "MCC o/i[%d]= %x%x%x/%x%x%x",
+ − ((int)rrd->arfcn),
+ − rrd->lai.mcc[0],
+ − rrd->lai.mcc[1],
+ − rrd->lai.mcc[2],
+ − sys_info_3->loc_area_ident.mcc[0],
+ − sys_info_3->loc_area_ident.mcc[1],
+ − sys_info_3->loc_area_ident.mcc[2]);
+ − TRACE_EVENT_WIN_P7 ( "MNC o/i[%d]= %x%x%x/%x%x%x",
+ − ((int)rrd->arfcn),
+ − rrd->lai.mnc[0],
+ − rrd->lai.mnc[1],
+ − rrd->lai.mnc[2],
+ − sys_info_3->loc_area_ident.mnc[0],
+ − sys_info_3->loc_area_ident.mnc[1],
+ − sys_info_3->loc_area_ident.mnc[2]);
+ − #endif
+ −
+ − if (index EQ SC_INDEX)
+ − {
+ − /*
+ − * compare only for serving cell
+ − *
+ − * check location area information and cell id
+ − */
+ − if (rrd->cell_id NEQ sys_info_3->cell_ident)
+ − lai_changed = TRUE;
+ − rrd->cell_id = sys_info_3->cell_ident;
+ −
+ − if (!dat_plmn_equal_req (sys_info_3->loc_area_ident.mcc,
+ − sys_info_3->loc_area_ident.mnc,
+ − rrd->lai.mcc,
+ − rrd->lai.mnc))
+ − lai_changed = TRUE;
+ − if (rrd->lai.lac NEQ sys_info_3->loc_area_ident.lac)
+ − lai_changed = TRUE;
+ − #if defined(_SIMULATION_)
+ − TRACE_EVENT_WIN_P3 ( "LAI[%d] changed=%d indicate_changes=%d",
+ − rr_data->nc_data[index].arfcn, lai_changed, indicate_changes);
+ − #endif
+ −
+ − if (lai_changed AND indicate_changes)
+ − {
+ − /*
+ − * If a change of location area code is detected,
+ − * signal this as a cell reselection to MM to force
+ − * a location updating.
+ − */
+ − att_reset_old_lai_rac();
+ − memcpy (&rrd->lai, &sys_info_3->loc_area_ident,
+ − sizeof (T_loc_area_ident));
+ − att_code_rr_act_ind ();
+ − #ifdef GPRS
+ − if (att_gprs_is_avail())
+ − {
+ − /*XY: inform GRR, and don't wait for CR_RSP */
+ − att_rrgrr_cr_ind(CR_ABNORMAL);
+ − /*
+ − * we are in the process of acquiring SIs because GRR
+ − * has requested it. We have to set the sys_info_read bitmap
+ − * to complete. in order to send a GPRS_SI13_IND to GRR below.
+ − * The missing SIs which RR requested will still arrive from ALR
+ − * so it does not matter that we reset the bitmap now.
+ − */
+ − att_set_sys_info_read(ALL_SYS_INFO_READ, SC_INDEX);
+ − rr_data->gprs_data.start_proc = START_PROC_NOTHING;
+ − att_signal_gprs_support();
+ − }
+ − #endif
+ − }
+ −
+ − /*
+ − * check imsi attach status and
+ − * periodic location update time
+ − */
+ − if ((rrd->control_descr.att NEQ sys_info_3->ctrl_chan_desc.att
+ − OR
+ − rrd->control_descr.t3212 NEQ sys_info_3->ctrl_chan_desc.t3212)
+ − AND
+ − indicate_changes)
+ − control_changed_mm = TRUE;
+ −
+ − /*
+ − * check control channel description
+ − */
+ − if (rrd->control_descr.bs_ag_blks_res NEQ
+ − sys_info_3->ctrl_chan_desc.bs_ag_blks_res)
+ − control_changed_l1 = TRUE;
+ − if (rrd->control_descr.ccch_conf NEQ
+ − sys_info_3->ctrl_chan_desc.ccch_conf)
+ − control_changed_l1 = TRUE;
+ − if (rrd->control_descr.bs_pa_mfrms NEQ
+ − sys_info_3->ctrl_chan_desc.bs_pa_mfrms)
+ − control_changed_l1 = TRUE;
+ −
+ − /*
+ − * check cell selection parameters
+ − */
+ − if (rrd->select_para.ms_txpwr_max_cch NEQ
+ − sys_info_3->cell_select.ms_txpwr_max_cch)
+ − {
+ − control_changed_l1 = TRUE;
+ − TRACE_EVENT_P2("txpwr_max_cch changed from %d -> %d in SI3", rrd->select_para.ms_txpwr_max_cch,
+ − sys_info_3->cell_select.ms_txpwr_max_cch);
+ − }
+ − }
+ −
+ − /*
+ − * store location information and control channel description
+ − */
+ − rrd->cell_id = sys_info_3->cell_ident;
+ − #ifdef REL99
+ − rrd->mscr_flag = sys_info_3->ctrl_chan_desc.mscr;
+ − #endif
+ − memcpy (&rrd->lai, &sys_info_3->loc_area_ident,
+ − sizeof (T_loc_area_ident));
+ − memcpy (&rrd->control_descr, &sys_info_3->ctrl_chan_desc,
+ − sizeof (T_ctrl_chan_desc));
+ −
+ − if (control_changed_mm)
+ − {
+ − /*
+ − * indicate changed imsi attach status or
+ − * changed periodic location updating time to MM
+ − */
+ − PALLOC (sync, RR_SYNC_IND);
+ −
+ − att_code_prr_mm_info (&sync->mm_info);
+ − sync->ciph = NOT_PRESENT_8BIT;
+ − sync->chm.ch_mode = NOT_PRESENT_8BIT;
+ − sync->synccs = SYNCCS_SYS_INFO_CHANGE;
+ − sync->bcch_info.v_bcch = FALSE;
+ − PSENDX (MM, sync);
+ − }
+ −
+ − /*
+ − * store the system information type 3 message parameter.
+ − */
+ − switch (index)
+ − {
+ − case SC_INDEX:
+ − rr_data->sc_data.cd.cell_options.pow_ctrl =
+ − sys_info_3->cell_opt_bcch.pow_ctrl;
+ − rr_data->sc_data.cd.cell_options.rlt =
+ − sys_info_3->cell_opt_bcch.rlt;
+ − rr_data->sc_data.cd.dtx =
+ − dtx_bcch [sys_info_3->cell_opt_bcch.dtx_b];
+ − rr_data->sc_data.cd.dtx_half =
+ − rr_data->sc_data.cd.dtx_full =
+ − rr_data->sc_data.cd.dtx;
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − /*The SI-2quater information that we store is useful for monitoring appropriate
+ − BCCH periodically: no need to change status in re-acquisition*/
+ − if ( sys_info_3->si3_rest_oct.v_si2quater_ind )
+ − {
+ − if ( rr_data->sc_data.cd.si2quater_status EQ SI2QUATER_ABSENT)
+ − rr_data->sc_data.cd.si2quater_status = SI2QUATER_CONFIGURE;
+ − rr_data->sc_data.cd.si2quater_pos = sys_info_3->si3_rest_oct.si2quater_ind.si2quater_pos;
+ − }
+ − #endif
+ − break;
+ −
+ − case CR_INDEX:
+ − rr_data->cr_data.cd.cell_options.pow_ctrl =
+ − sys_info_3->cell_opt_bcch.pow_ctrl;
+ − rr_data->cr_data.cd.cell_options.rlt =
+ − sys_info_3->cell_opt_bcch.rlt;
+ − rr_data->cr_data.cd.dtx =
+ − dtx_bcch [sys_info_3->cell_opt_bcch.dtx_b];
+ − rr_data->cr_data.cd.dtx_half =
+ − rr_data->cr_data.cd.dtx_full =
+ − rr_data->cr_data.cd.dtx;
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − /*The SI-2quater information that we store is used only if this cell is selected
+ − In case of serving cell, this information should be useful for monitoring appropriate
+ − BCCH periodically*/
+ − rr_data->cr_data.cd.si2quater_status = SI2QUATER_ABSENT;
+ − if ( sys_info_3->si3_rest_oct.v_si2quater_ind )
+ − {
+ − rr_data->cr_data.cd.si2quater_status = SI2QUATER_CONFIGURE;
+ − rr_data->cr_data.cd.si2quater_pos = sys_info_3->si3_rest_oct.si2quater_ind.si2quater_pos;
+ − }
+ − #endif
+ − break;
+ − }
+ −
+ − memcpy (&rrd->select_para, &sys_info_3->cell_select,
+ − sizeof (T_cell_select));
+ − att_copy_rach_parameter (index, &sys_info_3->rach_ctrl,
+ − indicate_changes);
+ −
+ −
+ − att_copy_c2_parameter_si3 (index, &sys_info_3->si3_rest_oct);
+ −
+ − att_set_sys_info_read (SYS_INFO_3_READ, index);
+ − att_check_2ter_read (index);
+ −
+ − if (control_changed_l1 AND indicate_changes)
+ − {
+ − /*
+ − * forward changed control channel description
+ − * to layer 1
+ − */
+ − att_build_idle_req (SC_INDEX, MODE_SYS_INFO_CHANGE);
+ − #ifdef REL99
+ − att_config_cbch ();
+ − #else
+ − att_build_cbch();
+ − #endif
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_sys_info_4_par |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : copy the relevant system information type 4 parameter after
+ − reception in idle mode or during cell selection or cell
+ − reselection.
+ − */
+ −
+ − GLOBAL void att_copy_sys_info_4_par (UBYTE index,
+ − T_D_SYS_INFO_4 *sys_info_4,
+ − UBYTE indicate_changes)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_NC_DATA * rrd = &rr_data->nc_data[index];
+ − UBYTE lai_changed = FALSE;
+ − UBYTE control_changed_l1 = FALSE;
+ − T_CELL_DATA * cd;
+ −
+ − TRACE_FUNCTION ("att_copy_sys_info_4_par()");
+ −
+ − switch (index)
+ − {
+ − case SC_INDEX:
+ − case CR_INDEX:
+ − /*
+ − * set pointer to data
+ − */
+ − if (index EQ SC_INDEX)
+ − cd = &rr_data->sc_data.cd;
+ − else
+ − cd = &rr_data->cr_data.cd;
+ −
+ − if (index EQ SC_INDEX)
+ − {
+ − /*
+ − * compare only for serving cell
+ − *
+ − * check location information
+ − */
+ − if (!dat_plmn_equal_req (sys_info_4->loc_area_ident.mcc,
+ − sys_info_4->loc_area_ident.mnc,
+ − rrd->lai.mcc,
+ − rrd->lai.mnc))
+ − lai_changed = TRUE;
+ − if (rrd->lai.lac NEQ sys_info_4->loc_area_ident.lac)
+ − lai_changed = TRUE;
+ −
+ − if (lai_changed AND indicate_changes)
+ − {
+ − /*
+ − * If a change of location area code is detected,
+ − * signal this as a cell reselection to MM to force
+ − * a location updating.
+ − */
+ − att_reset_old_lai_rac();
+ − memcpy (&rrd->lai, &sys_info_4->loc_area_ident,
+ − sizeof (T_loc_area_ident));
+ − att_code_rr_act_ind ();
+ − #ifdef GPRS
+ − if (att_gprs_is_avail())
+ − {
+ − /*XY: inform GRR, and don't wait for CR_RSP */
+ − att_rrgrr_cr_ind(CR_ABNORMAL);
+ − /*
+ − * we are in the process of acquiring SIs because GRR
+ − * has requested it. We have to set the sys_info_read bitmap
+ − * to complete. in order to send a GPRS_SI13_IND to GRR below.
+ − * The missing SIs which RR requested will still arrive from ALR
+ − * so it does not matter that we reset the bitmap now.
+ − */
+ − att_set_sys_info_read(ALL_SYS_INFO_READ, SC_INDEX);
+ − rr_data->gprs_data.start_proc = START_PROC_NOTHING;
+ − att_signal_gprs_support();
+ − }
+ − #endif
+ − }
+ −
+ − /*
+ − * check cell selection parameters
+ − */
+ − if (rrd->select_para.ms_txpwr_max_cch NEQ
+ − sys_info_4->cell_select.ms_txpwr_max_cch)
+ − {
+ − control_changed_l1 = TRUE;
+ − TRACE_EVENT_P2("txpwr_max_cch changed from %d -> %d in SI4", rrd->select_para.ms_txpwr_max_cch,
+ − sys_info_4->cell_select.ms_txpwr_max_cch);
+ − }
+ − }
+ −
+ − if (sys_info_4->v_chan_desc)
+ − {
+ − /*
+ − * the message contains a CBCH channel description.
+ − */
+ − if(memcmp(&cd->cbch_chan_desc,&sys_info_4->chan_desc, sizeof (T_chan_desc)) NEQ 0)
+ − {
+ − memcpy (&cd->cbch_chan_desc, &sys_info_4->chan_desc, sizeof (T_chan_desc));
+ − cd->cbch_chan_desc_avail = WITH_CONTENT;
+ − att_bits_to_byte (cd->cbch_mob_alloc,
+ − sys_info_4->mob_alloc.c_mac,
+ − sys_info_4->mob_alloc.mac);
+ − if( (index EQ SC_INDEX) AND ( GET_STATE(STATE_ATT) EQ ATT_IDLE))
+ − #ifdef REL99
+ − att_config_cbch();
+ − #else
+ − att_build_cbch();
+ − #endif
+ − }
+ − }
+ − else
+ − {
+ − memset (&cd->cbch_chan_desc, 0, sizeof (T_chan_desc));
+ − if( cd->cbch_chan_desc_avail NEQ NO_CONTENT)
+ − {
+ − cd->cbch_chan_desc_avail = NO_CONTENT;
+ − if( (index EQ SC_INDEX) AND ( GET_STATE(STATE_ATT) EQ ATT_IDLE))
+ − #ifdef REL99
+ − att_config_cbch();
+ − #else
+ − att_build_cbch();
+ − #endif
+ − }
+ − }
+ −
+ − /*
+ − * set system information bits
+ − */
+ − att_set_sys_info_read (SYS_INFO_4_READ, index);
+ − break;
+ − default:
+ − break;
+ − }
+ −
+ − /*
+ − * copy the parameters
+ − */
+ − rrd->v_acs = sys_info_4->cell_select.acs; /* use of sys info 7&8 */
+ − rrd->lai = sys_info_4->loc_area_ident;
+ − rrd->select_para = sys_info_4->cell_select;
+ − att_copy_rach_parameter (index, &sys_info_4->rach_ctrl, indicate_changes);
+ −
+ − if (control_changed_l1 AND indicate_changes)
+ − {
+ − /*
+ − * forward changed control channel description
+ − * to layer 1
+ − */
+ − att_build_idle_req (SC_INDEX, MODE_SYS_INFO_CHANGE);
+ − att_build_cbch ();
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_sys_info_7_8_par |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : copy the relevant system information type 7 and 8
+ − parameter after reception in idle mode for neighbourcells.
+ − The messages are only expected if system information
+ − message 4 does not contain the C2 parameter due to a big
+ − CBCH channel description and system information
+ − message 3 is hidden by the own paging block.
+ −
+ − */
+ −
+ − GLOBAL void att_copy_sys_info_7_8_par (UBYTE index,
+ − T_D_SYS_INFO_8 *sys_info_8)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_copy_sys_info_7_8_par()");
+ −
+ − /*
+ − * system info 4 does not contain enough information
+ − */
+ − if (rr_data->nc_data[index].v_acs)
+ − att_copy_c2_parameter_si4 (index, (T_si4_rest_oct *)&sys_info_8->si8_rest_oct);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_c2_parameter |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Rest Octets of System Information Type 3/4/7/8 are decoded
+ − to get the C2 parameter for cell reselection.
+ − */
+ −
+ − static void att_copy_c2_parameter (T_C2_PARAMETER *c2_par,
+ − T_si3_rest_oct *rest_oct )
+ − {
+ − TRACE_FUNCTION ("att_copy_c2_parameter()");
+ −
+ − /* Optional selection parameters */
+ − if(rest_oct->v_opt_sel_par)
+ − {
+ − c2_par->param_ind = TRUE;
+ − c2_par->cbq = rest_oct->opt_sel_par.cell_bar_qual;
+ − c2_par->cell_reselect_offset = rest_oct->opt_sel_par.cell_resel_offs;
+ − c2_par->temp_offset = rest_oct->opt_sel_par.temp_offs;
+ − c2_par->penalty_time = rest_oct->opt_sel_par.penalty_time;
+ − }
+ − else
+ − {
+ − c2_par->param_ind = FALSE;
+ − c2_par->cbq = 0;
+ − c2_par->cell_reselect_offset = 0;
+ − c2_par->temp_offset = 0;
+ − c2_par->penalty_time = 0;
+ − }
+ −
+ − /* Optional Power Offset */
+ − if(rest_oct->v_pow_offs)
+ − {
+ − c2_par->power_off_ind = TRUE;
+ − c2_par->power_off = rest_oct->pow_offs;
+ − }
+ − else
+ − {
+ − c2_par->power_off_ind = FALSE;
+ − c2_par->power_off = 0;
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_c2_parameter_si3 |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : rest octets of system information 3 are decoded
+ − to get the C2 parameter for cell reselection.
+ − */
+ −
+ − static void att_copy_c2_parameter_si3 (UBYTE index,
+ − T_si3_rest_oct *rest_oct)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_NC_DATA * nc = &rr_data->nc_data[index];
+ − T_C2_PARAMETER * c2_par = &nc->c2_par;
+ −
+ − TRACE_FUNCTION ("att_copy_c2_parameter_si3()");
+ −
+ − att_copy_c2_parameter (c2_par, rest_oct);
+ −
+ − if(index EQ SC_INDEX AND
+ − c2_par->two_ter NEQ rest_oct->v_si2ter_ind)
+ − {
+ − PALLOC (mph_sync_req, MPH_SYNC_REQ);
+ − mph_sync_req->cs = CS_SYS_INFO_2TER_SUPPORT_CHANGED;
+ − PSENDX (PL, mph_sync_req);
+ − }
+ − /* System Information 2ter Indicator */
+ − if (rest_oct->v_si2ter_ind)
+ − {
+ − c2_par->two_ter = TRUE;
+ − }
+ − else
+ − {
+ − c2_par->two_ter = FALSE;
+ − }
+ −
+ − /* Early Classmark Sending Control */
+ − if (rest_oct->v_es_ind_tag)
+ − {
+ − c2_par->ecsc = TRUE;
+ − }
+ − else
+ − {
+ − c2_par->ecsc = FALSE;
+ − }
+ −
+ − #ifdef GPRS
+ − if (rest_oct->v_gprs_indic)
+ − nc->rac = rest_oct->gprs_indic.ra_color;
+ − else
+ − nc->rac = NOT_PRESENT_8BIT;
+ − #endif
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_c2_parameter_si4 |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Rrest Octets of System Information Type 4 are decoded
+ − to get the C2 parameter for cell reselection.
+ −
+ − */
+ −
+ − GLOBAL void att_copy_c2_parameter_si4 (UBYTE index,
+ − T_si4_rest_oct *rest_oct)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_NC_DATA * nc = &rr_data->nc_data[index];
+ − T_C2_PARAMETER * c2_par = &nc->c2_par;
+ −
+ − TRACE_FUNCTION ("att_copy_c2_parameter_si4()");
+ −
+ − att_copy_c2_parameter (c2_par, (T_si3_rest_oct *)rest_oct);
+ −
+ − #ifdef GPRS
+ − if (rest_oct->v_gprs_indic)
+ − nc->rac = rest_oct->gprs_indic.ra_color;
+ − else
+ − nc->rac = NOT_PRESENT_8BIT;
+ − #endif
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_get_highest_c2_index |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : If a cell reselection shall be performed, a ranking
+ − of the available candidates is done. Depending on the
+ − calculated C2 values a C2-Index is used (because more
+ − than one cell can have the same C2-value or the value
+ − changes during cell reselection). This function looks
+ − for the cell with the highest c2 index.
+ −
+ − */
+ −
+ − static UBYTE att_get_highest_c2_index (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE max_result = SC_INDEX;
+ − UBYTE next_result;
+ − UBYTE result;
+ − UBYTE i;
+ −
+ − TRACE_FUNCTION ("att_get_highest_c2_index()");
+ −
+ − switch (rr_data->sc_data.selection_type)
+ − {
+ − case BACK_FROM_DEDICATED:
+ − case BACK_FROM_DEDICATED_RLF:
+ − /*
+ − * prepare cell reselection list
+ − */
+ − for (i = 0; i < 7; i++)
+ − {
+ − rr_data->nc_data[i].c2_used = FALSE;
+ − if (rr_data->nc_data[i].bcch_status NEQ EMPTY)
+ − if (rr_data->nc_data[i].rxlev > rr_data->nc_data[max_result].rxlev)
+ − max_result = i;
+ − }
+ −
+ − if (rr_data->sc_data.selection_type EQ BACK_FROM_DEDICATED)
+ − {
+ − /*
+ − * In case of normal release start with serving cell
+ − */
+ − rr_data->nc_data[SC_INDEX].c2_used = TRUE;
+ − max_result = SC_INDEX;
+ − rr_data->ms_data.c2_tab[SC_INDEX] = MAX_C2;
+ − result = SC_INDEX;
+ − }
+ − else
+ − {
+ − /*
+ − * In case of call re-establishment start with highest fieldstrength
+ − */
+ − rr_data->nc_data[max_result].c2_used = TRUE;
+ − rr_data->ms_data.c2_tab[max_result] = MAX_C2;
+ − result = max_result;
+ − }
+ − /*
+ − * Fill with the rest of neighbourcells
+ − */
+ − while ((next_result = att_get_next_highest_rx ())
+ − NEQ NO_AVAILABLE)
+ − {
+ − rr_data->ms_data.c2_tab[next_result] =
+ − rr_data->ms_data.c2_tab[result] - 1;
+ − result = next_result;
+ − }
+ − break;
+ −
+ − case CELL_RESELECTION_CR:
+ − /*
+ − * Start with CR cell (from idle mode to another cell)
+ − */
+ − rr_data->nc_data[CR_INDEX].c2_used = TRUE;
+ − max_result = CR_INDEX;
+ − result = max_result;
+ − rr_data->ms_data.c2_tab[CR_INDEX] = MAX_C2;
+ − /*
+ − * Fill with the rest of neighbourcells
+ − */
+ − while ((next_result = att_get_next_highest_rx ())
+ − NEQ NO_AVAILABLE)
+ − {
+ − rr_data->ms_data.c2_tab[next_result] =
+ − rr_data->ms_data.c2_tab[result] - 1;
+ − result = next_result;
+ − }
+ − break;
+ −
+ − default:
+ − /*
+ − * Normal Cell reselection
+ − *
+ − * use a corrected C2 value for calculation,
+ − * because for cell reselection due to C2(NC) > C2(SC) a
+ − * threshold value must be taken in account if serving
+ − * and neighbourcell are members of different location areas.
+ − */
+ − att_calculate_c2 (SC_INDEX);
+ − rr_data->nc_data[SC_INDEX].c2_corr = rr_data->nc_data[SC_INDEX].c2;
+ −
+ − /*
+ − * Calculate the corrected C2 values for all available neighbourcells.
+ − */
+ − for (i = 0; i < 6; i++)
+ − {
+ − if (rr_data->nc_data[i].bcch_status EQ DECODED)
+ − {
+ − att_calculate_c2 (i);
+ − rr_data->nc_data[i].c2_corr = rr_data->nc_data[i].c2;
+ −
+ − /*
+ − * Take the cell reselection hysterese threshold value in
+ − * account, if there are different location areas,
+ − * a cell reselection due to C2(NC) > C2(SC)
+ − */
+ − #ifdef GPRS
+ − /*
+ − * check if the location area has changed
+ − * or the routing area if the scell supports GPRS
+ − */
+ − {
+ − BOOL la_changed, ra_changed;
+ −
+ − if (!dat_plmn_equal_req (rr_data->nc_data[i].lai.mcc,
+ − rr_data->nc_data[i].lai.mnc,
+ − rr_data->nc_data[SC_INDEX].lai.mcc,
+ − rr_data->nc_data[SC_INDEX].lai.mnc)
+ − OR
+ − rr_data->nc_data[i].lai.lac NEQ
+ − rr_data->nc_data[SC_INDEX].lai.lac)
+ − la_changed = TRUE;
+ − else
+ − la_changed = FALSE;
+ −
+ − if(att_gprs_is_avail())
+ − {
+ − if(rr_data->nc_data[i].rac NEQ
+ − rr_data->nc_data[SC_INDEX].rac)
+ − ra_changed = TRUE;
+ − else
+ − ra_changed = FALSE;
+ − }
+ − else
+ − {
+ − ra_changed = FALSE;
+ − }
+ −
+ − if(la_changed OR ra_changed OR rr_data->gprs_data.ready_state)
+ − #else
+ − if (!dat_plmn_equal_req (rr_data->nc_data[i].lai.mcc,
+ − rr_data->nc_data[i].lai.mnc,
+ − rr_data->nc_data[SC_INDEX].lai.mcc,
+ − rr_data->nc_data[SC_INDEX].lai.mnc)
+ − OR
+ − rr_data->nc_data[i].lai.lac NEQ
+ − rr_data->nc_data[SC_INDEX].lai.lac)
+ − #endif
+ − {
+ − if ((rr_data->ms_data.imsi_available) AND
+ − (rr_data->ms_data.rr_service NEQ LIMITED_SERVICE) AND
+ − (rr_data->sc_data.selection_type EQ CELL_RESELECTION_NC))
+ − {
+ − rr_data->nc_data[i].c2_corr -=
+ − 2*rr_data->nc_data[SC_INDEX].select_para.cell_resel_hyst;
+ − }
+ − }
+ − else
+ − {
+ − /*
+ − * in case of a cell reselection occurring within the previous 15 seconds
+ − * in which case the C2 value for the new cell shall exceed the C2 value
+ − * of the serving cell by at least 5 dB for a period of 5 seconds
+ − */
+ −
+ − if ( IS_TIMER_ACTIVE (T_NO_RESELECT) )
+ − {
+ − rr_data->nc_data[i].c2_corr -= 5;
+ − }
+ − else
+ − {
+ − /* If the C2(NC)=C2(SC), this will get mis-interpreted in the check below
+ − * c2_corr must be adjusted to make this condition false. The '=' in the '>='
+ − * condition below only applies when T_NO_RESELECT timer is active and
+ − * c2_corr has been decremented by 5 (above statement)
+ − */
+ − if( rr_data->nc_data[i].c2_corr EQ rr_data->nc_data[max_result].c2_corr)
+ − {
+ − rr_data->nc_data[i].c2_corr -= 1;
+ − }
+ − }
+ − }
+ − #ifdef GPRS
+ − }
+ − #endif
+ − }
+ − }
+ −
+ − /*
+ − * Now find the the highest one
+ − */
+ − for (i = 0; i < 7; i++)
+ − {
+ − rr_data->nc_data[i].c2_used = FALSE;
+ − if (rr_data->nc_data[i].bcch_status EQ DECODED)
+ − {
+ − if (rr_data->nc_data[i].c2_corr >=
+ − rr_data->nc_data[max_result].c2_corr)
+ − max_result = i;
+ − }
+ − }
+ −
+ − result = max_result;
+ −
+ − rr_data->nc_data[result].c2_used = TRUE;
+ − rr_data->ms_data.c2_tab[result] = MAX_C2;
+ −
+ − /*
+ − * make the ranking for the other cells.
+ − */
+ − while ((next_result = att_get_next_highest_c2_val ())
+ − NEQ NO_AVAILABLE)
+ − {
+ − rr_data->ms_data.c2_tab[next_result] =
+ − rr_data->ms_data.c2_tab[result] - 1;
+ − result = next_result;
+ − }
+ − break;
+ − }
+ −
+ − /*
+ − * return the index of the highest C2.
+ − */
+ − return max_result;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_get_next_highest_c2_idx|
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : During the ranking of the cell reselection list this
+ − function searches for the cell with the next highest c2 index.
+ −
+ − */
+ −
+ − static UBYTE att_get_next_highest_c2_idx (UBYTE old_index)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE new_index = NO_AVAILABLE;
+ − UBYTE i;
+ −
+ − TRACE_FUNCTION ("att_get_next_highest_c2_idx()");
+ −
+ − /*
+ − * depending on the cell reselection type
+ − */
+ − switch (rr_data->sc_data.selection_type)
+ − {
+ − case BACK_FROM_DEDICATED:
+ − case BACK_FROM_DEDICATED_RLF:
+ − /*
+ − * check all cells
+ − */
+ − for (i = 0; i <= SC_INDEX; i++)
+ − {
+ − /*
+ − * If cell is available and is not selected yet
+ − */
+ − if (rr_data->nc_data[i].bcch_status NEQ EMPTY AND
+ − rr_data->ms_data.c2_tab[i] NEQ MAX_C2)
+ − {
+ − /*
+ − * it is the first candidate, then select it
+ − */
+ − if (new_index EQ NO_AVAILABLE)
+ − new_index = i;
+ − else
+ − {
+ − /*
+ − * else take the one with the higher C2
+ − */
+ − if (rr_data->ms_data.c2_tab[i] >
+ − rr_data->ms_data.c2_tab[new_index])
+ − new_index = i;
+ − }
+ − }
+ − }
+ −
+ − /*
+ − * If a new cell is found, mark this cell as selected
+ − * for the next round
+ − */
+ − if (new_index NEQ NO_AVAILABLE)
+ − rr_data->ms_data.c2_tab[new_index] = MAX_C2;
+ − break;
+ −
+ − default:
+ − /*
+ − * all other cell reselections
+ − */
+ − for (i = 0; i <= SC_INDEX; i++)
+ − {
+ − if (rr_data->nc_data[i].bcch_status EQ DECODED)
+ − {
+ − if (old_index NEQ i)
+ − {
+ − if (new_index EQ NO_AVAILABLE)
+ − {
+ − if (rr_data->ms_data.c2_tab[i] <
+ − rr_data->ms_data.c2_tab[old_index])
+ − {
+ − new_index = i;
+ − }
+ − }
+ − else
+ − {
+ − if (rr_data->ms_data.c2_tab[i] >
+ − rr_data->ms_data.c2_tab[new_index] AND
+ − rr_data->ms_data.c2_tab[i] <
+ − rr_data->ms_data.c2_tab[old_index])
+ − {
+ − new_index = i;
+ − }
+ − }
+ − }
+ − }
+ − }
+ − break;
+ − }
+ − return new_index;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_get_next_highest_c2_val|
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : During the ranking of the cell reselection list this
+ − function searches for the cell with the next highest c2 value.
+ −
+ − */
+ −
+ − static UBYTE att_get_next_highest_c2_val (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE new_index = NO_AVAILABLE;
+ − UBYTE i;
+ −
+ − TRACE_FUNCTION ("att_get_next_highest_c2_val()");
+ −
+ − /*
+ − * look for all cells inclusive the serving cell
+ − */
+ − for (i = 0; i < 7; i++)
+ − {
+ − /*
+ − * the cell must have read the system infos to calculate c2.
+ − */
+ − if (rr_data->nc_data[i].bcch_status EQ DECODED)
+ − {
+ − /*
+ − * if not ranked yet
+ − */
+ − if (rr_data->nc_data[i].c2_used EQ FALSE)
+ − {
+ − /*
+ − * if no cell is selected yet,
+ − * then select this one.
+ − */
+ − if (new_index EQ NO_AVAILABLE)
+ − {
+ − new_index = i;
+ − }
+ − else
+ − {
+ − /*
+ − * the corrected c2 value must be higher
+ − * then the selected one.
+ − */
+ − if (rr_data->nc_data[i].c2_corr >
+ − rr_data->nc_data[new_index].c2_corr)
+ − {
+ − new_index = i;
+ − }
+ − }
+ − }
+ − }
+ − }
+ −
+ − /*
+ − * a cell has been selected, then mark this cell
+ − * to be not selected the next time.
+ − */
+ − if (new_index NEQ NO_AVAILABLE)
+ − {
+ − rr_data->nc_data[new_index].c2_used = TRUE;
+ − }
+ −
+ − /*
+ − * return the selected cell index
+ − */
+ − return new_index;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_get_next_highest_rx |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : During the ranking of the cell reselection list this
+ − function searches for the cell with the next highest rx value.
+ −
+ − */
+ −
+ − static UBYTE att_get_next_highest_rx (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE new_index = NO_AVAILABLE;
+ − UBYTE i;
+ −
+ − TRACE_FUNCTION ("att_get_next_highest_rx()");
+ −
+ − /*
+ − * for all neighbourcells and the serving cell
+ − */
+ − for (i = 0; i < 7; i++)
+ − {
+ − /*
+ − * if something is stored for this index
+ − * that means at least MS is synchronized.
+ − */
+ − if (rr_data->nc_data[i].bcch_status NEQ EMPTY)
+ − {
+ − /*
+ − * the cell has not been ranked
+ − */
+ − if (rr_data->nc_data[i].c2_used EQ FALSE)
+ − {
+ − /*
+ − * if it is the first cell in this attempt,
+ − * then select it.
+ − */
+ − if (new_index EQ NO_AVAILABLE)
+ − {
+ − new_index = i;
+ − }
+ − else
+ − {
+ − /*
+ − * to select the cell it must have a higher
+ − * fieldstrength
+ − */
+ − if (rr_data->nc_data[i].rxlev >
+ − rr_data->nc_data[new_index].rxlev)
+ − {
+ − new_index = i;
+ − }
+ − }
+ − }
+ − }
+ − }
+ −
+ − /*
+ − * A cell has been selected, then mark it as
+ − * ranked to avoid selection in the next attempt.
+ − */
+ − if (new_index NEQ NO_AVAILABLE)
+ − {
+ − rr_data->nc_data[new_index].c2_used = TRUE;
+ − }
+ −
+ − /*
+ − * return the index of the selected cell.
+ − */
+ − return new_index;
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_handle_rr_act_req |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : MM requests with the primitive RR_ACTIVATE_REQ one of the
+ − following procedures:
+ −
+ − 1. Full Service - search for a specific PLMN
+ − 2. Limited Service - search for a cell for limited service
+ − 3. PLMN Search - request the PLMN available list.
+ −
+ − */
+ −
+ − GLOBAL void att_handle_rr_act_req (UBYTE initial_act,
+ − T_RR_ACTIVATE_REQ *rr_activate_req)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE count = 0;
+ − TRACE_FUNCTION ("att_handle_rr_act_req()");
+ −
+ − /*
+ − * stop time control for an ongoing cell reselection
+ − */
+ − TIMERSTOP (T_RESELECT);
+ − TIMERSTOP (TABORT);
+ −
+ − rr_data->net_lost = FALSE;
+ −
+ − if (initial_act)
+ − {
+ − /*
+ − * read ms data from e2prom
+ − */
+ − att_clear_parallel_search();
+ − rr_csf_ms_cap ();
+ − rr_csf_check_rfcap (FALSE);
+ − rr_csf_read_imei (&rr_data->ms_data.imei);
+ − rr_data->ms_data.rr_service = NO_SERVICE; /* set initial to no service */
+ − }
+ −
+ − for(count=0;count<MAX_MCC_SHIELD;count++)
+ − {
+ − TRACE_EVENT_P3 ( "Current software shielded MCC are =%d%d%d",
+ − rr_data->dyn_config.mcc_shield.mcc[count][0],
+ − rr_data->dyn_config.mcc_shield.mcc[count][1],
+ − rr_data->dyn_config.mcc_shield.mcc[count][2]);
+ − }
+ −
+ − /*
+ − * store EPLMN list, if available, for use in cell selection/reselection.
+ − */
+ − att_copy_eplmn_list(&rr_activate_req->eq_plmn_list);
+ −
+ − /*
+ − * store operation mode data
+ − */
+ − rr_data->ms_data.operation_mode =
+ − (rr_activate_req->op.ts << 7) +
+ − (rr_activate_req->op.m << 6) +
+ − (rr_activate_req->op.sim_ins << 5);
+ −
+ − /*
+ − * some optimisations must be removed for the FTA campaign.
+ − */
+ − if (dat_test_sim_available() EQ TRUE)
+ − test_house = TRUE;
+ −
+ − att_set_func (rr_activate_req->op.func);
+ − /*
+ − * start cell selection procedure
+ − */
+ − rr_data->ms_data.req_mm_service = att_get_func ();
+ −
+ − TRACE_EVENT_P6 ("RR_ACTIVATE_REQ: %s func op=%s %scurrent=%s rMM=%s RRs=%s",
+ − initial_act ? "initial" : "",
+ − _rr_str_FUNC[rr_activate_req->op.func],
+ − _rr_str_PARFUNC[rr_data->ms_data.parallel_net_plmn_search_type],
+ − _rr_str_FUNC[rr_data->ms_data.current_plmn_search_type],
+ − _rr_str_FUNC[rr_data->ms_data.req_mm_service],
+ − _rr_str_SERVICE[rr_data->ms_data.rr_service]);
+ −
+ − switch (rr_data->ms_data.req_mm_service)
+ − {
+ − case FUNC_LIM_SERV_ST_SRCH:
+ − #ifdef GPRS
+ − att_set_gprs_indication(rr_activate_req->gprs_indication);
+ − #endif
+ − rr_data->c1_offset = C1_OFFSET;
+ −
+ − att_begin_cs(FUNC_LIM_SERV_ST_SRCH);
+ − break;
+ −
+ − case FUNC_NET_SRCH_BY_MMI:
+ − /*
+ − * clear found_channel parameter from previous search
+ − */
+ − switch(GET_STATE(STATE_ATT))
+ − {
+ − case ATT_IDLE:
+ − case ATT_CON_EST:
+ − /* Cell Selection Improvements-LLD section:4.1.3.9 */
+ − att_start_cell_selection(MM_ORIGINATED, CS_PARALLEL,FULL_SEARCH_MODE);
+ − break;
+ −
+ − default:
+ − att_begin_cs(FUNC_NET_SRCH_BY_MMI);
+ − break;
+ − }
+ − EM_NET_SEARCH_STARTED;
+ − break;
+ −
+ − case FUNC_PLMN_SRCH:
+ − /*
+ − * copy full service parameter from SIM
+ − */
+ − #ifdef GPRS
+ − att_set_gprs_indication (rr_activate_req->gprs_indication);
+ − #endif
+ − if (rr_data->ms_data.rr_service NEQ NO_SERVICE)
+ − cs_clear_attributes (CHECKED_FLAG, rr_data->nc_data[SC_INDEX].arfcn);
+ −
+ − /* Implements Measure#32: Row 52,53 */
+ − att_print_mcc_mnc (NOT_PRESENT_16BIT, rr_data->ms_data.plmn.mcc,
+ − rr_data->ms_data.plmn.mnc, S2I_STRING("old_req_plmn"));
+ − att_print_mcc_mnc (NOT_PRESENT_16BIT, rr_activate_req->plmn.mcc,
+ − rr_activate_req->plmn.mnc, S2I_STRING("new_req_plmn"));
+ −
+ − memcpy (&rr_data->ms_data.plmn, &rr_activate_req->plmn, sizeof (T_plmn));
+ − rr_data->ms_data.cksn = rr_activate_req->cksn;
+ − memcpy (rr_data->ms_data.new_kc, &rr_activate_req->kcv.kc, KC_STRING_SIZE);
+ − rr_data->ms_data.access_classes = rr_activate_req->accc;
+ −
+ − #if defined(_SIMULATION_)
+ − /* Implements Measure#32: Row 52 */
+ − att_print_mcc_mnc(rr_data->nc_data[SC_INDEX].arfcn,
+ − rr_data->ms_data.plmn.mcc, rr_data->ms_data.plmn.mnc, S2I_STRING("req"));
+ − #endif
+ −
+ − /*
+ − * copy IMSI if available
+ − */
+ − if (rr_activate_req->imsi_struct.id_type EQ TYPE_IMSI)
+ − {
+ − memset (&rr_data->ms_data.imsi, 0, sizeof (rr_data->ms_data.imsi));
+ − rr_data->ms_data.imsi_available = TRUE;
+ − rr_data->ms_data.imsi.ident_type = TYPE_IMSI;
+ − rr_data->ms_data.imsi.v_ident_dig = TRUE;
+ − rr_data->ms_data.imsi.c_ident_dig =
+ − att_calculate_digits (rr_activate_req->imsi_struct.id);
+ − rr_data->ms_data.imsi.odd_even =
+ − rr_data->ms_data.imsi.c_ident_dig & 1;
+ − memcpy (rr_data->ms_data.imsi.ident_dig, rr_activate_req->imsi_struct.id, 16);
+ − }
+ − else
+ − rr_data->ms_data.imsi_available = FALSE;
+ −
+ − /*
+ − * copy TMSI if available
+ − */
+ − if (rr_activate_req->tmsi_struct.id_type EQ TYPE_TMSI)
+ − {
+ − rr_data->ms_data.tmsi_available = TRUE;
+ − rr_data->ms_data.tmsi_binary = rr_activate_req->tmsi_struct.tmsi_dig;
+ − }
+ − else
+ − rr_data->ms_data.tmsi_available = FALSE;
+ −
+ − if (dat_test_sim_available())
+ − rr_data->c1_offset = 0;
+ − else
+ − rr_data->c1_offset = C1_OFFSET;
+ −
+ − /* CSI-LLD section:4.1.1.11
+ − * First FUNC_PLMN_SEARCH
+ − */
+ − if(rr_data->cs_data.initial_plmn_search EQ INITIAL_PLMN_SEARCH_NOT_ACTIVE)
+ − {
+ − rr_data->cs_data.initial_plmn_search = INITIAL_PLMN_SEARCH_ACTIVE;
+ − }
+ −
+ − cs_set_bcch_info(&rr_activate_req->bcch_info);
+ −
+ − att_begin_cs(FUNC_PLMN_SRCH);
+ − EM_PLMN_SRCH_STARTED;
+ − break;
+ −
+ − case FUNC_ST_PWR_SCAN:
+ − SET_STATE(STATE_ATT, ATT_CS_INIT );
+ − att_start_cell_selection(MM_ORIGINATED, CS_NOT_PARALLEL,FULL_SEARCH_MODE);
+ − break;
+ − }
+ −
+ − PFREE (rr_activate_req);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_begin_cs |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function initializes a cell selection or starts BCCH
+ − reading depending on the ATT state on an RR_ACTIVATE_REQ
+ − */
+ −
+ − static void att_begin_cs (UBYTE req_mm_service)
+ − {
+ − GET_INSTANCE_DATA;
+ − if(GET_STATE(STATE_ATT) EQ ATT_CS_INIT )
+ − {
+ − SET_STATE(STATE_ATT, ATT_CS1 ); /*Boot Time*/
+ −
+ − /* Boot Time Performance Enhancement:
+ − * Start the TABORT timer on receiving the second activate_req
+ − */
+ − tstart_tabort(TABORT_VALUE);
+ −
+ − /* Invalidate whitelist if its PLMN ID does not match
+ − * with the requested PLMN
+ − */
+ − if((rr_data->ms_data.req_mm_service EQ FUNC_PLMN_SRCH) AND
+ − (!dat_plmn_equal_req(rr_data->cs_data.white_list.last_sc_lac.mcc,
+ − rr_data->cs_data.white_list.last_sc_lac.mnc,
+ − rr_data->ms_data.plmn.mcc,
+ − rr_data->ms_data.plmn.mnc)))
+ − {
+ − rr_data->cs_data.white_list.region = NOT_PRESENT_8BIT;
+ − }
+ − if( srv_check_stored_prim(MPH_POWER_CNF))
+ − {
+ − srv_use_stored_prim();
+ − }
+ − /*
+ − * reset the time for net search to finish
+ − * a PLMN available search during 60 seconds.
+ − */
+ − if( req_mm_service EQ FUNC_NET_SRCH_BY_MMI)
+ − TIMERSTART(T_PLMN_SEARCH, T_PLMN_SEARCH_VALUE);
+ − }
+ − else
+ − {
+ − /* Cell Selection Improvements-LLD section:4.1.3.9 */
+ − att_start_cell_selection(MM_ORIGINATED, CS_NOT_PARALLEL,FULL_SEARCH_MODE);
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_increase_power_on_time |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : A requirement in GSM 5.08 indicates that the list of
+ − forbidden location areas must be cleared all 12 hours.
+ − Therefore this counters power_on_low and power_on_high
+ − are used. They are triggered by the periodic measurement
+ − reports in idle mode.
+ −
+ − */
+ −
+ − GLOBAL void att_increase_power_on_time (USHORT inc)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_increase_power_on_time()");
+ −
+ − /*
+ − * increment power_on_low
+ − */
+ − rr_data->power_on_low += inc;
+ − if (rr_data->power_on_low >= 60)
+ − {
+ − /* if it is more then 60 seconds,
+ − * increment power_on_high = minutes
+ − */
+ − rr_data->power_on_low = 0;
+ − rr_data->power_on_high++;
+ − if (rr_data->power_on_high >= 720)
+ − {
+ − /*
+ − * after 12 hours = 720 minutes clear all forbidden location area lists
+ − */
+ − rr_data->power_on_low = 0;
+ − rr_data->power_on_high = 0;
+ − att_clear_forb_list (FORBIDDEN_LIST_NORMAL);
+ − att_clear_forb_list (FORBIDDEN_LIST_ROAMING);
+ − }
+ − }
+ − }
+ −
+ −
+ − #if defined (TI_PS_FF_RTD) AND defined (REL99)
+ − LOCAL void att_init_rtd_data(T_rr_enh_para *p_cur)
+ − {
+ − UBYTE j,k;
+ − TRACE_FUNCTION ("att_init_rtd_data");
+ −
+ − for(j = 0;j < MAX_NR_OF_NCELL; j++ )
+ − {
+ − p_cur->enh_para.enh_cell_list[j].v_rtd = FALSE;
+ − for(k = 0;k < MAX_NUM_OF_RTD_VALUES; k++)
+ − p_cur->enh_para.enh_cell_list[j].rtd[k]= RTD_NOT_AVAILABLE;
+ − p_cur->enh_para.enh_cell_list[j].c_rtd = 0;
+ − }/*for*/
+ − return;
+ − }
+ − #endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_init_cell_selection |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function initializes some variables needed for control
+ − of cell selection and cell reselection.
+ −
+ − */
+ −
+ − GLOBAL void att_init_cell_selection (UBYTE selection,
+ − BOOL initiator)
+ − {
+ − GET_INSTANCE_DATA;
+ − #if defined (TI_PS_FF_RTD) AND defined (REL99)
+ − T_rr_enh_para *p_cur = &rr_data->sc_data.emr_data_current;
+ − #endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */
+ −
+ − TRACE_FUNCTION ("att_init_cell_selection()");
+ −
+ − TRACE_EVENT_P2 ("init CELL_%sSELECTION %s_ORIGINATED",
+ − (selection EQ CELL_SELECTION) ? "" : "RE",
+ − (initiator EQ MM_ORIGINATED) ? "MM" : "RR" );
+ −
+ − /*
+ − * store type of selection and the initiator (RR or MM).
+ − */
+ − rr_data->sc_data.selection_type = selection;
+ − rr_data->sc_data.mm_started = initiator;
+ −
+ − /*
+ − * clear the list of the found PLMNs
+ − */
+ − rr_data->sc_data.found_entries = 0;
+ −
+ − /*
+ − * clear the bitmap indicating whether all system information
+ − * messages are read.
+ − */
+ − att_init_cr_data();
+ −
+ − #if defined (TI_PS_FF_RTD) AND defined (REL99)
+ − att_init_rtd_data(p_cur);
+ − #endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */
+ −
+ −
+ − /*
+ − * reset multiband parameter
+ − */
+ − rr_data->ncell_mb = 0;
+ −
+ − if (GET_STATE (STATE_ATT) EQ ATT_CS2 OR GET_STATE (STATE_ATT) EQ ATT_CS3)
+ − {
+ − rr_data->nc_data[SC_INDEX].lai.mcc[0] = NOT_PRESENT_8BIT;
+ − rr_data->nc_data[SC_INDEX].lai.mnc[0] = NOT_PRESENT_8BIT;
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_init_gsm_data |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Intialize all gsm specific variables of RR.
+ −
+ − */
+ −
+ − GLOBAL void att_init_gsm_data (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − SHORT i;
+ − UBYTE save;
+ − UBYTE gprs_ms, cmsp;
+ − BOOL ms_configured, cmsp_configured;
+ −
+ − #if defined (TI_PS_FF_RTD) AND defined (REL99)
+ − T_rr_enh_para *p_cur = &rr_data->sc_data.emr_data_current;
+ − #endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */
+ −
+ −
+ − TRACE_FUNCTION ("att_init_gsm_data()");
+ −
+ − save = rr_data->dyn_config.no_sys_time;
+ −
+ − /* save configured GPRS multislot class */
+ − ms_configured = rr_data->ms_data.multislot_class_configured;
+ − gprs_ms = rr_data->ms_data.rf_cap.rf_ms.gprs_ms_class;
+ − dat_emo_stop ( FALSE );
+ −
+ − /* save configured CMSP bit */
+ − cmsp_configured = rr_data->ms_data.cmsp_configured;
+ − cmsp = rr_data->ms_data.rf_cap.cmsp;
+ −
+ − /*
+ − * clar all RR data with the default value
+ − */
+ − memset (rr_data, 0, sizeof (T_RR_DATA));
+ −
+ − /*
+ − * some parameters have a different default value
+ − */
+ − rr_data->lup_rxlev = 63;
+ − rr_data->dyn_config.no_sys_time = save;
+ − rr_data->ms_data.cksn = CKSN_NOT_PRES;
+ −
+ − /* restore configured GPRS multislot class */
+ − if (ms_configured)
+ − {
+ − rr_data->ms_data.multislot_class_configured = TRUE;
+ − rr_data->ms_data.rf_cap.rf_ms.gprs_ms_class = gprs_ms;
+ − TRACE_EVENT_P1 ("'gprs_ms_class' configured to a value of %d", gprs_ms);
+ − }
+ −
+ − /* restore configured CMSP bit */
+ − if (cmsp_configured)
+ − {
+ − rr_data->ms_data.cmsp_configured = TRUE;
+ − rr_data->ms_data.rf_cap.cmsp= cmsp;
+ − TRACE_EVENT_P1 ("'cmsp' configured to a value of %d", cmsp);
+ − }
+ − #if defined (REL99) AND defined (TI_PS_FF_EMR) AND defined (GPRS)
+ − rr_data->ms_data.enable_ps_emr = TRUE;
+ − #endif
+ − for (i = 0; i < 8; i++)
+ − {
+ − rr_data->nc_data[i].lai.mcc[0] = NOT_PRESENT_8BIT;
+ − rr_data->nc_data[i].lai.mnc[0] = NOT_PRESENT_8BIT;
+ − }
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − /*Default initializations for EMR*/
+ − memset (rr_data->sc_data.rep_count, NOT_PRESENT_8BIT, MAX_NEIGHBOURCELLS);
+ − for_set_default_emr_data(&rr_data->sc_data.emr_data_current);
+ − for_set_default_emr_data(&rr_data->sc_data.emr_data_temp);
+ − rr_data->sc_data.enh_para_status = ENH_PARA_INVALID_STATE;
+ − rr_data->sc_data.ba_list_ded = rr_data->sc_data.ba_list_idle = FALSE;
+ − rr_data->sc_data.ba_index = NOT_PRESENT_8BIT;
+ − #endif
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − rr_data->sc_data.cd.band_indicator = NOT_PRESENT_8BIT;
+ − rr_data->cr_data.cd.band_indicator = NOT_PRESENT_8BIT;
+ − #endif
+ −
+ − #if defined (TI_PS_FF_RTD) AND defined (REL99)
+ − att_init_rtd_data(p_cur);
+ − #endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */
+ −
+ − /*
+ − * clear the forbidden lists
+ − */
+ − att_clear_forb_list (FORBIDDEN_LIST_NORMAL);
+ − att_clear_forb_list (FORBIDDEN_LIST_ROAMING);
+ −
+ − /*
+ − * indicate all RR timer as not running
+ − */
+ − memset (rr_data->t_running, 0, sizeof (rr_data->t_running));
+ − rr_data->t_expire[T3110] = tim_t3110;
+ − rr_data->t_expire[T3122] = tim_t3122;
+ − rr_data->t_expire[T3126] = tim_t3126;
+ − rr_data->t_expire[TREG] = tim_treg;
+ − rr_data->t_expire[T_RESELECT] = tim_treselect;
+ − rr_data->t_expire[TIM_EXT_MEAS] = tim_ext_meas;
+ − rr_data->t_expire[T_NO_RESELECT] = NULL;
+ − rr_data->t_expire[TCSVALID] = NULL;
+ − rr_data->t_expire[TABORT] = tim_tabort;
+ − rr_data->t_expire[T_PLMN_SEARCH] = tim_plmn_search_expiry;
+ − rr_data->t_expire[T_FAST_CS] = tim_tfast_cs;
+ − rr_data->t_expire[T_NORMAL_CS] = tim_tnormal_cs;
+ −
+ − rr_data->net_lost = FALSE;
+ −
+ −
+ − #if defined FF_EOTD
+ − rr_data->t_expire[TAPDU] = tim_apdu;
+ − #endif /* FF_EOTD */
+ − rr_data->t_expire[TNNN] = tim_tnnn;
+ − rr_data->t_expire[T_DEDICATED_MODE] = NULL;
+ − rr_data->treg_pending = FALSE;
+ −
+ − rr_data->old_serving_cell = NOT_PRESENT_8BIT;
+ − rr_data->ms_data.rr_service = NO_SERVICE;
+ −
+ − /* tch loop "open" */
+ − rr_data->tch_loop_subch = NOT_PRESENT_8BIT;
+ −
+ − #if defined FF_EOTD
+ − rr_data->eotd_req_id = NOT_PRESENT_16BIT;
+ − #endif /* FF_EOTD */
+ −
+ − #ifdef _SIMULATION_
+ − {
+ − int i=0;
+ − for (;i<NUM_OF_RR_TIMERS;i++)
+ − {
+ − #ifdef OPTION_TIMER
+ − /* vsi_t_config is removed in latest GPF */
+ − #endif
+ − }
+ − }
+ − #endif
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_init_pl_status |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : the function clear the cell channel description list
+ − of the serving cell.
+ −
+ − */
+ −
+ − GLOBAL void att_init_pl_status (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_init_pl_status()");
+ −
+ − rr_data->sc_data.cd.v_cell_chan_desc = NO_CONTENT;
+ − srv_clear_list (&rr_data->sc_data.cd.cell_chan_desc);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_insert_cell_in_data |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : During the analysis of an incoming measurement report a
+ − new cell has been detected. The data of this cell are
+ − stored in RR.
+ −
+ − */
+ −
+ − static void att_insert_cell_in_data (T_MPH_MEASUREMENT_IND *report,
+ − UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE i;
+ − T_NC_DATA * rrd;
+ −
+ − TRACE_FUNCTION ("att_insert_cell_in_data()");
+ −
+ − /*
+ − * search for a free entry in the RR storage area.
+ − */
+ − for (i = 0; i < 6; i++)
+ − if (rr_data->nc_data[i].bcch_status EQ EMPTY)
+ − break;
+ −
+ − if (i < 6)
+ − {
+ − /*
+ − * RR will ever find a free area, because all
+ − * cells which are not longer in the measurement
+ − * report, are removed from the RR storage.
+ − * The if-condition is only for security reasons
+ − *
+ − * Store the data from the measurement report and
+ − * wait for the system information messages.
+ − */
+ − rrd = &rr_data->nc_data[i];
+ − rrd->bsic = report->ncells.bsic[index];
+ − rrd->arfcn = report->ncells.arfcn[index];
+ − rrd->rxlev = report->ncells.rx_lev[index];
+ − rrd->bcch_status = NON_DECODED;
+ − rrd->avail_time = 0;
+ − #ifdef GPRS
+ − if( rr_data->gprs_data.use_c31 )
+ − att_insert_c31_cr_data_in_cell(i);
+ − #endif
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_max |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function calculates the maximum from two values a and b.
+ −
+ − */
+ −
+ − GLOBAL SHORT att_max (SHORT a,
+ − SHORT b)
+ − {
+ − TRACE_FUNCTION ("att_max()");
+ −
+ − return ((a > b) ? a : b);
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_plmn_in_found_list |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function checks whether a PLMN identification is already
+ − stored in the found list or not during cell selection / cell
+ − reselection.
+ −
+ − */
+ −
+ − GLOBAL T_FOUND_ELEMENT *att_plmn_in_found_list (UBYTE *mcc, UBYTE *mnc)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_FOUND_ELEMENT *result = NULL;
+ − int i;
+ − T_FOUND_ELEMENT *found;
+ −
+ − TRACE_FUNCTION ("att_plmn_in_found_list()");
+ −
+ − /*
+ − * look for all entries, which are already stored
+ − */
+ − /*att_print_mcc_mnc(0, mcc, mnc, "compare with");*/
+ − found = &rr_data->sc_data.found[0];
+ − for (i = 0; i < rr_data->sc_data.found_entries; i++, found++)
+ − {
+ − /*
+ − * compare mobile country code and mobile network code.
+ − */
+ − /*att_print_mcc_mnc(found->arfcn, found->plmn.mcc, found->plmn.mnc, "list");*/
+ − if (dat_plmn_equal_req (mcc, mnc,
+ − found->plmn.mcc, found->plmn.mnc))
+ − {
+ − result = found;
+ − /*att_print_mcc_mnc(found->arfcn, mcc, mnc, "found");*/
+ − break;
+ − }
+ − }
+ −
+ − /*
+ − * return the result of the compare.
+ − */
+ − return result;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_save_found_plmn |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function set all list elements of a found PLMN.
+ − It marks the element as present (V_PLMN_PRES)
+ −
+ − */
+ −
+ − GLOBAL void att_save_found_plmn (T_FOUND_ELEMENT *element, UBYTE mcc[], UBYTE mnc[],
+ − USHORT arfcn, UBYTE rxlev, USHORT lac, UBYTE cell_ok)
+ − {
+ − GET_INSTANCE_DATA;
+ − element->plmn.v_plmn = V_PLMN_PRES;
+ − memcpy (element->plmn.mcc, mcc, SIZE_MCC);
+ − memcpy (element->plmn.mnc, mnc, SIZE_MNC);
+ − element->arfcn = arfcn;
+ − element->rxlev = rxlev;
+ − element->lac = lac; /* LOL 02.01.2003: added for EONS support */
+ − element->cell_ok = cell_ok;
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − if (rr_data->cs_data.region EQ BOTH_REGIONS)
+ − {
+ − element->region = srv_get_region_from_std_arfcn (std, arfcn);
+ − }
+ − else
+ − {
+ − #endif
+ − element->region = rr_data->cs_data.region;
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − }
+ − #endif
+ −
+ − TRACE_EVENT_P9 ("[%u] save new PLMN (%x%x%x %x%x%x) region=%d, entries=%u",
+ − arfcn, mcc[0], mcc[1], mcc[2], mnc[0], mnc[1], mnc[2],element->region,
+ − rr_data->sc_data.found_entries+1);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_priority_check |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function checks the priority of a cell during cell
+ − selection. It must be differed between first and second
+ − attempt.
+ −
+ − During the first attempt it will be differed between
+ − low and high (or normal) priority.
+ −
+ − In the first attempt the cell bar qualifier is used to
+ − differ between low and high priority. Beside this there
+ − is an exception defined in GSM 5.08 chapter 9.
+ −
+ − In the second attempt the priority check is ever successful.
+ −
+ − */
+ −
+ − static BOOL att_priority_check (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_priority_check()");
+ −
+ − /*
+ − * During the First Scan second attempt and second scan the priority
+ − * check is ever successful.
+ − */
+ − if((rr_data->cs_data.scan_mode EQ CS_FIRST_SCAN_SECOND_ATTEMPT) OR
+ − (rr_data->cs_data.scan_mode EQ CS_SECOND_SCAN))
+ − {
+ − return TRUE;
+ − }
+ −
+ − /*
+ − * Exception condition handling according GSM 5.08, chapter 9
+ − */
+ − if (dat_hplmn (rr_data->nc_data[CR_INDEX].lai.mcc,
+ − rr_data->nc_data[CR_INDEX].lai.mnc) AND
+ − rr_data->cell_test_operation AND
+ − rr_data->nc_data[CR_INDEX].c2_par.cbq EQ 0 AND
+ − rr_data->nc_data[CR_INDEX].rach.cell_bar_access EQ 1 AND
+ − (rr_data->nc_data[CR_INDEX].rach.ac & 0x8000) NEQ 0)
+ − return TRUE;
+ −
+ − /*
+ − * Else the cell bar qualifier indicates the priority
+ − */
+ − if (rr_data->nc_data[CR_INDEX].c2_par.cbq)
+ − return FALSE;
+ − else
+ − return TRUE;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_remove_bad_ncell |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : During fieldtest sometimes a neighbourcell in the
+ − measurement report was equal to a neighbourcell. This
+ − function deletes neighbourcell data if the channel is
+ − equal to the serving cell.
+ −
+ − */
+ −
+ − GLOBAL void att_remove_bad_ncell (T_MPH_MEASUREMENT_IND *report)
+ − {
+ − GET_INSTANCE_DATA;
+ − SHORT i = 0;
+ − SHORT j;
+ − T_ncells *pncells = &report->ncells;
+ − TRACE_FUNCTION ("att_remove_bad_ncell()");
+ −
+ − /*
+ − * for all neighbourcells in the measurement report.
+ − */
+ − while (i < pncells->no_of_ncells)
+ − {
+ − /*
+ − * Do nothing if the channel number is different from
+ − * the serving cell.
+ − */
+ − if (pncells->arfcn[i] NEQ rr_data->nc_data[SC_INDEX].arfcn)
+ − i++;
+ − else
+ − {
+ − /*
+ − * else overwrite this entry by moving the rest one index
+ − */
+ − for (j = i + 1; j < pncells->no_of_ncells; j++)
+ − {
+ − pncells->arfcn[j-1] = pncells->arfcn[j];
+ − pncells->rx_lev[j-1] = pncells->rx_lev[j];
+ − pncells->bsic[j-1] = pncells->bsic[j];
+ − pncells->time_alignmt[j-1] = pncells->time_alignmt[j];
+ − pncells->frame_offset[j-1] = pncells->frame_offset[j];
+ − }
+ −
+ − /*
+ − * clean the last entry
+ − */
+ − pncells->bsic[j-1] = 0;
+ − pncells->arfcn[j-1] = 0;
+ − pncells->rx_lev[j-1] = 0;
+ − pncells->time_alignmt[j-1] = 0;
+ − pncells->frame_offset[j-1] = 0;
+ − pncells->no_of_ncells--;
+ − }
+ − }
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_allow_lai_in_list |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After a successful location updating request MM informs
+ − RR about this. If the location area information is stored
+ − inside of one of the forbidden location area lists this
+ − information must be updated. In this function, the entry
+ − is deleted from the given forbidden list.
+ −
+ − */
+ −
+ − LOCAL void att_allow_lai_in_list (int list_type,
+ − const T_plmn * plmn,
+ − USHORT lac)
+ − {
+ − GET_INSTANCE_DATA;
+ − unsigned int i;
+ − T_loc_area_ident *forb_list;
+ −
+ − if (list_type EQ FORBIDDEN_LIST_NORMAL)
+ − forb_list = &rr_data->ms_data.forb_lac_list[0];
+ − else
+ − forb_list = &rr_data->ms_data.roam_forb_lac_list[0];
+ −
+ − TRACE_FUNCTION ("att_allow_lai_in_list()");
+ −
+ − for (i = 0; i < (MAX_LAI-1); i++)
+ − {
+ − if ((lac EQ forb_list[i].lac) AND
+ − dat_plmn_equal_req (plmn->mcc, plmn->mnc,
+ − forb_list[i].mcc, forb_list[i].mnc))
+ − {
+ − /*
+ − * if the location area code is stored, overwrite this information
+ − * and fill the last issue with a default value
+ − */
+ − /* Implements Measure#32: Row 60 */
+ − att_print_forb_list ( list_type, plmn, lac, i, TRUE);
+ −
+ − memmove (&forb_list[i], &forb_list[i + 1],
+ − (MAX_LAI - i - 1) * sizeof (T_loc_area_ident));
+ − forb_list[MAX_LAI - 1].lac = NOT_PRESENT_16BIT;
+ − memset (forb_list[MAX_LAI - 1].mcc, NOT_PRESENT_8BIT, SIZE_MCC);
+ − memset (forb_list[MAX_LAI - 1].mnc, NOT_PRESENT_8BIT, SIZE_MNC);
+ − }
+ − }
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_rem_lai_from_forb_list |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : After a successful location updating request MM informs
+ − RR about this. If the location area information is stored
+ − inside of one of the forbidden location area lists this
+ − information must be updated. In this function, the entry
+ − is deleted from all forbidden lists.
+ −
+ − */
+ −
+ − GLOBAL void att_rem_lai_from_forb_list (const T_plmn * plmn,
+ − USHORT lac)
+ − {
+ − TRACE_FUNCTION ("att_rem_lai_from_forb_list()");
+ −
+ − att_allow_lai_in_list (FORBIDDEN_LIST_NORMAL, plmn, lac);
+ − att_allow_lai_in_list (FORBIDDEN_LIST_ROAMING, plmn, lac);
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_search_cell |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function searches for a cell for cell reselection.
+ −
+ − */
+ −
+ − static void att_search_cell (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_search_cell()");
+ − #if defined(_SIMULATION_)
+ − TRACE_SELECTION_TYPE (rr_data->sc_data.selection_type);
+ − #endif /* _SIMULATION_ */
+ −
+ − /*
+ − * depending on the cell reselection mode.
+ − */
+ − switch (rr_data->sc_data.selection_type)
+ − {
+ − case BACK_FROM_DEDICATED:
+ − case BACK_FROM_DEDICATED_RLF:
+ − /*
+ − * back from dedicated use the fieldstrength of the
+ − * cells as the criterion
+ − */
+ − rr_data->reselect_index = att_get_highest_c2_index ();
+ − att_select_cell_dedicated ();
+ − break;
+ −
+ − case CELL_RESELECTION:
+ − case CELL_RESELECTION_NC:
+ − #ifdef GPRS
+ − if( rr_data->gprs_data.use_c31 )
+ − {
+ − SHORT i;
+ − TRACE_EVENT ("c32 used in att_search_cell");
+ − for( i = 0; i < 6; i++ )
+ − rr_data->nc_data[i].c32_used = FALSE;
+ −
+ − rr_data->ms_data.old_cell_tried = FALSE;
+ −
+ − rr_data->reselect_index = att_get_next_best_c32_index (FALSE);
+ −
+ − /*
+ − * if no suitable cell is available
+ − * try the old cell again.
+ − */
+ − if (! att_check_cell_c31 ())
+ − att_try_old_cell ();
+ −
+ − break;
+ − }
+ − #endif
+ − /*lint -fallthrough*/
+ − default:
+ − /*
+ − * cell reselection in idle mode, use the cell
+ − * reselection criterion C2.
+ − */
+ − rr_data->ms_data.old_cell_tried = FALSE;
+ − rr_data->reselect_index = att_get_highest_c2_index ();
+ −
+ − /*
+ − * if no suitable cell is available
+ − * try the old cell again.
+ − */
+ − if (! att_check_cell ())
+ − att_try_old_cell ();
+ − break;
+ − }
+ − EM_FMM_RESEL_START_IND;
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_set_pl_in_idle_mode |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function configures the idle mode at the end of
+ − cell selection and cell reselection.
+ −
+ − */
+ −
+ − GLOBAL void att_set_pl_in_idle_mode (void)
+ − {
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − GET_INSTANCE_DATA;
+ − #endif
+ − TRACE_FUNCTION ("att_set_pl_in_idle_mode()");
+ − /* Update std value depending upon the band indicator value */
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − att_update_std_band_indicator (rr_data->sc_data.cd.band_indicator);
+ − #endif
+ − /*
+ − * forward classmark information to layer 1 for random access procedure
+ − */
+ − att_build_classmark_req ();
+ −
+ − /*
+ − * configure the idle mode (paging group etc.)
+ − */
+ − att_build_idle_req (SC_INDEX, MODE_CELL_SELECTION);
+ −
+ − /*
+ − * configure the CBCH channel
+ − */
+ − #ifdef REL99
+ − att_config_cbch ();
+ − #else
+ − att_build_cbch();
+ − #endif
+ −
+ − /*
+ − * configure the neighbourcell list
+ − */
+ − att_code_mph_ncell_req (SC_INDEX);
+ −
+ − /*
+ − * store the neighbourcell information in PCM and the SIM card.
+ − */
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_set_sys_info_read |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The decision whether all system information messages are
+ − read during cell selection / cell reselection is done with
+ − this function. The variable sys_info_read is organized as
+ − a bitmap. Each system information message belongs to one
+ − bit of the bitmap.
+ −
+ − */
+ −
+ − GLOBAL void att_set_sys_info_read (USHORT mess_bit, UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_CELL_DATA * cd;
+ − USHORT si;
+ −
+ − TRACE_FUNCTION ("att_set_sys_info_read()");
+ −
+ − /*
+ − * set pointer to data
+ − */
+ − if (index EQ SC_INDEX)
+ − cd = &rr_data->sc_data.cd;
+ − else if (index EQ CR_INDEX)
+ − cd = &rr_data->cr_data.cd;
+ − else
+ − {
+ − TRACE_EVENT ("unexpected index in att_set_sys_info_read()");
+ − return;
+ − }
+ −
+ − /*
+ − * set the bit of the bitmap according to the message type.
+ − */
+ − cd->sys_info_read |= mess_bit;
+ −
+ − #if !defined(NTRACE)
+ − si = cd->sys_info_read;
+ − /* if (GET_STATE (STATE_ATT) EQ ATT_CS2) comment out just for debugging */
+ − {
+ − if (si EQ ALL_SYS_INFO_READ)
+ − {
+ − TRACE_EVENT ("SI complete");
+ − }
+ − else
+ − {
+ − #ifdef GPRS
+ − TRACE_EVENT_P9 ( "await SI%s%s%s%s%s%s%s (%x) for %s",
+ − (si & SYS_INFO_1_READ ) ? "" : " 1" ,
+ − (si & SYS_INFO_2_READ ) ? "" : " 2" ,
+ − (si & SYS_INFO_2BIS_READ) ? "" : " 2bis",
+ − (si & SYS_INFO_3_READ ) ? "" : " 3" ,
+ − (si & SYS_INFO_4_READ ) ? "" : " 4" ,
+ − (si & SYS_INFO_2TER_READ) ? "" : " 2ter",
+ − (si & SYS_INFO_13_READ ) ? "" : " 13",
+ − si,
+ − (index EQ SC_INDEX ) ? "SC_INDEX" : "CR_INDEX");
+ − #else
+ − TRACE_EVENT_P8 ( "await SI%s%s%s%s%s%s (%x) for %s",
+ − (si & SYS_INFO_1_READ ) ? "" : " 1" ,
+ − (si & SYS_INFO_2_READ ) ? "" : " 2" ,
+ − (si & SYS_INFO_2BIS_READ) ? "" : " 2bis",
+ − (si & SYS_INFO_3_READ ) ? "" : " 3" ,
+ − (si & SYS_INFO_4_READ ) ? "" : " 4" ,
+ − (si & SYS_INFO_2TER_READ) ? "" : " 2ter",
+ − si,
+ − (index EQ SC_INDEX ) ? "SC_INDEX" : "CR_INDEX");
+ − #endif
+ − }
+ − }
+ − #endif /* !NTRACE */
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_start_cell_reselection |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function is called at the beginning of a cell
+ − reselection.
+ −
+ − */
+ −
+ − GLOBAL void att_start_cell_reselection (UBYTE mode)
+ − {
+ − GET_INSTANCE_DATA;
+ − #ifdef GPRS
+ − T_GPRS_DATA *gprs_data = &rr_data->gprs_data;
+ − #endif
+ −
+ − TRACE_FUNCTION ("att_start_cell_reselection()");
+ −
+ − if( mode EQ BACK_FROM_DEDICATED_RLF)
+ − {
+ − /*
+ − * Allow 20 seconds for call re-establishment
+ − * In case of data link failure
+ − */
+ − tstart_tabort(2 * TABORT_VALUE);
+ − }
+ − else
+ − {
+ − tstart_tabort(TABORT_VALUE);
+ − }
+ −
+ − #ifdef GPRS
+ − if ( gprs_data->rrgrr_ext_meas_req NEQ NULL )
+ − {
+ − /*
+ − * Save the request to perform Cell Reselection.
+ − * Then stop the Ext Meas procedure.
+ − * At the very end of the Ext Meas continue with CR.
+ − */
+ − gprs_data->ext_meas_ctrl |= EXT_MEAS_START_CR;
+ − rr_data->sc_data.selection_type = mode;
+ − att_rrgrr_ext_meas_stop_req(NULL);
+ − return;
+ − }
+ − #endif
+ −
+ − /*
+ − * intialise some variables
+ − */
+ − rr_data->dyn_config.fcr = 0;
+ − rr_data->dyn_config.scr = 0;
+ − rr_data->bcch_error = 0;
+ − rr_data->pag_rec = FALSE;
+ − srv_clear_stored_prim (MPH_PAGING_IND);
+ −
+ − /*
+ − * MM has initiated GPRS Activation. While waiting for SI13 a local reselection
+ − * is triggered in RR because of a better neighbour cell. Since MM initiated GPRS
+ − * is pending; RR should respond by RR_ACTIVATE_CNF and not by RR_ACTIVATE_IND.
+ − */
+ − #ifdef GPRS
+ − if ( GET_STATE(STATE_GPRS) EQ GPRS_ACTIVATED AND
+ − rr_data->sc_data.mm_started EQ MM_ORIGINATED AND
+ − rr_data->sc_data.selection_type EQ CELL_RESELECTION_ON_GPRS_ACT)
+ − {
+ − att_init_cell_selection (mode, MM_ORIGINATED);
+ − }
+ − else
+ − #endif
+ − {
+ − att_init_cell_selection (mode, RR_ORIGINATED);
+ − }
+ − dat_att_null();
+ − #ifdef GPRS
+ − gprs_init_data_cr();
+ − #endif
+ − /*
+ − * stop any reselection related timer.
+ − */
+ − if (rr_data->sc_data.selection_type EQ BACK_FROM_DEDICATED_RLF)
+ − {
+ − /*
+ − * according GSM 5.08, chapter 6.7.2 call re-establishment
+ − * shall be finished latest during 20 seconds (=2*normal
+ − * cell reselection time).
+ − */
+ − TIMERSTART (T_RESELECT, 2*TRESELECT_VALUE);
+ − }
+ − else
+ − {
+ − TIMERSTART (T_RESELECT, TRESELECT_VALUE);
+ − }
+ −
+ − /*
+ − * depending on the cell reselection type
+ − */
+ − switch (rr_data->sc_data.selection_type)
+ − {
+ − case CELL_RESELECTION:
+ − case CELL_RESELECTION_RACH:
+ − /*
+ − * exclude serving cell for five seconds
+ − */
+ − set_tnnn (SC_INDEX, 5*ONE_SEC);
+ − break;
+ − case BACK_FROM_DEDICATED:
+ − case BACK_FROM_DEDICATED_RLF:
+ − /*
+ − * remove serving cell from neighbourcell results
+ − */
+ − att_remove_bad_ncell (&rr_data->ms_data.measurement_report);
+ − att_build_back_from_dedicated ();
+ − break;
+ − case CELL_RESELECTION_CR:
+ − /*
+ − * check cell to get back in full service
+ − */
+ − break;
+ − }
+ −
+ − EM_CELL_RESEL_STARTED;
+ − /*
+ − * find a suitable cell for cell reselection
+ − */
+ − att_search_cell ();
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_start_cell_selection |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The function is called at the beginning of a cell selection.
+ − */
+ −
+ − GLOBAL void att_start_cell_selection (BOOL originator, BOOL parallel,
+ − U8 search_mode)
+ − {
+ − GET_INSTANCE_DATA;
+ − TRACE_FUNCTION ("att_start_cell_selection()");
+ −
+ −
+ − TRACE_EVENT_P2 ("START %sparallel CELL SELECTION %s_ORIGINATED",
+ − parallel?"":"not ",
+ − originator?"MM":"RR");
+ −
+ − tstart_tabort(TABORT_VALUE);
+ −
+ − rr_data->net_lost = FALSE;
+ − rr_data->cs_data.black_list_search_pending = FALSE;
+ −
+ − EM_CELL_SELECTION;
+ −
+ − /* Set the current search mode */
+ − CS_SET_CURRENT_SEARCH_MODE(search_mode);
+ −
+ − /*
+ − * initialize some variables
+ − */
+ − /* found channel handling */
+ − if(originator EQ MM_ORIGINATED)
+ − {
+ − if(rr_data->ms_data.req_mm_service EQ FUNC_PLMN_SRCH)
+ − cs_get_channel_from_found_list ();
+ − else
+ − cs_clear_channel_from_found_list();
+ − }
+ −
+ − att_init_cell_selection (CELL_SELECTION, originator);
+ −
+ − /* Reset scan mode */
+ − rr_data->cs_data.scan_mode = CS_FIRST_SCAN_FIRST_ATTEMPT;
+ −
+ − /*
+ − * init process and start power measurements
+ − */
+ − #ifdef GPRS
+ − rr_data->gprs_data.use_c31 = FALSE;
+ − #endif
+ − if(!parallel)
+ − {
+ − memset (rr_data->nc_data, 0, 6*sizeof (T_NC_DATA));
+ − cs_set_null ();
+ − }
+ − cs_set_all ();
+ −
+ − /*
+ − * Why should a RR caused cell selection forego the helpfully information
+ − * of the neighbor cell description of the last serving cell to speed up
+ − * the selection of the new cell?
+ − * But it is appropriate only for non PARALLEL search!
+ − * Therefore we collect first the (at cs_data) stored cell discription
+ − * and merge it into a channel list (cs_data.bcch_info). As second step
+ − * we use this list to level up the priority of the matching cells after
+ − * the scan execution (cs_mph_power_cnf).
+ − */
+ −
+ − #ifdef GPRS
+ − if(att_gprs_is_avail() AND !parallel)
+ − gprs_init_data_cr();
+ − #endif
+ −
+ − /* Cell Selection Improvements-LLD section:4.1.3.4.1.3
+ − * This function updates the new search mode based on the current
+ − * dynamic search mode configuration
+ − */
+ − att_check_dynamic_search_mode_config();
+ −
+ − cs_start_scan ();
+ −
+ − if(!parallel)
+ − {
+ − if( GET_STATE(STATE_ATT) EQ ATT_CS_INIT )
+ − {
+ − att_code_rr_act_cnf();
+ − }
+ − else
+ − {
+ − SET_STATE (STATE_ATT, ATT_CS1);
+ − }
+ − dat_att_null();
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_try_old_cell |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : if no suitable cell has been found during cell reselection
+ − the old serving cell is tried.
+ −
+ − */
+ −
+ − static void att_try_old_cell (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE search_mode =0;
+ − TRACE_FUNCTION ("att_try_old_cell()");
+ −
+ − /*
+ − * check whether the old serving cell is possible
+ − */
+ −
+ − if (rr_data->nc_data[SC_INDEX].arfcn EQ rr_data->nc_data[CR_INDEX].arfcn)
+ − rr_data->ms_data.old_cell_tried = TRUE;
+ −
+ − /* Implements RR Clone findings #8 */
+ − if (! att_cell_barred_status_cr_no_cr (SC_INDEX) AND
+ − ! is_tnnn (SC_INDEX) AND
+ − rr_data->nc_data[SC_INDEX].c1 > 0 AND
+ − rr_data->ms_data.old_cell_tried EQ FALSE)
+ −
+ − {
+ − if (rr_data->sc_data.selection_type EQ CELL_RESELECTION_RACH)
+ − {
+ − /*
+ − * go direct back to the old cell after random access failure
+ − */
+ − att_return_to_idle();
+ − }
+ − else
+ − {
+ − /*
+ − * serving cell is possible. Configure layer 1
+ − * to start cell reselection
+ − */
+ − rr_data->ms_data.old_cell_tried = TRUE;
+ − att_start_cr_in_pl(SC_INDEX);
+ − }
+ − }
+ − else
+ − {
+ − /*
+ − * old serving cell is not possible. Then start
+ − * cell selection for recovering.
+ − */
+ − /* Refer Cell Selection Improvements-LLD section:4.1.3.9 */
+ − if(rr_data->ms_data.rr_service EQ FULL_SERVICE)
+ − {
+ − /* RR is in full service and in ATT_IDLE state. Cell reselection started
+ − * and failed. Start Fast search
+ − */
+ − search_mode = FAST_SEARCH_MODE;
+ − }
+ − else
+ − {
+ − if((rr_data->ms_data.rr_service EQ LIMITED_SERVICE) AND
+ − (rr_data->ms_data.req_mm_service EQ FUNC_PLMN_SRCH))
+ − {
+ − /* RR is in Limited service and in ATT_IDLE state. The requested service
+ − * by MM is Full service. Cell reselection started and failed.
+ − * Obtain new search mode based on the current search mode.
+ − */
+ − search_mode = cs_get_new_search_mode();
+ − }
+ − else
+ − {
+ − /* RR is in Limited service and in ATT_IDLE state. The requested service
+ − * by MM is Limited service. Cell reselection started and failed.
+ − * Start Normal search
+ − */
+ − search_mode = NORMAL_SEARCH_MODE;
+ − }
+ − }
+ −
+ − /* XY:n don't inform GRR , and don't call start_cs_gprs !!!
+ − * we can go here after sending a CR_IND but not being in CS3
+ − */
+ − att_start_cell_selection (RR_ORIGINATED, CS_NOT_PARALLEL,search_mode);
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_mph_identity_req |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Forwards the mobile identities to Layer 1.
+ −
+ − */
+ −
+ − GLOBAL void att_mph_identity_req (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − PALLOC (mph_identity_req, MPH_IDENTITY_REQ);
+ −
+ − TRACE_FUNCTION ("att_mph_identity_req()");
+ −
+ − memset (mph_identity_req, 0, sizeof (T_MPH_IDENTITY_REQ));
+ −
+ − switch (rr_data->ms_data.rr_service)
+ − {
+ − case FULL_SERVICE:
+ − /*
+ − * identities are only forwarded in full service
+ − */
+ − if (rr_data->ms_data.imsi_available)
+ − {
+ − /*
+ − * fill in the international mobile identity, if available
+ − */
+ − mph_identity_req->mid.len_imsi = rr_data->ms_data.imsi.c_ident_dig;
+ − memcpy (mph_identity_req->mid.imsi, rr_data->ms_data.imsi.ident_dig,
+ − mph_identity_req->mid.len_imsi);
+ − }
+ −
+ − if (rr_data->ms_data.tmsi_available)
+ − {
+ − /*
+ − * fill in the temporary mobile identity, if available
+ − */
+ − mph_identity_req->mid.v_tmsi = TRUE;
+ − mph_identity_req->mid.tmsi = rr_data->ms_data.tmsi_binary;
+ − }
+ − break;
+ − }
+ − #ifdef GPRS
+ − if (rr_data->gprs_data.gprs_indic)
+ − {
+ − att_add_ptmsi (mph_identity_req);
+ − {
+ − PALLOC(ms_id, RRGRR_MS_ID_IND);
+ − ms_id->tmsi = rr_data->ms_data.tmsi_binary;
+ − PSENDX(GRR, ms_id);
+ − }
+ − }
+ − #endif
+ −
+ − /*
+ − * send the valid identities to layer 1.
+ − */
+ − PSENDX (PL, mph_identity_req);
+ −
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_start_registration_timer |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Checks whether to start, to stop or to let run the
+ − registration timer depending on the requested service by MM
+ − and the actual RR service.
+ −
+ − */
+ −
+ − /*
+ − * Time tables with increasing periods depending on the registration attempt
+ − */
+ − #ifdef TI_PS_FF_AT_P_CMD_CTREG
+ − /* Initialize the table with default values
+ − * All the values are in secs / 10 format
+ − * Ex: T_TEN_SEC = 10 Sec / 10
+ − */
+ − const UBYTE no_service_mode_time [25] =
+ − { T_TEN_SEC, T_TEN_SEC, T_TEN_SEC, T_TEN_SEC,
+ − T_TWENTY_SEC, T_TWENTY_SEC, T_TWENTY_SEC, T_TWENTY_SEC, T_TWENTY_SEC,
+ − T_THIRTY_SEC, T_THIRTY_SEC, T_THIRTY_SEC, T_THIRTY_SEC, T_THIRTY_SEC,
+ − T_SIXTY_SEC, T_SIXTY_SEC, T_SIXTY_SEC, T_SIXTY_SEC, T_SIXTY_SEC,
+ − T_TWO_MIN, T_TWO_MIN, T_TWO_MIN, T_TWO_MIN, T_TWO_MIN,
+ − T_SIX_MIN
+ − };
+ − #else
+ − const T_TIME no_service_mode_time [25] =
+ − { TEN_SEC, TEN_SEC, TEN_SEC, TEN_SEC,
+ − TWENTY_SEC, TWENTY_SEC, TWENTY_SEC, TWENTY_SEC, TWENTY_SEC,
+ − THIRTY_SEC, THIRTY_SEC, THIRTY_SEC, THIRTY_SEC, THIRTY_SEC,
+ − SIXTY_SEC, SIXTY_SEC, SIXTY_SEC, SIXTY_SEC, SIXTY_SEC,
+ − TWO_MIN, TWO_MIN, TWO_MIN, TWO_MIN, TWO_MIN,
+ − SIX_MIN
+ − };
+ − #endif /* TI_PS_FF_AT_P_CMD_CTREG */
+ −
+ − #ifdef TI_PS_FF_AT_P_CMD_CTREG
+ − const UBYTE lim_service_mode_time [25] =
+ − { T_TEN_SEC, T_TEN_SEC, T_TEN_SEC, T_TEN_SEC,
+ − T_TWENTY_SEC, T_TWENTY_SEC, T_TWENTY_SEC, T_TWENTY_SEC, T_TWENTY_SEC,
+ − T_THIRTY_SEC, T_THIRTY_SEC, T_THIRTY_SEC, T_THIRTY_SEC, T_THIRTY_SEC,
+ − T_SIXTY_SEC, T_SIXTY_SEC, T_SIXTY_SEC, T_SIXTY_SEC, T_SIXTY_SEC,
+ − T_TWO_MIN, T_TWO_MIN, T_TWO_MIN, T_TWO_MIN, T_TWO_MIN,
+ − T_SIX_MIN
+ − };
+ − #else
+ − const T_TIME lim_service_mode_time [25] =
+ − { TEN_SEC, TEN_SEC, TEN_SEC, TEN_SEC,
+ − TWENTY_SEC, TWENTY_SEC, TWENTY_SEC, TWENTY_SEC, TWENTY_SEC,
+ − THIRTY_SEC, THIRTY_SEC, THIRTY_SEC, THIRTY_SEC, THIRTY_SEC,
+ − SIXTY_SEC, SIXTY_SEC, SIXTY_SEC, SIXTY_SEC, SIXTY_SEC,
+ − TWO_MIN, TWO_MIN, TWO_MIN, TWO_MIN, TWO_MIN,
+ − SIX_MIN
+ − };
+ − #endif /* TI_PS_FF_AT_P_CMD_CTREG */
+ −
+ − #ifdef TI_PS_FF_AT_P_CMD_CTREG
+ − /*
+ − * Function used to convert the user given TREG timer values in secs to ms
+ − * Ex: User value TREG value
+ − * 0 1 Sec
+ − * 1 10 Sec
+ − * 2 20 Sec etc
+ − */
+ − T_TIME cast2T_Time(UBYTE tab_val)
+ − {
+ − return (tab_val)? tab_val*10000: 1000;
+ − }
+ − #endif /* TI_PS_FF_AT_P_CMD_CTREG */
+ −
+ − GLOBAL void att_start_registration_timer (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_TIME status = 0L;
+ −
+ − #ifdef TI_PS_FF_AT_P_CMD_CTREG
+ − BOOL ret;
+ − UBYTE tab_val;
+ − #endif /* TI_PS_FF_AT_P_CMD_CTREG */
+ −
+ − TRACE_FUNCTION ("att_start_registration_timer()");
+ −
+ − /*
+ − * is there a TREG expiry which couldn't be handled yet
+ − * so restart TREG with a short time
+ − */
+ − if (rr_data->treg_pending)
+ − {
+ − TIMERSTART (TREG, ONE_SEC);
+ − rr_data->treg_pending = FALSE;
+ − return;
+ − }
+ − /*
+ − * preparation to avoid the change of a running TREG
+ − */
+ − TIMER_STATUS (rr_handle, TREG, &status);
+ − /*
+ − * depending on the current service of RR
+ − */
+ − switch (rr_data->ms_data.rr_service)
+ − {
+ − case NO_SERVICE:
+ − /*
+ − * RR has no service, so it will try to come back
+ − * to limited or full service after timeout
+ − * set fieldstrength to 0.
+ − */
+ − #ifdef FF_PS_RSSI
+ − RX_SetValue ( 0, RX_QUAL_UNAVAILABLE, RX_ACCE_UNAVAILABLE);
+ − #else
+ − RX_SetValue (0);
+ − #endif
+ − cs_set_all ();
+ − if (status)
+ − return;
+ −
+ − if (!IS_TIMER_ACTIVE(TREG))
+ − {
+ − #ifdef TI_PS_HCOMM_CHANGE
+ − PSIGNAL (_hCommDL, DL_TRACE_REQ, NULL);
+ − #else
+ − PSIGNAL (hCommDL, DL_TRACE_REQ, NULL);
+ − #endif /* TI_PS_HCOMM_CHANGE */
+ − }
+ − #ifdef TI_PS_FF_AT_P_CMD_CTREG
+ − ret = cl_shrd_get_treg(RR_MOD_NOSERVICE_TIME,
+ − rr_data->ms_data.reg_counter,
+ − &tab_val);
+ − TRACE_EVENT_P1("Reading of the TREG value is %s",ret? "SUCCESS":"FAILURE");
+ − if (!ret)
+ − {
+ − /* Use default on failure */
+ − tab_val = no_service_mode_time [rr_data->ms_data.reg_counter];
+ − }
+ − if (rr_data->ms_data.reg_counter < 24)
+ − {
+ − rr_data->ms_data.reg_counter++;
+ − }
+ − TIMERSTART (TREG, cast2T_Time(tab_val));
+ − #else
+ − if (rr_data->ms_data.reg_counter < 24)
+ − {
+ − TIMERSTART (TREG, no_service_mode_time [rr_data->ms_data.reg_counter++]);
+ − }
+ − else
+ − {
+ − TIMERSTART (TREG, no_service_mode_time [24]);
+ − }
+ − #endif /* TI_PS_FF_AT_P_CMD_CTREG */
+ − break;
+ − case LIMITED_SERVICE:
+ − /*
+ − * RR has limited service
+ − * The timer is only started if MM has requested Full Service.
+ − */
+ − if (rr_data->ms_data.req_mm_service EQ FUNC_PLMN_SRCH)
+ − {
+ − cs_set_all ();
+ − if (status)
+ − return;
+ − #ifdef TI_PS_FF_AT_P_CMD_CTREG
+ − ret = cl_shrd_get_treg(RR_MOD_LIMSERVICE_TIME,
+ − rr_data->ms_data.reg_counter,
+ − &tab_val);
+ − TRACE_EVENT_P1("Reading of the TREG value is %s",ret? "SUCCESS":"FAILURE");
+ − if (!ret)
+ − {
+ − /* Use default on failure */
+ − tab_val = lim_service_mode_time [rr_data->ms_data.reg_counter];
+ − }
+ − if (rr_data->ms_data.reg_counter < 24)
+ − {
+ − rr_data->ms_data.reg_counter++;
+ − }
+ − TIMERSTART (TREG, cast2T_Time(tab_val));
+ − #else
+ − if (rr_data->ms_data.reg_counter < 24)
+ − {
+ − TIMERSTART (TREG, lim_service_mode_time [rr_data->ms_data.reg_counter++]);
+ − }
+ − else
+ − {
+ − TIMERSTART (TREG, lim_service_mode_time [24]);
+ − }
+ − #endif /* TI_PS_FF_AT_P_CMD_CTREG */
+ − }
+ − break;
+ − case FULL_SERVICE:
+ − /*
+ − * In full service mode, the timer is not started.
+ − */
+ − TIMERSTOP (TREG);
+ − break;
+ − }
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_set_func |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Sets the requested MM functionality. If MM requests a
+ − PLMN available search (FUNC_NET_SRCH_BY_MMI) the current
+ − search request must be stored until the PLMN available
+ − search has been finished.
+ −
+ − */
+ −
+ − GLOBAL void att_set_func (UBYTE func)
+ − {
+ − GET_INSTANCE_DATA;
+ −
+ − #if !defined(NTRACE)
+ − TRACE_EVENT_P5 ("att_set_func: func op=%s %scurrent=%s, service:rMM=%s RRs=%s",
+ − _rr_str_FUNC[func],
+ − _rr_str_PARFUNC[rr_data->ms_data.parallel_net_plmn_search_type],
+ − _rr_str_FUNC[rr_data->ms_data.current_plmn_search_type],
+ − _rr_str_FUNC[rr_data->ms_data.req_mm_service],
+ − _rr_str_SERVICE[rr_data->ms_data.rr_service]);
+ −
+ − if ((func EQ FUNC_NET_SRCH_BY_MMI) AND
+ − (func NEQ rr_data->ms_data.parallel_net_plmn_search_type))
+ − {
+ − TRACE_EVENT_P2 ("att_set_func: NEW %scurrent=%s",
+ − _rr_str_PARFUNC[func],
+ − _rr_str_FUNC[rr_data->ms_data.current_plmn_search_type]);
+ − }
+ − else if ((func NEQ rr_data->ms_data.current_plmn_search_type) OR
+ − (rr_data->ms_data.parallel_net_plmn_search_type NEQ 0))
+ − {
+ − TRACE_EVENT_P1 ("att_set_func: NEW current=%s", _rr_str_FUNC[func]);
+ − }
+ − #endif /* !NTRACE */
+ −
+ − switch (func)
+ − {
+ − case FUNC_LIM_SERV_ST_SRCH:
+ − case FUNC_PLMN_SRCH:
+ − case FUNC_ST_PWR_SCAN:
+ − rr_data->ms_data.parallel_net_plmn_search_type = 0;
+ − rr_data->ms_data.current_plmn_search_type = func;
+ − break;
+ −
+ − case FUNC_NET_SRCH_BY_MMI:
+ − rr_data->ms_data.parallel_net_plmn_search_type = func;
+ − break;
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_get_func |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Requests the MM functionality. If a temporary PLMN available
+ − search request is active (temp_func = FUNC_NET_SRCH_BY_MMI)
+ − this function is returned, else the last one which was
+ − requested by MMI.
+ −
+ − */
+ −
+ − static UBYTE att_get_func (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − switch (rr_data->ms_data.parallel_net_plmn_search_type)
+ − {
+ − case FUNC_NET_SRCH_BY_MMI:
+ − return (rr_data->ms_data.parallel_net_plmn_search_type);
+ −
+ − default:
+ − return (rr_data->ms_data.current_plmn_search_type);
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_clear_parallel_search |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Resets the temporary requested MM functionality.
+ −
+ − */
+ −
+ − static void att_clear_parallel_search (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − rr_data->ms_data.parallel_net_plmn_search_type = 0;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_init_cr_data |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The CR index contains the whole data of the new cell.
+ − It is cleared prior to cell reselection
+ − */
+ −
+ − GLOBAL void att_init_cr_data (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − /*
+ − * clear sys info bitmap
+ − */
+ − rr_data->cr_data.cd.sys_info_read = NO_SYS_INFO_READ;
+ −
+ − /*
+ − * reset BA list from SI 2, 2Bis and 2Ter
+ − */
+ − memset(&rr_data->cr_data.cr_white_list, 0 , sizeof(T_CR_WHITE_LIST));
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_cr_data |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The CR index contains the whole data of the new cell.
+ − It shall replace the content of the old serving cell.
+ − The content of the old serving cell is copied to the
+ − column of the new cell (in the neighbour cell area). The
+ − status is set to DECODED.
+ −
+ − */
+ −
+ − GLOBAL void att_copy_cr_data (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE i;
+ −
+ − /*
+ − * look for all neighbourcells.
+ − */
+ − for (i=0;i<6;i++)
+ − if (rr_data->nc_data[CR_INDEX].arfcn EQ rr_data->nc_data[i].arfcn)
+ − {
+ − /*
+ − * channel found, then update neighbourcell
+ − */
+ − memcpy (&rr_data->nc_data[i], &rr_data->nc_data[SC_INDEX],
+ − sizeof (T_NC_DATA));
+ − rr_data->nc_data[i].bcch_status = DECODED;
+ − rr_data->old_serving_cell = i;
+ − break;
+ − }
+ −
+ − /*
+ − * update SC_INDEX column.
+ − */
+ − memcpy (&rr_data->nc_data[SC_INDEX], &rr_data->nc_data[CR_INDEX],
+ − sizeof (T_NC_DATA));
+ − /*
+ − * implementation problem: must be set for RR originated search
+ − * e.g. limited to full service
+ − */
+ − rr_data->nc_data[SC_INDEX].bcch_status = DECODED;
+ −
+ − /*
+ − * fit classmarks depend on the values of 'std' and the new serving cell
+ − */
+ − rr_csf_fit_capability ();
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − /*When there's no SI-2quater in the reselected cell and in the current
+ − cell the reporting is of enhanced type, we can no longer use these
+ − enhanced parameters. */
+ − /*Reset EMR data and indicate the same to GRR and ALR*/
+ − if ( rr_data->sc_data.emr_data_current.is_data_valid EQ TRUE )
+ − {
+ − memset (rr_data->sc_data.rep_count, NOT_PRESENT_8BIT, MAX_NEIGHBOURCELLS);
+ − for_set_default_emr_data(&rr_data->sc_data.emr_data_current);
+ − for_send_enh_para(&rr_data->sc_data.emr_data_current);
+ − }
+ − #endif
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_eplmn_list |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function copies the EPLMN list to rr_data global area
+ −
+ − */
+ −
+ − GLOBAL void att_copy_eplmn_list (T_eq_plmn_list *eq_plmn_list)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE i;
+ −
+ − /* List only copied if valid and PLMNs greater than 1 (HPLMN included)
+ − else EPLMN list is marked as invalid */
+ − if(eq_plmn_list->v_eq_plmn)
+ − {
+ − for(i=0;i<RR_EPLMNLIST_SIZE;i++)
+ − {
+ − T_plmn local_copy;
+ − rr_unpack_plmn(&local_copy, eq_plmn_list->eq_plmn, i);
+ − rr_data->ms_data.eq_plmn_list[i] = local_copy; /*Struct copy*/
+ − }
+ −
+ − rr_data->ms_data.v_eq_plmn = TRUE;
+ − }
+ − else
+ − {
+ − rr_data->ms_data.v_eq_plmn = FALSE;
+ − memset(&rr_data->ms_data.eq_plmn_list, 0xFF, SIZE_EPLMN);
+ − }
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : rr_unpack_plmn |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Unpacks a PLMN from compressed form to uncompressed form.
+ −
+ − */
+ −
+ − LOCAL void rr_unpack_plmn (T_plmn *plmn, const UBYTE *packed, USHORT index)
+ − {
+ − index *= UBYTES_PER_PLMN;
+ − plmn->mcc[0] = packed[index] & 0x0f;
+ − plmn->mcc[1] = packed[index] >> 4;
+ − index++;
+ − plmn->mcc[2] = packed[index] & 0x0f;
+ − plmn->mnc[2] = packed[index] >> 4;
+ − index++;
+ − plmn->mnc[0] = packed[index] & 0x0f;
+ − plmn->mnc[1] = packed[index] >> 4;
+ − index++;
+ − if ((plmn->mcc[0] & 0x0F) EQ 0x0F)
+ − plmn->v_plmn = V_PLMN_NOT_PRES;
+ − else
+ − plmn->v_plmn = V_PLMN_PRES;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_build_back_from_dedicated |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : A list is build containing the serving cell followed by
+ − the synchronized neighbourcells in decreasing fieldstrenght.
+ − In case of call re-establishment the serving cell is not used.
+ − The list controls the cell reselection after dedicated mode.
+ −
+ − */
+ −
+ − static void att_build_back_from_dedicated (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE i;
+ − T_NC_DATA * rrd;
+ −
+ − /*
+ − * update the neighbourcell entries
+ − */
+ − for (i=0;i<6;i++)
+ − {
+ − /*
+ − * clear the data for this neighbourcell
+ − */
+ − memset (&rr_data->nc_data[i], 0, sizeof (T_NC_DATA));
+ − rrd = &rr_data->nc_data[i];
+ − if (rr_data->ms_data.measurement_report.ncells.no_of_ncells > i)
+ − {
+ − /*
+ − * if neighbourcell data is available in the measurement report
+ − * copy the data.
+ − */
+ − rrd->bsic = rr_data->ms_data.measurement_report.ncells.bsic[i];
+ − rrd->arfcn = rr_data->ms_data.measurement_report.ncells.arfcn[i];
+ − rrd->rxlev = rr_data->ms_data.measurement_report.ncells.rx_lev[i];
+ − rrd->bcch_status = NON_DECODED;
+ − rrd->avail_time = 0;
+ − }
+ − else
+ − rrd->bcch_status = EMPTY;
+ − }
+ −
+ − /*
+ − * update the serving cell fieldstrength data.
+ − * It is important to make distinction between rxlev of the traffic channel and of the BCCH channel
+ − * for C1 and C2 calculations
+ − */
+ − rr_data->nc_data[SC_INDEX].rxlev = rr_data->ms_data.measurement_report.bcch_rxlev_of_sc;
+ −
+ − /*
+ − * implementation problem: must be set to have a defined status.
+ − */
+ − rr_data->nc_data[SC_INDEX].bcch_status = DECODED;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_network |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function is a SW shield for the FTA campaign. Due
+ − to bad cables and public networks with high fieldstrength
+ − some FTA testcases (especially MM testcases with limited
+ − services) fails on our Anite System. To avoid this these
+ − function is a SW shield, excluding all german networks.
+ −
+ − This function must be adapted for other countries if needed.
+ −
+ − */
+ −
+ − GLOBAL UBYTE att_check_network (T_loc_area_ident * lai)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE count=0;
+ − /*
+ − * if a test SIM is inserted and "Software Shielding" is enabled then check mcc value with stored mcc value
+ − * (durig dynamic configuration command shield).The network exists in the list updated using dynamic
+ − * configauration command SHIELD then return FALSE.
+ − */
+ − if ((test_house EQ TRUE) AND (rr_data->dyn_config.mcc_shield.enabled EQ TRUE))
+ − {
+ − for(count=0;count< MAX_MCC_SHIELD;count++)
+ − {
+ − if (lai->mcc[0] EQ rr_data->dyn_config.mcc_shield.mcc[count][0] AND
+ − lai->mcc[1] EQ rr_data->dyn_config.mcc_shield.mcc[count][1] AND
+ − lai->mcc[2] EQ rr_data->dyn_config.mcc_shield.mcc[count][2] )
+ − {
+ − TRACE_EVENT ("att_check_network() returns FALSE");
+ − return FALSE;
+ − }
+ − }
+ − }
+ − return TRUE;
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_return_to_idle |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : The functions configures PL for idle mode without going
+ − through a complete cell reselection.
+ −
+ − */
+ −
+ − GLOBAL void att_return_to_idle (void)
+ − {
+ − GET_INSTANCE_DATA;
+ −
+ − TIMERSTOP (T_RESELECT);
+ − /*
+ − * configure layer 1
+ − */
+ − #ifdef GPRS
+ − if(att_gprs_cell_has_pbcch())
+ − {
+ − att_gprs_stop_pl();
+ − }
+ − else
+ − {
+ − #endif
+ − att_build_idle_req (SC_INDEX, MODE_CELL_SELECTION);
+ − #ifdef GPRS
+ − }
+ − #endif
+ −
+ − SET_STATE (STATE_ATT, ATT_IDLE);
+ − dat_att_cell_selected();
+ − srv_use_stored_prim ();
+ − #ifdef REL99
+ − #ifdef GPRS
+ − /* RR is returning to idle mode. If cbch info was received
+ − * during transfer mode, send it now.
+ − */
+ − if(rr_data->gprs_data.cbch_info_rxvd_in_ptm)
+ − {
+ − rr_data->gprs_data.cbch_info_rxvd_in_ptm = FALSE;
+ − att_config_cbch();
+ − }
+ − #endif
+ − #endif
+ − /*
+ − * Start registration timer if needed
+ − */
+ − att_start_registration_timer ();
+ −
+ − rr_data->bcch_error = 0;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_notify_stop_plmn_search |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Network search has to be aborted in lower layer so that
+ − NC monitoring can be resumed when going back to idle mode.
+ − */
+ −
+ − GLOBAL void att_notify_stop_plmn_search (UBYTE deactivate_pl)
+ − {
+ − GET_INSTANCE_DATA;
+ − UCHAR state = GET_STATE (STATE_ATT);
+ −
+ − TRACE_EVENT_P1("att_notify_stop_plmn_search - %x", rr_data->start_cell_reselection);
+ −
+ − if ( rr_data->ms_data.req_mm_service EQ FUNC_NET_SRCH_BY_MMI OR
+ − (rr_data->ms_data.rr_service EQ LIMITED_SERVICE AND !rr_data->start_cell_reselection) OR
+ − (rr_data->ms_data.rr_service EQ NO_SERVICE AND !rr_data->start_cell_reselection) OR
+ − state EQ ATT_CS1 OR
+ − state EQ ATT_CS2 OR
+ − CS_GET_CURRENT_SEARCH_MODE EQ BLACK_LIST_SEARCH_MODE)
+ − {
+ − PALLOC (mph_sync_req, MPH_SYNC_REQ);
+ −
+ − if(deactivate_pl)
+ − mph_sync_req->cs = CS_STOP_PLMN_SEARCH_AND_DEACTIVATE;
+ − else
+ − mph_sync_req->cs = CS_STOP_PLMN_SEARCH;
+ −
+ − PSENDX (PL, mph_sync_req);
+ −
+ − rr_data->start_cell_reselection = TRUE;
+ −
+ − TRACE_EVENT_P1("Start_cell_reselection %d",rr_data->start_cell_reselection);
+ −
+ − }
+ − }
+ −
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_reset_old_lai |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function is used to reset old lai parameters
+ − */
+ −
+ − GLOBAL void att_reset_old_lai_rac ()
+ − {
+ − GET_INSTANCE_DATA;
+ − rr_data->old_cell_id = NOT_PRESENT_16BIT;
+ − #ifdef GPRS
+ − rr_data->old_rac = NOT_PRESENT_8BIT;
+ − #endif
+ − memset (&rr_data->old_lai, NOT_PRESENT_8BIT, sizeof(T_loc_area_ident));
+ −
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_copy_old_lai_rac |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function is used to copy old lai and rac parameters
+ − */
+ −
+ − GLOBAL void att_copy_old_lai_rac (U8 index)
+ − {
+ − GET_INSTANCE_DATA;
+ − memcpy (&rr_data->old_lai, &rr_data->nc_data[index].lai,
+ − sizeof(T_loc_area_ident));
+ − rr_data->old_cell_id = rr_data->nc_data[index].cell_id;
+ − #ifdef GPRS
+ − rr_data->old_rac = rr_data->nc_data[index].rac;
+ − #endif
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_check_dynamic_search_mode_config |
+ − +--------------------------------------------------------------------------------+
+ −
+ − PURPOSE : This function updates the new search mode based on the current
+ − dynamic search mode configuration
+ − CSI-LLD section:4.1.3.4.1.8
+ − */
+ −
+ − GLOBAL void att_check_dynamic_search_mode_config(void)
+ − {
+ − GET_INSTANCE_DATA;
+ − U8 new_mode = CS_GET_CURRENT_SEARCH_MODE;
+ −
+ − TRACE_FUNCTION("att_check_dynamic_search_mode_config()");
+ −
+ − TRACE_EVENT_P1("New Search Mode : %d", CS_GET_CURRENT_SEARCH_MODE);
+ −
+ − if(new_mode EQ FAST_SEARCH_MODE AND !rr_data->dyn_config.tfast_cs_val)
+ − {
+ − /* If Fast search is disabled, use Normal search */
+ − new_mode = NORMAL_SEARCH_MODE;
+ −
+ − TRACE_EVENT("Fast Search is disabled");
+ − }
+ −
+ − if(new_mode EQ NORMAL_SEARCH_MODE AND !rr_data->dyn_config.tnormal_cs_val)
+ − {
+ − /* If Normal search is disabled, use Full search */
+ − new_mode = FULL_SEARCH_MODE;
+ −
+ − TRACE_EVENT("Normal Search is disabled");
+ − }
+ −
+ − if(new_mode NEQ CS_GET_CURRENT_SEARCH_MODE)
+ − TRACE_EVENT_P1("Allowed Search Mode : %d", new_mode);
+ −
+ − CS_SET_CURRENT_SEARCH_MODE(new_mode);
+ −
+ − }
+ −
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_get_txpwr_max_cch |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function is used to calculate the MAX power used on CCH
+ − */
+ −
+ − LOCAL UBYTE att_get_txpwr_max_cch (UBYTE index)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE max_cch = rr_data->nc_data[index].select_para.ms_txpwr_max_cch;
+ − UBYTE dbm_value;
+ −
+ − TRACE_FUNCTION("att_get_txpwr_max_cch()");
+ −
+ − switch(std)
+ − {
+ − case STD_1800:
+ − case STD_DUAL_EGSM:
+ − case STD_DUAL:
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − case STD_850_1800:
+ − case STD_850_900_1800:
+ − #endif
+ − /* Ref 04.18 Section 10.5.2.35
+ − * Rest Octets IE includes parameters which are used by the mobile station
+ − * for cell selection and reselection purposes. It may also include the
+ − * POWER OFFSET parameter used by DCS 1800 Class 3 MS.
+ − */
+ − if (INRANGE(LOW_CHANNEL_1800,rr_data->nc_data[index].arfcn,HIGH_CHANNEL_1800))
+ − {
+ − /* DCS 1800 */
+ − if (att_get_power() EQ 0x02 AND
+ − rr_data->nc_data[index].c2_par.power_off_ind)
+ − {
+ − TRACE_EVENT(" IDLE_REQ :DCS_PCLASS3");
+ − /* Class 3 & Valid Power offset */
+ − dbm_value = p_control_dcs[max_cch] +
+ − (rr_data->nc_data[index].c2_par.power_off << 1);
+ −
+ − /* dbm value is more than 36 return the max cch power level as 29 */
+ − if(dbm_value > 36) return 29;
+ −
+ − if(dbm_value NEQ 0)
+ − {
+ − /* map the calculated dbm value to power control value
+ − * Specification : 0505 Section : 4.1.1
+ − */
+ − max_cch = (30 - dbm_value) >= 0 ?
+ − (30 - dbm_value) >> 1 :
+ − (31 - ((dbm_value - 32) >> 1));
+ − }
+ − } /* if att_get_power */
+ − } /* if INRANGE */
+ − break;
+ − default :
+ − break;
+ − } /* switch std */
+ − return max_cch;
+ − }
+ −
+ −
+ −
+ − #if !defined(NTRACE)
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_print_mcc_mnc |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Outputs to trace mcc and mnc arrays
+ −
+ − */
+ − /* Implements Measure#32: Row 52,53, 124 & 125 */
+ − GLOBAL void att_print_mcc_mnc (USHORT arfcn, UBYTE *mcc, UBYTE *mnc, T_S2I_STRING titel)
+ − {
+ − TRACE_EVENT_P8 ( "[%d] MCC=%x%x%x MNC=%x%x%x %s",
+ − ((int)arfcn), mcc[0], mcc[1], mcc[2], mnc[0], mnc[1], mnc[2],
+ − S2I_STRING(titel)); /* A valid string is expected */
+ − }
+ −
+ − LOCAL void att_print_selection_type (UBYTE selection_type)
+ − {
+ − GET_INSTANCE_DATA;
+ − switch (rr_data->sc_data.selection_type)
+ − {
+ − case CELL_SELECTION:
+ − TRACE_EVENT_WIN ("selection_type=CELL_SELECTION");
+ − break;
+ − case CELL_RESELECTION:
+ − TRACE_EVENT_WIN ("selection_type=CELL_RESELECTION");
+ − break;
+ − case BACK_FROM_DEDICATED:
+ − TRACE_EVENT_WIN ("selection_type=BACK_FROM_DEDICATED");
+ − break;
+ − case CELL_RESELECTION_NC:
+ − TRACE_EVENT_WIN ("selection_type=CELL_RESELECTION_NC");
+ − break;
+ − case BACK_FROM_DEDICATED_RLF:
+ − TRACE_EVENT_WIN ("selection_type=BACK_FROM_DEDICATED_RLF");
+ − break;
+ − case CELL_RESELECTION_RACH:
+ − TRACE_EVENT_WIN ("selection_type=CELL_RESELECTION_RACH");
+ − break;
+ − case CELL_RESELECTION_CR:
+ − TRACE_EVENT_WIN ("selection_type=CELL_RESELECTION_CR");
+ − break;
+ − }
+ − }
+ −
+ − /* Implements Measure#32: Row 36, 39 and 40 */
+ − LOCAL void att_print_op (T_op *op, T_S2I_STRING titel)
+ − {
+ − T_S2I_STRING sim, mode;
+ − if (op->v_op)
+ − {
+ − sim = op->sim_ins ? (op->ts ? S2I_STRING("TEST") : S2I_STRING("NORMAL")) : S2I_STRING("NO");
+ − mode = op->m ? S2I_STRING("MAN") : S2I_STRING("AUTO");
+ −
+ − TRACE_EVENT_P5 ("%s: SIM=%s mode=%s func=%s service=%s",
+ − S2I_STRING(titel),
+ − S2I_STRING(sim),
+ − S2I_STRING(mode),
+ − S2I_STRING(_rr_str_FUNC[op->func]),
+ − S2I_STRING(_rr_str_SERVICE[op->service]));
+ − }
+ − }
+ − #endif /* !NTRACE */
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_set_rr_service_info |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Copies the RR related service info(service,PLMN,lac,cell_id)
+ − */
+ −
+ − GLOBAL void att_set_rr_service_info (void)
+ − {
+ − GET_INSTANCE_DATA;
+ −
+ − T_LOC_INFO loc_info;
+ − loc_info.service_mode = rr_data->ms_data.rr_service;
+ − if(rr_data->ms_data.rr_service NEQ NO_SERVICE)
+ − {
+ − loc_info.lac = rr_data->nc_data[SC_INDEX].lai.lac;
+ − loc_info.cell_id = rr_data->nc_data[SC_INDEX].cell_id;
+ − memcpy (loc_info.mcc,rr_data->nc_data[SC_INDEX].lai.mcc,SIZE_MCC);
+ − memcpy (loc_info.mnc,rr_data->nc_data[SC_INDEX].lai.mnc,SIZE_MNC);
+ − }
+ − cl_shrd_set_loc(&loc_info);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_set_tim_advance_info |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Copies the Timing Advance info.ME_STATUS is presently using
+ − only ME_STATUS_IDLE which is not as per the standard.We need to assign
+ − me_status correctly.
+ − */
+ −
+ − GLOBAL void att_set_tim_advance_info (void)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_TIM_ADV tim_adv;
+ − tim_adv.me_status = ME_STATUS_IDLE;
+ − tim_adv.tm_adv = rr_data->sc_data.new_ta;
+ − cl_shrd_set_tim_adv(&tim_adv);
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_func1_opt |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function sets the system information read flag of the passed
+ − si_type . It saves the neighbour cell list or Merges it to previously
+ − saved list if available.
+ − */
+ − LOCAL void att_copy_sys_info_2bis_2ter_par(UBYTE index,T_SI_TYPE si_type,
+ − T_LIST *new_2_bis_ter_list, BUF_neigh_cell_desc *neigh_cell_desc,
+ − UBYTE indicate_changes)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_LIST new_list;
+ − T_CELL_DATA *cd;
+ − BOOL modified = FALSE;
+ −
+ − USHORT cur_si_type_read; /*Current SI type*/
+ − USHORT oth_si_type_read; /*The other SI type in this function */
+ − USHORT oth_si_to_clean;/*The other SI to clean in this function*/
+ − UCHAR si_type_msg; /* 2BIS or 2TER type msg*/
+ −
+ − TRACE_FUNCTION("att_copy_sys_info_2bis_2ter_par()");
+ −
+ − switch (index)
+ − {
+ − case SC_INDEX:
+ − case CR_INDEX:
+ − if(si_type EQ SI_TYPE_2BIS)
+ − {
+ − cur_si_type_read = SYS_INFO_2BIS_READ;
+ − oth_si_type_read = SYS_INFO_2TER_READ;
+ − oth_si_to_clean = IND_SI_2TER;
+ − si_type_msg = SYS_INFO_2bis_MSG;
+ −
+ − }
+ − else
+ − {
+ − cur_si_type_read = SYS_INFO_2TER_READ;
+ − oth_si_type_read = SYS_INFO_2BIS_READ;
+ − oth_si_to_clean = IND_SI_2BIS;
+ − si_type_msg = SYS_INFO_2ter_MSG;
+ − }
+ − if (index EQ SC_INDEX)
+ − {
+ − cd = &rr_data->sc_data.cd;
+ − if ((cd->sys_info_read & cur_si_type_read) EQ cur_si_type_read)
+ − {
+ − srv_copy_list (&cd->ncell_list, new_2_bis_ter_list, sizeof (T_LIST));
+ − att_clean_buf ((USHORT)(IND_SI_2 | oth_si_to_clean));
+ − cd->sys_info_read &= ~(SYS_INFO_2_READ | oth_si_type_read);
+ − att_check_2ter_read (index);
+ − }
+ − else
+ − srv_merge_list (&cd->ncell_list, new_2_bis_ter_list);
+ − modified = TRUE;
+ − #ifdef GPRS
+ − rr_data->gprs_data.ba_bcch_modified= FALSE;
+ − #endif
+ −
+ − }
+ − else
+ − {
+ − cd = &rr_data->cr_data.cd;
+ − /*
+ − * merge 2bis list with the old neighbour cell list
+ − */
+ − srv_copy_list (&new_list, &cd->ncell_list,
+ − sizeof (T_LIST));
+ − srv_merge_list (&new_list, new_2_bis_ter_list);
+ −
+ − if (srv_compare_list (&cd->ncell_list, &new_list) EQ FALSE)
+ − {
+ − /*
+ − * both lists are different
+ − */
+ − srv_copy_list (&cd->ncell_list, &new_list, sizeof (T_LIST));
+ − modified = TRUE;
+ − }
+ − }
+ − /* CSI-LLD section:4.1.1.11
+ − * This function Updates the black list with the BA list received in si2bis/ter
+ − */
+ − cs_remove_BA_MA_from_black_list(rr_data->cs_data.region,&cd->ncell_list);
+ −
+ − dat_store_neigh_cell_desc (si_type_msg, index,
+ − neigh_cell_desc, &cd->ncell_list);
+ − /*
+ − * Indicate that the system information type 2bis message has been read.
+ − */
+ − att_set_sys_info_read (cur_si_type_read, index);
+ −
+ − /*
+ − * forward new neighbour cell list to layer 1
+ − * if changes shall be indicated
+ − */
+ − if (modified AND indicate_changes)
+ − {
+ − att_code_mph_ncell_req (index);
+ − }
+ −
+ − break;
+ − }
+ − }
+ −
+ −
+ −
+ − #if defined (REL99) && defined (TI_PS_FF_EMR)
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : attf_send_enh_para_to_alr |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function formats and sends a primitive with enhanced measurement
+ − parameters.
+ − */
+ − GLOBAL void attf_send_enh_para_to_alr(UBYTE rep_type, T_enh_para_struct *p_src)
+ − {
+ −
+ − PALLOC(p_enh, MPH_ENHPARA_UPDATE_REQ);
+ − TRACE_FUNCTION ("attf_send_enh_para_to_alr");
+ − p_enh->rep_type = rep_type;
+ − p_enh->enh_para = *p_src;
+ − PSENDX(PL,p_enh);
+ − return;
+ − }
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_print_mcc_mnc |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : Outputs to trace mcc and mnc arrays
+ −
+ − */
+ − LOCAL void att_check_for_si5ter_and_enhpara (UBYTE old_index)
+ − {
+ − GET_INSTANCE_DATA;
+ − T_rr_enh_para *p_temp = &rr_data->sc_data.emr_data_temp;
+ − T_rr_enh_para *p_cur = &rr_data->sc_data.emr_data_current;
+ −
+ − /*we have complete BA(SACCH) list (with or without SI-5 ter ) : we will
+ − discard any enhanced para that are present in previous state which are
+ − aligned to BA(BCCH)*/
+ − if (rr_data->sc_data.enh_para_status EQ ENH_PARA_IDLE )
+ − {
+ − /*The enhanced parameters available are from different state, reset them*/
+ − for_set_default_emr_data(p_temp);
+ − memset (rr_data->sc_data.rep_count, NOT_PRESENT_8BIT, MAX_NEIGHBOURCELLS);
+ − for_set_default_emr_data(p_cur);
+ − for_send_enh_para(p_cur);
+ − rr_data->sc_data.enh_para_status = ENH_PARA_INVALID_STATE;
+ − return;
+ − }
+ − if(old_index EQ rr_data->sc_data.ba_index)
+ − {
+ − /*Updation of enhanced cell list for EMR : we can receive
+ − SI-5ter and MI in any random order*/
+ − if ( (rr_data->sc_data.emr_data_temp.is_data_valid EQ TRUE) AND
+ − rr_data->sc_data.emr_data_temp.ba2bsic_map_pending )
+ − for_att_update_ba2bsic_mapping(&rr_data->sc_data.emr_data_temp);
+ − else if ( (rr_data->sc_data.emr_data_current.is_data_valid EQ TRUE) AND
+ − rr_data->sc_data.emr_data_current.ba2bsic_map_pending )
+ − for_att_update_ba2bsic_mapping(&rr_data->sc_data.emr_data_current);
+ −
+ − /*When BA(SACCH) is ready, check whether there are enhanced parameters in
+ − temp that needs attention*/
+ − if ( (rr_data->sc_data.enh_para_status EQ ENH_PARA_DEDICATED) AND
+ − (p_temp->is_data_valid EQ TRUE ) )
+ − {
+ − /* This means enhanced parameters were received before BA list - so update
+ − the enhanced list with actual ARFCN and update current EMR data*/
+ − if ( for_update_enh_cell_list( rr_data->act_ncell_list) EQ TRUE)
+ − {
+ − *p_cur = *p_temp;
+ − if (p_cur->ba2bsic_map_pending EQ 0) /*send parameters when there are no more mapping pending*/
+ − for_send_enh_para(p_cur);
+ − }
+ − /*Reset temporary, irrespective of whether updation is succesful or not*/
+ − for_set_default_emr_data(p_temp);
+ − }
+ − else if (p_cur->is_data_valid EQ TRUE AND p_cur->ba2bsic_map_pending EQ 0)
+ − for_send_enh_para(p_cur);
+ − }
+ − else
+ − {
+ − for_set_default_emr_data(p_temp);
+ − for_set_default_emr_data(p_cur);
+ − for_send_enh_para(p_cur);
+ − }
+ − return;
+ − }
+ − #endif
+ −
+ − /*
+ − +--------------------------------------------------------------------+
+ − | PROJECT : GSM-PS (6147) MODULE : RR_ATT |
+ − | STATE : code ROUTINE : att_update_std_band_indicator |
+ − +--------------------------------------------------------------------+
+ −
+ − PURPOSE : This function updates the std value depending upon the band
+ − indicator value received in the SI1 or SI6 message. band
+ − indicator will be ignored when received in the DCS 1800 or
+ − PCS 1900 frequency band. Depending upon the std value region
+ − will be updated.
+ −
+ − */
+ − #ifdef TI_PS_FF_QUAD_BAND_SUPPORT
+ − GLOBAL void att_update_std_band_indicator (UBYTE band_indicator)
+ − {
+ − GET_INSTANCE_DATA;
+ − UBYTE band_index = cs_get_band_index (rr_data->nc_data[SC_INDEX].arfcn);
+ − UBYTE freq_bands;
+ − UBYTE new_region = NOT_PRESENT_8BIT;
+ − UBYTE mscr = 0; /* variable to hold msc release version*/
+ −
+ − TRACE_FUNCTION ("att_update_std_band_indicator");
+ −
+ − get_msc_release_version(&mscr);
+ − TRACE_EVENT_P1("mscr (MSC release) version : 0x%X", mscr);
+ − if (mscr NEQ MSCR_99)
+ − {
+ − TRACE_EVENT ("band indicator received when MSC release is not R99");
+ − TRACE_ERROR ("band indicator received when MSC release is not R99");
+ − return;
+ − }
+ −
+ − if ((band_index EQ B_DCS_1800) OR (band_index EQ B_PCS_1900))
+ − {
+ − TRACE_EVENT_P2 ("band indicator %x received in band %x", band_indicator, band_index);
+ − return;
+ − }
+ −
+ − rr_csf_get_freq_bands (&freq_bands);
+ − if (band_indicator NEQ NOT_PRESENT_8BIT)
+ − {
+ − if (band_indicator EQ BAND_IND_1800)
+ − {
+ − if ((freq_bands & BAND_DCS_1800) EQ BAND_DCS_1800)
+ − {
+ − if (std EQ STD_DUAL_US)
+ − {
+ − std = STD_850_1800;
+ − new_region = BOTH_REGIONS;
+ − }
+ − else if (std EQ STD_850_900_1900)
+ − {
+ − std = STD_850_900_1800;
+ − new_region = BOTH_REGIONS;
+ − }
+ − }
+ − }
+ − else
+ − {
+ − if ((freq_bands & BAND_PCS_1900) EQ BAND_PCS_1900)
+ − {
+ − if (std EQ STD_DUAL_EGSM)
+ − {
+ − std = STD_900_1900;
+ − new_region = BOTH_REGIONS;
+ − }
+ − else if (std EQ STD_850_900_1800)
+ − {
+ − std = STD_850_900_1900;
+ − new_region = BOTH_REGIONS;
+ − }
+ − }
+ − }
+ − TRACE_EVENT_P1 ("Updated std to %x", std);
+ − }
+ −
+ − if (new_region NEQ NOT_PRESENT_8BIT)
+ − rr_data->cs_data.region = new_region;
+ − }
+ − #endif
+ −
+ − #endif /* RR_ATTF_C */