FreeCalypso > hg > fc-magnetite
view src/g23m-gsm/rr/rr_attg.c @ 228:d2cbdbffc528
aci2: FreeCalypso +CGSN logic implemented
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Wed, 16 Nov 2016 05:28:00 +0000 |
parents | 27a4235405c6 |
children |
line wrap: on
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 holds the functions for handling primitives | sent from entity GRR to entity RR and vice versa. +----------------------------------------------------------------------------- */ #ifndef RR_GPRS_C #define RR_GPRS_C #ifdef GPRS #define ENTITY_RR /*==== INCLUDES ===================================================*/ #include <string.h> #include <stddef.h> #include "typedefs.h" #include "pcm.h" #include "pconst.cdg" #include "mconst.cdg" #include "message.h" #include "vsi.h" #include "custom.h" #include "gsm.h" #include "prim.h" #include "pei.h" #include "tok.h" #include "rr_gprs.h" #include "rr.h" #include "rr_em.h" static void att_get_scell_info (T_serving_cell_info* sc, UBYTE idx); static void att_gprs_store_nc_mode(T_D_SYS_INFO_13* sys_info_13); #ifdef REL99 static void att_gprs_store_nw_release(T_D_SYS_INFO_13* sys_info_13); #endif static T_NC_DATA* is_nc_in_list(USHORT arfcn); static T_NC_DATA* find_empty_slot_in_list(void); static BOOL rr_ext_meas_can_start ( void ); /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_for_sysinfo_type13 | +--------------------------------------------------------------------+ PURPOSE : Checks GPRS availability of cell */ void att_for_sysinfo_type13 (T_MPH_UNITDATA_IND* data_ind, T_D_SYS_INFO_13* sys_info_13) { GET_INSTANCE_DATA; /* * get the storage area by checking the channel number */ UBYTE index = att_get_index (data_ind->arfcn); TRACE_FUNCTION("att_for_sysinfo_type13"); switch(GET_STATE(STATE_ATT)) { case ATT_CS2: /* * during cell selection * only CR_INDEX is taken in account */ if (index EQ CR_INDEX) { switch(GET_STATE(STATE_GPRS)) { case GPRS_ACTIVATED: case GPRS_PIM_BCCH: case GPRS_DEDI_SDCCH: if(sys_info_13->si13_rest_oct.v_si13_info AND sys_info_13->si13_rest_oct.si13_info.v_pbcch_des) { SET_STATE(STATE_GPRS, GPRS_PIM_PBCCH); } else { SET_STATE(STATE_GPRS, GPRS_PIM_BCCH); /* * check the NC mode, this is the default * value of the NW which can be overriden * by a Packet Measurement Order. * In CS2 we can always use the value provided * by the NW */ att_gprs_store_nc_mode(sys_info_13); } break; case GPRS_PIM_PBCCH: TRACE_EVENT("GPRS config changed during CS2!"); if(sys_info_13->si13_rest_oct.v_si13_info AND !sys_info_13->si13_rest_oct.si13_info.v_pbcch_des) { SET_STATE(STATE_GPRS, GPRS_PIM_BCCH); } break; default: break; } gprs_rrgrr_store_sdu(rr_data->gprs_data.si13_sdu, &data_ind->sdu); att_set_sys_info_read (SYS_INFO_13_READ, CR_INDEX); #ifdef REL99 att_gprs_store_nw_release(sys_info_13); #endif att_check_bcch_carrier_si (); } break; case ATT_CS3: /* * during cell reselection * only CR_INDEX is taken in account */ if (index EQ CR_INDEX) { switch(GET_STATE(STATE_GPRS)) { case GPRS_ACTIVATED: case GPRS_PIM_BCCH: if(sys_info_13->si13_rest_oct.v_si13_info AND sys_info_13->si13_rest_oct.si13_info.v_pbcch_des) { SET_STATE(STATE_GPRS, GPRS_PIM_PBCCH); } else { SET_STATE(STATE_GPRS, GPRS_PIM_BCCH); /* * check the NC mode, this is the default * value of the NW which can be overriden * by a Packet Measurement Order. * In CS3 we can always use the value provided * by the NW */ att_gprs_store_nc_mode(sys_info_13); } break; case GPRS_PIM_PBCCH: if(sys_info_13->si13_rest_oct.v_si13_info AND !sys_info_13->si13_rest_oct.si13_info.v_pbcch_des) { TRACE_EVENT("GPRS config changed during CS3!"); SET_STATE(STATE_GPRS, GPRS_PIM_BCCH); } break; default: break; } gprs_rrgrr_store_sdu(rr_data->gprs_data.si13_sdu, &data_ind->sdu); att_set_sys_info_read (SYS_INFO_13_READ, CR_INDEX); #ifdef REL99 att_gprs_store_nw_release(sys_info_13); #endif att_check_neighbourcell_si_reestab(); } break; case ATT_CON_EST: case ATT_IDLE: switch (index) { case SC_INDEX: /* now we want to send the SI13 to GRR */ if(/*rr_data->ms_data.rr_service EQ LIMITED_SERVICE OR*/ !att_gprs_is_avail()) return; switch(GET_STATE(STATE_GPRS)) { case GPRS_ACTIVATED: if((rr_data->sc_data.selection_type EQ CELL_RESELECTION_ON_GPRS_ACT) AND (rr_data->sc_data.cd.sys_info_read EQ ALL_SYS_INFO_READ)) { TRACE_EVENT(" CELL_RESELECTION_ON_GPRS_ACT should end"); TIMERSTOP(T_RESELECT); } if(sys_info_13->si13_rest_oct.v_si13_info AND sys_info_13->si13_rest_oct.si13_info.v_pbcch_des) { if(rr_data->sc_data.cd.sys_info_read EQ ALL_SYS_INFO_READ) { att_gprs_stop_pl(); } SET_STATE(STATE_GPRS, GPRS_PIM_PBCCH); } else { SET_STATE(STATE_GPRS, GPRS_PIM_BCCH); if(rr_data->sc_data.cd.sys_info_read EQ ALL_SYS_INFO_READ) { att_build_idle_req(SC_INDEX, MODE_SYS_INFO_CHANGE); } /* * check the NC mode, this is the default * value of the NW which can be overriden * by a Packet Measurement Order. * */ att_gprs_store_nc_mode(sys_info_13); } break; case GPRS_PIM_BCCH: case GPRS_PTM_BCCH: /* * check the NC mode, this is the default * value of the NW which can be overriden * by a Packet Measurement Order. * */ att_gprs_store_nc_mode(sys_info_13); /* XXX */ /* * Right now it not clear what how the activation of PBCCH * should be handled in RR/GRR */ if (GET_STATE(STATE_GPRS) EQ GPRS_PTM_BCCH) break; if(sys_info_13->si13_rest_oct.v_si13_info AND sys_info_13->si13_rest_oct.si13_info.v_pbcch_des) { att_gprs_stop_pl(); SET_STATE(STATE_GPRS, GPRS_PIM_PBCCH); } break; case GPRS_PIM_PBCCH: if(sys_info_13->si13_rest_oct.v_si13_info AND !sys_info_13->si13_rest_oct.si13_info.v_pbcch_des) { /*XXX not possible actually */ SET_STATE(STATE_GPRS, GPRS_PIM_BCCH); } break; default: break; } /* * GRR requests reading of SI13 every 30 seconds * pass the SI13 to GRR, for further evaluation */ TRACE_EVENT("idle sc"); TRACE_EVENT_P1("sys_info_read %d", rr_data->sc_data.cd.sys_info_read); gprs_rrgrr_store_sdu(rr_data->gprs_data.si13_sdu, &data_ind->sdu); att_set_sys_info_read (SYS_INFO_13_READ, SC_INDEX); #ifdef REL99 att_gprs_store_nw_release(sys_info_13); #endif att_signal_gprs_support(); break; case NOT_PRESENT_8BIT: /* * unexpected channel number */ break; default: /* * neighbour cell, ignore. * si13 info is not needed, at first * maybe get PBCCH info later */ break; } break; default: break; } } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS () MODULE : RR_GPRS | | STATE : code ROUTINE : att_set_gprs_indication | +-----------------------------------------------------------------------------+ PURPOSE : sets a flag indicating whether GPRS is wanted by upper layer or not. */ void att_set_gprs_indication (UBYTE gprs_indic) { GET_INSTANCE_DATA; TRACE_FUNCTION ("att_set_gprs_indication()"); /* * store parameter from MM */ rr_data->gprs_data.gprs_indic = gprs_indic; if(rr_data->gprs_data.gprs_indic) { SET_STATE(STATE_GPRS, GPRS_ACTIVATED); TRACE_EVENT("MM wants GPRS"); } else { SET_STATE(STATE_GPRS, GPRS_NULL); rr_data->gprs_data.ready_state = FALSE; /* Standby */ } } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_check_gprs_supp | +--------------------------------------------------------------------+ PURPOSE : Checks GPRS availability of cell and sets the internal flag */ void att_check_gprs_supp (UBYTE v_gprs_ind, T_gprs_indic * data) { GET_INSTANCE_DATA; TRACE_FUNCTION ("att_check_gprs_supp ()"); /* * if rest octets for GPRS are available * and GPRS support is requested by MM */ switch(GET_STATE(STATE_GPRS)) { case GPRS_NULL: if(GET_STATE(STATE_ATT) EQ ATT_CS2 OR GET_STATE(STATE_ATT) EQ ATT_CS3) att_set_sys_info_read(SYS_INFO_13_READ, CR_INDEX); else att_set_sys_info_read(SYS_INFO_13_READ, SC_INDEX); break; case GPRS_ACTIVATED: if(GET_STATE(STATE_ATT) EQ ATT_CS2 OR GET_STATE(STATE_ATT) EQ ATT_CS3) { if(v_gprs_ind) { SET_STATE(STATE_GPRS, GPRS_PIM_BCCH); /* * If System Info 13 is send on extended BCCH, * request reading, else it is reading on normal BCCH */ if(data->si13_pos EQ SI13_ON_BCCH_EXT) { PALLOC(mon_ctrl, MPH_MON_CTRL_REQ); mon_ctrl->action = START_MON_EBCCH; mon_ctrl->si_to_read = UPDATE_SI13; PSENDX(PL, mon_ctrl); } /* else we will read it on NBCCH */ } else { /* no state change */ att_set_sys_info_read(SYS_INFO_13_READ, CR_INDEX); } } /* * when GPRS is activated in idle this will be handled * when the SI13 is received */ break; case GPRS_PIM_BCCH: case GPRS_PIM_PBCCH: if(!v_gprs_ind) { SET_STATE(STATE_GPRS, GPRS_ACTIVATED); att_set_sys_info_read(SYS_INFO_13_READ, SC_INDEX); if(GET_STATE(STATE_ATT) EQ ATT_IDLE) { /*XY:n inform GRR, and wait for CR_RSP */ att_rrgrr_cr_ind(CR_ABNORMAL); rr_data->gprs_data.start_proc = START_PROC_CFG_CHG; } } else { /* * If System Info 13 is send on extended BCCH, * request reading, else it is reading on normal BCCH */ if(data->si13_pos EQ SI13_ON_BCCH_EXT) { PALLOC(mon_ctrl, MPH_MON_CTRL_REQ); mon_ctrl->action = START_MON_EBCCH; mon_ctrl->si_to_read = UPDATE_SI13; PSENDX(PL, mon_ctrl); } /* else we will read it on NBCCH */ } break; default: break; } } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_signal_gprs_support | +--------------------------------------------------------------------+ PURPOSE : Signal GPRS support to GRR */ void att_signal_gprs_support (void) { GET_INSTANCE_DATA; PALLOC_SDU(rrgrr_gprs_si13_ind, RRGRR_GPRS_SI13_IND, MAX_L2_FRAME_SIZE * BITS_PER_BYTE); TRACE_FUNCTION ("att_signal_gprs_support()"); /* * gprs_support : Y Y X N N * rr_service : F L N F L * sdu : Y Y - - - * servng_cell_info: */ if(GET_STATE(STATE_ATT) EQ ATT_IDLE AND rr_data->sc_data.cd.sys_info_read NEQ ALL_SYS_INFO_READ) { PFREE(rrgrr_gprs_si13_ind); return; } if((GET_STATE(STATE_GPRS) NEQ GPRS_ACTIVATED AND GET_STATE(STATE_GPRS) NEQ GPRS_NULL) AND !(rr_data->ms_data.rr_service EQ NO_SERVICE) ) { rrgrr_gprs_si13_ind->cause = GPRS_SUPPORTED; /* * copy message content */ gprs_rrgrr_fill_from_stored_sdu(&rrgrr_gprs_si13_ind->sdu, rr_data->gprs_data.si13_sdu); TRACE_EVENT_P4 ("cs=%u C[%d] bsic=%u #%u", rrgrr_gprs_si13_ind->cause, rrgrr_gprs_si13_ind->serving_cell_info.bcch_arfcn, rrgrr_gprs_si13_ind->serving_cell_info.bcch_bsic, __LINE__); /* * set system information state */ rrgrr_gprs_si13_ind->si_states.si1_state = SI1_RECEIVED; rrgrr_gprs_si13_ind->si_states.si3_state = SI3_RECEIVED; rrgrr_gprs_si13_ind->si_states.si13_state = SI13_RECEIVED; } else { rrgrr_gprs_si13_ind->cause = GPRS_NOT_SUPPORTED; } /*When BA (BCCH) list is modified by PMO/PCCO, this is just for RR, RR should not inform GRR as it would have already formed the list. Sending this new list will change all the index, as GRR would assume it to be new BA(BCCH). When New BA(BCCH) is received then this flag would be reset and GRR should update the BA list*/ if(rr_data->gprs_data.ba_bcch_modified AND (GET_STATE(STATE_ATT) NEQ ATT_CS3)) { rrgrr_gprs_si13_ind->serving_cell_info.arfcn[0]= RRGRR_INVALID_ARFCN; } else { memcpy(rrgrr_gprs_si13_ind->serving_cell_info.arfcn, rr_data->act_ncell_list, sizeof(rrgrr_gprs_si13_ind->serving_cell_info.arfcn)); } att_get_scell_info(&rrgrr_gprs_si13_ind->serving_cell_info, SC_INDEX); /* we have found a cell, reset cr flag*/ rr_data->gprs_data.cr_pbcch_active = FALSE; #if defined (REL99) AND defined (TI_PS_FF_EMR) rrgrr_gprs_si13_ind->serving_cell_info.ba_ind = rr_data->sc_data.ba_index; #endif PSENDX(GRR, rrgrr_gprs_si13_ind); } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_add_ptmsi | +-----------------------------------------------------------------------------+ PURPOSE : Add the ptmsi if GPRS is supported. */ void att_add_ptmsi (T_MPH_IDENTITY_REQ * mph_identity_req) { GET_INSTANCE_DATA; TRACE_FUNCTION ("att_add_ptmsi()"); /* * check availability of GPRS */ switch (GET_STATE(STATE_GPRS)) { case GPRS_NULL: mph_identity_req->mid.v_ptmsi = FALSE; mph_identity_req->mid.ptmsi = 0L; mph_identity_req->mid.v_ptmsi2 = FALSE; mph_identity_req->mid.ptmsi2 = 0L; break; default: if (rr_data->gprs_data.ptmsi EQ 0xFFFFFFFF OR rr_data->gprs_data.ptmsi EQ 0L) { mph_identity_req->mid.v_ptmsi = FALSE; mph_identity_req->mid.ptmsi = 0L; } else { mph_identity_req->mid.v_ptmsi = TRUE; mph_identity_req->mid.ptmsi = rr_data->gprs_data.ptmsi; } /*the candidate PTMSI has to be checked also*/ if (rr_data->gprs_data.ptmsi2 EQ 0xFFFFFFFF OR rr_data->gprs_data.ptmsi2 EQ 0L) { mph_identity_req->mid.v_ptmsi2 = FALSE; mph_identity_req->mid.ptmsi2 = 0L; } else { mph_identity_req->mid.v_ptmsi2 = TRUE; mph_identity_req->mid.ptmsi2 = rr_data->gprs_data.ptmsi2; } break; } } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_rrgrr_cr_ind | +-----------------------------------------------------------------------------+ PURPOSE : IN : OUT : */ void att_rrgrr_cr_ind (UBYTE type) { GET_INSTANCE_DATA; /* * Following cases are possible: * BCCH always CR_ABNORMAL * after con est fail, after dedi, normal cr in idle and ptm, * plmn search,activate req, cell selection * PBCCH: * after con est fail(CR_REQ_CANDIDATE), * activate_req (CR_NORMAL) * after dedi (CR_ABNORMAL) * after imm_ass_rej (CR_SUSPENDED_IDLE) */ if(GET_STATE(STATE_GPRS) NEQ GPRS_NULL) { PALLOC(cr_ind, RRGRR_CR_IND); cr_ind->cr_type = type; TRACE_EVENT("cr_ind"); PSENDX(GRR, cr_ind); } else { TRACE_EVENT("cr already active"); } } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_rrgrr_cr_req | +-----------------------------------------------------------------------------+ PURPOSE : */ void att_rrgrr_cr_req (T_RRGRR_CR_REQ *cr_req) { GET_INSTANCE_DATA; int gprs_state; TRACE_FUNCTION ("gprs_rrgrr_cell_reselection_req ()"); gprs_state = GET_STATE(STATE_GPRS); TRACE_EVENT_P3 ("RRGRR_CR_REQ %u [%u] st=%u", cr_req->cr_type, cr_req->arfcn, gprs_state); TRACE_EVENT_P6 ( "r MCC/MNC=%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]); switch(gprs_state) { case GPRS_PAM_PBCCH: case GPRS_ACTIVATED: case GPRS_PIM_PBCCH: case GPRS_PTM_PBCCH: case GPRS_SUSPENDED_PBCCH: switch(cr_req->cr_type) { case CR_NEW: { UBYTE i; UBYTE found=0; for(i=0; i<6;i++) { if(rr_data->nc_data[i].arfcn EQ cr_req->arfcn AND rr_data->nc_data[i].bsic EQ cr_req->bsic) { found=1; break; } } if(!found) { TRACE_ERROR("cr_req(NEW): data does not match");} memset (&rr_data->nc_data[CR_INDEX], 0, sizeof (T_NC_DATA)); rr_data->gprs_data.cr_pbcch_active = TRUE; rr_data->nc_data[CR_INDEX].arfcn = cr_req->arfcn; rr_data->nc_data[CR_INDEX].bsic = cr_req->bsic; if (rr_data->ms_data.req_mm_service EQ FUNC_NET_SRCH_BY_MMI) { /*inform MM and stop everything*/ /* * send Please Retry to the MMI */ rr_data->sc_data.found_entries = 0; att_code_rr_abort_ind (RRCS_ABORT_CEL_SEL_FAIL); } if( ! rr_data->gprs_data.cr_pcco_active ) att_start_cell_reselection_pbcch(MODE_CELL_RESELECTION); else { att_init_cell_selection(CELL_RESELECTION, RR_ORIGINATED); 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); gprs_init_data_cr(); dat_att_null(); SET_STATE (STATE_ATT, ATT_CS3); att_build_idle_req (CR_INDEX, MODE_CELL_RESELECTION); } break; } case CR_NEW_NOT_SYNCED: { /* * a Packet Cell Change Order for a cell which * is not in the BA list has been requested by the NW */ PALLOC (mph_bsic_req, MPH_BSIC_REQ); { PALLOC (mph_sync_req, MPH_SYNC_REQ); mph_sync_req->cs = CS_START_PCCO; PSENDX (PL, mph_sync_req); } TRACE_EVENT("cco w/ not synced"); /* * remember we got a CR_REQ, will be used * on reception of the BSIC_CNF and the UNITDATA_IND */ rr_data->gprs_data.cr_pbcch_active = TRUE; SET_STATE (STATE_CELL_SEL, CS_CCO); /* avoid reacting to PAGINGs and ESTABLISH_REQ */ dat_att_null(); SET_STATE (STATE_ATT, ATT_CS3); cs_set_all(); rr_data->gprs_data.bsic = cr_req->bsic; rr_data->gprs_data.arfcn = mph_bsic_req->arfcn = cr_req->arfcn; PSENDX (PL, mph_bsic_req); } break; case CR_CS: if (rr_data->gprs_data.cr_pcco_active EQ FALSE) rr_data->gprs_data.cr_pbcch_active = TRUE; else rr_data->gprs_data.cr_pcco_active = FALSE; if (rr_data->ms_data.req_mm_service EQ FUNC_NET_SRCH_BY_MMI) { /*inform MM and stop everything*/ /* * send Please Retry to the MMI */ rr_data->sc_data.found_entries = 0; att_code_rr_abort_ind (RRCS_ABORT_CEL_SEL_FAIL); } /*XY:n don't inform GRR */ att_start_cell_selection(RR_ORIGINATED, CS_NOT_PARALLEL,NORMAL_SEARCH_MODE); break; case CR_CONT: att_continue_cell_reselect (); break; case CR_COMPLETE: /* * we have a PBCCH and GRR has read it * we inform MM now */ if( rr_data->gprs_data.cr_pcco_active EQ FALSE ) { PALLOC(mph_mon_ctrl_req, MPH_MON_CTRL_REQ ); mph_mon_ctrl_req->action = ENTER_PIM_PBCCH; PSENDX (PL, mph_mon_ctrl_req); } rr_data->gprs_data.cr_pcco_active = FALSE; /* initialize the ba_bcch_modified value when Cell reselection is done */ rr_data->gprs_data.ba_bcch_modified= FALSE; dat_att_cell_selected (); SET_STATE (STATE_ATT, ATT_IDLE); dat_send_bcchinfo_mm (rr_data->cr_data.cr_white_list.si2); if(rr_data->ms_data.rr_service EQ LIMITED_SERVICE AND rr_data->ms_data.req_mm_service EQ FUNC_PLMN_SRCH) { 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 att_code_rr_abort_ind (RRCS_ABORT_CEL_SEL_FAIL); } else { if (rr_data->sc_data.mm_started) att_code_rr_act_cnf (); else att_code_rr_act_ind (); } /* * This instruction disdurbs test case 20.7 * unclear why it is coded here * rr_data->first_meas_received = TRUE; TRACE_EVENT("first_meas_received set to TRUE(rr_attg.c)"); */ /* * Start registration timer if needed */ att_start_registration_timer (); att_mph_identity_req (); #ifdef REL99 if(rr_data->ms_data.rr_service EQ FULL_SERVICE) att_config_cbch (); #else if(rr_data->ms_data.rr_service EQ FULL_SERVICE) att_build_cbch (); #endif srv_use_stored_prim (); /* * 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); } break; default: break; } break; case GPRS_PTM_BCCH: if( (cr_req->cr_type EQ CR_NEW) || (cr_req->cr_type EQ CR_NEW_NOT_SYNCED)) { UBYTE i; UBYTE found=0; for(i=0; i<6;i++) { if(rr_data->nc_data[i].arfcn EQ cr_req->arfcn AND rr_data->nc_data[i].bsic EQ cr_req->bsic) { found=1; break; } } memset (&rr_data->nc_data[CR_INDEX], 0, sizeof (T_NC_DATA)); rr_data->nc_data[CR_INDEX].arfcn = cr_req->arfcn; rr_data->nc_data[CR_INDEX].bsic = cr_req->bsic; if (rr_data->ms_data.req_mm_service EQ FUNC_NET_SRCH_BY_MMI) { /*inform MM and stop everything*/ /* * send Please Retry to the MMI */ rr_data->sc_data.found_entries = 0; att_code_rr_abort_ind (RRCS_ABORT_CEL_SEL_FAIL); } rr_data->gprs_data.cr_pcco_active = TRUE; if(!found) { /* * a Packet Cell Change Order for a cell which * is not in the BA list has been requested by the NW */ PALLOC (mph_bsic_req, MPH_BSIC_REQ); { PALLOC (mph_sync_req, MPH_SYNC_REQ); mph_sync_req->cs = CS_START_PCCO; PSENDX (PL, mph_sync_req); } TRACE_EVENT("cco w/ not synced"); SET_STATE (STATE_CELL_SEL, CS_CCO); /* avoid reacting to PAGINGs and ESTABLISH_REQ */ dat_att_null(); SET_STATE (STATE_ATT, ATT_CS3); cs_set_all(); rr_data->gprs_data.bsic = cr_req->bsic; rr_data->gprs_data.arfcn = mph_bsic_req->arfcn = cr_req->arfcn; PSENDX (PL, mph_bsic_req); } else { att_init_cell_selection(CELL_RESELECTION, RR_ORIGINATED); 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); gprs_init_data_cr(); dat_att_null(); SET_STATE (STATE_ATT, ATT_CS3); att_build_idle_req (CR_INDEX, MODE_CELL_RESELECTION); } } break; case GPRS_PIM_BCCH: /* * we have a BCCH * we have informed GRR of this (with additional info that we * have Full or Limited service) * we inform MM now */ switch(cr_req->cr_type) { case CR_COMPLETE: case CR_NEW: case CR_NEW_NOT_SYNCED: case CR_CS: case CR_CONT: dat_att_cell_selected (); SET_STATE (STATE_ATT, ATT_IDLE); dat_send_bcchinfo_mm (rr_data->cr_data.cr_white_list.si2); if(rr_data->ms_data.rr_service EQ LIMITED_SERVICE AND rr_data->ms_data.req_mm_service EQ FUNC_PLMN_SRCH) { 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 att_code_rr_abort_ind (RRCS_ABORT_CEL_SEL_FAIL); } else { if (rr_data->sc_data.mm_started) att_code_rr_act_cnf (); else att_code_rr_act_ind (); } /* * Start registration timer if needed */ att_start_registration_timer (); att_mph_identity_req (); srv_use_stored_prim (); /* * 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); } break; default: break; } break; case GPRS_PAM_BCCH: TRACE_EVENT("CR during PA"); break; default: break; } PFREE (cr_req); } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_rrgrr_stop_mon_ccch_req | +-----------------------------------------------------------------------------+ PURPOSE : Process the primitive RRGRR_STOP_MON_CCCH_REQ received from GRR. This primitive indicates to the RR that the monitoring of CCCH should be stopped. This is only a signal, no parameters needed. This signal is forwarded to ALR. */ /* * This is reqiured for Testmode A */ /*lint -esym(526,grlc_test_mode_active) : not defined | defined in GRR */ EXTERN UBYTE grlc_test_mode_active(void); void att_rrgrr_stop_mon_ccch_req (T_RRGRR_STOP_MON_CCCH_REQ* stop_mon) { GET_INSTANCE_DATA; TRACE_FUNCTION ("att_rrgrr_stop_mon_ccch_req ()"); switch(GET_STATE(STATE_GPRS)) { /* * This case is necessary to stop ccch monitoring in STATE_GPRS=GPRS_PTM * during Testmode A/B */ case GPRS_PTM_BCCH: if(grlc_test_mode_active()) { PALLOC(mph_mon_ctrl_req, MPH_MON_CTRL_REQ ); mph_mon_ctrl_req->action = STOP_MON_CCCH; PSENDX (PL, mph_mon_ctrl_req); TRACE_EVENT("STOP_MON_CCCH during TESTMODE !!"); } break; case GPRS_PIM_BCCH: case GPRS_PIM_PBCCH: case GPRS_PAM_PBCCH: case GPRS_PTM_PBCCH: case GPRS_PAM_BCCH: TRACE_EVENT("stop ccch"); break; default: break; } PFREE (stop_mon); } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_rrgrr_start_mon_ccch_req | +-----------------------------------------------------------------------------+ PURPOSE : Process the primitive RRGRR_START_MON_CCCH_REQ received from GRR. This primitive indicates to the RR that the monitoring of CCCH should be started. This is only a signal, no parameters needed. This signal is forwarded to ALR. */ void att_rrgrr_start_mon_ccch_req (T_RRGRR_START_MON_CCCH_REQ* mon_req) { GET_INSTANCE_DATA; TRACE_EVENT ("att_rrgrr_start_mon_ccch_req ()"); /* * store split paging cycle */ if (mon_req->split_pg EQ RRGRR_NO_DRX) rr_data->gprs_data.split_pg=256; else rr_data->gprs_data.split_pg=mon_req->split_pg; switch(GET_STATE(STATE_GPRS)) { case GPRS_PIM_BCCH: TRACE_EVENT("GPRS_PIM_BCCH"); if(mon_req->pag_mode EQ PAG_MODE_DEFAULT AND rr_data->gprs_data.page_mode EQ PAG_MODE_REORG) { /* * go back to normal page mode */ rr_data->gprs_data.page_mode = PAG_MODE_DEFAULT; att_build_idle_req (SC_INDEX, MODE_SYS_INFO_CHANGE); } if(mon_req->pag_mode EQ PAG_MODE_PTM_NP) { /* * we may have received dl assignment in reog idle mode * we shopuld directly switch to transfer mode with: * NORMAL_PAGING */ SET_STATE(STATE_GPRS, GPRS_PTM_BCCH); SET_STATE(STATE_ATT, ATT_IDLE); SET_STATE(STATE_DAT, DAT_IDLE); att_build_idle_req(SC_INDEX, MODE_PACKET_TRANSFER); { PALLOC(mon_ctrl, MPH_MON_CTRL_REQ); mon_ctrl->action = START_MON_CCCH; mon_ctrl->si_to_read = 0; PSENDX(PL, mon_ctrl); } } break; case GPRS_PTM_BCCH: /* * set page_mode to REORG if NON_DRX_TIMER is used * set page_mode to DEFAULT if NON_DRX_TIMER is not used or 0 */ TRACE_EVENT("GPRS_PTM"); if(mon_req->pag_mode EQ PAG_MODE_PTM_NP) { PALLOC(mon_ctrl, MPH_MON_CTRL_REQ); mon_ctrl->action = START_MON_CCCH; mon_ctrl->si_to_read = 0; PSENDX(PL, mon_ctrl); } else if(mon_req->pag_mode EQ PAG_MODE_PIM_NP) { /* * Go into idle mode but start ONLY CCCH reading with the following order: * start with REORG and then switch to NORMAL paging */ TRACE_EVENT("PTM->IDLE: CCCH reorg+normal"); } else { rr_data->gprs_data.page_mode = mon_req->pag_mode; SET_STATE (STATE_GPRS, GPRS_PIM_BCCH); att_return_to_idle(); /* Transition from PTM to PIM state. PBCCH is not present in the * cell. Check if the previous PTM state was used for 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_GPRS_1P) AND (rr_data->cs_data.black_list_search_pending EQ TRUE)) { 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 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); } #endif else { #ifdef TI_PS_FF_QUAD_BAND_SUPPORT 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 #endif { TRACE_EVENT("Black List empty"); } } } } } break; case GPRS_PAM_BCCH: rr_data->gprs_data.page_mode = mon_req->pag_mode; SET_STATE (STATE_GPRS, GPRS_PIM_BCCH); att_return_to_idle(); break; case GPRS_PAM_PBCCH: //break; case GPRS_PTM_PBCCH: case GPRS_PIM_PBCCH: { PALLOC(mon_ctrl, MPH_MON_CTRL_REQ); mon_ctrl->action = START_MON_CCCH; mon_ctrl->si_to_read = 0; PSENDX(PL, mon_ctrl); SET_STATE(STATE_DAT, DAT_IDLE); /* allow reception of pagings */ } break; default: TRACE_EVENT("start_mon_ccch_req not handled"); break; } PFREE (mon_req); } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_rrgrr_start_mon_bcch_req | +-----------------------------------------------------------------------------+ PURPOSE : Process the primitive RRGRR_START_MON_BCCH_REQ received from GRR. This primitive indicates to the RR to read SI. This primitive contains the parameter which indicates to the SI to read on BCCH. This signal is forwarded to ALR. */ void att_rrgrr_start_mon_bcch_req (T_RRGRR_START_MON_BCCH_REQ *rrgrr_start_mon_bcch_req) { GET_INSTANCE_DATA; TRACE_EVENT ("att_rrgrr_start_mon_bcch_req ()"); switch (GET_STATE(STATE_GPRS)) { case GPRS_PIM_BCCH: case GPRS_PIM_PBCCH: case GPRS_PAM_PBCCH: case GPRS_PTM_PBCCH: case GPRS_PTM_BCCH: { UBYTE si_to_read = rrgrr_start_mon_bcch_req->si_to_read; USHORT sys_info_read = rr_data->sc_data.cd.sys_info_read; PALLOC ( mph_mon_ctrl_req, MPH_MON_CTRL_REQ ); TRACE_EVENT_P1 ( "rrgrr_start_mon_bcch_req->si_to_read = %d", rrgrr_start_mon_bcch_req->si_to_read ); mph_mon_ctrl_req->action = START_MON_NBCCH; switch ( si_to_read ) { case UNSPECIFIED_SI: sys_info_read = 0; break; case UPDATE_SI1: sys_info_read &= ~ SYS_INFO_1_READ; break; case UPDATE_SI2_SI2BIS_OR_SI2TER: sys_info_read &= ~ (SYS_INFO_2_READ | SYS_INFO_2BIS_READ | SYS_INFO_2TER_READ); #if defined (REL99) AND defined (TI_PS_FF_EMR) /*Once SI-2,2Bis,2Ter are read, we can configure ALR for acquiring SI-2quater*/ if ( (rr_data->sc_data.cd.si2quater_status NEQ SI2QUATER_ABSENT ) AND (rr_data->sc_data.cd.si2quater_pos EQ SI2QUATER_ON_EBCCH) ) rr_data->sc_data.cd.si2quater_status = SI2QUATER_CONFIGURE; #endif break; case UPDATE_SI3_SI4_SI7_OR_SI8: sys_info_read &= ~ (SYS_INFO_3_READ | SYS_INFO_4_READ ); break; case UPDATE_SI9: /* no support of SI 9 */ break; case COMPLETE_SI: sys_info_read = 0; break; case UPDATE_SI13: sys_info_read &= ~ SYS_INFO_13_READ; if(rr_data->nc_data[SC_INDEX].si13_loc_ind EQ EXTENDED_BCCH) { mph_mon_ctrl_req->action = START_MON_EBCCH; } break; default: break; } rr_data->sc_data.cd.sys_info_read = sys_info_read; TRACE_EVENT_P1("st_bcch:sys_info_read %d", rr_data->sc_data.cd.sys_info_read); mph_mon_ctrl_req->si_to_read = si_to_read; PSENDX (PL, mph_mon_ctrl_req); } break; case GPRS_ACTIVATED: if(GET_STATE(STATE_ATT) EQ ATT_IDLE) att_signal_gprs_support(); break; default: break; } PFREE (rrgrr_start_mon_bcch_req); } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_convert_idle_c31_cr | +-----------------------------------------------------------------------------+ PURPOSE : */ GLOBAL void att_convert_idle_c31_cr(UBYTE index) { GET_INSTANCE_DATA; TRACE_EVENT_P2("att_convert_idle_c31_cr called for index=%d conversion = %d", index, rr_data->nc_data[index].v_cr_par); switch(rr_data->nc_data[index].v_cr_par) { case CR_PAR_VALID: /* * For the first neighbour cell in the message where the struct exists, * the following default values shall be used for missing parameters: * GPRS_RXLEV_ACCESS_MIN = Serving cell RXLEV_ACCESS_MIN * GPRS_MS_TXPWR_MAX_CCH = Serving cell MS_TXPWR_MAX_CCH * GPRS_TEMPORARY_OFFSET = Serving cell TEMPORARY_OFFSET * GPRS_PENALTY_TIME = Serving cell PENALTY_TIME * GPRS_RESELECT_OFFSET = 0 * HCS_THR = infinity * PRIORITY_CLASS = undefined * SI13_PBCCH_LOCATION = undefined * */ if( rr_data->nc_data[index].cr_par.gprs_rxlev_access_min EQ GPRS_RXLEV_ACCESS_MIN_INVALID) rr_data->nc_data[index].cr_par.gprs_rxlev_access_min = rr_data->nc_data[SC_INDEX].select_para.rxlev_access_min; if(rr_data->nc_data[index].cr_par.gprs_ms_txpwr_max_cch EQ GPRS_MS_TXPWR_MAX_CCH_INVALID) rr_data->nc_data[index].cr_par.gprs_ms_txpwr_max_cch = rr_data->nc_data[SC_INDEX].select_para.ms_txpwr_max_cch; if(rr_data->nc_data[index].cr_par.gprs_temporary_offset EQ GPRS_TEMPORARY_OFFSET_INVALID) rr_data->nc_data[index].cr_par.gprs_temporary_offset = rr_data->nc_data[SC_INDEX].c2_par.temp_offset; if(rr_data->nc_data[index].cr_par.gprs_penalty_time EQ GPRS_PENALTY_TIME_INVALID) rr_data->nc_data[index].cr_par.gprs_penalty_time = rr_data->nc_data[SC_INDEX].c2_par.penalty_time * 2; if(rr_data->nc_data[index].cr_par.gprs_reselect_offset EQ GPRS_RESEL_OFF_INVALID) rr_data->nc_data[index].cr_par.gprs_reselect_offset = GPRS_RESEL_OFF_DEFAULT; /* * 31 is the maximum 5 bit value. * hcs_thr = 31 , means infinity */ if(rr_data->nc_data[index].cr_par.hcs_thr EQ GPRS_HCS_THR_INVALID) { rr_data->nc_data[index].cr_par.hcs_thr = 31; } /* priority class remains undefined */ break; case CR_PAR_INVALID: /* * initialise values * conversion of idle mode parameters to GPRS cell reselection parameters * GPRS_RXLEV_ACCESS_MIN = RXLEV_ACCESS_MIN * GPRS_MS_TXPWR_MAX_CCH = MS_TXPWR_MAX_CCH * C31 = 0 * GPRS_RESELECT_OFFSET(n) = * CELL_RESELECT_OFFSET(n) - CELL_RESELECT_OFFSET(s) * GPRS_TEMPORARY_OFFSET = TEMPORARY OFFSET * GPRS_PENALTY_TIME = PENALTY_TIME * PRIORITY_CLASS = 0 * C32_QUAL = 0 * GPRS_CELL_RESELECT_HYSTERESIS = CELL_RESELECT_HYSTERESIS * RA_RESELECT_HYSTERESIS = CELL_RESELECT_HYSTERESIS */ rr_data->nc_data[index].cr_par.gprs_rxlev_access_min = rr_data->nc_data[index].select_para.rxlev_access_min; rr_data->nc_data[index].cr_par.gprs_ms_txpwr_max_cch = rr_data->nc_data[index].select_para.ms_txpwr_max_cch; /* * NOTE: If PENALTY_TIME = 11111 for a cell, * the sign of CELL_RESELECT_OFFSET shall be changed * and TEMPORARY OFFSET set to 0 for that cell. */ if( rr_data->nc_data[index].c2_par.penalty_time EQ 31) { rr_data->nc_data[index].cr_par.gprs_temporary_offset = GPRS_RESEL_OFF_DEFAULT; } else { rr_data->nc_data[index].cr_par.gprs_temporary_offset = rr_data->nc_data[index].c2_par.temp_offset; } rr_data->nc_data[index].cr_par.priority_class = 0; rr_data->nc_data[index].cr_par.gprs_penalty_time = rr_data->nc_data[index].c2_par.penalty_time; rr_data->nc_data[index].cr_par.gprs_penalty_time *= 2; if( rr_data->nc_data[index].bcch_status NEQ DECODED ) TRACE_EVENT_P1("Error BCCH in NOT DECODED for arfcn %d in att_convert_idle_cr" , rr_data->nc_data[index].arfcn); break; default : break; } } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6147) MODULE : RR_ATT | | STATE : code ROUTINE : att_insert_c31_cr_data_in_cell | +--------------------------------------------------------------------+ PURPOSE : To store T_cr_par in rr_data->nc_data */ GLOBAL void att_insert_c31_cr_data_in_cell (UBYTE index) { GET_INSTANCE_DATA; UBYTE i; TRACE_EVENT_P1("att_insert_c31_cr_data_in_cell called for arfcn=%d" , rr_data->nc_data[index].arfcn ); rr_data->nc_data[index].v_cr_par = CR_PAR_INVALID; for( i = 0; i < rr_data->gprs_data.num_add_freq_list; i++ ) { if( rr_data->nc_data[index].arfcn EQ rr_data->gprs_data.add_freq_lists[i].arfcn ) { if( rr_data->gprs_data.add_freq_lists[i].v_cr_par ) { TRACE_EVENT_P2(" Cr_par found for arfcn =%d, index in add freq list = %d", rr_data->nc_data[index].arfcn, i); rr_data->nc_data[index].v_cr_par = CR_PAR_VALID; memcpy( & (rr_data->nc_data[index].cr_par), & (rr_data->gprs_data.add_freq_lists[i].cr_par), sizeof( T_cr_par) ); att_convert_idle_c31_cr(index); } else { TRACE_EVENT_P1("Cr_par found for arfcn =%d in add freq list", rr_data->nc_data[index].arfcn); if( rr_data->nc_data[index].bcch_status EQ DECODED) att_convert_idle_c31_cr(index); } return; } } if( rr_data->nc_data[index].bcch_status EQ DECODED) att_convert_idle_c31_cr(index); } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_init_c31_cr_par | +-----------------------------------------------------------------------------+ PURPOSE : */ static void att_init_nc_data_c31_cr_par(void) { GET_INSTANCE_DATA; UBYTE index = 0; TRACE_EVENT(" att_init_nc_data_c31_cr_par called"); for(index = 0; index <= SC_INDEX; index++) { rr_data->nc_data[index].v_cr_par = CR_PAR_INVALID; if( rr_data->nc_data[index].bcch_status NEQ EMPTY ) { att_insert_c31_cr_data_in_cell(index); } } } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_init_cr_c31_data | +-----------------------------------------------------------------------------+ PURPOSE : */ static void att_init_cr_c31_data(void) { GET_INSTANCE_DATA; UBYTE i = 0; UBYTE first_valid_par = 0; UBYTE valid_found = FALSE; for( ; (i < rr_data->gprs_data.num_add_freq_list) AND (valid_found EQ FALSE) ; i++) { if( rr_data->gprs_data.add_freq_lists[i].v_cr_par) { valid_found = TRUE; first_valid_par = i; } } if( valid_found ) { for( i = first_valid_par + 1 ; i < rr_data->gprs_data.num_add_freq_list ; i++) { if( rr_data->gprs_data.add_freq_lists[i].v_cr_par EQ FALSE ) { /* entire structure is missing */ rr_data->gprs_data.add_freq_lists[i].v_cr_par = TRUE; memcpy(& (rr_data->gprs_data.add_freq_lists[i].cr_par), & (rr_data->gprs_data.add_freq_lists[i-1].cr_par), sizeof(T_cr_par)); } else { /* a part of the structure may be missing */ if( rr_data->gprs_data.add_freq_lists[i].cr_par.gprs_rxlev_access_min EQ GPRS_RXLEV_ACCESS_MIN_INVALID) rr_data->gprs_data.add_freq_lists[i].cr_par.gprs_rxlev_access_min = rr_data->gprs_data.add_freq_lists[i-1].cr_par.gprs_rxlev_access_min; if(rr_data->gprs_data.add_freq_lists[i].cr_par.gprs_ms_txpwr_max_cch EQ GPRS_MS_TXPWR_MAX_CCH_INVALID) rr_data->gprs_data.add_freq_lists[i].cr_par.gprs_ms_txpwr_max_cch = rr_data->gprs_data.add_freq_lists[i-1].cr_par.gprs_ms_txpwr_max_cch; if(rr_data->gprs_data.add_freq_lists[i].cr_par.gprs_temporary_offset EQ GPRS_TEMPORARY_OFFSET_INVALID ) rr_data->gprs_data.add_freq_lists[i].cr_par.gprs_temporary_offset = rr_data->gprs_data.add_freq_lists[i-1].cr_par.gprs_temporary_offset; if(rr_data->gprs_data.add_freq_lists[i].cr_par.gprs_penalty_time EQ GPRS_PENALTY_TIME_INVALID) rr_data->gprs_data.add_freq_lists[i].cr_par.gprs_penalty_time = rr_data->gprs_data.add_freq_lists[i-1].cr_par.gprs_penalty_time; if(rr_data->gprs_data.add_freq_lists[i].cr_par.gprs_reselect_offset EQ GPRS_RESEL_OFF_INVALID) rr_data->gprs_data.add_freq_lists[i].cr_par.gprs_reselect_offset = rr_data->gprs_data.add_freq_lists[i-1].cr_par.gprs_reselect_offset; if(rr_data->gprs_data.add_freq_lists[i].cr_par.hcs_thr EQ GPRS_HCS_THR_INVALID ) rr_data->gprs_data.add_freq_lists[i].cr_par.hcs_thr = rr_data->gprs_data.add_freq_lists[i-1].cr_par.hcs_thr; if(rr_data->gprs_data.add_freq_lists[i].cr_par.priority_class EQ GPRS_PRIORITY_CLASS_INVALID) rr_data->gprs_data.add_freq_lists[i].cr_par.priority_class = rr_data->gprs_data.add_freq_lists[i-1].cr_par.priority_class; if(rr_data->gprs_data.add_freq_lists[i].cr_par.si13_location EQ SI13_LOC_INVALID) rr_data->gprs_data.add_freq_lists[i].cr_par.si13_location = rr_data->gprs_data.add_freq_lists[i-1].cr_par.si13_location; if(rr_data->gprs_data.add_freq_lists[i].cr_par.psi1_repeat_period EQ PSI1_RPT_PRD_INVALID ) rr_data->gprs_data.add_freq_lists[i].cr_par.psi1_repeat_period = rr_data->gprs_data.add_freq_lists[i-1].cr_par.psi1_repeat_period; if(rr_data->gprs_data.add_freq_lists[i].cr_par.pbcch_location EQ PBCCH_LOC_INVALID ) rr_data->gprs_data.add_freq_lists[i].cr_par.pbcch_location = rr_data->gprs_data.add_freq_lists[i-1].cr_par.pbcch_location; } } } } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6147) MODULE : RR_ATT | | STATE : code ROUTINE : att_calculate_c31_c32 | +--------------------------------------------------------------------+ PURPOSE : calculates C31 and C32 criterion of cell reselection is still valid. */ GLOBAL void att_calculate_c31_c32 (UBYTE index) { GET_INSTANCE_DATA; USHORT h; T_NC_DATA *rrd; T_NC_DATA *sc_ptr; UBYTE prios, prion; /*lint -esym(526,p_res_off) : not defined | defined in GRR */ extern const SHORT p_res_off[]; /* C1 = A - MAX(B,0) */ /* A = RLA_P - GPRS_RXLEV _ACCESS_MIN */ /* B = GPRS_MS_TXPWR_MAX_CCH - P */ /* P = max MS RF output power */ /* C31 = RLA_P(s) - HCS_THR(s) (serving cell) */ /* C31 = RLA_P(n) - HCS_THR(n) - TO(n)*L(n) (neighbour cell) */ /* TO(n) = GPRS_TEMPORARY_OFFSET(n) * H( GPRS_PENALTY_TIME(N) - T(n) ) */ /* L(n) = 0 if PRIORITY_CLASS(n) = PRIORITY_CLASS(s) */ /* 1 if PRIORITY_CLASS(n) = PRIORITY_CLASS(s) */ /* C32(s) = C1(s) */ /* C32(n) = C1(n) + GPRS_RESELECT OFFSET(n) - TO(n)*(1-L(n)) */ /* TO(n) = GPRS_TEMPORARY_OFFSET(n) * H( GPRS_PENALTY_TIME(N) - T(n) ) */ /* L(n) = 0 if PRIORITY_CLASS(n) = PRIORITY_CLASS(s) */ /* 1 if PRIORITY_CLASS(n) = PRIORITY_CLASS(s) */ /* H(n) = 0 for x < 0 */ /* 1 for x >= 0 */ TRACE_FUNCTION ("att_calculate_c31_c32()"); /* * get a pointer to the data of the cell */ rrd = &(rr_data->nc_data[index]); att_calculate_c1(index); TRACE_EVENT_P5("arfcn = %d, C1 = %d, rxlev = %d, GPRS_RXLEV _ACCESS_MIN = %d, GPRS_MS_TXPWR_MAX_CCH = %d", rrd->arfcn, rrd->c1, rrd->rxlev, rrd->cr_par.gprs_rxlev_access_min, rrd->cr_par.gprs_ms_txpwr_max_cch); TRACE_EVENT_P3 ("hcs thr = %d, prion = %d, valid = %d", rrd->cr_par.hcs_thr, rrd->cr_par.priority_class , rrd->v_cr_par); if( rrd->v_cr_par EQ CR_PAR_INVALID) { rrd->c31 = 0; /* conversion of idle mode to GPRS cell reselection parameters */ } else { rrd->c31 = rrd->rxlev; if( rrd->cr_par.hcs_thr NEQ 31 ) rrd->c31 -= ( 2 * rrd->cr_par.hcs_thr ); else rrd->c31 = -32768; } rrd->c32 = rrd->c1; /* C31 = RLA_P(s) - HCS_THR(s) (serving cell) */ /* C32(s) = C1(s) */ TRACE_EVENT_P1("C31 = %d", rrd->c31); if( index EQ SC_INDEX ) return; sc_ptr = &rr_data->nc_data[SC_INDEX]; prios = (sc_ptr->cr_par.priority_class NEQ GPRS_PRIORITY_CLASS_INVALID) ? sc_ptr->cr_par.priority_class : (UBYTE) 0; prion = (rrd->cr_par.priority_class NEQ GPRS_PRIORITY_CLASS_INVALID ) ? rrd->cr_par.priority_class : (UBYTE) 0; if( rr_data->nc_data[index].v_cr_par EQ CR_PAR_INVALID) { /* * conversion of idle mode paramter to GPRS cell reselection parameters. * the conversion for GPRS_RESELECTION_OFFSET is done here. * the coversion of other parameters are done as * part of the function att_calculate_c1() * * GPRS_RESELECT_OFFSET(n) = * CELL_RESELECT_OFFSET(n) - CELL_RESELECT_OFFSET(s) * * NOTE: If PENALTY_TIME = 11111 for a cell, * the sign of CELL_RESELECT_OFFSET shall be changed * */ SHORT nc_sign = 1; SHORT sc_sign = 1; if(rrd->cr_par.gprs_penalty_time EQ 31 ) { nc_sign = -1; } if(sc_ptr->cr_par.gprs_penalty_time EQ 31 ) { sc_sign = -1; } rrd->c32 += (USHORT)(nc_sign * 2 * rrd->c2_par.cell_reselect_offset ) - (USHORT)(sc_sign * 2 * sc_ptr->c2_par.cell_reselect_offset); TRACE_EVENT_P5(" nc_sign =%d, sc_sign =%d, nc_rese_off=%d, sc_res_off=%d, c31=%d", nc_sign, sc_sign, rrd->c2_par.cell_reselect_offset, sc_ptr->c2_par.cell_reselect_offset, rrd->c32); } else { /* a valid paramter for GPRS_RESELECT_OFFSET is received */ rrd->c32 += p_res_off[rrd->cr_par.gprs_reselect_offset]; TRACE_EVENT_P3(" gprs_reselect_offset = %d, c32 = %d, res off index =%d", p_res_off[rrd->cr_par.gprs_reselect_offset], rrd->c32, rrd->cr_par.gprs_reselect_offset); } /* c32_qual is not known, if there is no PCCCH */ /* * Calculate x for H(x) and set H(x) */ TRACE_EVENT_P2("penalty time = %d, avail time = %d", (rrd->cr_par.gprs_penalty_time + 1) * 10, rrd->avail_time/PERIOD_1_SEC ); if ((USHORT)((rrd->cr_par.gprs_penalty_time + 1) * 10) >= (USHORT)(rrd->avail_time/PERIOD_1_SEC)) h = TRUE; else h = FALSE; if (h) { SHORT *criterion; TRACE_EVENT_P3("penalty timer not expired: %d offset will be applied to (prios)%d NEQ (prion)%d (if 0 c31, if 1 C32 )", 10 * rrd->cr_par.gprs_temporary_offset, prios, prion); if( prios NEQ prion ) { criterion = & (rrd->c31); } else { criterion = & (rrd->c32); } if( rrd->cr_par.gprs_temporary_offset EQ 7 ) { *criterion = -32768; } else { *criterion -= 10 * rrd->cr_par.gprs_temporary_offset; } } else { TRACE_EVENT("penalty timer expired"); } if( rrd->cr_par.same_ra_as_serving_cell NEQ SAME_RA_AS_SCELL) { TRACE_EVENT(" cell is NOT SAME_RA_AS_SCELL"); rrd->c32 -= 2 * rrd->select_para.cell_resel_hyst ; } else if(rr_data->gprs_data.ready_state) { /* MS in ready state*/ TRACE_EVENT(" MS in ready state"); rrd->c32 -= (SHORT)2 * rrd->select_para.cell_resel_hyst ; /* C31 hyst not known */ } if ( IS_TIMER_ACTIVE (T_NO_RESELECT) ) { rrd->c32 -= 5; } } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6147) MODULE : RR_ATT | | STATE : code ROUTINE : att_get_next_best_c32_index | +--------------------------------------------------------------------+ PURPOSE : This function returns the index of a target cell, for reselection. If no target cell is available then NO_AVAILABLE is returned. The parameter c31_calculated, indicates if C31 and C32 are already calculated. */ GLOBAL UBYTE att_get_next_best_c32_index (BOOL c31_calculated) { GET_INSTANCE_DATA; UBYTE max_result = SC_INDEX; UBYTE result = NO_AVAILABLE; UBYTE i; T_cr_par *ncell_first, *ncell_next, *scell_ptr; BOOL c31_bc, c31_nc, c31_sc; TRACE_FUNCTION ("att_get_next_best_c32_index()"); switch (rr_data->sc_data.selection_type) { case BACK_FROM_DEDICATED: case BACK_FROM_DEDICATED_RLF: case CELL_RESELECTION_CR: case CELL_RESELECTION_RACH: /* not valid */ break; default: /* * Normal Cell reselection * */ if( ! c31_calculated ) { att_calculate_c31_c32(SC_INDEX); for (i = 0; i < SC_INDEX; i++) { if (rr_data->nc_data[i].bcch_status EQ DECODED) { att_calculate_c31_c32 (i); } } } TRACE_EVENT_P1("Scell = %d", rr_data->nc_data[SC_INDEX].arfcn); for(i=0;i<=SC_INDEX; i++) { if( rr_data->nc_data[i].bcch_status EQ DECODED ) { ncell_first = &rr_data->nc_data[i].cr_par; TRACE_EVENT_P6("First best Cell(arfcn=%d), C1=%d,C31=%d,C32=%d,Priority class =%d,hcs_thr=%d", rr_data->nc_data[i].arfcn, rr_data->nc_data[i].c1, rr_data->nc_data[i].c31, rr_data->nc_data[i].c32, ncell_first->priority_class, ncell_first->hcs_thr ); } } /* * Now find the the first one which is not previously used */ for (i = 0; (i < SC_INDEX) AND (max_result EQ SC_INDEX) ; i++) { if ( (!rr_data->nc_data[i].c32_used) AND rr_data->nc_data[i].bcch_status EQ DECODED AND rr_data->nc_data[i].c1 > 0) { max_result = i; } } if( max_result EQ SC_INDEX ) { return NO_AVAILABLE; } ncell_first = &( rr_data->nc_data[max_result].cr_par); c31_bc = ( (rr_data->nc_data[max_result].c31 >= 0) AND (ncell_first->priority_class NEQ GPRS_PRIORITY_CLASS_INVALID) ); /* choose the best cell */ for (i = max_result+1; i < 6 ; i++) { if ( (!rr_data->nc_data[i].c32_used) AND (rr_data->nc_data[i].bcch_status EQ DECODED) AND (rr_data->nc_data[i].c1) > 0 ) { ncell_next = &( rr_data->nc_data[i].cr_par); c31_nc = ( (rr_data->nc_data[i].c31 >= 0) AND (ncell_next->priority_class NEQ GPRS_PRIORITY_CLASS_INVALID )); if( ( c31_bc AND c31_nc AND (ncell_next->priority_class EQ ncell_first->priority_class ) AND (rr_data->nc_data[i].c32 > rr_data->nc_data[max_result].c32) ) OR ( c31_bc AND c31_nc AND (ncell_next->priority_class > ncell_first->priority_class ) ) ) { /* prioritized cell reselection must be done */ max_result = i; ncell_first = ncell_next; } else if ( !c31_bc ) { if( c31_nc) { c31_bc = c31_nc; max_result = i; ncell_first = ncell_next; } else if ( rr_data->nc_data[i].c32 > rr_data->nc_data[max_result].c32 ) { /* no prioritized cell found yet */ max_result = i; ncell_first = ncell_next; } } } } /* compare with Serving cell */ scell_ptr = &( rr_data->nc_data[SC_INDEX].cr_par); c31_sc = ( (rr_data->nc_data[SC_INDEX].c31 >= 0 ) AND (scell_ptr->priority_class NEQ GPRS_PRIORITY_CLASS_INVALID)); TRACE_EVENT_P2("c31_sc=%d and c31_bc=%d",c31_sc, c31_bc); if( ( c31_sc AND c31_bc AND (ncell_first->priority_class EQ scell_ptr->priority_class) AND (rr_data->nc_data[max_result].c32 > rr_data->nc_data[SC_INDEX].c32) ) OR ( c31_sc AND c31_bc AND (ncell_first->priority_class > scell_ptr->priority_class) ) OR ( c31_bc AND !c31_sc ) OR ( !c31_bc AND !c31_sc AND (rr_data->nc_data[max_result].c32 > rr_data->nc_data[SC_INDEX].c32) ) ) { result = max_result; } else { result = NO_AVAILABLE; } } /* * return the index of the highest C2. */ return result; } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6147) MODULE : RR_ATT | | STATE : code ROUTINE : att_check_c31_reselect_decision| +--------------------------------------------------------------------+ PURPOSE : */ GLOBAL void att_check_c31_reselect_decision (UBYTE start_now) { GET_INSTANCE_DATA; UBYTE index; UBYTE delta = 1; TRACE_FUNCTION("att_check_c31_reselect_decision()"); index = att_get_next_best_c32_index(TRUE); while( index NEQ NO_AVAILABLE ) { /* found a suitable cell to reselect */ TRACE_EVENT_P1("found a cell %d", index); /* * 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( index)) delta =0; /* * 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(index)) delta = 0; #if defined(_SIMULATION_) TRACE_EVENT_WIN_P1 ("delta=%d", delta); #endif } if (delta > 0) { if (start_now) { att_start_cell_reselection_gprs(CELL_RESELECTION_NC); } return; } rr_data->nc_data[index].c32_used = TRUE; index = att_get_next_best_c32_index(TRUE); } TRACE_EVENT(" index == NO_AVAILABLE in function att_check_c31_reselect_decision"); } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6147) MODULE : RR_ATT | | STATE : code ROUTINE : att_check_cell_c31 | +--------------------------------------------------------------------+ PURPOSE : if a cell reselection decision has been taken based on C31 and C32 criterion , 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. */ GLOBAL BOOL att_check_cell_c31 (void) { GET_INSTANCE_DATA; BOOL cell_ok = FALSE; TRACE_FUNCTION ("att_check_cell_c31()"); /* * 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 C31 and C32 for the candidate */ //att_calculate_c31_c32(rr_data->reselect_index); /* * attempting reselection. Mark used */ rr_data->nc_data[rr_data->reselect_index].c32_used = TRUE; /* * Cell should not be barred. * If GPRS cell reselection parameters are received, look for CELL_BAR_ACCESS_2 * else SI3 RACH parameters would indicate cell barred */ if( rr_data->nc_data[rr_data->reselect_index].v_cr_par EQ CR_PAR_VALID) { TRACE_EVENT_P2("Cell Barred arfcn = %d, barred = %d", rr_data->nc_data[rr_data->reselect_index].arfcn, rr_data->nc_data[rr_data->reselect_index].cr_par.cell_bar_access_2 ); cell_ok = (rr_data->nc_data[rr_data->reselect_index].cr_par.cell_bar_access_2 EQ CBA_2_NORMAL) ? TRUE : FALSE; } else { /* Implements RR Clone findings #8 */ cell_ok = !att_cell_barred_status_cr_no_cr (rr_data->reselect_index); } /* * 1. the cell shall be not temporarily excluded (tnnn e.g. after random access * failure). * 2. C31-C32 parameters are still valid */ if( cell_ok ) cell_ok = ( ! is_tnnn (rr_data->reselect_index) AND att_check_c31_criterion(rr_data->reselect_index) ); if ((cell_ok) AND (rr_data->ms_data.rr_service EQ FULL_SERVICE)) { if(rr_data->nc_data[rr_data->reselect_index].v_cr_par EQ CR_PAR_VALID) { cell_ok = (rr_data->nc_data[rr_data->reselect_index].cr_par.exc_acc EQ EXC_ACC_NORMAL) ? TRUE : FALSE; } else cell_ok = dat_roam_forb_lai_check (rr_data->reselect_index); } if (cell_ok) { /* * suitable cell has been found */ TRACE_EVENT ("use neighbour cell"); /* * 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); rr_data->gprs_data.ready_state = FALSE; /* Standby */ 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_best_c32_index(TRUE); } } /* * return whether a cell has been found or not. */ return cell_ok; } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6147) MODULE : RR_ATT | | STATE : code ROUTINE : att_check_c31_criterion | +--------------------------------------------------------------------+ PURPOSE : returns TRUE if C31 and C32 criterion of cell reselection is still valid. */ GLOBAL BOOL att_check_c31_criterion(UBYTE index) { GET_INSTANCE_DATA; T_cr_par *ncell_ptr, *scell_ptr; BOOL c31_bc, c31_sc; att_calculate_c31_c32(index); ncell_ptr = & rr_data->nc_data[index].cr_par; scell_ptr = & rr_data->nc_data[SC_INDEX].cr_par; c31_bc = ((rr_data->nc_data[index].c31 >= 0) AND (ncell_ptr->priority_class NEQ GPRS_PRIORITY_CLASS_INVALID )); c31_sc = ( (rr_data->nc_data[SC_INDEX].c31 >= 0 ) AND (scell_ptr->priority_class NEQ GPRS_PRIORITY_CLASS_INVALID)); /* * check for C32 values, depending on priority */ if( ( c31_sc AND c31_bc AND (ncell_ptr->priority_class > scell_ptr->priority_class) ) OR ( c31_sc AND c31_bc AND (ncell_ptr->priority_class EQ scell_ptr->priority_class) AND (rr_data->nc_data[index].c32 > rr_data->nc_data[SC_INDEX].c32) ) OR ( c31_bc AND (!c31_sc) ) OR ( (!c31_sc) AND (!c31_bc) AND (rr_data->nc_data[index].c32 > rr_data->nc_data[SC_INDEX].c32) ) ) { return TRUE; } return FALSE; } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_rrgrr_update_ba_req | +-----------------------------------------------------------------------------+ PURPOSE : */ void att_rrgrr_update_ba_req(T_RRGRR_UPDATE_BA_REQ* ba_req) { GET_INSTANCE_DATA; UBYTE i; T_LIST new_list; T_CELL_DATA * cd; TRACE_EVENT ("update_ba"); switch(GET_STATE(STATE_GPRS)) { case GPRS_PIM_BCCH: case GPRS_PTM_BCCH: case GPRS_PIM_PBCCH: case GPRS_PAM_PBCCH: case GPRS_PTM_PBCCH: #if 0 if (ba_req->cell_type EQ SYNC_SCELL) #endif cd = &rr_data->sc_data.cd; #if 0 /* this is not used anymore but it still in the SAp for some time */ else cd = &rr_data->cr_data.cd; #endif if(!rr_data->gprs_data.ba_bcch_modified) srv_copy_list (&rr_data->ba_ncell_list, &cd->ncell_list, sizeof (T_LIST)); /* store GPRS cell reselection parameters */ memcpy(rr_data->gprs_data.add_freq_lists, ba_req->add_freq_list, sizeof(T_add_freq_list) * RRGRR_BA_LIST_SIZE ); srv_copy_list (&new_list, &rr_data->ba_ncell_list, sizeof (T_LIST)); if( ba_req->add_freq_list[0].arfcn NEQ NOT_PRESENT_16BIT ) rr_data->gprs_data.use_c31 = FALSE; for(i=0; i<RRGRR_BA_LIST_SIZE AND ba_req->add_freq_list[i].arfcn NEQ NOT_PRESENT_16BIT; i++) { /*add*/ srv_set_channel (&new_list, ba_req->add_freq_list[i].arfcn); if((ba_req->add_freq_list[i].v_cr_par) AND ( ba_req->cell_type EQ SYNC_SCELL)) { rr_data->gprs_data.use_c31 = TRUE; TRACE_EVENT("use_c31 turned TRUE"); } } if( rr_data->gprs_data.use_c31) { rr_data->gprs_data.num_add_freq_list = i; TRACE_EVENT_P1("number of added arfcn = %d", i); att_init_cr_c31_data(); att_init_nc_data_c31_cr_par(); } else { rr_data->gprs_data.num_add_freq_list = 0; } for(i=0; i<RRGRR_BA_LIST_SIZE AND ba_req->rm_freq_list[i].arfcn NEQ NOT_PRESENT_16BIT; i++) { /*remove*/ srv_unset_channel(&new_list, ba_req->rm_freq_list[i].arfcn); } /* * The NC2 indication in the Packet Measurement Order * overrides the setting in the SI13 */ rr_data->gprs_data.nc_mode_of_pmo = ba_req->nc_mode; rr_data->gprs_data.ba_bcch_modified=TRUE; if (srv_compare_list (&cd->ncell_list, &new_list) EQ FALSE) { /* both lists are different or it is the first one */ srv_copy_list (&cd->ncell_list, &new_list, sizeof (T_LIST)); /* forward new neighbour cell list to layer 1 if * changes shall be indicated */ if (ba_req->cell_type EQ SYNC_SCELL) { att_remove_bad_rr_data_ncells(); att_code_mph_ncell_req (SC_INDEX); #if defined (REL99) AND defined (TI_PS_FF_EMR) /*BA(BCCH) is modified by GRR, this list now loses the indices' sync that it has with NW, so can no longer use it when moved to dedicated mode, for reporting*/ rr_data->sc_data.ba_list_idle = FALSE; rr_data->sc_data.enh_para_status = ENH_PARA_INVALID_STATE; for_set_default_emr_data(&rr_data->sc_data.emr_data_current); for_set_default_emr_data(&rr_data->sc_data.emr_data_temp); #endif } } break; default: break; } PFREE(ba_req); } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_rrgrr_meas_rep_req | +-----------------------------------------------------------------------------+ PURPOSE : */ void att_rrgrr_meas_rep_req (T_RRGRR_MEAS_REP_REQ* rrgrr_meas_rep_req) { PALLOC(mph_meas_rep_req,MPH_MEAS_REP_REQ); TRACE_FUNCTION("rrgrr_meas_rep_req()"); mph_meas_rep_req->cause = rrgrr_meas_rep_req->meas_cause; PFREE(rrgrr_meas_rep_req); PSENDX(PL,mph_meas_rep_req); } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_check_sync_results | +-----------------------------------------------------------------------------+ PURPOSE : */ BOOL att_check_sync_results(T_MPH_MEASUREMENT_IND* mph_measurement_ind) { UBYTE i=0; T_NC_DATA* pnc = NULL; if(mph_measurement_ind->gprs_sync EQ SYNC_RESULTS) { PALLOC(bsic_ind, RRGRR_NCELL_SYNC_IND); TRACE_EVENT ("check_sync_res:TRUE"); for(; i < mph_measurement_ind->ncells.no_of_ncells ;i++) { bsic_ind->sync_result[i].arfcn = mph_measurement_ind->ncells.arfcn[i]; if(mph_measurement_ind->ncells.bsic[i] EQ NOT_PRESENT_8BIT OR mph_measurement_ind->ncells.bsic[i] EQ MPH_BSIC_UNKNOWN) { if((pnc = is_nc_in_list(mph_measurement_ind->ncells.arfcn[i])) NEQ NULL) { pnc->arfcn = NOT_PRESENT_16BIT; pnc->bcch_status = EMPTY; } if(mph_measurement_ind->ncells.bsic[i] EQ NOT_PRESENT_8BIT) bsic_ind->sync_result[i].sb_flag = NO_SB_FOUND; else bsic_ind->sync_result[i].sb_flag = SB_UNKNOWN; /*UNKNOWN*/ } else { /* cell with BSIC */ if( (pnc = is_nc_in_list(mph_measurement_ind->ncells.arfcn[i])) NEQ NULL) { /*found -> update if needed */ pnc->bsic = bsic_ind->sync_result[i].bsic = mph_measurement_ind->ncells.bsic[i]; pnc->bcch_status=NON_DECODED; } else { /*add*/ pnc = find_empty_slot_in_list(); if(pnc) { pnc->bsic = bsic_ind->sync_result[i].bsic = mph_measurement_ind->ncells.bsic[i]; pnc->arfcn = mph_measurement_ind->ncells.arfcn[i]; pnc->bcch_status=NON_DECODED; } } bsic_ind->sync_result[i].sb_flag = SB_FOUND; } } if(i>0) { for(/*dont init i*/; i < RRGRR_MAX_RSLT_NCELL_SYNC_IND; i++) { bsic_ind->sync_result[i].arfcn = NOT_PRESENT_16BIT; } PSENDX(GRR, bsic_ind); } else { PFREE(bsic_ind); } PFREE(mph_measurement_ind); return TRUE; } return FALSE; } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_check_bsic | +-----------------------------------------------------------------------------+ PURPOSE : */ void att_check_bsic(T_RRGRR_NCELL_SYNC_REQ* check_bsic) { 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_EVENT ("acq_bsic"); switch(GET_STATE(STATE_GPRS)) { case GPRS_PIM_PBCCH: case GPRS_PTM_PBCCH: case GPRS_PAM_PBCCH: { #if defined (TI_PS_FF_RTD) AND defined (REL99) UBYTE i,j; #else UBYTE i; #endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */ PALLOC(ncell_req, MPH_NEIGHBOURCELL_REQ); if(check_bsic->sync_type EQ SYNC_RECONFIRM) ncell_req->sync_only = RECONFIRM_SYNC_LIST; else ncell_req->sync_only = SYNC_LIST; memset(ncell_req->arfcn, NOT_PRESENT_8BIT, 2*MAX_NEIGHBOURCELLS); for(i=0; i<RRGRR_MAX_ARFCN_NCELL_SYNC_REQ AND #if defined (TI_PS_FF_RTD) AND defined (REL99) check_bsic->ncell_sync_list[i].arfcn #else check_bsic->arfcn[i] #endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */ NEQ NOT_PRESENT_16BIT; i++) { #if defined (TI_PS_FF_RTD) AND defined (REL99) ncell_req->arfcn[i] = check_bsic->ncell_sync_list[i].arfcn; #else ncell_req->arfcn[i] = check_bsic->arfcn[i]; #endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */ #if defined (TI_PS_FF_RTD) AND defined (REL99) for(j = 0;j < check_bsic->ncell_sync_list[i].c_rtd;j++) p_cur->enh_para.enh_cell_list[i].rtd[j] = check_bsic->ncell_sync_list[i].rtd[j]; p_cur->enh_para.enh_cell_list[i].v_rtd = check_bsic->ncell_sync_list[i].v_rtd; p_cur->enh_para.enh_cell_list[i].c_rtd = check_bsic->ncell_sync_list[i].c_rtd; #endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */ } #if defined (TI_PS_FF_RTD) AND defined (REL99) attf_send_enh_para_to_alr(p_cur->rep_type,&p_cur->enh_para); #endif /* #if defined (TI_PS_FF_RTD) AND defined (REL99) */ PSENDX(PL, ncell_req); } break; default: break; } PFREE(check_bsic); } static T_NC_DATA* is_nc_in_list(USHORT arfcn) { GET_INSTANCE_DATA; T_NC_DATA* p_nc = &rr_data->nc_data[0]; T_NC_DATA* p_nc_l = &rr_data->nc_data[6]; for(; p_nc < p_nc_l;p_nc++) if(p_nc->arfcn EQ arfcn) return p_nc; return NULL; } static T_NC_DATA* find_empty_slot_in_list(void) { GET_INSTANCE_DATA; T_NC_DATA* p_nc = &rr_data->nc_data[0]; T_NC_DATA* p_nc_l = &rr_data->nc_data[6]; for(; p_nc < p_nc_l;p_nc++) if(p_nc->bcch_status EQ EMPTY) return p_nc; return NULL; } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_is_gprs_avail | +-----------------------------------------------------------------------------+ PURPOSE : */ BOOL att_gprs_is_avail(void) { GET_INSTANCE_DATA; if(GET_STATE(STATE_GPRS) NEQ GPRS_NULL) return TRUE; else return FALSE; } BOOL att_gprs_cell_has_pbcch(void) { GET_INSTANCE_DATA; UBYTE st = GET_STATE(STATE_GPRS); switch(st) { case GPRS_PIM_PBCCH: case GPRS_PAM_PBCCH: case GPRS_PTM_PBCCH: case GPRS_SUSPENDED_PBCCH: return TRUE; default: return FALSE; } } void att_gprs_stop_pl (void) { GET_INSTANCE_DATA; att_build_classmark_req (); att_build_idle_req(SC_INDEX, MODE_CONFIG_PL); #if defined (REL99) AND defined (TI_PS_FF_EMR) /*reset SI-2quater information and enhanced parameters if present when PBCCH is present*/ rr_data->sc_data.cd.si2quater_status = SI2QUATER_ABSENT; rr_data->sc_data.cd.si2quater_pos = 0; memset (rr_data->sc_data.rep_count, NOT_PRESENT_8BIT, MAX_NEIGHBOURCELLS); for_set_default_emr_data(&rr_data->sc_data.emr_data_current); 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; #endif /* * use neighbour cell list from system info 2/2bis/2ter */ memset (rr_data->act_ncell_list, 0xFF, 2*MAX_NEIGHBOURCELLS); srv_create_list (&rr_data->sc_data.cd.ncell_list, rr_data->act_ncell_list, MAX_NEIGHBOURCELLS, FALSE, 0); } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_gprs_cr_rsp | +-----------------------------------------------------------------------------+ PURPOSE : */ void att_gprs_cr_rsp(T_RRGRR_CR_RSP* cr_rsp) { GET_INSTANCE_DATA; TRACE_FUNCTION("att_gprs_cr_rsp()"); switch(rr_data->gprs_data.start_proc) { case START_PROC_ACTIVATE_REQ: srv_use_stored_prim(); break; case START_PROC_ENTER_LIMITED: att_signal_gprs_support(); if(att_gprs_cell_has_pbcch()) { /* * RR is now in idle and configures L1 for * BCCH only */ SET_STATE(STATE_GPRS, GPRS_PIM_BCCH); switch(GET_STATE(STATE_ATT)) { case ATT_IDLE: case ATT_CON_EST: att_build_idle_req(SC_INDEX, MODE_SYS_INFO_CHANGE); break; default: break; } } else { switch(GET_STATE(STATE_GPRS)) { case GPRS_PTM_BCCH: case GPRS_PAM_BCCH: /* * Due to the CR_IND GRR has released the TBF so we * have to correctly enter idle mode * to respect transition rules */ rr_data->gprs_data.page_mode = PAG_MODE_DEFAULT; SET_STATE (STATE_GPRS, GPRS_PIM_BCCH); att_return_to_idle(); break; default: break; } } break; case START_PROC_NORMAL_CR: att_start_cell_reselection (rr_data->gprs_data.cr_type); break; case START_PROC_NORMAL_CS: att_start_cell_selection (rr_data->gprs_data.cr_orig, CS_NOT_PARALLEL, rr_data->cs_data.current_search_mode); break; case START_PROC_CFG_CHG: /*XXX not ok*/ att_signal_gprs_support(); break; case START_PROC_GPRS_ACT: { PALLOC ( mph_mon_ctrl_req, MPH_MON_CTRL_REQ ); TRACE_EVENT("START_PROC_GPRS_ACT----- "); rr_data->sc_data.mm_started = MM_ORIGINATED; if((rr_data->sc_data.cd.sys_info_read EQ ALL_SYS_INFO_READ) OR ((rr_data->sc_data.cd.sys_info_read | SYS_INFO_13_READ) EQ ALL_SYS_INFO_READ)) { rr_data->sc_data.cd.sys_info_read &= ~(SYS_INFO_13_READ); mph_mon_ctrl_req->si_to_read = UPDATE_SI13; if( rr_data->nc_data[SC_INDEX].si13_loc_ind EQ NORMAL_BCCH) { mph_mon_ctrl_req->action = START_MON_NBCCH; mph_mon_ctrl_req->si_to_read = UPDATE_SI13_GSM; } else mph_mon_ctrl_req->action = START_MON_EBCCH; } else { TRACE_EVENT ("START_PROC_GPRS_ACT: Read Complete SI"); rr_data->sc_data.cd.sys_info_read = NO_SYS_INFO_READ; mph_mon_ctrl_req->si_to_read = COMPLETE_SI; mph_mon_ctrl_req->action = START_MON_NBCCH; } PSENDX (PL, mph_mon_ctrl_req); TIMERSTART (T_RESELECT, TRESELECT_VALUE); } break; default: break; } rr_data->gprs_data.start_proc = START_PROC_NOTHING; PFREE(cr_rsp); } void att_gprs_sync_req( T_RRGRR_SYNC_REQ *sync_req) { GET_INSTANCE_DATA; switch(GET_STATE(STATE_GPRS)) { case GPRS_PIM_PBCCH: case GPRS_PTM_PBCCH: { memset (&rr_data->nc_data[CR_INDEX], 0, sizeof (T_NC_DATA)); rr_data->nc_data[CR_INDEX].arfcn = sync_req->arfcn; rr_data->nc_data[CR_INDEX].bsic = sync_req->bsic; att_start_cell_reselection_pbcch(MODE_CELL_RESELECTION_SYNC_ONLY); } break; default: break; } PFREE(sync_req); } void att_start_cell_reselection_gprs(UBYTE mode) { GET_INSTANCE_DATA; /*set start proc */ if(att_gprs_is_avail()) { UBYTE st = GET_STATE(STATE_ATT); if(st NEQ ATT_CS3 AND st NEQ ATT_CS2 AND st NEQ ATT_CS1) { att_rrgrr_cr_ind(CR_ABNORMAL); rr_data->gprs_data.start_proc = START_PROC_NORMAL_CR; rr_data->gprs_data.cr_type = mode; return; } } att_start_cell_reselection(mode); } void att_start_cell_selection_gprs(UBYTE originator,UBYTE search_mode) { GET_INSTANCE_DATA; /*set start proc*/ if(att_gprs_is_avail()) { UBYTE st = GET_STATE(STATE_ATT); if(st NEQ ATT_CS3 AND st NEQ ATT_CS2 AND st NEQ ATT_CS1) { if(originator EQ MM_ORIGINATED) { TRACE_EVENT("start_cs_gprs: MM_ORIG!"); att_rrgrr_cr_ind(CR_NORMAL); } else { att_rrgrr_cr_ind(CR_ABNORMAL); EM_FMM_RESEL_START_IND; } CS_SET_CURRENT_SEARCH_MODE(search_mode); rr_data->gprs_data.start_proc = START_PROC_NORMAL_CS; rr_data->gprs_data.cr_orig = originator; return; } } att_start_cell_selection(originator, CS_NOT_PARALLEL,search_mode); } void att_gprs_idle_req(T_MPH_IDLE_REQ* idle_req) { GET_INSTANCE_DATA; /* * At this point it is necessary to do the following things: * o send a CR_IND (optional) * o set the parameter gprs_support (if needed, the default has already been set before) * o set the parameter reorg_only (if needed) */ switch(idle_req->mod) { case MODE_CELL_SELECTION: if(GET_STATE(STATE_GPRS) EQ GPRS_NULL OR GET_STATE(STATE_GPRS) EQ GPRS_ACTIVATED) { idle_req->gprs_support = MPH_GPRS_PROCS_NOT_USED; } else { idle_req->gprs_support = MPH_GPRS_PROCS_USED; if(rr_data->gprs_data.page_mode EQ PAG_MODE_REORG) idle_req->reorg_only = REORG_ONLY; else idle_req->reorg_only = NORMAL_PGM; } break; case MODE_CONFIG_PL: idle_req->gprs_support = MPH_GPRS_PROCS_USED; break; case MODE_SYS_INFO_CHANGE: if(GET_STATE(STATE_GPRS) EQ GPRS_NULL OR GET_STATE(STATE_GPRS) EQ GPRS_ACTIVATED OR rr_data->ms_data.rr_service EQ LIMITED_SERVICE) { idle_req->gprs_support = MPH_GPRS_PROCS_NOT_USED; } else { idle_req->gprs_support = MPH_GPRS_PROCS_USED; idle_req->reorg_only = NORMAL_PGM; } break; default: break; } } BOOL att_gprs_check_ncell(void) { GET_INSTANCE_DATA; BOOL cell_ok; BOOL cell_barred; BOOL check_nw; BOOL roam_forb; BOOL plmn_equal; TRACE_EVENT_P1 ("att_gprs_check_ncell() cr_pbcch_active=%d", rr_data->gprs_data.cr_pbcch_active); if((rr_data->gprs_data.cr_pbcch_active) OR (rr_data->gprs_data.cr_pcco_active)) { /* Implements RR Clone findings #8 */ cell_barred = !att_cell_barred_status_cr_no_cr (CR_INDEX); check_nw = att_check_network (&rr_data->nc_data[CR_INDEX].lai); roam_forb = dat_roam_forb_lai_check (CR_INDEX); plmn_equal = dat_plmn_equal_req (rr_data->nc_data[CR_INDEX].lai.mcc, rr_data->nc_data[CR_INDEX].lai.mnc, rr_data->ms_data.plmn.mcc, rr_data->ms_data.plmn.mnc); TRACE_EVENT_P6 ( "r MCC/MNC=%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_P6 ( "i MCC/MNC=%x%x%x/%x%x%x", rr_data->nc_data[CR_INDEX].lai.mcc[0], rr_data->nc_data[CR_INDEX].lai.mcc[1], rr_data->nc_data[CR_INDEX].lai.mcc[2], rr_data->nc_data[CR_INDEX].lai.mnc[0], rr_data->nc_data[CR_INDEX].lai.mnc[1], rr_data->nc_data[CR_INDEX].lai.mnc[2]); TRACE_EVENT_P5 ("[%u] cb %d nw %d ro %d pl %d", rr_data->nc_data[CR_INDEX].arfcn, cell_barred, check_nw, roam_forb, plmn_equal); cell_ok = cell_barred AND check_nw AND roam_forb AND plmn_equal; if (cell_ok) { TRACE_EVENT_P2 ("NEW SC [%d]->[%d]", rr_data->nc_data[SC_INDEX].arfcn, rr_data->nc_data[CR_INDEX].arfcn); if(rr_data->ms_data.rr_service EQ LIMITED_SERVICE AND rr_data->ms_data.req_mm_service NEQ FUNC_LIM_SERV_ST_SRCH) { /* * 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. */ rr_data->ms_data.rr_service = FULL_SERVICE; } /* TIMERSTOP (T_RESELECT);*/ att_copy_cr_data (); memcpy (&rr_data->sc_data.cd, &rr_data->cr_data.cd, sizeof (T_CELL_DATA)); if (rr_data->old_serving_cell < SC_INDEX) rr_data->nc_data[rr_data->old_serving_cell].avail_time = PERIOD_700_SEC; rr_data->nc_data[CR_INDEX].arfcn = NOT_PRESENT_16BIT; rr_data->gprs_data.cr_pbcch_active = FALSE; if(GET_STATE(STATE_GPRS) EQ GPRS_PIM_BCCH OR GET_STATE(STATE_GPRS) EQ GPRS_ACTIVATED) { att_set_pl_in_idle_mode (); } else if (att_gprs_cell_has_pbcch()) { att_gprs_stop_pl(); } att_signal_gprs_support(); /* XXX ??? * 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); } else { /* * inform GRR that the currently chosen cell * not suitable and wait for the next rrgrr_cr_req */ att_cell_reselection_gprs_failed(); } return TRUE; } else { return FALSE; } } void att_cell_reselection_gprs_failed(void) { GET_INSTANCE_DATA; PALLOC_SDU(gprs_ind, RRGRR_GPRS_SI13_IND, MAX_L2_FRAME_SIZE * BITS_PER_BYTE); gprs_ind->cause = GPRS_CELL_NOT_SUITABLE; gprs_ind->serving_cell_info.bcch_arfcn = rr_data->nc_data[CR_INDEX].arfcn; gprs_ind->serving_cell_info.bcch_bsic = rr_data->nc_data[CR_INDEX].bsic; TRACE_EVENT_P4 ("cs=%u C[%d] bsic=%u #%u", gprs_ind->cause, gprs_ind->serving_cell_info.bcch_arfcn, gprs_ind->serving_cell_info.bcch_bsic, __LINE__); SET_STATE(STATE_GPRS, GPRS_PIM_PBCCH); PSENDX(GRR, gprs_ind); } void att_gprs_start_task (T_RRGRR_START_TASK_REQ* start_task) { GET_INSTANCE_DATA; switch(GET_STATE(STATE_GPRS)) { case GPRS_PIM_PBCCH: case GPRS_PAM_PBCCH: case GPRS_PTM_PBCCH: switch(start_task->state) { case TASK_STATE_PTM: { PALLOC(mph_mon_ctrl_req, MPH_MON_CTRL_REQ ); mph_mon_ctrl_req->action = ENTER_PTM_PBCCH; PSENDX (PL, mph_mon_ctrl_req); SET_STATE(STATE_GPRS, GPRS_PTM_PBCCH); } break; case TASK_STATE_PIM: { PALLOC(mph_mon_ctrl_req, MPH_MON_CTRL_REQ ); mph_mon_ctrl_req->action = ENTER_PIM_PBCCH; PSENDX (PL, mph_mon_ctrl_req); SET_STATE(STATE_GPRS, GPRS_PIM_PBCCH); #ifdef REL99 if(rr_data->gprs_data.cbch_info_rxvd_in_ptm) { rr_data->gprs_data.cbch_info_rxvd_in_ptm = FALSE; att_config_cbch(); } #endif } break; case TASK_STATE_PAM: { SET_STATE(STATE_GPRS, GPRS_PAM_PBCCH); } break; default: break; } break; case GPRS_PIM_BCCH: case GPRS_PAM_BCCH: case GPRS_PTM_BCCH: switch(start_task->state) { case TASK_STATE_PTM: { PALLOC(mph_mon_ctrl_req, MPH_MON_CTRL_REQ ); mph_mon_ctrl_req->action = ENTER_PTM_BCCH; PSENDX (PL, mph_mon_ctrl_req); } break; default: break; } break; default: break; } { PALLOC(stop_cnf, RRGRR_START_TASK_CNF); PSENDX(GRR, stop_cnf); } PFREE(start_task); } void att_start_cell_reselection_pbcch(UBYTE mode) { GET_INSTANCE_DATA; att_init_cell_selection(CELL_RESELECTION, RR_ORIGINATED); att_init_pl_status (); rr_data->bcch_error = 0; 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); gprs_init_data_cr(); srv_clear_list (&rr_data->cr_data.cd.ncell_list); dat_att_null(); rr_data->gprs_data.cr_pbcch_active = TRUE; SET_STATE (STATE_ATT, ATT_CS3); att_build_idle_req (CR_INDEX, mode); } /* +------------------------------------------------------------------------------ | Function : rr_ext_meas_can_start +------------------------------------------------------------------------------ | Description : Check if the Ext Meas procedure can be performed. | | Parameters : none | +------------------------------------------------------------------------------ */ static BOOL rr_ext_meas_can_start ( void ) { GET_INSTANCE_DATA; if( GET_STATE(STATE_ATT) EQ ATT_IDLE ) { if( (GET_STATE(STATE_GPRS) EQ GPRS_PIM_BCCH) OR (GET_STATE ( STATE_GPRS ) EQ GPRS_PIM_PBCCH) ) return TRUE; } return FALSE; } /* +------------------------------------------------------------------------------ | Function : att_rrgrr_ext_meas_req +------------------------------------------------------------------------------ | Description : Start Extended Measurement (GPRS) at RR. | | Parameters : *rrgrr_ext_meas_req - Frequencies to be measured | +------------------------------------------------------------------------------ */ GLOBAL void att_rrgrr_ext_meas_req ( T_RRGRR_EXT_MEAS_REQ *rrgrr_ext_meas_req ) { GET_INSTANCE_DATA; T_GPRS_DATA *gprs_data = &rr_data->gprs_data; UBYTE i,n; T_arfcn_idx *s; USHORT *d; /* Store the primitive 'rrgrr_ext_meas_req' during the Ext Meas procedure. */ if ( gprs_data->rrgrr_ext_meas_req NEQ NULL AND gprs_data->rrgrr_ext_meas_req NEQ rrgrr_ext_meas_req ) { PFREE ( gprs_data->rrgrr_ext_meas_req ); } gprs_data->rrgrr_ext_meas_req = rrgrr_ext_meas_req; if ( rr_ext_meas_can_start() ) { PALLOC ( mph_ext_meas_req, MPH_EXT_MEAS_REQ ); SET_STATE (STATE_CELL_SEL, CS_XMEAS); n = rrgrr_ext_meas_req->c_arfcn_idx; if ( n > RRGRR_MAX_ARFCN_EXT_MEAS ) { n = RRGRR_MAX_ARFCN_EXT_MEAS; TRACE_ERROR ( "cut freq list (in)" ); } if ( n > MAX_CHANNELS ) { n = MAX_CHANNELS; TRACE_ERROR ( "cut freq list (out)" ); } mph_ext_meas_req->num_of_chan = n; d = &mph_ext_meas_req->arfcn[0]; s = &rrgrr_ext_meas_req->arfcn_idx[0]; for ( i = 0; i < n; ++i, ++s, ++d ) *d = s->arfcn; /* Other elements of mph_ext_meas_req remain uninitialized here (set in ALR). */ mph_ext_meas_req->freq_bands = cs_get_freq_band(FALSE); gprs_data->ext_meas_ctrl = 0; PSENDX (PL, mph_ext_meas_req); } else { PALLOC ( rrgrr_ext_meas_cnf, RRGRR_EXT_MEAS_CNF ); rrgrr_ext_meas_cnf->c_xmeas_res = 0; rrgrr_ext_meas_cnf->xmeas_cause = EXT_MEAS_RESET; rrgrr_ext_meas_cnf->call_ref = rrgrr_ext_meas_req->call_ref; PSENDX ( GRR, rrgrr_ext_meas_cnf ); PFREE ( rrgrr_ext_meas_req ); gprs_data->rrgrr_ext_meas_req = NULL; } return; } /* +------------------------------------------------------------------------------ | Function : rr_ext_meas_idx +------------------------------------------------------------------------------ | Description : Retrieve InDeX associated to ARFCN during | Extended Measurement (GPRS). | | Parameters : ARFCN - channel number in 'rrgrr_ext_meas_req' | +------------------------------------------------------------------------------ */ GLOBAL UBYTE rr_ext_meas_idx (USHORT arfcn ) { GET_INSTANCE_DATA; UBYTE i,n; T_arfcn_idx *arfcn_idx; T_RRGRR_EXT_MEAS_REQ *rrgrr_ext_meas_req; rrgrr_ext_meas_req = rr_data->gprs_data.rrgrr_ext_meas_req; if ( rrgrr_ext_meas_req EQ NULL ) { return NOT_PRESENT_8BIT; } n = rrgrr_ext_meas_req->c_arfcn_idx; arfcn_idx = &rrgrr_ext_meas_req->arfcn_idx[0]; for ( i = 0; i < n; ++i, ++arfcn_idx ) { if ( arfcn_idx->arfcn EQ arfcn ) return arfcn_idx->idx; } return NOT_PRESENT_8BIT; } /* +------------------------------------------------------------------------------ | Function : rr_ext_meas_end +------------------------------------------------------------------------------ | Description : Deallocate memory used during Extended Measurement (GPRS). | This function is called in three scenarios: | (1) System start (pei_init) | (2) Cell Change triggered via gprs_init_gprs_data | (3) End of Extended Measurement procedure | | Parameters : none | +------------------------------------------------------------------------------ */ GLOBAL void att_ext_meas_end ( BOOL destory_rrgrr_ext_meas_cnf ) { GET_INSTANCE_DATA; /* Note : If att_ext_meas_end is called via gprs_init_gprs_data and * Ext Meas is running, then a repsonse from Lower Layers is still * to be expected, which will be ignored. */ T_GPRS_DATA *gprs_data = &rr_data->gprs_data; if ( gprs_data->rrgrr_ext_meas_req NEQ NULL ) { PFREE ( gprs_data->rrgrr_ext_meas_req ); gprs_data->rrgrr_ext_meas_req = NULL; } if ( gprs_data->rrgrr_ext_meas_cnf NEQ NULL ) { if ( destory_rrgrr_ext_meas_cnf ) { PFREE ( gprs_data->rrgrr_ext_meas_cnf ); } gprs_data->rrgrr_ext_meas_cnf = NULL; } if ( gprs_data->mph_ext_meas_cnf NEQ NULL ) { PFREE ( gprs_data->mph_ext_meas_cnf ); gprs_data->mph_ext_meas_cnf = NULL; } if ( (gprs_data->ext_meas_ctrl & EXT_MEAS_START_CR) NEQ 0 ) { /* handle the outstanding Cell Reselection request */ gprs_data->ext_meas_ctrl &= ~EXT_MEAS_START_CR; att_start_cell_reselection(rr_data->sc_data.selection_type); } gprs_data->ext_meas_ctrl = 0; } /* +------------------------------------------------------------------------------ | Function : att_ext_meas_next_bsic +------------------------------------------------------------------------------ | Description : Check for next BSIC scanning during Extended Measurement. | If all BSICs are scanned, then a response is sent to GRR. | | Parameters : none | +------------------------------------------------------------------------------ */ GLOBAL void att_ext_meas_next_bsic (void) { GET_INSTANCE_DATA; T_GPRS_DATA *gprs_data = &rr_data->gprs_data; T_RRGRR_EXT_MEAS_REQ *rrgrr_ext_meas_req = gprs_data->rrgrr_ext_meas_req; T_RRGRR_EXT_MEAS_CNF *rrgrr_ext_meas_cnf = gprs_data->rrgrr_ext_meas_cnf; T_MPH_EXT_MEAS_CNF *mph_ext_meas_cnf = gprs_data->mph_ext_meas_cnf; UBYTE n = gprs_data->mph_ext_meas_num; UBYTE stop = gprs_data->ext_meas_ctrl & EXT_MEAS_RESET; if ( (rrgrr_ext_meas_req->report_type EQ REP_TYPE_1 OR rrgrr_ext_meas_req->report_type EQ REP_TYPE_2 ) AND rrgrr_ext_meas_cnf->c_xmeas_res >= 6 OR rrgrr_ext_meas_cnf->c_xmeas_res >= RRGRR_MAX_ARFCN_EXT_MEAS OR n >= mph_ext_meas_cnf->num_of_chan OR stop ) { /* stop PLMN search at lower layers */ PALLOC (mph_sync_req, MPH_SYNC_REQ); mph_sync_req->cs = CS_STOP_PLMN_SEARCH; PSENDX (PL, mph_sync_req); } else { PALLOC ( mph_bsic_req, MPH_BSIC_REQ ); mph_bsic_req->arfcn = mph_ext_meas_cnf->arfcn[n]; PSENDX ( PL, mph_bsic_req ); } } /* +------------------------------------------------------------------------------ | Function : att_mph_ext_meas_cnf +------------------------------------------------------------------------------ | Description : Evaluate rxlev during Start Extended Measurement (GPRS). | | Parameters : *mph_ext_meas_cnf - ARFCNs and RXLEVs from MPH | +------------------------------------------------------------------------------ */ GLOBAL void att_mph_ext_meas_cnf ( T_MPH_EXT_MEAS_CNF *mph_ext_meas_cnf ) { GET_INSTANCE_DATA; T_GPRS_DATA *gprs_data = &rr_data->gprs_data; T_RRGRR_EXT_MEAS_REQ *rrgrr_ext_meas_req = gprs_data->rrgrr_ext_meas_req; /* Check that Ext Meas is still active at RR */ if ( rrgrr_ext_meas_req NEQ NULL ) { PALLOC ( rrgrr_ext_meas_cnf, RRGRR_EXT_MEAS_CNF ); gprs_data->rrgrr_ext_meas_cnf = rrgrr_ext_meas_cnf; /* Depending on the Reporting Type either obtain BSICs or respond immediately. */ if ( rrgrr_ext_meas_req->report_type EQ REP_TYPE_1 OR rrgrr_ext_meas_req->report_type EQ REP_TYPE_2 ) { /* allocate RRGRR_EXT_MEAS_CNF, save mph_ext_meas_cnf and send initial BSIC_REQ */ rrgrr_ext_meas_cnf->c_xmeas_res = 0; gprs_data->mph_ext_meas_cnf = mph_ext_meas_cnf; gprs_data->mph_ext_meas_num = 0; att_ext_meas_next_bsic(); } else { /* With REP_TYPE_3 no BSIC scanning is needed, RRGRR_EXT_MEAS_CNF is sent. */ /* Note: the premature end of the Ext Meas procedure could have been requested, */ /* but this 'normal' procedure end is performed instead (now). */ UBYTE i,n; T_xmeas_res *xmeas_res; PALLOC (mph_sync_req, MPH_SYNC_REQ); n = mph_ext_meas_cnf->num_of_chan; if ( n > RRGRR_MAX_ARFCN_EXT_MEAS ) { n = RRGRR_MAX_ARFCN_EXT_MEAS; TRACE_ERROR ( "cut freq list (meas_cnf)" ); } rrgrr_ext_meas_cnf->c_xmeas_res = n; xmeas_res = &rrgrr_ext_meas_cnf->xmeas_res[0]; for ( i = 0; i < n; ++i, ++xmeas_res ) { USHORT arfcn = mph_ext_meas_cnf->arfcn[i] & ARFCN_MASK; UBYTE idx = rr_ext_meas_idx ( arfcn ); xmeas_res->arfcn_idx.arfcn = arfcn; xmeas_res->arfcn_idx.idx = idx; xmeas_res->rxlev = mph_ext_meas_cnf->rx_lev[i]; xmeas_res->bsic = 0; } PFREE(mph_ext_meas_cnf); mph_sync_req->cs = CS_STOP_PLMN_SEARCH; PSENDX (PL, mph_sync_req); } } else { PFREE(mph_ext_meas_cnf); } return; } /* +------------------------------------------------------------------------------ | Function : att_rrgrr_ext_meas_stop_req +------------------------------------------------------------------------------ | Description : Premature stop of Extended Measurement (GPRS). | | Parameters : *rrgrr_ext_meas_stop_req - Signal. | +------------------------------------------------------------------------------ */ GLOBAL void att_rrgrr_ext_meas_stop_req ( T_RRGRR_EXT_MEAS_STOP_REQ *rrgrr_ext_meas_stop_req) { GET_INSTANCE_DATA; T_GPRS_DATA *gprs_data = &rr_data->gprs_data; T_RRGRR_EXT_MEAS_REQ *rrgrr_ext_meas_req = gprs_data->rrgrr_ext_meas_req; if (rrgrr_ext_meas_stop_req) PFREE ( rrgrr_ext_meas_stop_req ); if ( rrgrr_ext_meas_req EQ NULL ) return; gprs_data->ext_meas_ctrl |= EXT_MEAS_RESET; /* Continue the stop of the Ext Meas procedure with the next response from Lower Layers. */ } /* +------------------------------------------------------------------------------ | Function : att_rrgrr_standby_ind +------------------------------------------------------------------------------ | Description : Store information that MS is in Standby State. | | Parameters : T_RRGRR_STANDBY_STATE_IND | +------------------------------------------------------------------------------ */ void att_rrgrr_standby_ind(T_RRGRR_STANDBY_STATE_IND* stdby) { GET_INSTANCE_DATA; rr_data->gprs_data.ready_state = FALSE; PFREE(stdby); } /* +------------------------------------------------------------------------------ | Function : att_rrgrr_ready_ind +------------------------------------------------------------------------------ | Description : Store information that MS is in Ready State. | | Parameters : T_RRGRR_READY_STATE_IND | +------------------------------------------------------------------------------ */ void att_rrgrr_ready_ind(T_RRGRR_READY_STATE_IND* rdy) { GET_INSTANCE_DATA; rr_data->gprs_data.ready_state = TRUE; PFREE(rdy); } #ifdef REL99 /* +------------------------------------------------------------------------------ | Function : att_rrgrr_cbch_info_ind +------------------------------------------------------------------------------ | Description : Stores the CBCH configuration received from GRR. | This information corresponds to the cbch channel | description received on PBCCH. | This primitive upates RR with latest CBCH information | whenever new PSI8, PSI2 or PSI13 is received by GRR | in PIM or PTM. This primitive should not come when | the mobile has camped on a R97/98 network. | | Parameters : cbch_inf points to RRGRR_CBCH_INFO_IND primitive. | +------------------------------------------------------------------------------ */ void att_rrgrr_cbch_info_ind(T_RRGRR_CBCH_INFO_IND* cbch_inf) { GET_INSTANCE_DATA; /* we should now be in R-99 network and in a cell having * pbcch. Otherwise this message should not be received. */ TRACE_FUNCTION("att_rrgrr_cbch_info_ind"); if(att_gprs_get_nw_release() AND att_gprs_cell_has_pbcch()) { /* Store the CBCH configuration in serving cell context */ TRACE_EVENT("CBCH info on PBCCH received"); rr_data->gprs_data.cbch_psi_valid = TRUE; memcpy(&(rr_data->gprs_data.cbch_psi8),&(cbch_inf->cbch),sizeof(T_cbch)); if( GET_STATE(STATE_ATT) EQ ATT_IDLE) { att_config_cbch(); } } else { TRACE_ERROR( " CBCH Info received when PBCCH not present or when in R97 cell"); } PFREE(cbch_inf); } #endif /* +------------------------------------------------------------------------------ | Function : is_nc2_used +------------------------------------------------------------------------------ | Description : Check if the cell reselection should be done | or not. | Parameters : none | Return Value: TRUE -> don't do cell reselection | FALSE -> do cell reselection +------------------------------------------------------------------------------ */ BOOL is_nc2_used(void) { GET_INSTANCE_DATA; BOOL ret = FALSE; UBYTE st = GET_STATE(STATE_GPRS); TRACE_FUNCTION("is_nc2_used()"); if (st EQ GPRS_NULL OR st EQ GPRS_ACTIVATED) return ret; /* * Check: * Ready State * NC mode of PMO * NC mode of SI13 */ TRACE_EVENT_P3("nc2: rdy: %d, pmo_nc: %d, si13 nc: %d", rr_data->gprs_data.ready_state, rr_data->gprs_data.nc_mode_of_pmo, rr_data->gprs_data.is_nc2_used_in_si13); if(rr_data->gprs_data.ready_state) { /* we are in READY */ if(rr_data->gprs_data.nc_mode_of_pmo NEQ NC_MODE_RESET) { if(rr_data->gprs_data.nc_mode_of_pmo EQ NC2_USED) ret = TRUE; else ret = FALSE; } else { /* use SI13 NC mode */ ret = rr_data->gprs_data.is_nc2_used_in_si13; } } else { /* we are in STANDBY */ /* in STANDBY NC mode is not used */ ret = FALSE; } return ret; } #ifdef REL99 /* +------------------------------------------------------------------------------ | Function : att_gprs_get_nw_release +------------------------------------------------------------------------------ | Description : returns the network release received in SI 13 | Parameters : none | Return Value: RR_GPRS_R97, RR_GPRS_R99 depending on the network | release. +------------------------------------------------------------------------------ */ UBYTE att_gprs_get_nw_release(void) { GET_INSTANCE_DATA; return rr_data->gprs_data.nw_release; } #endif /*===========================================================================*/ /* */ /* */ /* L O C A L S */ /* */ /* */ /*===========================================================================*/ /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_get_scell_info | +-----------------------------------------------------------------------------+ PURPOSE : fill the serving cell struct */ static void att_get_scell_info (T_serving_cell_info* sc, UBYTE idx) { GET_INSTANCE_DATA; sc->gprs_ms_txpwr_max_cch = rr_data->nc_data[idx].select_para.ms_txpwr_max_cch; sc->gprs_rxlev_access_min = rr_data->nc_data[idx].select_para.rxlev_access_min; sc->bcch_arfcn = rr_data->nc_data[idx].arfcn; sc->bcch_bsic = rr_data->nc_data[idx].bsic; sc->ac_class = rr_data->nc_data[idx].rach.ac; /* air-interface coding */ sc->pwr_offset = (UBYTE)(rr_data->nc_data[idx].c2_par.power_off_ind ? rr_data->nc_data[idx].c2_par.power_off : 0); if (!(dat_forb_lai_check (SC_INDEX) AND dat_roam_forb_lai_check (SC_INDEX))) sc->limited = 1; else sc->limited = (UBYTE)(rr_data->ms_data.rr_service EQ LIMITED_SERVICE); sc->rr_cell_env.plmn.v_plmn = TRUE; memcpy (sc->rr_cell_env.plmn.mcc, rr_data->nc_data[idx].lai.mcc, SIZE_MCC); memcpy (sc->rr_cell_env.plmn.mnc, rr_data->nc_data[idx].lai.mnc, SIZE_MNC); sc->rr_cell_env.lac = rr_data->nc_data[idx].lai.lac; sc->rr_cell_env.cid = rr_data->nc_data[idx].cell_id; sc->bs_pa_mfrms = rr_data->nc_data[idx].control_descr.bs_pa_mfrms; } /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_gprs_store_nc_mode | +-----------------------------------------------------------------------------+ PURPOSE : store the NC mode of SI13 */ static void att_gprs_store_nc_mode(T_D_SYS_INFO_13* sys_info_13) { GET_INSTANCE_DATA; if( !sys_info_13->si13_rest_oct.si13_info.flag1 AND sys_info_13->si13_rest_oct.si13_info.nco EQ NCO_NET_RESEL) { rr_data->gprs_data.is_nc2_used_in_si13 = TRUE; } else { rr_data->gprs_data.is_nc2_used_in_si13 = FALSE; } } #ifdef REL99 /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : att_gprs_store_nw_release | +-----------------------------------------------------------------------------+ PURPOSE : stores the network release received in si13 */ static void att_gprs_store_nw_release(T_D_SYS_INFO_13* sys_info_13) { GET_INSTANCE_DATA; if(sys_info_13->si13_rest_oct.v_si13_info) { if(sys_info_13->si13_rest_oct.si13_info.v_sgsnr) { rr_data->gprs_data.nw_release = RR_GPRS_R99; } else { rr_data->gprs_data.nw_release = RR_GPRS_R97; } } else { rr_data->gprs_data.nw_release = RR_GPRS_R97; } } #ifdef TI_PS_FF_EMR /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-GPRS (6147) MODULE : RR_GPRS | | STATE : code ROUTINE : attg_send_enh_para_to_grr | +-----------------------------------------------------------------------------+ PURPOSE : To format and send a primitive indicating enhanced measurement parameters received from SI-2quater. */ GLOBAL void att_send_enh_para_to_grr(T_rr_enh_para *p_src) { PALLOC(p_enh,RRGRR_SI2QUATER_IND); TRACE_FUNCTION("att_send_enh_para_to_grr"); p_enh->rep_type = p_src->grr_rep_type; p_enh->enh_para_struct = p_src->enh_para; p_enh->nc_para_struct = p_src->nc_para; PSENDX(GRR, p_enh); return; } #endif #endif #endif /* GPRS */ #endif /* !RR_GPRS_C */