lib/iec_std_lib.h
changeset 375 7a11f9e9e703
parent 348 990ecb883731
parent 350 2c3c4dc34979
child 378 7f2e120c7352
--- a/lib/iec_std_lib.h	Wed Sep 07 19:28:10 2011 +0200
+++ b/lib/iec_std_lib.h	Thu Sep 08 20:25:00 2011 +0200
@@ -1,5 +1,6 @@
 /*
- * (c) 2008 Edouard TISSERANT
+ * copyright 2008 Edouard TISSERANT
+ * copyright 2011 Mario de Sousa (msousa@fe.up.pt)
  *
  * Offered to the public under the terms of the GNU Lesser General Public
  * License as published by the Free Software Foundation; either version 2
@@ -18,6 +19,13 @@
  * IEC 61131-3 standard function library
  */
 
+/* NOTE: This file is full of (what may seem at first) very strange macros.
+ *       If you want to know what all these strange macros are doing,
+ *       just parse this file through a C preprocessor (e.g. cpp), 
+ *       and analyse the output!
+ *       $gcc -E iec_std_lib.h 
+ */
+
 #include <limits.h>
 #include <float.h>
 #include <math.h>
@@ -51,7 +59,7 @@
     u_int16_t body[STR_MAX_LEN];
 } WSTRING;
 */
-
+/*
 # if __WORDSIZE == 64
 #define __32b_sufix
 #define __64b_sufix L
@@ -59,6 +67,19 @@
 #define __32b_sufix L
 #define __64b_sufix LL
 #endif
+*/
+
+# if __WORDSIZE == 64
+#define __32b_sufix
+#define __64b_sufix L
+#else
+#define __32b_sufix L
+/* changed this from LL to L temporarily. It was causing a bug when compiling resulting code with gcc.
+ * I have other things to worry about at the moment.. 
+ */
+#define __64b_sufix L   
+#endif
+
 
 #define __lit(type,value,sfx) (type)value##sfx
 // Keep this macro expention step to let sfx change into L or LL
@@ -83,7 +104,7 @@
 #define __BYTE_LITERAL(value) __literal(BYTE,value,)
 #define __WORD_LITERAL(value) __literal(WORD,value,)
 #define __DWORD_LITERAL(value) __literal(DWORD,value,__32b_sufix)
-#define __LWORD_LITERAL(value) __literal(LWORD,value,__32b_sufix)
+#define __LWORD_LITERAL(value) __literal(LWORD,value,__64b_sufix)
 
 
 
@@ -136,17 +157,33 @@
     DATE    DATEvar;
 } __IL_DEFVAR_T;
 
-/*****************/
-/* Misc internal */
-/*****************/
+
+/**********************************************************************/
+/**********************************************************************/
+/*****                                                            *****/
+/*****      Some helper functions...                              *****/
+/*****                     ...used later:                         *****/
+/*****    - when declaring the IEC 61131-3 standard functions     *****/
+/*****    - in the C source code itself in SFC and ST expressions *****/
+/*****                                                            *****/
+/**********************************************************************/
+/**********************************************************************/
+
+
+/****************************/
+/* Notify IEC runtime error */
+/****************************/
 
 /* function that generates an IEC runtime error */
