diff src/g23m-gprs/cl/clqos.c @ 1:fa8dc04885d8

src/g23m-*: import from Magnetite
author Mychaela Falconia <falcon@freecalypso.org>
date Fri, 16 Oct 2020 06:25:50 +0000
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/g23m-gprs/cl/clqos.c	Fri Oct 16 06:25:50 2020 +0000
@@ -0,0 +1,464 @@
+/*-----------------------------------------------------------------------------
+|  Project :  3G UMTS PS
+|  Module  :  CL
++------------------------------------------------------------------------------
+|             Copyright 2003 Texas Instruments.
+|             All rights reserved. 
+| 
+|             This file is confidential and a trade secret of Texas 
+|             Instruments .
+|             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. 
++------------------------------------------------------------------------------
+| Purpose:    QoS handling functions.
+|             For design details, see:
+|             8010.149 CL SAP
++----------------------------------------------------------------------------*/
+
+#ifndef CLQOS_C
+#define CLQOS_C
+#endif
+
+#ifndef ENTITY_CL
+#define ENTITY_CL
+#endif
+#define ENTITY_CLT
+
+/*==== INCLUDES =============================================================*/
+
+#include "typedefs.h"   /* to get TI data types */
+#include "vsi.h"        /* to get a lot of macros */
+#include "gsm.h"        /* to get a lot of macros */
+#include "message.h"    /* to get the message definitions for sm */
+#include "prim.h"       /* to get the definitions of used SAP and directions */
+#include "p_cl.val"  /* to get the type of sgsn_rel*/
+#include "p_8010_152_ps_include.h" /*to get T_PS_qos_rXX*/
+#include "cl_inline.h"
+
+
+typedef U8 T_CL_result; /*I really don't like this ! Can we do something in cl.sap ?*/
+
+/*==== CONST ================================================================*/
+
+/*
+ * Conversion table for 1/n in T_ratio.
+ * Contains 1/n * 1E9 entries used for scaling.
+ */
+static const U32 cl_qos_ratio_table[10] = {
+           0UL,
+  1000000000UL,  /* 1E9 / 1 */
+   500000000UL,  /* 1E9 / 2 */
+   333333333UL,  /* 1E9 / 3 */
+   250000000UL,  /* 1E9 / 4 */
+   200000000UL,  /* 1E9 / 5 */
+   166666667UL,  /* 1E9 / 6 */
+   142857143UL,  /* 1E9 / 7 */
+   125000000UL,  /* 1E9 / 8 */
+   111111111UL   /* 1E9 / 9 */
+};
+
+/*==== LOCAL VARS ===========================================================*/
+
+/*==== PRIVATE FUNCTIONS ====================================================*/
+
+/*-----------------------------------------------------------------------------
+| Function    : cl_qos_ratio_to_U32
++------------------------------------------------------------------------------
+| Description : Conversion function: T_ratio to integer
+|
+| Parameters  : T_ratio
+|
+| Return      : ratio as integer
++----------------------------------------------------------------------------*/
+static U32 cl_qos_ratio_to_U32(U8 ratio_mant, U8 ratio_exp)
+{
+  U32   value;
+  int   count;
+
+  /* Error checks */
+  if (ratio_exp > (U8)9) 
+  {
+    /*TRACE_ERROR( "QoS ratio exponent parameter out of bounds!" );*/
+    return 0;
+  }
+  else
+  {
+    if (ratio_mant > (U8)9)
+    {
+      /*TRACE_ERROR( "QoS ratio exponent parameter out of bounds!" );*/
+      return 0;
+    }
+    else
+    {
+      if (ratio_exp == (U8)9 && ratio_mant < (U8)5)
+      {
+        /*TRACE_ERROR( "QoS ratio parameter exceeds data type range!" );*/
+        return 0;
+      }
+    }
+  }
+
+  for (count = 9-(int)ratio_exp, value = 1UL; count > 0; count--)
+  {
+    value *= 10;
+  }
+  /* Get real value by rounding 1/x and dividing by 10^(9-ratio_exp) */
+  value = (cl_qos_ratio_table[(U16)ratio_mant] + (value >> 1)) / value;
+
+  return value;
+}
+
+/*-----------------------------------------------------------------------------
+| Function    : cl_qos_r99_to_delay
++------------------------------------------------------------------------------
+| Description : Given an input R99 QoS, return the corresponding R97 delay
+|               parameter
+|
+| Parameters  : R99 Qos
+|
+| Return      : Delay
++----------------------------------------------------------------------------*/
+static T_PS_delay cl_qos_r99_to_delay(T_PS_qos_r99 *src_qos_r99)
+{
+  T_PS_delay delay;
+
+  /* Traffic class / handling priority */
+  switch ((T_PS_tc)src_qos_r99->tc)
+  {
+    case PS_TC_SUB:       /* 0 - Subscribed */
+      delay               = PS_DELAY_SUB;
+      break;
+    case PS_TC_CONV:      /* 1 - Conversational */
+      delay               = PS_DELAY_1;
+      break;
+    case PS_TC_STREAM:    /* 2 - Streaming */
+      delay               = PS_DELAY_1;
+      break;
+    case PS_TC_INTER:     /* 3 - Interactive */
+      switch (src_qos_r99->handling_pri) 
+      {
+        case PS_HANDLING_PRI_1:
+          delay             = PS_DELAY_1;
+          break;
+        case PS_HANDLING_PRI_2:
+          delay             = PS_DELAY_2;
+          break;
+        case PS_HANDLING_PRI_3:
+        default:            /* For safety */
+          delay             = PS_DELAY_3;
+          break;
+      }   /* switch */
+      break;
+    case PS_TC_BG:        /* 4 - Background */
+      delay               = PS_DELAY_4;
+      break;
+    default:              /* 5, 6, 7 or other garbage */
+      delay               = PS_DELAY_SUB;
+      break;
+  }     /* switch */
+
+  return delay;
+} /* cl_qos_r99_to_delay */
+
+/*-----------------------------------------------------------------------------
+| Function    : cl_qos_r99_to_relclass
++------------------------------------------------------------------------------
+| Description : Given an input R99 QoS, return the corresponding R97 relclass
+|               parameter
+|
+| Parameters  : R99 Qos
+|
+| Return      : Reliability class
++----------------------------------------------------------------------------*/
+static T_PS_relclass cl_qos_r99_to_relclass(T_PS_qos_r99 *src_qos_r99)
+{
+  U32             sdu_err;
+  T_PS_relclass  relclass;
+
+  /* First, convert SDU error ratio to integral number */
+  sdu_err = cl_qos_ratio_to_U32(src_qos_r99->sdu_err_ratio.ratio_mant,
+                                src_qos_r99->sdu_err_ratio.ratio_exp);
+
+  if (sdu_err == 0UL)
+  {             /* SDU err ratio == subscribed value */
+    relclass       = PS_RELCLASS_SUB;      /* 0 */
+  }
+  else
+  {
+    if (sdu_err >= 100000UL)
+    { /* SDU err ratio <= 1E-5 */
+      relclass         = PS_LLC_RLC_PROT;    /* 2 */
+    }
+    else
+    {
+      if (sdu_err >= 2000UL)
+      {  /* 1E-5 < SDU err ratio <= 5E-4 */
+        relclass         = PS_RLC_PROT;        /* 3 */
+      }
+      else
+      {                        /* SDU err ratio > 5E-4 */
+        if (src_qos_r99->ber.ratio_exp > (U8)4 ||
+            (src_qos_r99->ber.ratio_exp == (U8)4 && src_qos_r99->ber.ratio_mant >= (U8)2))
+        {
+                                  /* BER <= 2E-4 */
+          relclass       = PS_PROT;              /* 4 */
+        }
+        else
+        {                    /* BER > 2E-4 */
+          relclass       = PS_NO_REL;            /* 5 */
+        }
+      }
+    }
+  }  /* if (sdu_err ...) */
+
+  return relclass;
+}
+
+/*==== PUBLIC FUNCTIONS =====================================================*/
+
+/*-----------------------------------------------------------------------------
+| Function    : cl_qos_convert_r99_to_r97_req
++------------------------------------------------------------------------------
+| Description : Function for converting a R99 QoS parameter set to a R97/R98
+|               ditto. Conversion rules according to [3G 23.107, sec 9]
+|
+| Parameters  : Originating R99 QoS parameter struct
+|               Destination R97 QoS parameter struct (overwritten in this function)
+|
+| Return      : Conversion result and Destination R97 QoS parameter struct.
++----------------------------------------------------------------------------*/
+T_CL_result cl_qos_convert_r99_to_r97(T_PS_qos_r99 *src_qos_r99,
+                                      T_PS_qos_r97 *dst_qos_r97)
+{
+  T_CL_result  success = (T_CL_result)TRUE;
+  U16          count, max_rate;
+
+  dst_qos_r97->delay    = (U8)cl_qos_r99_to_delay(src_qos_r99);
+
+  /*
+   * SDU error ratio / Residual bit error ratio
+   */
+  dst_qos_r97->relclass = (U8)cl_qos_r99_to_relclass(src_qos_r99);
+
+  /*
+   * Maximum bit-rate
+   *
+   * In case max uplink and downlink data rates differ, [3G 23.107]
+   * says to map according to max(R99_uplink, R99_downlink),
+   * i.e. the higher of the two.
+   */
+  max_rate = (src_qos_r99->max_rate_ul > src_qos_r99->max_rate_dl ?
+              src_qos_r99->max_rate_ul : src_qos_r99->max_rate_dl);
+
+  if (max_rate == (U16)PS_MAX_BR_UL_SUB)
+  { /* Subscribed */
+    dst_qos_r97->peak = (U8)PS_PEAK_SUB;
+  }
+  else
+  { /* ! Subscribed */
+    /* Perform pseudo log2(bit-rate).  Really, we are finding the highest set
+     * bit to use in our comparison rather than making a *huge*
+     * if-[else if]*-else clause.
+     * Fixed ceiling to 2048 (count < 12) => max_rate <= 2^11 */
+    for (count = (U16)4; ((max_rate >> count) != 0) && count < (U16)12; count++);
+
+    /* The quick / code optimal way:
+     * dst_qos_r97->peak = count - 3; */
+
+    switch (count)
+    {
+      case 4:  dst_qos_r97->peak = (U8)PS_PEAK_1K;   break;
+      case 5:  dst_qos_r97->peak = (U8)PS_PEAK_2K;   break;
+      case 6:  dst_qos_r97->peak = (U8)PS_PEAK_4K;   break;
+      case 7:  dst_qos_r97->peak = (U8)PS_PEAK_8K;   break;
+      case 8:  dst_qos_r97->peak = (U8)PS_PEAK_16K;  break;
+      case 9:  dst_qos_r97->peak = (U8)PS_PEAK_32K;  break;
+      case 10: dst_qos_r97->peak = (U8)PS_PEAK_64K;  break;
+      case 11: dst_qos_r97->peak = (U8)PS_PEAK_128K; break;
+      case 12: dst_qos_r97->peak = (U8)PS_PEAK_256K; break;
+    }   /* switch */
+  }     /* if - Subscribed */
+
+  /* R97 peak throughput class is fixed to 31 (best effort) */
+  dst_qos_r97->mean   = (U8)PS_MEAN_BEST;
+  dst_qos_r97->preced = (U8)PS_PRECED_SUB;
+
+  return success;
+}
+
+/*-----------------------------------------------------------------------------
+| Function    : cl_qos_convert_r97_to_r99_req
++------------------------------------------------------------------------------
+| Description : Function for converting a R97/R98 QoS parameter set to a R99
+|               ditto. Conversion rules according to [3G 23.107, sec 9]
+|
+| Parameters  : Originating R97 QoS parameter struct
+|               Destination R99 QoS parameter struct (overwritten in this function)
+|
+| Return      : Conversion result and Destination R99 QoS parameter struct.
++----------------------------------------------------------------------------*/
+T_CL_result cl_qos_convert_r97_to_r99(T_PS_qos_r97 *src_qos_r97,
+                                      T_PS_qos_r99 *dst_qos_r99)
+{
+  T_CL_result  success = (T_CL_result)TRUE;
+
+  /* Delay parameter */
+  switch ((T_PS_delay)src_qos_r97->delay)
+  {
+    case PS_DELAY_SUB:               /* 0 */
+      dst_qos_r99->tc           = (U8)PS_TC_SUB;
+      dst_qos_r99->handling_pri = (U8)PS_HANDLING_PRI_SUB;
+      break;
+    case PS_DELAY_1:                 /* 1 */
+      dst_qos_r99->tc           = (U8)PS_TC_INTER;
+      dst_qos_r99->handling_pri = (U8)PS_HANDLING_PRI_1;
+      break;
+    case PS_DELAY_2:                 /* 2 */
+      dst_qos_r99->tc           = (U8)PS_TC_INTER;
+      dst_qos_r99->handling_pri = (U8)PS_HANDLING_PRI_2;
+      break;
+    case PS_DELAY_3:                 /* 3 */
+      dst_qos_r99->tc           = (U8)PS_TC_INTER;
+      dst_qos_r99->handling_pri = (U8)PS_HANDLING_PRI_3;
+      break;
+    case PS_DELAY_4:                 /* 4 */
+      dst_qos_r99->tc           = (U8)PS_TC_BG;
+      dst_qos_r99->handling_pri = (U8)PS_HANDLING_PRI_SUB;
+      break;
+    default:                      /* 5 - 7 or other garbage */
+      /* Error handling? */
+      success = (T_CL_result)FALSE;
+      break;
+  }     /* switch */
+
+  /* Reliability class */
+  switch ((T_PS_relclass)src_qos_r97->relclass)
+  {
+    case PS_RELCLASS_SUB:                      /* 0 - Subscribed */
+      dst_qos_r99->sdu_err_ratio.ratio_mant = dst_qos_r99->sdu_err_ratio.ratio_exp = (U8)0;
+      dst_qos_r99->ber.ratio_mant = dst_qos_r99->ber.ratio_exp = (U8)0;
+      dst_qos_r99->del_err_sdu = (U8)PS_DEL_ERR_SUB;
+      break;
+    case PS_GTP_LLC_RLC_PROT:                  /* 1 - Acknowledged GTP, LLC, and RLC */
+      dst_qos_r99->sdu_err_ratio.ratio_mant = (U8)1;
+      dst_qos_r99->sdu_err_ratio.ratio_exp  = (U8)6; /* SDU error ratio = 1E-6 */
+      dst_qos_r99->ber.ratio_mant           = (U8)1;
+      dst_qos_r99->ber.ratio_exp            = (U8)5; /* Residual BER = 1E-5 */
+      dst_qos_r99->del_err_sdu = (U8)PS_DEL_ERR_NO;
+      break;
+    case PS_LLC_RLC_PROT:                      /* 2 - Unacknowledged GTP; Acknowledged LLC and RLC */
+      dst_qos_r99->sdu_err_ratio.ratio_mant = (U8)1;
+      dst_qos_r99->sdu_err_ratio.ratio_exp  = (U8)6; /* SDU error ratio = 1E-6 */
+      dst_qos_r99->ber.ratio_mant           = (U8)1;
+      dst_qos_r99->ber.ratio_exp            = (U8)5; /* Residual BER = 1E-5 */
+      dst_qos_r99->del_err_sdu = (U8)PS_DEL_ERR_NO;
+      break;
+    case PS_RLC_PROT:                          /* 3 - Unacknowledged GTP and LLC; Acknowledged RLC */
+      dst_qos_r99->sdu_err_ratio.ratio_mant = (U8)1;
+      dst_qos_r99->sdu_err_ratio.ratio_exp  = (U8)4; /* SDU error ratio = 1E-4 */
+      dst_qos_r99->ber.ratio_mant           = (U8)1;
+      dst_qos_r99->ber.ratio_exp            = (U8)5; /* Residual BER = 1E-5 */
+      dst_qos_r99->del_err_sdu = (U8)PS_DEL_ERR_NO;
+      break;
+    case PS_PROT:                              /* 4 - Unacknowledged GTP, LLC, and RLC */
+      dst_qos_r99->sdu_err_ratio.ratio_mant = (U8)1;
+      dst_qos_r99->sdu_err_ratio.ratio_exp  = (U8)3; /* SDU error ratio = 1E-3 */
+      dst_qos_r99->ber.ratio_mant           = (U8)1;
+      dst_qos_r99->ber.ratio_exp            = (U8)5; /* Residual BER = 1E-5 */
+      dst_qos_r99->del_err_sdu = (U8)PS_DEL_ERR_NO;
+      break;
+    case PS_NO_REL:                            /* 5 - Unacknowledged GTP, LLC, and RLC (Unprotected) */
+      dst_qos_r99->sdu_err_ratio.ratio_mant = (U8)1;
+      dst_qos_r99->sdu_err_ratio.ratio_exp  = (U8)3; /* SDU error ratio = 1E-3 */
+      dst_qos_r99->ber.ratio_mant           = (U8)4;
+      dst_qos_r99->ber.ratio_exp            = (U8)3; /* Residual BER = 4E-3 */
+      dst_qos_r99->del_err_sdu = (U8)PS_DEL_ERR_YES;
+      break;
+    default:                                      /* 6, 7 or other garbage */
+      /* Error handling? */
+      success = (T_CL_result)FALSE;
+      break;
+  }     /* switch */
+
+  /* Peak throughput parameter */
+  /* NOTE: Could be optimized heavily into the following:
+   * dst_qos_r99->max_br_dl = dst_qos_r99->max_br_ul = (peak == 0 ? 0 : 4 << peak)
+   */
+
+  switch ((T_PS_peak)src_qos_r97->peak)
+  {
+    case PS_PEAK_SUB:             /* 0 */
+      dst_qos_r99->max_rate_dl = dst_qos_r99->max_rate_ul = (U16)PS_MAX_BR_UL_SUB;
+      break;
+    case PS_PEAK_1K:              /* 1 */
+      dst_qos_r99->max_rate_dl = dst_qos_r99->max_rate_ul = (U16)8;
+      break;
+    case PS_PEAK_2K:              /* 2 */
+      dst_qos_r99->max_rate_dl = dst_qos_r99->max_rate_ul = (U16)16;
+      break;
+    case PS_PEAK_4K:              /* 3 */
+      dst_qos_r99->max_rate_dl = dst_qos_r99->max_rate_ul = (U16)32;
+      break;
+    case PS_PEAK_8K:              /* 4 */
+      dst_qos_r99->max_rate_dl = dst_qos_r99->max_rate_ul = (U16)64;
+      break;
+    case PS_PEAK_16K:             /* 5 */
+      dst_qos_r99->max_rate_dl = dst_qos_r99->max_rate_ul = (U16)128;
+      break;
+    case PS_PEAK_32K:             /* 6 */
+      dst_qos_r99->max_rate_dl = dst_qos_r99->max_rate_ul = (U16)256;
+      break;
+    case PS_PEAK_64K:             /* 7 */
+      dst_qos_r99->max_rate_dl = dst_qos_r99->max_rate_ul = (U16)512;
+      break;
+    case PS_PEAK_128K:            /* 8 */
+      dst_qos_r99->max_rate_dl = dst_qos_r99->max_rate_ul = (U16)1024;
+      break;
+    case PS_PEAK_256K:            /* 9 */
+      dst_qos_r99->max_rate_dl = dst_qos_r99->max_rate_ul = (U16)2048;
+      break;
+    default:
+      /* Handling? */
+      success = (T_CL_result)FALSE;
+      break;
+  }     /* switch */
+
+  /* R99 max SDU size is fixed to 1500 */
+  dst_qos_r99->max_sdu  = (U16)1500;
+
+  /* Remaining R99 fields are not specified. We fill them in with subscribed */
+  dst_qos_r99->order      =  (U8)PS_ORDER_SUB;
+  dst_qos_r99->xfer_delay = (U16)PS_XFER_DELAY_SUB;
+  dst_qos_r99->guar_br_ul = (U16)PS_GUAR_BR_UL_SUB;
+  dst_qos_r99->guar_br_dl = (U16)PS_GUAR_BR_DL_SUB;
+
+  return success;
+}
+
+/*-----------------------------------------------------------------------------
+| Function    : cl_qos_use_ack_mode_llc_req
++------------------------------------------------------------------------------
+| Description : Query function: Given input QoS, should NSAPI use acknowledged
+|               LLC mode?
+|
+| Parameters  : Originating R99 QoS parameter struct
+|
+| Return      : LLC mode
++----------------------------------------------------------------------------*/
+T_PS_llc_mode cl_qos_use_ack_mode_llc(T_PS_qos_r99 *src_qos_r99)
+{
+  T_PS_relclass  relclass;
+  T_PS_llc_mode   ret_val = PS_LLC_UNACKNOWLEDGED;
+
+  relclass = cl_qos_r99_to_relclass(src_qos_r99);
+
+  if (relclass == PS_GTP_LLC_RLC_PROT || relclass == PS_LLC_RLC_PROT)
+  {
+    ret_val = PS_LLC_ACKNOWLEDGED;
+  }
+
+  return ret_val;
+}