view src/g23m-gprs/sndcp/sndcp_f.c @ 638:cab2f315827e

FFS dev.c: added Spansion PL032J to the "generic" table With the discovery of first GTM900 and then Tango, it now appears that Openmoko was not the only manuf after all who kept TI's TCS211 firmware largely intact (as opposed to changing it beyond all recognition like Compal, Chi-Mei and BenQ did), thus we are now getting new "alien" targets on which we reuse the original manuf's FFS with IMEI and RF calibration tables as if it were native. On these targets we use the original device table for FFS, even though we previously thought that it would never apply to any target other than dsample, leonardo and gtamodem. We have previously added Samsung K5L33xxCAM (a new kind of multi-ID device) to the generic table to support its use in Huawei GTM900-B modules; now we got news that some slightly older GTM900-B specimen used S71PL032J instead, so we are now adding PL032J as well.
author Mychaela Falconia <falcon@freecalypso.org>
date Thu, 30 Jan 2020 17:45:48 +0000
parents 219afcfc6250
children
line wrap: on
line source

/*
+-----------------------------------------------------------------------------
|  Project :  GPRS (8441)
|  Modul   :  sndcp_f.c
+-----------------------------------------------------------------------------
|  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 :  Contains global functions of SNDCP
+-----------------------------------------------------------------------------
*/

#define ENTITY_SNDCP

/*==== INCLUDES =============================================================*/

#include "typedefs.h"    /* to get Condat data types */
#include "vsi.h"        /* to get a lot of macros */
#include "gsm.h"        /* to get a lot of macros */
#include "prim.h"       /* to get the definitions of used SAP and directions */
#include "macdef.h"

#include "dti.h"
#include "sndcp.h"        /* to get the global entity definitions */
#include "sndcp_f.h"       /* to get the functions to access the global arrays*/

#include "sndcp_nup.h"      /* to get nu functions that will be called from
                               sndcp_sig_callback() */
#include "sndcp_ndp.h"      /* to get nu functions that will be called from
                               sndcp_sig_callback() */


#include <string.h>     /* to get memcpy() */


/*==== CONST ================================================================*/

#ifdef SNDCP_TRACE_IP_DATAGRAM
#ifndef CF_FAST_EXEC
static U8 bin_trace_ip_buf[1500];
#endif /* CF_FAST_EXEC */
#endif
/*==== LOCAL VARS ===========================================================*/




#ifdef TI_PS_OP_ICUT_SNDCP

/* PDP_TBR add SNDCP terminal loopback capability */
extern U8 SNDCP_LOOPBACK;
extern BOOL bufFull[SNDCP_NUMBER_OF_NSAPIS];
extern T_DTI2_DATA_IND *saveDti2_data_ind[SNDCP_NUMBER_OF_NSAPIS] ;

#endif /* TI_PS_OP_ICUT_SNDCP */





/*==== PRIVATE FUNCTIONS ====================================================*/

/*==== PUBLIC FUNCTIONS =====================================================*/


