diff src/cs/services/atp/atp_cmd.c @ 0:4e78acac3d88

src/{condat,cs,gpf,nucleus}: import from Selenite
author Mychaela Falconia <falcon@freecalypso.org>
date Fri, 16 Oct 2020 06:23:26 +0000
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/cs/services/atp/atp_cmd.c	Fri Oct 16 06:23:26 2020 +0000
@@ -0,0 +1,2638 @@
+/********************************************************************************
+ *
+ * File Name : atp_cmd.c
+ *
+ * Functions handling the command translation between txt and command
+ * (and vice-versa)
+ *
+ * (C) Texas Instruments, all rights reserved
+ *
+ * Version number   : 0.1 - 03-03-2000
+ *
+ * History          : 0.1 - Created by E. Baissus
+ *                    0.2 - Reviewed : More generic AT commands handling and
+ *                                     copy optimization (especially on L2CAP
+ *                                     buffers)
+ *                    0.3 - '+CME ERROR' and '+CMS ERROR' support
+ *
+ * Author           : Eric Baissus : e-baissus@ti.com
+ *
+ *   (C) Copyright 2000 by Texas Instruments Incorporated, All Rights Reserved
+ *
+ ********************************************************************************/
+#include "rv/rv_general.h"
+#include "rvf/rvf_api.h"
+#include "atp/atp_api.h"
+#include "atp/atp_i.h"
+#include "atp/atp_config.h"
+#include "atp/atp_cmd.h"
+#include "rvm/rvm_use_id_list.h"
+#include <string.h>
+
+#ifdef BLUETOOTH
+#include "l2cap_data_hdlr.h"	/* For L2CAP data handling.                     */
+#endif
+
+/********************************************************************************
+* Function name	: dtoa
+*
+* Description   : Convert the digits of the given decimal value to a character 
+*				  string
+*
+* Parameters    : decimal = decimal value to be converted
+*                 ascii_p = string result
+*
+* Return        : Digit number
+*
+* Note          : digit and remainder declared as volatile to avoid bugs due to
+*				  optimization.
+*
+* History       : 0.1 (25-August-2000) - Created
+*
+*********************************************************************************/
+UINT8 dtoa (UINT16         decimal,
+			T_ATP_TXT_CMD  ascii_p)
+{
+    /* Declare local variables.                                                 */
+	T_ATP_TXT_CMD    str_p     = ascii_p;
+	volatile UINT8   digit     = 0x00;
+	volatile UINT16  remainder = decimal;
+
+/***************************** dtoa function begins *****************************/
+
+	/* Check to see if the string result is non-null.                           */
+	if (str_p == NULL)
+	{
+		return (0x00);
+	}
+
+	/* Convert the fifth digit: remainder = [65535,9999[.                       */
+	for (digit = 0x00;
+		 remainder > 9999;
+		 remainder -= 10000,
+		 digit++)
+	{
+	}
+	if (digit > 0x00)
+	{
+		*str_p++ = (INT8) ('0' + digit);
+	}
+
+	/* Convert the fourth digit: remainder = [9xxx,999[.                        */
+	for (digit = 0x00;
+		 remainder > 999;
+		 remainder -= 1000,
+		 digit++)
+	{
+	}
+	if ((digit > 0x00) || (str_p != ascii_p))
+	{
+		*str_p++ = (INT8) ('0' + digit);
+	}
+
+	/* Convert the third digit: remainder = [9xx,99[.                           */
+	for (digit = 0x00;
+		 remainder > 99;
+		 remainder -= 100,
+		 digit++)
+	{
+	}
+	if ((digit > 0x00) || (str_p != ascii_p))
+	{
+		*str_p++ = (INT8) ('0' + digit);
+	}
+
+	/* Convert the second digit: remainder = [9x,9[.                            */
+	for (digit = 0x00;
+		 remainder > 9;
+		 remainder -= 10,
+		 digit++)
+	{
+	}
+	if ((digit > 0x00) || (str_p != ascii_p))
+	{
+		*str_p++ = (INT8) ('0' + digit);
+	}
+
+	/* Convert the last digit: remainder = [9,0].                               */
+	*str_p++ = (INT8) ('0' + remainder);
+
+	/* Return the length of the string.                                         */
+	return ((UINT8) (str_p - ascii_p));
+
+} /**************************** End of dtoa function ****************************/
+
+
+/********************************************************************************
+* Function name : atp_cmd_init_dce_info
+*
+* Description   : Initialize the whole of the information stored in the 'DCE 
+*                 information' structure
+*
+* Parameter     : dce_info_p = pointer on the DCE information
+*
+* Return        : RV_OK
+*
+* History       : 0.1 (28-August-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_cmd_init_dce_info (T_ATP_DCE_INFO  *dce_info_p)
+{
+    /* Declare a local variable.                                                */
+	UINT8  count = 0;
+
+/********************* atp_cmd_init_dce_info function begins ********************/
+
+	rvf_send_trace ("ATP : DCE information initialized ",
+					34,
+					NULL_PARAM,
+					RV_TRACE_LEVEL_DEBUG_LOW,
+					ATP_USE_ID);
+
+	/* Define the default command line termination character (See ITU-T         */
+	/* Recommendation V.250 ter page 21).                                       */
+	dce_info_p->cr_character = ATP_CR_CHARACTER;
+
+	/* Define the default response formatting character (See ITU-T              */
+	/* Recommendation V.250 ter page 22).                                       */
+	dce_info_p->lf_character = ATP_LF_CHARACTER;
+
+	/* Define the default command line editing character (See ITU-T             */
+	/* Recommendation V.250 ter page 22).                                       */
+	dce_info_p->bs_character = ATP_BS_CHARACTER;
+
+	/* Define the escape sequence (See ITU-T Recommendation V.250 ter page 24). */
+	memcpy ((void *) (dce_info_p->escape_sequence),
+			ATP_ESCAPE_SEQUENCE,
+			MAX_NB_OF_CHARACTER_FOR_END_SEQUENCE);
+
+	/* Define the command echo (See ITU-T Recommendation V.250 ter page 23).    */
+	dce_info_p->echo_mode = ECHO_OFF;
+
+	/* Define the result code suppression (See ITU-T Recommendation V.250 ter   */
+	/* page 23).                                                                */
+	dce_info_p->result_code_mode = RESULT_CODE_ON;
+
+	/* Define the DCE response format (See ITU-T Recommendation V.250 ter page  */
+	/* 24).                                                                     */
+	dce_info_p->verbose_mode = ATP_VERBOSE_1;
+
+	/* Initialize the masks associated with the commands executed by the ATP.   */
+	memset ((void *) (dce_info_p->dce_mask),
+			0x00,
+			sizeof (T_ATP_DCE_MASK));
+
+	/* Initialize pointers on temporary buffers containing the escape sequence. */
+	for (count = 0;
+		 count < MAX_NB_OF_CHARACTER_FOR_END_SEQUENCE;
+		 count++)
+	{
+
+		/* Initialize the pointer on a temporary data buffer containing part of */
+		/* the assumed escape sequence.                                         */
+		dce_info_p->escape_sequence_tmp_buffer_p[count] = NULL;
+
+		/* Initialize the number of data included in buffer hereinabove.        */
+		dce_info_p->length_of_escape_sequence_tmp_buffer_p[count] = 0;
+	}
+	dce_info_p->nb_plus_received = 0;
+	return (RV_OK);
+
+} /******************** End of atp_cmd_init_dce_info function *******************/
+
+
+/********************************************************************************
+* Function name : atp_get_custom_info_from_txt
+*
+* Description   : Etract custom information from a text string
+*
+* Parameters    : text_pp = string containing the command (0-terminated)
+*                 cmd_format = related structure
+*                 mb_id = memory bank used to get the custom command information
+*                         buffer
+*                 termination_char = termination character ('\x00' or <CR>)
+*
+* Return        : cmd_info_pp and RV_OK if the command is found,
+*                 RV_NOT_SUPPORTED
+*
+* Note          : Space characters are ignored and may be used freely for formatting
+*                 purposes, unless they are embedded in numeric or string constants
+*                 (See ITU-T Recommendation V.250 ter sections 5.4.2.1 or 5.4.2.2 on
+*                 pages 6 and 7). The DCE shall be capable of accepting at least 40
+*                 characters in the body.
+*
+* History       : 0.1 (29-August-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_get_custom_info_from_txt (T_ATP_TXT_CMD     *text_pp,
+										T_ATP_CMD_FORMAT  cmd_format,
+										T_RVF_MB_ID       mb_id,
+										T_ATP_CMD         **cmd_info_pp,
+										const char        termination_char)
+{
+
+/***************** atp_get_custom_info_from_txt function begins *****************/
+
+	switch (cmd_format)
+	{
+
+		/* Basic syntax command does not expect any <number>.                   */
+		case ATP_NO_PARAM:
+			{
+				*cmd_info_pp = NULL;
+				return (RV_OK);
+			}
+
+		/* Basic syntax command.                                                */
+		case ATP_BASIC_PARAM:
+			{
+
+				/* Allocate memory in order to store the <number> associated    */
+				/* with basic syntax commands. If insufficient resources        */
+				/* available, then report an internal memory error and abort.   */
+				if (rvf_get_buf (mb_id, \
+								 sizeof (T_ATP_BASIC_CMD), \
+								 (void **) cmd_info_pp) == RVF_RED)
+				{
+					atp_error (ATP_ERROR_TX_MB_RED);
+					return (RV_MEMORY_ERR);
+				}
+
+				/* Get the <number> of the basic syntax command (See ITU-T      */
+				/* Recommendation V.250 ter page 5).                            */
+				memset ((void *) (*cmd_info_pp),
+						0x00,
+						sizeof (T_ATP_BASIC_CMD));
+				ATP_GET_NUMBER (*text_pp,
+								&((*((T_ATP_BASIC_CMD **) cmd_info_pp))->number),
+								'0');
+				return (RV_OK);
+			}
+
+		/* Dial.                                                                */
+		case ATP_DIAL_PARAM:
+			{
+
+				/* Allocate memory in order to store the <dial_string>          */
+				/* associated with the dial. If insufficient resources          */
+				/* available, then report an internal memory error and abort.   */
+				if (rvf_get_buf (mb_id, \
+								 sizeof (T_ATP_DIAL), \
+								 (void **) cmd_info_pp) == RVF_RED)
+				{
+					atp_error (ATP_ERROR_TX_MB_RED);
+					return (RV_MEMORY_ERR);
+				}
+
+				/* Get the <dial_string> of the dial (See ITU-T Recommendation  */
+				/* V.250 ter page 31). All characters appearing on the same     */
+				/* command line after D are considered part of the call         */
+				/* addressing information to be signalled to the network, or    */
+				/* modifiers used to control the signalling process, up to a    */
+				/* semicolon character or the end of the command line. If the   */
+				/* <dial_string> is terminated by a semicolon, the DCE does not */
+				/* start the call origination procedure, but instead returns to */
+				/* command state after completion of the signalling of call     */
+				/* addressing information to the network.                       */
+				memset ((void *) (*cmd_info_pp),
+						0x00,
+						sizeof (T_ATP_DIAL));
+				ATP_GET_DIAL_STRING (*text_pp,
+									 (*((T_ATP_DIAL **) cmd_info_pp))->dial_string_p,
+									 &((*((T_ATP_DIAL **) cmd_info_pp))->dial_string_length),
+									 &((*((T_ATP_DIAL **) cmd_info_pp))->call_type),
+									 termination_char);
+				return (RV_OK);
+			}
+
+		/* S-parameter.                                                         */
+		case ATP_S_PARAM:
+			{
+
+				/* Allocate memory in order to store the <value> associated     */
+				/* with S-parameters. If insufficient resources available, then */
+				/* report an internal memory error and abort.                   */
+				if (rvf_get_buf (mb_id, \
+								 sizeof (T_ATP_S_PARAM), \
+								 (void **) cmd_info_pp) == RVF_RED)
+				{
+					atp_error (ATP_ERROR_TX_MB_RED);
+					return (RV_MEMORY_ERR);
+				}
+
+				/* Get the characters that immediately follow                   */
+				/* <parameter_number>. "=" is used to set the indicated         */
+				/* S-parameter to a new value (See ITU-T Recommendation V.250   */
+				/* ter page 5). Note that if no value is given, the S-parameter */
+				/* specified may be set to 0, or an ERROR result code issued    */
+				/* and the stored value left unchanged. "?" is used to read the */
+				/* current value of the indicated S-parameter (See ITU-T        */
+				/* Recommendation V.250 ter page 5).                            */
+				memset ((void *) (*cmd_info_pp),
+						0x00,
+						sizeof (T_ATP_S_PARAM));
+				ATP_GET_PARAMETER_VALUE (*text_pp,
+										 &((*((T_ATP_S_PARAM **) cmd_info_pp))->s_operator),
+										 &((*((T_ATP_S_PARAM **) cmd_info_pp))->value));
+				return (RV_OK);
+			}
+
+		/* Extended syntax command does not expect any <value>.                 */
+		case ATP_NO_EXTENDED_PARAM:
+			{
+
+				/* Allocate memory in order to store the <value> associated     */
+				/* with extended syntax commands. If insufficient resources     */
+				/* available, then report an internal memory error and abort.   */
+				if (rvf_get_buf (mb_id, \
+								 sizeof (T_ATP_NO_SUBPARAMETER), \
+								 (void **) cmd_info_pp) == RVF_RED)
+				{
+					atp_error (ATP_ERROR_TX_MB_RED);
+					return (RV_MEMORY_ERR);
+				}
+
+				/* Get the characters that immediately follow <name>. "=?" is   */
+				/* used to test whether the extended syntax command is          */
+				/* implemented in the DCE (See ITU-T Recommendation V.250 ter   */
+				/* page 9).                                                     */
+				memset ((void *) (*cmd_info_pp),
+						0x00,
+						sizeof (T_ATP_NO_SUBPARAMETER));
+				ATP_GET_OPERATOR (*text_pp,
+								  &((*((T_ATP_NO_SUBPARAMETER **) cmd_info_pp))->extended_operator));
+				return (RV_OK);
+			}
+
+		/* Extended syntax command whose subparameter is a numeric constant.    */
+		case ATP_SINGLE_EXTENDED_PARAM:
+			{
+
+				/* Allocate memory in order to store the <value> associated     */
+				/* with extended syntax commands. If insufficient resources     */
+				/* available, then report an internal memory error and abort.   */
+				if (rvf_get_buf (mb_id, \
+								 sizeof (T_ATP_SINGLE_SUBPARAMETER), \
+								 (void **) cmd_info_pp) == RVF_RED)
+				{
+					atp_error (ATP_ERROR_TX_MB_RED);
+					return (RV_MEMORY_ERR);
+				}
+
+				/* Get the characters that immediately follow <name>. "=" is    */
+				/* used to set the indicated extended syntax command to a new   */
+				/* value (See ITU-T Recommendation V.250 ter page 8). Note that */
+				/* if no value is given, the extended syntax command specified  */
+				/* may be set to 0. "?" is used to read the current value of    */
+				/* the indicated extended syntax command (See ITU-T             */
+				/* Recommendation V.250 ter page 9). "=?" is used to test       */
+				/* whether the extended syntax command is implemented in the    */
+				/* DCE.                                                         */
+				memset ((void *) (*cmd_info_pp),
+						0x00,
+						sizeof (T_ATP_SINGLE_SUBPARAMETER));
+				ATP_GET_VALUE (*text_pp,
+							   &((*((T_ATP_SINGLE_SUBPARAMETER **) cmd_info_pp))->extended_operator),
+							   &((*((T_ATP_SINGLE_SUBPARAMETER **) cmd_info_pp))->value),
+							   termination_char);
+				return (RV_OK);
+			}
+
+		/* Keypad control command.                                              */
+		case ATP_PLUS_CKPD_PARAM:
+			{
+
+				/* Allocate memory in order to store the <keys>,<time> and      */
+				/* <pause> associated with +CKPD extended syntax command. If    */
+				/* insufficient resources available, then report an internal    */
+				/* memory error and abort.                                      */
+				if (rvf_get_buf (mb_id, \
+								 sizeof (T_ATP_PLUS_CKPD), \
+								 (void **) cmd_info_pp) == RVF_RED)
+				{
+					atp_error (ATP_ERROR_TX_MB_RED);
+					return (RV_MEMORY_ERR);
+				}
+
+				/* Get the <keys>,<time> and <pause> of the keypad control      */
+				/* command. Note that <keys> shall consist of a string constant */
+				/* and <time> and <pause> shall consist of numeric constants in */
+				/* tenths of a second (See ETS 300 916 (GSM 07.07) Version      */
+				/* 5.8.1 page 62). "=?" is used to test whether the extended    */
+				/* syntax command is implemented in the DCE.                    */
+				memset ((void *) (*cmd_info_pp),
+						0x00,
+						sizeof (T_ATP_PLUS_CKPD));
+				ATP_GET_CKPD_PARAM (*text_pp,
+									(*((T_ATP_PLUS_CKPD **) cmd_info_pp)),
+									termination_char);
+				return (RV_OK);
+			}
+
+		/* AT command undefined or not supported for now.                       */
+		default:
+			{
+				break;
+			}
+	}
+	return (RV_NOT_SUPPORTED);
+
+} /**************** End of atp_get_custom_info_from_txt function ****************/
+
+
+/********************************************************************************
+* Function name : atp_translate_raw_data_to_cmd
+*
+* Description   : Translate raw data into an interpreted command
+*
+* Parameters    : dce_info_p = pointer on the port structure
+*                 text_p = raw data containing the command (0-terminated)
+*                 cmd_type = command type of the text
+*                 cmd_type_p = type of the interpreted command
+*                 cmd_nb_p = binary related code of the interpreted command
+*                 skip_prefix = indicate whether the prefix shall be skipped in
+*                               order to proceed the translation
+*                 mb_id = memory bank used to get the custom command information
+*                         buffer
+*                 cmd_info_pp = pointer on the custom command information structure
+*
+* Return        : RV_OK,
+*                 RV_NOT_SUPPORTED if the command is not recognized,
+*                 RV_INVALID_PARAMETER
+*
+* Note          : A command line is made up of three elements: the prefix, the body,
+*                 and the termination character. The command line prefix consists of
+*                 the characters "AT" or "at", or, to repeat the execution of the 
+*                 previous command line, the characters "A/" or "a/". The body is
+*                 made up of individual commands. Space characters are ignored and
+*                 may be used freely for formatting purposes, unless they are embedded
+*                 in numeric or string constants (See ITU-T Recommendation V.250 ter
+*                 sections 5.4.2.1 or 5.4.2.2 on pages 6 and 7). The termination
+*                 character may not appear in the body. The DCE shall be capable of
+*                 accepting at least 40 characters in the body. Note that the
+*                 termination character may be selected by a user option (parameter
+*                 S3), the default being CR.
+*
+* History       : 0.1 (25-August-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_translate_raw_data_to_cmd (T_ATP_PORT_STRUCT  *port_p,
+										 T_ATP_TXT_CMD      *text_pp,
+										 T_ATP_CMD_TYPE     cmd_type,
+										 T_ATP_CMD_TYPE     *cmd_type_p,
+										 T_ATP_CMD_NB       *cmd_nb_p,
+										 T_RVF_MB_ID        mb_id,
+										 T_ATP_CMD          **cmd_info_pp)
+{
+	/* Declare a local variable.                                                */
+	BOOLEAN  equal = FALSE;
+
+/***************** atp_translate_raw_data_to_cmd function begins ****************/
+
+	/* Check to see if the text command line is valid.                          */
+	if ((text_pp == NULL) || (*text_pp == NULL))
+	{
+		return (RV_INVALID_PARAMETER);
+	}
+
+	/* Initialize information for AT commands do not expect any number or       */
+	/* value.                                                                   */
+	*cmd_info_pp = NULL;
+
+	/* DTE command lines (See ITU-T Recommendation V.250 ter page 4). By the    */
+	/* way, note that the prefix consists of the characters "AT" or "at".       */
+	if ((cmd_type == UNKNOWN) || (cmd_type == AT_CMD))
+	{
+
+		/* Declare a local block variable.                                      */
+		UINT8  prefix_len = 0x00;
+
+		/* Check to see whether the prefix shall be skipped in order to proceed */
+		/* the translation.                                                     */
+		if ((port_p == NULL) || ((port_p->cmd_info).status == NOT_STARTED))
+		{
+			prefix_len = ATP_AT_PREFIX_LEN;
+		}
+
+		/* If the prefix is either "AT" or "at", then the DCE shall proceed the */
+		/* command line body (See ITU-T Recommendation V.250 ter page 4).       */
+		ATP_MEM_I_CMP (ATP_AT_PREFIX,
+					   *text_pp,
+					   prefix_len,
+					   &equal);
+		if (equal == TRUE)
+		{
+
+			/* Declare local block variables.                                   */
+			const char  *cmd_p  = NULL;
+			UINT8       cmd_len = 0x00;
+			
+			for (*cmd_nb_p   = 0x00, \
+				 *cmd_type_p = AT_CMD, \
+				 equal       = FALSE, \
+				 *text_pp    += prefix_len;
+				 *cmd_nb_p < ATP_MAX_NB_OF_AT_COMMANDS;
+				 (*cmd_nb_p)++)
+			{
+
+				/* If needed, skip this empty entry.                            */
+				if (ATP_AT_INFO[*cmd_nb_p][ATP_AT_PARAM_COLUMN] == ATP_CMD_NOT_DEFINED)
+				{
+					continue;
+				}
+
+				/* Get the length of the remainder.                             */
+				cmd_len = (UINT8) (ATP_AT_INFO[*cmd_nb_p + 0x01][ATP_AT_OFFSET_COLUMN] - ATP_AT_INFO[*cmd_nb_p][ATP_AT_OFFSET_COLUMN]);
+				cmd_p   = &(ATP_AT_TXT_TABLE[ATP_AT_INFO[*cmd_nb_p][ATP_AT_OFFSET_COLUMN]]);
+				ATP_MEM_SP_I_CMP (cmd_p,
+								  *text_pp,
+								  &cmd_len,
+								  &equal);
+
+				/* If both AT commands match, then get the custom information.  */
+				if (equal == TRUE)
+				{
+					rvf_send_trace ("ATP : Translate an AT command from text to command ",
+									51,
+									NULL_PARAM,
+									RV_TRACE_LEVEL_DEBUG_LOW,
+									ATP_USE_ID);
+					*text_pp += cmd_len;
+					(void) atp_get_custom_info_from_txt (text_pp,
+														 ATP_AT_INFO[*cmd_nb_p][ATP_AT_PARAM_COLUMN],
+														 mb_id,
+														 cmd_info_pp,
+														 (const char) ((port_p == NULL) ? ('\x00') : ((port_p->dce_info_p)->cr_character)));
+					return (RV_OK);
+				}
+			}
+			return (RV_NOT_SUPPORTED);
+		}
+	}
+
+	/* DCE responses (See ITU-T Recommendation V.250 ter page 10).              */
+	if ((cmd_type == UNKNOWN) || (cmd_type == RESULT_CODE) || \
+		(cmd_type == UNSOLICITED_RESULT))
+	{
+
+		/* Declare local block variables.                                       */
+		const char  *result_code_p  = NULL;
+		UINT8       header_len      = 0x00;
+		UINT8       result_code_len = 0x00;
+
+		/* If verbose responses are enabled, check to see whether leading <CR>  */
+		/* and <LF> characters shall be skipped in order to proceed the         */
+		/* translation.                                                         */
+		if ((port_p != NULL) && ((port_p->dce_info_p)->verbose_mode == ATP_VERBOSE_1))
+		{
+			equal = TRUE;
+			equal &= ((*text_pp)[0x00] == (port_p->dce_info_p)->cr_character);
+			equal &= ((*text_pp)[0x01] == (port_p->dce_info_p)->lf_character);
+
+			/* If leading characters do not match <CR><LF> headers of           */
+			/* information responses, then abort (See ETS 300 916 (GSM 07.07)   */
+			/* Version 4.1 page 13).                                            */
+			if (equal == FALSE)
+			{
+				*cmd_type_p = UNKNOWN;
+				return (RV_NOT_SUPPORTED);
+			}
+			header_len = ATP_RESULT_CODE_HEADER_LEN;
+		}
+		for (*cmd_nb_p   = 0x00, \
+			 *cmd_type_p = RESULT_CODE, \
+			 *text_pp    += header_len;
+			 *cmd_nb_p < ATP_MAX_NB_OF_RESULT_CODES;
+			 (*cmd_nb_p)++)
+		{
+
+			/* If needed, skip this empty entry.                                */
+			if (ATP_RESULT_CODE_INFO[*cmd_nb_p][ATP_RESULT_PARAM_COLUMN] == ATP_RESULT_CODE_NOT_DEFINED)
+			{
+				continue;
+			}
+
+			/* If verbose responses are disabled, then get the length of the    */
+			/* result code from the dedicated list.                             */
+			if ((port_p != NULL) && ((port_p->dce_info_p)->verbose_mode == ATP_VERBOSE_0))
+			{
+				result_code_len = (UINT8) (ATP_RESULT_CODE_INFO[*cmd_nb_p + 0x01][ATP_RESULT_OFFSET_V0_COLUMN] - ATP_RESULT_CODE_INFO[*cmd_nb_p][ATP_RESULT_OFFSET_V0_COLUMN]);
+				result_code_p   = &(ATP_RESULT_CODE_TXT_TABLE_V0[ATP_RESULT_CODE_INFO[*cmd_nb_p][ATP_RESULT_OFFSET_V0_COLUMN]]);
+			}
+			else
+			{
+				result_code_len = (UINT8) (ATP_RESULT_CODE_INFO[*cmd_nb_p + 0x01][ATP_RESULT_OFFSET_V1_COLUMN] - ATP_RESULT_CODE_INFO[*cmd_nb_p][ATP_RESULT_OFFSET_V1_COLUMN]);
+				result_code_p   = &(ATP_RESULT_CODE_TXT_TABLE_V1[ATP_RESULT_CODE_INFO[*cmd_nb_p][ATP_RESULT_OFFSET_V1_COLUMN]]);
+			}
+			ATP_MEM_I_CMP (result_code_p,
+						   *text_pp,
+						   result_code_len,
+						   &equal);
+			if (equal == FALSE)
+			{
+				continue;
+			}
+			rvf_send_trace ("ATP : Translate a result from text to command ",
+							46,
+							NULL_PARAM,
+							RV_TRACE_LEVEL_DEBUG_LOW,
+							ATP_USE_ID);
+			*text_pp += result_code_len;
+			switch (ATP_RESULT_CODE_INFO[*cmd_nb_p][ATP_RESULT_PARAM_COLUMN])
+			{
+
+				/* Extended syntax result code.                                 */
+				case ATP_EXTENDED_RESULT_CODE:
+					{
+
+						/* Allocate memory in order to store the <value>        */
+						/* associated with extended syntax result codes. If     */
+						/* insufficient resources available, then report an     */
+						/* internal memory error and abort.                     */
+						if (rvf_get_buf (mb_id, \
+										 sizeof (T_ATP_SINGLE_RESULT_CODE_VALUE), \
+										 (void **) cmd_info_pp) == RVF_RED)
+						{
+							atp_error (ATP_ERROR_TX_MB_RED);
+							return (RV_MEMORY_ERR);
+						}
+
+						/* Get the value associated with the extended result    */
+						/* codes (See Headset Specification, Section 4.7.3).    */
+						memset ((void *) (*cmd_info_pp),
+								0x00,
+								sizeof (T_ATP_SINGLE_RESULT_CODE_VALUE));
+						ATP_GET_RESULT_CODE_VALUE (*text_pp,
+												   &((*((T_ATP_SINGLE_RESULT_CODE_VALUE **) cmd_info_pp))->value));
+						break;
+					}
+
+				/* +CME ERROR: <error> and +CMS ERROR: <error> result codes.    */
+				case ATP_PLUS_ERROR_RESULT_CODE:
+					{
+
+						/* Allocate memory in order to store the <error>        */
+						/* associated with +CME ERROR or +CMS ERROR result      */
+						/* codes. If insufficient resources available, then     */
+						/* report an internal memory error and abort.           */
+						if (rvf_get_buf (mb_id, \
+										 sizeof (T_ATP_PLUS_ERROR_RESULT_CODE), \
+										 (void **) cmd_info_pp) == RVF_RED)
+						{
+							atp_error (ATP_ERROR_TX_MB_RED);
+							return (RV_MEMORY_ERR);
+						}
+
+						/* Get the <error> associated with the +CME ERROR or    */
+						/* +CMS ERROR result codes.                             */
+						memset ((void *) (*cmd_info_pp),
+								0x00,
+								sizeof (T_ATP_PLUS_ERROR_RESULT_CODE));
+						ATP_PLUS_ERROR_STRING (*text_pp,
+											   ((T_ATP_PLUS_ERROR_RESULT_CODE *) (*cmd_info_pp))->error_p,
+											   &(((T_ATP_PLUS_ERROR_RESULT_CODE *) (*cmd_info_pp))->error_length),
+											   '\x00');
+						break;
+					}
+				default:
+					{
+
+						/* Check to see if any text is associated with the      */
+						/* CONNECT result code.                                 */
+						if (*cmd_nb_p == ATP_CONNECT_NB)
+						{
+
+							/* Declare a local block variable.                  */
+							UINT16  connect_text = 0x0000;
+
+							/* Get the <text> associated with the CONNECT       */
+							/* result codes (See ITU-T Recommendation V.250 ter */
+							/* page 11).                                        */
+							ATP_GET_CONNECT_TXT (*text_pp,
+												 &connect_text);
+
+							/* If no <text> is associated with the CONNECT      */
+							/* result code, then abort.                         */
+							if (connect_text == 0x0000)
+							{
+								break;
+							}
+
+							/* Otherwise, allocate memory in order to store the */
+							/* <text> associated with the CONNECT result code.  */
+							/* If insufficient resources available, then report */
+							/* an internal memory error and abort.              */
+							if (rvf_get_buf (mb_id, \
+											 sizeof (T_ATP_CONNECT_TXT_PARAM), \
+											 (void **) cmd_info_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								return (RV_MEMORY_ERR);
+							}
+
+							/* Return the <text> associated with the CONNECT    */
+							/* result code.                                     */
+							(*((T_ATP_CONNECT_TXT_PARAM **) cmd_info_pp))->value = connect_text;
+							*cmd_nb_p = ATP_CONNECT_TXT_NB;
+						}
+						break;
+					}
+			}
+			return (RV_OK);
+		}
+	}
+	*cmd_type_p = UNKNOWN;
+	return (RV_NOT_SUPPORTED);
+
+} /**************** End of atp_translate_raw_data_to_cmd function ***************/
+
+
+/********************************************************************************
+* Function name : atp_translate_txt_to_cmd
+*
+* Description   : Translate a text string into an interpreted command
+*
+* Parameters    : text_p = text string containing the command (0-terminated)
+*                 cmd_type = command type of the text
+*                 cmd_type_p = type of the interpreted command
+*                 cmd_nb_p = binary related code of the interpreted command
+*                 mb_id = memory bank used to get the custom command information
+*                         buffer
+*                 cmd_info_pp = pointer on the custom command information structure
+*
+* Return        : RV_OK,
+*                 RV_NOT_SUPPORTED if the command is not recognized,
+*                 RV_INVALID_PARAMETER
+*
+* History       : 0.1 (25-August-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_translate_txt_to_cmd (T_ATP_TXT_CMD   text_p,
+									T_ATP_CMD_TYPE  cmd_type,
+									T_ATP_CMD_TYPE  *cmd_type_p,
+									T_ATP_CMD_NB    *cmd_nb_p,
+									T_RVF_MB_ID     mb_id,
+									T_ATP_CMD       **cmd_info_pp)
+{
+	/* Declare a local variable.                                                */
+	T_ATP_RET  ret_status = RV_OK;
+		
+/******************* atp_translate_txt_to_cmd function begins *******************/
+
+	ret_status = atp_translate_raw_data_to_cmd (NULL,
+												&text_p,
+												cmd_type,
+												cmd_type_p,
+												cmd_nb_p,
+												mb_id,
+												cmd_info_pp);
+	return (ret_status);
+
+} /****************** End of atp_translate_txt_to_cmd function ******************/
+
+
+/********************************************************************************
+* Function name : atp_interpret_raw_data
+*
+* Description   : Fetch and interpret (if applicable and DCE mode) a new command
+*                 from the raw data buffer stored available in port_structure
+*
+* Parameters    : port_p = pointer on the port structure
+*                 mb_id = memory bank which the text buffer should be counted on
+*                 cmd_type_p = found command type (if UNKNOWN, should check text
+*                              field)
+*                 cmd_nb_p = found command number
+*                 cmd_info_p = pointer on the custom command information structure
+*                 text_pp = result of interpretation: contain next command to be
+*                           sent by ATP in text format (0-terminated) or NULL if
+*                           no command to be sent
+*                 text_length_p = length of the text command, '\x00' not included
+*
+* Return        : RV_MEMORY_ERR in case 'memory bank' get RED,
+*                 RV_NOT_SUPPORTED if the buffer does not contain proper data,
+*                 RV_OK otherwise
+*
+* Note          : The first data that should be checked are:
+*                 (port_p->cmd_info).cmd_txt_p[(port_p->cmd_info).next_position]
+*                 if (port_p->cmd_info).status = NOT_STARTED, an 'AT' should be
+*                 found
+*                 if the function has processed all the chain, it should set state
+*                 to FINISHED
+*                 (port_p->cmd_info).next_position must be updated by the function
+*
+* History       : 0.1 (25-August-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_interpret_raw_data (T_ATP_PORT_STRUCT	 *port_p,
+								  T_RVF_MB_ID        mb_id,
+								  T_ATP_CMD_TYPE     *cmd_type_p,
+								  T_ATP_CMD_NB       *cmd_nb_p,
+								  T_ATP_CMD          **cmd_info_pp,
+								  T_ATP_TXT_CMD      *text_pp,
+								  UINT16             *text_length_p)
+{
+	/* Declare local variables.                                                 */
+	T_ATP_RET      ret_status = RV_OK;
+	T_ATP_TXT_CMD  text_p     = NULL;
+
+/******************** atp_interpret_raw_data function begins ********************/
+
+	rvf_send_trace ("ATP : Interpret raw data ",
+					25,
+					NULL_PARAM,
+					RV_TRACE_LEVEL_DEBUG_LOW,
+					ATP_USE_ID);
+
+	*text_pp = NULL;
+
+	/* Interpret all AT commands or result codes.                               */
+	while (((port_p->cmd_info).status == NOT_STARTED) || \
+		   (((port_p->cmd_info).status == ON_GOING) && \
+		   ((port_p->cmd_info).cmd_txt_p[(port_p->cmd_info).next_position] != (port_p->dce_info_p)->cr_character) && \
+		   ((port_p->cmd_info).next_position < (port_p->cmd_info).cmd_txt_length)))
+	{
+		text_p     = &((port_p->cmd_info).cmd_txt_p[(port_p->cmd_info).next_position]);
+		ret_status = atp_translate_raw_data_to_cmd (port_p,
+													&text_p,
+													UNKNOWN,
+													cmd_type_p,
+													cmd_nb_p,
+													mb_id,
+													cmd_info_pp);
+		
+		/* If any memory error occurred, then abort.                            */
+		if (ret_status == RV_MEMORY_ERR)
+		{
+			return (RV_MEMORY_ERR);
+		}
+		
+		/* Backup the next character to be interpreted.                         */
+		(port_p->cmd_info).status        = ON_GOING;
+		(port_p->cmd_info).next_position = (UINT16) (text_p - (port_p->cmd_info).cmd_txt_p);
+		
+		/* Intrepret the extracted command.                                     */
+		switch (*cmd_type_p)
+		{
+			
+			/* Interpret the extracted AT command.                              */
+			case AT_CMD:
+				{
+
+					/* Interpret AT commands only if ATP is acting as a DCE.	*/
+					if (port_p->port_config != DCE_CONFIG)
+					{
+						break;
+					}
+
+					/* Set the result of interpretation to NULL.                */
+					*text_pp = NULL;
+					switch (*cmd_nb_p)
+					{
+						
+						/* Answer. Note that any additional commands that       */
+						/* appear after A on the same command line are ignored  */
+						/* (See ITU-T Recommendation V.250 ter page 35).        */
+						case ATP_ATA_NB:
+						
+						/* Dial. Note that all characters appearing on the same */
+						/* command line after D are considered part of the call */
+						/* addressing information to be signalled to the        */
+						/* network, or modifiers used to control the signalling */
+						/* process, up to a semicolon character or the end of   */
+						/* the command line (See ITU-T Recommendation V.250 ter */
+						/* page 31). Note also that the ITU-T Recommendation    */
+						/* V.250 ter does not describe DCE behaviour in some    */
+						/* situations. Thus, additional characters that appear  */
+						/* on the same command line after a semicolon that      */
+						/* terminates dial string are either ignored or         */
+						/* processed as commands (See ITU-T Recommendation      */
+						/* V.250 ter page 14). In our implementation, such      */
+						/* additional characters are ignored.                   */
+						case ATP_ATD_NB:
+						
+						/* Reset to default configuration. Note that any        */
+						/* additional commands that appear on the same command  */
+						/* line after Z are ignored (See ITU-T Recommendation   */
+						/* V.250 ter page  15).                                 */
+						case ATP_ATZ_NB:
+							{
+								(port_p->cmd_info).next_position = (port_p->cmd_info).cmd_txt_length;
+								break;
+							}
+						
+						/* Command echo.                                        */
+						case ATP_ATE_NB:
+							{
+								
+								/* Check to see if the specified value is valid */
+								/* (See ITU-T Recommendation V.250 ter page     */
+								/* 23).                                         */
+								if (((*((T_ATP_ATE_PARAM **) cmd_info_pp))->number != ECHO_OFF) && \
+									((*((T_ATP_ATE_PARAM **) cmd_info_pp))->number != ECHO_ON))
+								{
+									rvf_send_trace ("ATP : Command echo value invalid ",
+													33,
+													NULL_PARAM,
+													RV_TRACE_LEVEL_WARNING,
+													ATP_USE_ID);
+									rvf_free_buf (*cmd_info_pp);
+									*cmd_info_pp = NULL;
+									return (RV_NOT_SUPPORTED);
+								}
+								rvf_send_trace ("ATP : Command echo updated ",
+												27,
+												NULL_PARAM,
+												RV_TRACE_LEVEL_DEBUG_LOW,
+												ATP_USE_ID);
+								(port_p->dce_info_p)->echo_mode = (*((T_ATP_ATE_PARAM **) cmd_info_pp))->number;
+								*cmd_type_p                     = UNKNOWN;
+								rvf_free_buf (*cmd_info_pp);
+								*cmd_info_pp = NULL;
+								
+								/* Interpret the next AT command or result code */
+								/* to come.                                     */
+								continue;
+							}
+							
+						/* Result code suppression.                             */
+						case ATP_ATQ_NB:
+							{
+								
+								/* Check to see if the specified value is valid */
+								/* (See ITU-T Recommendation V.250 ter page     */
+								/* 23).                                         */
+								if (((*((T_ATP_ATQ_PARAM **) cmd_info_pp))->number != RESULT_CODE_ON) && \
+									((*((T_ATP_ATQ_PARAM **) cmd_info_pp))->number != RESULT_CODE_OFF))
+								{
+									rvf_send_trace ("ATP : Result code suppression value invalid ",
+													44,
+													NULL_PARAM,
+													RV_TRACE_LEVEL_WARNING,
+													ATP_USE_ID);
+									rvf_free_buf (*cmd_info_pp);
+									*cmd_info_pp = NULL;
+									return (RV_NOT_SUPPORTED);
+								}
+								rvf_send_trace ("ATP : Result code suppression updated ",
+												38,
+												NULL_PARAM,
+												RV_TRACE_LEVEL_DEBUG_LOW,
+												ATP_USE_ID);
+								(port_p->dce_info_p)->result_code_mode = (*((T_ATP_ATQ_PARAM **) cmd_info_pp))->number;
+								*cmd_type_p                            = UNKNOWN;
+								rvf_free_buf (*cmd_info_pp);
+								*cmd_info_pp = NULL;
+								
+								/* Interpret the next AT command or result code */
+								/* to come.                                     */
+								continue;
+							}
+							
+						/* DCE response format.                                 */
+						case ATP_ATV_NB:
+							{
+								
+								/* Check to see if the specified value is valid */
+								/* (See ITU-T Recommendation V.250 ter page     */
+								/* 24).                                         */
+								if (((*((T_ATP_ATV_PARAM **) cmd_info_pp))->number != ATP_VERBOSE_0) && \
+									((*((T_ATP_ATV_PARAM **) cmd_info_pp))->number != ATP_VERBOSE_1))
+								{
+									rvf_send_trace ("ATP : DCE response format value invalid ",
+													40,
+													NULL_PARAM,
+													RV_TRACE_LEVEL_WARNING,
+													ATP_USE_ID);
+									rvf_free_buf (*cmd_info_pp);
+									*cmd_info_pp = NULL;
+									return (RV_NOT_SUPPORTED);
+								}
+								rvf_send_trace ("ATP : DCE response format updated ",
+												34,
+												NULL_PARAM,
+												RV_TRACE_LEVEL_DEBUG_LOW,
+												ATP_USE_ID);
+								(port_p->dce_info_p)->verbose_mode = (*((T_ATP_ATV_PARAM **) cmd_info_pp))->number;
+								*cmd_type_p                        = UNKNOWN;
+								rvf_free_buf (*cmd_info_pp);
+								*cmd_info_pp = NULL;
+								
+								/* Interpret the next AT command or result code */
+								/* to come.                                     */
+								continue;
+							}
+							
+						/* Command line termination character.                  */
+						case ATP_ATS3_NB:
+							{
+								
+								/* Check to see if the specified value is valid */
+								/* (See ITU-T Recommendation V.250 ter pages 21 */
+								/* and 22).                                     */
+								if ((((*((T_ATP_ATS3_PARAM **) cmd_info_pp))->s_operator != READ_S_PARAM) && \
+									((*((T_ATP_ATS3_PARAM **) cmd_info_pp))->s_operator != SET_S_PARAM)) || \
+									((*((T_ATP_ATS3_PARAM **) cmd_info_pp))->value > 0x007F))
+								{
+									rvf_send_trace ("ATP : Command line termination character invalid ",
+													49,
+													NULL_PARAM,
+													RV_TRACE_LEVEL_WARNING,
+													ATP_USE_ID);
+									rvf_free_buf (*cmd_info_pp);
+									*cmd_info_pp = NULL;
+									return (RV_NOT_SUPPORTED);
+								}
+								rvf_send_trace ("ATP : Command line termination character updated ",
+												49,
+												NULL_PARAM,
+												RV_TRACE_LEVEL_DEBUG_LOW,
+												ATP_USE_ID);
+								
+								/* Check to see if the S-parameter is requested */
+								/* to be set to a new value.                    */
+								if ((*((T_ATP_ATS3_PARAM **) cmd_info_pp))->s_operator == SET_S_PARAM)
+								{
+									(port_p->dce_info_p)->cr_character = (char) ((*((T_ATP_ATS3_PARAM **) cmd_info_pp))->value);
+									*cmd_type_p                        = UNKNOWN;
+									rvf_free_buf (*cmd_info_pp);
+									*cmd_info_pp = NULL;
+									
+									/* Interpret the next AT command or result  */
+									/* code to come.                            */
+									continue;
+								}
+								break;
+							}
+							
+						/* Response formatting character.                       */
+						case ATP_ATS4_NB:
+							{
+								
+								/* Check to see if the specified value is valid */
+								/* (See ITU-T Recommendation V.250 ter page     */
+								/* 22).                                         */
+								if ((((*((T_ATP_ATS4_PARAM **) cmd_info_pp))->s_operator != READ_S_PARAM) && \
+									((*((T_ATP_ATS4_PARAM **) cmd_info_pp))->s_operator != SET_S_PARAM)) || \
+									((*((T_ATP_ATS4_PARAM **) cmd_info_pp))->value > 0x007F))
+								{
+									rvf_send_trace ("ATP : Response formatting character invalid ",
+													44,
+													NULL_PARAM,
+													RV_TRACE_LEVEL_WARNING,
+													ATP_USE_ID);
+									rvf_free_buf (*cmd_info_pp);
+									*cmd_info_pp = NULL;
+									return (RV_NOT_SUPPORTED);
+								}
+								rvf_send_trace ("ATP : Response formatting character updated ",
+												44,
+												NULL_PARAM,
+												RV_TRACE_LEVEL_DEBUG_LOW,
+												ATP_USE_ID);
+								
+								/* Check to see if the S-parameter is requested */
+								/* to be set to a new value.                    */
+								if ((*((T_ATP_ATS4_PARAM **) cmd_info_pp))->s_operator == SET_S_PARAM)
+								{
+									(port_p->dce_info_p)->lf_character = (char) ((*((T_ATP_ATS4_PARAM **) cmd_info_pp))->value);
+									*cmd_type_p                        = UNKNOWN;
+									rvf_free_buf (*cmd_info_pp);
+									*cmd_info_pp = NULL;
+									
+									/* Interpret the next AT command or result  */
+									/* code to come.                            */
+									continue;
+								}
+								break;
+							}
+							
+						/* Command line editing character.                      */
+						case ATP_ATS5_NB:
+							{
+								
+								/* Check to see if the specified value is valid */
+								/* (See ITU-T Recommendation V.250 ter page     */
+								/* 22).                                         */
+								if ((((*((T_ATP_ATS5_PARAM **) cmd_info_pp))->s_operator != READ_S_PARAM) && \
+									((*((T_ATP_ATS5_PARAM **) cmd_info_pp))->s_operator != SET_S_PARAM)) || \
+									((*((T_ATP_ATS5_PARAM **) cmd_info_pp))->value > 0x007F))
+								{
+									rvf_send_trace ("ATP : Command line editing character invalid ",
+													45,
+													NULL_PARAM,
+													RV_TRACE_LEVEL_WARNING,
+													ATP_USE_ID);
+									rvf_free_buf (*cmd_info_pp);
+									*cmd_info_pp = NULL;
+									return (RV_NOT_SUPPORTED);
+								}
+								rvf_send_trace ("ATP : Command line editing character updated ",
+												45,
+												NULL_PARAM,
+												RV_TRACE_LEVEL_DEBUG_LOW,
+												ATP_USE_ID);
+								
+								/* Check to see if the S-parameter is requested */
+								/* to be set to a new value.                    */
+								if ((*((T_ATP_ATS5_PARAM **) cmd_info_pp))->s_operator == SET_S_PARAM)
+								{
+									(port_p->dce_info_p)->bs_character = (char) ((*((T_ATP_ATS5_PARAM **) cmd_info_pp))->value);
+									*cmd_type_p                        = UNKNOWN;
+									rvf_free_buf (*cmd_info_pp);
+									*cmd_info_pp = NULL;
+									
+									/* Interpret the next AT command or result  */
+									/* code to come.                            */
+									continue;
+								}
+								break;
+							}
+
+						/* AT command not recognized.                           */
+						case ATP_MAX_NB_OF_AT_COMMANDS:
+							{
+
+								/* Get the length of the AT command to be       */
+								/* returned, '\x00' not included.               */
+								ATP_GET_UNKNOWN_AT_CMD_LEN (text_p,
+															text_length_p,
+															(port_p->dce_info_p)->cr_character);
+
+								/* Allocate memory in order to return the AT    */
+								/* command ('\x00' included). Note that the     */
+								/* prefix must be taken into account. If        */
+								/* insufficient resources available, then       */
+								/* report an internal memory error and abort.   */
+								if (rvf_get_buf (mb_id, \
+												 ATP_AT_PREFIX_LEN + *text_length_p + 0x0001, \
+												 (void **) text_pp) == RVF_RED)
+								{
+									atp_error (ATP_ERROR_TX_MB_RED);
+									ret_status = RV_MEMORY_ERR;
+									break;
+								}
+
+								/* Copy the prefix into the buffer (See ITU-T   */
+								/* Recommendation V.250 ter page 4).            */
+								memcpy ((void *) *text_pp,
+										ATP_AT_PREFIX,
+										ATP_AT_PREFIX_LEN);
+
+								/* Then, copy the command line body into the    */
+								/* buffer.                                      */
+								memcpy ((void *) &((*text_pp)[ATP_AT_PREFIX_LEN]),
+										(void *) text_p,
+										*text_length_p);
+								(*text_pp)[*text_length_p + ATP_AT_PREFIX_LEN] = '\x00';
+								text_p                                         += *text_length_p;
+								break;
+							}
+
+						/* Other recognized AT commands.                        */
+						default:
+							{
+								break;
+							}
+					}
+
+					/* If the extracted AT command is an extended syntax        */
+					/* command, then update the position of the next character  */
+					/* to be interpreted.                                       */
+					switch (ATP_AT_INFO[*cmd_nb_p][ATP_AT_PARAM_COLUMN])
+					{
+
+						/* AT command undefined or not supported for now.       */
+						case ATP_CMD_NOT_DEFINED:
+
+						/* Extended syntax command does not expect any <value>. */
+						case ATP_NO_EXTENDED_PARAM:
+
+						/* Extended syntax command whose subparameter is a      */
+						/* numeric constant.                                    */
+						case ATP_SINGLE_EXTENDED_PARAM:
+
+						/* Keypad control command.                              */
+						case ATP_PLUS_CKPD_PARAM:
+							{
+								text_p                           += ((*text_p == (port_p->dce_info_p)->cr_character) ? (0x0000) : (0x0001));
+								(port_p->cmd_info).next_position = (UINT16) (text_p - (port_p->cmd_info).cmd_txt_p);
+							}
+						default:
+							{
+								break;
+							}
+					}
+					break;
+				}
+
+			/* Return raw data as received.									    */
+			case UNKNOWN:
+				{
+
+					/* Get the length of raw data to be returned, '\x00' not    */
+					/* included.                                                */
+					ATP_GET_UNKNOWN_CMD_LEN (text_p,
+											 text_length_p,
+											 (port_p->dce_info_p)->cr_character);
+
+					/* Allocate memory in order to return raw data ('\x00'      */
+					/* included). If insufficient resources available, then     */
+					/* report an internal memory error and abort.               */
+					if (rvf_get_buf (mb_id, \
+									 *text_length_p + 0x0001, \
+									 (void **) text_pp) == RVF_RED)
+					{
+						atp_error (ATP_ERROR_TX_MB_RED);
+						ret_status = RV_MEMORY_ERR;
+						break;
+					}
+
+					/* Copy raw data into the buffer.                           */
+					memcpy ((void *) *text_pp,
+							(void *) text_p,
+							*text_length_p);
+					(*text_pp)[*text_length_p] = '\x00';
+					text_p                     += *text_length_p;
+
+					/* Backup the next character to be interpreted.             */
+					(port_p->cmd_info).next_position = (UINT16) (text_p - (port_p->cmd_info).cmd_txt_p);
+					break;
+				}
+
+			/* Else, result codes not supported.                                */
+			case RESULT_CODE:
+			case UNSOLICITED_RESULT:
+			default:
+				{
+					break;
+				}
+		}
+		break;
+	}
+
+	/* Then, check to see whether the interpretation is over.                   */
+	if (((port_p->cmd_info).cmd_txt_p[(port_p->cmd_info).next_position] == (port_p->dce_info_p)->cr_character) || \
+		((port_p->cmd_info).next_position >= (port_p->cmd_info).cmd_txt_length))
+	{
+		(port_p->cmd_info).status = FINISHED;
+	}
+	return (RV_OK);
+
+} /******************* End of atp_interpret_raw_data function *******************/
+
+
+/********************************************************************************
+* Function name	: atp_translate_cmd_to_txt
+*
+* Description   : Translate a command in interpreted format to text format. Buffer 
+*                 containing the command is assumed to be BTF buffer and is freed
+*                 by this function. Text buffer is a BTF buffer
+*
+* Parameters    : cmd_type = type of the command (AT_CMD, RESULT_CODE and 
+*                 UNSOLICITED_RESULT)
+*                 cmd_nb = binary related code
+*                 cmd_info_p = pointer on the custom command information structure
+*                 mb_id = memory bank used to get the text buffer
+*                 text_pp = pointer on the text chain (0-terminated)
+*                 text_length_p = length of the text chain, '\x00' not included
+*
+* Return        : RV_OK,
+*                 RV_NOT_SUPPORTED if the command is not recognized
+*
+* History       : 0.1 (25-August-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_translate_cmd_to_txt (T_ATP_CMD_TYPE  cmd_type,
+									T_ATP_CMD_NB    cmd_nb,
+									T_ATP_CMD       *cmd_info_p,
+									T_RVF_MB_ID     mb_id,
+									T_ATP_TXT_CMD   *text_pp,
+									UINT16          *text_length_p)
+{
+    /* Declare local variables.                                                 */
+	const char  *table     = NULL;
+	UINT16      offset     = 0x0000;
+	T_ATP_RET   ret_status = RV_OK;
+	
+/******************* atp_translate_cmd_to_txt function begins *******************/
+
+	switch (cmd_type)
+	{
+
+		/* Translate AT commands into text.                                     */
+		case AT_CMD:
+			{
+				rvf_send_trace ("ATP : Translate an AT command into text ",
+								40,
+								NULL_PARAM,
+								RV_TRACE_LEVEL_DEBUG_LOW,
+								ATP_USE_ID);
+				table = ATP_AT_TXT_TABLE;
+
+				/* Get the related offset.                                      */
+				offset = ATP_AT_INFO[cmd_nb][ATP_AT_OFFSET_COLUMN];
+
+				/* Get the length of the AT command.                            */
+				*text_length_p = (UINT16) (ATP_AT_PREFIX_LEN + \
+										   ATP_AT_INFO[cmd_nb + 0x01][ATP_AT_OFFSET_COLUMN] - ATP_AT_INFO[cmd_nb][ATP_AT_OFFSET_COLUMN]);
+
+				/* Get the related structure.                                   */
+				switch (ATP_AT_INFO[cmd_nb][ATP_AT_PARAM_COLUMN])
+				{
+
+					/* Basic syntax command does not expect any <number>.       */
+					case ATP_NO_PARAM:
+						{
+
+							/* Create a buffer and copy text string into it     */
+							/* ('\x00' included). If insufficient resources     */
+							/* available, then report an internal memory error  */
+							/* and abort.                                       */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + 0x0001, \
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+	
+							/* Copy the command line prefix into the buffer.    */
+							memcpy ((void *) (*text_pp),
+									ATP_AT_PREFIX,
+									ATP_AT_PREFIX_LEN);
+
+							/* Copy the text into the buffer.                   */
+							memcpy ((void *) &((*text_pp)[ATP_AT_PREFIX_LEN]),
+									(void *) &(table[offset]),
+									*text_length_p - ATP_AT_PREFIX_LEN);
+							(*text_pp)[*text_length_p] = '\x00';
+							break;
+						}
+
+					/* Basic syntax command.                                    */
+					case ATP_BASIC_PARAM:
+						{
+
+							/* Declare a local block variable.                  */
+							T_ATP_BASIC_CMD  *basic_cmd_param_p = (T_ATP_BASIC_CMD *) cmd_info_p;
+
+							/* Create a buffer and copy text string into it     */
+							/* ('\x00' included). If insufficient resources     */
+							/* available, then report an internal memory error  */
+							/* and abort.                                       */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + MAX_BASIC_CMD_PARAM_LEN + 0x0001, \
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+	
+							/* Copy the command line prefix into the buffer.    */
+							memcpy ((void *) (*text_pp),
+									ATP_AT_PREFIX,
+									ATP_AT_PREFIX_LEN);
+
+							/* Copy the text into the buffer.                   */
+							memcpy ((void *) &((*text_pp)[ATP_AT_PREFIX_LEN]),
+									(void *) &(table[offset]),
+									*text_length_p - ATP_AT_PREFIX_LEN);
+
+							/* If needed, copy the buffer describing the        */
+							/* command in interpreted format and free it.       */
+							if (basic_cmd_param_p != NULL)
+							{
+								*text_length_p = (UINT16) (*text_length_p + \
+														   dtoa (basic_cmd_param_p->number,
+																 &((*text_pp)[*text_length_p])));
+							}
+							(*text_pp)[*text_length_p] = '\x00';
+							break;
+						}
+
+					/* Dial.                                                    */
+					case ATP_DIAL_PARAM:
+						{
+
+							/* Declare local block variables.                   */
+							UINT8       dial_param_length = 0x00;
+							UINT8       dial_semicolon    = DATA_CALL;
+							T_ATP_DIAL  *dial_param_p     = (T_ATP_DIAL *) cmd_info_p;
+
+							/* If needed, take the buffer describing the        */
+							/* command in interpreted format into account.      */
+							if (dial_param_p != NULL)
+							{
+								dial_param_length = dial_param_p->dial_string_length;
+								dial_semicolon    = (UINT8) (dial_param_p->call_type);
+							}
+
+							/* Create a buffer and copy text string into it     */
+							/* ('\x00' included). If insufficient resources     */
+							/* available, then report an internal memory error  */
+							/* and abort.                                       */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + dial_param_length + dial_semicolon + 0x0001, \
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+	
+							/* Copy the command line prefix into the buffer.    */
+							memcpy ((void *) (*text_pp),
+									ATP_AT_PREFIX,
+									ATP_AT_PREFIX_LEN);
+
+							/* Copy the text into the buffer.                   */
+							memcpy ((void *) &((*text_pp)[ATP_AT_PREFIX_LEN]),
+									(void *) &(table[offset]),
+									*text_length_p - ATP_AT_PREFIX_LEN);
+
+							/* If needed, copy the buffer describing the        */
+							/* command in interpreted format and free it.       */
+							if (dial_param_p != NULL)
+							{
+								memcpy ((void *) &((*text_pp)[*text_length_p]),
+										(void *) (dial_param_p->dial_string_p),
+										dial_param_length);
+								if (dial_semicolon == VOICE_CALL)
+								{
+									(*text_pp)[*text_length_p + dial_param_length] = ';';
+								}
+								*text_length_p = (UINT16) (*text_length_p + \
+														   dial_param_length + dial_semicolon);
+							}
+							(*text_pp)[*text_length_p] = '\x00';
+							break;
+						}
+
+					/* S-parameter.                                             */
+					case ATP_S_PARAM:
+						{
+
+							/* Declare a local block variable.                  */
+							T_ATP_S_PARAM  *s_param_p = (T_ATP_S_PARAM *) cmd_info_p;
+
+							/* Create a buffer and copy text string into it     */
+							/* ('\x00' included). If insufficient resources     */
+							/* available, then report an internal memory error  */
+							/* and abort.                                       */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + MAX_S_PARAM_LEN + 0x0001,
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+	
+							/* Copy the command line prefix into the buffer.    */
+							memcpy ((void *) (*text_pp),
+									ATP_AT_PREFIX,
+									ATP_AT_PREFIX_LEN);
+
+							/* Copy the text into the buffer.                   */
+							memcpy ((void *) &((*text_pp)[ATP_AT_PREFIX_LEN]),
+									(void *) &(table[offset]),
+									*text_length_p - ATP_AT_PREFIX_LEN);
+
+							/* If needed, copy the buffer describing the        */
+							/* command in interpreted format and free it.       */
+							if (s_param_p == NULL)
+							{
+								(*text_pp)[*text_length_p] = '\x00';
+								break;
+							}
+							switch (s_param_p->s_operator)
+							{
+
+								/* Parameter read command syntax.               */
+								case READ_S_PARAM:
+									{
+										(*text_pp)[(*text_length_p)++] = '?';
+										(*text_pp)[*text_length_p]     = '\x00';
+										break;
+									}
+
+								/* Parameter set command syntax.                */
+								case SET_S_PARAM:
+									{
+										(*text_pp)[(*text_length_p)++] = '=';
+										*text_length_p = (UINT16) (*text_length_p + \
+																   dtoa (s_param_p->value,
+																		 &((*text_pp)[*text_length_p])));
+										(*text_pp)[*text_length_p] = '\x00';
+										break;
+									}
+								default:
+									{
+										rvf_free_buf (*text_pp);
+										*text_pp   = NULL;
+										ret_status = RV_NOT_SUPPORTED;
+										break;
+									}
+							}
+							break;
+						}
+
+					/* Extended syntax command does not expect any <value>.     */
+					case ATP_NO_EXTENDED_PARAM:
+						{
+
+							/* Declare a local block variable.                  */
+							T_ATP_NO_SUBPARAMETER  *extended_cmd_param_p = (T_ATP_NO_SUBPARAMETER *) cmd_info_p;
+
+							/* Create a buffer and copy text string into it     */
+							/* ('\x00' included). If insufficient resources     */
+							/* available, then report an internal memory error  */
+							/* and abort.                                       */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + MAX_NO_SUBPARAMETER_LEN + 0x0001, \
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+	
+							/* Copy the command line prefix into the buffer.    */
+							memcpy ((void *) (*text_pp),
+									ATP_AT_PREFIX,
+									ATP_AT_PREFIX_LEN);
+
+							/* Copy the text into the buffer.                   */
+							memcpy ((void *) &((*text_pp)[ATP_AT_PREFIX_LEN]),
+									(void *) &(table[offset]),
+									*text_length_p - ATP_AT_PREFIX_LEN);
+
+							/* If needed, copy the buffer describing the        */
+							/* command in interpreted format and free it.       */
+							if (extended_cmd_param_p == NULL)
+							{
+								(*text_pp)[*text_length_p] = '\x00';
+								break;
+							}
+							switch (extended_cmd_param_p->extended_operator)
+							{
+
+								/* No subparameter.                             */
+								case NO_SUBPARAMETER:
+									{
+										(*text_pp)[*text_length_p] = '\x00';
+										break;
+									}
+
+								/* Action test command syntax.                  */
+								case TEST_EXTENDED_CMD:
+									{
+										(*text_pp)[(*text_length_p)++] = '=';
+										(*text_pp)[(*text_length_p)++] = '?';
+										(*text_pp)[*text_length_p]     = '\x00';
+										break;
+									}
+								default:
+									{
+										rvf_free_buf (*text_pp);
+										*text_pp   = NULL;
+										ret_status = RV_NOT_SUPPORTED;
+										break;
+									}
+							}
+							break;
+						}
+
+					/* Extended syntax command whose subparameter is a numeric  */
+					/* constant.                                                */
+					case ATP_SINGLE_EXTENDED_PARAM:
+						{
+
+							/* Declare a local block variable.                  */
+							T_ATP_SINGLE_SUBPARAMETER  *extended_cmd_param_p = (T_ATP_SINGLE_SUBPARAMETER *) cmd_info_p;
+
+							/* Create a buffer and copy text string into it     */
+							/* ('\x00' included). If insufficient resources     */
+							/* available, then report an internal memory error  */
+							/* and abort.                                       */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + MAX_SINGLE_SUBPARAMETER_LEN + 0x0001, \
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+	
+							/* Copy the command line prefix into the buffer.    */
+							memcpy ((void *) (*text_pp),
+									ATP_AT_PREFIX,
+									ATP_AT_PREFIX_LEN);
+
+							/* Copy the text into the buffer.                   */
+							memcpy ((void *) &((*text_pp)[ATP_AT_PREFIX_LEN]),
+									(void *) &(table[offset]),
+									*text_length_p - ATP_AT_PREFIX_LEN);
+
+							/* If needed, copy the buffer describing the        */
+							/* command in interpreted format and free it.       */
+							if (extended_cmd_param_p == NULL)
+							{
+								(*text_pp)[*text_length_p] = '\x00';
+								break;
+							}
+							switch (extended_cmd_param_p->extended_operator)
+							{
+
+								/* No subparameter.                             */
+								case NO_SUBPARAMETER:
+									{
+										(*text_pp)[*text_length_p] = '\x00';
+										break;
+									}
+
+								/* Action test command syntax.                  */
+								case TEST_EXTENDED_CMD:
+									{
+										(*text_pp)[(*text_length_p)++] = '=';
+									}
+
+								/* Parameter read command syntax.               */
+								case READ_EXTENDED_CMD:
+									{
+										(*text_pp)[(*text_length_p)++] = '?';
+										(*text_pp)[*text_length_p]     = '\x00';
+										break;
+									}
+
+								/* Parameter set command syntax.                */
+								case SET_EXTENDED_CMD:
+									{
+										(*text_pp)[(*text_length_p)++] = '=';
+										*text_length_p = (UINT16) (*text_length_p + \
+																   dtoa (extended_cmd_param_p->value,
+																		 &((*text_pp)[*text_length_p])));
+										(*text_pp)[*text_length_p] = '\x00';
+										break;
+									}
+								default:
+									{
+										rvf_free_buf (*text_pp);
+										*text_pp   = NULL;
+										ret_status = RV_NOT_SUPPORTED;
+										break;
+									}
+							}
+							break;
+						}
+
+					/* Keypad control command. Note that <keys> is a string of  */
+					/* characters representing keys (See See ETS 300 916 (GSM   */
+					/* 07.07) Version 5.8.1 page 62). Colon character followed  */
+					/* by one character can be used to indicate a manufacturer  */
+					/* specific key not listed here. All characters from a      */
+					/* semicolon character to the next single semicolon         */
+					/* character are treated as alpha entries and are not       */
+					/* converted to key equivalents. All semicolon characters   */
+					/* inside alpha entries should be duplicated in the DTE.    */
+					/* Pause characters "W" and "w" can be used to pause        */
+					/* between key pressings for a time specified by <pause>.   */
+					case ATP_PLUS_CKPD_PARAM:
+						{
+
+							/* Declare local block variables.                   */
+							UINT8            nb_keypressed = 0x00;
+							T_ATP_PLUS_CKPD  *ckpd_param_p = (T_ATP_PLUS_CKPD *) cmd_info_p;
+
+							/* Create a buffer and copy text string into it     */
+							/* ('\x00' included). If insufficient resources     */
+							/* available, then report an internal memory error  */
+							/* and abort.                                       */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + MAX_CKPD_PARAM_LEN + 0x0001, \
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+
+							/* Copy the command line prefix into the buffer.    */
+							memcpy ((void *) (*text_pp),
+									ATP_AT_PREFIX,
+									ATP_AT_PREFIX_LEN);
+
+							/* Copy the text into the buffer.                   */
+							memcpy ((void *) &((*text_pp)[ATP_AT_PREFIX_LEN]),
+									(void *) &(table[offset]),
+									*text_length_p - ATP_AT_PREFIX_LEN);
+
+							/* If needed, copy the buffer describing the        */
+							/* command in interpreted format and free it.       */
+							if (ckpd_param_p == NULL)
+							{
+								(*text_pp)[*text_length_p] = '\x00';
+								break;
+							}
+							switch (ckpd_param_p->extended_operator)
+							{
+
+								/* Action test command syntax.                  */
+								case TEST_EXTENDED_CMD:
+									{
+										(*text_pp)[(*text_length_p)++] = '=';
+										(*text_pp)[(*text_length_p)++] = '?';
+										(*text_pp)[*text_length_p]     = '\x00';
+										break;
+									}
+
+								/* Parameter set command syntax.                */
+								case SET_EXTENDED_CMD:
+									{
+										(*text_pp)[(*text_length_p)++] = '=';
+										(*text_pp)[(*text_length_p)++] = '"';
+
+										/* Store each keypressed into the       */
+										/* buffer.                              */
+										for (nb_keypressed = 0x00;
+											 nb_keypressed < ckpd_param_p->nb_keys;
+											 nb_keypressed++)
+										{
+											(*text_pp)[(*text_length_p)++] = ';';
+											*text_length_p = (UINT16) (*text_length_p + \
+																	   dtoa (ckpd_param_p->keys[nb_keypressed],
+																			 &((*text_pp)[*text_length_p])));
+											(*text_pp)[(*text_length_p)++] = ';';
+										}
+										(*text_pp)[(*text_length_p)++] = '"';
+
+										/* Store <time> subparameter into the   */
+										/* buffer.                              */
+										(*text_pp)[(*text_length_p)++] = ',';
+										if ((ckpd_param_p->pause != DEFAULT_TIME) || \
+											(ckpd_param_p->pause != TIME_DO_NOT_CARE))
+										{
+											*text_length_p = (UINT16) (*text_length_p + \
+																	   dtoa (ckpd_param_p->time,
+																			 &((*text_pp)[*text_length_p])));
+										}
+
+										/* Store <pause> subparameter into the  */
+										/* buffer.                              */
+										(*text_pp)[(*text_length_p)++] = ',';
+										if ((ckpd_param_p->pause != DEFAULT_PAUSE) || \
+											(ckpd_param_p->pause != PAUSE_DO_NOT_CARE))
+										{
+											*text_length_p = (UINT16) (*text_length_p + \
+																	   dtoa (ckpd_param_p->pause,
+																			 &((*text_pp)[*text_length_p])));
+										}
+										(*text_pp)[*text_length_p] = '\x00';
+										break;
+									}
+								default:
+									{
+										rvf_free_buf (*text_pp);
+										*text_pp   = NULL;
+										ret_status = RV_NOT_SUPPORTED;
+										break;
+									}
+							}
+							break;
+						}
+					default:
+						{
+							rvf_send_trace ("ATP : Received an unknown command ",
+											34,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+							*text_pp   = NULL;
+							ret_status = RV_NOT_SUPPORTED;
+							break;
+						}
+				}
+				break;
+			}
+
+		/* Translate DCE responses into text (See ITU-T Recommendation V.250    */
+		/* ter page 10).                                                        */
+		case RESULT_CODE:
+		case UNSOLICITED_RESULT:
+			{
+				rvf_send_trace ("ATP : Translate a result into text ",
+								35,
+								NULL_PARAM,
+								RV_TRACE_LEVEL_DEBUG_LOW,
+								ATP_USE_ID);
+				table = ATP_RESULT_CODE_TXT_TABLE_V1;
+
+				/* Get the related offset.                                      */
+				offset = ATP_RESULT_CODE_INFO[cmd_nb][ATP_RESULT_OFFSET_V1_COLUMN];
+
+				/* Get the length of the result code.                           */
+				*text_length_p = (UINT16) (ATP_RESULT_CODE_INFO[cmd_nb + 0x01][ATP_RESULT_OFFSET_V1_COLUMN] - ATP_RESULT_CODE_INFO[cmd_nb][ATP_RESULT_OFFSET_V1_COLUMN]);
+
+				/* Get the related structure.                                   */
+				switch (ATP_RESULT_CODE_INFO[cmd_nb][ATP_RESULT_PARAM_COLUMN])
+				{
+
+					/* Basic syntax result code.                                */
+					case ATP_BASIC_RESULT_CODE:
+						{
+
+							/* Create a buffer and copy text string into it     */
+							/* ('\x00' included). If insufficient resources     */
+							/* available, then report an internal memory error  */
+							/* and abort.                                       */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + 0x0001, \
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+
+							/* Copy text into the buffer.                       */
+							memcpy ((void *) (*text_pp),
+									(void *) &(table[offset]),
+									*text_length_p);
+							(*text_pp)[*text_length_p] = '\x00';
+							break;
+						}
+
+					/* Extended syntax result code.                             */
+					case ATP_EXTENDED_RESULT_CODE:
+						{
+
+							/* Declare a local block variable.                  */
+							T_ATP_SINGLE_RESULT_CODE_VALUE  *result_code_param_p = (T_ATP_SINGLE_RESULT_CODE_VALUE *) cmd_info_p;
+
+							/* Create a buffer and copy text string into it     */
+							/* ('=' and '\x00' included). If insufficient       */
+							/* resources available, then report an internal     */
+							/* memory error and abort.                          */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + MAX_SINGLE_RESULT_CODE_VALUE_LEN + 0x0002, \
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+
+							/* Copy text into the buffer.                       */
+							memcpy ((void *) (*text_pp),
+									(void *) &(table[offset]),
+									*text_length_p);
+
+							/* If needed, copy the buffer describing the        */
+							/* command in interpreted format and free it.       */
+							if (result_code_param_p != NULL)
+							{
+								(*text_pp)[(*text_length_p)++] = '=';
+								*text_length_p = (UINT16) (*text_length_p + \
+														   dtoa (result_code_param_p->value,
+																 &((*text_pp)[*text_length_p])));
+							}
+							(*text_pp)[*text_length_p] = '\x00';
+							break;
+						}
+
+					/* CONNECT <text> result code.                              */
+					case ATP_CONNECT_TXT_PARAM:
+						{
+
+							/* Declare a local block variable.                  */
+							T_ATP_CONNECT_TXT_PARAM  *connect_txt_param_p = (T_ATP_CONNECT_TXT_PARAM *) cmd_info_p;
+
+							/* Create a buffer and copy text string into it     */
+							/* (' ' and '\x00' included). If insufficient       */
+							/* resources available, then report an internal     */
+							/* memory error and abort.                          */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + MAX_CONNECT_TXT_LEN + 0x0002, \
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+
+							/* Copy text into the buffer.                       */
+							memcpy ((void *) (*text_pp),
+									(void *) &(table[offset]),
+									*text_length_p);
+
+							/* If needed, copy the buffer describing the        */
+							/* command in interpreted format and free it.       */
+							if (connect_txt_param_p != NULL)
+							{
+								(*text_pp)[(*text_length_p)++] = ' ';
+								*text_length_p = (UINT16) (*text_length_p + \
+														   dtoa (connect_txt_param_p->value,
+																 &((*text_pp)[*text_length_p])));
+							}
+							(*text_pp)[*text_length_p] = '\x00';
+							break;
+						}
+
+					/* +CME ERROR: <error> and +CMS ERROR: <error> result       */
+					/* codes.                                                   */
+					case ATP_PLUS_ERROR_RESULT_CODE:
+						{
+
+							/* Declare local block variables.                   */
+							UINT8                         error_length              = 0x00;
+							T_ATP_PLUS_ERROR_RESULT_CODE  *plus_error_result_code_p = cmd_info_p;
+
+							/* If needed, take the buffer describing the        */
+							/* command in interpreted format into account.      */
+							if (plus_error_result_code_p != NULL)
+							{
+								error_length = plus_error_result_code_p->error_length;
+							}
+
+							/* Create a buffer and copy text string into it     */
+							/* (' ' and '\x00' included). If insufficient       */
+							/* resources available, then report an internal     */
+							/* memory error and abort.                          */
+							if (rvf_get_buf (mb_id, \
+											 *text_length_p + error_length + 0x0002, \
+											 (void **) text_pp) == RVF_RED)
+							{
+								atp_error (ATP_ERROR_TX_MB_RED);
+								ret_status = RV_MEMORY_ERR;
+								break;
+							}
+	
+							/* Copy the command line prefix into the buffer.    */
+							memcpy ((void *) (*text_pp),
+									ATP_AT_PREFIX,
+									ATP_AT_PREFIX_LEN);
+
+							/* Copy the text into the buffer.                   */
+							memcpy ((void *) &((*text_pp)[ATP_AT_PREFIX_LEN]),
+									(void *) &(table[offset]),
+									*text_length_p - ATP_AT_PREFIX_LEN);
+
+							/* If needed, copy the buffer describing the        */
+							/* command in interpreted format and free it.       */
+							if (plus_error_result_code_p != NULL)
+							{
+								(*text_pp)[(*text_length_p)++] = ' ';
+								memcpy ((void *) &((*text_pp)[*text_length_p]),
+										(void *) (plus_error_result_code_p->error_p),
+										error_length);
+								*text_length_p = (UINT16) (*text_length_p + \
+														   error_length);
+							}
+							(*text_pp)[*text_length_p] = '\x00';
+							break;
+						}
+					default:
+						{
+							rvf_send_trace ("ATP : Received an unknown command ",
+											34,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+							*text_pp   = NULL;
+							ret_status = RV_NOT_SUPPORTED;
+							break;
+						}
+				}
+				break;
+			}
+		default:
+			{
+				rvf_send_trace ("ATP : Received an unknown command ",
+								34,
+								NULL_PARAM,
+								RV_TRACE_LEVEL_WARNING,
+								ATP_USE_ID);
+				*text_pp   = NULL;
+				ret_status = RV_NOT_SUPPORTED;
+				break;
+			}
+	}
+
+	/* If needed, free the buffer describing the command in interpreted format. */
+	if (cmd_info_p != NULL)
+	{
+		rvf_free_buf (cmd_info_p);
+		cmd_info_p = NULL;
+	}
+	return (ret_status);
+
+} /****************** End of atp_translate_cmd_to_txt function ******************/
+
+
+/********************************************************************************
+* Function name : atp_copy_buffer
+* 
+* Description   : Copy 'data size' bytes of 'in buffer' to 'out buffer'
+*
+* Parameters    : in_buffer_p = buffer to copy from
+*                 out_buffer_p = new buffer
+*                 data_size = number of bytes to be copied
+*
+* Return        : RV_OK
+*
+* History       : 0.1 (05-May-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_copy_buffer (UINT8   *in_buffer_p,
+						   UINT8   *out_buffer_p,
+						   UINT32  data_size)
+{
+
+/************************ atp_copy_buffer function begins ***********************/
+
+	memcpy ((void *) out_buffer_p,
+			(void *) in_buffer_p,
+			data_size);
+  	return (RV_OK);
+
+} /*********************** End of atp_copy_buffer function **********************/
+
+
+/********************************************************************************
+* Function name : atp_create_data_buffer_from_cmd
+* 
+* Description   : create a data buffer based on port features and on the command 
+*                 which is interpreted or in text format
+*
+* Parameters    : cmd_mode = interpreted or text
+*                 header = header size
+*                 trailer = trailer size
+*                 dce_info_p = pointer on the DCE information
+*                 mb_id = memory bank used to get the data buffer
+*                 cmd_type = type of the command
+*                 cmd_nb = related binary code of the command (not used if TXT
+*                          format)
+*                 text_p = pointer on the text string (0-terminated) (not used
+*                          if INTERPRETED format)
+*                 cmd_info_p = pointer on the custom command information structure
+*                              (not used if TXT format)
+*                 buffer_pp = pointer on the data buffer generated by the function
+*                 length_p = length of the data buffer
+*
+* Return        : RV_OK,
+*                 RV_NOT_SUPPORTED if the command is not recognized
+*
+* History       : 0.1 (01-March-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_create_data_buffer_from_cmd (T_ATP_CMD_MODE  cmd_mode,
+										   UINT16          header,
+										   UINT16          trailer,
+										   T_ATP_DCE_INFO  *dce_info_p,
+										   T_RVF_MB_ID     mb_id,
+										   T_ATP_CMD_TYPE  cmd_type,
+										   T_ATP_CMD_NB    cmd_nb,
+										   T_ATP_TXT_CMD   text_p,
+										   T_ATP_CMD       *cmd_info_p,
+										   T_ATP_BUFFER    *buffer_pp,
+										   UINT16          *length_p)
+{
+    /* Declare local variables.                                                 */
+	UINT16         txt_length   = 0x0000;
+	T_ATP_BUFFER   atp_buffer_p = NULL;
+	T_ATP_TXT_CMD  text_cmd_p   = NULL;
+	
+/**************** atp_create_data_buffer_from_cmd function begins ***************/
+
+	/* Need to have the command type provided.                                  */
+	if (cmd_type == UNKNOWN)
+	{
+		return (RV_NOT_SUPPORTED);
+	}
+
+	/* Get text version of the AT command and point on it via text_cmd_p. If    */
+	/* the command has been provided already in text format, then...            */
+	if (cmd_mode == TXT_MODE)
+	{
+		text_cmd_p = text_p;
+
+		/* The length does not include '\x00'.                                  */
+		txt_length = (UINT16) (strlen (text_cmd_p));
+	}
+	else
+	{
+
+		/* Declare a local block variable.                                      */
+		T_ATP_RET  ret_status = RV_OK;
+
+		/* Translate the command into text.                                     */
+		ret_status = atp_translate_cmd_to_txt (cmd_type,
+											   cmd_nb,
+											   cmd_info_p,
+											   mb_id,
+											   &text_cmd_p,
+											   &txt_length);
+
+		/* If any error occurred, then abort.                                   */
+		if (ret_status != RV_OK)
+		{
+			return (RV_NOT_SUPPORTED);
+		}
+	}
+
+	/* Create real buffer to send for a AT Command.                             */
+	switch (dce_info_p->verbose_mode)
+	{
+
+		/* Verbose responses enabled.                                           */
+		case ATP_VERBOSE_0:
+			{
+				switch (cmd_type)
+				{
+
+					/* DCE responses (See ITU-T Recommendation V.250 ter page   */
+					/* 10).                                                     */
+					case RESULT_CODE:
+					case UNSOLICITED_RESULT:
+						{
+							*length_p = (UINT16) (trailer + header + 0x0002);
+
+							/* Create buffer and copy text string into it. If   */
+							/* insufficient resources available, then report an */
+							/* internal memory error and abort.                 */
+							if (rvf_get_buf (mb_id, \
+											 *length_p, \
+											 (void **) buffer_pp) == RVF_RED)
+							{
+								*length_p = 0x0000;
+								atp_error (ATP_ERROR_TX_MB_RED);
+								return (RV_MEMORY_ERR);
+							}
+							atp_buffer_p = *buffer_pp;
+
+							/* Add code.                                        */
+							atp_buffer_p[header] = (char) (cmd_nb + '0');
+
+							/* Add special characters.                          */
+							atp_buffer_p[header + 0x0001] = dce_info_p->cr_character;
+
+							/* Release text buffer.                             */
+							rvf_free_buf (text_cmd_p);
+							return (RV_OK);
+						}
+					case PRELIMINARY_RESULT_CODE:
+						{
+
+							/* Add 2 special characters <CR> and <LF>.          */
+							*length_p = (UINT16) (trailer + header + txt_length + 0x0002);
+			
+							/* Create buffer and copy text string into it. If   */
+							/* insufficient resources available, then report an */
+							/* internal memory error and abort.                 */
+							if (rvf_get_buf (mb_id, \
+											 *length_p, \
+											 (void **) buffer_pp) == RVF_RED)
+							{
+								*length_p = 0x0000;
+								atp_error (ATP_ERROR_TX_MB_RED);
+								return (RV_MEMORY_ERR);
+							}
+							atp_buffer_p = *buffer_pp;
+
+							/* Copy text into the buffer.                       */
+							memcpy ((void *) (atp_buffer_p + header),
+									(void *) text_cmd_p,
+									txt_length);
+
+							/* Add special characters.                          */
+							atp_buffer_p[header + txt_length]          = dce_info_p->cr_character;
+							atp_buffer_p[header + txt_length + 0x0001] = dce_info_p->lf_character;
+
+							/* Release text buffer.                             */
+							rvf_free_buf (text_cmd_p);
+							return (RV_OK);
+						}
+					default:
+						{
+							break;
+						}
+				}
+			}
+
+		/* Verbose responses disabled.                                          */
+		case ATP_VERBOSE_1:
+			{
+				switch (cmd_type)
+				{
+
+					/* DTE command lines (See ITU-T Recommendation V.250 ter    */
+					/* page 4).                                                 */
+					case AT_CMD:
+						{
+
+							/* The buffer contains AT command and <CR>          */
+							/* character.                                       */
+							*length_p = (UINT16) (trailer + header + txt_length + 0x0001);
+
+							/* Create buffer and copy text string into it. If   */
+							/* insufficient resources available, then report an */
+							/* internal memory error and abort.                 */
+							if (rvf_get_buf (mb_id, \
+											 *length_p, \
+											 (void **) buffer_pp) == RVF_RED)
+							{
+								*length_p = 0x0000;
+								atp_error (ATP_ERROR_TX_MB_RED);
+								return (RV_MEMORY_ERR);
+							}
+							atp_buffer_p = *buffer_pp;
+
+							/* Copy text into the buffer.                       */
+							memcpy ((void *) (atp_buffer_p + header),
+									(void *) text_cmd_p,
+									txt_length);
+
+							/* Add special characters.                          */
+							atp_buffer_p[header + txt_length] = dce_info_p->cr_character;
+
+							/* Release text buffer.                             */
+							rvf_free_buf (text_cmd_p);
+							return (RV_OK);
+						}
+
+					/* DCE responses (See ITU-T Recommendation V.250 ter page   */
+					/* 10).                                                     */
+					case RESULT_CODE:
+					case UNSOLICITED_RESULT:
+					case PRELIMINARY_RESULT_CODE:
+						{
+
+							/* Add 4 special characters <CR> and <LF> (twice).  */
+							*length_p = (UINT16) (trailer + header + txt_length + 0x0004);
+			
+							/* Create buffer and copy text string into it. If   */
+							/* insufficient resources available, then report an */
+							/* internal memory error and abort.                 */
+							if (rvf_get_buf (mb_id, \
+											 *length_p, \
+											 (void **) buffer_pp) == RVF_RED)
+							{
+								*length_p = 0x0000;
+								atp_error (ATP_ERROR_TX_MB_RED);
+								return (RV_MEMORY_ERR);
+							}
+							atp_buffer_p = *buffer_pp;
+
+							/* Copy text into the buffer.                       */
+							memcpy ((void *) (atp_buffer_p + header + 0x0002),
+									(void *) text_cmd_p,
+									txt_length);
+
+							/* Add special characters.                          */
+							atp_buffer_p[header]                       = dce_info_p->cr_character;
+							atp_buffer_p[header + 0x0001]              = dce_info_p->lf_character;
+							atp_buffer_p[header + txt_length + 0x0002] = dce_info_p->cr_character;
+							atp_buffer_p[header + txt_length + 0x0003] = dce_info_p->lf_character;
+
+							/* Release text buffer.                             */
+							rvf_free_buf (text_cmd_p);
+							return (RV_OK);
+						}
+					default:
+						{
+							rvf_send_trace ("ATP : Tried to create a buffer for an unknown command ",
+											54,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+							*length_p  = 0x0000;
+							*buffer_pp = NULL;
+							break;
+						}
+				}
+				break;
+			}
+		default:
+			{
+				rvf_send_trace ("ATP : Verbose mode invalid ",
+								27,
+								NULL_PARAM,
+								RV_TRACE_LEVEL_WARNING,
+								ATP_USE_ID);
+				*length_p  = 0x0000;
+				*buffer_pp = NULL;
+				break;
+			}
+	}
+	return (RV_NOT_SUPPORTED);
+
+} /*************** End of atp_create_data_buffer_from_cmd function **************/
+
+
+/********************************************************************************
+* Function name : atp_copy_buffer_from_l2cap
+*
+* Description   : Check the text command
+*
+* Parameters    : l2cap_buffer_p = type is L2CAP_ACL_DATA
+*                 copy_buffer_p = pointer on the buffer to copy data in 
+*                 buffer_length = data length to read
+*
+* Return        : RV_OK if it fits
+*
+* Note          : l2cap_buffer_p is not freed by this function !
+*
+* History       : 0.1 (21-March-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_copy_buffer_from_l2cap (void    *l2cap_buffer_p,
+									  void    *copy_buffer_p,
+									  UINT32  buffer_length,
+									  UINT32  offset)
+{
+
+#ifdef BLUETOOTH
+
+	/* Declare local variables.                                                 */
+	UINT16            nb_byte    = 0x0000;
+	T_RECV_DATA_ADDR  acl_data_p = {NULL, \
+									NULL};
+	
+/****************** atp_copy_buffer_from_l2cap function begins ******************/
+
+	rvf_send_trace ("ATP : Translate L2CAP buffer into a ATP buffer ",
+					47,
+					NULL_PARAM,
+					RV_TRACE_LEVEL_DEBUG_LOW,
+					ATP_USE_ID);
+	l2cap_read_next_uint ((T_L2CAP_ACL_DATA *) l2cap_buffer_p,
+						  sizeof (UINT8),
+						  ((UINT8 *) copy_buffer_p),
+						  &acl_data_p,
+						  (UINT8) offset);
+	for (nb_byte = 1;
+		 nb_byte < buffer_length;
+		 nb_byte++)
+	{
+		l2cap_read_next_uint ((T_L2CAP_ACL_DATA *) l2cap_buffer_p,
+							  sizeof (UINT8),
+							  ((UINT8 *) copy_buffer_p) + nb_byte,
+							  &acl_data_p,
+							  0x0000);
+	}
+	return (RV_OK);
+#else
+	rvf_send_trace ("ATP : Tried to read buffer in L2CAP format whereas Bluetooth is not enabled ",
+					76,
+					NULL_PARAM,
+					RV_TRACE_LEVEL_WARNING,
+					ATP_USE_ID);
+	return (RV_NOT_SUPPORTED);
+#endif
+	
+} /***************** End of atp_copy_buffer_from_l2cap function *****************/
+
+
+/********************************************************************************
+* Function name : atp_free_l2cap_buffer
+*
+* Description   : Release a L2CAP buffer
+*
+* Parameter     : l2cap_buffer_p = type is L2CAP_ACL_DATA
+*
+* Return        : RV_OK if free is OK
+*
+* History       : 0.1 (19-Dec-2001) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_free_l2cap_buffer (UINT8  *atp_buffer_p)
+{
+	
+/********************* atp_free_l2cap_buffer function begins ********************/
+
+#ifdef BLUETOOTH
+	return ((T_ATP_RET) (l2cap_free_data ((T_L2CAP_ACL_DATA *) atp_buffer_p)));
+#else
+	rvf_send_trace ("ATP : Tried to read buffer in L2CAP format whereas Bluetooth is not enabled ",
+					76,
+					NULL_PARAM,
+					RV_TRACE_LEVEL_WARNING,
+					ATP_USE_ID);
+	return (RV_NOT_SUPPORTED);
+#endif
+
+} /******************** End of atp_free_l2cap_buffer function *******************/
+
+
+/********************************************************************************
+* Function name : atp_escape_sequence_process
+*
+* Description   : This function is used to detect the escape sequence in the data
+*                 flow. This function should not be called if the port is not
+*                 configured in DCE mode. The escape sequence should start in a new
+*                 packet and the last character of the exit sequence should be the
+*                 last character of a packet. Note that each data buffer checked
+*                 that may be part of the escape sequence is temporarily stored so
+*                 that it can be sent to the SWE later on in case it was the escape
+*                 sequence.
+*
+* Parameters	: port_p = structure of the port
+*                 atp_buffer_p = pointer on the buffer received by ATP (can be a 
+*                 NORMAL data packet or a SEGMENTED data packet)
+*                 data_length = number of payload data in the packet pointed by
+*                 atp_buffer_p
+*                 packet_mode = indicates the mode of the data packet (NORMAL or
+*                 SEGMENTED)
+*
+* Return        : ATP_ESCAPE_SEQUENCE_SUCCESS,
+*                 ATP_ESCAPE_SEQUENCE_FAILED otherwise
+*
+* History       : 0.1 (06-Feb-2001) - Created
+*
+*********************************************************************************/
+T_ATP_ESCAPE_SEQUENCE_STATUS atp_escape_sequence_process (T_ATP_PORT_STRUCT  *port_p,
+														  UINT8              *atp_buffer_p,
+														  UINT32             data_length,
+														  T_ATP_PACKET_MODE  packet_mode)
+{
+	/* Declare local variables.                                                 */
+	UINT8  count   = 0;
+	UINT8  data_sequence[MAX_NB_OF_CHARACTER_FOR_END_SEQUENCE];
+	
+/****************** atp_escape_sequence_process function begins *****************/
+
+	/* Check the sequence. Indeed, there are data after the last character of   */
+	/* the escape sequence.                                                     */
+	if (((port_p->dce_info_p)->nb_plus_received + data_length) > MAX_NB_OF_CHARACTER_FOR_END_SEQUENCE)
+	{
+		return (ATP_ESCAPE_SEQUENCE_FAILED);
+	}
+
+	/* Get data from the buffer.                                                */
+	if (packet_mode == SEGMENTED_PACKET)
+	{
+		atp_copy_buffer_from_l2cap (atp_buffer_p,
+									data_sequence,
+									data_length,
+									0);
+	}
+	else
+	{
+		memcpy (data_sequence,
+				atp_buffer_p,
+				data_length);
+	}
+
+	/* Check every character.                                                   */
+	for (count = 0;
+		 count < data_length;
+		 count++)
+	{
+		if (data_sequence[count] != (port_p->dce_info_p)->escape_sequence[count + (port_p->dce_info_p)->nb_plus_received])
+		{
+			return (ATP_ESCAPE_SEQUENCE_FAILED);
+		}
+	}
+
+	/* Keep temporarily the pointer on the buffer.                              */
+	for (count = 0;
+		 (port_p->dce_info_p)->escape_sequence_tmp_buffer_p[count] != NULL;
+		 count++);
+	(port_p->dce_info_p)->escape_sequence_tmp_buffer_p[count]           = atp_buffer_p;
+	(port_p->dce_info_p)->length_of_escape_sequence_tmp_buffer_p[count]	= (UINT8) data_length;
+	if (((port_p->dce_info_p)->nb_plus_received + data_length) !=  MAX_NB_OF_CHARACTER_FOR_END_SEQUENCE)
+	{
+		(port_p->dce_info_p)->nb_plus_received = (UINT8) ((port_p->dce_info_p)->nb_plus_received + data_length);		
+		return (ATP_ESCAPE_SEQUENCE_WAIT);
+	}
+
+	/* Otherwise, all characters have been found.                               */
+	return (ATP_ESCAPE_SEQUENCE_SUCCESS);
+
+} /***************** End of atp_escape_sequence_process function ****************/
+
+  
+/********************************************************************************
+* Function name : atp_pipe_extra_character
+*
+* Description   : This function is called when it has been found that the escape
+*                 sequence search has failed and when other SWE is in copy mode.
+*                 In this case, data that has been received and that have not been
+*                 sent to the other SWE because we thought they may contain the
+*                 escape sequence must finally be sent.
+*                 For this reason, all the buffer to sent has been previously stored
+*                 in a temporary buffer called dce_info_p->escape_sequence_tmp_buffer_p.
+*                 This function pipes all this buffer into the data pipe of the SWE.
+*                 Note that such a complex mechanism is essentially due to the fact
+*                 that we may have to deal with L2CAP packet (SEGMENTED_MODE).
+*
+* Parameters    : port_p = pointer on the port structure
+*                 other_port_info_p = pointer on the port information structure of
+*                 the SWE which will receive the data.
+*
+* Return        : RV_OK
+*
+* History       : 0.1 (01-March-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_pipe_extra_character (T_ATP_PORT_STRUCT      *port_p,
+									T_ATP_PORT_END_STRUCT  *other_port_info_p)
+{
+	/* Declare local variables.                                                 */
+	UINT8            count        = 0;
+	UINT8            nb_packet    = 0;
+	T_ATP_RX_PACKET  *rx_packet_p = NULL;
+	
+/******************* atp_pipe_extra_character function begins *******************/
+
+	for (nb_packet = 0;
+		 ((port_p->dce_info_p)->escape_sequence_tmp_buffer_p[nb_packet] != NULL) && \
+		 (nb_packet < MAX_NB_OF_CHARACTER_FOR_END_SEQUENCE);
+		 nb_packet++);
+	for (count = 0; count < nb_packet; count++)
+	{
+		if (rvf_get_buf (other_port_info_p->rx_mb, \
+						 sizeof (T_ATP_RX_PACKET), \
+						 (void **) &rx_packet_p) == RVF_RED)
+		{
+			atp_error_switch (ATP_ERROR_FAILED_TO_SEND_DATA,
+							  ATP_MEMORY_ERROR,NULL);
+			return (RV_MEMORY_ERR);
+		}
+		rx_packet_p->first_byte        = 0;
+		rx_packet_p->last_byte         = ((port_p->dce_info_p)->length_of_escape_sequence_tmp_buffer_p[count]) - 1;
+		rx_packet_p->atp_buffer_p      = (port_p->dce_info_p)->escape_sequence_tmp_buffer_p[count];
+		rx_packet_p->next_byte_to_read = rx_packet_p->first_byte;
+		rvf_enqueue (&(other_port_info_p->rx_queue),
+					 rx_packet_p);
+		other_port_info_p->rx_data_left += (port_p->dce_info_p)->length_of_escape_sequence_tmp_buffer_p[count];
+	}
+	return (RV_OK);
+
+} /****************** End of atp_pipe_extra_character function ******************/
+
+  
+/********************************************************************************
+* Function name : atp_reset_escape_sequence
+*
+* Description   : This function resets all internal values used by the algorithm
+*                 of escape sequence search (information stored in the dce_info_p
+*                 structure)
+*
+* Parameter     : port_p = pointer on the port structure
+*
+* Return        : RV_OK
+*
+* History       : 0.1 (01-March-2000) - Created
+*
+*********************************************************************************/
+T_ATP_RET atp_reset_escape_sequence (T_ATP_PORT_STRUCT  *port_p)
+{
+	/* Declare local variables.                                                 */
+	UINT8  count     = 0;
+	UINT8  *buffer_p = NULL;
+	
+/******************* atp_reset_escape_sequence function begins ******************/
+
+	(port_p->dce_info_p)->nb_plus_received = 0;
+	
+	/* Clear pointers on temporary buffer containing potentially escape         */
+	/* sequence.                                                                */
+	for (count = 0;
+		 count < MAX_NB_OF_CHARACTER_FOR_END_SEQUENCE;
+		 count++)
+	{
+		if ((buffer_p = (port_p->dce_info_p)->escape_sequence_tmp_buffer_p[count]) != NULL)
+		{
+
+#ifdef BLUETOOTH
+
+			/* Check if the buffer has been issued by SPP. Not very clean way to */
+			/* check what is the packet mode.                                    */
+			if ((strcmp ((char *) atp_sw_entity_table_p[(port_p->port_info[0]).sw_id]->sw_entity_name, \
+						 ATP_SPP_NAME) == 0) || \
+				(strcmp ((char *) atp_sw_entity_table_p[(port_p->port_info[1]).sw_id]->sw_entity_name, \
+						 ATP_SPP_NAME) == 0))
+			{
+
+				/* Mode is SEGMENTED PACKET.                                     */
+				l2cap_free_data ((T_L2CAP_ACL_DATA *) buffer_p);
+			}
+			else
+			{
+
+				/* Mode  is NORMAL_PACKET.                                       */
+				rvf_free_buf (buffer_p);
+			}
+#else
+			rvf_free_buf (buffer_p);
+#endif
+			(port_p->dce_info_p)->escape_sequence_tmp_buffer_p[count]           = NULL;
+			(port_p->dce_info_p)->length_of_escape_sequence_tmp_buffer_p[count] = 0;
+		}
+	}
+	return (RV_OK);
+
+} /****************** End of atp_reset_escape_sequence function ******************/