FreeCalypso > hg > fc-magnetite
diff src/g23m-gsm/dl/dl_com.c @ 104:27a4235405c6
src/g23m-gsm: import from LoCosto source
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Tue, 04 Oct 2016 18:24:05 +0000 |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/g23m-gsm/dl/dl_com.c Tue Oct 04 18:24:05 2016 +0000 @@ -0,0 +1,1327 @@ +/* ++----------------------------------------------------------------------------- +| Project : +| Modul : ++----------------------------------------------------------------------------- +| Copyright 2002 Texas Instruments Berlin, AG +| All rights reserved. +| +| This file is confidential and a trade secret of Texas +| Instruments Berlin, AG +| The receipt of or possession of this file does not convey +| any rights to reproduce or disclose its contents or to +| manufacture, use, or sell anything it may describe, in +| whole, or in part, without the specific written consent of +| Texas Instruments Berlin, AG. ++----------------------------------------------------------------------------- +| Purpose : This Modul defines the common functions +| for the component DL of the mobile station. ++----------------------------------------------------------------------------- +*/ + +#ifndef DL_COM_C +#define DL_COM_C + +#define ENTITY_DL + +/*==== INCLUDES ===================================================*/ +#include "typedefs.h" +#include <string.h> +#include "vsi.h" +#if !defined(DL_2TO1) +//#include "p_8010_147_l1_include.h" +#endif /* DL_2TO1 */ +#include "pconst.cdg" +#include "custom.h" +#include "gsm.h" +#include "mon_dl.h" +#include "prim.h" +#include "pei.h" +#include "tok.h" +#include "ccdapi.h" +#include "dl.h" +#include "dl_trc.h" +#include "dl_em.h" + +/*==== TEST TRACE ===================================================*/ +#define TEST_ENTITY_DL + +/*==== EXPORT =====================================================*/ + +/*==== PRIVAT =====================================================*/ +LOCAL void com_build_frame ( + UBYTE ch_type, + UBYTE type, + UBYTE sapi, + UBYTE cr, + UBYTE ns, + UBYTE nr, + UBYTE p_bit, + UBYTE length, + UBYTE m_bit, + UBYTE * pInfoBuffer, + UBYTE InfoOffset); + +/*==== VARIABLES ==================================================*/ + +/*==== FUNCTIONS ==================================================*/ + +/* ++------------------------------------------------------------------------------ +| Function : com_free_pointer ++------------------------------------------------------------------------------ +| Description : frees the pointer given by parameter ++------------------------------------------------------------------------------ +*/ +GLOBAL void com_free_pointer (void * pointer) +{ + GET_INSTANCE_DATA; + TRACE_EVENT_WIN_P1 ("com_free_pointer(,p=%08x)", pointer); + +#if defined(INVOKE_SIGNAL) + if (dl_data->interrupt_context) + { + sig_invoke_com_free_pointer (pointer); + return; + } +#endif /* INVOKE_SIGNAL */ + +TRACE_ASSERT(pointer); + MY_PFREE (pointer); +} + +GLOBAL void com_free_queue_buffer (T_QUEUE * queue, USHORT index) +{ + T_DL_DATA_REQ **pp; + + switch (index) + { + default: + if (index <= INDEX_MAX_STORE_BUFFER) + pp = &queue->store_buffer[index]; + else + pp = NULL; + break; + case INDEX_SENDING_BUFFER: + pp = &queue->sending_buffer; + break; + case INDEX_SWITCH_BUFFER: + pp = &queue->switch_buffer; + break; + } + if (pp AND *pp) + { + COM_FREE_POINTER (*pp); + *pp = NULL; + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_clear_queue | ++--------------------------------------------------------------------+ + + PURPOSE : Clearing a DL queue. + +*/ + +GLOBAL void com_clear_queue (UBYTE sapi) +{ + GET_INSTANCE_DATA; + USHORT i; + T_QUEUE *queue; + TRACE_FUNCTION ("com_clear_queue()"); + + if (sapi EQ PS_SAPI_0) + { + queue = &dl_data->dcch0_queue; + } + else + { + queue = &dl_data->dcch3_queue; + } + + for (i=0;i<MAX_QUEUED_MESSAGES;i++) + if (queue->store_buffer [i] NEQ NULL) + { + COM_FREE_QUEUE_BUFFER (queue, i); + } + + if (queue->sending_buffer NEQ NULL) + { + COM_FREE_QUEUE_BUFFER (queue, INDEX_SENDING_BUFFER); + } + + if (queue->switch_buffer NEQ NULL) + { + COM_FREE_QUEUE_BUFFER (queue, INDEX_SWITCH_BUFFER); + } + queue->act_length = queue->act_offset = 0; + queue->no_of_stored_messages = 0; + memset (&queue->transmit_buffer, 0, sizeof (T_FRAME)); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_restore_queue | ++--------------------------------------------------------------------+ + + PURPOSE : Restoring a DL queue. If a new connection shall be + established, the message (ASSIGNMENT or HANDOVER COMPLETE) + is transmitted first. So the message is stored in the + switch buffer. A previously not complete send message will + be put back into the storing buffer. + +*/ + +GLOBAL void com_restore_queue (UBYTE sapi, T_DL_DATA_REQ* est_req) +{ + GET_INSTANCE_DATA; + T_QUEUE *queue = sapi EQ PS_SAPI_0 ? &dl_data->dcch0_queue:&dl_data->dcch3_queue; + + TRACE_FUNCTION ("com_restore_queue()"); + + if (queue->switch_buffer NEQ NULL) + { + TRACE_EVENT_WIN ("free old switch_buffer"); + COM_FREE_QUEUE_BUFFER (queue, INDEX_SWITCH_BUFFER); + } + + if (est_req AND est_req->sdu.l_buf) + { + PPASS (est_req, data_req, DL_DATA_REQ); + queue->switch_buffer = data_req; /* only valid sdu */ + + TRACE_EVENT_WIN_P2 ("new fill of switch_buffer:%p l=%u", + data_req, est_req->sdu.l_buf>>3); + } + + if (queue->sending_buffer NEQ NULL) + { + TRACE_EVENT_WIN ("restore sending_buffer"); + queue->act_length = queue->sending_buffer->sdu.l_buf; + queue->act_offset = queue->sending_buffer->sdu.o_buf; + } + TRACE_EVENT_WIN ("delete transmit_buffer"); + memset (&queue->transmit_buffer, 0, sizeof (T_FRAME)); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_recover_queue | ++--------------------------------------------------------------------+ + + PURPOSE : Recover a DL queue after a Reject condition. + +*/ + +GLOBAL void com_recover_queue (UBYTE sapi) +{ + GET_INSTANCE_DATA; + T_QUEUE *queue = sapi EQ PS_SAPI_0 ? &dl_data->dcch0_queue:&dl_data->dcch3_queue; + + TRACE_FUNCTION ("com_recover_queue()"); + + if (queue->sending_buffer NEQ NULL) + { + if(queue->act_length NEQ 0) /* For last buf, act_length is set to 0 */ + { /* and the act_offset is not modified */ + queue->act_offset -= queue->transmit_buffer.l_buf; + } + queue->act_length += queue->transmit_buffer.l_buf; + } + memset (&queue->transmit_buffer, 0, sizeof (T_FRAME)); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_read_queue | ++--------------------------------------------------------------------+ + + PURPOSE : Reading the next segment of a message from a DL queue. + If a message in the switch buffer is stored, this message + is used. If a message is stored in the sending buffer which + is not send completely the next segment is copied to the + transmit buffer. Else the next sending buffer is copied from + the store buffer of the queue. The first segment of this + message is copied to the transfer buffer. + +*/ + +GLOBAL void com_read_queue (UBYTE ch_type, UBYTE sapi, + UBYTE * m_bit) +{ + GET_INSTANCE_DATA; + USHORT i; + USHORT length, bit_length; + T_DL_DATA_REQ * dl_data_req; + T_QUEUE *queue = sapi EQ PS_SAPI_0 ? &dl_data->dcch0_queue : &dl_data->dcch3_queue; + + TRACE_FUNCTION ("com_read_queue()"); + TRACE_EVENT_WIN_P1 ("read sapi_%u_queue", sapi); + + switch (ch_type) + { + case L2_CHANNEL_SDCCH: + length = N201_SDCCH; + break; + + case L2_CHANNEL_SACCH: + length = N201_SACCH; + break; + + case L2_CHANNEL_FACCH_F: + case L2_CHANNEL_FACCH_H: + length = N201_FACCH; + break; + + default: + length = 0; + bit_length = 0; + break; + } + bit_length = length << 3; + + if (queue->switch_buffer NEQ NULL) + { + T_DL_DATA_REQ *switch_buffer = queue->switch_buffer; + + TRACE_EVENT_WIN_P2 ("fill transmit_buffer with bytes %u-%u of switch_buffer, no bytes left", + (switch_buffer->sdu.o_buf>>3), + ((switch_buffer->sdu.o_buf+switch_buffer->sdu.l_buf)>>3)-1); + + queue->m_bit = * m_bit = 0; + queue->transmit_buffer.o_buf = 24; + queue->transmit_buffer.l_buf = switch_buffer->sdu.l_buf; + for (i=0;i<(switch_buffer->sdu.l_buf>>3);i++) + queue->transmit_buffer.buf[i+3] = + switch_buffer->sdu.buf[i+(switch_buffer->sdu.o_buf>>3)]; + + /* + should be freed first after acknowledgement! + COM_FREE_QUEUE_BUFFER (dl_data, queue, INDEX_SWITCH_BUFFER); + */ + return; + } + else + { + if (queue->act_length EQ 0) + { + if (queue->sending_buffer NEQ NULL) + { + COM_FREE_QUEUE_BUFFER (queue, INDEX_SENDING_BUFFER); + } + queue->sending_buffer = queue->store_buffer[0]; + queue->act_offset = queue->sending_buffer->sdu.o_buf; + queue->act_length = queue->sending_buffer->sdu.l_buf; + TRACE_EVENT_WIN_P2 ("fill sending_buffer with bytes %u-%u of store_buffer[0]", + (queue->sending_buffer->sdu.o_buf>>3), + ((queue->sending_buffer->sdu.o_buf+queue->sending_buffer->sdu.l_buf)>>3)-1); + + for (i=0;i<MAX_QUEUED_MESSAGES-1;i++) + queue->store_buffer[i] = queue->store_buffer[i+1]; + queue->no_of_stored_messages--; + + TRACE_EVENT_WIN_P1 ("left no_of_stored_messages=%u", queue->no_of_stored_messages); + } + dl_data_req = queue->sending_buffer; + } + + + if (queue->act_length > bit_length) + { /* + * lint Info 702: Shift right of signed quantity: + * not possible because of the compare one line before + */ + TRACE_EVENT_WIN_P3 ("fill transmit_buffer with bytes %u-%u of sending_buffer, %u bytes left", + queue->act_offset>>3, (queue->act_offset>>3)+length-1, + (queue->act_length-bit_length)>>3); + + queue->m_bit = * m_bit = 1; + queue->transmit_buffer.o_buf = 24; + queue->transmit_buffer.l_buf = bit_length; + for (i=0;i<length;i++) + queue->transmit_buffer.buf[i+3] = + dl_data_req->sdu.buf[i+(queue->act_offset>>3)]; + queue->act_offset += bit_length; + queue->act_length -= bit_length; + } + else + { + TRACE_EVENT_WIN_P2 ("fill transmit_buffer with bytes %u-%u of sending_buffer, no bytes left", + queue->act_offset>>3, ((queue->act_offset+queue->act_length)>>3)-1); + + queue->m_bit = * m_bit = 0; + queue->transmit_buffer.o_buf = 24; + queue->transmit_buffer.l_buf = queue->act_length; + for (i=0;i<(queue->act_length>>3);i++) + queue->transmit_buffer.buf[i+3] = + dl_data_req->sdu.buf[i+(queue->act_offset>>3)]; + queue->act_length = 0; + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_store_queue | ++--------------------------------------------------------------------+ + + PURPOSE : Storing a message into the queue. + +*/ + +GLOBAL void com_store_queue (UBYTE sapi, T_DL_DATA_REQ * data_req) +{ + GET_INSTANCE_DATA; + T_QUEUE * queue; + + TRACE_FUNCTION ("com_store_queue()"); + + if (sapi EQ PS_SAPI_0) + { + queue = &dl_data->dcch0_queue; + } + else + { + queue = &dl_data->dcch3_queue; + } + + if (queue->no_of_stored_messages < MAX_QUEUED_MESSAGES) + { + queue->store_buffer[queue->no_of_stored_messages++] = data_req; + TRACE_EVENT_WIN_P3 ("sapi_%u_queue: add entry with %u bytes, no_of_stored_messages=%u", + sapi, data_req->sdu.l_buf>>3, queue->no_of_stored_messages); + } + else + { + COM_FREE_POINTER (data_req); + TRACE_EVENT_WIN_P1 ("sapi_%u_queue overflowed", sapi); + } +} + +/* ++------------------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_queue_awaiting_transmission | ++------------------------------------------------------------------------------+ + + PURPOSE : The function checks whether any segment has to sended. + Function returns TRUE if a frame/segment is awaiting transmission. + Function returns FALSE if not. + +*/ + +GLOBAL BOOL com_queue_awaiting_transmission (UBYTE sapi) +{ + GET_INSTANCE_DATA; + T_QUEUE * queue; + BOOL ret; + + TRACE_EVENT_WIN_P1 ("com_queue_awaiting_transmission(SAPI=%u)", sapi); + + queue = (sapi EQ PS_SAPI_0) ? &dl_data->dcch0_queue : &dl_data->dcch3_queue; + if (sapi EQ PS_SAPI_0) + { + queue = &dl_data->dcch0_queue; + } + else + { + queue = &dl_data->dcch3_queue; + } + if (queue->switch_buffer NEQ NULL) + { + TRACE_EVENT_WIN_P1 ("sapi_%u_queue: switch_buffer is awaiting", sapi); + ret = TRUE; + } + else + { + if (queue->act_length EQ 0) + { + ret = queue->no_of_stored_messages NEQ 0; + if (ret) + { + TRACE_EVENT_WIN_P2 ("sapi_%u_queue: store_buffer is awaiting (no_of_stored_messages=%u)", + sapi, queue->no_of_stored_messages); + } + } + else + { + TRACE_EVENT_WIN_P2 ("sapi_%u_queue: transmit_buffer is awaiting (%u bytes)", + sapi, queue->act_length); + ret = TRUE; + } + } + + return ret; +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_leave_dedicated | ++--------------------------------------------------------------------+ + + PURPOSE : Leave dedicated mode. + +*/ + +GLOBAL void com_leave_dedicated (UBYTE ch_type) +{ + GET_INSTANCE_DATA; + dl_data->RR_dedicated = FALSE; /* RR is leaving the dedicated mode */ + DL_OFFLINE_TRACE (TRACE_DL_EVENT, TRACE_CH_UNKNOWN, ch_type, "RR_dedicated:=FALSE"); +} + +/* ++-----------------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : possible_reset_dcch0_ch_type | ++-----------------------------------------------------------------------------+ + + PURPOSE : Reset dedicated channel. + +*/ +GLOBAL void possible_reset_dcch0_ch_type (void) +{ + GET_INSTANCE_DATA; + if ( + #if defined(DELAYED_SABM) + (dl_data->dcch0_sabm_flag NEQ NOT_PRESENT_8BIT) AND + #endif /* DELAYED_SABM */ + #if defined(DELAYED_RELEASE_IND) + (dl_data->release_ind_ch_type NEQ NOT_PRESENT_8BIT) AND + #endif /* DELAYED_RELEASE_IND */ + (dl_data->state[C_DCCH0] <= STATE_IDLE_DL) AND + (dl_data->cch[C_DCCH0].vtx EQ EMPTY_CMD)) + { + TRACE_EVENT_WIN_P1 ("reset dcch0_ch_type=%s ->0", CH_TYPE_NAME[dl_data->dcch0_ch_type]); + dl_data->dcch0_ch_type = 0; + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_compare_L3_msg | ++--------------------------------------------------------------------+ + + PURPOSE : The function compares two layer 3 messages. + + +*/ + +GLOBAL UBYTE com_compare_L3_msg (T_DL_DATA_REQ * data_ind1, UBYTE * data_ind2) +{ + USHORT length1; + USHORT length2; + USHORT pos1; + USHORT pos2; + USHORT i; + + TRACE_FUNCTION ("com_compare_L3_msg()"); + + /* + * Calculates Length of SABM and UA layer 3 message + */ + length1 = data_ind1->sdu.l_buf>>3; /* length of l3 msg inside SABM frame */ + length2 = ((data_ind2[2] & 0xFC)>>2); /* length of l3 msg inside UA frame */ + + if (length1 NEQ length2) + return FALSE; + + pos1 = data_ind1->sdu.o_buf >> 3; + pos2 = 3; + + + for (i=0; i<length1; i++) + { + if (data_ind1->sdu.buf[i+pos1] NEQ data_ind2[i+pos2]) + { +#if defined(DL_TRACE_ENABLED) + UBYTE sapi = data_ind1->sapi; + UBYTE trace_channel = TRACE_CH_UNKNOWN; + switch (data_ind1->ch_type) + { + case L2_CHANNEL_SDCCH: + if (sapi EQ PS_SAPI_0) + trace_channel = C_DCCH0; + else if (sapi EQ PS_SAPI_3) + trace_channel = C_DCCH3; + break; + case L2_CHANNEL_FACCH_F: + case L2_CHANNEL_FACCH_H: + trace_channel = C_DCCH0; + break; + default: + break; + }/* endswitch chan */ + DL_OFFLINE_TRACE(TRACE_DL_EVENT, trace_channel, data_ind1->ch_type, "UA doesn´t match"); +#endif /* DL_TRACE_ENABLED */ + return FALSE; + } + } + return TRUE; +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_concatenate | ++--------------------------------------------------------------------+ + + PURPOSE : Concenate an imcoming segment. + +*/ + +GLOBAL void com_concatenate (T_DL_DATA_IND ** in_msg, + UBYTE * new_data_in) +{ + GET_INSTANCE_DATA; + USHORT end_pos; + USHORT length; + USHORT start_pos; + #define L2_HEADER_BYTESIZE 3 + + TRACE_FUNCTION ("com_concatenate()"); + +#if defined(INVOKE_SIGNAL) + if (dl_data->interrupt_context) + { + sig_invoke_com_concatenate (in_msg, new_data_in); + return; + } +#endif /* INVOKE_SIGNAL */ + + length = new_data_in[2] >> 2; + + if (*in_msg EQ NULL) + { + /* + * Nothing stored yet + */ + USHORT len_in_bits = (length + L2_HEADER_BYTESIZE) << 3; + PALLOC_SDU (first_data, DL_DATA_IND, len_in_bits ); + + first_data->sdu.l_buf = length << 3; /* = length * BITS_PER_BYTE */ + first_data->sdu.o_buf = L2_HEADER_BYTESIZE << 3;/* = L2_HEADER_BYTESIZE * BITS_PER_BYTE */ + /*lint -e419 (Warning -- Apparent data overrun) */ + memset (&first_data->sdu.buf[0], 0, L2_HEADER_BYTESIZE); + /*lint +e419 (Warning -- Apparent data overrun) */ + /*lint -e416 (Warning -- creation of out-of-bounds pointer) */ + memcpy (&first_data->sdu.buf[L2_HEADER_BYTESIZE], + &new_data_in[L2_HEADER_BYTESIZE], + length); + /*lint +e416 (Warning -- creation of out-of-bounds pointer) */ + *in_msg = first_data; + } + else + { + T_DL_DATA_IND *previous_data = *in_msg; + + start_pos = previous_data->sdu.l_buf + previous_data->sdu.o_buf; + end_pos = (length << 3) + start_pos; + { + PALLOC_SDU (entire_data, DL_DATA_IND, end_pos ); + + /*lint -e415 (Warning -- access of out-of-bounds pointer) */ + memcpy (entire_data->sdu.buf, previous_data->sdu.buf, start_pos >> 3); + memcpy (&entire_data->sdu.buf[start_pos >> 3], + &new_data_in[L2_HEADER_BYTESIZE], + length); + /*lint +e415 (Warning -- access of out-of-bounds pointer) */ + + entire_data->sdu.l_buf = previous_data->sdu.l_buf + (length << 3); + entire_data->sdu.o_buf = previous_data->sdu.o_buf; + + COM_FREE_POINTER (*in_msg); + *in_msg = entire_data; + } + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_check_nr | ++--------------------------------------------------------------------+ + + PURPOSE : Check the receive number. + +*/ + +GLOBAL UBYTE com_check_nr (UBYTE va, UBYTE vs, UBYTE nr) +{ + BYTE a,b; + + TRACE_FUNCTION ("com_check_nr()"); + /* + * under GSM 4.06 subclause 3.5.2.3: + * nr is valid, if and only if ((nr-va) mod 8) <= ((vs-va) mod 8) + */ + a = (nr+8-va) & 7; + b = (vs+8-va) & 7; + + return (a <= b); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_prepare_DISC | ++--------------------------------------------------------------------+ + + PURPOSE : Prepares a DISC command. + +*/ + +GLOBAL void com_prepare_DISC (UBYTE channel, UBYTE sapi) +{ + GET_INSTANCE_DATA; + T_CCH *pcch = &dl_data->cch[channel]; + + pcch->vtx = DISC_CMD; + pcch->time_flag = TRUE; + pcch->T200_counter = 0; + pcch->rc = 0; + set_channel_state (channel, STATE_AWAITING_RELEASE); + TRACE_EVENT_WIN_P3 ("RELEASE_REQ: %s SAPI=%u vtx=%s", CH_TYPE_NAME[pcch->ch_type], + sapi, VTX_NAME[pcch->vtx]); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_UA_response | ++--------------------------------------------------------------------+ + + PURPOSE : Build an UA response. + +*/ + +GLOBAL void com_build_UA_response (UBYTE ch_type, UBYTE sapi, UBYTE f_bit) +{ + TRACE_FUNCTION ("com_build_UA_response()"); + com_build_frame (ch_type, UA_FRAME, sapi, MS2BS_RSP, 0, 0, f_bit, 0, 0, NULL, 0); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_RR_response | ++--------------------------------------------------------------------+ + + PURPOSE : Build an RR response. + +*/ + +GLOBAL void com_build_RR_response (UBYTE ch_type, + UBYTE sapi, UBYTE nr, UBYTE f_bit) +{ + TRACE_FUNCTION ("com_build_RR_response()"); + com_build_frame (ch_type, RR_FRAME, sapi, MS2BS_RSP, 0, nr, f_bit, 0, 0, NULL, 0); +} + +#if 0 +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_RR_command | ++--------------------------------------------------------------------+ + + PURPOSE : Build an RR command. + +*/ + +GLOBAL void com_build_RR_command (T_DL_DATA_STORE * dl_data, UBYTE ch_type, + UBYTE sapi, UBYTE nr, UBYTE p_bit) +{ + TRACE_FUNCTION ("com_build_RR_command()"); + com_build_frame (dl_data, ch_type, RR_FRAME, sapi, MS2BS_CMD, 0, nr, p_bit, 0, 0, NULL, 0); +} +#endif /* 0 */ +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_REJ_response| ++--------------------------------------------------------------------+ + + PURPOSE : Build an REJ response. + +*/ + +GLOBAL void com_build_REJ_response (UBYTE ch_type, + UBYTE sapi, UBYTE nr, UBYTE f_bit) +{ + TRACE_FUNCTION ("com_build_REJ_response()"); + com_build_frame (ch_type, REJ_FRAME, sapi, MS2BS_RSP, 0, nr, f_bit, 0, 0, NULL, 0); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_DISC_command | ++--------------------------------------------------------------------+ + + PURPOSE : Build a DISC command. + +*/ + +GLOBAL void com_build_DISC_command (UBYTE ch_type, + UBYTE sapi, UBYTE p_bit) +{ + TRACE_FUNCTION ("com_build_DISC_command()"); + com_build_frame (ch_type, DISC_FRAME, sapi, MS2BS_CMD, 0, 0, p_bit, 0, 0, NULL, 0); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_SABM | ++--------------------------------------------------------------------+ + + PURPOSE : Build a SABM command with or without Layer 3 message. + +*/ + +GLOBAL void com_build_SABM (UBYTE ch_type, + UBYTE sapi, BOOL contention_resultion) +{ + GET_INSTANCE_DATA; + T_QUEUE *queue = &dl_data->dcch0_queue; + TRACE_FUNCTION ("com_build_SABM()"); + TRACE_EVENT_WIN_P3 ("com_build_SABM %s SAPI=%u %s", CH_TYPE_NAME[ch_type], sapi, + (contention_resultion AND queue->switch_buffer) ? "+L3" : ""); + + if (contention_resultion AND queue->switch_buffer) + { + com_build_frame (ch_type, SABM_FRAME, sapi, MS2BS_CMD, 0, 0, 1, + (UBYTE)(queue->switch_buffer->sdu.l_buf >> 3), 0, + queue->switch_buffer->sdu.buf, (UBYTE)(queue->switch_buffer->sdu.o_buf >> 3)); + } + else + { + com_build_frame (ch_type, SABM_FRAME, sapi, MS2BS_CMD, 0, 0, 1, 0, 0, NULL, 0); + } +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_DM_response | ++--------------------------------------------------------------------+ + + PURPOSE : Build an DM response. + +*/ + +GLOBAL void com_build_DM_response (UBYTE ch_type, + UBYTE sapi, UBYTE f_bit) +{ + TRACE_FUNCTION ("com_build_DM_response()"); + com_build_frame (ch_type, DM_FRAME, sapi, MS2BS_RSP, 0, 0, f_bit, 0, 0, NULL, 0); +} + + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_I_command | ++--------------------------------------------------------------------+ + + PURPOSE : Build an I command. + +*/ + +GLOBAL void com_build_I_command (UBYTE ch_type, + UBYTE sapi, UBYTE ns, UBYTE nr, UBYTE p_bit,UBYTE m_bit, T_QUEUE * queue) +{ + TRACE_FUNCTION ("com_build_I_command()"); + com_build_frame (ch_type, I_FRAME, sapi, MS2BS_CMD, ns, nr, p_bit, + (UBYTE)(queue->transmit_buffer.l_buf >> 3), m_bit, + queue->transmit_buffer.buf, 0); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_UI_command | ++--------------------------------------------------------------------+ + + PURPOSE : Build an UI command. + +*/ + +GLOBAL void com_build_UI_command (UBYTE ch_type, + UBYTE sapi, const T_FRAME * buffer) +{ + TRACE_FUNCTION ("com_build_UI_command()"); + com_build_frame (ch_type, UI_FRAME, sapi, MS2BS_CMD, 0, 0, 0, + (UBYTE)(buffer->l_buf >> 3), 0, (UBYTE*)buffer->buf, (UBYTE)(buffer->o_buf >> 3)); +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_UI_Bter | ++--------------------------------------------------------------------+ + + PURPOSE : Build an UI frame in Bter format. + +*/ + +GLOBAL void com_build_UI_Bter (UBYTE ch_type) +{ + GET_INSTANCE_DATA; + unsigned off = dl_data->rr_short_pd_buffer.o_buf>>3; + unsigned len = dl_data->rr_short_pd_buffer.l_buf>>3; + unsigned foff, maxlen; + + if (ch_type EQ L2_CHANNEL_SACCH) + { + maxlen = DL_N201_SACCH_Bter; + foff = 2; +#ifndef DL_2TO1 + dl_data->l2_frame.A[0] = dl_data->l2_frame.A[1] = 0; /* place holder for L1 header */ +#else + dl_data->l2_frame.frame_array[0] = dl_data->l2_frame.frame_array[1] = 0; /* place holder for L1 header */ +#endif + + } + else + { + foff = 0; + maxlen = DL_N201_DCCH_Bter; + } + + if (len > maxlen) + len = maxlen; + /* + * LINT Warning 662: Possible creation of out-of-bounds pointer ... and + * LINT Warning 669: Possible data overrun for function 'memset ... + * can be ignored, because of the right alignment between foff and maxlen. + */ + /*lint -e662 -e669 Possible creation of out-of-bounds pointer or Possible data overrun*/ +#ifndef DL_2TO1 + memcpy (&dl_data->l2_frame.A[foff], dl_data->rr_short_pd_buffer.buf+off, len); + memset (&dl_data->l2_frame.A[foff+len], 0x2b, maxlen-len); +#else + memcpy (&dl_data->l2_frame.frame_array[foff], dl_data->rr_short_pd_buffer.buf+off, len); + memset (&dl_data->l2_frame.frame_array[foff+len], 0x2b, maxlen-len); +#endif +/* lint +e662 +e669 Possible creation of out-of-bounds pointer or Possible data overrun */ +/* mark message as handled */ + dl_data->rr_short_pd_buffer.l_buf = 0; +} + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_build_frame | ++--------------------------------------------------------------------+ + + PURPOSE : Builds any frame. + +*/ +LOCAL void com_build_frame (UBYTE ch_type, + UBYTE type, UBYTE sapi, UBYTE cr, UBYTE ns, + UBYTE nr, UBYTE p_bit, UBYTE length, UBYTE m_bit, + UBYTE* pInfoBuffer, UBYTE InfoOffset) +{ + GET_INSTANCE_DATA; + UBYTE offset; + UBYTE maxlength; +#ifndef DL_2TO1 + UBYTE *pOutFrame = dl_data->l2_frame.A; +#else + UBYTE *pOutFrame = dl_data->l2_frame.frame_array; +#endif /* DL_2TO1*/ +#define SPARE 0 +#define LPD 0 +#define EA 1 +#define EL 1 + + if (ch_type EQ L2_CHANNEL_SACCH) + { + offset = 2; + maxlength = DL_N201_SACCH_A_B; + pOutFrame[0] = pOutFrame[1] = 0; /* reset layer 1 header */ + } + else + { + offset = 0; + maxlength = DL_N201_DCCH_A_B; + } + + if (length > maxlength) + { + TRACE_EVENT_P1 ("Error: framelength to big %d", length); + length = maxlength; /* for safety's sake */ + } + + /* + * set header bytes (bit 8 7 6 5 4 3 2 1) + * address field | | | | | | | + * Spare --LPD- ----SAPI--- C/R EA + */ + pOutFrame[offset] = (SPARE << 7) | (LPD << 5) | (sapi << 2) | (cr << 1) | EA; + + /* + * control field 8 7 6 5 4 3 2 1 + * I format ----N(R)--- P ----N(S)--- 0 + * S format ----N(R)--- P/F S S 0 1 + * U format U U U P/F U U 1 1 + */ + if ((type & 0x01) EQ 0) + pOutFrame[offset+1] = (nr << 5) | (p_bit << 4) | (ns << 1); /* I format */ + else if ((type & 0x02) EQ 0) + pOutFrame[offset+1] = (nr << 5) | (p_bit << 4) | type; /* S format */ + else + pOutFrame[offset+1] = (p_bit << 4) | type; /* U format */ + + /* + * length field 8 3 2 1 + * -----length indicator----- M EL + */ + pOutFrame[offset+2] = (length << 2) | (m_bit << 1) | EL; + + /* + * LINT Warning 662: Possible creation of out-of-bounds pointer ... and + * LINT Warning 669: Possible data overrun for function 'memcpy ... and + * LINT Warning 671: Possibly passing to function 'memset ... + * can be ignored because the sum of offset, 3 plus maxlength (length) does + * never exceed the maximal size of pOutFrame[]. + */ + if (length && pInfoBuffer) + {/* copy info bits */ + memcpy (&pOutFrame[offset+3], pInfoBuffer + (InfoOffset ? InfoOffset : 3), length); + offset += length; + maxlength -= length; + } + + /* fill remain of the frame with 0x2b */ + memset (&pOutFrame[offset+3], 0x2b, maxlength); + +#undef SPARE +#undef LPD +#undef EA +#undef EL +}/* endfunc com_build_frame */ + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_data_ind | ++--------------------------------------------------------------------+ + + PURPOSE : Indicates a downlinked message to RR. The queue which is + taken depend on the sapi value. In case of sapi=0 the function + checks the layer 3 message for a channel release additionally. + This is done to speed up the release procedure. + +*/ +GLOBAL void com_data_ind (UBYTE ch_type, UBYTE sapi, ULONG fn) +{ + GET_INSTANCE_DATA; + T_DL_DATA_IND *in_msg; + +#if defined(INVOKE_SIGNAL) + if (dl_data->interrupt_context) + { + sig_invoke_com_data_ind (ch_type, sapi, fn); + return; + } +#endif /* INVOKE_SIGNAL */ + + if (sapi EQ PS_SAPI_0) + { + in_msg = dl_data->dcch0_in_msg; + + if (in_msg) + { /* + * Purpose: Check if message is channel release, then start release + * of connection. This will be done to speed up the DISC process. + */ + /*lint -e415 (Warning -- access of out-of-bounds pointer) */ + /*lint -e416 (Warning -- creation of out-of-bounds pointer) */ + if ((in_msg->sdu.buf[3] EQ 0x06) AND /* PD RR */ + (in_msg->sdu.buf[4] EQ 0x0D)) /* MT channel release */ + { + /*lint +e416 (Warning -- creation of out-of-bounds pointer) */ + /*lint +e415 (Warning -- access of out-of-bounds pointer) */ + /* + * Send immediately (on the next uplink opportunity) a DISC frame + * on DCCH. + * Disable dedicated mode -> restrain measurement reports + */ + dl_data->dcch0_disc_request = TRUE; + DL_OFFLINE_TRACE(TRACE_DL_EVENT, C_DCCH0, ch_type, "CHANNEL REL received"); +#if !defined(LATE_LEAVING_DEDICATED) + com_leave_dedicated (ch_type); +#endif /* LATE_LEAVING_DEDICATED */ + } + } + } + else + { + in_msg = dl_data->dcch3_in_msg; + } + + if (in_msg) + { + in_msg->ch_type = ch_type; + in_msg->sapi = sapi; + drr_dl_data_ind (sapi, fn); + } + else + { + DL_OFFLINE_TRACE (TRACE_DL_EVENT, + (sapi EQ PS_SAPI_0) ? C_DCCH0 : C_DCCH3, ch_type, "dcchx_in_msg=NULL"); + } +} + +#if !defined(NTRACE) +#if defined(DL_TRACE_ENABLED) && defined(DL_IMMEDIATE_TRACE) +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_l2trace | ++--------------------------------------------------------------------+ + + PURPOSE : Send L2 trace. + +*/ + +GLOBAL void com_l2trace (UBYTE trace_type, + UBYTE channel, UBYTE ch_type, T_TIME trace_time, UBYTE* data) +{ + GET_INSTANCE_DATA; + ULONG trace_mask; + +#ifdef TI_PS_HCOMM_CHANGE + vsi_gettracemask(_hCommDL, _hCommDL, &trace_mask); +#else + vsi_gettracemask(hCommDL, hCommDL, &trace_mask); +#endif + if ((trace_mask & TC_USER1) EQ 0) + return; + + if (trace_time EQ 0) + vsi_t_time (VSI_CALLER &trace_time); + +#if defined(INVOKE_SIGNAL) + if (dl_data->interrupt_context) + { + sig_invoke_com_l2trace (trace_type, channel, ch_type, trace_time, data); + return; + } +#endif /* INVOKE_SIGNAL */ + dl_fast_trace(trace_type, channel, ch_type, trace_time, trace_mask, data); +} +#endif /* DL_TRACE_ENABLED && DL_IMMEDIATE_TRACE */ + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_l3trace | ++--------------------------------------------------------------------+ + + PURPOSE : Send L3 trace. + +*/ + +GLOBAL void com_l3trace (UBYTE type, UBYTE ch_type, UBYTE *frame) +{ + GET_INSTANCE_DATA; + ULONG trace_mask; + +#ifdef TI_PS_HCOMM_CHANGE + vsi_gettracemask(_hCommDL, _hCommDL, &trace_mask); +#else + vsi_gettracemask(hCommDL, hCommDL, &trace_mask); +#endif + if ((trace_mask & TC_USER4) EQ 0) + return; + +#if defined(INVOKE_SIGNAL) + if (dl_data->interrupt_context) + { + sig_invoke_com_l3trace (type, ch_type, frame); + return; + } +#endif /* INVOKE_SIGNAL */ + + if (type EQ TRACE_UPLINK) + { /* uplink acknowledged */ + T_DL_DATA_REQ *d = (T_DL_DATA_REQ *)frame; + com_print_l3trace (TRACE_UPLINK, ch_type, d->sapi, (UBYTE *)&d->sdu); + } + else if (type EQ TRACE_DOWNLINK) + { /* downlink */ + T_DL_DATA_IND *d = (T_DL_DATA_IND *)frame; + com_print_l3trace (TRACE_DOWNLINK, ch_type, d->sapi, (UBYTE *)&d->sdu); + } + else if ((type EQ TRACE_UACK_UP) OR (type EQ TRACE_UACK_DN)) + { + com_print_l3trace (type, ch_type, PS_SAPI_0, frame); + } +} + +GLOBAL void com_print_l3trace (UBYTE type, UBYTE ch_type, UBYTE sapi, UBYTE *l3msg) +{ + char description[40]; + + switch (type) + { + case TRACE_UPLINK: + case TRACE_DOWNLINK: + { + T_sdu *sdu = (T_sdu *)l3msg; + sprintf (description, "L3 %s ch=%u SAPI%u", + type EQ TRACE_UPLINK ? "UP" : "DN", ch_type, sapi); +#ifdef TI_PS_HCOMM_CHANGE + TRACE_BINDUMP(_hCommDL, TC_USER4, description, + sdu->buf+(sdu->o_buf>>3), (sdu->l_buf>>3)); +#else + TRACE_BINDUMP(hCommDL, TC_USER4, description, + sdu->buf+(sdu->o_buf>>3), (sdu->l_buf>>3)); +#endif + } + break; + case TRACE_UACK_UP: + case TRACE_UACK_DN: + if (GET_BTER_FORMAT (&l3msg[0]) EQ SHORT_L2_HEADER_TYPE_1) + { + sprintf (description, "L3 %s ch=%u SAPI%u RR Short PD header", + type EQ TRACE_UACK_UP ? "UP" : "DN", ch_type, sapi); +#ifdef TI_PS_HCOMM_CHANGE + TRACE_BINDUMP(_hCommDL, TC_USER4, description, l3msg, DL_N201_SACCH_Bter); +#else + TRACE_BINDUMP(hCommDL, TC_USER4, description, l3msg, DL_N201_SACCH_Bter); +#endif + } + else + { + sprintf (description, "L3 %s ch=%u SAPI%u", + type EQ TRACE_UACK_UP ? "UP" : "DN", ch_type, sapi); +#ifdef TI_PS_HCOMM_CHANGE + TRACE_BINDUMP(_hCommDL, TC_USER4, description, l3msg, DL_N201_SACCH_A_B); +#else + TRACE_BINDUMP(hCommDL, TC_USER4, description, l3msg, DL_N201_SACCH_A_B); +#endif + } + break; + } +} +#endif /* !NTRACE */ + +/* ++--------------------------------------------------------------------+ +| PROJECT : GSM-PS (6147) MODULE : DL_COM | +| STATE : code ROUTINE : com_init_data | ++--------------------------------------------------------------------+ + + PURPOSE : Initializes the data for one instance. + +*/ + +GLOBAL void com_init_data (void) +{ + GET_INSTANCE_DATA; + TRACE_FUNCTION ("com_init_data()"); + + memset (dl_data, 0, sizeof (T_DL_DATA_STORE)); + + dl_data->cch[C_DCCH0].T200_counter = 0; + dl_data->cch[C_DCCH3].T200_counter = 0; + dl_data->cch[C_SACCH0].T200_counter = 0; + dl_data->dcch0_in_msg = NULL; + dl_data->dcch3_in_msg = NULL; + + dl_data->dcch0_disc_request = FALSE; + dl_data->dcch3_disc_request = FALSE; + dl_data->RR_dedicated = FALSE; + dl_data->fn = NOT_PRESENT_32BIT; + +#if defined(INVOKE_SIGNAL) + sig_init_signal_data (); +#endif /* INVOKE_SIGNAL */ +} + + +#if defined (DL_TRACE_ENABLED) || defined (FF_EM_MODE) +/* +* Some of the functions originally designed for DL tracing are used as well for +* the engineering mode and therefore defined here. +*/ + +GLOBAL void com_semaphore_err (void) +{ +#if defined(_SIMULATION_) + SYST_TRACE ("DL:error:semaphore"); + TRACE_ASSERT (1); +#else /* _SIMULATION_ */ +static UCHAR out = 0; + if (!out) + { + out = 1; + SYST_TRACE ("DL:error:semaphore"); + vsi_t_sleep(VSI_CALLER 10000); + TRACE_ASSERT (1); + } +#endif /* _SIMULATION_ */ +} + +GLOBAL int com_enter_critical_section (T_HANDLE sem) +{ + if (vsi_s_get (VSI_CALLER sem) NEQ VSI_OK) + { + com_semaphore_err(); + return -1; + } + else + { + return 0; + } +}/*endfunc com_enter_critical_section*/ + +GLOBAL int com_leave_critical_section (T_HANDLE sem) +{ + if (vsi_s_release (VSI_CALLER sem) NEQ VSI_OK) + { + com_semaphore_err(); + return -1; + } + else + { + return 0; + } +}/* endfunc com_leave_critical_section */ + +#if !defined(DL_IMMEDIATE_TRACE) +GLOBAL int com_semaphore_state (T_HANDLE sem) +{ + USHORT semCount; + + if (vsi_s_status (VSI_CALLER sem, &semCount) NEQ VSI_OK) + { + com_semaphore_err (); + return -1; + } + if (semCount EQ 0) + { + SYST_TRACE ("DL:semCount == 0"); + return 1; + } + else + return 0; +} +#endif /* !DL_IMMEDIATE_TRACE */ +#endif /* defined (DL_TRACE_ENABLED) || defined (FF_EM_MODE) */ + +#endif /* DL_COM_C */ +