/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_ack
+------------------------------------------------------------------------------
| Description : gets nsapi_ack_ra[nsapi]
|
| Parameters  : UBYTE nsapi, BOOL* ack
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_ack(UBYTE nsapi,
                                  BOOL* ack)
{
  TRACE_FUNCTION("sndcp_get_nsapi_ack");
  {
    *ack = sndcp_data->nsapi_ack_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

#ifdef _SNDCP_DTI_2_
/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_direction
+------------------------------------------------------------------------------
| Description : gets nsapi_ack_direction[nsapi]
|               Values: HOME or NEIGHBOR from dti.h.
|
| Parameters  : UBYTE nsapi, BOOL* direction
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_direction(UBYTE nsapi,
                                      U8* direction)
{
  TRACE_FUNCTION("sndcp_get_nsapi_direction");
  {
    *direction = (U8)sndcp_data->nsapi_direction_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_interface
+------------------------------------------------------------------------------
| Description : gets nsapi_interface_ra[nsapi]
|
| Parameters  : UBYTE nsapi, U8 interface
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_interface(UBYTE nsapi,
                                      U8* interfac)
{
  TRACE_FUNCTION("sndcp_get_nsapi_interface");
  {
    *interfac = sndcp_data->nsapi_interface_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_linkid
+------------------------------------------------------------------------------
| Description : gets nsapi_linkid_ra[nsapi]
|
| Parameters  : UBYTE nsapi, ULONG* linkid
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_linkid(UBYTE nsapi,
                                   ULONG* linkid)
{
  TRACE_FUNCTION("sndcp_get_nsapi_linkid");
  {
    *linkid = sndcp_data->nsapi_linkid_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_neighbor
+------------------------------------------------------------------------------
| Description : gets nsapi_neighbor_ra[nsapi]
|
| Parameters  : UBYTE nsapi, UBYTE** neighbor
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_neighbor(UBYTE nsapi,
                                     UBYTE** neighbor)
{
  TRACE_FUNCTION("sndcp_get_nsapi_neighbor");
  {
    *neighbor = sndcp_data->nsapi_neighbor_ra[nsapi];
  }
}
#endif /* CF_FAST_EXEC */

#else /*_SNDCP_DTI_2_*/
/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_direction
+------------------------------------------------------------------------------
| Description : gets nsapi_ack_direction[nsapi]
|               Values: HOME or NEIGHBOR from dti.h.
|
| Parameters  : UBYTE nsapi, BOOL* direction
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_direction(UBYTE nsapi,
                                      BOOL* direction)
{
  TRACE_FUNCTION("sndcp_get_nsapi_direction");
  {
    *direction = sndcp_data->nsapi_direction_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_interface
+------------------------------------------------------------------------------
| Description : gets nsapi_interface_ra[nsapi]
|
| Parameters  : UBYTE nsapi, U8 interface
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_interface(UBYTE nsapi,
                                      U8* interfac)
{
  TRACE_FUNCTION("sndcp_get_nsapi_interface");
  {
    *interfac = sndcp_data->nsapi_interface_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_linkid
+------------------------------------------------------------------------------
| Description : gets nsapi_linkid_ra[nsapi]
|
| Parameters  : UBYTE nsapi, ULONG* linkid
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_linkid(UBYTE nsapi,
                                   ULONG* linkid)
{
  TRACE_FUNCTION("sndcp_get_nsapi_linkid");
  {
    *linkid = sndcp_data->nsapi_linkid_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_neighbor
+------------------------------------------------------------------------------
| Description : gets nsapi_neighbor_ra[nsapi]
|
| Parameters  : UBYTE nsapi, UBYTE** neighbor
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_neighbor(UBYTE nsapi,
                                     UBYTE** neighbor)
{
  TRACE_FUNCTION("sndcp_get_nsapi_neighbor");
  {
    *neighbor = sndcp_data->nsapi_neighbor_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

#endif /*_SNDCP_DTI_2_*/

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_sapi_ack
+------------------------------------------------------------------------------
| Description : gets sapi_ack_ra["index of sapi"]
|
| Parameters  : UBYTE sapi, BOOL* ack
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_sapi_ack(UBYTE sapi,
                                  BOOL* ack)
{
  TRACE_FUNCTION("sndcp_get_sapi_ack");
  {
    UBYTE sapi_index = 0;

    sndcp_get_sapi_index(sapi, &sapi_index);
    *ack = sndcp_data->sapi_ack_ra[sapi_index];
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_sapi_stat
+------------------------------------------------------------------------------
| Description : gets sapi_state_ra[sapi]
|
| Parameters  : UBYTE sapi, UBYTE* stat
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_sapi_state(UBYTE sapi,
                                 USHORT* stat)
{
  TRACE_FUNCTION("sndcp_get_sapi_state");
  {
    UBYTE sapi_index = 0;

    sndcp_get_sapi_index(sapi, &sapi_index);
    *stat = sndcp_data->sapi_state_ra[sapi_index];
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_sapi_index
+------------------------------------------------------------------------------
| Description : gets the index (0, 1, 2, 3) for the given sapi (3, 5, 9, 11).
| This function is used when a sapi number serves as a key for one of the 4
| possible instances of services su and sd.
|
| Parameters  : UBYTE sapi, UBYTE* index
|
+------------------------------------------------------------------------------
*/
/*#if defined(CF_FAST_EXEC) || defined(_SIMULATION_) || \
   defined(SNDCP_2to1) */

GLOBAL void sndcp_get_sapi_index(UBYTE sapi,
                                 UBYTE* index)
{
  TRACE_FUNCTION("sndcp_get_sapi_index");
  {
    if (sapi == 3) {
      *index = 0;
    } else if (sapi == 5) {
      *index = 1;
    } else if (sapi == 9) {
      *index = 2;
    } else if (sapi == 11) {
      *index = 3;
    }
  }
}

/*#endif */ /* CF_FAST_EXEC || _SIMULATION_ || !REL99 || SNDCP_2to1 */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_prio
+------------------------------------------------------------------------------
| Description : sets prio to nsapi_prio_ra[nsapi]
|
| Parameters  : UBYTE nsapi, UBYTE* prio
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_prio(UBYTE nsapi,
                                 UBYTE* prio)
{
  TRACE_FUNCTION("sndcp_get_nsapi_prio");
  {
    *prio = sndcp_data->nsapi_prio_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

#ifdef REL99 
/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_pktflowid
+------------------------------------------------------------------------------
| Description : gets packet flow id form nsapi_pktflowid_ra[nsapi]
|
| Parameters  : UBYTE nsapi, UBYTE* pkt_flow_id
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_pktflowid(U8 nsapi, U16* pkt_flow_id)

{
  TRACE_FUNCTION("sndcp_get_nsapi_pktflowid");
  {
    *pkt_flow_id = sndcp_data->nsapi_pktflowid_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

#endif /*REL99*/

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_qos
+------------------------------------------------------------------------------
| Description : gets nsapi_qos_ra[nsapi]
|
| Parameters  : UBYTE nsapi, T_snsm_qos* qos
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_qos(UBYTE nsapi,
                                 T_snsm_qos* qos)
{
  TRACE_FUNCTION("sndcp_get_nsapi_qos");
  {
    qos->delay = sndcp_data->nsapi_qos_ra[nsapi].delay;
    qos->relclass = sndcp_data->nsapi_qos_ra[nsapi].relclass;
    qos->peak = sndcp_data->nsapi_qos_ra[nsapi].peak;
    qos->preced = sndcp_data->nsapi_qos_ra[nsapi].preced;
    qos->mean = sndcp_data->nsapi_qos_ra[nsapi].mean;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_sapi
+------------------------------------------------------------------------------
| Description : gets nsapi_sapi_ra[nsapi]
|
| Parameters  : UBYTE nsapi, UBYTE* sapi
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_sapi(UBYTE nsapi,
                                 UBYTE* sapi)
{
  TRACE_FUNCTION("sndcp_get_nsapi_sapi");
  {
    *sapi = sndcp_data->nsapi_sapi_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_state
+------------------------------------------------------------------------------
| Description : gets nsapi_state_ra[nsapi]
|
| Parameters  : UBYTE nsapi, USHORT* state
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_get_nsapi_state(UBYTE nsapi,
                                  USHORT* state)
{
  TRACE_FUNCTION("sndcp_get_nsapi_state");
  {
    *state = sndcp_data->nsapi_state_ra[nsapi];
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_is_nsapi_data_compressed
+------------------------------------------------------------------------------
| Description : If nsapi uses data compression then compressed is set to TRUE,
|               else to FALSE.
|
| Parameters  : UBYTE nsapi, BOOL* compressed
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_is_nsapi_data_compressed(UBYTE nsapi,
                                           BOOL* compressed)
{
  USHORT nsapis = sndcp_data->cia.cur_xid_block.v42.nsapis;

  TRACE_FUNCTION("sndcp_is_nsapi_data_compressed");

  *compressed = (nsapis & (1 << nsapi)) > 0;

}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_is_nsapi_used
+------------------------------------------------------------------------------
| Description : sets b to TRUE if the nsapi is already in use, otherwise to
|               FALSE
|
| Parameters  : UBYTE nsapi, BOOL* b
|
+------------------------------------------------------------------------------
*/
/*#if defined(CF_FAST_EXEC) || defined(_SIMULATION_) || \
   defined (SNDCP_2to1) */

GLOBAL void sndcp_is_nsapi_used(UBYTE nsapi,
                                BOOL* b)
{
  TRACE_FUNCTION("sndcp_is_nsapi_used");
  {
    *b = sndcp_data->nsapi_used_ra[nsapi];
  }
}

/*#endif */ /* CF_FAST_EXEC || _SIMULATION_ || !REL99 || SNDCP_2to1 */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_is_nsapi_header_compressed
+------------------------------------------------------------------------------
| Description : If nsapi uses header compression then compressed is set to TRUE,
|               else to FALSE.
|
| Parameters  : UBYTE nsapi, BOOL* compressed
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_is_nsapi_header_compressed(UBYTE nsapi,
                                             BOOL* compressed)
{
  USHORT nsapis = sndcp_data->cia.cur_xid_block.vj.nsapis;

  TRACE_FUNCTION("sndcp_is_nsapi_header_compressed");

  *compressed = ((nsapis & (1 << nsapi)) > 0) &&
    sndcp_data->cia.cur_xid_block.vj.is_set;

}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_mean_trace
+------------------------------------------------------------------------------
| Description : traces the mean thruput for given
|
| Parameters  : nsapi, direction, ack_mode
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_mean_trace(UBYTE nsapi,
                             UBYTE direction,
                             UBYTE ack_mode,
                             USHORT len)
{
  T_TIME* start_time_ra = NULL;
  T_TIME* cur_time_ra = NULL;
  ULONG* cur_num_ra = NULL;
  ULONG* cur_pac_ra = NULL;

  TRACE_FUNCTION("sndcp_mean_trace");

  if (direction == SNDCP_MEAN_UP) {
    if (ack_mode == SNDCP_MEAN_ACK) {
      start_time_ra = sndcp_data->start_time_uplink_ack;
      cur_time_ra = sndcp_data->cur_time_uplink_ack;
      cur_num_ra = sndcp_data->cur_num_uplink_ack;
      cur_pac_ra = sndcp_data->cur_pac_uplink_ack;
    } else {
      start_time_ra = sndcp_data->start_time_uplink_unack;
      cur_time_ra = sndcp_data->cur_time_uplink_unack;
      cur_num_ra = sndcp_data->cur_num_uplink_unack;
      cur_pac_ra = sndcp_data->cur_pac_uplink_unack;
    }
  } else {
    if (ack_mode == SNDCP_MEAN_ACK) {
      start_time_ra = sndcp_data->start_time_downlink_ack;
      cur_time_ra = sndcp_data->cur_time_downlink_ack;
      cur_num_ra = sndcp_data->cur_num_downlink_ack;
      cur_pac_ra = sndcp_data->cur_pac_downlink_ack;
    } else {
      start_time_ra = sndcp_data->start_time_downlink_ack;
      cur_time_ra = sndcp_data->cur_time_downlink_ack;
      cur_num_ra = sndcp_data->cur_num_downlink_unack;
      cur_pac_ra = sndcp_data->cur_pac_downlink_unack;
    }
  }

  /*
   * Increment number of packets.
   */
  cur_pac_ra[nsapi] ++;


  if (start_time_ra[nsapi] == 0) {
    vsi_t_time(VSI_CALLER  &start_time_ra[nsapi]);
    cur_num_ra[nsapi] += len;
  } else {

    ULONG mean = 0;
    ULONG delta_millis = 0;

    vsi_t_time(VSI_CALLER  &cur_time_ra[nsapi]);
    cur_num_ra[nsapi] += len;

    delta_millis = cur_time_ra[nsapi] -
                   start_time_ra[nsapi];
    if (direction == SNDCP_MEAN_UP) {
      if (delta_millis > 0) {
        mean = (cur_num_ra[nsapi] * 1000) / delta_millis;
        TRACE_EVENT_P4(
          "nsapi %d  up. %d octets in %d ms. %d octets per sec.",
          nsapi,
          cur_num_ra[nsapi],
          delta_millis,
          mean
        );
      } else {
        TRACE_EVENT_P3(
          "nsapi %d  up. %d octets in %d ms.",
          nsapi,
          cur_num_ra[nsapi],
          delta_millis
        );
      }

    } else { /* not (direction == SNDCP_MEAN_UP) */
      if (delta_millis > 0) {
        mean = (cur_num_ra[nsapi] * 1000) / delta_millis;
        TRACE_EVENT_P4(
          "nsapi %d  down. %d octets in %d ms. %d octets per sec.",
          nsapi,
          cur_num_ra[nsapi],
          delta_millis,
          mean
        );
      } else {
        TRACE_EVENT_P3(
          "nsapi %d  down. %d octets in %d ms.",
          nsapi,
          cur_num_ra[nsapi],
          delta_millis
        );
      }
    } /* not (direction == SNDCP_MEAN_UP) */

  }

}

#endif /* CF_FAST_EXEC */


/*
+------------------------------------------------------------------------------
| Function    : sndcp_reset_xid_block
+------------------------------------------------------------------------------
| Description : Resets the given T_XID_BLOCK to default values.
|
| Parameters  : T_XID_BLOCK* xid_block
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_reset_xid_block (T_XID_BLOCK* xid_block)
{
  TRACE_FUNCTION( "sndcp_reset_xid_block" );

  /*
   * Reset all flags for optional compression field parameters.
   */
  xid_block->version_set = FALSE;
  xid_block->v42.is_set = FALSE;
  xid_block->vj.is_set = FALSE;

  xid_block->v42.nsapis_set = FALSE;
  xid_block->v42.p0_set = FALSE;
  xid_block->v42.p1_set = FALSE;
  xid_block->v42.p2_set = FALSE;
  xid_block->vj.nsapis_set = FALSE;
  xid_block->vj.s0_m_1_set = FALSE;
  /*
   * Set fields to default values.
   */
  xid_block->v42.nsapis = SNDCP_NSAPIS_DEFAULT;
  xid_block->v42.p0 = SNDCP_V42_DEFAULT_DIRECTION;
  xid_block->v42.p1 = SNDCP_V42_DEFAULT_P1;
  xid_block->v42.p2 = SNDCP_V42_DEFAULT_P2;

  xid_block->vj.nsapis = SNDCP_NSAPIS_DEFAULT;
  xid_block->vj.s0_m_1 = SNDCP_VJ_DEFAULT_S0_M_1;


} /* sndcp_reset_xid_block() */

#endif /* CF_FAST_EXEC */

#ifdef _SIMULATION_
/*
+------------------------------------------------------------------------------
| Function    : sndcp_sdu_to_desc_list
+------------------------------------------------------------------------------
| Description : copies the given sdu to the given desc_list
|
| Parameters  : T_sdu* sdu,
|               T_desc_list* desc_list
|
+------------------------------------------------------------------------------
*/
#ifdef _SNDCP_DTI_2_
GLOBAL void sndcp_sdu_to_desc_list(T_sdu* sdu, T_desc_list2* desc_list) {
  T_desc2           *desc = NULL;
  T_desc2           *last_desc = NULL;
#else /*_SNDCP_DTI_2_*/
GLOBAL void sndcp_sdu_to_desc_list(T_sdu* sdu, T_desc_list* desc_list) {
  T_desc            *desc = NULL;
  T_desc            *last_desc = NULL;
#endif /*_SNDCP_DTI_2_*/
  USHORT               sdu_index;
  USHORT               length;


  /*
   * Begin at the first relevant octet.
   */
  sdu_index = sdu->o_buf/8;

  /*
   * Initialise descriptor list length.
   */
  desc_list->list_len = 0;


  /*
   * Copy complete SDU to descriptor list using descriptors of max. 10 bytes.
   */
  while (sdu_index < sdu->l_buf / 8)
  {
    /*
     * Calculate length of descriptor data (= length of remaining sdu buffer
     * with a maximum of 10)
     */
    length = (sdu_index + 10 < sdu->l_buf / 8) ?
                      10 : (sdu->l_buf / 8 - sdu_index);

    /*
     * Allocate the necessary size for the data descriptor. The size is
     * calculated as follows:
     * - take the size of a descriptor structure
     * - subtract one because of the array buffer[1] to get the size of
     *   descriptor control information
     * - add number of octets of descriptor data
     */
#ifdef _SNDCP_DTI_2_
    MALLOC (desc, (USHORT)(sizeof(T_desc2) - 1 + length));
#else /*_SNDCP_DTI_2_*/
    MALLOC (desc, (USHORT)(sizeof(T_desc) - 1 + length));
#endif /*_SNDCP_DTI_2_*/
    /*
     * Fill descriptor control information.
     */
    desc->next  = (ULONG)NULL;
    desc->len   = length;
#ifdef _SNDCP_DTI_2_
    desc->offset = 0;
    desc->size = desc->len;
#endif
    /*
     * Add length of descriptor data to list length.
     */
    desc_list->list_len += length;

    /*
     * Copy user data from SDU to descriptor.
     */
    if (length>0)
    {
      memcpy (desc->buffer, &sdu->buf[sdu_index], length);
    }
    sdu_index += length;

    if (last_desc)
    {
      /*
       * Add this descriptor (not the first) to the descriptor list.
       */
      last_desc->next = (ULONG)desc;
    }
    else
    {
      /*
       * Insert first descriptor in descriptor list.
       */
      desc_list->first     = (ULONG)desc;
    }

    /*
     * Store this descriptor for later use.
     */
    last_desc = desc;
  }

 }
#endif /* _SIMULATION_ */



/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_ack
+------------------------------------------------------------------------------
| Description : sets nsapi_ack_ra[nsapi] to ack
|
| Parameters  : UBYTE nsapi, BOOL ack
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_nsapi_ack(UBYTE nsapi,
                                 BOOL ack)
{
  TRACE_FUNCTION(" sndcp_set_nsapi_ack ");
  {
    sndcp_data->nsapi_ack_ra[nsapi] = ack;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_direction
+------------------------------------------------------------------------------
| Description : sets nsapi_direction_ra[nsapi] to direction
|
| Parameters  : UBYTE nsapi, BOOL direction
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_nsapi_direction(UBYTE nsapi,
                                      BOOL direction)
{
  TRACE_FUNCTION(" sndcp_set_nsapi_direction ");
#ifdef _SNDCP_DTI_2_
  if (direction == DTI_CHANNEL_TO_LOWER_LAYER) {
    sndcp_data->nsapi_direction_ra[nsapi] = DTI_CHANNEL_TO_LOWER_LAYER;
  } else {
    sndcp_data->nsapi_direction_ra[nsapi] = DTI_CHANNEL_TO_HIGHER_LAYER;
#else /*_SNDCP_DTI_2_*/
  if (direction == SN_HOME) {
    sndcp_data->nsapi_direction_ra[nsapi] = HOME;
  } else {
    sndcp_data->nsapi_direction_ra[nsapi] = NEIGHBOR;
#endif /*_SNDCP_DTI_2_*/
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_interface
+------------------------------------------------------------------------------
| Description : sets nsapi_interface_ra[nsapi] to interfac
|
| Parameters  : UBYTE nsapi, U8 interfac
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_nsapi_interface(UBYTE nsapi,
                                      U8 interfac)
{
  TRACE_FUNCTION(" sndcp_set_nsapi_interface ");
  {
    sndcp_data->nsapi_interface_ra[nsapi] = interfac;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_linkid
+------------------------------------------------------------------------------
| Description : sets nsapi_linkid_ra[nsapi] to linkid
|
| Parameters  : UBYTE nsapi, U32 linkid
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_nsapi_linkid(UBYTE nsapi,
                                   U32 linkid)
{
  TRACE_FUNCTION(" sndcp_set_nsapi_linkid ");
  {
    sndcp_data->nsapi_linkid_ra[nsapi] = linkid;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_neighbor
+------------------------------------------------------------------------------
| Description : sets nsapi_neighbor_ra[nsapi] to neighbor
|
| Parameters  : UBYTE nsapi, U8* neighbor
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_nsapi_neighbor(UBYTE nsapi,
                                     U8* neighbor)
{
  TRACE_FUNCTION(" sndcp_set_nsapi_neighbor ");
  {
    sndcp_data->nsapi_neighbor_ra[nsapi] = neighbor;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_no_xid
+------------------------------------------------------------------------------
| Description : XID negotiation during each context (default in our
|               implementation) is switched off.
|
| Parameters  :
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_no_xid(void)
{
  TRACE_FUNCTION("sndcp_no_xid");
  {
    sndcp_data->always_xid = FALSE;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_prio
+------------------------------------------------------------------------------
| Description : sets nsapi_prio_ra[nsapi] to prio - 1:
|               SNSM prio val is mapped to LL prio val, which is the internal
|               representation.
|
| Parameters  : UBYTE nsapi, UBYTE prio
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_nsapi_prio(UBYTE nsapi,
                                 UBYTE prio)
{
  TRACE_FUNCTION("sndcp_set_nsapi_prio");
  {
    sndcp_data->nsapi_prio_ra[nsapi] = prio - 1;
  }
}

#endif /* CF_FAST_EXEC */

#ifdef REL99
/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_pktflowid
+------------------------------------------------------------------------------
| Description : sets pktflowid_ra[nsapi] to packet_flow_identifier:
|               SNSM packet flow identifier is mapped to LL pkt_flow_id, which is the internal
|               representation.
|
| Parameters  : UBYTE nsapi, UBYTE packet_flow_identifier
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_nsapi_pktflowid(UBYTE nsapi,
                                 U16 packet_flow_identifier)
{
  TRACE_FUNCTION("sndcp_set_nsapi_pktflowid");
  {
    sndcp_data->nsapi_pktflowid_ra[nsapi] = packet_flow_identifier;
  }
}
#endif /* CF_FAST_EXEC */

#endif /*REL99*/

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_qos
+------------------------------------------------------------------------------
| Description : sets nsapi_qos_ra[nsapi] to qos
|
| Parameters  : UBYTE nsapi, T_snsm_qos qos
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_nsapi_qos(UBYTE nsapi,
                                T_snsm_qos qos)
{
  TRACE_FUNCTION("sndcp_set_nsapi_qos");
  {
    sndcp_data->nsapi_qos_ra[nsapi].delay = qos.delay;
    sndcp_data->nsapi_qos_ra[nsapi].relclass = qos.relclass;
    sndcp_data->nsapi_qos_ra[nsapi].peak = qos.peak;
    sndcp_data->nsapi_qos_ra[nsapi].preced = qos.preced;
    sndcp_data->nsapi_qos_ra[nsapi].mean = qos.mean;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_sapi
+------------------------------------------------------------------------------
| Description : sets nsapi_sapi_ra[nsapi] to sapi
|
| Parameters  : UBYTE nsapi, UBYTE sapi
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_nsapi_sapi(UBYTE nsapi,
                                 UBYTE sapi)
{
  TRACE_FUNCTION("sndcp_set_nsapi_sapi");
  {
    sndcp_data->nsapi_sapi_ra[nsapi] = sapi;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_state
+------------------------------------------------------------------------------
| Description : adds stat to nsapi_state_ra[nsapi]
|
| Parameters  : UBYTE nsapi, UBYTE stat
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC
GLOBAL void sndcp_set_nsapi_state(UBYTE nsapi,
                                 USHORT stat)
{
  TRACE_FUNCTION("sndcp_set_nsapi_state");
  {
    sndcp_data->nsapi_state_ra[nsapi] |= stat;
  }
}

#endif /* CF_FAST_EXEC */


/*
+------------------------------------------------------------------------------
| Function    : sndcp_ip_packet_filter
+------------------------------------------------------------------------------
| Description : The function filters TCP and UDP packets on certain ports
|
| Parameters  : T_desc_list2 *desc_list, BOOL *discard
|
+------------------------------------------------------------------------------
*/
#ifdef _SNDCP_DTI_2_
LOCAL void sndcp_ip_packet_filter(T_desc_list2 *desc_list, BOOL *discard)
{
  T_desc2 *desc = (T_desc2 *)desc_list->first;
  UBYTE *pack_hdr = &desc->buffer[desc->offset];
  UBYTE *tcp_hdr = NULL, 
        *udp_hdr = NULL;
  USHORT offset = 0;
  USHORT src_port   = 0;
  USHORT dest_port  = 0;

  TRACE_FUNCTION("sndcp_ip_packet_filter");

  if(desc_list == NULL || desc_list->first == NULL || desc_list->list_len == 0){
    *discard = TRUE; /* discard packet */
    return; 
  }

  offset = (pack_hdr[0] & 0x0f) * 4; /* IP header length in bytes*/
  /* Verify, that the desc has a proper length */
  if(desc->len < (offset + 4)){
    return;
  }

  switch(pack_hdr[9]){
    /* ICMP Protocol */
    case 1:
      *discard = FALSE; /* don't discard ICMP packets */
      /* TRACE_EVENT("INFO IP FILTER: ICMP packet is sent"); */
     return;
    /* TCP Protocol */
    case 6:
      tcp_hdr = &pack_hdr[offset];
      src_port  |= tcp_hdr[0] << 8;
      src_port  |= tcp_hdr[1];
      dest_port |= tcp_hdr[2] << 8;
      dest_port |= tcp_hdr[3];
      if(src_port  == 42   || /* Host Name Server */
         dest_port == 42   || 
         src_port  == 111  || /* Remote Procedure Call */
         dest_port == 111  || 
         ((src_port  > 134) && (src_port  < 140))  || /* NetBIOS */
         ((dest_port > 134) && (dest_port < 140))  || /* NetBIOS */
         src_port  == 445   || /* Microsoft-DS */
         dest_port == 445   || 
         src_port  == 1025  || /* network blackjack */
         dest_port == 1025  || 
         src_port  == 1196  || /* listener RFS */
         dest_port == 1196  || 
         src_port  == 1433  || /* Microsoft-SQL-Server */
         dest_port == 1433  || 
         src_port  == 1434  || /* Microsoft-SQL-Monitor */
         dest_port == 1434   
       ){
        *discard = TRUE; /* discard this TCP packet */
        /*TRACE_EVENT_P2("INFO IP FILTER: TCP packet on dest_port %d, src_port %d is discarded",
                                                                   dest_port, src_port);*/
      } else {                                                             
        *discard = FALSE; /* dont't discard this TCP packet */
        /*TRACE_EVENT_P2("INFO IP FILTER: TCP packet on dest_port %d, src_port %d is sent",
                                                                   dest_port, src_port);*/
      }
     return;
    /* UDP Protocol */
    case 17:
      udp_hdr = &pack_hdr[offset];
      src_port  |= udp_hdr[0] << 8;
      src_port  |= udp_hdr[1];
      dest_port |= udp_hdr[2] << 8;
      dest_port |= udp_hdr[3];
      if(src_port  == 42  || /* Host Name Server */
         dest_port == 42  || 
         src_port  == 67  || /* DHCP BOOTP Protocol Server */
         dest_port == 67  || 
         src_port  == 68  || /* DHCP BOOTP Protocol Client */
         dest_port == 68  || 
         src_port  == 69  || /* Trivial FTP */
         dest_port == 69  || 
         src_port  == 111  || /* Remote Procedure Call */
         dest_port == 111  || 
         src_port  == 135  || /* NetBIOS & co */
         dest_port == 135  || 
         src_port  == 137  || /* NetBIOS & co*/
         dest_port == 137  || 
         src_port  == 138  || /* NetBIOS & co*/
         dest_port == 138  || 
         src_port  == 139  || /* NetBIOS & co*/
         dest_port == 139  || 
         src_port  == 389  || /* LDAP */
         dest_port == 389  ||
         src_port  == 445  || /* Microsoft-DS */
         dest_port == 445  || 
         src_port  == 1023 || /* network blackjack & co */
         dest_port == 1023 || 
         src_port  == 1026 || /* network blackjack & co */
         dest_port == 1026 || 
         src_port  == 1027 || /* network blackjack & co */
         dest_port == 1027 || 
         src_port  == 1028 || /* network blackjack & co */
         dest_port == 1028 || 
         src_port  == 1029 || /* network blackjack & co */
         dest_port == 1029 
       ){
        *discard = TRUE; /* discard this UDP packet */
        /*TRACE_EVENT_P2("INFO IP FILTER: UDP packet on dest_port %d, src_port %d is discarded",
                                                                   dest_port, src_port);*/
      } else {                                                             
        *discard = FALSE; /* dont't discard this UDP packet */
        /*TRACE_EVENT_P2("INFO IP FILTER: UDP packet on dest_port %d, src_port %d is sent",
                                                                   dest_port, src_port);*/
    }
    return;
    default:
      *discard = TRUE; /* discard all other packets */
      TRACE_EVENT_P1("INFO IP FILTER: protocol type %d is discarded", pack_hdr[9]);
      return;
  }
}
#endif /* _SNDCP_DTI_2_ */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_sig_callback
+------------------------------------------------------------------------------
| Description : callback function for dti lib.
|
| Parameters  :
|
+------------------------------------------------------------------------------
*/
#ifdef _SNDCP_DTI_2_
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_sig_callback(U8 instance,
                               U8 interfac,
                               U8 channel,
                               U8 reason,
                               T_DTI2_DATA_IND *dti2_data_ind)
{
  T_SN_DATA_REQ* sn_data_req = NULL;
  T_SN_UNITDATA_REQ* sn_unitdata_req = NULL;
  BOOL ack = FALSE;

  TRACE_FUNCTION("sndcp_sig_callback");

  
  switch(reason)
  {
    case DTI_REASON_CONNECTION_OPENED:
      /*
       * set rx and tx state machine of the given interface to idle state
       */
      /*
       * The nsapi is given here with the parameter 'channel'.
       */
      nu_connection_state(channel, TRUE);
      break;
    case DTI_REASON_CONNECTION_CLOSED:
      /*
       * set the given interface to closed state
       */
      break;
    case DTI_REASON_DATA_RECEIVED:
      /*
       * process received data
       */
      /*
       * Trace p_id.
       */



#ifdef TI_PS_OP_ICUT_SNDCP

      /* PDP_TBR added SNDCP terminal loopback */
      if ( SNDCP_LOOPBACK == 1)
      {
        if (bufFull[channel] == FALSE)
        {
          TRACE_EVENT_P1("LOOPBACK len=%d", dti2_data_ind->desc_list2.list_len);

          dti_send_data(sndcp_data->hDTI,
                    instance, // U8 instance,
                    SNDCP_INTERFACE_UNACK, // U8 interface,
                    channel,// U8 channel,
                    dti2_data_ind // T_DTI2_DATA_IND *dti_data_ind)
                    );
        }
        else
        {
          saveDti2_data_ind[channel] = dti2_data_ind;
        }
        return;
      }
      


#endif /* TI_PS_OP_ICUT_SNDCP */





#ifdef SNDCP_TRACE_ALL
      switch (dti2_data_ind->parameters.p_id) {
      case DTI_PID_IP:
        TRACE_EVENT_P2("dti2_data_ind->parameters.p_id: %02x (%s)",
                       dti2_data_ind->parameters.p_id,
                       "DTI_PID_IP");
        break;
      case DTI_PID_CTCP:
        TRACE_EVENT_P2("dti2_data_ind->parameters.p_id: %02x (%s)",
                       dti2_data_ind->parameters.p_id,
                       "DTI_PID_CTCP");
        break;
      case DTI_PID_UTCP:
        TRACE_EVENT_P2("dti2_data_ind->parameters.p_id: %02x (%s)",
                       dti2_data_ind->parameters.p_id,
                       "DTI_PID_UTCP");
        break;
      case DTI_PID_UOS:
        TRACE_EVENT_P2("dti2_data_ind->parameters.p_id: %02x (%s)",
                       dti2_data_ind->parameters.p_id,
                       "DTI_PID_UOS");
        break;
      default:
        TRACE_EVENT_P2("dti2_data_ind->parameters.p_id: %02x (%s)",
                       dti2_data_ind->parameters.p_id,
                       "unknown");


      }
#endif /* SNDCP_TRACE_ALL */
      /*
       * Discard DTI primitive if the config prim "DISCARD" was received
       */
      sndcp_data->nu = &sndcp_data->nu_base[channel];
      if(sndcp_data->nu->discard){
        TRACE_EVENT_P1("INFO DISCARD: UL packet with length %d is discarded",
                                        dti2_data_ind->desc_list2.list_len );
        sndcp_data->nu->discarded_data += dti2_data_ind->desc_list2.list_len;
        PFREE_DESC2(dti2_data_ind);
        dti2_data_ind = NULL;
        break;
      }
      /*
       * If the IP filter is enabled (config prim "IP_FILTER_ON" was received), 
       * filter the packets.
       */
      if(sndcp_data->ip_filter){
        BOOL discard;
        /* Statistics Trace */
        TRACE_EVENT_P3("INFO NSAPI[%d]: sent bytes: %d, discarded bytes: %d",
          channel, sndcp_data->nu->sent_data, sndcp_data->nu->discarded_data);
        sndcp_ip_packet_filter(&dti2_data_ind->desc_list2, &discard);
        if(discard){
          sndcp_data->nu->discarded_data += dti2_data_ind->desc_list2.list_len;
          PFREE_DESC2(dti2_data_ind);
          dti2_data_ind = NULL;
          break;
        } else {
          sndcp_data->nu->sent_data += dti2_data_ind->desc_list2.list_len;
        }
      } 
      /*
       * Copy dti_data_ind to sn_[unit]data_ind and enter nu service the
       * old way.
       */
      sndcp_get_nsapi_ack(channel, &ack);
      if (ack) {
        /*
         * Allocate a "normal" SN_DATA_REQ primitive and copy the data
         * of the test primitive to that one
         */
        MALLOC(sn_data_req, sizeof(T_SN_DATA_REQ));
        sn_data_req->nsapi = channel;
        sn_data_req->p_id = dti2_data_ind->parameters.p_id;
        sn_data_req->desc_list2 = dti2_data_ind->desc_list2;

        /*
         * Free the received dti primitive.
         */
        PFREE (dti2_data_ind);
        dti2_data_ind = NULL;

        nu_sn_data_req(sn_data_req);

      } else {
        /*
         * Allocate a "normal" SN_UNITDATA_REQ primitive and copy the data
         * of the test primitive to that one
         */
        MALLOC(sn_unitdata_req, sizeof(T_SN_UNITDATA_REQ));

        sn_unitdata_req->nsapi = channel;
        sn_unitdata_req->p_id = dti2_data_ind->parameters.p_id;
        sn_unitdata_req->desc_list2 = dti2_data_ind->desc_list2;

        /*
         * Free the received dti 2 primitive.
         */
        PFREE (dti2_data_ind);
        dti2_data_ind = NULL;

        nu_sn_unitdata_req(sn_unitdata_req);

      }

      break;
    case DTI_REASON_TX_BUFFER_FULL:
      /*
       * set tx state machine of the given interface to TX_IDLE state
       */



#ifdef TI_PS_OP_ICUT_SNDCP

     /* PDP_TBR added SNDCP terminal loopback */
      if (SNDCP_LOOPBACK == 1)
      {
        dti_stop(sndcp_data->hDTI,
                    instance,
                    interfac,
                    channel);
        bufFull[channel] = TRUE;
        TRACE_EVENT("SNDCP BUFFER IS FULL");
      }
      
#endif /* TI_PS_OP_ICUT_SNDCP */



      /*
       * Will be ignored.
       */
      break;
    case DTI_REASON_TX_BUFFER_READY:
      /*
       * set tx state machine of the given interface to TX_READY state
       */




#ifdef TI_PS_OP_ICUT_SNDCP
    /* PDP_TBR added SNDCP terminal loopback */
      if ( SNDCP_LOOPBACK == 1)
      {
        dti_start(sndcp_data->hDTI,
                    instance,
                    interfac,
                    channel);
        bufFull[channel] = FALSE;
        TRACE_ERROR("SNDCP BUFFER IS READY");
        if (saveDti2_data_ind[channel] != NULL)
        {
            TRACE_EVENT_P1("LOOPBACK len=%d", dti2_data_ind->desc_list2.list_len);

            dti_send_data(sndcp_data->hDTI,
                      instance, // U8 instance,
                      SNDCP_INTERFACE_UNACK, // U8 interface,
                      channel,// U8 channel,
                      saveDti2_data_ind[channel] // T_DTI2_DATA_IND *dti_data_ind)
                      );
            saveDti2_data_ind[channel] = NULL;

        }
        break;
      }
#endif /* TI_PS_OP_ICUT_SNDCP */




      /*
       * Reaction will be like reaction to SN_GET[UNIT]DATA_REQ.
       */
      nd_dti_buffer_ready(channel);
      break;
    default:
      TRACE_ERROR("sndcp_sig_callback called with undefined reason");
  }
}

#endif /* CF_FAST_EXEC */


#else /*_SNDCP_DTI_2_*/

#ifndef CF_FAST_EXEC

GLOBAL void sndcp_sig_callback(U8 instance,
                               U8 interfac,
                               U8 channel,
                               U8 reason,
                               T_DTI_DATA_IND *dti_data_ind)
{
  TRACE_FUNCTION("sndcp_sig_callback");

  switch(reason)
  {
    case DTI_REASON_CONNECTION_OPENED:
      /*
       * set rx and tx state machine of the given interface to idle state
       */
      /*
       * The nsapi is given here with the parameter 'channel'.
       */
      nu_connection_state(channel, TRUE);
      break;
    case DTI_REASON_CONNECTION_CLOSED:
      /*
       * set the given interface to closed state
       */
      break;
    case DTI_REASON_DATA_RECEIVED:
      /*
       * process received data
       */
      /*
       * Trace p_id.
       */
#ifdef SNDCP_TRACE_ALL
      switch (dti_data_ind->p_id) {
      case DTI_PID_IP:
        TRACE_EVENT_P2("dti_data_ind->p_id: %02x (%s)",
                       dti_data_ind->p_id,
                       "DTI_PID_IP");
        break;
      case DTI_PID_CTCP:
        TRACE_EVENT_P2("dti_data_ind->p_id: %02x (%s)",
                       dti_data_ind->p_id,
                       "DTI_PID_CTCP");
        break;
      case DTI_PID_UTCP:
        TRACE_EVENT_P2("dti_data_ind->p_id: %02x (%s)",
                       dti_data_ind->p_id,
                       "DTI_PID_UTCP");
        break;
      case DTI_PID_FRAME:
        TRACE_EVENT_P2("dti_data_ind->p_id: %02x (%s)",
                       dti_data_ind->p_id,
                       "DTI_PID_FRAME");
        break;
      default:
        TRACE_EVENT_P2("dti_data_ind->p_id: %02x (%s)",
                       dti_data_ind->p_id,
                       "unknown");


      }
#endif /* SNDCP_TRACE_ALL */
      /*
       * Copy dti_data_ind to sn_[unit]data_ind and enter nu service the
       * old way.
       */
      if (interfac == SNDCP_INTERFACE_ACK) {
        /*
         * Allocate a "normal" SN_DATA_REQ primitive and copy the data
         * of the test primitive to that one
         */
        PALLOC_DESC (sn_data_req, SN_DATA_REQ);

        sn_data_req->nsapi = channel;
        sn_data_req->p_id = dti_data_ind->p_id;
        sn_data_req->desc_list = dti_data_ind->desc_list;

        /*
         * Free the received dti primitive.
         */
        PFREE (dti_data_ind);
        dti_data_ind = NULL;

        nu_sn_data_req(sn_data_req);

      } else {
        /*
         * Allocate a "normal" SN_UNITDATA_REQ primitive and copy the data
         * of the test primitive to that one
         */
        PALLOC_DESC (sn_unitdata_req, SN_UNITDATA_REQ);

        sn_unitdata_req->nsapi = channel;
        sn_unitdata_req->p_id = dti_data_ind->p_id;
        sn_unitdata_req->desc_list = dti_data_ind->desc_list;

        /*
         * Free the received test primitive.
         */
        PFREE (dti_data_ind);
        dti_data_ind = NULL;

        nu_sn_unitdata_req(sn_unitdata_req);

      }

      break;
    case DTI_REASON_TX_BUFFER_FULL:
      /*
       * set tx state machine of the given interface to TX_IDLE state
       */
      /*
       * Will be ignored.
       */
      break;
    case DTI_REASON_TX_BUFFER_READY:
      /*
       * set tx state machine of the given interface to TX_READY state
       */
      /*
       * Reaction will be like reaction to SN_GET[UNIT]DATA_REQ.
       */
      nd_dti_buffer_ready(channel);
      break;
    default:
      TRACE_ERROR("sndcp_sig_callback called with undefined reason");
  }
}
#endif /* CF_FAST_EXEC */

#endif /*_SNDCP_DTI_2_*/

/*
+------------------------------------------------------------------------------
| Function    : sndcp_unset_nsapi_state
+------------------------------------------------------------------------------
| Description : subtracts stat from nsapi_state_ra[nsapi]
|
| Parameters  : UBYTE nsapi, UBYTE stat
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_unset_nsapi_state(UBYTE nsapi,
                                 USHORT stat)
{
  TRACE_FUNCTION("sndcp_unset_nsapi_state");
  {
    sndcp_data->nsapi_state_ra[nsapi] &= ~ stat;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_used
+------------------------------------------------------------------------------
| Description : sets nsapi_used_ra[nsapi] to b
|
| Parameters  : UBYTE nsapi, BOOL b
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_nsapi_used(UBYTE nsapi,
                                 BOOL b)
{
  TRACE_FUNCTION("sndcp_set_nsapi_used");
  {
    sndcp_data->nsapi_used_ra[nsapi] = b;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_sapi_ack
+------------------------------------------------------------------------------
| Description : sets sapi_ack_ra[sapi] to ack
|
| Parameters  : UBYTE sapi, BOOL ack
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_sapi_ack(UBYTE sapi,
                                 BOOL ack)
{
  TRACE_FUNCTION(" sndcp_set_sapi_ack ");
  {
    UBYTE sapi_index = 0;

    sndcp_get_sapi_index(sapi, &sapi_index);
    sndcp_data->sapi_ack_ra[sapi_index] = ack;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_snsm_qos_to_ll_qos
+------------------------------------------------------------------------------
| Description : transfers an snsm_qos to an ll_qos
|
| Parameters  : T_snsm_qos, T_ll_qos*
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_snsm_qos_to_ll_qos(T_snsm_qos snsm_qos,
                                     T_ll_qos* ll_qos)
{
  TRACE_FUNCTION("sndcp_snsm_qos_to_ll_qos");
  {
    ll_qos->delay = snsm_qos.delay;
    ll_qos->relclass = snsm_qos.relclass;
    ll_qos->peak = snsm_qos.peak;
    ll_qos->preced = snsm_qos.preced;
    ll_qos->mean = snsm_qos.mean;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_sapi_state
+------------------------------------------------------------------------------
| Description : adds stat to sapi_state_ra[sapi] bitwise
|
| Parameters  : UBYTE sapi, UBYTE stat
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_set_sapi_state(UBYTE sapi,
                                 USHORT stat)
{
  TRACE_FUNCTION(" sndcp_set_sapi_state ");
  {
    UBYTE sapi_index = 0;
    sndcp_get_sapi_index(sapi, &sapi_index);
    sndcp_data->sapi_state_ra[sapi_index] |= stat;
  }
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_unset_sapi_state
+------------------------------------------------------------------------------
| Description : subtracts stat from sapi_state_ra[sapi] bitwise
|
| Parameters  : UBYTE sapi, UBYTE stat
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_unset_sapi_state(UBYTE sapi,
                                   USHORT stat)
{
  TRACE_FUNCTION(" sndcp_unset_sapi_state ");
  {
    UBYTE sapi_index = 0;
    sndcp_get_sapi_index(sapi, &sapi_index);
    sndcp_data->sapi_state_ra[sapi_index] &= ~ stat;
  }
}

#endif /* CF_FAST_EXEC */

#ifdef FLOW_TRACE
/*
+------------------------------------------------------------------------------
| Function    : sndcp_trace_flow_control
+------------------------------------------------------------------------------
| Description : traces flow control
|
| Parameters  : UBYTE entity, UBYTE uplink, UBYTE sent, BOOL close
|
+------------------------------------------------------------------------------
*/
/*
 * 0 GRR
 * 1 LLC
 * 2 SNDCP
 * 3 PPP
 * 4 UART
 */

/*
#define FLOW_TRACE_GRR    0
#define FLOW_TRACE_LLC    1
#define FLOW_TRACE_SNDCP  2
#define FLOW_TRACE_PPP    3
#define FLOW_TRACE_UART   4

direction of data transfer

#define FLOW_TRACE_UP     0
#define FLOW_TRACE_DOWN   1

is sap sitting on top of entity or is lower entity sap used?

#define FLOW_TRACE_TOP    0
#define FLOW_TRACE_BOTTOM 1
*/


#ifndef CF_FAST_EXEC

GLOBAL void sndcp_trace_flow_control(UBYTE entity, UBYTE transfer, UBYTE position, BOOL opened)
{
  TRACE_FUNCTION(" sndcp_trace_flow_control ");
#ifdef SNDCP_TRACE_ALL
  TRACE_EVENT_P4("sndcp_trace_flow_control[%d][%d][%d] = %d",
                 entity,
                 transfer,
                 position,
                 opened);
#endif
  {
    sndcp_data->flow_control_ra[entity][transfer][position] = opened;
  }
}

#endif /* CF_FAST_EXEC */

#endif /* FLOW_TRACE */


#ifdef _SNDCP_DTI_2_

#ifdef SNDCP_TRACE_ALL

/*
+------------------------------------------------------------------------------
| Function    : sndcp_trace_desc_list3_content
+------------------------------------------------------------------------------
| Description : traces content of a desc3 descriptor list
|
| Parameters  : desc_list3 descriptor list
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_trace_desc_list3_content(T_desc_list3 desc_list3)
{
  U16 current_pos = 0;  /*  */
  U16 data_len = 0;     /* The length of the data to be traced including data offset */
  U8* p_data = NULL;    /* Pointer to byte data element */
  T_desc3* p_desc3 = (T_desc3*)desc_list3.first; /* Pointer to the actual desc3 descriptor element */

  while(p_desc3 != NULL)
  {
    current_pos = p_desc3->offset; /* In case it is the sndcp header allocation, ENCODE_OFFSET has been considered*/
    p_data = (U8*)p_desc3->buffer;
    data_len = current_pos + p_desc3->len; /* p_desc3->len is the total allocation length for sndcp header description */
    while(current_pos <= data_len)
    {
      TRACE_EVENT_P1
        ("%02x",
          p_data[current_pos]
        );
      current_pos ++;
    }

    p_desc3 = (T_desc3*)p_desc3->next;
  }
}

#endif /* CF_FAST_EXEC */

#endif /* SNDCP_TRACE_ALL */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_cl_desc2_attach
+------------------------------------------------------------------------------
| Description : This function attaches a decriptor list to already allocated
|               memory. There is no return value. In case an attempt is made
|               to attach to freed or invalid memory, then the FRAME generates
|               a trace SYSTEM ERROR. This function does not allocate any new
|               memory.
|
| Parameters  : T_desc2* pointer to desc2 descriptor.
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

void sndcp_cl_desc2_attach(T_desc2* p_desc2)
{

  MATTACH(p_desc2);

}

#endif /* CF_FAST_EXEC */
/*
+------------------------------------------------------------------------------
| Function    : sndcp_cl_desc3_free
+------------------------------------------------------------------------------
| Description : Frees the descriptor connected to the desc3 descriptor. This
|               free will when applicable cause the frame to decrease a
|               connected counter attached to the allocation or really free
|               the memory in case the counter is zero.
|
| Parameters  : T_desc3* pointer to desc3 descriptor.
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

void sndcp_cl_desc3_free(T_desc3* p_desc3)
{

  MFREE(p_desc3->buffer);
  p_desc3->buffer = NULL;
}

#endif /* CF_FAST_EXEC */

#endif /* _SNDCP_DTI_2_ */


#ifdef SNDCP_TRACE_ALL

/*
+------------------------------------------------------------------------------
| Function    : sndcp_trace_sdu
+------------------------------------------------------------------------------
| Description : traces content of one sdu
|
| Parameters  : pointer to sdu
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_trace_sdu(T_sdu* sdu)
{

  USHORT pos = sdu->o_buf >> 3;
  USHORT  frame_len = (sdu->l_buf + 7) / 8;


  TRACE_FUNCTION("sndcp_trace_sdu");

  while(pos < (frame_len + (sdu->o_buf >> 3)))
  {
    if (pos + 8 <= (frame_len + (sdu->o_buf >> 3))) {
      TRACE_EVENT_P8
        ("%02x %02x %02x %02x %02x %02x %02x %02x",
          sdu->buf[pos],
          sdu->buf[pos + 1],
          sdu->buf[pos + 2],
          sdu->buf[pos + 3],
          sdu->buf[pos + 4],
          sdu->buf[pos + 5],
          sdu->buf[pos + 6],
          sdu->buf[pos + 7]
        );
      pos += 8;
    } else if (pos + 7 <= (frame_len + (sdu->o_buf >> 3))){
      TRACE_EVENT_P7
        ("%02x %02x %02x %02x %02x %02x %02x",
          sdu->buf[pos],
          sdu->buf[pos + 1],
          sdu->buf[pos + 2],
          sdu->buf[pos + 3],
          sdu->buf[pos + 4],
          sdu->buf[pos + 5],
          sdu->buf[pos + 6]
        );
      pos += 7;
    } else if (pos + 6 <= (frame_len + (sdu->o_buf >> 3))){
      TRACE_EVENT_P6
        ("%02x %02x %02x %02x %02x %02x",
          sdu->buf[pos],
          sdu->buf[pos + 1],
          sdu->buf[pos + 2],
          sdu->buf[pos + 3],
          sdu->buf[pos + 4],
          sdu->buf[pos + 5]
        );
      pos += 6;
    } else if (pos + 5 <= (frame_len + (sdu->o_buf >> 3))){
      TRACE_EVENT_P5
        ("%02x %02x %02x %02x %02x",
          sdu->buf[pos],
          sdu->buf[pos + 1],
          sdu->buf[pos + 2],
          sdu->buf[pos + 3],
          sdu->buf[pos + 4]
        );
      pos += 5;
    } else if (pos + 4 <= (frame_len + (sdu->o_buf >> 3))){
      TRACE_EVENT_P4
        ("%02x %02x %02x %02x",
          sdu->buf[pos],
          sdu->buf[pos + 1],
          sdu->buf[pos + 2],
          sdu->buf[pos + 3]
        );
      pos += 4;
    } else if (pos + 3 <= (frame_len + (sdu->o_buf >> 3))){
      TRACE_EVENT_P3
        ("%02x %02x %02x",
          sdu->buf[pos],
          sdu->buf[pos + 1],
          sdu->buf[pos + 2]
        );
      pos += 3;
    } else if (pos + 2 <= (frame_len + (sdu->o_buf >> 3))){
      TRACE_EVENT_P2
        ("%02x %02x",
          sdu->buf[pos],
          sdu->buf[pos + 1]
        );
      pos += 2;
    } else if (pos + 1 <= (frame_len + (sdu->o_buf >> 3))){
      TRACE_EVENT_P1
        ("%02x",
          sdu->buf[pos]
        );
      pos++;
    }

  }
}

#endif /* CF_FAST_EXEC */


#endif /* SNDCP_TRACE_ALL */

#ifdef  SNDCP_TRACE_BUFFER

/*
+------------------------------------------------------------------------------
| Function    : sndcp_trace_desc_list
+------------------------------------------------------------------------------
| Description : traces content of one desc_list
|
| Parameters  : pointer to desc_list
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

#ifdef _SNDCP_DTI_2_
GLOBAL void sndcp_trace_desc_list(T_desc_list2* desc_list)
{
#else /*_SNDCP_DTI_2_*/
GLOBAL void sndcp_trace_desc_list(T_desc_list* desc_list)
{
#endif /*_SNDCP_DTI_2_*/
  USHORT  frame_len = desc_list->list_len;
#ifdef _SNDCP_DTI_2_
  T_desc2* desc = (T_desc2*)desc_list->first;
  USHORT  desc_pos  = desc->offset;
  USHORT  desc_end  = desc->len + desc->offset;
#else /*_SNDCP_DTI_2_*/
  T_desc* desc = (T_desc*)desc_list->first;
  USHORT  desc_pos = 0;
  USHORT  desc_end  = desc->len;
#endif /*_SNDCP_DTI_2_*/
  USHORT  list_pos = 0;

  TRACE_FUNCTION("sndcp_trace_desc_list");

  while(list_pos < frame_len)
  {
    if (desc != NULL) {
      if (desc_pos >= desc_end) {
#ifdef _SNDCP_DTI_2_
        desc = (T_desc2*)desc->next;
        desc_pos = desc->offset;
        desc_end = desc->len + desc->offset;
#else /*_SNDCP_DTI_2_*/
        desc = (T_desc*)desc->next;
        desc_pos = 0;
        desc_end = desc->len;
#endif /*_SNDCP_DTI_2_*/
      }
    }
    if (desc == NULL) {
      return;
    }
    if (desc_pos + 8 <= desc_end) {
      TRACE_EVENT_P8 ("%02x %02x %02x %02x %02x %02x %02x %02x ",
                      desc->buffer[desc_pos],
                      desc->buffer[desc_pos + 1],
                      desc->buffer[desc_pos + 2],
                      desc->buffer[desc_pos + 3],
                      desc->buffer[desc_pos + 4],
                      desc->buffer[desc_pos + 5],
                      desc->buffer[desc_pos + 6],
                      desc->buffer[desc_pos + 7]
                     );
      list_pos+= 8;
      desc_pos+= 8;
    } else if (desc_pos + 7 <= desc_end) {
      TRACE_EVENT_P7 ("%02x %02x %02x %02x %02x %02x %02x ",
                      desc->buffer[desc_pos],
                      desc->buffer[desc_pos + 1],
                      desc->buffer[desc_pos + 2],
                      desc->buffer[desc_pos + 3],
                      desc->buffer[desc_pos + 4],
                      desc->buffer[desc_pos + 5],
                      desc->buffer[desc_pos + 6]
                     );
      list_pos+= 7;
      desc_pos+= 7;
    } else if (desc_pos + 6 <= desc_end) {
      TRACE_EVENT_P6 ("%02x %02x %02x %02x %02x %02x ",
                      desc->buffer[desc_pos],
                      desc->buffer[desc_pos + 1],
                      desc->buffer[desc_pos + 2],
                      desc->buffer[desc_pos + 3],
                      desc->buffer[desc_pos + 4],
                      desc->buffer[desc_pos + 5]
                     );
      list_pos+= 6;
      desc_pos+= 6;
    } else if (desc_pos + 5 <= desc_end) {
      TRACE_EVENT_P5 ("%02x %02x %02x %02x %02x ",
                      desc->buffer[desc_pos],
                      desc->buffer[desc_pos + 1],
                      desc->buffer[desc_pos + 2],
                      desc->buffer[desc_pos + 3],
                      desc->buffer[desc_pos + 4]
                     );
      list_pos+= 5;
      desc_pos+= 5;
    } else if (desc_pos + 4 <= desc_end) {
      TRACE_EVENT_P4 ("%02x %02x %02x %02x ",
                      desc->buffer[desc_pos],
                      desc->buffer[desc_pos + 1],
                      desc->buffer[desc_pos + 2],
                      desc->buffer[desc_pos + 3]
                     );
      list_pos+= 4;
      desc_pos+= 4;
    } else if (desc_pos + 3 <= desc_end) {
      TRACE_EVENT_P3 ("%02x %02x %02x ",
                      desc->buffer[desc_pos],
                      desc->buffer[desc_pos + 1],
                      desc->buffer[desc_pos + 2]
                     );
      list_pos+= 3;
      desc_pos+= 3;
    } else if (desc_pos + 2 <= desc_end) {
      TRACE_EVENT_P2 ("%02x %02x ",
                      desc->buffer[desc_pos],
                      desc->buffer[desc_pos + 1]
                     );
      list_pos+= 2;
      desc_pos+= 2;
    } else if (desc_pos + 1 <= desc_end) {
      TRACE_EVENT_P1 ("%02x ",
                      desc->buffer[desc_pos]
                     );
      list_pos++;
      desc_pos++;
    }

    /* break tracing if payload shall not be traced*/
    if(!sndcp_data->trace_ip_datagram && (list_pos >= 40))
      break;

  } /* while(list_pos < frame_len) */

}

#endif /* CF_FAST_EXEC */

#endif /* SNDCP_TRACE_BUFFER */



#ifdef _SNDCP_MEAN_TRACE_
/*
+------------------------------------------------------------------------------
| Function    : sndcp_sn_count_req
+------------------------------------------------------------------------------
| Description : reaction to prim SN_COUNT_REQ
|
| Parameters  : primitive payload
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL void sndcp_sn_count_req(T_SN_COUNT_REQ* sn_count_req)
{
  TRACE_FUNCTION("sndcp_sn_count_req");

  {
    BOOL ack = FALSE;
    PALLOC(sn_count_cnf, SN_COUNT_CNF);

    sn_count_cnf->nsapi = sn_count_req->nsapi;

    sndcp_get_nsapi_ack(sn_count_req->nsapi, &ack);
    if (ack) {
      sn_count_cnf->octets_uplink =
        sndcp_data->cur_num_uplink_ack[sn_count_req->nsapi];
      sn_count_cnf->octets_downlink =
        sndcp_data->cur_num_downlink_ack[sn_count_req->nsapi];

      sn_count_cnf->packets_uplink =
        sndcp_data->cur_pac_uplink_ack[sn_count_req->nsapi];
      sn_count_cnf->packets_downlink =
        sndcp_data->cur_pac_downlink_ack[sn_count_req->nsapi];

#ifdef SNDCP_UPM_INCLUDED 
      if (sn_count_req->reset == NAS_RESET_YES) {
#else 
      if (sn_count_req->reset == SN_RESET_YES) { 
#endif   /*SNDCP_UPM_INCLUDED*/
        sndcp_data->cur_num_uplink_ack[sn_count_req->nsapi] = 0;
        sndcp_data->cur_num_downlink_ack[sn_count_req->nsapi] = 0;
        sndcp_data->cur_pac_uplink_ack[sn_count_req->nsapi] = 0;
        sndcp_data->cur_pac_downlink_ack[sn_count_req->nsapi] = 0;
      }

    } else {

      sn_count_cnf->octets_uplink =
        sndcp_data->cur_num_uplink_unack[sn_count_req->nsapi];
      sn_count_cnf->octets_downlink =
        sndcp_data->cur_num_downlink_unack[sn_count_req->nsapi];

      sn_count_cnf->packets_uplink =
        sndcp_data->cur_pac_uplink_unack[sn_count_req->nsapi];
      sn_count_cnf->packets_downlink =
        sndcp_data->cur_pac_downlink_unack[sn_count_req->nsapi];

#ifdef SNDCP_UPM_INCLUDED 
      if (sn_count_req->reset == NAS_RESET_YES) {
#else 
      if (sn_count_req->reset == SN_RESET_YES) {
#endif  /*SNDCP_UPM_INCLUDED*/
        sndcp_data->cur_num_uplink_unack[sn_count_req->nsapi] = 0;
        sndcp_data->cur_num_downlink_unack[sn_count_req->nsapi] = 0;
        sndcp_data->cur_pac_uplink_unack[sn_count_req->nsapi] = 0;
        sndcp_data->cur_pac_downlink_unack[sn_count_req->nsapi] = 0;
      }
    }

#ifdef SNDCP_UPM_INCLUDED 
	PSEND(hCommUPM, sn_count_cnf);  /*count_cnf goes via UPM*/
#else
    PSEND(hCommMMI, sn_count_cnf);
#endif  /*SNDCP_UPM_INCLUDED*/

  }

  PFREE(sn_count_req);
}

#endif /* CF_FAST_EXEC */

#endif /*_SNDCP_MEAN_TRACE */

#ifdef SNDCP_TRACE_IP_DATAGRAM

/*
+------------------------------------------------------------------------------
| Function    : sndcp_trace_ip_datagram
+------------------------------------------------------------------------------
| Description : traces content of one desc_list
|
| Parameters  : pointer to desc_list
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

#ifdef _SNDCP_DTI_2_
GLOBAL void sndcp_trace_ip_datagram(T_desc_list2* desc_list)
{
#else /*_SNDCP_DTI_2_*/
GLOBAL void sndcp_trace_ip_datagram(T_desc_list* desc_list)
{
#endif /*_SNDCP_DTI_2_*/
  USHORT  frame_len = desc_list->list_len;
#ifdef _SNDCP_DTI_2_
  T_desc2* desc = (T_desc2*)desc_list->first;
  USHORT  desc_pos = desc->offset;
  USHORT  desc_end  = desc->len + desc->offset;
#else /*_SNDCP_DTI_2_*/
  T_desc* desc = (T_desc*)desc_list->first;
  USHORT  desc_pos = 0;
  USHORT  desc_end  = desc->len;
#endif /*_SNDCP_DTI_2_*/
  USHORT  list_pos = 0;
  UBYTE   *pack_hdr, *ip_hdr;
  USHORT  t_length   = 0;
  USHORT  length     = 0;
  USHORT  pid        = 0;
  USHORT  f_offset   = 0;
  USHORT  h_checksum = 0;
  USHORT  src_port   = 0;
  USHORT  dest_port  = 0;
  ULONG   seq_num    = 0;
  ULONG   ack_num    = 0;
  USHORT  window     = 0;
  USHORT  checksum   = 0;
  USHORT  urg_ptr    = 0;
  USHORT  ip_hlen    = 0; /* IP header length */
  USHORT  tcp_hlen   = 0; /* TCP header length */
  UBYTE   tmp_flag   = 0;
  UBYTE   type       = 0;
  UBYTE   code       = 0;
  /*TRACE_FUNCTION("sndcp_trace_ip_datagram");*/

  if(!sndcp_data->trace_ip_datagram && !sndcp_data->trace_ip_header){
    return;
  }

  while(list_pos < frame_len)
  {
    if (desc != NULL) {
      if (desc_pos >= desc_end) {
#ifdef _SNDCP_DTI_2_
        desc = (T_desc2*)desc->next;
        desc_pos = desc->offset;
        desc_end = desc->len + desc->offset;
#else /*_SNDCP_DTI_2_*/
        desc = (T_desc*)desc->next;
        desc_pos = 0;
        desc_end = desc->len;
#endif /*_SNDCP_DTI_2_*/
        if (desc == NULL) {
          break;
        }
      }
    } else {
      return;
    }
    if((list_pos == 0) || (list_pos >= t_length))
    {
      ip_hdr = &desc->buffer[desc_pos];
      if((ip_hdr[9] != 1) && (ip_hdr[9] != 6) && (ip_hdr[9] != 17)) {
        TRACE_EVENT_P1("INFO TRACE: Tracing of protocol type %d not supported",
                                                                    ip_hdr[9]);
        return;
      }

      /* Trace IP Header */
      ip_hlen = (ip_hdr[0] & 0x0F) << 2;
      if ((desc_pos + ip_hlen) <= desc_end){
        TRACE_EVENT("IP Header");
        TRACE_EVENT_P1("  Protocol Version:         %d",
                      ((ip_hdr[0] & 0xF0) >> 4));
        TRACE_EVENT_P1("  Header Length:            %d", (ip_hlen >> 2));
        TRACE_EVENT_P1("  Type Of Service:          %d", ip_hdr[1]);
        t_length |= ip_hdr[2] << 8;
        t_length |= ip_hdr[3];
        TRACE_EVENT_P1("  Total Length:             %d", t_length);
        pid |= ip_hdr[4] << 8;
        pid |= ip_hdr[5];
        TRACE_EVENT_P1("  Packet ID:                %d", pid);
        TRACE_EVENT_P2("  MF|DF:                    %d|%d",
                            (ip_hdr[6] & 0x20) >> 5,
                            (ip_hdr[6] & 0x40) >> 6);
        f_offset |= (ip_hdr[6] & 0x1F) << 8;
        f_offset |= ip_hdr[7];
        TRACE_EVENT_P1("  Fragment offset:          %d", f_offset);
        TRACE_EVENT_P1("  Time To Live:             %d", ip_hdr[8]);
        if(ip_hdr[9] == 17) {
          TRACE_EVENT_P1("  Protocol:                 UDP(%d)",
                                             ip_hdr[9]);
        } else if (ip_hdr[9] == 6) {
          TRACE_EVENT_P1("  Protocol:                 TCP(%d)",
                                             ip_hdr[9]);
        } else {
          TRACE_EVENT_P1("  Protocol:                 %d",
                                             ip_hdr[9]);
        }
        h_checksum |= ip_hdr[10] << 8;
        h_checksum |= ip_hdr[11];
        TRACE_EVENT_P1("  Header Checksum:          %d", h_checksum);
        TRACE_EVENT_P4("  Source Address:           %d.%d.%d.%d",
                          ip_hdr[12], ip_hdr[13],
                          ip_hdr[14], ip_hdr[15]);
        TRACE_EVENT_P4("  Destination Address:      %d.%d.%d.%d",
                          ip_hdr[16], ip_hdr[17],
                          ip_hdr[18], ip_hdr[19]);
        list_pos+= ip_hlen;
        desc_pos+= ip_hlen;
      }
      /* Trace TCP Header */
      if((ip_hdr[9] == 6) && ((desc_pos + 20) <= desc_end))
      {
        TRACE_EVENT("TCP Header");
        pack_hdr = &ip_hdr[ip_hlen];
        tcp_hlen = (pack_hdr[12] & 0xF0) >> 2; /* the same as (x>>4)*4 */
        src_port |= pack_hdr[0] << 8;
        src_port |= pack_hdr[1];
        TRACE_EVENT_P1("  Source Port:              %d", src_port);
        dest_port |= pack_hdr[2] << 8;
        dest_port |= pack_hdr[3];
        TRACE_EVENT_P1("  Destination Port:         %d", dest_port);
        seq_num |= pack_hdr[4] << 24;
        seq_num |= pack_hdr[5] << 16;
        seq_num |= pack_hdr[6] << 8;
        seq_num |= pack_hdr[7];
        TRACE_EVENT_P1("  Sequence Number:          %u", seq_num);
        ack_num |= pack_hdr[8] << 24;
        ack_num |= pack_hdr[9] << 16;
        ack_num |= pack_hdr[10] << 8;
        ack_num |= pack_hdr[11];
        TRACE_EVENT_P1("  Acknowledgment Number:    %u", ack_num);
        TRACE_EVENT_P1("  Data Offset:              %d", (tcp_hlen >> 2));
        TRACE_EVENT_P6("  URG|ACK|PSH|RST|SYN|FIN:  %d|%d|%d|%d|%d|%d",
                                       (pack_hdr[13] & 0x20) >> 5,
                                       (pack_hdr[13] & 0x10) >> 4,
                                       (pack_hdr[13] & 0x08) >> 3,
                                       (pack_hdr[13] & 0x04) >> 2,
                                       (pack_hdr[13] & 0x02) >> 1,
                                       (pack_hdr[13] & 0x01)    );
        window |= pack_hdr[14] << 8;
        window |= pack_hdr[15];
        TRACE_EVENT_P1("  Window Size:              %d", window);
        checksum |= pack_hdr[16] << 8;
        checksum |= pack_hdr[17];
        TRACE_EVENT_P1("  Checksum:                 %d", checksum);
        urg_ptr |= pack_hdr[18] << 8;
        urg_ptr |= pack_hdr[19];
        TRACE_EVENT_P1("  Ungent Pointer:           0x%04x", urg_ptr);
        list_pos+= tcp_hlen;
        desc_pos+= tcp_hlen;
      }
      /* Trace UDP Header */
      else if((ip_hdr[9] == 17) && ((desc_pos + 8) <= desc_end))
      {
        TRACE_EVENT("UDP Header");
        pack_hdr = &ip_hdr[ip_hlen];
        src_port |= pack_hdr[0] << 8;
        src_port |= pack_hdr[1];
        TRACE_EVENT_P1("  Source Port:              %d", src_port);
        dest_port |= pack_hdr[2] << 8;
        dest_port |= pack_hdr[3];
        TRACE_EVENT_P1("  Destination Port:         %d", dest_port);
        length |= pack_hdr[4] << 8;
        length |= pack_hdr[5];
        TRACE_EVENT_P1("  Length:                   %d", length);
        checksum |= pack_hdr[6] << 8;
        checksum |= pack_hdr[7];
        TRACE_EVENT_P1("  Checksum:                 %d", checksum);
        list_pos+= 8;
        desc_pos+= 8;
      }
      /* Trace ICMP Header */
      else if((ip_hdr[9] == 1) && ((desc_pos + 8) <= desc_end))
      {
        TRACE_EVENT("ICMP Header");
        pack_hdr = &ip_hdr[ip_hlen];
        type = pack_hdr[0];
        TRACE_EVENT_P1("  Message Type:             %d", type);
        code = pack_hdr[1];
        TRACE_EVENT_P1("  Message Code:             %d", code);
        checksum |= pack_hdr[2] << 8;
        checksum |= pack_hdr[3];
        TRACE_EVENT_P1("  Checksum:                 %d", checksum);
        list_pos+= 8;
        desc_pos+= 8;
      }
    }/* if((list_pos == 0) || (list_pos >= t_length)) */
    /* Shall payload be traced? */
    if(!sndcp_data->trace_ip_datagram){
      break;
    }
  /* Trace Payload */
    if(!tmp_flag && (desc_pos < frame_len)){
      TRACE_EVENT("Payload");
      tmp_flag = 1;
    }
    /*lint -e661 -e662 possible access/creation of out-of-bounds pointer*/	
    if (desc_pos + 16 <= desc_end) {
      TRACE_EVENT_P16 ("  %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x ",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1], desc->buffer[desc_pos+2], 
        desc->buffer[desc_pos+3], desc->buffer[desc_pos+4], desc->buffer[desc_pos+5], 
        desc->buffer[desc_pos+6], desc->buffer[desc_pos+7], desc->buffer[desc_pos+8], 
        desc->buffer[desc_pos+9], desc->buffer[desc_pos+10], desc->buffer[desc_pos+11], 
        desc->buffer[desc_pos+12], desc->buffer[desc_pos+13], desc->buffer[desc_pos+14],
        desc->buffer[desc_pos+15]);
      list_pos+= 16;
      desc_pos+= 16;
    } else if (desc_pos + 14 <= desc_end) {
      TRACE_EVENT_P14 ("  %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1], desc->buffer[desc_pos+2], 
        desc->buffer[desc_pos+3], desc->buffer[desc_pos+4], desc->buffer[desc_pos+5], 
        desc->buffer[desc_pos+6], desc->buffer[desc_pos+7], desc->buffer[desc_pos+8], 
        desc->buffer[desc_pos+9], desc->buffer[desc_pos+10], desc->buffer[desc_pos+11], 
        desc->buffer[desc_pos+12], desc->buffer[desc_pos+13]);
      list_pos+= 14;
      desc_pos+= 14;
    } else if (desc_pos + 12 <= desc_end) {
      TRACE_EVENT_P12("  %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x ",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1], desc->buffer[desc_pos+2], 
        desc->buffer[desc_pos+3], desc->buffer[desc_pos+4], desc->buffer[desc_pos+5], 
        desc->buffer[desc_pos+6], desc->buffer[desc_pos+7], desc->buffer[desc_pos+8], 
        desc->buffer[desc_pos+9], desc->buffer[desc_pos+10], desc->buffer[desc_pos+11]);
      list_pos+= 12;
      desc_pos+= 12;
    } else if (desc_pos + 10 <= desc_end) {
      TRACE_EVENT_P10("  %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1], desc->buffer[desc_pos+2], 
        desc->buffer[desc_pos+3], desc->buffer[desc_pos+4], desc->buffer[desc_pos+5], 
        desc->buffer[desc_pos+6], desc->buffer[desc_pos+7], desc->buffer[desc_pos+8], 
        desc->buffer[desc_pos+9]);
      list_pos+= 10;
      desc_pos+= 10;
    } else if (desc_pos + 8 <= desc_end) {
      TRACE_EVENT_P8 ("  %02x %02x %02x %02x %02x %02x %02x %02x ",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1], desc->buffer[desc_pos+2], 
        desc->buffer[desc_pos+3], desc->buffer[desc_pos+4], desc->buffer[desc_pos+5], 
        desc->buffer[desc_pos+6], desc->buffer[desc_pos+7]);
      list_pos+= 8;
      desc_pos+= 8;
    } else if (desc_pos + 7 <= desc_end) {
      TRACE_EVENT_P7 ("  %02x %02x %02x %02x %02x %02x %02x ",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1], desc->buffer[desc_pos+2], 
        desc->buffer[desc_pos+3], desc->buffer[desc_pos+4], desc->buffer[desc_pos+5], 
        desc->buffer[desc_pos+6]);
      list_pos+= 7;
      desc_pos+= 7;
    } else if (desc_pos + 6 <= desc_end) {
      TRACE_EVENT_P6 ("  %02x %02x %02x %02x %02x %02x ",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1], desc->buffer[desc_pos+2], 
        desc->buffer[desc_pos+3], desc->buffer[desc_pos+4], desc->buffer[desc_pos+5]);
      list_pos+= 6;
      desc_pos+= 6;
    } else if (desc_pos + 5 <= desc_end) {
      TRACE_EVENT_P5 ("  %02x %02x %02x %02x %02x ",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1], desc->buffer[desc_pos+2], 
        desc->buffer[desc_pos+3], desc->buffer[desc_pos+4]);
      list_pos+= 5;
      desc_pos+= 5;
    } else if (desc_pos + 4 <= desc_end) {
      TRACE_EVENT_P4 ("  %02x %02x %02x %02x ",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1], 
        desc->buffer[desc_pos+2], desc->buffer[desc_pos+3]);
      list_pos+= 4;
      desc_pos+= 4;
    } else if (desc_pos + 3 <= desc_end) {
      TRACE_EVENT_P3 ("  %02x %02x %02x ",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1], 
        desc->buffer[desc_pos+2]);
      list_pos+= 3;
      desc_pos+= 3;
    } else if (desc_pos + 2 <= desc_end) {
      TRACE_EVENT_P2 ("  %02x %02x ",
        desc->buffer[desc_pos+0], desc->buffer[desc_pos+1]);
      list_pos+= 2;
      desc_pos+= 2;
    } else if (desc_pos + 1 <= desc_end) {
      TRACE_EVENT_P1 ("  %02x ", desc->buffer[desc_pos+0]);
      list_pos++;
      desc_pos++;
    }
  } /* while(list_pos < frame_len) */
    /*lint -e661 -e662 possible access/creation of out-of-bounds pointer*/
}
/*
+------------------------------------------------------------------------------
| Function    : sndcp_bin_trace_ip
+------------------------------------------------------------------------------
| Description : traces content of one desc_list using TRACE_IP macro
|
| Parameters  : pointer to desc_list
|               direction: SNDCP_UL_PACKET (0x1)
|                          SNDCP_DL_PACKET (0x2)
|
+------------------------------------------------------------------------------
*/
#ifdef _SNDCP_DTI_2_
GLOBAL void sndcp_bin_trace_ip(T_desc_list2* desc_list, U8 direction)
{
#else /*_SNDCP_DTI_2_*/
GLOBAL void sndcp_bin_trace_ip(T_desc_list* desc_list, U8 direction)
{
#endif /*_SNDCP_DTI_2_*/
  USHORT  frame_len = desc_list->list_len;
#ifdef _SNDCP_DTI_2_
  T_desc2* desc = (T_desc2*)desc_list->first;
#else /*_SNDCP_DTI_2_*/
  T_desc* desc = (T_desc*)desc_list->first;
#endif /*_SNDCP_DTI_2_*/

  U16 pid = 0;
  U16 pos = 0;

  TRACE_FUNCTION("sndcp_bin_trace_ip");

  if(frame_len > 1500){
    TRACE_EVENT("TRACE IP ERROR: IP packet too long.");
    return;
  }

  while(desc != NULL)
  {
#ifdef _SNDCP_DTI_2_
    memcpy(&bin_trace_ip_buf[pos], (U8*)&desc->buffer[desc->offset], desc->len );
    pos += desc->len;
    desc = (T_desc2*)desc->next;
#else /*_SNDCP_DTI_2_*/
    memcpy(&bin_trace_ip_buf[pos], (U8*)&desc->buffer[0], desc->len );
    pos += desc->len;
    desc = (T_desc*)desc->next;
#endif /*_SNDCP_DTI_2_*/
  }

#ifdef TRACE_IP
  if(direction == SNDCP_DL_PACKET){
    TRACE_IP(SNDCP_handle, SNDCP_handle, DOWNLINK_OPC, \
                    (U8*)&bin_trace_ip_buf[0], frame_len);
    pid = bin_trace_ip_buf[5] | (bin_trace_ip_buf[4] << 8); 
    TRACE_EVENT_P1("TRACE IP INFO: Downlink IP Packet, ID: %d", pid);
  } else if (direction == SNDCP_UL_PACKET){
    TRACE_IP(SNDCP_handle, SNDCP_handle, UPLINK_OPC, \
                  (U8*)&bin_trace_ip_buf[0], frame_len);
    pid = bin_trace_ip_buf[5] | (bin_trace_ip_buf[4] << 8); 
    TRACE_EVENT_P1("TRACE IP INFO: Uplink IP Packet, ID: %d", pid);
  } else {
    TRACE_EVENT("TRACE IP ERROR: unknown transfer direction.");
  }
#else /*TRACE_IP*/
  TRACE_EVENT("SNDCP WARNING: TRACE_IP macro not defined");
#endif /*TRACE_IP*/ 
}


/*
+------------------------------------------------------------------------------
| Function    : sndcp_default_ip_trace
+------------------------------------------------------------------------------
| Description : traces TCP, UDP or ICMP header
|
| Parameters  : pointer to desc_list
|               direction: SNDCP_UL_PACKET (0x1)
|                          SNDCP_DL_PACKET (0x2)
|
+------------------------------------------------------------------------------
*/
#ifdef _SNDCP_DTI_2_
GLOBAL void sndcp_default_ip_trace(T_desc_list2 *desc_list, U8 direction)
#else
GLOBAL void sndcp_default_ip_trace(T_desc_list *desc_list, U8 direction)
#endif
{
#ifdef _SNDCP_DTI_2_  
  T_desc2 *desc = (T_desc2 *)desc_list->first;
  U8  *ip_hdr = &desc->buffer[desc->offset];
#else
  T_desc *desc  = (T_desc *)desc_list->first;
  U8  *ip_hdr = &desc->buffer[0];
#endif
  U8  *pk_hdr = NULL; 
  U16  ip_len   = 0;
  U16  icmp_seq = 0;
  U32  seq_num  = 0,
       ack_num  = 0;
  U16  t_length = 0;
  U16  pid      = 0;
  U8   type, code;
  U16  win_size = 0;

  TRACE_FUNCTION("sndcp_default_ip_trace");

  if(desc_list == NULL || desc_list->first == NULL || desc_list->list_len == 0){
    return; 
  }

  ip_len = (ip_hdr[0] & 0x0f) * 4; /* IP header length in bytes*/
  if(desc->len < ip_len){
    TRACE_EVENT("IP TRACE: can not trace ip header, desc too small");
    return;
  }

  pk_hdr = &ip_hdr[ip_len];        /* Pointer to the header on top of IP hdr */  
  t_length |= ip_hdr[2] << 8;      /* Total packet length */
  t_length |= ip_hdr[3];    
  pid |= ip_hdr[4] << 8;           /* Unequal packet ID */
  pid |= ip_hdr[5];

  switch(ip_hdr[9]){
    /* ICMP Protocol */
    case 1:
      if(desc->len < (ip_len + 8)){
        TRACE_EVENT("IP TRACE: can not trace icmp header, desc to small");
        return;
      }
      /* Trace ICMP Header */
      type  = pk_hdr[0];
      if(type == 8 || type == 0){
        icmp_seq |= pk_hdr[6];
        icmp_seq |= pk_hdr[7] << 8;
        TRACE_EVENT_P4("ICMP Echo: type %d, len %d, pid %d, icmp_seq %d", 
                                          type, t_length, pid, icmp_seq);
      } else {
        code = pk_hdr[1];
        TRACE_EVENT_P5("ICMP Packet: dir %d, len %d, pid %d, type: %d, code: %d", 
                                           direction, t_length, pid, type, code);
      }      
     break;
    /* TCP Protocol */
    case 6:
      if(desc->len < (ip_len + 20)){
        TRACE_EVENT("IP TRACE: can not trace tcp header, desc to small");
        return;
      }
      seq_num |= pk_hdr[4] << 24;
      seq_num |= pk_hdr[5] << 16;
      seq_num |= pk_hdr[6] << 8;
      seq_num |= pk_hdr[7];
      ack_num |= pk_hdr[8] << 24;
      ack_num |= pk_hdr[9] << 16;
      ack_num |= pk_hdr[10] << 8;
      ack_num |= pk_hdr[11];
      win_size |= pk_hdr[14] << 8;
      win_size |= pk_hdr[15];

      TRACE_EVENT_P14("TCP Packet: dir %d, len %d, pid %d, seq_num %u, ack_num %u, src_ip %d.%d.%d.%d, dst_ip %d.%d.%d.%d, win_size %d", 
                                      direction, t_length, pid, seq_num, ack_num, 
                                      ip_hdr[12], ip_hdr[13], ip_hdr[14], ip_hdr[15],
                                      ip_hdr[16], ip_hdr[17], ip_hdr[18], ip_hdr[19], win_size);
        break;
    /* UDP Protocol */
    case 17:
      if(desc->len < (ip_len + 8)){
        TRACE_EVENT("IP TRACE: can not trace udp header, desc to small");
        return;
      }
      TRACE_EVENT_P3("UDP Packet: dir %d, len %d, pid %d", 
                                 direction, t_length, pid);
     break;
    default:
        TRACE_EVENT_P1("INFO TRACE: Tracing of protocol type %d not supported",
                                                                    ip_hdr[9]);
      break;
  }

  return;
}

#endif /* CF_FAST_EXEC */
#endif /* SNDCP_TRACE_IP_DATAGRAM */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_swap2
+------------------------------------------------------------------------------
| Description : This routine converts (2 byte) short n from network byte order
|               to host byte order.
|
| Parameters  : USHORT n
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC

GLOBAL USHORT sndcp_swap2(USHORT n)
{
  USHORT tmp = n;
  return n = (((tmp & 0xff00) >> 8) | ((tmp & 0x00ff) << 8) );
}

#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_swap4
+------------------------------------------------------------------------------
| Description : This routine converts (4 byte) long n from network byte order
|               to host byte order.
|
| Parameters  : ULONG n
|
+------------------------------------------------------------------------------
*/

#ifndef CF_FAST_EXEC

GLOBAL ULONG sndcp_swap4(ULONG n) {
  ULONG tmp = n;
  return n = ((tmp & 0xff000000) >> 24) | ((tmp & 0x00ff0000) >> 8 ) |
             ((tmp & 0x0000ff00) << 8 ) | ((tmp & 0x000000ff) << 24);
}

#endif /* CF_FAST_EXEC */
/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_nsapi_rec_state
+------------------------------------------------------------------------------
| Description : The function returns the receiving state for given NSAPI
|
| Parameters  : IN  : nsapi
|               OUT : state 
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC
GLOBAL U8 sndcp_get_nsapi_rec_state (U8 nsapi)
{
  TRACE_FUNCTION( "sndcp_get_nsapi_rec_state" );

  return sndcp_data->rec_states[nsapi];

} /* sndcp_get_nsapi_rec_state() */
#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_nsapi_rec_state
+------------------------------------------------------------------------------
| Description : The function sets the receiving state for given NSAPI
|
| Parameters  : IN : nsapi
|               IN : state 
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC
GLOBAL void sndcp_set_nsapi_rec_state (U8 nsapi, U8 state)
{
  TRACE_FUNCTION( "sndcp_set_nsapi_rec_state" );

  sndcp_data->rec_states[nsapi] = state;

} /* sndcp_set_nsapi_rec_state() */
#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_npdu_num
+------------------------------------------------------------------------------
| Description : E X T R A convenience function, not in SDL.
|               Gets the N-PDU number from the segment header.
|
| Parameters  :  ll_unitdata_ind T_LL_UNITDATA_IND*, npdu_num*
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC
LOCAL void sndcp_get_npdu_num (T_LL_UNITDATA_IND* ll_unitdata_ind, USHORT* npdu_num)
{ 
  USHORT msn_off = 1;
  USHORT lsb_off = 2;
  UBYTE msn = 0;
  UBYTE lsb = 0;

  if ((ll_unitdata_ind->sdu.buf[ll_unitdata_ind->sdu.o_buf / 8]) & 0x40) {
    msn_off = 2;
    lsb_off = 3;
  }

  msn =
    ll_unitdata_ind->sdu.buf[(ll_unitdata_ind->sdu.o_buf / 8) + msn_off];
  lsb = 
    ll_unitdata_ind->sdu.buf[(ll_unitdata_ind->sdu.o_buf / 8) + lsb_off];

  *npdu_num = ((msn & 0xf) << 8) + lsb;
} /* sndcp_get_npdu_num() */
#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_get_seg_num
+------------------------------------------------------------------------------
| Description : E X T R A convenience function, not in SDL.
|               Gets the NSAPI from the segment header.
|
| Parameters  :  ll_unitdata_ind T_LL_UNITDATA_IND*, seg_num*
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC
LOCAL void sndcp_get_seg_num (T_LL_UNITDATA_IND* ll_unitdata_ind, UBYTE* seg_num)
{ 
  *seg_num = (ll_unitdata_ind->sdu.buf[(ll_unitdata_ind->sdu.o_buf / 8) + 2] >> 4);
} /* sd_get_seg_num() */
#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_unack_transfer_params
+------------------------------------------------------------------------------
| Description : There is one N-PDU, whose segments are sent to CIA one by one,
| and that is expected reassembled and decompressed back from CIA. Some
| informations are only given in the first segment. 
| This procedure stores the needed elements in service variables that can
| be read before sending a SIG_SD_CIA_TRANSFER_REQ.
|
| Parameters  : ll_unitdata_req*
| Pre         : Correct instance of sd service must be activated.
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC
GLOBAL void sndcp_set_unack_transfer_params (T_LL_UNITDATA_IND* ll_unitdata_ind) 
{ 

  UBYTE nsapi = 
           (ll_unitdata_ind->sdu.buf[(ll_unitdata_ind->sdu.o_buf / 8)]) & 0xf;

  /*
   * The dcomp value in the first segment of the currently reassembled N-PDU.
   */
  sndcp_data->cur_dcomp[nsapi] = 
   (ll_unitdata_ind->sdu.buf[ll_unitdata_ind->sdu.o_buf / 8 + 1] & 0xf0) >> 4 ;
  /*
   * The pcomp value in the first segment of the currently reassembled N-PDU.
   */
  sndcp_data->cur_pcomp[nsapi] = 
          (ll_unitdata_ind->sdu.buf[ll_unitdata_ind->sdu.o_buf / 8 + 1]) & 0xf;
  /*
   * Reference to N-PDU.
   */
  sndcp_data->cur_pdu_ref[nsapi].ref_nsapi = nsapi;
  sndcp_get_npdu_num(ll_unitdata_ind, &sndcp_data->cur_pdu_ref[nsapi].ref_npdu_num);
  sndcp_get_seg_num(ll_unitdata_ind, &sndcp_data->cur_pdu_ref[nsapi].ref_seg_num);
  /*
   * First and/or last segment?
   */
  sndcp_data->cur_seg_pos[nsapi] = 0;
  /* 
   * if f bit is set 
   */
  if ((ll_unitdata_ind->sdu.buf[ll_unitdata_ind->sdu.o_buf / 8] & 0x40)) {
    sndcp_data->cur_seg_pos[nsapi] += SEG_POS_FIRST;
  }
  /* 
   * if m bit is not set 
   */
  if (!(ll_unitdata_ind->sdu.buf[ll_unitdata_ind->sdu.o_buf / 8] & 0x10)) {
    sndcp_data->cur_seg_pos[nsapi] += SEG_POS_LAST;
  }
} /* sndcp_set_unack_transfer_params() */
#endif /* CF_FAST_EXEC */

/*
+------------------------------------------------------------------------------
| Function    : sndcp_set_ack_transfer_params
+------------------------------------------------------------------------------
| Description : There is one N-PDU, whose segments are sent to CIA one by one,
| and that is expected reassembled and decompressed back from CIA. Some
| informations are only given in the first segment. 
| This procedure stores the needed elements in service variables that can
| be read before sending a SIG_SDA_CIA_TRANSFER_REQ.
|
| Parameters  : ll_data_ind*
| Pre         : Correct instance of sda service must be activated.
|
+------------------------------------------------------------------------------
*/
#ifndef CF_FAST_EXEC
GLOBAL void sndcp_set_ack_transfer_params (T_LL_DATA_IND* ll_data_ind) 
{ 
  UBYTE nsapi = (ll_data_ind->sdu.buf[(ll_data_ind->sdu.o_buf / 8)]) & 0xf;

  /*
   * The dcomp value in the first segment of the currently reassembled N-PDU.
   */
  sndcp_data->cur_dcomp[nsapi] = 
         (ll_data_ind->sdu.buf[ll_data_ind->sdu.o_buf / 8 + 1] & 0xf0) >> 4 ;
  /*
   * The pcomp value in the first segment of the currently reassembled N-PDU.
   */
  sndcp_data->cur_pcomp[nsapi] = 
                  ll_data_ind->sdu.buf[ll_data_ind->sdu.o_buf / 8 + 1] & 0xf;
  /*
   * Reference to N-PDU.
   */
  sndcp_data->cur_pdu_ref[nsapi].ref_nsapi = nsapi;
  sndcp_data->cur_pdu_ref[nsapi].ref_npdu_num = 
                     ll_data_ind->sdu.buf[(ll_data_ind->sdu.o_buf >> 3) + 2];

  
  /*
   * First and/or last segment?
   */
  sndcp_data->cur_seg_pos[nsapi] = 0;
  /*
   * if f bit is set
   */ 
  if ((ll_data_ind->sdu.buf[ll_data_ind->sdu.o_buf / 8]) & 0x40) {
    sndcp_data->cur_seg_pos[nsapi] += SEG_POS_FIRST;
  }
  /*
   * if m bit is not set
   */
  if (!((ll_data_ind->sdu.buf[ll_data_ind->sdu.o_buf / 8]) & 0x10)) {
    sndcp_data->cur_seg_pos[nsapi] += SEG_POS_LAST;
  }
} /* sndcp_set_ack_transfer_params() */
#endif /* CF_FAST_EXEC */