-static inline void IEC_error(void) {
+static inline void __iec_error(void) {
   /* TODO... */
   fprintf(stderr, "IEC 61131-3 runtime error.\n");
   /*exit(1);*/
 }
 
+/*******************************/
+/* Time normalization function */
+/*******************************/
 
 static inline void __normalize_timespec (IEC_TIMESPEC *ts) {
   if( ts->tv_nsec < -1000000000 || (( ts->tv_sec > 0 ) && ( ts->tv_nsec < 0 ))){
@@ -159,6 +196,10 @@
   }
 }
 
+/**********************************************/
+/* Time conversion to/from timespec functions */
+/**********************************************/
+
 static inline IEC_TIMESPEC __time_to_timespec(int sign, double mseconds, double seconds, double minutes, double hours, double days) {
   IEC_TIMESPEC ts;
 
@@ -197,7 +238,7 @@
   epoch_seconds = mktime(&broken_down_time); /* determine number of seconds since the epoch, i.e. Jan 1st 1970 */
 
   if ((time_t)(-1) == epoch_seconds)
-    IEC_error();
+    __iec_error();
 
   ts.tv_sec = epoch_seconds;
   ts.tv_nsec = 0;
@@ -223,56 +264,35 @@
 
   epoch_seconds = mktime(&broken_down_time); /* determine number of seconds since the epoch, i.e. Jan 1st 1970 */
   if ((time_t)(-1) == epoch_seconds)
-    IEC_error();
+    __iec_error();
 
   ts.tv_sec += epoch_seconds;
   if (ts.tv_sec < epoch_seconds)
     /* since the TOD is always positive, if the above happens then we had an overflow */
-    IEC_error();
+    __iec_error();
 
   return ts;
 }
 
-/********************/
-/*   EN/ENO PARAMS  */
-/********************/
-
-#define EN_ENO_PARAMS BOOL EN, BOOL *ENO
-#define TEST_EN(TYPENAME)\
-  if (!EN) {\
-    if (ENO != NULL)\
-      *ENO = __BOOL_LITERAL(FALSE);\
-    return __INIT_##TYPENAME;\
-  }\
-  else if (ENO != NULL)\
-    *ENO = __BOOL_LITERAL(TRUE);
-#define TEST_EN_COND(TYPENAME, COND)\
-  if (!EN || (COND)) {\
-    if (ENO != NULL)\
-      *ENO = __BOOL_LITERAL(FALSE);\
-    return __INIT_##TYPENAME;\
-  }\
-  else if (ENO != NULL)\
-    *ENO = __BOOL_LITERAL(TRUE);
-
-/***************/
-/*   Time ops  */
-/***************/
-#define __TIME_CMP(t1, t2) (t2.tv_sec == t1.tv_sec ? t1.tv_nsec - t2.tv_nsec : t1.tv_sec - t2.tv_sec)
-
-static inline TIME __TIME_ADD(TIME IN1, TIME IN2){
+/*******************/
+/* Time operations */
+/*******************/
+
+#define __time_cmp(t1, t2) (t2.tv_sec == t1.tv_sec ? t1.tv_nsec - t2.tv_nsec : t1.tv_sec - t2.tv_sec)
+
+static inline TIME __time_add(TIME IN1, TIME IN2){
   TIME res ={IN1.tv_sec + IN2.tv_sec,
              IN1.tv_nsec + IN2.tv_nsec };
   __normalize_timespec(&res);
   return res;
 }
-static inline TIME __TIME_SUB(TIME IN1, TIME IN2){
+static inline TIME __time_sub(TIME IN1, TIME IN2){
   TIME res ={IN1.tv_sec - IN2.tv_sec,
              IN1.tv_nsec - IN2.tv_nsec };
   __normalize_timespec(&res);
   return res;
 }
-static inline TIME __TIME_MUL(TIME IN1, LREAL IN2){
+static inline TIME __time_mul(TIME IN1, LREAL IN2){
   LREAL s_f = IN1.tv_sec * IN2;
   time_t s = s_f;
   div_t ns = div((LREAL)IN1.tv_nsec * IN2, 1000000000);
@@ -281,7 +301,7 @@
   __normalize_timespec(&res);
   return res;
 }
-static inline TIME __TIME_DIV(TIME IN1, LREAL IN2){
+static inline TIME __time_div(TIME IN1, LREAL IN2){
   LREAL s_f = IN1.tv_sec / IN2;
   time_t s = s_f;
   TIME res = {s,
@@ -290,183 +310,6 @@
   return res;
 }
 
-static inline TIME __date_and_time_to_time_of_day(EN_ENO_PARAMS, TIME IN){
-  TEST_EN(TIME)
-  return (TIME){IN.tv_sec % 86400, IN.tv_nsec};
-}
-static inline TIME __date_and_time_to_date(EN_ENO_PARAMS, TIME IN){
-  TEST_EN(TIME)
-  return (TIME){IN.tv_sec - (IN.tv_sec % (24*60*60)), 0};
-}
-static inline TIME __time_add(EN_ENO_PARAMS, TIME IN1, TIME IN2){
-  TEST_EN(TIME)
-  return __TIME_ADD(IN1, IN2);
-}
-static inline TIME __time_sub(EN_ENO_PARAMS, TIME IN1, TIME IN2){
-  TEST_EN(TIME)
-  return __TIME_SUB(IN1, IN2);
-}
-static inline TIME __time_mul(EN_ENO_PARAMS, TIME IN1, LREAL IN2){
-  TEST_EN(TIME)
-  return __TIME_MUL(IN1, IN2);
-}
-static inline TIME __time_div(EN_ENO_PARAMS, TIME IN1, LREAL IN2){
-  TEST_EN(TIME)
-  return __TIME_DIV(IN1, IN2);
-}
-
-/***************/
-/* String ops  */
-/***************/
-#define __STR_CMP(str1, str2) memcmp((char*)&str1.body,(char*)&str2.body, str1.len < str2.len ? str1.len : str2.len)
-
-static inline __strlen_t __len(EN_ENO_PARAMS, STRING IN){
-    TEST_EN(UINT)
-    return IN.len;
-}
-static inline STRING __left(EN_ENO_PARAMS, STRING IN, __strlen_t L){
-    STRING res;
-    TEST_EN_COND(STRING, L < 0)
-    res = __INIT_STRING;
-    L = L < IN.len ? L : IN.len;
-    memcpy(&res.body, &IN.body, L);
-    res.len = L;
-    return res;
-}
-static inline STRING __right(EN_ENO_PARAMS, STRING IN, __strlen_t L){
-    STRING res;
-    TEST_EN_COND(STRING, L < 0)
-    res = __INIT_STRING;
-    L = L < IN.len ? L : IN.len;
-    memcpy(&res.body, &IN.body[IN.len - L], L);
-    res.len = L;
-    return res;
-}
-static inline STRING __mid(EN_ENO_PARAMS, STRING IN, __strlen_t L, __strlen_t P){
-    STRING res;
-    TEST_EN_COND(STRING, L < 0 || P < 0)
-    res = __INIT_STRING;
-    if(P <= IN.len){
-	    P -= 1; /* now can be used as [index]*/
-	    L = L + P <= IN.len ? L : IN.len - P;
-	    memcpy(&res.body, &IN.body[P] , L);
-        res.len = L;
-    }
-    return res;
-}
-static inline STRING __concat(EN_ENO_PARAMS, UINT param_count, ...){
-  UINT i;
-  STRING res;
-  va_list ap;
-  __strlen_t charcount;
-  TEST_EN(STRING)
-  charcount = 0;
-  res = __INIT_STRING;
-
-  va_start (ap, param_count);         /* Initialize the argument list.  */
-
-  for (i = 0; i < param_count && charcount < STR_MAX_LEN; i++)
-  {
-    STRING tmp = va_arg(ap, STRING);
-    __strlen_t charrem = STR_MAX_LEN - charcount;
-    __strlen_t to_write = tmp.len > charrem ? charrem : tmp.len;
-    memcpy(&res.body[charcount], &tmp.body , to_write);
-    charcount += to_write;
-  }
-
-  res.len = charcount;
-
-  va_end (ap);                  /* Clean up.  */
-  return res;
-}
-static inline STRING __insert(EN_ENO_PARAMS, STRING IN1, STRING IN2, __strlen_t P){
-    STRING res;
-    __strlen_t to_copy;
-    TEST_EN_COND(STRING, P < 0)
-    res = __INIT_STRING;
-
-    to_copy = P > IN1.len ? IN1.len : P;
-    memcpy(&res.body, &IN1.body , to_copy);
-    P = res.len = to_copy;
-
-    to_copy = IN2.len + res.len > STR_MAX_LEN ? STR_MAX_LEN - res.len : IN2.len;
-    memcpy(&res.body[res.len], &IN2.body , to_copy);
-    res.len += to_copy;
-
-    to_copy = IN1.len - P < STR_MAX_LEN - res.len ? IN1.len - P : STR_MAX_LEN - res.len ;
-    memcpy(&res.body[res.len], &IN1.body[P] , to_copy);
-    res.len += to_copy;
-
-    return res;
-}
-static inline STRING __delete(EN_ENO_PARAMS, STRING IN, __strlen_t L, __strlen_t P){
-    STRING res;
-    __strlen_t to_copy;
-    TEST_EN_COND(STRING, L < 0 || P < 0)
-    res = __INIT_STRING;
-
-    to_copy = P > IN.len ? IN.len : P-1;
-    memcpy(&res.body, &IN.body , to_copy);
-    P = res.len = to_copy;
-
-    if( IN.len > P + L ){
-        to_copy = IN.len - P - L;
-        memcpy(&res.body[res.len], &IN.body[P + L], to_copy);
-        res.len += to_copy;
-    }
-
-    return res;
-}
-static inline STRING __replace(EN_ENO_PARAMS, STRING IN1, STRING IN2, __strlen_t L, __strlen_t P){
-    STRING res;
-    __strlen_t to_copy;
-    TEST_EN_COND(STRING, L < 0 || P < 0)
-    res = __INIT_STRING;
-
-    to_copy = P > IN1.len ? IN1.len : P-1;
-    memcpy(&res.body, &IN1.body , to_copy);
-    P = res.len = to_copy;
-
-    to_copy = IN2.len < L ? IN2.len : L;
-
-    if( to_copy + res.len > STR_MAX_LEN )
-       to_copy = STR_MAX_LEN - res.len;
-
-    memcpy(&res.body[res.len], &IN2.body , to_copy);
-    res.len += to_copy;
-
-    P += L;
-    if( res.len <  STR_MAX_LEN && P < IN1.len)
-    {
-        to_copy = IN1.len - P;
-        memcpy(&res.body[res.len], &IN1.body[P] , to_copy);
-        res.len += to_copy;
-    }
-
-    return res;
-}
-
-
-
-static inline __strlen_t __pfind(STRING* IN1, STRING* IN2){
-    UINT count1 = 0; /* offset of first matching char in IN1 */
-    UINT count2 = 0; /* count of matching char */
-    while(count1 + count2 < IN1->len && count2 < IN2->len)
-    {
-        if(IN1->body[count1 + count2] != IN2->body[count2]){
-            count1 += count2 + 1;
-            count2 = 0;
-        }
-        else {
-            count2++;
-        }
-    }
-    return count2 == IN2->len -1 ? 0 : count1 + 1;
-}
-static inline __strlen_t __find(EN_ENO_PARAMS, STRING IN1, STRING IN2){
-    TEST_EN(UINT)
-    return __pfind(&IN1, &IN2);
-}
 
 /***************/
 /* Convertions */
@@ -474,67 +317,49 @@
     /*****************/
     /*  REAL_TO_INT  */
     /*****************/
-static inline LINT __real_round(LREAL IN)
-{
+static inline LINT __real_round(LREAL IN) {
 	return fmod(IN, 1) == 0 ? ((LINT)IN / 2) * 2 : (LINT)IN;
 }
-static inline LINT __preal_to_sint(LREAL IN)
-{
+static inline LINT __preal_to_sint(LREAL IN) {
    return IN >= 0 ? __real_round(IN + 0.5) : __real_round(IN - 0.5);
 }
-static inline LINT __preal_to_uint(LREAL IN)
-{
+static inline LINT __preal_to_uint(LREAL IN) {
    return IN >= 0 ? __real_round(IN + 0.5) : 0;
 }
-static inline LINT __real_to_sint(EN_ENO_PARAMS, LREAL IN){
-    TEST_EN(LINT)
-    return (LINT)__preal_to_sint(IN);
-}
-static inline LWORD __real_to_bit(EN_ENO_PARAMS, LREAL IN){
-    TEST_EN(LWORD)
-    return (LWORD)__preal_to_uint(IN);
-}
-static inline ULINT __real_to_uint(EN_ENO_PARAMS, LREAL IN){
-    TEST_EN(ULINT)
-    return (ULINT)__preal_to_uint(IN);
-}
+static inline LINT __real_to_sint(LREAL IN)  {return (LINT)__preal_to_sint(IN);}
+static inline LWORD __real_to_bit(LREAL IN)  {return (LWORD)__preal_to_uint(IN);}
+static inline ULINT __real_to_uint(LREAL IN) {return (ULINT)__preal_to_uint(IN);}
+
     /***************/
     /*  TO_STRING  */
     /***************/
-static inline STRING __bool_to_string(EN_ENO_PARAMS, BOOL IN)
-{
-    TEST_EN(STRING)
-    if(IN)
-        return (STRING){4, "TRUE"};
+static inline STRING __bool_to_string(BOOL IN) {
+    if(IN) return (STRING){4, "TRUE"};
     return (STRING){5,"FALSE"};
 }
-static inline STRING __bit_to_string(EN_ENO_PARAMS, LWORD IN){
+static inline STRING __bit_to_string(LWORD IN) {
     STRING res;
-    TEST_EN(STRING)
     res = __INIT_STRING;
     res.len = snprintf((char*)res.body, STR_MAX_LEN, "16#%llx", IN);
     if(res.len > STR_MAX_LEN) res.len = STR_MAX_LEN;
     return res;
 }
-static inline STRING __real_to_string(EN_ENO_PARAMS, LREAL IN){
+static inline STRING __real_to_string(LREAL IN) {
     STRING res;
-    TEST_EN(STRING)
     res = __INIT_STRING;
     res.len = snprintf((char*)res.body, STR_MAX_LEN, "%.10g", IN);
     if(res.len > STR_MAX_LEN) res.len = STR_MAX_LEN;
     return res;
 }
-static inline STRING __sint_to_string(EN_ENO_PARAMS, LINT IN){
+static inline STRING __sint_to_string(LINT IN) {
     STRING res;
-    TEST_EN(STRING)
     res = __INIT_STRING;
     res.len = snprintf((char*)res.body, STR_MAX_LEN, "%lld", IN);
     if(res.len > STR_MAX_LEN) res.len = STR_MAX_LEN;
     return res;
 }
-static inline STRING __uint_to_string(EN_ENO_PARAMS, ULINT IN){
+static inline STRING __uint_to_string(ULINT IN) {
     STRING res;
-    TEST_EN(STRING)
     res = __INIT_STRING;
     res.len = snprintf((char*)res.body, STR_MAX_LEN, "%llu", IN);
     if(res.len > STR_MAX_LEN) res.len = STR_MAX_LEN;
@@ -543,12 +368,11 @@
     /***************/
     /* FROM_STRING */
     /***************/
-static inline BOOL __string_to_bool(EN_ENO_PARAMS, STRING IN){
-    TEST_EN(BOOL)
+static inline BOOL __string_to_bool(STRING IN) {
     return IN.len == 5 ? !memcmp(&IN.body,"TRUE", IN.len) : 0;
 }
 
-static inline LINT __pstring_to_sint(STRING* IN){
+static inline LINT __pstring_to_sint(STRING* IN) {
     LINT res = 0;
     __strlen_t l;
     unsigned int shift = 0;
@@ -609,22 +433,12 @@
     return res;
 }
 
-static inline LINT __string_to_sint(EN_ENO_PARAMS, STRING IN){
-    TEST_EN(LINT)
-    return (LINT)__pstring_to_sint(&IN);
-}
-static inline LWORD __string_to_bit(EN_ENO_PARAMS, STRING IN){
-    TEST_EN(LWORD)
-    return (LWORD)__pstring_to_sint(&IN);
-}
-static inline ULINT __string_to_uint(EN_ENO_PARAMS, STRING IN){
-    TEST_EN(ULINT)
-    return (ULINT)__pstring_to_sint(&IN);
-}
-static inline LREAL __string_to_real(EN_ENO_PARAMS, STRING IN){
-	__strlen_t l;
-	TEST_EN(LREAL)
-	l = IN.len;
+static inline LINT  __string_to_sint(STRING IN) {return (LINT)__pstring_to_sint(&IN);}
+static inline LWORD __string_to_bit (STRING IN) {return (LWORD)__pstring_to_sint(&IN);}
+static inline ULINT __string_to_uint(STRING IN) {return (ULINT)__pstring_to_sint(&IN);}
+static inline LREAL __string_to_real(STRING IN) {
+    __strlen_t l;
+    l = IN.len;
     /* search the dot */
     while(--l > 0 && IN.body[l] != '.');
     if(l != 0){
@@ -637,18 +451,10 @@
     /***************/
     /*   TO_TIME   */
     /***************/
-static inline TIME __int_to_time(EN_ENO_PARAMS, LINT IN){
-    TEST_EN(TIME)
-    return (TIME){IN, 0};
-}
-
-static inline TIME __real_to_time(EN_ENO_PARAMS, LREAL IN){
-    TEST_EN(TIME)
-    return (TIME){IN, (IN - (LINT)IN) * 1000000000};
-}
-static inline TIME __string_to_time(EN_ENO_PARAMS, STRING IN){
-	__strlen_t l;
-	TEST_EN(TIME)
+static inline TIME    __int_to_time(LINT IN)  {return (TIME){IN, 0};}
+static inline TIME   __real_to_time(LREAL IN) {return (TIME){IN, (IN - (LINT)IN) * 1000000000};}
+static inline TIME __string_to_time(STRING IN){
+    __strlen_t l;
     /* TODO :
      *
      *  Duration literals without underlines: T#14ms    T#-14ms   T#14.7s   T#14.7m
@@ -671,7 +477,7 @@
      */
     /* Quick hack : only transform seconds */
     /* search the dot */
-	l = IN.len;
+    l = IN.len;
     while(--l > 0 && IN.body[l] != '.');
     if(l != 0){
         LREAL IN_val = atof((const char *)&IN.body);
@@ -684,18 +490,13 @@
     /***************/
     /*  FROM_TIME  */
     /***************/
-static inline LREAL __time_to_real(EN_ENO_PARAMS, TIME IN){
-    TEST_EN(LREAL)
+static inline LREAL __time_to_real(TIME IN){
     return (LREAL)IN.tv_sec + ((LREAL)IN.tv_nsec/1000000000);
 }
-static inline LINT __time_to_int(EN_ENO_PARAMS, TIME IN){
-    TEST_EN(LINT)
-    return IN.tv_sec;
-}
-static inline STRING __time_to_string(EN_ENO_PARAMS, TIME IN){
+static inline LINT __time_to_int(TIME IN) {return IN.tv_sec;}
+static inline STRING __time_to_string(TIME IN){
     STRING res;
     div_t days;
-    TEST_EN(STRING)
     /*t#5d14h12m18s3.5ms*/
     res = __INIT_STRING;
     days = div(IN.tv_sec ,86400);
@@ -721,32 +522,30 @@
     if(res.len > STR_MAX_LEN) res.len = STR_MAX_LEN;
     return res;
 }
-static inline STRING __date_to_string(EN_ENO_PARAMS, DATE IN){
+static inline STRING __date_to_string(DATE IN){
     STRING res;
     struct tm* broken_down_time;
     time_t seconds;
-	TEST_EN(STRING)
     /* D#1984-06-25 */
     res = __INIT_STRING;
     seconds = IN.tv_sec;
     if (NULL == (broken_down_time = localtime(&seconds))){ /* get the UTC (GMT) broken down time */
-        IEC_error();
+        __iec_error();
         return (STRING){7,"D#ERROR"};
     }
     res.len = snprintf((char*)&res.body, STR_MAX_LEN, "D#%d-%2.2d-%2.2d", broken_down_time->tm_year + 1900, broken_down_time->tm_mon + 1, broken_down_time->tm_mday);
     if(res.len > STR_MAX_LEN) res.len = STR_MAX_LEN;
     return res;
 }
-static inline STRING __tod_to_string(EN_ENO_PARAMS, TOD IN){
+static inline STRING __tod_to_string(TOD IN){
     STRING res;
     struct tm* broken_down_time;
     time_t seconds;
-    TEST_EN(STRING)
     /* TOD#15:36:55.36 */
     res = __INIT_STRING;
     seconds = IN.tv_sec;
     if (NULL == (broken_down_time = localtime(&seconds))){ /* get the UTC (GMT) broken down time */
-        IEC_error();
+        __iec_error();
         return (STRING){9,"TOD#ERROR"};
     }
     if(IN.tv_nsec == 0){
@@ -757,15 +556,14 @@
     if(res.len > STR_MAX_LEN) res.len = STR_MAX_LEN;
     return res;
 }
-static inline STRING __dt_to_string(EN_ENO_PARAMS, DT IN){
+static inline STRING __dt_to_string(DT IN){
     STRING res;
     struct tm* broken_down_time;
     time_t seconds;
-    TEST_EN(STRING)
     /* DT#1984-06-25-15:36:55.36 */
     seconds = IN.tv_sec;
     if (NULL == (broken_down_time = localtime(&seconds))){ /* get the UTC (GMT) broken down time */
-        IEC_error();
+        __iec_error();
         return (STRING){8,"DT#ERROR"};
     }
     if(IN.tv_nsec == 0){
@@ -788,12 +586,21 @@
     if(res.len > STR_MAX_LEN) res.len = STR_MAX_LEN;
     return res;
 }
-    /* BCD */
+
+    /**********************************************/
+    /*  [ANY_DATE | TIME] _TO_ [ANY_DATE | TIME]  */
+    /**********************************************/
+
+static inline TOD __date_and_time_to_time_of_day(DT IN) {return (TOD){IN.tv_sec % 86400, IN.tv_nsec};}
+static inline DATE __date_and_time_to_date(DT IN){return (DATE){IN.tv_sec - (IN.tv_sec % (24*60*60)), 0};}
+
+    /*****************/
+    /*  FROM/TO BCD  */
+    /*****************/
 #define __bcd_digit(fac)
-static inline ULINT __bcd_to_uint(EN_ENO_PARAMS, LWORD IN){
+static inline ULINT __bcd_to_uint(LWORD IN){
     ULINT res;
     ULINT i;
-    TEST_EN(ULINT)
 
     res = IN & 0xf;
     for(i = 10ULL; i <= 1000000000000000ULL; i *= 10){
@@ -804,10 +611,9 @@
     return res;
 }
 
-static inline LWORD __uint_to_bcd(EN_ENO_PARAMS, ULINT IN){
+static inline LWORD __uint_to_bcd(ULINT IN){
     LWORD res;
     USINT i;
-    TEST_EN(LWORD)
 
     res = IN % 10;
     for(i = 4; i<=60; i += 4){
@@ -818,6 +624,296 @@
     return res;
 }
 
+
+    /************/
+    /*  MOVE_*  */
+    /************/
+
+/* some helpful __move_[ANY] functions, used in the *_TO_** and MOVE  standard functions */
+/* e.g. __move_BOOL, __move_BYTE, __move_REAL, __move_TIME, ... */
+#define __iec_(TYPENAME)\
+static inline TYPENAME __move_##TYPENAME(TYPENAME op1) {return op1;}
+__ANY(__iec_)
+#undef __iec_
+
+
+
+/*****************************************************************/
+/*****************************************************************/
+/*****                                                       *****/
+/*****                 IEC 61131-3                           *****/
+/*****      S T A N D A R D     F U N C T I O N S            *****/
+/*****                                                       *****/
+/*****************************************************************/
+/*****************************************************************/
+
+/* NOTE: If you want to know what all these strange macros are doing,
+ *       just parse this file through a C preprocessor, and analyse the output!
+ *       $gcc -E iec_std_lib.h 
+ */
+
+/* NOTE: We only define and declare the explicitly typed standard functions
+ *       (e.g., SIN_REAL, SIN_LREAL, ..., ADD_SINT, ADD_INT, ADD_LINT, ...)
+ *       We do not declare/define the overloaded functions
+ *       (SIN, ADD, ...). 
+ *       When handling a call to an overloaded function, the iec2c compiler 
+ *       will determine in stage3 the data type of the parameter being passed, 
+ *       and in stage4 generate the C code to call the correct
+ *       typed standard function.
+ */
+
+/* NOTE on explicit typing of:
+ *           - Table 25 - Standard bit shift functions
+ *           - Table 29 - Character string Functions
+ *
+ *  In section 2.5.1.4 (Typing, overloading, and type conversion) of the IEC 61131-3 (version 2)
+ *  of the standard, it is stated:
+ * "A standard function, [...] is said to be overloaded when it
+ * can operate on input data elements of various types within a generic type designator as defined in
+ * 2.3.2. For instance, an overloaded addition function on generic type ANY_NUM can operate on data
+ * of types LREAL, REAL, DINT, INT, and SINT."
+ * [...]
+ * "When a function which normally represents an overloaded operator is to be typed, i.e., the types
+ * of its inputs and outputs restricted to a particular elementary or derived data type as defined in
+ * 2.3, this shall be done by appending an "underline" character followed by the required type, as
+ * shown in table 21."
+ *
+ * However, this explanation (as well as the example in table 21) only refers to functions where the same
+ * generic data type is used for the single input and the output parameter.
+ * How can we create explicitly types functions when this is not the case?
+ * It does not seem to be covered by the standard.
+ *
+ * For this reason, we do not define the LEN_SINT, LEN_INT, LEN_STRING, LEN_[ANY_INT], LEN_[ANY_STRING] functions...
+ */
+ 
+
+/********************/
+/*   EN/ENO PARAMS  */
+/********************/
+
+#define EN_ENO_PARAMS BOOL EN, BOOL *ENO
+
+#define TEST_EN(TYPENAME)\
+  if (!EN) {\
+    if (ENO != NULL)\
+      *ENO = __BOOL_LITERAL(FALSE);\
+    return __INIT_##TYPENAME;\
+  }\
+  else if (ENO != NULL)\
+    *ENO = __BOOL_LITERAL(TRUE);
+
+#define TEST_EN_COND(TYPENAME, COND)\
+  if (!EN || (COND)) {\
+    if (ENO != NULL)\
+      *ENO = __BOOL_LITERAL(FALSE);\
+    return __INIT_##TYPENAME;\
+  }\
+  else if (ENO != NULL)\
+    *ENO = __BOOL_LITERAL(TRUE);
+
+  
+  
+/*****************************************/  
+/*****************************************/  
+/*  2.5.1.5.1 Type Conversion Functions  */
+/*****************************************/  
+/*****************************************/  
+
+#define __convert_type(from_TYPENAME,to_TYPENAME, oper) \
+static inline to_TYPENAME from_TYPENAME##_TO_##to_TYPENAME(EN_ENO_PARAMS, from_TYPENAME op){\
+  TEST_EN(to_TYPENAME)\
+  return (to_TYPENAME)oper(op);\
+}
+
+
+#define __to_anynum_(from_TYPENAME)   __ANY_NUM_1(__iec_,from_TYPENAME)
+#define __to_anyint_(from_TYPENAME)   __ANY_INT_1(__iec_,from_TYPENAME)
+#define __to_anybit_(from_TYPENAME)   __ANY_BIT_1(__iec_,from_TYPENAME)
+#define __to_anynbit_(from_TYPENAME) __ANY_NBIT_1(__iec_,from_TYPENAME)
+#define __to_anysint_(from_TYPENAME) __ANY_SINT_1(__iec_,from_TYPENAME)
+#define __to_anyuint_(from_TYPENAME) __ANY_UINT_1(__iec_,from_TYPENAME)
+#define __to_anyreal_(from_TYPENAME) __ANY_REAL_1(__iec_,from_TYPENAME)
+#define __to_anydate_(from_TYPENAME) __ANY_DATE_1(__iec_,from_TYPENAME)
+
+/******** [ANY_BIT]_TO_[ANY_NUM | ANT_BIT]   ************/ 
+#define __iec_(to_TYPENAME,from_TYPENAME) __convert_type(from_TYPENAME, to_TYPENAME, __move_##to_TYPENAME)
+__ANY_BIT(__to_anynum_)
+__ANY_BIT(__to_anybit_)
+#undef __iec_
+
+/******** [ANY_INT]_TO_[ANY_NUM | ANT_BIT]   ************/ 
+#define __iec_(to_TYPENAME,from_TYPENAME) __convert_type(from_TYPENAME, to_TYPENAME, __move_##to_TYPENAME)
+__ANY_INT(__to_anynum_)
+__ANY_INT(__to_anybit_)
+#undef __iec_
+
+/******** [ANY_REAL]_TO_[ANY_BIT]   ************/ 
+#define __iec_(to_TYPENAME,from_TYPENAME) __convert_type(from_TYPENAME, to_TYPENAME, __real_to_bit)
+__ANY_REAL(__to_anybit_)
+#undef __iec_
+
+/******** [ANY_REAL]_TO_[ANY_INT]   ************/ 
+#define __iec_(to_TYPENAME,from_TYPENAME) __convert_type(from_TYPENAME, to_TYPENAME, __real_to_sint)
+__ANY_REAL(__to_anysint_)
+#undef __iec_
+#define __iec_(to_TYPENAME,from_TYPENAME) __convert_type(from_TYPENAME, to_TYPENAME, __real_to_uint)
+__ANY_REAL(__to_anyuint_)
+#undef __iec_
+
+/******** [ANY_REAL]_TO_[ANY_REAL]   ************/ 
+#define __iec_(to_TYPENAME,from_TYPENAME) __convert_type(from_TYPENAME, to_TYPENAME, __move_##to_TYPENAME)
+__ANY_REAL(__to_anyreal_)
+#undef __iec_
+
+/******** [ANY_BIT | ANY_INT]_TO_[TIME | ANY_DATE]   ************/ 
+#define __iec_(from_TYPENAME) __convert_type(from_TYPENAME, TIME, __int_to_time)
+__ANY_BIT(__iec_)
+__ANY_INT(__iec_)
+#undef __iec_
+#define __iec_(to_TYPENAME,from_TYPENAME) __convert_type(from_TYPENAME, to_TYPENAME, __int_to_time)
+__ANY_BIT(__to_anydate_)
+__ANY_INT(__to_anydate_)
+#undef __iec_
+
+/******** [ANY_REAL]_TO_[TIME | ANY_DATE]   ************/ 
+#define __iec_(from_TYPENAME) __convert_type(from_TYPENAME, TIME, __real_to_time)
+__ANY_REAL(__iec_)
+#undef __iec_
+#define __iec_(to_TYPENAME,from_TYPENAME) __convert_type(from_TYPENAME, to_TYPENAME, __real_to_time)
+__ANY_REAL(__to_anydate_)
+#undef __iec_
+
+/******** [TIME | ANY_DATE]_TO_[ANY_BIT | ANY_INT]   ************/ 
+#define __iec_(to_TYPENAME,from_TYPENAME) __convert_type(from_TYPENAME, to_TYPENAME, __time_to_int)
+__to_anyint_(TIME)
+__to_anybit_(TIME)
+__ANY_DATE(__to_anyint_)
+__ANY_DATE(__to_anybit_)
+#undef __iec_
+
+/******** [TIME | ANY_DATE]_TO_[ANY_REAL]   ************/ 
+#define __iec_(to_TYPENAME,from_TYPENAME) __convert_type(from_TYPENAME, to_TYPENAME, __time_to_real)
+__to_anyreal_(TIME)
+__ANY_DATE(__to_anyreal_)
+#undef __iec_
+
+
+/******** [ANY_DATE]_TO_[ANY_DATE | TIME]   ************/ 
+/* Not supported: DT_TO_TIME */
+__convert_type(DT, DATE,  __date_and_time_to_date)
+__convert_type(DT, DT,    __move_DT)
+__convert_type(DT, TOD,   __date_and_time_to_time_of_day)
+/* Not supported: DATE_TO_TIME */
+__convert_type(DATE, DATE, __move_DATE)
+/* Not supported: DATE_TO_DT */
+/* Not supported: DATE_TO_TOD */
+/* Not supported: TOD_TO_TIME */
+/* Not supported: TOD_TO_DATE */
+/* Not supported: TOD_TO_DT */
+__convert_type(TOD, TOD, __move_TOD)
+
+
+/******** TIME_TO_[ANY_DATE]   ************/ 
+/* Not supported: TIME_TO_DATE */
+/* Not supported: TIME_TO_DT */
+/* Not supported: TIME_TO_TOD */
+
+/******** TIME_TO_TIME   ************/ 
+__convert_type(TIME, TIME, __move_TIME)
+
+
+/******** [ANY_BIT]_TO_STRING   ************/ 
+__convert_type(BOOL, STRING, __bool_to_string)
+#define __iec_(from_TYPENAME) __convert_type(from_TYPENAME, STRING, __bit_to_string)
+__ANY_NBIT(__iec_)
+#undef __iec_
+
+/******** [ANY_INT]_TO_STRING   ************/ 
+#define __iec_(from_TYPENAME) __convert_type(from_TYPENAME, STRING, __sint_to_string)
+__ANY_SINT(__iec_)
+#undef __iec_
+#define __iec_(from_TYPENAME) __convert_type(from_TYPENAME, STRING, __uint_to_string)
+__ANY_UINT(__iec_)
+#undef __iec_
+
+/******** [ANY_REAL]_TO_STRING   ************/ 
+#define __iec_(from_TYPENAME) __convert_type(from_TYPENAME, STRING, __real_to_string)
+__ANY_REAL(__iec_)
+#undef __iec_
+
+/******** [ANY_DATE]_TO_STRING   ************/ 
+__convert_type(DATE, STRING, __date_to_string)
+__convert_type(DT,   STRING, __dt_to_string)
+__convert_type(TOD,  STRING, __tod_to_string)
+
+/******** TIME_TO_STRING   ************/ 
+__convert_type(TIME, STRING, __time_to_string)
+
+
+/******** STRING_TO_[ANY_BIT]   ************/ 
+__convert_type(STRING, BOOL, __string_to_bool)
+#define __iec_(to_TYPENAME) __convert_type(STRING, to_TYPENAME, __string_to_bit)
+__ANY_NBIT(__iec_)
+#undef __iec_
+
+/******** STRING_TO_[ANY_INT]   ************/ 
+#define __iec_(to_TYPENAME) __convert_type(STRING, to_TYPENAME, __string_to_sint)
+__ANY_SINT(__iec_)
+#undef __iec_
+#define __iec_(to_TYPENAME) __convert_type(STRING, to_TYPENAME, __string_to_uint)
+__ANY_UINT(__iec_)
+#undef __iec_
+
+/******** STRING_TO_[ANY_REAL]   ************/ 
+#define __iec_(to_TYPENAME) __convert_type(STRING, to_TYPENAME, __string_to_real)
+__ANY_REAL(__iec_)
+#undef __iec_
+
+/******** STRING_TO_[ANY_DATE]   ************/ 
+#define __iec_(to_TYPENAME) __convert_type(STRING, to_TYPENAME, __string_to_time)
+__ANY_DATE(__iec_)
+#undef __iec_
+                                   
+/******** STRING_TO_TIME   ************/ 
+__convert_type(STRING, TIME, __string_to_time)
+
+
+/********   TRUNC   ************/ 
+#define __iec_(to_TYPENAME,from_TYPENAME) \
+static inline to_TYPENAME TRUNC__##to_TYPENAME##__##from_TYPENAME(EN_ENO_PARAMS, from_TYPENAME op){\
+  TEST_EN(to_TYPENAME)\
+  return (to_TYPENAME)__move_##to_TYPENAME(op);\
+}
+__ANY_REAL(__to_anyint_)
+#undef __iec_
+
+
+/********   _TO_BCD   ************/ 
+#define __iec_(to_TYPENAME,from_TYPENAME) \
+static inline to_TYPENAME from_TYPENAME##_TO_BCD_##to_TYPENAME(EN_ENO_PARAMS, from_TYPENAME op){\
+  TEST_EN(to_TYPENAME)\
+  return (to_TYPENAME)__uint_to_bcd(op);\
+}
+__ANY_UINT(__to_anynbit_)
+#undef __iec_
+
+
+/********   BCD_TO_   ************/ 
+#define __iec_(to_TYPENAME,from_TYPENAME) \
+static inline to_TYPENAME from_TYPENAME##_BCD_TO_##to_TYPENAME(EN_ENO_PARAMS, from_TYPENAME op){\
+  TEST_EN(to_TYPENAME)\
+  return (to_TYPENAME)__bcd_to_uint(op);\
+}
+__ANY_NBIT(__to_anyuint_)
+#undef __iec_
+
+
+/***********************************/  
+/***********************************/  
+/*  2.5.1.5.2 Numerical Functions  */
+/***********************************/  
+/***********************************/  
+
 /* workaround for va-args limitation on shorter than int params */
 #define VA_ARGS_REAL LREAL
 #define VA_ARGS_LREAL LREAL
@@ -841,11 +937,187 @@
 #define VA_ARGS_TOD TOD
 #define VA_ARGS_DT DT
 
-/*******************************************/
-/*     Arithmetic and bitwise functions    */
-/*******************************************/
+
+#define __numeric(fname,TYPENAME, FUNC) \
+static inline TYPENAME fname##TYPENAME(EN_ENO_PARAMS, TYPENAME op){\
+  TEST_EN(TYPENAME)\
+  return FUNC(op);\
+}
+
+/******************************************************************/
+/***   Table 23 - Standard functions of one numeric variable    ***/
+/******************************************************************/
+
+  /**************/
+  /*    ABS     */
+  /**************/
+/* explicitly typed function */
+#define __iec_(TYPENAME) \
+static inline TYPENAME ABS_##TYPENAME(EN_ENO_PARAMS, TYPENAME op){\
+  TEST_EN(TYPENAME)\
+  if (op < 0)\
+    return -op;\
+  return op;\
+}
+__ANY_REAL(__iec_)
+__ANY_SINT(__iec_)
+#undef __iec_
+
+#define __iec_(TYPENAME) \
+static inline TYPENAME ABS_##TYPENAME(EN_ENO_PARAMS, TYPENAME op){\
+  TEST_EN(TYPENAME)\
+  return op;\
+}
+__ANY_UINT(__iec_)
+#undef __iec_
+
+/* overloaded function */
+#define __iec_(TYPENAME) \
+static inline TYPENAME ABS__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, TYPENAME op){\
+  TEST_EN(TYPENAME)\
+  if (op < 0)\
+    return -op;\
+  return op;\
+}
+__ANY_REAL(__iec_)
+__ANY_SINT(__iec_)
+#undef __iec_
+
+#define __iec_(TYPENAME) \
+static inline TYPENAME ABS__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, TYPENAME op){\
+  TEST_EN(TYPENAME)\
+  return op;\
+}
+__ANY_UINT(__iec_)
+#undef __iec_
+
+
+  /**************/
+  /*    SQRT    */
+  /**************/
+/* explicitly typed function */
+#define __iec_(TYPENAME) \
+__numeric(SQRT_, TYPENAME, sqrt)                             /* explicitly typed function */\
+__numeric(SQRT__##TYPENAME##__, TYPENAME, sqrt)              /* overloaded function */
+__ANY_REAL(__iec_)
+#undef __iec_
+
+  /**************/
+  /*     LN     */
+  /**************/
+#define __iec_(TYPENAME) \
+__numeric(LN_, TYPENAME, log)                             /* explicitly typed function */\
+__numeric(LN__##TYPENAME##__, TYPENAME, log)              /* overloaded function */
+__ANY_REAL(__iec_)
+#undef __iec_
+
+
+  /**************/
+  /*     LOG    */
+  /**************/
+#define __iec_(TYPENAME) \
+__numeric(LOG_, TYPENAME, log10)                             /* explicitly typed function */\
+__numeric(LOG__##TYPENAME##__, TYPENAME, log10)              /* overloaded function */
+__ANY_REAL(__iec_)
+#undef __iec_
+
+  /**************/
+  /*     EXP    */
+  /**************/
+#define __iec_(TYPENAME) \
+__numeric(EXP_, TYPENAME, exp)                             /* explicitly typed function */\
+__numeric(EXP__##TYPENAME##__, TYPENAME, exp)              /* overloaded function */
+__ANY_REAL(__iec_)
+#undef __iec_
+
+
+  /**************/
+  /*     SIN    */
+  /**************/
+#define __iec_(TYPENAME) \
+__numeric(SIN_, TYPENAME, sin)                              /* explicitly typed function */\
+__numeric(SIN__##TYPENAME##__, TYPENAME, sin)               /* overloaded function */
+__ANY_REAL(__iec_)
+#undef __iec_
+
+
+  /**************/
+  /*     COS    */
+  /**************/
+#define __iec_(TYPENAME) \
+__numeric(COS_, TYPENAME, cos)                             /* explicitly typed function */\
+__numeric(COS__##TYPENAME##__, TYPENAME, cos)              /* overloaded function */
+__ANY_REAL(__iec_)
+#undef __iec_
+
+  /**************/
+  /*     TAN    */
+  /**************/
+#define __iec_(TYPENAME) \
+__numeric(TAN_, TYPENAME, tan)                             /* explicitly typed function */\
+__numeric(TAN__##TYPENAME##__, TYPENAME, tan)              /* overloaded function */
+__ANY_REAL(__iec_)
+#undef __iec_
+
+
+  /**************/
+  /*    ASIN    */
+  /**************/
+#define __iec_(TYPENAME) \
+__numeric(ASIN_, TYPENAME, asin)                             /* explicitly typed function */\
+__numeric(ASIN__##TYPENAME##__, TYPENAME, asin)              /* overloaded function */
+__ANY_REAL(__iec_)
+#undef __iec_
+
+  /**************/
+  /*    ACOS    */
+  /**************/
+#define __iec_(TYPENAME) \
+__numeric(ACOS_, TYPENAME, acos)                             /* explicitly typed function */\
+__numeric(ACOS__##TYPENAME##__, TYPENAME, acos)              /* overloaded function */
+__ANY_REAL(__iec_)
+#undef __iec_
+
+  /**************/
+  /*    ATAN    */
+  /**************/
+#define __iec_(TYPENAME) \
+__numeric(ATAN_, TYPENAME, atan)                             /* explicitly typed function */\
+__numeric(ATAN__##TYPENAME##__, TYPENAME, atan)              /* overloaded function */
+__ANY_REAL(__iec_)
+#undef __iec_
+
+
+
+/*****************************************************/
+/***   Table 24 - Standard arithmetic functions    ***/
+/*****************************************************/
+/*
+   Unfortunately, the following does not work!!
+   
+#define NUMARGS(...) (sizeof((int[]){__VA_ARGS__})/sizeof(int))
+
 #define __arith_expand(fname,TYPENAME, OP) \
-static inline TYPENAME fname##TYPENAME(EN_ENO_PARAMS, UINT param_count, TYPENAME op1, ...){\
+#define fname(EN, ENO, ...) fname__(EN, ENO, NUMARGS(__VA_ARGS__), __VA_ARGS__)\
+static inline TYPENAME fname__(EN_ENO_PARAMS, UINT param_count, TYPENAME op1, ...){\
+  va_list ap;\
+  UINT i;\
+  TEST_EN(TYPENAME)\
+  \
+  va_start (ap, op1);         \
+  \
+  for (i = 0; i < param_count - 1; i++){\
+    op1 = op1 OP va_arg (ap, VA_ARGS_##TYPENAME);\
+  }\
+  \
+  va_end (ap);                  \
+  return op1;\
+}
+*/
+
+
+#define __arith_expand(fname,TYPENAME, OP) \
+static inline TYPENAME fname(EN_ENO_PARAMS, UINT param_count, TYPENAME op1, ...){\
   va_list ap;\
   UINT i;\
   TEST_EN(TYPENAME)\
@@ -861,292 +1133,353 @@
 }
 
 #define __arith_static(fname,TYPENAME, OP) \
-static inline TYPENAME fname##TYPENAME(EN_ENO_PARAMS, TYPENAME op1, TYPENAME op2){\
+static inline TYPENAME fname(EN_ENO_PARAMS, TYPENAME op1, TYPENAME op2){\
   TEST_EN(TYPENAME)\
   return op1 OP op2;\
 }
 
-/**************/
-/*     ADD    */
-/**************/
-#define __add_(TYPENAME) __arith_expand(__add_, TYPENAME, + )
-ANY_NUM(__add_)
-
-/**************/
-/*     MUL    */
-/**************/
-#define __mul_(TYPENAME) __arith_expand(__mul_, TYPENAME, * )
-ANY_NUM(__mul_)
-
-/**************/
-/*     SUB    */
-/**************/
-#define __sub_(TYPENAME) __arith_static(__sub_, TYPENAME, - )
-ANY_NUM(__sub_)
-
-/**************/
-/*     DIV    */
-/**************/
-#define __div_(TYPENAME)\
-static inline TYPENAME __div_##TYPENAME(EN_ENO_PARAMS, TYPENAME op1, TYPENAME op2){\
+  /**************/
+  /*     ADD    */
+  /**************/
+#define __iec_(TYPENAME) \
+__arith_expand(ADD_##TYPENAME, TYPENAME, +) 			 /* explicitly typed function */\
+__arith_expand(ADD__##TYPENAME##__##TYPENAME, TYPENAME, +)	 /* overloaded function */
+__ANY_NUM(__iec_)
+#undef __iec_
+
+  /**************/
+  /*     MUL    */
+  /**************/
+#define __iec_(TYPENAME) \
+__arith_expand(MUL_##TYPENAME, TYPENAME, *) 			 /* explicitly typed function */\
+__arith_expand(MUL__##TYPENAME##__##TYPENAME, TYPENAME, *)	 /* overloaded function */
+__ANY_NUM(__iec_)
+#undef __iec_
+
+  /**************/
+  /*     SUB    */
+  /**************/
+#define __iec_(TYPENAME) \
+__arith_expand(SUB_##TYPENAME, TYPENAME, -) 			 /* explicitly typed function */\
+__arith_expand(SUB__##TYPENAME##__##TYPENAME##__##TYPENAME, TYPENAME, -)	 /* overloaded function */
+__ANY_NUM(__iec_)
+#undef __iec_
+
+  /**************/
+  /*     DIV    */
+  /**************/
+/* The explicitly typed standard functions */
+#define __iec_(TYPENAME)\
+static inline TYPENAME DIV_##TYPENAME(EN_ENO_PARAMS, TYPENAME op1, TYPENAME op2){\
   TEST_EN_COND(TYPENAME, op2 == 0)\
   return op1 / op2;\
 }
-ANY_NUM(__div_)
-
-/**************/
-/*     MOD    */
-/**************/
-#define __mod_(TYPENAME) __arith_static(__mod_, TYPENAME, % )
-ANY_INT(__mod_)
-
-/**************/
-/*     AND    */
-/**************/
-__arith_expand(__and_, BOOL, && )
-#define __and_(TYPENAME) __arith_expand(__and_, TYPENAME, & )
-ANY_NBIT(__and_)
-
-/*************/
-/*     OR    */
-/*************/
-__arith_expand(__or_, BOOL, || )
-#define __or_(TYPENAME) __arith_expand(__or_, TYPENAME, |)
-ANY_NBIT(__or_)
-
-/**************/
-/*     XOR    */
-/**************/
-static inline BOOL __xor_BOOL(EN_ENO_PARAMS, UINT param_count, BOOL op1, ...){
-  va_list ap;
-  UINT i;
-  TEST_EN(BOOL)
-
-  va_start (ap, op1);         /* Initialize the argument list.  */
-
-  for (i = 0; i < param_count - 1; i++){
-    BOOL tmp = va_arg (ap, VA_ARGS_BOOL);
-    op1 = (op1 && !tmp) || (!op1 && tmp);
-  }
-
-  va_end (ap);                  /* Clean up.  */
-  return op1;
-}
-#define __xor_(TYPENAME) __arith_expand(__xor_, TYPENAME, ^)
-ANY_NBIT(__xor_)
-
-/**************/
-/*     NOT    */
-/**************/
-static inline BOOL __not_BOOL(EN_ENO_PARAMS, BOOL op1){
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/* The overloaded standard functions */
+#define __iec_(TYPENAME)\
+static inline TYPENAME DIV__##TYPENAME##__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, TYPENAME op1, TYPENAME op2){\
+  TEST_EN_COND(TYPENAME, op2 == 0)\
+  return op1 / op2;\
+}
+__ANY_NUM(__iec_)
+#undef __iec_
+
+  /**************/
+  /*     MOD    */
+  /**************/
+/* The explicitly typed standard functions */
+#define __iec_(TYPENAME)\
+__arith_expand(MOD_##TYPENAME, TYPENAME, %) 			 /* explicitly typed function */\
+__arith_expand(MOD__##TYPENAME##__##TYPENAME##__##TYPENAME, TYPENAME, %)	 /* overloaded function */
+__ANY_INT(__iec_)
+#undef __iec_
+
+  /**************/
+  /*    EXPT    */
+  /**************/
+/* overloaded function */
+#define __iec_(in1_TYPENAME,in2_TYPENAME) \
+static inline in1_TYPENAME EXPT__##in1_TYPENAME##__##in1_TYPENAME##__##in2_TYPENAME\
+  (EN_ENO_PARAMS, in1_TYPENAME IN1, in2_TYPENAME IN2){\
+  TEST_EN(in1_TYPENAME)\
+  return pow(IN1, IN2);\
+}
+#define __in1_anyreal_(in2_TYPENAME)   __ANY_REAL_1(__iec_,in2_TYPENAME)
+__ANY_NUM(__in1_anyreal_)
+#undef __iec_
+
+  
+
+  /***************/
+  /*     MOVE    */
+  /***************/
+/* The explicitly typed standard functions */
+#define __iec_(TYPENAME)\
+static inline TYPENAME MOVE_##TYPENAME(EN_ENO_PARAMS, TYPENAME op1){\
+  TEST_EN(TYPENAME)\
+  return op1;\
+}
+__ANY(__iec_)
+#undef __iec_
+
+/* Overloaded function */
+#define __iec_(TYPENAME)\
+static inline TYPENAME MOVE__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, TYPENAME op1){\
+  TEST_EN(TYPENAME)\
+  return op1;\
+}
+__ANY(__iec_)
+#undef __iec_
+
+
+
+
+
+
+/***********************************/  
+/***********************************/  
+/*  2.5.1.5.3 Bit String Functions */
+/***********************************/  
+/***********************************/  
+
+/****************************************************/
+/***   Table 25 - Standard bit shift functions    ***/
+/****************************************************/
+
+/* We do not delcare explcitly typed versions of the functions in table 25.
+ * See note above regarding explicitly typed functions for more details.
+ */
+#define __in1_anynbit_(in2_TYPENAME)   __ANY_NBIT_1(__iec_,in2_TYPENAME)
+
+#define __shift_(fname, in1_TYPENAME, in2_TYPENAME, OP)\
+static inline in1_TYPENAME fname(EN_ENO_PARAMS, in1_TYPENAME IN, in2_TYPENAME N) {\
+  TEST_EN(in1_TYPENAME)\
+  return IN OP N;\
+}
+
+  /**************/
+  /*     SHL    */
+  /**************/
+#define __iec_(TYPENAME) \
+/* Overloaded function */\
+static inline BOOL SHL__BOOL__##TYPENAME(EN_ENO_PARAMS, BOOL IN, TYPENAME N) { \
+  TEST_EN(BOOL);\
+  return (N==0)? IN : __INIT_BOOL;  /* shifting by N>1 will always introduce a 0 */\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+#define __iec_(in1_TYPENAME,in2_TYPENAME) \
+__shift_(SHL__##in1_TYPENAME##__##in1_TYPENAME##__##in2_TYPENAME, in1_TYPENAME, in2_TYPENAME, << )/* Overloaded function */
+__ANY_INT(__in1_anynbit_)
+#undef __iec_
+
+
+  /**************/
+  /*     SHR    */
+  /**************/
+#define __iec_(TYPENAME) \
+/* Overloaded function */\
+static inline BOOL SHR__BOOL__##TYPENAME(EN_ENO_PARAMS, BOOL IN, TYPENAME N) { \
+  TEST_EN(BOOL);\
+  return (N==0)? IN : __INIT_BOOL;  /* shifting by N>1 will always introduce a 0 */\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+#define __iec_(in1_TYPENAME,in2_TYPENAME) \
+__shift_(SHR__##in1_TYPENAME##__##in1_TYPENAME##__##in2_TYPENAME, in1_TYPENAME, in2_TYPENAME, >> )/* Overloaded function */
+__ANY_INT(__in1_anynbit_)
+#undef __iec_
+
+
+  /**************/
+  /*     ROR    */
+  /**************/
+#define __iec_(TYPENAME) \
+/* Overloaded function */\
+static inline BOOL ROR__BOOL__##TYPENAME(EN_ENO_PARAMS, BOOL IN, TYPENAME N) { \
+  TEST_EN(BOOL);\
+  return IN; /* rotating a single bit by any value N will not change that bit! */\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+#define __iec_(in1_TYPENAME,in2_TYPENAME) \
+static inline in1_TYPENAME ROR__##in1_TYPENAME##__##in1_TYPENAME##__##in2_TYPENAME(EN_ENO_PARAMS, in1_TYPENAME IN, in2_TYPENAME N){\
+  TEST_EN(in1_TYPENAME)\
+  N %= 8*sizeof(in1_TYPENAME);\
+  return (IN >> N) | (IN << (8*sizeof(in1_TYPENAME)-N));\
+}
+__ANY_INT(__in1_anynbit_)
+#undef __iec_
+
+
+  /**************/
+  /*     ROL    */
+  /**************/
+#define __iec_(TYPENAME) \
+/* Overloaded function */\
+static inline BOOL ROL__BOOL__##TYPENAME(EN_ENO_PARAMS, BOOL IN, TYPENAME N) { \
+  TEST_EN(BOOL);\
+  return IN; /* rotating a single bit by any value N will not change that bit! */\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+#define __iec_(in1_TYPENAME,in2_TYPENAME) \
+static inline in1_TYPENAME ROL__##in1_TYPENAME##__##in1_TYPENAME##__##in2_TYPENAME(EN_ENO_PARAMS, in1_TYPENAME IN, in2_TYPENAME N){\
+  TEST_EN(in1_TYPENAME)\
+  N %= 8*sizeof(in1_TYPENAME);\
+  return (IN << N) | (IN >> (8*sizeof(in1_TYPENAME)-N));\
+}
+__ANY_INT(__in1_anynbit_)
+#undef __iec_
+
+
+
+/*********************/
+/***   Table 26    ***/
+/*********************/
+
+  /**************/
+  /*     AND    */
+  /**************/
+__arith_expand(AND_BOOL, BOOL, && )         /* The explicitly typed standard functions */
+__arith_expand(AND__BOOL__BOOL, BOOL, && )  /* Overloaded function */
+
+#define __iec_(TYPENAME) \
+__arith_expand(AND_##TYPENAME, TYPENAME, &)  /* The explicitly typed standard functions */\
+__arith_expand(AND__##TYPENAME##__##TYPENAME, TYPENAME, &)  /* Overloaded function */
+__ANY_NBIT(__iec_)
+#undef __iec_
+
+  /*************/
+  /*     OR    */
+  /*************/
+__arith_expand(OR_BOOL, BOOL, || )         /* The explicitly typed standard functions */
+__arith_expand(OR__BOOL__BOOL, BOOL, || )  /* Overloaded function */
+
+#define __iec_(TYPENAME) \
+__arith_expand(OR_##TYPENAME, TYPENAME, |)  /* The explicitly typed standard functions */\
+__arith_expand(OR__##TYPENAME##__##TYPENAME, TYPENAME, |)  /* Overloaded function */
+__ANY_NBIT(__iec_)
+#undef __iec_
+
+  /**************/
+  /*     XOR    */
+  /**************/
+#define __xorbool_expand(fname) \
+static inline BOOL fname(EN_ENO_PARAMS, UINT param_count, BOOL op1, ...){ \
+  va_list ap; \
+  UINT i; \
+  TEST_EN(BOOL) \
+\
+  va_start (ap, op1);         /* Initialize the argument list.  */ \
+\
+  for (i = 0; i < param_count - 1; i++){ \
+    BOOL tmp = va_arg (ap, VA_ARGS_BOOL); \
+    op1 = (op1 && !tmp) || (!op1 && tmp); \
+  } \
+\
+  va_end (ap);                  /* Clean up.  */ \
+  return op1; \
+}
+
+__xorbool_expand(XOR_BOOL) /* The explicitly typed standard functions */
+__xorbool_expand(XOR__BOOL__BOOL) /* Overloaded function */
+
+#define __iec_(TYPENAME) \
+__arith_expand(XOR_##TYPENAME, TYPENAME, ^) /* The explicitly typed standard functions */\
+__arith_expand(XOR__##TYPENAME##__##TYPENAME, TYPENAME, ^) /* Overloaded function */\
+__ANY_NBIT(__iec_)
+#undef __iec_
+
+
+  /**************/
+  /*     NOT    */
+  /**************/
+/* The explicitly typed standard functions */
+static inline BOOL NOT_BOOL(EN_ENO_PARAMS, BOOL op1){
   TEST_EN(BOOL)
   return !op1;
 }
 
-#define __not_(TYPENAME)\
-static inline TYPENAME __not_##TYPENAME(EN_ENO_PARAMS, TYPENAME op1){\
+/* Overloaded function */
+static inline BOOL NOT__BOOL__BOOL(EN_ENO_PARAMS, BOOL op1){
+  TEST_EN(BOOL)
+  return !op1;
+}
+
+/* The explicitly typed standard functions */
+#define __iec_(TYPENAME)\
+static inline TYPENAME NOT_##TYPENAME(EN_ENO_PARAMS, TYPENAME op1){\
   TEST_EN(TYPENAME)\
   return ~op1;\
 }
-ANY_NBIT(__not_)
-
-/***************/
-/*     MOVE    */
-/***************/
-#define __move_(TYPENAME)\
-static inline TYPENAME __move_##TYPENAME(EN_ENO_PARAMS, TYPENAME op1){\
+__ANY_NBIT(__iec_)
+#undef __iec_
+
+/* Overloaded function */
+#define __iec_(TYPENAME)\
+static inline TYPENAME NOT__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, TYPENAME op1){\
   TEST_EN(TYPENAME)\
-  return op1;\
-}
-ANY(__move_)
-
-/**************/
-/* Binary ops */
-/**************/
-#define __shift_(fname, TYPENAME, OP)\
-static inline TYPENAME fname##TYPENAME(EN_ENO_PARAMS, TYPENAME IN, USINT N) {\
-  TEST_EN(TYPENAME)\
-  return IN OP N;\
-}
-
-#define __shl_(TYPENAME) __shift_(__shl_, TYPENAME, << )
-/* Call previously defined macro for each ANY_NBIT */
-ANY_NBIT(__shl_)
-
-#define __shr_(TYPENAME) __shift_(__shr_, TYPENAME, >> )
-/* Call previously defined macro for each ANY_NBIT */
-ANY_NBIT(__shr_)
-
-#define __ror_(TYPENAME)\
-static inline TYPENAME __ror_##TYPENAME(EN_ENO_PARAMS, TYPENAME IN, USINT N){\
-  TEST_EN(TYPENAME)\
-  N %= 8*sizeof(TYPENAME);\
-  return (IN >> N) | (IN << (8*sizeof(TYPENAME)-N));\
-}
-/* Call previously defined macro for each ANY_NBIT */
-ANY_NBIT(__ror_)
-
-#define __rol_(TYPENAME)\
-static inline TYPENAME __rol_##TYPENAME(EN_ENO_PARAMS, TYPENAME IN, USINT N){\
-  TEST_EN(TYPENAME)\
-  N %= 8*sizeof(TYPENAME);\
-  return (IN << N) | (IN >> (8*sizeof(TYPENAME)-N));\
-}
-/* Call previously defined macro for each ANY_NBIT */
-ANY_NBIT(__rol_)
-
-/*******************************************/
-/*     Arithmetic and bitwise functions    */
-/*******************************************/
-
-#define __numeric(fname,TYPENAME, FUNC) \
-static inline TYPENAME fname##TYPENAME(EN_ENO_PARAMS, TYPENAME op){\
-  TEST_EN(TYPENAME)\
-  return FUNC(op);\
-}
-
-  /**************/
-  /*    ABS     */
-  /**************/
-#define __abs_signed(TYPENAME) \
-static inline TYPENAME __abs_##TYPENAME(EN_ENO_PARAMS, TYPENAME op){\
-  TEST_EN(TYPENAME)\
-  if (op < 0)\
-    return -op;\
-  return op;\
-}
-ANY_REAL(__abs_signed)
-ANY_SINT(__abs_signed)
-
-#define __abs_unsigned(TYPENAME) \
-static inline TYPENAME __abs_##TYPENAME(EN_ENO_PARAMS, TYPENAME op){\
-  TEST_EN(TYPENAME)\
-  return op;\
-}
-ANY_UINT(__abs_unsigned)
-
-  /**************/
-  /*    SQRT    */
-  /**************/
-#define __sqrt_(TYPENAME) __numeric(__sqrt_, TYPENAME, sqrt)
-ANY_REAL(__sqrt_)
-
-  /**************/
-  /*     LN     */
-  /**************/
-#define __ln_(TYPENAME) __numeric(__ln_, TYPENAME, log)
-ANY_REAL(__ln_)
-
-  /**************/
-  /*     LOG    */
-  /**************/
-#define __log_(TYPENAME) __numeric(__log_, TYPENAME, log10)
-ANY_REAL(__log_)
-
-  /**************/
-  /*     EXP    */
-  /**************/
-#define __exp_(TYPENAME) __numeric(__exp_, TYPENAME, exp)
-ANY_REAL(__exp_)
-
-  /**************/
-  /*     SIN    */
-  /**************/
-#define __sin_(TYPENAME) __numeric(__sin_, TYPENAME, sin)
-ANY_REAL(__sin_)
-
-  /**************/
-  /*     COS    */
-  /**************/
-#define __cos_(TYPENAME) __numeric(__cos_, TYPENAME, cos)
-ANY_REAL(__cos_)
-
-  /**************/
-  /*     TAN    */
-  /**************/
-#define __tan_(TYPENAME) __numeric(__tan_, TYPENAME, tan)
-ANY_REAL(__tan_)
-
-  /**************/
-  /*    ASIN    */
-  /**************/
-#define __asin_(TYPENAME) __numeric(__asin_, TYPENAME, asin)
-ANY_REAL(__asin_)
-
-  /**************/
-  /*    ACOS    */
-  /**************/
-#define __acos_(TYPENAME) __numeric(__acos_, TYPENAME, acos)
-ANY_REAL(__acos_)
-
-  /**************/
-  /*    ATAN    */
-  /**************/
-#define __atan_(TYPENAME) __numeric(__atan_, TYPENAME, atan)
-ANY_REAL(__atan_)
-
-  /**************/
-  /*    EXPT    */
-  /**************/
-#define __expt_(TYPENAME)\
-static inline TYPENAME __expt_##TYPENAME(EN_ENO_PARAMS, TYPENAME IN1, LREAL IN2){\
-  TEST_EN(TYPENAME)\
-  return pow(IN1, IN2);\
-}
-ANY_REAL(__expt_)
-
-/**************/
-/* Selection  */
-/**************/
-
-  /**************/
-  /*    SEL     */
-  /**************/
-
-#define __sel_(TYPENAME)\
-static inline TYPENAME __sel_##TYPENAME(EN_ENO_PARAMS, BOOL G, TYPENAME op0, TYPENAME op1){\
+  return ~op1;\
+}
+__ANY_NBIT(__iec_)
+#undef __iec_
+
+
+
+
+
+
+/***************************************************/  
+/***************************************************/  
+/*  2.5.1.5.4  Selection and comparison Functions  */
+/***************************************************/  
+/***************************************************/  
+
+/*********************/
+/***   Table 27    ***/
+/*********************/
+
+
+    /**************/
+    /*    SEL     */
+    /**************/
+
+/* The explicitly typed standard functions */
+#define __iec_(TYPENAME)\
+static inline TYPENAME SEL_##TYPENAME(EN_ENO_PARAMS, BOOL G, TYPENAME op0, TYPENAME op1){\
   TEST_EN(TYPENAME)\
   return G ? op1 : op0;\
 }
-ANY(__sel_)
-
-	/**************/
-	/*   limit    */
-	/**************/
-
-#define __limit_(TYPENAME)\
-static inline TYPENAME __limit_##TYPENAME(EN_ENO_PARAMS, TYPENAME MN, TYPENAME IN, TYPENAME MX){\
+__ANY(__iec_)
+#undef __iec_
+
+/* Overloaded function */
+#define __iec_(TYPENAME)\
+static inline TYPENAME SEL__##TYPENAME##__BOOL__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, BOOL G, TYPENAME op0, TYPENAME op1){\
   TEST_EN(TYPENAME)\
-  return IN > MN ? IN < MX ? IN : MX : MN;\
-}
-
-/* Call previously defined macro for each concerned type */
-ANY_NBIT(__limit_)
-ANY_NUM(__limit_)
-
-#define __limit_time(TYPENAME)\
-static inline TYPENAME __limit_##TYPENAME(EN_ENO_PARAMS, TYPENAME MN, TYPENAME IN, TYPENAME MX){\
-    TEST_EN(TYPENAME)\
-    return __TIME_CMP(IN, MN) > 0 ? /* IN>MN ?*/\
-           __TIME_CMP(IN, MX) < 0 ? /* IN<MX ?*/\
-           IN : MX : MN;\
-}
-
-/* Call previously defined macro for each concerned type */
-ANY_DATE(__limit_time)
-__limit_time(TIME)
-
-static inline STRING __limit_STRING(EN_ENO_PARAMS, STRING MN, STRING IN, STRING MX){
-    TEST_EN(STRING)
-    return __STR_CMP(IN, MN) > 0 ? __STR_CMP(IN, MX) < 0 ? IN : MX : MN;
-}
+  return G ? op1 : op0;\
+}
+__ANY(__iec_)
+#undef __iec_
+
 
     /**************/
     /*     MAX    */
     /**************/
 
 #define __extrem_(fname,TYPENAME, COND) \
-static inline TYPENAME fname##TYPENAME(EN_ENO_PARAMS, UINT param_count, TYPENAME op1, ...){\
+static inline TYPENAME fname(EN_ENO_PARAMS, UINT param_count, TYPENAME op1, ...){\
   va_list ap;\
   UINT i;\
   TEST_EN(TYPENAME)\
@@ -1162,52 +1495,130 @@
   return op1;\
 }
 
-#define __max_num(TYPENAME) __extrem_(__max_,TYPENAME, op1 < tmp)
-ANY_NBIT(__max_num)
-ANY_NUM(__max_num)
-
-__extrem_(__max_, STRING, __STR_CMP(op1,tmp) < 0)
-#define __max_time(TYPENAME) __extrem_(__max_, TYPENAME, __TIME_CMP(op1, tmp) < 0)
-
-/* Call previously defined macro for each concerned type */
-ANY_DATE(__max_time)
-__max_time(TIME)
+/* Max for numerical data types */	
+#define __iec_(TYPENAME) \
+__extrem_(MAX_##TYPENAME,TYPENAME, op1 < tmp) /* The explicitly typed standard functions */\
+__extrem_(MAX__##TYPENAME##__##TYPENAME,TYPENAME, op1 < tmp) /* Overloaded function */
+__ANY_BIT(__iec_)
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/* Max for time data types */	
+#define __iec_(TYPENAME) \
+__extrem_(MAX_##TYPENAME, TYPENAME, __time_cmp(op1, tmp) < 0) /* The explicitly typed standard functions */\
+__extrem_(MAX__##TYPENAME##__##TYPENAME, TYPENAME, __time_cmp(op1, tmp) < 0) /* Overloaded function */
+__ANY_DATE(__iec_)
+__iec_(TIME)
+#undef __iec_
+
+/* Max for string data types */	
+__extrem_(MAX_STRING, STRING, __STR_CMP(op1,tmp) < 0) /* The explicitly typed standard functions */
+__extrem_(MAX__STRING__STRING, STRING, __STR_CMP(op1,tmp) < 0) /* Overloaded function */
 
     /**************/
     /*     MIN    */
     /**************/
-#define __min_num(TYPENAME) __extrem_(__min_, TYPENAME, op1 > tmp)
-ANY_NBIT(__min_num)
-ANY_NUM(__min_num)
-
-__extrem_(__min_, STRING, __STR_CMP(op1,tmp) > 0)
-
-#define __min_time(TYPENAME) __extrem_(__min_, TYPENAME, __TIME_CMP(op1, tmp) > 0)
-
-/* Call previously defined macro for each concerned type */
-ANY_DATE(__min_time)
-__min_time(TIME)
+/* Min for numerical data types */	
+#define __iec_(TYPENAME) \
+__extrem_(MIN_##TYPENAME, TYPENAME, op1 > tmp) /* The explicitly typed standard functions */\
+__extrem_(MIN__##TYPENAME##__##TYPENAME, TYPENAME, op1 > tmp) /* Overloaded function */
+__ANY_NBIT(__iec_)
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/* Min for time data types */	
+#define __iec_(TYPENAME) \
+__extrem_(MIN_##TYPENAME, TYPENAME, __time_cmp(op1, tmp) > 0) /* The explicitly typed standard functions */\
+__extrem_(MIN__##TYPENAME##__##TYPENAME, TYPENAME, __time_cmp(op1, tmp) > 0) /* Overloaded function */
+__ANY_DATE(__iec_)
+__iec_(TIME)
+#undef __iec_
+
+/* Min for string data types */
+__extrem_(MIN_STRING, STRING, __STR_CMP(op1,tmp) > 0) /* The explicitly typed standard functions */
+__extrem_(MIN__STRING__STRING, STRING, __STR_CMP(op1,tmp) > 0) /* Overloaded function */
+
+    /**************/
+    /*   LIMIT    */
+    /**************/
+
+/* Limit for numerical data types */
+#define __iec_(TYPENAME)\
+/* The explicitly typed standard functions */\
+static inline TYPENAME LIMIT_##TYPENAME(EN_ENO_PARAMS, TYPENAME MN, TYPENAME IN, TYPENAME MX){\
+  TEST_EN(TYPENAME)\
+  return IN > MN ? IN < MX ? IN : MX : MN;\
+}\
+/* Overloaded function */\
+static inline TYPENAME LIMIT__##TYPENAME##__##TYPENAME##__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, TYPENAME MN, TYPENAME IN, TYPENAME MX){\
+  TEST_EN(TYPENAME)\
+  return IN > MN ? IN < MX ? IN : MX : MN;\
+}
+__ANY_NBIT(__iec_)
+__ANY_NUM(__iec_)
+#undef __iec_
+
+
+/* Limit for time data types */	
+#define __iec_(TYPENAME)\
+/* The explicitly typed standard functions */\
+static inline TYPENAME LIMIT_##TYPENAME(EN_ENO_PARAMS, TYPENAME MN, TYPENAME IN, TYPENAME MX){\
+    TEST_EN(TYPENAME)\
+    return __time_cmp(IN, MN) > 0 ? /* IN>MN ?*/\
+           __time_cmp(IN, MX) < 0 ? /* IN<MX ?*/\
+           IN : MX : MN;\
+}\
+/* Overloaded function */\
+static inline TYPENAME LIMIT__##TYPENAME##__##TYPENAME##__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, TYPENAME MN, TYPENAME IN, TYPENAME MX){\
+    TEST_EN(TYPENAME)\
+    return __time_cmp(IN, MN) > 0 ? /* IN>MN ?*/\
+           __time_cmp(IN, MX) < 0 ? /* IN<MX ?*/\
+           IN : MX : MN;\
+}
+
+__ANY_DATE(__iec_)
+__iec_(TIME)
+#undef __iec_
+
+/* Limit for string data types */	
+/* The explicitly typed standard functions */
+static inline STRING LIMIT_STRING(EN_ENO_PARAMS, STRING MN, STRING IN, STRING MX){
+    TEST_EN(STRING)
+    return __STR_CMP(IN, MN) > 0 ? __STR_CMP(IN, MX) < 0 ? IN : MX : MN;
+}
+
+/* Overloaded function */
+static inline STRING LIMIT__STRING__STRING__STRING__STRING(EN_ENO_PARAMS, STRING MN, STRING IN, STRING MX){
+    TEST_EN(STRING)
+    return __STR_CMP(IN, MN) > 0 ? __STR_CMP(IN, MX) < 0 ? IN : MX : MN;
+}
+
 
     /**************/
     /*     MUX    */
     /**************/
-#define __mux_(TYPENAME) \
-static inline TYPENAME __mux_##TYPENAME(EN_ENO_PARAMS, UINT param_count, UINT K, ...){\
+/* The standard states that the inputs for SEL and MUX must be named starting off from 0,
+ * unlike remaining functions, that start off at 1.
+ */    
+/* The explicitly typed standard functions */
+#define __in1_anyint_(in2_TYPENAME)   __ANY_INT_1(__iec_,in2_TYPENAME)
+#define __iec_(in1_TYPENAME,in2_TYPENAME) \
+static inline in2_TYPENAME MUX__##in2_TYPENAME##__##in1_TYPENAME##__##in2_TYPENAME(EN_ENO_PARAMS, UINT param_count, in1_TYPENAME K, ...){\
   va_list ap;\
   UINT i;\
-  TYPENAME tmp;\
-  TEST_EN_COND(TYPENAME, K >= param_count)\
-  tmp = __INIT_##TYPENAME;\
+  in2_TYPENAME tmp;\
+  TEST_EN_COND(in2_TYPENAME, K >= param_count)\
+  tmp = __INIT_##in2_TYPENAME;\
   \
   va_start (ap, K);         /* Initialize the argument list.  */\
   \
   for (i = 0; i < param_count; i++){\
     if(K == i){\
-        tmp = va_arg (ap, VA_ARGS_##TYPENAME);\
+        tmp = va_arg (ap, VA_ARGS_##in2_TYPENAME);\
         va_end (ap);                  /* Clean up.  */\
         return tmp;\
     }else{\
-        va_arg (ap, VA_ARGS_##TYPENAME);\
+        va_arg (ap, VA_ARGS_##in2_TYPENAME);\
     }\
   }\
   \
@@ -1215,14 +1626,17 @@
   return tmp;\
 }
 
-ANY(__mux_)
-
-/**************/
-/* Comparison */
-/**************/
+__ANY(__in1_anyint_)
+#undef __iec_
+
+
+/******************************************/
+/***             Table 28               ***/
+/***   Standard comparison functions    ***/
+/******************************************/
 
 #define __compare_(fname,TYPENAME, COND) \
-static inline BOOL fname##TYPENAME(EN_ENO_PARAMS, UINT param_count, TYPENAME op1, ...){\
+static inline BOOL fname(EN_ENO_PARAMS, UINT param_count, TYPENAME op1, ...){\
   va_list ap;\
   UINT i;\
   TEST_EN(BOOL)\
@@ -1247,91 +1661,537 @@
 }
 
 #define __compare_num(fname, TYPENAME, TEST) __compare_(fname, TYPENAME, op1 TEST tmp )
-#define __compare_time(fname, TYPENAME, TEST) __compare_(fname, TYPENAME, __TIME_CMP(op1, tmp) TEST 0)
+#define __compare_time(fname, TYPENAME, TEST) __compare_(fname, TYPENAME, __time_cmp(op1, tmp) TEST 0)
 #define __compare_string(fname, TEST) __compare_(fname, STRING, __STR_CMP(op1, tmp) TEST 0 )
 
 
     /**************/
     /*     GT     */
     /**************/
-
-#define __gt_num(TYPENAME) __compare_num(__gt_, TYPENAME, > )
-ANY_NBIT(__gt_num)
-ANY_NUM(__gt_num)
-
-#define __gt_time(TYPENAME) __compare_time(__gt_, TYPENAME, > )
-ANY_DATE(__gt_time)
-__gt_time(TIME)
-
-__compare_string(__gt_, > )
+/* Comparison for numerical data types */
+#define __iec_(TYPENAME) \
+__compare_num(GT_##TYPENAME, TYPENAME, > ) /* The explicitly typed standard functions */\
+__compare_num(GT__BOOL__##TYPENAME, TYPENAME, > ) /* Overloaded function */
+__ANY_NBIT(__iec_)
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/* Comparison for time data types */	
+#define __iec_(TYPENAME) \
+__compare_time(GT_##TYPENAME, TYPENAME, > ) /* The explicitly typed standard functions */\
+__compare_time(GT__BOOL__##TYPENAME, TYPENAME, > ) /* Overloaded function */
+__ANY_DATE(__iec_)
+__iec_(TIME)
+#undef __iec_
+
+/* Comparison for string data types */	
+__compare_string(GT_STRING, > ) /* The explicitly typed standard functions */
+__compare_string(GT__BOOL__STRING, > ) /* Overloaded function */
 
     /**************/
     /*     GE     */
     /**************/
-
-#define __ge_num(TYPENAME) __compare_num(__ge_, TYPENAME, >= )
-ANY_BIT(__ge_num)
-ANY_NUM(__ge_num)
-
-#define __ge_time(TYPENAME) __compare_time(__ge_, TYPENAME, >= )
-ANY_DATE(__ge_time)
-__ge_time(TIME)
-
-__compare_string(__ge_, >=)
+/* Comparison for numerical data types */
+#define __iec_(TYPENAME) \
+__compare_num(GE_##TYPENAME, TYPENAME, >= ) /* The explicitly typed standard functions */\
+__compare_num(GE__BOOL__##TYPENAME, TYPENAME, >= ) /* Overloaded function */
+__ANY_NBIT(__iec_)
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/* Comparison for time data types */	
+#define __iec_(TYPENAME) \
+__compare_time(GE_##TYPENAME, TYPENAME, >= ) /* The explicitly typed standard functions */\
+__compare_time(GE__BOOL__##TYPENAME, TYPENAME, >= ) /* Overloaded function */
+__ANY_DATE(__iec_)
+__iec_(TIME)
+#undef __iec_
+
+/* Comparison for string data types */	
+__compare_string(GE_STRING, >= ) /* The explicitly typed standard functions */
+__compare_string(GE__BOOL__STRING, >= ) /* Overloaded function */
+
+
 
     /**************/
     /*     EQ     */
     /**************/
-
-#define __eq_num(TYPENAME) __compare_num(__eq_, TYPENAME, == )
-ANY_BIT(__eq_num)
-ANY_NUM(__eq_num)
-
-#define __eq_time(TYPENAME) __compare_time(__eq_, TYPENAME, == )
-ANY_DATE(__eq_time)
-__eq_time(TIME)
-
-__compare_string(__eq_, == )
+/* Comparison for numerical data types */
+#define __iec_(TYPENAME) \
+__compare_num(EQ_##TYPENAME, TYPENAME, == ) /* The explicitly typed standard functions */\
+__compare_num(EQ__BOOL__##TYPENAME, TYPENAME, == ) /* Overloaded function */
+__ANY_NBIT(__iec_)
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/* Comparison for time data types */	
+#define __iec_(TYPENAME) \
+__compare_time(EQ_##TYPENAME, TYPENAME, == ) /* The explicitly typed standard functions */\
+__compare_time(EQ__BOOL__##TYPENAME, TYPENAME, == ) /* Overloaded function */
+__ANY_DATE(__iec_)
+__iec_(TIME)
+#undef __iec_
+
+/* Comparison for string data types */	
+__compare_string(EQ_STRING, == ) /* The explicitly typed standard functions */
+__compare_string(EQ__BOOL__STRING, == ) /* Overloaded function */
+
 
     /**************/
     /*     LT     */
     /**************/
-
-#define __lt_num(TYPENAME) __compare_num(__lt_, TYPENAME, < )
-ANY_BIT(__lt_num)
-ANY_NUM(__lt_num)
-
-#define __lt_time(TYPENAME) __compare_time(__lt_, TYPENAME, < )
-ANY_DATE(__lt_time)
-__lt_time(TIME)
-
-__compare_string(__lt_, < )
+/* Comparison for numerical data types */
+#define __iec_(TYPENAME) \
+__compare_num(LT_##TYPENAME, TYPENAME, < ) /* The explicitly typed standard functions */\
+__compare_num(LT__BOOL__##TYPENAME, TYPENAME, < ) /* Overloaded function */
+__ANY_NBIT(__iec_)
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/* Comparison for time data types */	
+#define __iec_(TYPENAME) \
+__compare_time(LT_##TYPENAME, TYPENAME, < ) /* The explicitly typed standard functions */\
+__compare_time(LT__BOOL__##TYPENAME, TYPENAME, < ) /* Overloaded function */
+__ANY_DATE(__iec_)
+__iec_(TIME)
+#undef __iec_
+
+/* Comparison for string data types */	
+__compare_string(LT_STRING, < ) /* The explicitly typed standard functions */
+__compare_string(LT__BOOL__STRING, < ) /* Overloaded function */
+
 
     /**************/
     /*     LE     */
     /**************/
-
-#define __le_num(TYPENAME) __compare_num(__le_, TYPENAME, <= )
-ANY_BIT(__le_num)
-ANY_NUM(__le_num)
-
-#define __le_time(TYPENAME) __compare_time(__le_, TYPENAME, <= )
-ANY_DATE(__le_time)
-__le_time(TIME)
-
-__compare_string(__le_, <= )
+/* Comparison for numerical data types */
+#define __iec_(TYPENAME) \
+__compare_num(LE_##TYPENAME, TYPENAME, <= ) /* The explicitly typed standard functions */\
+__compare_num(LE__BOOL__##TYPENAME, TYPENAME, <= ) /* Overloaded function */
+__ANY_NBIT(__iec_)
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/* Comparison for time data types */	
+#define __iec_(TYPENAME) \
+__compare_time(LE_##TYPENAME, TYPENAME, <= ) /* The explicitly typed standard functions */\
+__compare_time(LE__BOOL__##TYPENAME, TYPENAME, <= ) /* Overloaded function */
+__ANY_DATE(__iec_)
+__iec_(TIME)
+#undef __iec_
+
+/* Comparison for string data types */	
+__compare_string(LE_STRING, <= ) /* The explicitly typed standard functions */
+__compare_string(LE__BOOL__STRING, <= ) /* Overloaded function */
+
 
     /**************/
     /*     NE     */
     /**************/
-
-#define __ne_num(TYPENAME) __compare_num(__ne_, TYPENAME, != )
-ANY_BIT(__ne_num)
-ANY_NUM(__ne_num)
-
-#define __ne_time(TYPENAME) __compare_time(__ne_, TYPENAME, != )
-ANY_DATE(__ne_time)
-__ne_time(TIME)
-
-__compare_string(__ne_, != )
-
+/* Comparison for numerical data types */
+#define __iec_(TYPENAME) \
+__compare_num(NE_##TYPENAME, TYPENAME, != ) /* The explicitly typed standard functions */\
+__compare_num(NE__BOOL__##TYPENAME##__##TYPENAME, TYPENAME, != ) /* Overloaded function */
+__ANY_NBIT(__iec_)
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/* Comparison for time data types */	
+#define __iec_(TYPENAME) \
+__compare_time(NE_##TYPENAME, TYPENAME, != ) /* The explicitly typed standard functions */\
+__compare_time(NE__BOOL__##TYPENAME##__##TYPENAME, TYPENAME, != ) /* Overloaded function */
+__ANY_DATE(__iec_)
+__iec_(TIME)
+#undef __iec_
+
+/* Comparison for string data types */	
+__compare_string(NE_STRING, != ) /* The explicitly typed standard functions */
+__compare_string(NE__BOOL__STRING__STRING, != ) /* Overloaded function */
+
+
+
+
+
+
+/*********************************************/  
+/*********************************************/  
+/*  2.5.1.5.5   Character string  Functions  */
+/*********************************************/  
+/*********************************************/  
+
+/*************************************/
+/***           Table 29            ***/
+/***  Character string  Functions  ***/
+/*************************************/
+
+/* We do not delcare explcitly typed versions of the functions in table 29.
+ * See note above regarding explicitly typed functions for more details.
+ */
+ 
+
+#define __STR_CMP(str1, str2) memcmp((char*)&str1.body,(char*)&str2.body, str1.len < str2.len ? str1.len : str2.len)
+
+
+    /***************/
+    /*     LEN     */
+    /***************/
+static inline __strlen_t __len(STRING IN) {return IN.len;}
+
+/* A function, with 1 input paramter, implementing a generic OPERATION */
+#define __genoper_1p_(fname,ret_TYPENAME, par_TYPENAME, OPERATION) \
+static inline ret_TYPENAME fname(EN_ENO_PARAMS, par_TYPENAME par1){\
+  TEST_EN(ret_TYPENAME)\
+  return (ret_TYPENAME)OPERATION(par1);\
+}
+
+#define __iec_(TYPENAME) __genoper_1p_(LEN__##TYPENAME##__STRING, TYPENAME, STRING, __len)
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+    /****************/
+    /*     LEFT     */
+    /****************/
+
+static inline STRING __left(STRING IN, __strlen_t L){
+    STRING res;
+    res = __INIT_STRING;
+    L = L < IN.len ? L : IN.len;
+    memcpy(&res.body, &IN.body, L);
+    res.len = L;
+    return res;
+}
+
+#define __iec_(TYPENAME) \
+static inline STRING LEFT__STRING__STRING__##TYPENAME(EN_ENO_PARAMS, STRING str, TYPENAME L){\
+  TEST_EN_COND(STRING, L < 0)\
+  return (STRING)__left(str,L);\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+    /*****************/
+    /*     RIGHT     */
+    /*****************/
+
+static inline STRING __right(STRING IN, __strlen_t L){
+    STRING res;
+    res = __INIT_STRING;
+    L = L < IN.len ? L : IN.len;
+    memcpy(&res.body, &IN.body[IN.len - L], L);
+    res.len = L;
+    return res;
+}
+
+
+#define __iec_(TYPENAME) \
+static inline STRING RIGHT__STRING__STRING__##TYPENAME(EN_ENO_PARAMS, STRING str, TYPENAME L){\
+  TEST_EN_COND(STRING, L < 0)\
+  return (STRING)__right(str,L);\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+    /***************/
+    /*     MID     */
+    /***************/
+
+static inline STRING __mid(STRING IN, __strlen_t L, __strlen_t P){
+    STRING res;
+    res = __INIT_STRING;
+    if(P <= IN.len){
+	    P -= 1; /* now can be used as [index]*/
+	    L = L + P <= IN.len ? L : IN.len - P;
+	    memcpy(&res.body, &IN.body[P] , L);
+        res.len = L;
+    }
+    return res;
+}
+
+#define __iec_(TYPENAME) \
+static inline STRING MID__STRING__STRING__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, STRING str, TYPENAME L, TYPENAME P){\
+  TEST_EN_COND(STRING, L < 0 || P < 0)\
+  return (STRING)__mid(str,L,P);\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+    /******************/
+    /*     CONCAT     */
+    /******************/
+
+static inline STRING CONCAT__STRING__STRING(UINT param_count, ...){
+  UINT i;
+  STRING res;
+  va_list ap;
+  __strlen_t charcount;
+  charcount = 0;
+  res = __INIT_STRING;
+
+  va_start (ap, param_count);         /* Initialize the argument list.  */
+
+  for (i = 0; i < param_count && charcount < STR_MAX_LEN; i++)
+  {
+    STRING tmp = va_arg(ap, STRING);
+    __strlen_t charrem = STR_MAX_LEN - charcount;
+    __strlen_t to_write = tmp.len > charrem ? charrem : tmp.len;
+    memcpy(&res.body[charcount], &tmp.body , to_write);
+    charcount += to_write;
+  }
+
+  res.len = charcount;
+
+  va_end (ap);                  /* Clean up.  */
+  return res;
+}
+
+    /******************/
+    /*     INSERT     */
+    /******************/
+
+static inline STRING __insert(STRING IN1, STRING IN2, __strlen_t P){
+    STRING res;
+    __strlen_t to_copy;
+    res = __INIT_STRING;
+
+    to_copy = P > IN1.len ? IN1.len : P;
+    memcpy(&res.body, &IN1.body , to_copy);
+    P = res.len = to_copy;
+
+    to_copy = IN2.len + res.len > STR_MAX_LEN ? STR_MAX_LEN - res.len : IN2.len;
+    memcpy(&res.body[res.len], &IN2.body , to_copy);
+    res.len += to_copy;
+
+    to_copy = IN1.len - P < STR_MAX_LEN - res.len ? IN1.len - P : STR_MAX_LEN - res.len ;
+    memcpy(&res.body[res.len], &IN1.body[P] , to_copy);
+    res.len += to_copy;
+
+    return res;
+}
+
+#define __iec_(TYPENAME) \
+static inline STRING INSERT__STRING__STRING__STRING__##TYPENAME(EN_ENO_PARAMS, STRING str1, STRING str2, TYPENAME P){\
+  TEST_EN_COND(STRING, P < 0)\
+  return (STRING)__insert(str1,str2,P);\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+    /******************/
+    /*     DELETE     */
+    /******************/
+
+static inline STRING __delete(STRING IN, __strlen_t L, __strlen_t P){
+    STRING res;
+    __strlen_t to_copy;
+    res = __INIT_STRING;
+
+    to_copy = P > IN.len ? IN.len : P-1;
+    memcpy(&res.body, &IN.body , to_copy);
+    P = res.len = to_copy;
+
+    if( IN.len > P + L ){
+        to_copy = IN.len - P - L;
+        memcpy(&res.body[res.len], &IN.body[P + L], to_copy);
+        res.len += to_copy;
+    }
+
+    return res;
+}
+
+#define __iec_(TYPENAME) \
+static inline STRING DELETE__STRING__STRING__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, STRING str, TYPENAME L, TYPENAME P){\
+  TEST_EN_COND(STRING, L < 0 || P < 0)\
+  return (STRING)__delete(str,L,P);\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+    /*******************/
+    /*     REPLACE     */
+    /*******************/
+
+static inline STRING __replace(STRING IN1, STRING IN2, __strlen_t L, __strlen_t P){
+    STRING res;
+    __strlen_t to_copy;
+    res = __INIT_STRING;
+
+    to_copy = P > IN1.len ? IN1.len : P-1;
+    memcpy(&res.body, &IN1.body , to_copy);
+    P = res.len = to_copy;
+
+    to_copy = IN2.len < L ? IN2.len : L;
+
+    if( to_copy + res.len > STR_MAX_LEN )
+       to_copy = STR_MAX_LEN - res.len;
+
+    memcpy(&res.body[res.len], &IN2.body , to_copy);
+    res.len += to_copy;
+
+    P += L;
+    if( res.len <  STR_MAX_LEN && P < IN1.len)
+    {
+        to_copy = IN1.len - P;
+        memcpy(&res.body[res.len], &IN1.body[P] , to_copy);
+        res.len += to_copy;
+    }
+
+    return res;
+}
+
+#define __iec_(TYPENAME) \
+static inline STRING REPLACE__STRING__STRING__STRING__##TYPENAME##__##TYPENAME(EN_ENO_PARAMS, STRING str1, STRING str2, TYPENAME L, TYPENAME P){\
+  TEST_EN_COND(STRING, L < 0 || P < 0)\
+  return (STRING)__replace(str1,str2,L,P);\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+    /****************/
+    /*     FIND     */
+    /****************/
+
+static inline __strlen_t __pfind(STRING* IN1, STRING* IN2){
+    UINT count1 = 0; /* offset of first matching char in IN1 */
+    UINT count2 = 0; /* count of matching char */
+    while(count1 + count2 < IN1->len && count2 < IN2->len)
+    {
+        if(IN1->body[count1 + count2] != IN2->body[count2]){
+            count1 += count2 + 1;
+            count2 = 0;
+        }
+        else {
+            count2++;
+        }
+    }
+    return count2 == IN2->len -1 ? 0 : count1 + 1;
+}
+
+#define __iec_(TYPENAME) \
+static inline TYPENAME FIND__##TYPENAME##__STRING__STRING(EN_ENO_PARAMS, STRING str1, STRING str2){\
+  TEST_EN(TYPENAME)\
+  return (TYPENAME)__pfind(&str1,&str2);\
+}
+__ANY_INT(__iec_)
+#undef __iec_
+
+
+/*********************************************/  
+/*********************************************/  
+/*  2.5.1.5.6  Functions of time data types  */
+/*********************************************/  
+/*********************************************/  
+
+/**************************************/
+/***           Table 30             ***/
+/***  Functions of time data types  ***/
+/**************************************/
+
+
+static inline TIME ADD_TIME(EN_ENO_PARAMS, TIME IN1, TIME IN2){
+  TEST_EN(TIME)
+  return __time_add(IN1, IN2);
+}
+
+static inline TOD ADD_TOD_TIME(EN_ENO_PARAMS, TOD IN1, TIME IN2){
+  TEST_EN(TOD)
+  return __time_add(IN1, IN2);
+}
+
+static inline DT ADD_DT_TIME(EN_ENO_PARAMS, DT IN1, TIME IN2){
+  TEST_EN(DT)
+  return __time_add(IN1, IN2);
+}
+
+static inline TIME SUB_TIME(EN_ENO_PARAMS, TIME IN1, TIME IN2){
+  TEST_EN(TIME)
+  return __time_sub(IN1, IN2);
+}
+
+static inline TIME SUB_DATE_DATE(EN_ENO_PARAMS, DATE IN1, DATE IN2){
+  TEST_EN(TIME)
+  return __time_sub(IN1, IN2);
+}
+
+static inline TOD SUB_TOD_TIME(EN_ENO_PARAMS, TOD IN1, TIME IN2){
+  TEST_EN(TOD)
+  return __time_sub(IN1, IN2);
+}
+
+static inline TIME SUB_TOD_TOD(EN_ENO_PARAMS, TOD IN1, TOD IN2){
+  TEST_EN(TIME)
+  return __time_sub(IN1, IN2);
+}
+
+static inline DT SUB_DT_TIME(EN_ENO_PARAMS, DT IN1, TIME IN2){
+  TEST_EN(DT)
+  return __time_sub(IN1, IN2);
+}
+
+static inline TIME SUB_DT_DT(EN_ENO_PARAMS, DT IN1, DT IN2){
+  TEST_EN(TIME)
+  return __time_sub(IN1, IN2);
+}
+
+
+/***  MULTIME  ***/
+#define __iec_(TYPENAME)\
+static inline TIME MULTIME__TIME__TIME__##TYPENAME(EN_ENO_PARAMS, TIME IN1, TYPENAME IN2){\
+  TEST_EN(TIME)\
+  return __time_mul(IN1, IN2);\
+}
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/***  MUL  ***/
+#define __iec_(TYPENAME)\
+static inline TIME MUL__TIME__TIME__##TYPENAME(EN_ENO_PARAMS, TIME IN1, TYPENAME IN2){\
+  TEST_EN(TIME)\
+  return __time_mul(IN1, IN2);\
+}
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/***  DIVTIME  ***/
+#define __iec_(TYPENAME)\
+static inline TIME DIVTIME__TIME__TIME__##TYPENAME(EN_ENO_PARAMS, TIME IN1, TYPENAME IN2){\
+  TEST_EN(TIME)\
+  return __time_div(IN1, IN2);\
+}
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/***  DIV  ***/
+#define __iec_(TYPENAME)\
+static inline TIME DIV__TIME__TIME__##TYPENAME(EN_ENO_PARAMS, TIME IN1, TYPENAME IN2){\
+  TEST_EN(TIME)\
+  return __time_div(IN1, IN2);\
+}
+__ANY_NUM(__iec_)
+#undef __iec_
+
+/*** CONCAT_DATE_TOD ***/
+static inline DT CONCAT_DATE_TOD(EN_ENO_PARAMS, DATE IN1, TOD IN2){
+  TEST_EN(DT)
+  return __time_add(IN1, IN2);
+}
+
+
+
+/****************************************************/  
+/****************************************************/  
+/*  2.5.1.5.6   Functions of enumerated data types  */
+/****************************************************/  
+/****************************************************/  
+
+/********************************************/
+/***              Table 31                ***/
+/***  Functions of enumerated data types  ***/
+/********************************************/
+
+/* Do we support this? */