FreeCalypso > hg > fc-tourmaline
diff src/g23m-gprs/grlc/grlc_f.c @ 1:fa8dc04885d8
src/g23m-*: import from Magnetite
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Fri, 16 Oct 2020 06:25:50 +0000 |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/g23m-gprs/grlc/grlc_f.c Fri Oct 16 06:25:50 2020 +0000 @@ -0,0 +1,2394 @@ +/* ++----------------------------------------------------------------------------- +| Project : GPRS (8441) +| Modul : GRLC ++----------------------------------------------------------------------------- +| 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 implements global functions for GRLC ++----------------------------------------------------------------------------- +*/ + +#ifndef GRLC_F_C +#define GRLC_F_C +#endif + +#define ENTITY_GRLC + + +/*==== INCLUDES =============================================================*/ + +#include <stdio.h> +#include <string.h> /* to get definition of memcpy() */ +#include <math.h> + +#include "typedefs.h" /* to get Condat data types */ +#include "vsi.h" /* to get a lot of macros */ +#include "macdef.h" +#include "gprs.h" +#include "gsm.h" /* to get a lot of macros */ +#include "ccdapi.h" /* to get CCD API */ +#include "cnf_grlc.h" /* to get cnf-definitions */ +#include "mon_grlc.h" /* to get mon-definitions */ +#include "prim.h" /* to get the definitions of used SAP and directions */ +#include "message.h" /* to get message describtion */ +#include "pcm.h" + +#include "grlc.h" /* to get the global entity definitions */ + +#include "grlc_f.h" /* to check own definitions */ +#include "grlc_tmf.h" /* to get definition of tm_grlc_init() */ +#include "grlc_gfff.h" /* to get definition of gff_init() */ +#include "grlc_rdf.h" /* to get definition of rd_init() */ +#include "grlc_ruf.h" /* to get definition of ru_init() */ +#include "grlc_rus.h" +#include "grlc_rds.h" +#include "grlc_tms.h" +#include "grlc_tpcs.h" +#include "grlc_meass.h" +#include "cl_rlcmac.h" + +/*==== CONST ================================================================*/ + +/*==== DIAGNOSTICS ==========================================================*/ + +/*==== LOCAL VARS ===========================================================*/ + +/*==== GLOBAL VARS ===========================================================*/ +/*==== LOCAL MACROS =========================================================*/ + + +/*==== FUNCTIONS PROTOTYPES =================================================*/ + +LOCAL void grlc_encode_dl_acknack ( UBYTE * ptr_out ); + +LOCAL void grlc_encode_ul_dummy ( UBYTE * ptr_out ); + +LOCAL void grlc_encode_pca ( UBYTE * ptr_out ); + +LOCAL UBYTE grlc_decode_ul_acknack ( UBYTE * ptr_blk ); + +LOCAL UBYTE grlc_ccd_error_handling ( UBYTE entity_i ); + +LOCAL USHORT grlc_convert_11bit_2_etsi ( USHORT eleven_bit ); + +/*==== FUNCTIONS ============================================================*/ +/* ++------------------------------------------------------------------------------ +| Function : grlc_encode_dl_acknack ++------------------------------------------------------------------------------ +| Description : The function grlc_encode_dl_acknack() encodes the packet downlink +| ack/nack block without CCD +| +| Parameters : *ptr_out_i- ptr to the buffer where the air message will be placed +| +| ++------------------------------------------------------------------------------ +*/ +LOCAL void grlc_encode_dl_acknack ( UBYTE * ptr_out) +{ + MCAST (u_dl_ack,U_GRLC_DL_ACK); /* T_U_GRLC_DL_ACK */ + + UBYTE i=0, bit=0,byte,bit_in_byte,j; + + TRACE_FUNCTION( "grlc_encode_dl_acknack" ); + + /******************* mandatory elements ****************************/ + + /* MESSAGE TYPE 6 bit */ + ptr_out[0] = u_dl_ack->msg_type << 2; + bit +=6; + /* DL TFI 5 bit */ + ptr_out[0] |= (u_dl_ack->dl_tfi >> 3) & 0x03; + ptr_out[1] = (u_dl_ack->dl_tfi << 5); + bit +=5; + /* ACK NACK DESCRIPTION */ + /* final ack indication 1 bit */ + ptr_out[1] |= (u_dl_ack->ack_nack_des.f_ack_ind << 4); + bit +=1; + /* ssn 7 bit */ + ptr_out[1] |= (u_dl_ack->ack_nack_des.ssn >> 3); + ptr_out[2] = (u_dl_ack->ack_nack_des.ssn << 5); + bit +=7; + for (i=0;i<64; i++) + { + byte = bit / 8; /* byte pos */ + bit_in_byte = bit % 8; /* rel bit pos in the current byte */ + ptr_out[byte] |= u_dl_ack->ack_nack_des.rbb[i] << (7-bit_in_byte); + bit +=1; + } + /* CHANNEL REQUEST DESCRIPTION */ + /* valid flag*/ + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= u_dl_ack->v_chan_req_des << (7-bit_in_byte); + bit +=1; + if(u_dl_ack->v_chan_req_des) + { + /* peak_thr_class */ + for(i=0;i<4;i++) + { + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (u_dl_ack->chan_req_des.peak_thr_class >> (3-i)) << (7-bit_in_byte); + bit +=1; + } + /* radio prio */ + for(i=0;i<2;i++) + { + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (u_dl_ack->chan_req_des.radio_prio >> (1-i)) << (7-bit_in_byte); + bit +=1; + } + /* rlc_mode*/ + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (u_dl_ack->chan_req_des.rlc_mode) << (7-bit_in_byte); + bit +=1; + /* llc pdu type*/ + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (u_dl_ack->chan_req_des.llc_pdu_type) << (7-bit_in_byte); + bit +=1; + /* rlc_oct_cnt */ + for(i=0;i<16;i++) + { + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (u_dl_ack->chan_req_des.rlc_octet_cnt >> (15-i)) << (7-bit_in_byte); + bit +=1; + } + } + /* CHANNEL QUALITY REPORT */ + /* c_value */ + for(i=0;i<6;i++) + { + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (u_dl_ack->chan_qual_rep.c_value >> (5-i)) << (7-bit_in_byte); + bit +=1; + } + /* rxqual */ + for(i=0;i<3;i++) + { + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (u_dl_ack->chan_qual_rep.rxqual >> (2-i)) << (7-bit_in_byte); + bit +=1; + } + /* signvar */ + for(i=0;i<6;i++) + { + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (u_dl_ack->chan_qual_rep.signvar >> (5-i)) << (7-bit_in_byte); + bit +=1; + } + /* c_value */ + for(j=0;j<8;j++) + { + UBYTE flag,value; + + switch(j) + { + case 0: + flag = u_dl_ack->chan_qual_rep.ilev.v_ilev0; + value = u_dl_ack->chan_qual_rep.ilev.ilev0; + break; + case 1: + flag = u_dl_ack->chan_qual_rep.ilev.v_ilev1; + value = u_dl_ack->chan_qual_rep.ilev.ilev1; + break; + case 2: + flag = u_dl_ack->chan_qual_rep.ilev.v_ilev2; + value = u_dl_ack->chan_qual_rep.ilev.ilev2; + break; + case 3: + flag = u_dl_ack->chan_qual_rep.ilev.v_ilev3; + value = u_dl_ack->chan_qual_rep.ilev.ilev3; + break; + case 4: + flag = u_dl_ack->chan_qual_rep.ilev.v_ilev4; + value = u_dl_ack->chan_qual_rep.ilev.ilev4; + break; + case 5: + flag = u_dl_ack->chan_qual_rep.ilev.v_ilev5; + value = u_dl_ack->chan_qual_rep.ilev.ilev5; + break; + case 6: + flag = u_dl_ack->chan_qual_rep.ilev.v_ilev6; + value = u_dl_ack->chan_qual_rep.ilev.ilev6; + break; + case 7: + flag = u_dl_ack->chan_qual_rep.ilev.v_ilev7; + value = u_dl_ack->chan_qual_rep.ilev.ilev7; + break; + default: + TRACE_EVENT_P1("no valid j=%d value during grlc_encode_dl_acknack should not appear ",j); + flag = 0; + value = 0; + break; + } + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (flag) << (7-bit_in_byte); + bit +=1; + if(flag) + { + for(i=0;i<4;i++) + { + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (value >> (3-i)) << (7-bit_in_byte); + bit +=1; + } + } + } + + /* RELEASE 99*/ +#ifdef REL99 + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= u_dl_ack->v_release_99_str_u_grlc_dl_ack << (7-bit_in_byte); + bit +=1; + + if(u_dl_ack->v_release_99_str_u_grlc_dl_ack) + { + byte = bit / 8; + bit_in_byte = bit % 8; + + if(u_dl_ack->release_99_str_u_grlc_dl_ack.v_pfi) + { + ptr_out[byte] |= u_dl_ack->release_99_str_u_grlc_dl_ack.v_pfi << (7-bit_in_byte); + bit +=1; + for(i=0;i<6;i++) + { + byte = bit / 8; + bit_in_byte = bit % 8; + ptr_out[byte] |= (u_dl_ack->release_99_str_u_grlc_dl_ack.pfi >> (6-i)) << (7-bit_in_byte); + bit +=1; + } + + } + else + { + ptr_out[byte] |= u_dl_ack->release_99_str_u_grlc_dl_ack.v_pfi << (7-bit_in_byte); + bit +=1; + } + } + + bit++; +#endif + + /* SPARE PADDINGS */ + byte = bit / 8; + bit_in_byte = bit % 8; + + if(bit_in_byte < 7) + { + UBYTE mask; + + mask= 0xff >> (bit_in_byte); + mask &= 0x2B; + ptr_out[byte] |= mask; + } + for(i=byte;i<22;i++) + { + byte++; + ptr_out[byte] = 0x2B; + + } +} /* grlc_encode_dl_acknack() */ + +/* ++------------------------------------------------------------------------------ +| Function : grlc_encode_ul_dummy ++------------------------------------------------------------------------------ +| Description : The function grlc_encode_ul_dummy() encodes the packet uplink +| dummy block without CCD +| +| Parameters : *ptr_out_i- ptr to the buffer where the air message will be placed +| +| ++------------------------------------------------------------------------------ +*/ +LOCAL void grlc_encode_ul_dummy ( UBYTE * ptr_out) +{ + MCAST (ul_dummy,U_GRLC_UL_DUMMY); /* T_U_GRLC_UL_DUMMY */ + + UBYTE i; + + TRACE_FUNCTION( "grlc_encode_ul_dummy" ); + + /******************* mandatory elements ****************************/ + + ptr_out[0] = ul_dummy->msg_type << 2; + ptr_out[0] |= (UBYTE) (grlc_data->uplink_tbf.tlli >> 30); /* 1100 0000 0000 0000 0000 0000 0000 0000 */ + ptr_out[1] = (UBYTE) ((grlc_data->uplink_tbf.tlli & 0x3FC00000) >> 22); /* 0011 1111 1100 0000 0000 0000 0000 0000 */ + ptr_out[2] = (UBYTE) ((grlc_data->uplink_tbf.tlli & 0x003FC000) >> 14); /* 0000 0000 0011 1111 1100 0000 0000 0000 */ + ptr_out[3] = (UBYTE) ((grlc_data->uplink_tbf.tlli & 0x00003FC0) >> 6); /* 0000 0000 0000 0000 0011 1111 1100 0000 */ + ptr_out[4] = (UBYTE) ((grlc_data->uplink_tbf.tlli & 0x0000003F) << 2); /* 0000 0000 0000 0000 0000 0000 0011 1111 */ + ptr_out[4] |= 0x01; /* spare paddings */ + + for(i=5;i<22;i++) + ptr_out[i] = 0x2B; +} /* grlc_encode_ul_dummy() */ + +/* ++------------------------------------------------------------------------------ +| Function : grlc_encode_pca ++------------------------------------------------------------------------------ +| Description : The function grlc_encode_pca() encodes the packet control +| acknowledgement block without CCD +| +| Parameters : *ptr_out_i- ptr to the buffer where the air message will be placed +| +| ++------------------------------------------------------------------------------ +*/ +LOCAL void grlc_encode_pca ( UBYTE * ptr_out) +{ + MCAST(u_ctrl_ack,U_GRLC_CTRL_ACK); /* T_U_GRLC_CTRL_ACK */ + + UBYTE i; + + TRACE_FUNCTION( "grlc_encode_pca" ); + + /******************* mandatory elements ****************************/ + + ptr_out[0] = u_ctrl_ack->msg_type << 2; + ptr_out[0] |= (UBYTE) (grlc_data->uplink_tbf.tlli >> 30); /* 1100 0000 0000 0000 0000 0000 0000 0000 */ + ptr_out[1] = (UBYTE) ((grlc_data->uplink_tbf.tlli & 0x3FC00000) >> 22); /* 0011 1111 1100 0000 0000 0000 0000 0000 */ + ptr_out[2] = (UBYTE) ((grlc_data->uplink_tbf.tlli & 0x003FC000) >> 14); /* 0000 0000 0011 1111 1100 0000 0000 0000 */ + ptr_out[3] = (UBYTE) ((grlc_data->uplink_tbf.tlli & 0x00003FC0) >> 6); /* 0000 0000 0000 0000 0011 1111 1100 0000 */ + ptr_out[4] = (UBYTE) ((grlc_data->uplink_tbf.tlli & 0x0000003F) << 2); /* 0000 0000 0000 0000 0000 0000 0011 1111 */ + ptr_out[4] |= u_ctrl_ack->pctrl_ack & 0x03; + + for(i=5;i<22;i++) + ptr_out[i] = 0x2B; +} /* grlc_encode_pca() */ +/* ++------------------------------------------------------------------------------ +| Function : grlc_decode_ul_acknack ++------------------------------------------------------------------------------ +| Description : The function grlc_decode_ul_acknack() decodes the packet uplink +| ack/nack without CCD +| +| Parameters : *ptr_blk- ptr to the air message +| +| Return value: returns decode status of air message:ccdOK,ccdWarning,ccdError +| ++------------------------------------------------------------------------------ +*/ +LOCAL UBYTE grlc_decode_ul_acknack ( UBYTE *ptr_blk ) +{ + MCAST(d_ul_ack,D_GRLC_UL_ACK);/* T_D_GRLC_UL_ACK */ + + UBYTE i,j,k; + UBYTE bit_pos; +#ifdef REL99 + UBYTE flag; +#endif + + UBYTE result =ccdOK; + + TRACE_FUNCTION( "grlc_decode_ul_acknack" ); + + /******************* mandatory elements ****************************/ + + d_ul_ack->msg_type = (ptr_blk[0] & 0xFC) >> 2 ; + d_ul_ack->page_mode = ptr_blk[0] & 0x03; + + if((ptr_blk[1] & 0xC0)) /* distrubiton part error check */ + { + TRACE_ERROR("P UL ACK: DISTRUBITION PART ERROR"); + TRACE_EVENT_P2("P UL ACK: DISTRUBITION PART ERROR byte = 0x2%x res = 0x%2x",ptr_blk[1] ,ptr_blk[1] & 0xC0); + return ccdError; + } + d_ul_ack->ul_tfi = (ptr_blk[1] & 0x3E) >> 1; + + if((ptr_blk[1] & 0x01)) /* message escape bit check */ + { +#ifdef REL99 + d_ul_ack->egprs_flag = TRUE; +#endif + TRACE_ERROR("P UL ACK: MESSAGE ESCAPE ERROR"); + TRACE_EVENT_P2("P UL ACK: MESSAGE ESCAPE ERROR byte = 0x2%x res = 0x%2x",ptr_blk[1] ,ptr_blk[1] & 0x01); + return ccdError; + } + + d_ul_ack->v_gprs_ul_ack_nack_info = TRUE; + + d_ul_ack->gprs_ul_ack_nack_info.chan_coding_cmd = (ptr_blk[2] & 0xC0) >> 6; + +#ifdef _SIMULATION_ + TRACE_EVENT_P2("tfi=%d ch_c_cmd=%d",d_ul_ack->ul_tfi,d_ul_ack->gprs_ul_ack_nack_info.chan_coding_cmd); +#endif /* _SIMULATION_ */ + /******************* Ack/Nack description ****************************/ + d_ul_ack->gprs_ul_ack_nack_info.ack_nack_des.f_ack_ind = (ptr_blk[2] & 0x20) >> 5; + d_ul_ack->gprs_ul_ack_nack_info.ack_nack_des.ssn = (ptr_blk[2] << 2) & 0x7C; + d_ul_ack->gprs_ul_ack_nack_info.ack_nack_des.ssn |= (ptr_blk[3] >> 6) & 0x03; + + j= 3; /* inital byte of rbb field */ + k= 2; /* inital bit of rbb field */ + + for(i=0; i< 64;i++) + { + d_ul_ack->gprs_ul_ack_nack_info.ack_nack_des.rbb[i] = (ptr_blk[j] & (0x80>>k)) >> (7-k); + k++; + if(k EQ 8) + { + k=0; + j++; + } + } +#ifdef _SIMULATION_ + TRACE_EVENT_P2("fai=%d ssn=%d",d_ul_ack->gprs_ul_ack_nack_info.ack_nack_des.f_ack_ind,d_ul_ack->gprs_ul_ack_nack_info.ack_nack_des.ssn); +#endif /* _SIMULATION_ */ + + bit_pos = 91; /* abs bit position */ + + /******************* contention resolution tlli ***********************/ + + j = bit_pos / 8; /* byte pos */ + k = bit_pos % 8; /* rel bit pos */ + + d_ul_ack->gprs_ul_ack_nack_info.v_cr_tlli = (ptr_blk[j] >> (8-k)) & 0x01; + + bit_pos++; + + if(d_ul_ack->gprs_ul_ack_nack_info.v_cr_tlli) + { + d_ul_ack->gprs_ul_ack_nack_info.cr_tlli.l_cr_tlli = 32; + d_ul_ack->gprs_ul_ack_nack_info.cr_tlli.o_cr_tlli = 3; + d_ul_ack->gprs_ul_ack_nack_info.cr_tlli.b_cr_tlli[0] = ptr_blk[j]; + d_ul_ack->gprs_ul_ack_nack_info.cr_tlli.b_cr_tlli[1] = ptr_blk[j+1]; + d_ul_ack->gprs_ul_ack_nack_info.cr_tlli.b_cr_tlli[2] = ptr_blk[j+2]; + d_ul_ack->gprs_ul_ack_nack_info.cr_tlli.b_cr_tlli[3] = ptr_blk[j+3]; + d_ul_ack->gprs_ul_ack_nack_info.cr_tlli.b_cr_tlli[4] = ptr_blk[j+4]; + bit_pos+=32; + } + +#ifdef REL99 + /******************* packet timing advance ****************************/ + j = bit_pos / 8; /* byte pos */ + k = bit_pos % 8; /* rel bit pos */ + flag = (ptr_blk[j] >> (8-k)) & 0x01; + bit_pos++; + if(flag) + { + } + /******************* power control params ****************************/ + j = bit_pos / 8; /* byte pos */ + k = bit_pos % 8; /* rel bit pos */ + flag = (ptr_blk[j] >> (8-k)) & 0x01; + bit_pos++; + if(flag) + { + } + /******************* Extension bits ****************************/ + j = bit_pos / 8; /* byte pos */ + k = bit_pos % 8; /* rel bit pos */ + flag = (ptr_blk[j] >> (8-k)) & 0x01; + bit_pos++; + if(flag) + { + } + /******************* Fixed Alloc ****************************/ + j = bit_pos / 8; /* byte pos */ + k = bit_pos % 8; /* rel bit pos */ + flag = (ptr_blk[j] >> (8-k)) & 0x01; + bit_pos++; + if(flag) + { + } + /******************* R99 FLAG ****************************/ + j = bit_pos / 8; /* byte pos */ + k = bit_pos % 8; /* rel bit pos */ + bit_pos++; + d_ul_ack->gprs_ul_ack_nack_info.v_release_99_str_d_ul_ack = (ptr_blk[j] >> (8-k)) & 0x01; + /******************* Extended PTA ****************************/ + j = bit_pos / 8; /* byte pos */ + k = bit_pos % 8; /* rel bit pos */ + flag = (ptr_blk[j] >> (8-k)) & 0x01; + bit_pos++; + if(flag) + { + } + /**********************************************************************/ + j = bit_pos / 8; /* byte pos */ + k = bit_pos % 8; /* rel bit pos */ + d_ul_ack->gprs_ul_ack_nack_info.release_99_str_d_ul_ack.tbf_est = (ptr_blk[j] >> (8-k)) & 0x01; + +#endif + + return(result); + +} /* grlc_decode_ul_acknack() */ + +/* ++------------------------------------------------------------------------------ +| Function : grlc_ccd_error_handling ++------------------------------------------------------------------------------ +| Description : The function grlc_ccd_error_handling() ... +| +| Parameters : entity_i - the CCD was called for this entity +| ++------------------------------------------------------------------------------ +*/ +LOCAL UBYTE grlc_ccd_error_handling ( UBYTE entity_i ) +{ + UBYTE result = DELETE_MESSAGE; + USHORT parlist [MAX_ERR_PAR]; + UBYTE first_error; + + TRACE_FUNCTION( "grlc_ccd_error_handling" ); + + + memset (parlist, 0, sizeof (parlist)); + + first_error = ccd_getFirstError (entity_i, parlist); + + switch (first_error) + { + + case ERR_PATTERN_MISMATCH: /* A spare pattern does not match with */ + /* the specified content */ + /* Error params[0] = bitposition */ + { + MCAST(ptr,D_GRLC_UL_ACK); + result = ptr->msg_type; + } + break; + + default: + /* SZML-GLBL/010 */ + TRACE_ERROR( "Ctrl-Message will be deleted" ); + break; + } + + return(result); + +} /* grlc_ccd_error_handling() */ + + +/*==== PUBLIC FUNCTIONS =====================================================*/ + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_buffer2ulong ++------------------------------------------------------------------------------ +| Description : The function grlc_buffer2ulong() copy a 32-Bit-Buffer in a ULONG +| variable +| +| SZML-GLBL/002 +| +| Parameters : ptmsi - pointer to buffer that contains the 32bit for the ULONG +| ++------------------------------------------------------------------------------ +*/ +GLOBAL ULONG grlc_buffer2ulong ( BUF_cr_tlli *tlli) +{ + ULONG ul; + + UBYTE l, dummy; + USHORT i, ii; + UBYTE off1, off2; + + TRACE_FUNCTION( "grlc_buffer2ulong "); + + ul= 0; + + l = (UBYTE)tlli->l_cr_tlli; + + off1 = tlli->o_cr_tlli / 8; + off2 = tlli->o_cr_tlli % 8; + + dummy = 0; + dummy = tlli->b_cr_tlli[off1] << off2; + + if(l <= (8-off2)) + { + dummy = dummy >> (8-l); + ul |= dummy; + return ul; + } + dummy = dummy >> off2; + ul |= dummy; + l -= (8-off2); + + do + { + off1++; + + if(l < 8) + { + dummy = tlli->b_cr_tlli[off1] >> (8-l); + ii = 1; + ul = ul << l; + for(i=0; i< l; i++) + { + ul = ul | (dummy & ii); + ii *= 2; + } + return ul; + } + else + { + ul = ul << 8; + ul |= tlli->b_cr_tlli[off1]; + l -= 8; + if(l EQ 0) + return ul; + } + } + while(TRUE); +} + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_delete_prim ++------------------------------------------------------------------------------ +| Description : The function grlc_delete_prim() deletes the primitive that is +| pointed by the grlc_data->prim_start_tbf and sets the +| grlc_data->prim_start_tbf to the next entry in the tbf list. +| +| Parameters : void +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_delete_prim ( void ) +{ + TRACE_FUNCTION( "grlc_delete_prim" ); + + /* + * access type is reseted + */ + grlc_data->uplink_tbf.access_type = CGRLC_AT_NULL; + + if(grlc_data->prim_start_tbf < PRIM_QUEUE_SIZE_TOTAL) + { + UBYTE i; + + /* + * get first entry from tbf list + */ + i = grlc_prim_get_first (&grlc_data->prim_start_tbf); + + + if(i >= PRIM_QUEUE_SIZE) + { + TRACE_EVENT_P5("delete prim VOR i=%d ps=%d, pf=%d,sps=%d,spf=%d", + i, + grlc_data->prim_start_tbf, + grlc_data->prim_start_free, + grlc_data->save_prim_start_tbf, + grlc_data->save_prim_start_free); + } + + /* + * estimate new user data amount in queue + */ + grlc_data->prim_user_data -= BYTELEN(grlc_data->prim_queue[i].prim_ptr->sdu.l_buf); + + /* + * free primitive before reset old primitive entry + */ + PFREE ( grlc_data->prim_queue[i].prim_ptr ); + + /* + * reset old primitive + */ + grlc_data->prim_queue[i].prim_ptr = NULL; + grlc_data->prim_queue[i].prim_type = CGRLC_LLC_PRIM_TYPE_NULL; + grlc_data->prim_queue[i].cv_status = FALSE; + grlc_data->prim_queue[i].rlc_status = FALSE; + grlc_data->prim_queue[i].re_allocation = FALSE; + grlc_data->prim_queue[i].start_new_tbf = FALSE; + grlc_data->prim_queue[i].last_bsn = 0xff; + grlc_data->prim_queue[i].previous = 0xff; + + /* + * put new entry at the end of free list + */ + + if(i < PRIM_QUEUE_SIZE) + { + grlc_prim_put(&grlc_data->prim_start_free,i,END_OF_LIST); + } + else if(grlc_data->gmm_procedure_is_running) + { + grlc_prim_put(&grlc_data->prim_start_free,i,END_OF_LIST); + TRACE_EVENT_P5("delete prim %d AFTER PST=%d, PSF=%d,spst=%d,spsf=%d", + i, + grlc_data->prim_start_tbf, + grlc_data->prim_start_free, + grlc_data->save_prim_start_tbf, + grlc_data->save_prim_start_free); + } + + /* + * update LLC flow control state + */ + if ((grlc_data->tm.send_grlc_ready_ind EQ PRIM_QUEUE_FULL) AND /* PRIM QUEUE IS FULL */ + ( + (!grlc_data->gmm_procedure_is_running AND (i < PRIM_QUEUE_SIZE)) OR /* LLC QUEUE IS ACTIVE, PDU FROM LLC QUEUE DELETED */ + (grlc_data->gmm_procedure_is_running AND !(i < PRIM_QUEUE_SIZE)))) /* GMM QUEUE IS ACTIVE, PDU FROM GMM QUEUE DELETED */ + { + TRACE_EVENT_P3("Flow control activated gmm_q=%d,i=%d,ready=%d",grlc_data->gmm_procedure_is_running,i,grlc_data->tm.send_grlc_ready_ind); + grlc_data->tm.send_grlc_ready_ind = SEND_A_GRLC_READY_IND; + } + + grlc_data->grlc_data_req_cnt--; + } + else + { + TRACE_EVENT_P3("PST=%d PSF=%d PDU=%d: grlc_delete_prim" + ,grlc_data->prim_start_tbf + ,grlc_data->prim_start_free + ,grlc_data->grlc_data_req_cnt); + return; + } + + +} /* grlc_delete_prim() */ + + + + + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_calc_new_poll_pos ++------------------------------------------------------------------------------ +| Description : The function grlc_calc_new_poll_pos() calculates the fn of the +| new poll position +| +| Parameters : fn_i - framenumber +| rrbp_i - relative position +| ++------------------------------------------------------------------------------ +*/ +GLOBAL ULONG grlc_calc_new_poll_pos ( ULONG fn_i, UBYTE rrbp_i ) +{ + ULONG result=0; + TRACE_FUNCTION( "grlc_calc_new_poll_pos" ); + + switch( rrbp_i ) + { + case 0: + result = (fn_i+13); + break; + case 1: + if((fn_i+18)%13) + result = (fn_i+17); + else + result = (fn_i+18); + break; + case 2: + if(((fn_i+21)%13) EQ 8) + result = (fn_i+21); + else + result = (fn_i+22); + break; + case 3: + result = (fn_i+26); + break; + default: + TRACE_ERROR( "unexpected rrbp value" ); + break; + } /* switch (rrbp_i) */ + result = result % 0x297000; + + return result; + +} /* grlc_calc_new_poll_pos() */ + + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_get_new_poll_index ++------------------------------------------------------------------------------ +| Description : The function grlc_get_new_poll_index() +| +| Parameters : +| ++------------------------------------------------------------------------------ +*/ +LOCAL UBYTE grlc_get_new_poll_index ( UBYTE * ptr_list_start_i ) +{ + UBYTE result; + TRACE_FUNCTION( "grlc_get_new_poll_index" ); + + result = *ptr_list_start_i; + /* set to new first entry */ + if (result NEQ 0xFF) + { + *ptr_list_start_i = grlc_data->next_poll_array[*ptr_list_start_i].next; + /* remove first entry from list */ + grlc_data->next_poll_array[result].next = 0xff; + } + else + { + TRACE_EVENT ("Poll array is full"); + } + + return(result); + +} /* grlc_get_new_poll_index() */ + + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_save_poll_pos ++------------------------------------------------------------------------------ +| Description : The function grlc_save_poll_pos() +| +| Parameters : fn_i - framenumber +| tn_i - timeslot number +| rrbp_i - fn of the poll block| +| poll_type_i - kind of dl data +| pctrl_ack_i - packet control ack value, needed fo p ctr ack msg +| +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_save_poll_pos ( ULONG fn_i, USHORT tn_i, UBYTE rrbp_i, UBYTE poll_type_i,UBYTE pctrl_ack_i) +{ + ULONG new_poll_pos; + UBYTE i, next, help_index; + + TRACE_FUNCTION( "grlc_save_poll_pos" ); + + if( tn_i >= POLL_TYPE_ARRAY_SIZE ) + { + TRACE_EVENT_P5( "grlc_save_poll_pos: fn = %d, tn = %d, rrbp = %d, poll_type = %d, pctrl_ack = %d", + fn_i, tn_i, rrbp_i, poll_type_i, pctrl_ack_i ); + + return; + } + + if(0xFF NEQ rrbp_i) + { + new_poll_pos = grlc_calc_new_poll_pos(fn_i, rrbp_i); + } + else + { + new_poll_pos = fn_i; + } + + /*TRACE_EVENT_P6("SAVE BEF: fn_i= %ld,rrbp=%ld,new_poll_pos=%ld, poll_type=%d,ps=%d ps_fn=%ld", + fn_i, + rrbp_i, + new_poll_pos, + poll_type_i, + grlc_data->poll_start_tbf, + grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn); + + */ + /* delete the poll position when it is older than 26 frames */ + while( grlc_data->poll_start_tbf NEQ 0xFF AND + grlc_check_dist(new_poll_pos, + grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn, 26) EQ FALSE) + { + /* move the expired poll position to the list of unused entries */ + TRACE_EVENT_P1("Remove expired poll at pst=%d",grlc_data->poll_start_tbf); + help_index = grlc_data->poll_start_tbf; + grlc_data->poll_start_tbf = grlc_data->next_poll_array[help_index].next; + grlc_data->next_poll_array[help_index].next = grlc_data->poll_start_free; + grlc_data->poll_start_free = help_index; + + /* clear the poll position */ + grlc_data->next_poll_array[help_index].fn = 0xFFFFFFFF; + grlc_data->next_poll_array[help_index].cnt = 0; + + for( i = 0; i < POLL_TYPE_ARRAY_SIZE; i++ ) + { + grlc_data->next_poll_array[help_index].poll_type[i] = CGRLC_POLL_NONE; + } + } + + next = 0xFF; + help_index = grlc_data->poll_start_tbf; + + + /* + * find the position of the new fn + */ + + while( help_index NEQ 0xFF ) + { + if( grlc_data->next_poll_array[help_index].fn EQ new_poll_pos ) + { + next = help_index; + help_index = 0xFF; + } + else if( grlc_check_dist( new_poll_pos, grlc_data->next_poll_array[help_index].fn, 26 ) ) + { + next = help_index; + help_index = grlc_data->next_poll_array[help_index].next; + } + else + { + help_index = 0xFF; + } + } + + /* + * new_poll_pos is present in the poll array + */ + if( next NEQ 0xFF AND + grlc_data->next_poll_array[next].fn EQ new_poll_pos AND + grlc_data->next_poll_array[next].poll_type[tn_i] EQ CGRLC_POLL_NONE ) + { + /* + * no collision + */ + grlc_data->next_poll_array[next].poll_type[tn_i] = poll_type_i; + grlc_data->next_poll_array[next].cnt++; + grlc_data->next_poll_array[next].ctrl_ack = pctrl_ack_i; + } + else if( next NEQ 0xFF AND + grlc_data->next_poll_array[next].fn EQ new_poll_pos ) + + { + /* + * collision detected, if both pos were received with data blocks, the sent + * RLC/MAC block otherwise send packet control ack. + */ + if ((poll_type_i NEQ CGRLC_POLL_DATA) + OR + (grlc_data->next_poll_array[next].poll_type[tn_i] NEQ CGRLC_POLL_DATA)) + { + grlc_data->next_poll_array[next].poll_type[tn_i] = CGRLC_POLL_COLLISION; + TRACE_EVENT("collision detected: pca will sent"); + } + else + { + TRACE_EVENT("collision detected: rlc/mac will be sent"); + } + if(pctrl_ack_i NEQ 3) + grlc_data->next_poll_array[next].ctrl_ack = pctrl_ack_i; + } + else + { + /* + * fn does not exist in the poll list, included in poll list + * + * get new free index from the free list + */ + i = grlc_get_new_poll_index(&(grlc_data->poll_start_free)); + if( i EQ 0xFF ) + { + TRACE_EVENT ("Poll array is full"); /*This should not happen */ + return; + } + + grlc_data->next_poll_array[i].cnt = 1; + grlc_data->next_poll_array[i].fn = new_poll_pos; + grlc_data->next_poll_array[i].poll_type[tn_i] = poll_type_i; + grlc_data->next_poll_array[i].ctrl_ack = pctrl_ack_i; + + if( next EQ 0xFF ) + { + /* + * first entry in poll array + */ + grlc_data->next_poll_array[i].next = grlc_data->poll_start_tbf; + grlc_data->poll_start_tbf = i; + } + else if( next < NEXT_POLL_ARRAY_SIZE ) + { + /* + * include in description list, is not first element + */ + grlc_data->next_poll_array[i].next = grlc_data->next_poll_array[next].next; + grlc_data->next_poll_array[next].next = i; + } + else + { + TRACE_ASSERT( next < NEXT_POLL_ARRAY_SIZE ); + } + } + + + /*TRACE_EVENT_P6("SAVE AFTER: fn_i= %ld,rrbp=%ld,new_poll_pos=%ld, poll_type=%d,ps=%d ps_fn=%ld", + fn_i, + rrbp_i, + new_poll_pos, + poll_type_i, + grlc_data->poll_start_tbf, + grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn); + */ + +} /* grlc_save_poll_pos() */ + + + + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_encode_ctrl ++------------------------------------------------------------------------------ +| Description : The function grlc_encode_ctrl() build a T_SDU buffer that +| contains the encode Ctrl Block ready to transmit. +| +| Parameters : ULONG ptr_in_i - ptr to the input structure +| ULONG ptr_out_i - ptr to begin of output buffer +| UBYTE r_bit_i - value of r_bit ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_encode_ctrl ( UBYTE * ptr_in_i, + T_MSGBUF * ptr_out_i, + UBYTE r_bit_i) +{ + TRACE_FUNCTION( "grlc_encode_ctrl" ); + + ptr_out_i->buf[0] = 0x40 | r_bit_i; + ptr_out_i->o_buf = BIT_UL_CTRL_BLOCK_MAC_HEADER; + ptr_out_i->l_buf = BIT_UL_CTRL_BLOCK_CONTENTS; + + + if(!grlc_data->grlc_wo_ccd) + { + ccd_codeMsg ( CCDENT_GRLC, + UPLINK, + ptr_out_i, + ptr_in_i, + NOT_PRESENT_8BIT); + } + else + { + switch(ptr_in_i[0]) /* msg_type */ + { + case U_GRLC_UL_DUMMY_c: + grlc_encode_ul_dummy(&ptr_out_i->buf[1]); + break; + case U_GRLC_CTRL_ACK_c: + grlc_encode_pca(&ptr_out_i->buf[1]); + break; + case U_GRLC_DL_ACK_c: + memset(&ptr_out_i->buf[1],0,22); /*lint !e419*/ + grlc_encode_dl_acknack(&ptr_out_i->buf[1]); + break; + default: + ccd_codeMsg ( CCDENT_GRLC, + UPLINK, + ptr_out_i, + ptr_in_i, + NOT_PRESENT_8BIT); + break; + } + } + ptr_out_i->l_buf += ptr_out_i->o_buf; + ptr_out_i->o_buf = 0; + +} /* grlc_encode_ctrl() */ + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_init ++------------------------------------------------------------------------------ +| Description : The function grlc_init initializes the entity GRLC +| +| Parameters : void +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_init ( void ) +{ + TRACE_FUNCTION( "grlc_init" ); + + /* initialize all GRLC data */ + grlc_data = &grlc_data_base; + memset( grlc_data, 0, sizeof( T_GRLC_DATA ) ); + + grlc_data->grlc_data_req_cnt = 0; + grlc_data->testmode.mode = CGRLC_NO_TEST_MODE; + + grlc_data->meas.sq_restart = TRUE; + + /* + * call of service init functions + */ + tm_grlc_init(); + rd_init(); + ru_init(); + meas_grlc_init(); + tpc_grlc_init(); + gff_init(); + + ccd_register(CCD_REENTRANT); + +#ifdef _SIMULATION_ + /* + * to make sure that the structure definitions of T_GRLC_DATA_REQ and + * T_GRLC_UNITDATA_REQ have the same layout. + */ + { + T_GRLC_DATA_REQ * ptr_grlc_data_req = (T_GRLC_DATA_REQ *)_decodedMsg; + T_GRLC_UNITDATA_REQ * ptr_grlc_unitdata_req = (T_GRLC_UNITDATA_REQ *)_decodedMsg; + + if( &(ptr_grlc_data_req->grlc_qos) NEQ &(ptr_grlc_unitdata_req->grlc_qos) AND + &(ptr_grlc_data_req->radio_prio) NEQ &(ptr_grlc_unitdata_req->radio_prio) AND + &(ptr_grlc_data_req->sdu) NEQ &(ptr_grlc_unitdata_req->sdu) AND + &(ptr_grlc_data_req->tlli) NEQ &(ptr_grlc_unitdata_req->tlli) ) + { + /* + * In this case the primitive handling will not work correctly!!! + */ + TRACE_ERROR("Fatal ERROR: T_GRLC_UNITDATA_REQ and T_GRLC_DATA_REQ are not equal!!"); + } + } /* _SIMULATION_ */ +#endif + + grlc_data->t3164_to_cnt = 0; + grlc_data->ul_tfi_changed = FALSE; + grlc_data->uplink_tbf.access_type = CGRLC_AT_NULL; /* NO CELL UPDATE NEED */ + grlc_data->grlc_wo_ccd = 0; /* as default ccd used for air message handling */ + + /* + * Ready Timer state initialization + */ +#ifdef FF_GRLC_4_TWO_2_ONE + grlc_data->ready_timer.handling = READY_TIMER_HANDLING_ENABLED; +#else + grlc_data->ready_timer.handling = READY_TIMER_HANDLING_DISABLED; +#endif + + grlc_data->ready_timer.state = STANDBY_STATE; + grlc_data->ready_timer.value = CGRLC_T3314_DEFAULT; + +} /* grlc_init() */ + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_prim_put ++------------------------------------------------------------------------------ +| Description : The function grlc_prim_put() put a Element (object_i) behind +| pos_i in the prim_queue. +| +| Parameters : list_start_i - address of the list should be manipulted +| object_i - index of element that should be added to the list +| pos_i - index of the position behind that the object +| should be added +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_prim_put(UBYTE * list_start_i, UBYTE object_i, UBYTE pos_i) +{ + TRACE_FUNCTION( "grlc_prim_put" ); + + if(* list_start_i EQ 0xff) + { /* no elements in the list */ + * list_start_i = object_i; + } + else + { /* elements in the list */ + UBYTE i = * list_start_i; + UBYTE j; + BOOL put= FALSE; + + /* + * SZML-GLBL/004 + */ + for(j=0;j<PRIM_QUEUE_SIZE_TOTAL;j++) + { + if (grlc_data->prim_queue[i].next NEQ pos_i) + { + i = grlc_data->prim_queue[i].next; + } + else + { + /* + * put new object at pos_i of queue + */ + put = TRUE; + grlc_data->prim_queue[object_i].next = grlc_data->prim_queue[i].next; + grlc_data->prim_queue[i].next = object_i; + break; + } + } + if(!put) + TRACE_EVENT_P3("PST=%d PSF=%d PDU=%d: grlc_prim_put failed" + ,grlc_data->prim_start_tbf + ,grlc_data->prim_start_free + ,grlc_data->grlc_data_req_cnt); + + } +} /* grlc_prim_put() */ + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_prim_get_first ++------------------------------------------------------------------------------ +| Description : The function grlc_prim_get_first() removes the first element +| and returns the index of the first element. +| +| Parameters : list_start_i - address of the list that should be used +| ++------------------------------------------------------------------------------ +*/ +GLOBAL UBYTE grlc_prim_get_first(UBYTE * list_start_i) +{ + UBYTE result; + TRACE_FUNCTION( "grlc_prim_get_first" ); + + result = *list_start_i; + + /* + * set to new first entry + */ + *list_start_i = grlc_data->prim_queue[*list_start_i].next; + + /* + * remove first entry from list + */ + grlc_data->prim_queue[result].next = 0xff; + + return(result); + +} /* grlc_prim_get_first() */ + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_set_packet_ctrl_ack ++------------------------------------------------------------------------------ +| Description : The function grlc_set_packet_ctrl_ack() | +| Parameters : +| ++------------------------------------------------------------------------------ +*/ +GLOBAL UBYTE * grlc_set_packet_ctrl_ack(void) +{ + MCAST(u_ctrl_ack,U_GRLC_CTRL_ACK); + UBYTE * result; + TRACE_FUNCTION( "grlc_set_packet_ctrl_ack" ); + + u_ctrl_ack->msg_type = U_GRLC_CTRL_ACK_c; + + grlc_set_buf_tlli( &u_ctrl_ack->tlli_value, grlc_data->uplink_tbf.tlli ); + + if(grlc_data->poll_start_tbf NEQ 0xFF) + u_ctrl_ack->pctrl_ack = grlc_data->next_poll_array[grlc_data->poll_start_tbf].ctrl_ack; + else /* response to IA poll */ + u_ctrl_ack->pctrl_ack = 3; + + result = _decodedMsg; + + return(result); + +} /* grlc_set_packet_ctrl_ack() */ + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_send_access_burst ++------------------------------------------------------------------------------ +| Description : The function grlc_send_access_burst() sents the poll as +| four access burst type +| +| Parameters : tn_i : timeslot where the access burst shall send +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_send_access_burst(UBYTE tn_i) +{ + TRACE_FUNCTION( "grlc_send_access_burst" ); + + + + if(grlc_data->poll_start_tbf NEQ 0xFF) + { + grlc_data->next_poll_array[grlc_data->poll_start_tbf].cnt--; + grlc_data->next_poll_array[grlc_data->poll_start_tbf].poll_type[tn_i] = CGRLC_POLL_NONE; + grlc_data->next_poll_fn = grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn; + } + grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].tn = tn_i; + if(grlc_data->ab_type EQ CGRLC_AB_11_BIT) + { + /*11 bit access burst*/ + USHORT elevenBit = 0x07E4+grlc_data->next_poll_array[grlc_data->poll_start_tbf].ctrl_ack; + grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].block_status = 8; + grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].ul_block[0] = grlc_convert_11bit_2_etsi(elevenBit); + /* + TRACE_EVENT_P2("PCA AB_11_BIT sent fn_i= %ld PRACH11: %d" + ,grlc_data->next_poll_fn + ,grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].ul_block[0]); + */ + } + else + { + /*8 bit access burst*/ + grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].block_status = 7; + grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].ul_block[0] = (0x7C+grlc_data->next_poll_array[grlc_data->poll_start_tbf].ctrl_ack); + /* + TRACE_EVENT_P1 ("PCA 8 bit access burst sent fn_i= %ld ",grlc_data->next_poll_fn); + */ + } + +#ifdef _SIMULATION_ + { + PALLOC(mac_poll_req,MAC_POLL_REQ); + memset(mac_poll_req,0, + sizeof(T_MAC_POLL_REQ)); + memcpy(&(mac_poll_req->ul_poll_resp), + &(grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index]), + sizeof(T_ul_poll_resp)); + PSEND(hCommL1,mac_poll_req); + } +#else /* #ifdef _SIMULATION_ */ + { + TRACE_MEMORY_PRIM ( hCommGRLC, hCommL1, MAC_POLL_REQ, + &grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index], + sizeof(T_ul_poll_resp) ); + } +#endif /* #ifdef _SIMULATION_ */ + + TRACE_BINDUMP + ( hCommGRLC, TC_USER4, + cl_rlcmac_get_msg_name( U_MSG_TYPE_CHANNEL_REQ_c, RLC_MAC_ROUTE_UL ), + grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].ul_block, + RLC_MAC_MAX_LEN_CHANNEL_REQ ); /*lint !e569*/ + + grlc_data->ul_poll_pos_index++; + /* + * next poll block invalid + */ + grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].block_status = 0; + + +} /* grlc_send_access_burst() */ + +/* ++------------------------------------------------------------------------------ +| Function : grlc_send_normal_burst ++------------------------------------------------------------------------------ +| Description : The function grlc_send_normal_burst() sents the poll as +| normal burst type +| +| Parameters : ptr_block_i : ptr to the ctrl message +| Parameters : tn_i : timeslot where the access burst shall send +| Parameters : r_bit_i : r bit needed for the mac header +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_send_normal_burst + (UBYTE * struct_data, UBYTE * encoded_data, UBYTE tn_i) +{ + + TRACE_FUNCTION( "grlc_send_normal_burst" ); + + if( (grlc_data->poll_start_tbf NEQ 0xFF) AND + (grlc_data->ta_value EQ 0xFF) ) + { + TRACE_EVENT_P2("No TA VALUE IN GRLC --> NB POLL NOT SENT ta=%d poll_st_tbf=%d" + ,grlc_data->ta_value + ,grlc_data->poll_start_tbf); + + TRACE_ERROR("No TA VALUE IN GRLC --> NB POLL NOT SENT"); + + grlc_data->next_poll_array[grlc_data->poll_start_tbf].cnt--; + grlc_data->next_poll_array[grlc_data->poll_start_tbf].poll_type[tn_i] = CGRLC_POLL_NONE; + return; + } + + /* + * either encoded data or structured data are passed by reference + */ + + + if( struct_data NEQ NULL ) + { + grlc_encode_ul_ctrl_block( ( UBYTE* )grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].ul_block, struct_data ); + } + else if( encoded_data NEQ NULL ) + { + memcpy( ( UBYTE* )grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].ul_block, + encoded_data, + BYTE_UL_CTRL_BLOCK ); + } + else + { + TRACE_ERROR( "grlc_send_normal_burst: no data available" ); + } + + grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].tn = tn_i; + grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].block_status = 3; + + if(grlc_data->poll_start_tbf NEQ 0xFF) + { + grlc_data->next_poll_array[grlc_data->poll_start_tbf].cnt--; + grlc_data->next_poll_array[grlc_data->poll_start_tbf].poll_type[tn_i] = CGRLC_POLL_NONE; + grlc_data->next_poll_fn = grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn; + } + +#ifdef _SIMULATION_ + { + PALLOC(mac_poll_req,MAC_POLL_REQ); + memset(mac_poll_req,0, + sizeof(T_MAC_POLL_REQ)); + memcpy(&(mac_poll_req->ul_poll_resp), + &(grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index]), + sizeof(T_ul_poll_resp)); + PSEND(hCommL1,mac_poll_req); + } +#else /* #ifdef _SIMULATION_ */ + { + TRACE_MEMORY_PRIM ( hCommGRLC, hCommL1, MAC_POLL_REQ, + &grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index], + sizeof(T_ul_poll_resp) ); + } +#endif /* #ifdef _SIMULATION_ */ + + { + UBYTE* ul_block = ( UBYTE* )grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].ul_block; + + TRACE_BINDUMP( hCommGRLC, TC_USER4, + cl_rlcmac_get_msg_name + ( ( UBYTE )( ul_block[1] >> 2 ), RLC_MAC_ROUTE_UL ), + ul_block, MAX_L2_FRAME_SIZE ); /*lint !e569*/ + } + + grlc_data->ul_poll_pos_index++; + /* + * next poll block invalid + */ + grlc_data->ul_poll_resp[grlc_data->ul_poll_pos_index].block_status = 0; + + /* + TRACE_EVENT_P1 ("Normal burst sent at fn_i= %ld ",grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn); + */ +} /* grlc_send_normal_burst() */ + +/* ++------------------------------------------------------------------------------ +| Function : grlc_del_sent_poll ++------------------------------------------------------------------------------ +| Description : The function grlc_del_sent_poll() deletes the poll pos which was +| sent to L1 +| +| Parameters + +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_del_sent_poll(void ) +{ + UBYTE help_index; + UBYTE next_index; + + TRACE_FUNCTION( "grlc_del_sent_poll" ); + + + if( grlc_data->next_poll_array[grlc_data->poll_start_tbf].cnt EQ 0 ) + { + help_index = grlc_data->poll_start_tbf; + grlc_data->poll_start_tbf = grlc_data->next_poll_array[grlc_data->poll_start_tbf].next; + next_index = grlc_data->poll_start_free; + while(grlc_data->next_poll_array[next_index].next NEQ 0xFF) + { + next_index = grlc_data->next_poll_array[next_index].next; + } + grlc_data->next_poll_array[next_index].next = help_index; + grlc_data->next_poll_array[help_index].next = 0xFF; + } + + + +} /* grlc_del_sent_poll() */ + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_decode_tbf_start_rel +|------------------------------------------------------------------------------ +| Description : The function grlc_decode_tbf_start_rel() translates the TBF- +| Starting-Time-Relative into full frame number. Therefore the +| received frame number is needed in start_fn !! +| +| Parameters : rel_pos - number in blocks added to current framenuber +| ++------------------------------------------------------------------------------ +*/ +GLOBAL ULONG grlc_decode_tbf_start_rel(ULONG start_fn, USHORT rel_pos) +{ + ULONG result; + + TRACE_FUNCTION( "grlc_decode_tbf_start_rel" ); + + result = 4+4*rel_pos + start_fn + rel_pos/3; + + if ((12 EQ (result%13)) OR + (7 EQ (result%13)) OR + (3 EQ (result%13))) + { + result += 1; + } + if(FN_MAX <= result) + { + result %= FN_MAX; + } + + return result; +} /* grlc_decode_tbf_start_rel */ + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_get_sdu_len_and_used_ts ++------------------------------------------------------------------------------ +| Description : +| +| Parameters : * sdu_len - len of the current sdu in progress +| * used_ul_resources - used uplink resources in fixed alloc mode +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_get_sdu_len_and_used_ts ( T_RLC_VALUES * values ) +{ + TRACE_FUNCTION( "grlc_get_sdu_len_and_used_ts" ); + + switch( GET_STATE( RU ) ) + { + case RU_NULL: + values->sdu_len = grlc_data->prim_queue[grlc_data->prim_start_tbf].prim_ptr->sdu.l_buf/8; + values->cnt_ts = 0; + break; + default: + values->sdu_len = grlc_data->ru.sdu_len; + values->cnt_ts = grlc_data->ru.cnt_ts; + break; + } + /* grlc_get_sdu_len_and_used_ts() */ +} + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_check_dist ++------------------------------------------------------------------------------ +| Description : The function grlc_check_dist() checks if high_i is bigger/equal +| than low_i(modulo calculation). +| The return value is true, if high_i is equal to low_i or +| bigger than low_i. +| Parameters : high_i - expected high value +| low_i - expected low value +| dist_i - max. allowed distance between high_i and low_i ++------------------------------------------------------------------------------ +*/ +GLOBAL BOOL grlc_check_dist ( ULONG high_i, ULONG low_i, ULONG dist_i) +{ + BOOL result = FALSE; + ULONG real_dist; + + TRACE_FUNCTION( "grlc_check_dist" ); + + if (high_i >= low_i) + real_dist = high_i - low_i; + else + real_dist = high_i + (FN_MAX-low_i); + + if (real_dist <= dist_i ) + { + result = TRUE; + } + return result; +} /* grlc_check_dist() */ + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_handle_poll_pos ++------------------------------------------------------------------------------ +| Description : The function grlc_handle_poll_pos() +| +| Parameters : - ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_handle_poll_pos (ULONG current_fn) +{ + UBYTE help_index; + UBYTE next_index; + TRACE_FUNCTION( "grlc_handle_poll_pos" ); + + while ( (grlc_data->poll_start_tbf NEQ 0xFF) AND + (grlc_check_dist(current_fn,grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn,26)) AND + (current_fn NEQ grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn)) + { + TRACE_EVENT_P4("current_fn= %ld missed poll_fn=%ld ps_tbf=%d cnt=%d", + current_fn, + grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn, + grlc_data->poll_start_tbf, + grlc_data->next_poll_array[grlc_data->poll_start_tbf].cnt); + /* + * delete elememnt from tbf_list if all pollType = POLL_NONE, and add to free list + */ + grlc_data->missed_poll_fn = grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn; + grlc_data->next_poll_array[grlc_data->poll_start_tbf].cnt = 0; + help_index = grlc_data->poll_start_tbf; + grlc_data->poll_start_tbf = grlc_data->next_poll_array[grlc_data->poll_start_tbf].next; + next_index = grlc_data->poll_start_free; + while(grlc_data->next_poll_array[next_index].next NEQ 0xFF) + { + next_index = grlc_data->next_poll_array[next_index].next; + } + grlc_data->next_poll_array[next_index].next = help_index; + grlc_data->next_poll_array[help_index ].next = 0xFF; + TRACE_ERROR( "Poll Position missed" ); + } +} /* grlc_handle_poll_pos() */ + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_send_rem_poll_pos ++------------------------------------------------------------------------------ +| Description : The function grlc_send_rem_poll_pos() +| +| Parameters : - ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_send_rem_poll_pos (ULONG current_fn) +{ + ULONG delta_fn = 0; + UBYTE *ptr_block=NULL; + + + TRACE_FUNCTION( "grlc_send_rem_poll_pos" ); + + if (grlc_data->poll_start_tbf EQ 0xFF) + return; + + + +#ifdef _TARGET_ + delta_fn = grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn; + if(current_fn EQ FN_MAX-5) + delta_fn += 5; + else + delta_fn -= current_fn; +#endif + +#ifdef _SIMULATION_ + if(current_fn EQ grlc_data->next_poll_array[grlc_data->poll_start_tbf].fn) + delta_fn = 4; + else + delta_fn = 0; +#endif + + + if( (delta_fn EQ 4) OR + (delta_fn EQ 5) ) + { + UBYTE tn=0; + UBYTE index; + while(grlc_data->next_poll_array[grlc_data->poll_start_tbf].cnt + AND (tn < 8)) + { + switch(grlc_data->next_poll_array[grlc_data->poll_start_tbf].poll_type[tn]) + { + case CGRLC_POLL_CTRL: + case CGRLC_POLL_UACK: + case CGRLC_POLL_COLLISION: + if(grlc_data->burst_type EQ CGRLC_BURST_TYPE_NB) + { + ptr_block = grlc_set_packet_ctrl_ack(); + grlc_send_normal_burst(ptr_block, NULL, tn); + } + else + grlc_send_access_burst(tn); + break; + case CGRLC_POLL_RES_AB: + grlc_send_access_burst(tn); + break; + case CGRLC_POLL_RES_NB: + ptr_block = grlc_set_packet_ctrl_ack(); + grlc_send_normal_burst(ptr_block, NULL, tn); + break; + case CGRLC_POLL_DATA: + if( grlc_data->tbf_type EQ TBF_TYPE_UL ) + { + /* + * no downlink active , send ctrl block or pca + */ + if( tm_get_num_ctrl_blck( ) NEQ 0 ) + { + ptr_block = tm_get_ctrl_blk( &index, TRUE ); + grlc_send_normal_burst(NULL, ptr_block, tn); + grlc_data->rd.next_poll_block = NEXT_POLL_BLOCK_DL_DATA; + } + else if(grlc_data->burst_type EQ CGRLC_BURST_TYPE_NB) + { + ptr_block = grlc_set_packet_ctrl_ack(); + grlc_send_normal_burst(ptr_block, NULL, tn); + } + else + grlc_send_access_burst(tn); + } + break; +#if defined REL99 AND defined TI_PS_FF_TBF_EST_PACCH + case CGRLC_POLL_RE_ASS: /* TBF re assignment on PACCH */ + + if(grlc_data->burst_type EQ CGRLC_BURST_TYPE_NB) + { + T_U_GRLC_RESOURCE_REQ resource_req; + + tm_build_res_req( &resource_req,R_BUILD_2PHASE_ACCESS); + grlc_send_normal_burst((UBYTE *)&resource_req, NULL, tn); + } + else /* PCA CTRL ACK 00 */ + { + grlc_data->next_poll_array[grlc_data->poll_start_tbf].ctrl_ack = 0; + grlc_send_access_burst(tn); + } + + break; +#endif + } + tn++; + } + grlc_del_sent_poll(); + } +}/* grlc_send_rem_poll_pos() */ + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_test_mode_active ++------------------------------------------------------------------------------ +| Description : This functions returns 0 if the GPRS test mode is not activated. +| Otherwise a value greater then 0. +| +| Parameters : no parameters +| ++------------------------------------------------------------------------------ +*/ + +GLOBAL UBYTE grlc_test_mode_active () +{ + if(grlc_data->testmode.mode NEQ CGRLC_NO_TEST_MODE) + return TRUE; + else + return FALSE; + +}/* grlc_test_mode_active*/ + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_prbs ++------------------------------------------------------------------------------ +| Description : +| This function generates a pseodo random bis sequence. +| The implementation is related to CCITT O.151 Okt. 92 chapter 2.1 . +| +| This functions generates the next length_i bytes of a 32767-bit +| pseudo-random test sequence if action_i is set to COMPUTE_DATA +| and writes the data at address ptr_i. +| +| The function stores the position where it stops to calculate. +| In case of action_i EQ INITIALIZE the function erase his history. +| On its next call the function continues with its intial values +| +| Parameters : UBYTE action_i (INITIALIZE or COMPUTE_DATA) +| and only +| UBYTE length_i (number of data bytes which shall callculated and copied) +| UBYTE * out_i (location where this dada have to be placed, only valid) +| ++------------------------------------------------------------------------------ +*/ + +GLOBAL void grlc_prbs(UBYTE action_i, UBYTE length_i, UBYTE * ptr_i) +{ + TRACE_FUNCTION( "grlc_prbs" ); + + switch(action_i) + { + case INITIALIZE: + +#define FEED_BACK_MASK 0x6000 + /* + * 14th and 15 stage will be feeded back + */ + + + /* + * Initialize the prbs generation + * + * This value is the value of the shift register 8 cycles + * before shift register value 0x7fff + */ + grlc_data->testmode.prbs_shift_reg = 0x55ff; + break; + case COMPUTE_DATA: + { + /* + * get prbs computing values + */ + USHORT i,reg = grlc_data->testmode.prbs_shift_reg; + + UBYTE * out_ptr = ptr_i; + + for(i=0; i < (8 * length_i); i++) + { + /* + * write a byte in output if neccessary + */ + if(i % 8 EQ 0) + { + *out_ptr++ = (UBYTE)(0x00ff®); + } + + + { + USHORT temp = reg & FEED_BACK_MASK; + /* + * shift the register and but new data in + */ + reg = reg << 1; + if ( (temp EQ FEED_BACK_MASK) OR (temp EQ 0) ) + { + /* + * put a "0" in + */; + } + else + { + /* + * put a "1" in + */ + reg +=1; + } + } /* for(i=0; i < (8 * length_i); i++) */ + } + /* + * store prbs computing values + */ + grlc_data->testmode.prbs_shift_reg = reg; + } + break; + + default: + break; + } + + return; + +}/*grlc_prbs*/ + + + + + + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_trace_tbf_par ++------------------------------------------------------------------------------ +| Description : +| +| Parameters : +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_trace_tbf_par ( UBYTE tbf_index ) +{ + TRACE_FUNCTION( "grlc_trace_tbf_par" ); + + TRACE_EVENT_P9("tbf_type=%d;start_fn=%ld (%ld);end_fn=%ld(%ld);rlc_oct_cnt=%d; pdu_cnt=%ld;vs_vr=%d;va_vq=%d ", + grlc_data->tbf_ctrl[tbf_index].tbf_type, + grlc_data->tbf_ctrl[tbf_index].start_fn, + grlc_data->tbf_ctrl[tbf_index].start_fn%42432, + grlc_data->tbf_ctrl[tbf_index].end_fn, + grlc_data->tbf_ctrl[tbf_index].end_fn%42432, + grlc_data->tbf_ctrl[tbf_index].rlc_oct_cnt, + grlc_data->tbf_ctrl[tbf_index].pdu_cnt, + grlc_data->tbf_ctrl[tbf_index].vs_vr, + grlc_data->tbf_ctrl[tbf_index].va_vq); + + TRACE_EVENT_P9("cnt_ts=%d;ack_cnt=%d;fbi=%d;ret_bsn=%d;N_ACC=%d,tlli=%lx rem_ul_data=%d,PST=%d,PSF=%d", + grlc_data->tbf_ctrl[tbf_index].cnt_ts, + grlc_data->tbf_ctrl[tbf_index].ack_cnt, + grlc_data->tbf_ctrl[tbf_index].fbi, + grlc_data->tbf_ctrl[tbf_index].ret_bsn, + grlc_data->tm.n_acc_req_procedures, + grlc_data->uplink_tbf.tlli, + grlc_data->grlc_data_req_cnt, + grlc_data->prim_start_tbf, + grlc_data->prim_start_free); + +} /* grlc_trace_tbf_par */ + + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_set_buf_tlli ++------------------------------------------------------------------------------ +| Description : The function grlc_set_buf_tlli() fills the TLLI buffer. +| +| Parameters : +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_set_buf_tlli ( BUF_tlli_value *buf_tlli_o, ULONG tlli_i ) +{ + TRACE_FUNCTION( "grlc_set_buf_tlli" ); + + grlc_set_tlli( &buf_tlli_o->l_tlli_value, + &buf_tlli_o->o_tlli_value, + &buf_tlli_o->b_tlli_value[0], + tlli_i ); + +} /* grlc_set_buf_tlli */ + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_set_tlli ++------------------------------------------------------------------------------ +| Description : The function grlc_set_tlli() fills the TLLI buffer. +| +| Parameters : +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_set_tlli + ( USHORT *l_tlli, USHORT *o_tlli, UBYTE *b_tlli, ULONG tlli ) +{ + TRACE_FUNCTION( "grlc_set_tlli" ); + + *l_tlli = 32; + *o_tlli = 0; + + b_tlli[0] = (UBYTE)((tlli >> 24) & 0x000000ff); + b_tlli[1] = (UBYTE)((tlli >> 16) & 0x000000ff); + b_tlli[2] = (UBYTE)((tlli >> 8 ) & 0x000000ff); + b_tlli[3] = (UBYTE)((tlli ) & 0x000000ff); + + /* unused byte must be set to 0x00, otherwise CCD has some problems */ + b_tlli[4] = 0; + +} /* grlc_set_tlli */ + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_encode_ul_ctrl_block ++------------------------------------------------------------------------------ +| Description : +| +| Parameters : +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_encode_ul_ctrl_block ( UBYTE *ul_ctrl_block, + UBYTE *ul_ctrl_data ) +{ + T_CTRLBUF enc_block; + + TRACE_FUNCTION( "grlc_encode_ul_ctrl_block" ); + + grlc_encode_ctrl( ul_ctrl_data, ( T_MSGBUF* )&enc_block , grlc_data->r_bit ); + memcpy( ul_ctrl_block, enc_block.buf, BYTELEN( enc_block.l_buf ) ); + +} /* grlc_encode_ul_ctrl_block */ + + + + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_check_if_tbf_start_is_elapsed ++------------------------------------------------------------------------------ +| Description : The function grlc_check_if_tbf_start_is_elapsed() checks if +| tbf starting time is elapsed or not, modulo calculation is +| needed +| Parameters : start_fn - tbf starting time +| current_fn - current frame number ++------------------------------------------------------------------------------ +*/ +GLOBAL BOOL grlc_check_if_tbf_start_is_elapsed ( ULONG start_fn, ULONG current_fn) +{ + BOOL result = FALSE; + ULONG d1; + /* ULONG d2; */ + /* FN_MAX=0x297000 == 2715648 ==125463 seconds(4.62ms per frame) + * the starting time is within current_fn-10808 and current_fn+31623 + * modulo operation must be taken in account + */ + TRACE_FUNCTION( "grlc_check_if_tbf_start_is_elapsed" ); + + /* + * handle maximum distance for tbf starting time + */ + + if(start_fn EQ CGRLC_STARTING_TIME_NOT_PRESENT) /*lint !e650*/ + { + result =TRUE; + return result; + } + d1 = 10808; + /* d2 = 31623; */ + if( (start_fn <= current_fn) AND + ((current_fn-start_fn) <= d1)) + { + result = TRUE; + /*TRACE_EVENT_P2(" case 1: st time elapsed st_fn=%ld c_fn=%ld",start_fn,current_fn);*/ + } + else if((start_fn >= current_fn) AND + (FN_MAX-start_fn+current_fn) <= d1) + { + result = TRUE; + /* TRACE_EVENT_P2("case 2: st time elapsed st_fn=%ld c_fn=%ld",start_fn,current_fn);*/ + } +/* else + { + TRACE_EVENT_P2("case 3: WAIT FOR ST TIME st_fn=%ld c_fn=%ld",start_fn,current_fn); + } +*/ + + return result; +} /* grlc_check_if_tbf_start_is_elapsed() */ + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_decode_grlc ++------------------------------------------------------------------------------ +| Description : The function grlc_decode_grlc() calls the function ccd_decodeMsg. +| After the call the decoded Message is in _decodeCtrlMsg. +| +| Parameters : msg_ptr_i - pointer to buffer that should be decoded +| ++------------------------------------------------------------------------------ +*/ +GLOBAL UBYTE grlc_decode_grlc (T_MSGBUF * msg_ptr_i) +{ + UBYTE result; + UBYTE msg_type = msg_ptr_i->buf[0] >> 2;; + + TRACE_FUNCTION( "grlc_decode_grlc" ); + + /* + * Offset must be zero, else code to get msg_type is illegal + */ + TRACE_ASSERT (msg_ptr_i->o_buf==0); + + + + switch (msg_type) + { + case D_UL_ACK: + if(!grlc_data->grlc_wo_ccd) + { + result = ccd_decodeMsg (CCDENT_GRLC, DOWNLINK, msg_ptr_i, _decodedMsg, NOT_PRESENT_8BIT); + } + else + { + result = grlc_decode_ul_acknack(msg_ptr_i->buf); + } + break; + default: + TRACE_ERROR(" not Packet ul ack decoded: should nor happen "); + result = DELETE_MESSAGE; + break; + } + + + if ( result EQ ccdError ) + { + return grlc_ccd_error_handling( CCDENT_GRLC ); + } + + return msg_type; + +} /* grlc_decode_grlc() */ + + + +/* ++------------------------------------------------------------------------------ +| Function : grlc_activate_tfi ++------------------------------------------------------------------------------ +| Description : +| grlc_activate_tfi modifies tfi´s if tbf starting time is reached +| +| +| Parameters : fn_i is the current framenuber in tbf mode +| ++------------------------------------------------------------------------------ +*/ + +GLOBAL void grlc_activate_tfi (ULONG fn_i) +{ + + TRACE_FUNCTION( "grlc_activate_tfi" ); + + + /* 1. check if starting time is reached */ + switch(grlc_data->tfi_change) + { + case TFI_CHANGE_UL: + if(grlc_check_if_tbf_start_is_elapsed ( grlc_data->ul_tbf_start_time, ((fn_i+5)%FN_MAX))) + { + grlc_data->tfi_change = TFI_CHANGE_NULL; + grlc_data->ul_tfi = grlc_data->start_fn_ul_tfi; + grlc_data->start_fn_ul_tfi = 0xFF; + TRACE_EVENT_P2("UL TFI CHANGE St reached st_fn=%ld c_fn=%ld",grlc_data->ul_tbf_start_time,fn_i ); + grlc_data->ul_tbf_start_time = CGRLC_STARTING_TIME_NOT_PRESENT; + grlc_data->tbf_ctrl[grlc_data->ul_index].tfi = grlc_data->ul_tfi; + grlc_data->ul_tfi_changed = TRUE; + } + break; + case TFI_CHANGE_DL: + if(grlc_check_if_tbf_start_is_elapsed ( grlc_data->dl_tbf_start_time, fn_i)) + { + grlc_data->tfi_change = TFI_CHANGE_NULL; + grlc_data->dl_tfi = grlc_data->start_fn_dl_tfi; + grlc_data->start_fn_dl_tfi = 0xFF; + TRACE_EVENT_P2("DL TFI CHANGE St reached st_fn=%ld c_fn=%ld",grlc_data->dl_tbf_start_time,fn_i ); + grlc_data->dl_tbf_start_time = CGRLC_STARTING_TIME_NOT_PRESENT; + grlc_data->tbf_ctrl[grlc_data->dl_index].tfi = grlc_data->dl_tfi; + } + break; + case TFI_CHANGE_ALL: + if(grlc_check_if_tbf_start_is_elapsed ( grlc_data->ul_tbf_start_time, ((fn_i+5)%FN_MAX))) + { + grlc_data->tfi_change = TFI_CHANGE_DL; + grlc_data->ul_tfi = grlc_data->start_fn_ul_tfi; + grlc_data->start_fn_ul_tfi = 0xFF; + TRACE_EVENT_P2("UL TFI CHANGE(ALL) St reached st_fn=%ld c_fn=%ld",grlc_data->ul_tbf_start_time,fn_i ); + grlc_data->ul_tbf_start_time = CGRLC_STARTING_TIME_NOT_PRESENT; + grlc_data->tbf_ctrl[grlc_data->ul_index].tfi = grlc_data->ul_tfi; + grlc_data->tbf_ctrl[grlc_data->dl_index].tfi = grlc_data->dl_tfi; + grlc_data->ul_tfi_changed = TRUE; + } + if(grlc_check_if_tbf_start_is_elapsed ( grlc_data->dl_tbf_start_time, fn_i)) + { + if (grlc_data->tfi_change EQ TFI_CHANGE_DL) + grlc_data->tfi_change = TFI_CHANGE_NULL; + else + grlc_data->tfi_change = TFI_CHANGE_UL; + grlc_data->dl_tfi = grlc_data->start_fn_dl_tfi; + grlc_data->start_fn_dl_tfi = 0xFF; + TRACE_EVENT_P2("DL TFI CHANGE(ALL) St reached st_fn=%ld c_fn=%ld",grlc_data->dl_tbf_start_time,fn_i ); + grlc_data->dl_tbf_start_time = CGRLC_STARTING_TIME_NOT_PRESENT; + } + break; + } +} + +/* ++------------------------------------------------------------------------------ +| Function : grlc_t_status ++------------------------------------------------------------------------------ +| Description : This function returns the remaining time in milliseconds. +| A value of 0L is returned in case the timer is not existing or +| not running. +| +| Parameters : t_index : timer index +| ++------------------------------------------------------------------------------ +*/ +GLOBAL T_TIME grlc_t_status( USHORT t_index ) +{ + T_TIME t_time = 0L; + + TRACE_FUNCTION( "grlc_t_status" ); + + vsi_t_status( GRLC_handle, t_index, &t_time ); + + return( t_time ); +} /* grlc_t_status */ + +/* ++------------------------------------------------------------------------------ +| Function : grlc_enter_standby_state ++------------------------------------------------------------------------------ +| Description : This function is called in case the STANDBY state should be +| entered. +| +| Parameters : +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_enter_standby_state ( void ) +{ + TRACE_FUNCTION( "grlc_enter_standby_state" ); + + if( grlc_data->ready_timer.handling EQ READY_TIMER_HANDLING_ENABLED ) + { + if( grlc_data->ready_timer.state EQ READY_STATE ) + { + /* The ready_timer.state is set to STANDBY_STATE just in case we are not already in this state. + The CGRLC_STANDBY_STATE_IND primitives are sent if we move from READY_STATE to STANDBY_STATE only. + */ + if( grlc_data->ready_timer.value NEQ CGRLC_DEACTIVATED ) + { + /* If the timer T3314 expires while we are in READY_STATE but the timer is deactivated then + the transition to STANDBY_STATE will never occur + */ + /* If the primitive CGRLC_FORCE_TO_STANDBY_REQ is received when we are in Ready State but + the timer is deactivated then the transition to STANDBY_STATE will never occur + */ + grlc_data->ready_timer.state = STANDBY_STATE; + vsi_t_stop( GRLC_handle, T3314 ); + + { + PALLOC(cgrlc_standby_state_ind,CGRLC_STANDBY_STATE_IND); /* T_CGRLC_STANDBY_STATE_IND sent to GMM */ + PSEND(hCommGMM,cgrlc_standby_state_ind); + } + + { + PALLOC(cgrlc_standby_state_ind,CGRLC_STANDBY_STATE_IND); /* T_CGRLC_STANDBY_STATE_IND sent to GRR */ + PSEND(hCommGRR,cgrlc_standby_state_ind); + } + } + else + { + TRACE_EVENT( "grlc_enter_standby_state: MS enters STANDBY state while T3314 is deactivated" ); + } + } + else + { + TRACE_EVENT( "grlc_enter_standby_state: MS is already in STANDBY state" ); + } + } +} /* grlc_enter_standby_state */ + +/* ++------------------------------------------------------------------------------ +| Function : grlc_enter_ready_state ++------------------------------------------------------------------------------ +| Description : This function is called in case the READY state should be +| entered. +| +| Parameters : +| ++------------------------------------------------------------------------------ +*/ +GLOBAL void grlc_enter_ready_state ( void ) +{ + TRACE_FUNCTION( "grlc_enter_ready_state" ); + + if (grlc_data->ready_timer.state EQ STANDBY_STATE) + { + /* The ready_timer.state is set to READY_STATE just in case we are not already in this state. + The CGRLC_READY_STATE_IND primitives are sent only if we move from STANDBY_STATE to READY_STATE */ + + /* If we receive the primitive CGRLC_READY_TIMER_CONFIG_REQ with timer value CGRLC_DEACTIVATED when + we are in CGRLC_STANDBY, the state will immediately switch to READY and the CGRLC_READY_STATE_IND + primitives will be sent */ + grlc_data->ready_timer.state = READY_STATE; + + { + PALLOC(cgrlc_ready_state_ind,CGRLC_READY_STATE_IND); /* T_CGRLC_READY_STATE_IND sent to GMM */ + PSEND(hCommGMM,cgrlc_ready_state_ind); + } + + { + PALLOC(cgrlc_ready_state_ind,CGRLC_READY_STATE_IND); /* T_CGRLC_READY_STATE_IND sent to GRR */ + PSEND(hCommGRR,cgrlc_ready_state_ind); + } + } + else + { + TRACE_EVENT( "grlc_enter_ready_state: MS is already in READY state" ); + } +} /* grlc_enter_ready_state */ +/* ++------------------------------------------------------------------------------ +| Function : grlc_convert_11bit_2_etsi ++------------------------------------------------------------------------------ +| Description : Converts the 11 bit access burst value into ETSI format +| +| Parameters : In: eleven bit value +| Out: converted eleven bit +| ++------------------------------------------------------------------------------ +*/ +LOCAL USHORT grlc_convert_11bit_2_etsi ( USHORT eleven_bit ) +{ + USHORT etsi11bit; + USHORT dummy1 = 0, dummy2 = 0; + + TRACE_FUNCTION( "grlc_convert_11bit_2_etsi" ); + + /* + * 11 Bit access burst + * b: bit + * b10 b9 b8 b7 b6 b5 b4 b3 b2 b1 b0 + * should be sent to the network -according 04.60 and 0404- in the + * following 16-bit format: + * 0 0 0 0 0 b2 b1 b0 b10 b9 b8 b7 b6 b5 b4 b3 + */ + + /* + * get b2 b1 b0 + */ + dummy1 = 0x0007 & eleven_bit; + + /* + * shift it 8 bits to left + */ + dummy1 = ( dummy1 << 8 ); + + /* + * get b10 b9 b8 b7 b6 b5 b4 b3 + */ + dummy2 = 0xFFF8 & eleven_bit; + + /* + * shift it 3 bits to right + */ + dummy2 = ( dummy2 >> 3 ); + + /* + * compose dummy1 and dummy2 to the target 16-bit format + */ + etsi11bit = dummy1 | dummy2; + + return etsi11bit; + +} /* grlc_convert_11bit_2_etsi() */