diff src/cs/services/atp/atp_uart_handle_msg.c @ 0:b6a5e36de839

src/cs: initial import from Magnetite
author Mychaela Falconia <falcon@freecalypso.org>
date Sun, 15 Jul 2018 04:39:26 +0000 (2018-07-15)
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/cs/services/atp/atp_uart_handle_msg.c	Sun Jul 15 04:39:26 2018 +0000
@@ -0,0 +1,1058 @@
+/********************************************************************************/
+/*                                                                              */
+/*    File Name:         atp_uart_handle_msg.c                                  */
+/*                                                                              */
+/*    Purpose:           This file contains the internal function related to    */
+/*                       the ATP-UART interface and used to handle incoming     */
+/*                       messages                                               */
+/*                                                                              */
+/*    Note:              None.                                                  */
+/*                                                                              */
+/*    Revision History:                                                         */
+/*       10/04/01        Pascal Pompei                                          */
+/*                          - Create.                                           */
+/*                                                                              */
+/* (C) Copyright 2001 by Texas Instruments Incorporated, All Rights Reserved.   */
+/*                                                                              */
+/********************************************************************************/
+#include "atp/atp_i.h"
+#include "atp/atp_uart_i.h"
+#include "rvm/rvm_use_id_list.h"
+
+
+/********************************************************************************/
+/*                                                                              */
+/*    Function Name:     atp_uart_handle_msg                                    */
+/*                                                                              */
+/*    Purpose:           This function handles incoming messages available from */
+/*                       the mailbox dedicated to the ATP-UART interface.       */
+/*                                                                              */
+/*    Input Parameters:  None.                                                  */
+/*                                                                              */
+/*    Output Parameters: None.                                                  */
+/*                                                                              */
+/*    Global Parameters: None.                                                  */
+/*                                                                              */
+/*    Notes:             The GSM protocol stack must be started first.          */
+/*                       Nevertheless, note that incoming data calls are not    */
+/*						 supported for now. Moreover, the AT Parser is alerted  */
+/*                       that the call is terminated by receiving an 'Ok' or a  */
+/*                       'No Carrier' final result code only. At last, the GSM  */
+/*                       protocol stack is initialized by invoking the          */
+/*                       following AT commands:                                 */
+/*                          ATE0      : Echo mode switched off,                 */
+/*                          AT+CFUN=1 : Set Phone Functionality (Full),         */
+/*                          AT+COPS=0 : Operator Selection.                     */
+/*                                                                              */
+/*    Revision History:                                                         */
+/*       10/04/01        Pascal Pompei                                          */
+/*                          - Create.                                           */
+/*                                                                              */
+/********************************************************************************/
+T_ATP_UART_ERROR_CODES atp_uart_handle_msg (void)
+{
+	/* Declare local variables.                                                 */
+	UINT16                event_pending          = 0x0000;
+	UINT32                in_buffer_size         = 0x00000000;
+	BOOLEAN               echo_mode_switched_off = FALSE;
+	BOOLEAN               error_occurred         = FALSE;
+	T_ATP_CALLBACK        return_path            = {0x00, \
+													NULL};
+	T_ATP_ENTITY_MODE     mode                   = {CMD_SUPPORT_ON, \
+													TXT_MODE, \
+													COPY_ON};
+	T_ATP_SW_ENTITY_NAME  name                   = ATP_GSM_NAME;
+
+/********************** atp_uart_handle_msg function begins *********************/
+
+	/* First, wait until the COM port is activated.                             */
+	while ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).com_port_handle == INVALID_HANDLE_VALUE)
+	{
+
+		/* Wait for the next event(s) of interest.                              */
+		event_pending = rvf_wait (ATP_UART_ALL_EVENT_FLAGS,
+								  0x00000000);
+
+		/* If an expected event occurred, then get the associated message from  */
+		/* the mailbox dedicated to the ATP-UART interface.                     */
+		if (event_pending & ATP_UART_EXPECTED_EVENT)
+		{
+
+			/* Declare a local block variable.                                  */
+			T_RV_HDR  *incoming_message_p = NULL;
+
+			/* Get the incoming message from the mailbox dedicated to the       */
+			/* ATP-UART interface.                                              */
+			incoming_message_p = (T_RV_HDR *) rvf_read_mbox (ATP_UART_MAILBOX);
+
+			/* Check whether the incoming message is valid.                     */
+			if (incoming_message_p == NULL)
+			{
+				ATP_UART_UNRECOVERABLE_ERROR (RVM_INTERNAL_ERR,
+											  "  ATP-UART (handle_msg). Invalid event occurred ");
+				return (RVM_INTERNAL_ERR);
+			}
+			switch (incoming_message_p->msg_id)
+			{
+
+				/* 'ATP-UART Open COM Port'.                                    */
+				case ATP_UART_OPEN_COM_PORT:
+					{
+
+						/* Declare a local block variable.                      */
+						UINT8  gsm_basic_init_p[] = {'A', 'T', 'E', '0', \
+													 ATP_CR_CHARACTER};
+
+						/* Open the COM port whose number and baud rate are     */
+						/* specified. If any error occurred, then call the      */
+						/* function used whenever any unrecoverable error       */
+						/* occurs and abort.                                    */
+						if (atp_uart_create_com_port ((unsigned char) (((T_ATP_UART_OPEN_COM_PORT *) (incoming_message_p))->com_port), \
+													  ((T_ATP_UART_OPEN_COM_PORT *) (incoming_message_p))->baud_rate) != RV_OK)
+						{
+							ATP_UART_UNRECOVERABLE_ERROR (RVM_INTERNAL_ERR,
+														  "  ATP-UART (handle_msg). Unable to activate the COM port ");
+							return (RVM_INTERNAL_ERR);
+						}
+						rvf_send_trace ("  ATP-UART (handle_msg). COM port properly opened ",
+										50,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_LOW,
+										ATP_USE_ID);
+
+						/* Configure the GSM especially not to echo characters  */
+						/* during 'Command State' and 'Online Command State'.   */
+						/* If any error occurred, then call the function used   */
+						/* whenever any unrecoverable error occurs and abort.   */
+						if (atp_uart_write_com_port (gsm_basic_init_p, \
+													 sizeof (gsm_basic_init_p)) != RV_OK)
+						{
+							ATP_UART_UNRECOVERABLE_ERROR (RVM_INTERNAL_ERR,
+														  "  ATP-UART (handle_msg). GSM configuration failed ");
+							return (RVM_INTERNAL_ERR);
+						}
+						break;
+					}
+
+				/* Unexpected message.                                          */
+				default:
+					{
+						rvf_send_trace ("  ATP-UART (handle_msg). Unexpected message received ",
+										53,
+										incoming_message_p->msg_id,
+										RV_TRACE_LEVEL_WARNING,
+										ATP_USE_ID);
+						break;
+					}
+			}
+			(void) rvf_free_buf ((T_RVF_BUFFER *) (incoming_message_p));
+		}
+	}
+
+	/* Then, wait for the echo mode being switched off.                         */
+	while (echo_mode_switched_off == FALSE)
+	{
+
+		/* Wait for the next event(s) of interest.                              */
+		event_pending = rvf_wait (ATP_UART_ALL_EVENT_FLAGS,
+								  0x00000001);
+
+		/* If an expected event occurred, then get the associated message from  */
+		/* the mailbox dedicated to the ATP-UART interface.                     */
+		if (event_pending & ATP_UART_EXPECTED_EVENT)
+		{
+
+			/* Declare a local block variable.                                  */
+			T_RV_HDR  *incoming_message_p = NULL;
+
+			/* Get the incoming message from the mailbox dedicated to the       */
+			/* ATP-UART interface.                                              */
+			incoming_message_p = (T_RV_HDR *) rvf_read_mbox (ATP_UART_MAILBOX);
+
+			/* Check whether the incoming message is valid.                     */
+			if (incoming_message_p == NULL)
+			{
+				ATP_UART_UNRECOVERABLE_ERROR (RVM_INTERNAL_ERR,
+											  "  ATP-UART (handle_msg). Invalid event occurred ");
+				return (RVM_INTERNAL_ERR);
+			}
+			rvf_send_trace ("  ATP-UART (handle_msg). Unexpected message received ",
+							53,
+							incoming_message_p->msg_id,
+							RV_TRACE_LEVEL_WARNING,
+							ATP_USE_ID);
+			(void) rvf_free_buf ((T_RVF_BUFFER *) (incoming_message_p));
+		}
+
+		/* Now, attempt forwarding data to the AT Parser. Thus, initialize the  */
+		/* amount of data to be forwarded.                                      */
+		in_buffer_size = (UINT32) (ATP_UART_MAX_PACKET_SIZE - \
+								   (gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left);
+
+		/* Then, wait for data from the COM port. If any error occurred, then   */
+		/* abort.                                                               */
+		if (atp_uart_read_com_port (&((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left]), \
+									&in_buffer_size) == RV_OK)
+		{
+
+			/* Declare local block variables.                                   */
+			UINT32          in_buffer_begin = 0x00000000;
+			UINT32          in_buffer_end   = 0x00000000;
+			T_ATP_CMD       *cmd_info_p     = NULL;
+			T_ATP_CMD_NB    cmd_nb          = 0x0000;
+			T_ATP_CMD_TYPE  cmd_type        = UNKNOWN;
+
+			/* Send every 'Text Command' to the AT Parser separately.           */
+			while (in_buffer_end < in_buffer_size)
+			{
+
+				/* Skip leading <CR> and <LF> characters.                       */
+				for (in_buffer_begin = in_buffer_end;
+					 (in_buffer_begin < in_buffer_size) && \
+					 (((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_begin] == ATP_LF_CHARACTER) || \
+					  ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_begin] == ATP_CR_CHARACTER));
+					 in_buffer_begin++)
+				{
+				}
+
+				/* Get the 'Text Command' length by searching for the ending    */
+				/* <CR> character.                                              */
+				for (in_buffer_end = in_buffer_begin;
+					 (in_buffer_end < in_buffer_size) && \
+					 ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_end] != ATP_CR_CHARACTER);
+					 in_buffer_end++)
+				{
+				}
+
+				/* Abort whether no ending <CR> character found.                */
+				if ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_end] != ATP_CR_CHARACTER)
+				{
+
+					/* Update the number of bytes left.                         */
+					in_buffer_size -= in_buffer_begin;
+
+					/* Defrag.                                                  */
+					memcpy ((void *) ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p),
+							(void *) (&((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_begin])),
+							in_buffer_size);
+
+					/* Update the number of bytes left.                         */
+					(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left = in_buffer_size;
+					break;
+				}
+
+				/* Otherwise, replace the ending <CR> character by NULL.        */
+				/* Indeed, such a 'Text Command' must be a NULL-terminated      */
+				/* string.                                                      */
+				(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_end++] = '\x00';
+				rvf_send_trace (&((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_begin]),
+								(UINT8) (in_buffer_end - in_buffer_begin - 0x01),
+								NULL_PARAM,
+								RV_TRACE_LEVEL_DEBUG_LOW,
+								ATP_USE_ID);
+
+				/* Check whether the 'Text Command' is a final result code.     */
+				if ((atp_translate_txt_to_cmd (&((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_begin]), \
+											   UNKNOWN, \
+											   &cmd_type, \
+											   &cmd_nb, \
+											   (gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).atp_id, \
+											   &cmd_info_p) == RV_OK) && \
+					(cmd_type == RESULT_CODE))
+				{
+
+					/* If needed, de-allocate the command-related information.  */
+					if (cmd_info_p != NULL)
+					{
+						(void) rvf_free_buf ((T_RVF_BUFFER *) (cmd_info_p));
+					}
+
+					/* Check whether the 'Text Command' corresponds to the 'OK' */
+					/* final result code.                                       */
+					if (cmd_nb == ATP_OK_NB)
+					{
+						rvf_send_trace ("  ATP-UART (handle_msg). Echo mode switched off ",
+										48,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_LOW,
+										ATP_USE_ID);
+
+						/* The echo mode is properly switched off.              */
+						echo_mode_switched_off = TRUE;
+						break;
+					}
+					ATP_UART_UNRECOVERABLE_ERROR (RVM_INTERNAL_ERR,
+												  "  ATP-UART (handle_msg). Echo mode not switched off ");
+					return (RVM_INTERNAL_ERR);
+				}
+
+				/* If needed, de-allocate the command-related information.      */
+				if (cmd_info_p != NULL)
+				{
+					(void) rvf_free_buf ((T_RVF_BUFFER *) (cmd_info_p));
+				}
+			}
+			continue;
+		}
+		rvf_send_trace ("  ATP-UART (handle_msg). Failed in getting data from the COM port ",
+						66,
+						NULL_PARAM,
+						RV_TRACE_LEVEL_WARNING,
+						ATP_USE_ID);
+	}
+
+	/* At last, initialize the remainder of the return path in order to receive */
+	/* events from the AT parser.                                               */
+	return_path.addr_id = gbl_atp_uart_ctrl_blk_p->addr_id;
+	
+	/* Register the ATP-UART interface to the AT Parser. If any error occurred, */
+	/* then call the function used whenever any unrecoverable error occurs and  */
+	/* abort.                                                                   */
+	if (atp_reg (name, \
+				 return_path, \
+				 mode, \
+				 &((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).atp_id)) != RV_OK)
+	{
+		ATP_UART_UNRECOVERABLE_ERROR (RVM_INTERNAL_ERR,
+									  "  ATP-UART (handle_msg). Registration failed ");
+		return (RVM_INTERNAL_ERR);
+	}
+	rvf_send_trace ("  ATP-UART (handle_msg). Registration completed ",
+					48,
+					NULL_PARAM,
+					RV_TRACE_LEVEL_DEBUG_LOW,
+					ATP_USE_ID);
+
+	/* Now, wait for the next events to be handled as long as no error occurs.  */
+	while (error_occurred == FALSE)
+	{
+
+		/* Wait for the next event(s) of interest.                              */
+		event_pending = rvf_wait (ATP_UART_ALL_EVENT_FLAGS,
+								  0x00000001);
+
+		/* If an expected event occurred, then get the associated message from  */
+		/* the mailbox dedicated to the ATP-UART interface.                     */
+		if (event_pending & ATP_UART_EXPECTED_EVENT)
+		{
+
+			/* Declare a local block variable.                                  */
+			T_RV_HDR  *incoming_message_p = NULL;
+
+			/* Get the incoming message from the mailbox dedicated to the       */
+			/* ATP-UART interface.                                              */
+			incoming_message_p = (T_RV_HDR *) rvf_read_mbox (ATP_UART_MAILBOX);
+
+			/* Check whether the incoming message is valid.                     */
+			if (incoming_message_p == NULL)
+			{
+				ATP_UART_UNRECOVERABLE_ERROR (RVM_INTERNAL_ERR,
+											  "  ATP-UART (handle_msg). Invalid event occurred ");
+				return (RVM_INTERNAL_ERR);
+			}
+			switch (incoming_message_p->msg_id)
+			{
+
+				/* 'ATP Open Port Indication'.                                  */
+				case ATP_OPEN_PORT_IND:
+					{
+
+						/* Declare local block variables.                       */
+						T_ATP_PORT_INFO             gsm_port_info      = {NOT_DEFINED_CONFIG, \
+																		  ATP_NO_RING_TYPE, \
+																		  ATP_ALL_THE_SIGNAL_UNMASK, \
+																		  0x0000};
+						T_ATP_NO_COPY_INFO          gsm_no_copy_info   = {RVF_INVALID_MB_ID, \
+																		  RVF_INVALID_MB_ID, \
+																		  TX_HEADER_OFF, \
+																		  0x00, \
+																		  0x00, \
+																		  RX_HEADER_OFF, \
+																		  0x00, \
+																		  0x00, \
+																		  NORMAL_PACKET};
+						T_ATP_CUSTOM_FROM_GSM_INFO  *gsm_custom_info_p = NULL;
+
+						rvf_send_trace ("  ATP-UART (handle_msg). 'ATP Open Port Indication' received ",
+										61,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_MEDIUM,
+										ATP_USE_ID);
+
+						/* Set custom information associated with the pseudo    */
+						/* GSM Software Entity. If insufficient resources, then */
+						/* call the function used whenever any unrecoverable    */
+						/* error occurs and abort.                              */
+						if (rvf_get_buf (gbl_atp_uart_ctrl_blk_p->mb_id, \
+										 sizeof (T_ATP_CUSTOM_FROM_GSM_INFO), \
+										 (T_RVF_BUFFER **) (&gsm_custom_info_p)) == RVF_RED)
+						{
+							ATP_UART_UNRECOVERABLE_ERROR (RVM_MEMORY_ERR,
+														  "  ATP-UART (handle_msg). Insufficient resources ");
+							return (RVM_MEMORY_ERR);
+						}
+						gsm_custom_info_p->custom_type                 = ATP_FROM_GSM_INFO;
+						gsm_custom_info_p->optimal_gsm_max_packet_size = ATP_UART_MAX_PACKET_SIZE;
+
+						/* Open the virtual modem port with the AT Parser. If   */
+						/* any error occurred, then de-allocate custom          */
+						/* information associated with the pseudo GSM Software  */
+						/* Entity, call the function used whenever any          */
+						/* unrecoverable error occurs and abort.                */
+						if (atp_open_port_rsp (((T_ATP_OPEN_PORT_IND *) (incoming_message_p))->initiator_id, \
+											   ((T_ATP_OPEN_PORT_IND *) (incoming_message_p))->initiator_port_nb, \
+											   (gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).atp_id, \
+											   ATP_UART_GSM_PORT_NB, \
+											   gsm_port_info, \
+											   gsm_no_copy_info, \
+											   (T_ATP_CUSTOM_INFO *) (gsm_custom_info_p), \
+											   OPEN_PORT_OK) != RV_OK)
+						{
+							(void) rvf_free_buf ((T_RVF_BUFFER *) (gsm_custom_info_p));
+							ATP_UART_UNRECOVERABLE_ERROR (RVM_INTERNAL_ERR,
+														  "  ATP-UART (handle_msg). Unable to open the virtual modem port with the AT Parser ");
+							return (RVM_INTERNAL_ERR);
+						}
+						rvf_send_trace ("  ATP-UART (handle_msg). Virtual modem port properly established with the AT Parser ",
+										84,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_LOW,
+										ATP_USE_ID);
+
+						/* The virtual modem port opened with the AT Parser is  */
+						/* now properly initialized.                            */
+						(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).virtual_modem_port_established = TRUE;
+
+						/* If needed, de-allocate custom information contained  */
+						/* in the 'ATP Open Port Indication'.                   */
+						if (((T_ATP_OPEN_PORT_IND *) incoming_message_p)->custom_info_p != NULL)
+						{
+							rvf_free_buf ((T_RVF_BUFFER *) (((T_ATP_OPEN_PORT_IND *) (incoming_message_p))->custom_info_p));
+						}
+						break;
+					}
+
+				/* 'ATP Text Command Ready'.                                    */
+				case ATP_TXT_CMD_RDY:
+					{
+
+						/* Declare local block variables.                       */
+						UINT8              txt_cmd_length = 0x00;
+						T_ATP_TXT_CMD_RDY  *txt_cmd_rdy_p = (T_ATP_TXT_CMD_RDY *) (incoming_message_p);
+
+						rvf_send_trace ("  ATP-UART (handle_msg). 'ATP Text Command Ready' received ",
+										59,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_MEDIUM,
+										ATP_USE_ID);
+
+						/* If any error occurred, then abort.                   */
+						if ((txt_cmd_rdy_p->port_nb != ATP_UART_GSM_PORT_NB) || \
+							(txt_cmd_rdy_p->txt_cmd_p == NULL))
+						{
+							rvf_send_trace ("  ATP-UART (handle_msg). 'Text Command' transfer failed ",
+											56,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+							break;
+						}
+
+						/* Get the length of the 'Text Command'. Note that such */
+						/* a 'Text Command' is a NULL-terminated string.        */
+						txt_cmd_length = (UINT8) (strlen (txt_cmd_rdy_p->txt_cmd_p));
+						rvf_send_trace (txt_cmd_rdy_p->txt_cmd_p,
+										txt_cmd_length,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_LOW,
+										ATP_USE_ID);
+
+						/* Forward the 'Text Command' to the COM port. If any   */
+						/* error occurred, then proceed.                        */
+						txt_cmd_rdy_p->txt_cmd_p[txt_cmd_length] = ATP_CR_CHARACTER;
+						if (atp_uart_write_com_port ((UINT8 *) (txt_cmd_rdy_p->txt_cmd_p), \
+													 (txt_cmd_length + 0x00000001)) != RV_OK)
+						{
+							rvf_send_trace ("  ATP-UART (handle_msg). 'Text Command' transfer failed ",
+											56,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+						}
+
+						/* De-allocate the data buffer provided with the 'ATP   */
+						/* Command Ready'.                                      */
+						(void) rvf_free_buf ((T_RVF_BUFFER *) (txt_cmd_rdy_p->txt_cmd_p));
+						break;
+					}
+
+				/* 'ATP No Copy Data Ready'.                                    */
+				case ATP_NO_COPY_DATA_RDY:
+					{
+
+						/* Declare a local block variable.                      */
+						T_ATP_NO_COPY_DATA_RDY  *no_copy_data_rdy_p = (T_ATP_NO_COPY_DATA_RDY *) (incoming_message_p);
+
+						rvf_send_trace ("  ATP-UART (handle_msg). 'ATP No Copy Data Ready' received ",
+										59,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_MEDIUM,
+										ATP_USE_ID);
+
+						/* If any error occurred, then abort.                   */
+						if ((no_copy_data_rdy_p->port_nb != ATP_UART_GSM_PORT_NB) || \
+							(no_copy_data_rdy_p->atp_buffer_p == NULL))
+						{
+							rvf_send_trace ("  ATP-UART (handle_msg). Data transfer to the COM port failed ",
+											62,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+							break;
+						}
+						rvf_send_trace ("  ATP-UART (handle_msg). Data size ",
+										35,
+										no_copy_data_rdy_p->buffer_size,
+										RV_TRACE_LEVEL_DEBUG_LOW,
+										ATP_USE_ID);
+
+						/* Forward data to the COM port. If any error occurred, */
+						/* then proceed.                                        */
+						if (atp_uart_write_com_port (no_copy_data_rdy_p->atp_buffer_p, \
+													 no_copy_data_rdy_p->buffer_size) != RV_OK)
+						{
+							rvf_send_trace ("  ATP-UART (handle_msg). Data transfer to the COM port failed ",
+											62,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+						}
+
+						/* De-allocate the data buffer provided with the 'ATP   */
+						/* No Copy Data Ready'.                                 */
+						(void) rvf_free_buf ((T_RVF_BUFFER *) (no_copy_data_rdy_p->atp_buffer_p));
+						break;
+					}
+
+				/* 'ATP Signal Changed'.                                        */
+				case ATP_SIGNAL_CHANGED:
+					{
+
+						/* Declare a local block variable.                      */
+						T_ATP_SIGNAL_CHANGED  *signal_changed_p = (T_ATP_SIGNAL_CHANGED *) (incoming_message_p);
+
+						rvf_send_trace ("  ATP-UART (handle_msg). 'ATP Signal Changed' received ",
+										55,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_MEDIUM,
+										ATP_USE_ID);
+
+						/* If any error occurred, then abort.                   */
+						if (signal_changed_p->port_nb != ATP_UART_GSM_PORT_NB)
+						{
+							rvf_send_trace ("  ATP-UART (handle_msg). Invalid port number ",
+											45,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+							break;
+						}
+						rvf_send_trace ("  ATP-UART (handle_msg). Bit mask ",
+										34,
+										signal_changed_p->signal,
+										RV_TRACE_LEVEL_DEBUG_LOW,
+										ATP_USE_ID);
+
+						/* Update the RS232 control signals associated with the */
+						/* virtual modem port.                                  */
+						(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).control_signals = signal_changed_p->signal;
+						break;
+					}
+
+				/* 'ATP Port Mode Changed'.                                     */
+				case ATP_PORT_MODE_CHANGED:
+					{
+
+						/* Declare a local block variable.                      */
+						T_ATP_PORT_MODE_CHANGED  *port_mode_changed_p = (T_ATP_PORT_MODE_CHANGED *) (incoming_message_p);
+
+						rvf_send_trace ("  ATP-UART (handle_msg). 'ATP Port Mode Changed' received ",
+										58,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_MEDIUM,
+										ATP_USE_ID);
+
+						/* If any error occurred, then abort.                   */
+						if (port_mode_changed_p->port_nb != ATP_UART_GSM_PORT_NB)
+						{
+							rvf_send_trace ("  ATP-UART (handle_msg). Invalid port number ",
+											45,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+							break;
+						}
+						rvf_send_trace ("  ATP-UART (handle_msg). Mode ",
+										30,
+										port_mode_changed_p->mode,
+										RV_TRACE_LEVEL_DEBUG_LOW,
+										ATP_USE_ID);
+
+						/* If the virtual modem port is in 'Data' mode, then    */
+						/* send the escape sequence first.                      */
+						if ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).virtual_modem_port_mode == ATP_PORT_DATA_MODE)
+						{
+
+							/* Forward the escape sequence to the COM port. If  */
+							/* any error occurred, then call the function used  */
+							/* whenever any unrecoverable error occurs and      */
+							/* abort.                                           */
+							if (atp_uart_write_com_port ((UINT8 *) (ATP_ESCAPE_SEQUENCE), \
+														 (sizeof (ATP_ESCAPE_SEQUENCE) - 0x00000001)) != RV_OK)
+							{
+								ATP_UART_UNRECOVERABLE_ERROR (RVM_INTERNAL_ERR,
+															  "  ATP-UART (handle_msg). 'Escape sequence' not transmitted ");
+								return (RVM_INTERNAL_ERR);
+							}
+							rvf_send_trace ("  ATP-UART (handle_msg). 'Escape sequence' properly transmitted ",
+											64,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_DEBUG_LOW,
+											ATP_USE_ID);
+						}
+
+						/* Update the mode of the virtual modem port.           */
+						(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).virtual_modem_port_mode = port_mode_changed_p->mode;
+						break;
+					}
+
+				/* 'ATP Port Closed'.                                           */
+				case ATP_PORT_CLOSED:
+					{
+						rvf_send_trace ("  ATP-UART (handle_msg). 'ATP Port Closed' received ",
+										52,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_MEDIUM,
+										ATP_USE_ID);
+
+						/* If any error occurred, then abort.                   */
+						if (((T_ATP_PORT_CLOSED *) (incoming_message_p))->port_nb != ATP_UART_GSM_PORT_NB)
+						{
+							rvf_send_trace ("  ATP-UART (handle_msg). Invalid port number ",
+											45,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+							break;
+						}
+
+						/* The virtual modem port opened with the AT Parser is  */
+						/* now closed.                                          */
+						(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).virtual_modem_port_established = FALSE;
+						break;
+					}
+
+				/* 'ATP-UART Start GSM'.                                        */
+				case ATP_UART_START_GSM:
+					{
+
+						/* Declare a local block variable.                      */
+						UINT8  gsm_init_p[] = {'A', 'T', '+', 'C', 'F', 'U', 'N', '=', '1', ';', '+', 'C', 'O', 'P', 'S', '=', '0', \
+												ATP_CR_CHARACTER};
+
+						rvf_send_trace ("  ATP-UART (handle_msg). 'ATP-UART Start GSM' received ",
+										55,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_MEDIUM,
+										ATP_USE_ID);
+
+						/* If any error occurred, then abort.                   */
+						if (atp_uart_write_com_port (gsm_init_p, \
+													 sizeof (gsm_init_p)) != RV_OK)
+						{
+							rvf_send_trace ("  ATP-UART (handle_msg). Unable to start the GSM protocol stack ",
+											64,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+							break;
+						}
+						break;
+					}
+
+				/* Unexpected message.                                          */
+				default:
+					{
+						rvf_send_trace ("  ATP-UART (handle_msg). Unexpected message received ",
+										53,
+										incoming_message_p->msg_id,
+										RV_TRACE_LEVEL_WARNING,
+										ATP_USE_ID);
+						break;
+					}
+			}
+			(void) rvf_free_buf ((T_RVF_BUFFER *) (incoming_message_p));
+		}
+
+		/* Now, if the flow control is activated, then proceed without getting  */
+		/* any data from the COM port.                                          */
+		if (((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).control_signals & ATP_RX_FLOW_UNMASK) == ATP_RX_FLOW_ON)
+		{
+			continue;
+		}
+
+		/* Otherwise, attempt forwarding data to the AT Parser. Thus,           */
+		/* initialize the amount of data to be forwarded.                       */
+		in_buffer_size = (UINT32) (ATP_UART_MAX_PACKET_SIZE - \
+								   (gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left);
+
+		/* Then, wait for data from the COM port. If any error occurred, then   */
+		/* abort.                                                               */
+		if (atp_uart_read_com_port (&((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left]), \
+									&in_buffer_size) != RV_OK)
+		{
+			rvf_send_trace ("  ATP-UART (handle_msg). Failed in getting data from the COM port ",
+							66,
+							NULL_PARAM,
+							RV_TRACE_LEVEL_WARNING,
+							ATP_USE_ID);
+			continue;
+		}
+
+		/* Take new data into account.                                          */
+		in_buffer_size = (UINT32) ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left + \
+								   in_buffer_size);
+
+		/* If no virtual modem port established with the AT Parser or if no     */
+		/* data pending for transmission to the AT Parser, then update the      */
+		/* number of bytes left and proceed.                                    */
+		if (((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).virtual_modem_port_established == FALSE) || \
+			(in_buffer_size == 0x00000000))
+		{
+			(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left = in_buffer_size;
+			continue;
+		}
+
+		/* Otherwise, re-initialize the number of bytes left.                   */
+		(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left = 0x00000000;
+		rvf_send_trace ("  ATP-UART (handle_msg). Get data from the COM port ",
+						52,
+						in_buffer_size,
+						RV_TRACE_LEVEL_DEBUG_LOW,
+						ATP_USE_ID);
+
+		/* At last, depending on the mode, transfert data from the COM port as  */
+		/* 'Text Command' or 'Data'.                                            */
+		if ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).virtual_modem_port_mode == ATP_PORT_CMD_MODE)
+		{
+
+			/* Declare local block variables.                                   */
+			UINT32          in_buffer_begin = 0x00000000;
+			UINT32          in_buffer_end   = 0x00000000;
+			T_ATP_CMD       *cmd_info_p     = NULL;
+			T_ATP_CMD_NB    cmd_nb          = 0x0000;
+			T_ATP_TXT_CMD   txt_cmd_p       = NULL;
+			T_ATP_CMD_TYPE  cmd_type        = UNKNOWN;
+
+			/* Send every 'Text Command' to the AT Parser separately.           */
+			while (in_buffer_end < in_buffer_size)
+			{
+
+				/* Skip leading <CR> and <LF> characters.                       */
+				for (in_buffer_begin = in_buffer_end;
+					 (in_buffer_begin < in_buffer_size) && \
+					 (((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_begin] == ATP_LF_CHARACTER) || \
+					  ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_begin] == ATP_CR_CHARACTER));
+					 in_buffer_begin++)
+				{
+				}
+
+				/* Get the 'Text Command' length by searching for the ending    */
+				/* <CR> character.                                              */
+				for (in_buffer_end = in_buffer_begin;
+					 (in_buffer_end < in_buffer_size) && \
+					 ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_end] != ATP_CR_CHARACTER);
+					 in_buffer_end++)
+				{
+				}
+
+				/* Abort whether no ending <CR> character found.                */
+				if ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_end] != ATP_CR_CHARACTER)
+				{
+
+					/* Update the number of bytes left.                         */
+					in_buffer_size -= in_buffer_begin;
+
+					/* Defrag.                                                  */
+					memcpy ((void *) ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p),
+							(void *) (&((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_begin])),
+							in_buffer_size);
+
+					/* Update the number of bytes left.                         */
+					(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left = in_buffer_size;
+					break;
+				}
+
+				/* Otherwise, replace the ending <CR> character by NULL.        */
+				/* Indeed, such a 'Text Command' must be a NULL-terminated      */
+				/* string.                                                      */
+				(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_end++] = '\x00';
+
+				/* Allocate memory in order to forward the 'Text Command' to    */
+				/* the AT Parser. If insufficient resources, then call the      */
+				/* function used whenever any unrecoverable error occurs and    */
+				/* abort.                                                       */
+				if (rvf_get_buf (gbl_atp_uart_ctrl_blk_p->mb_id, \
+								 (in_buffer_end - in_buffer_begin), \
+								 (T_RVF_BUFFER **) (&txt_cmd_p)) == RVF_RED)
+				{
+					ATP_UART_UNRECOVERABLE_ERROR (RVM_MEMORY_ERR,
+												  "  ATP-UART (handle_msg). Insufficient resources ");
+					break;
+				}
+
+				/* Copy the 'Text Command'.                                     */
+				memcpy ((void *) (txt_cmd_p),
+						(void *) (&((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_begin])),
+						(in_buffer_end - in_buffer_begin));
+				rvf_send_trace (txt_cmd_p,
+								(UINT8) (in_buffer_end - in_buffer_begin - 0x01),
+								NULL_PARAM,
+								RV_TRACE_LEVEL_DEBUG_LOW,
+								ATP_USE_ID);
+
+				/* Check whether the 'Text Command' is a final or an            */
+				/* unsolicited result code.                                     */
+				if ((atp_translate_txt_to_cmd (txt_cmd_p, \
+											   UNKNOWN, \
+											   &cmd_type, \
+											   &cmd_nb, \
+											   (gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).atp_id, \
+											   &cmd_info_p) == RV_OK) && \
+					(cmd_type == RESULT_CODE))
+				{
+
+					/* If the 'Text Command' corresponds to the 'Connect' final */
+					/* result code, then turn the 'Data Carrier Detect' on      */
+					/* before sending it to the AT Parser (See ITU-T            */
+					/* Recommendation V.250 ter page 32). If any error          */
+					/* occurred, then de-allocate the 'Text Command' and abort. */
+					if (cmd_nb == ATP_CONNECT_NB)
+					{
+						if (atp_set_signal ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).atp_id, \
+											ATP_UART_GSM_PORT_NB, \
+											ATP_DCD_1, \
+											ATP_DCD_UNMASK) != RV_OK)
+						{
+							(void) rvf_free_buf ((T_RVF_BUFFER *) (txt_cmd_p));
+
+							/* If needed, de-allocate the command-related       */
+							/* information.                                     */
+							if (cmd_info_p != NULL)
+							{
+								(void) rvf_free_buf ((T_RVF_BUFFER *) (cmd_info_p));
+							}
+							rvf_send_trace ("  ATP-UART (handle_msg). 'Data Carrier Detect' not turned on ",
+											61,
+											NULL_PARAM,
+											RV_TRACE_LEVEL_WARNING,
+											ATP_USE_ID);
+							break;
+						}
+						rvf_send_trace ("  ATP-UART (handle_msg). 'Data Carrier Detect' properly turned on ",
+										66,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_LOW,
+										ATP_USE_ID);
+					}
+				}
+
+				/* Otherwise, the 'Text Command' is recognized as a preliminary */
+				/* result code.                                                 */
+				else
+				{
+					cmd_type = PRELIMINARY_RESULT_CODE;
+				}
+
+				/* If needed, de-allocate the command-related information.      */
+				if (cmd_info_p != NULL)
+				{
+					(void) rvf_free_buf ((T_RVF_BUFFER *) (cmd_info_p));
+				}
+
+				/* Forward the 'Text Command' to the AT Parser. If any error    */
+				/* occurred, then de-allocate the 'Text Command' and abort.     */
+				if (atp_send_txt_cmd ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).atp_id, \
+									  ATP_UART_GSM_PORT_NB, \
+									  cmd_type, \
+									  txt_cmd_p) != RV_OK)
+				{
+					(void) rvf_free_buf ((T_RVF_BUFFER *) (txt_cmd_p));
+					rvf_send_trace ("  ATP-UART (handle_msg). 'Text Command' failed ",
+									47,
+									NULL_PARAM,
+									RV_TRACE_LEVEL_WARNING,
+									ATP_USE_ID);
+					break;
+				}
+			}
+		}
+		else
+		{
+
+			/* Declare local block variables.                                   */
+			UINT8         no_carrier_count = 0x00;
+			UINT8         no_carrier_p[]   = {ATP_CR_CHARACTER, \
+											  ATP_LF_CHARACTER, \
+											  'N', 'O', ' ', 'C', 'A', 'R', 'R', 'I', 'E', 'R', \
+											  ATP_CR_CHARACTER, \
+											  ATP_LF_CHARACTER};
+			UINT8         ok_count         = 0x00;
+			UINT8         ok_p[]           = {ATP_CR_CHARACTER, \
+											  ATP_LF_CHARACTER, \
+											  'O', 'K', \
+											  ATP_CR_CHARACTER, \
+											  ATP_LF_CHARACTER};
+			UINT32        in_buffer_count  = 0x00000000;
+			UINT32        in_data_size     = 0x00000000;
+			T_ATP_CMD_NB  result_code_nb   = ATP_MAX_NB_OF_RESULT_CODES;
+
+			/* First, search for 'Ok' or 'No Carrier' final result codes.       */
+			for (in_buffer_count = 0x00000000, \
+				 in_data_size    = in_buffer_size;
+				 in_buffer_count < in_buffer_size;
+				 in_buffer_count++)
+			{
+
+				/* Search for an 'Ok' final result code.                        */
+				if ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_count] == ok_p[ok_count])
+				{
+
+					/* 'Ok' final result code detected.                         */
+					if (++ok_count == sizeof (ok_p))
+					{
+						result_code_nb = ATP_OK_NB;
+						rvf_send_trace ("  ATP-UART (handle_msg). 'Ok' final result code detected ",
+										57,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_LOW,
+										ATP_USE_ID);
+
+						/* Send the remainder to the AT Parser.                 */
+						in_data_size = in_buffer_count - (sizeof (ok_p) - 0x00000001);
+						break;
+					}
+				}
+				else
+				{
+					ok_count = 0x00;
+				}
+
+				/* Search for a 'No Carrier' final result code.                 */
+				if ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_count] == no_carrier_p[no_carrier_count])
+				{
+
+					/* 'No Carrier' final result code detected.                 */
+					if (++no_carrier_count == sizeof (no_carrier_p))
+					{
+						result_code_nb = ATP_NO_CARRIER_NB;
+						rvf_send_trace ("  ATP-UART (handle_msg). 'No Carrier' final result code detected ",
+										65,
+										NULL_PARAM,
+										RV_TRACE_LEVEL_DEBUG_LOW,
+										ATP_USE_ID);
+
+						/* Send the remainder to the AT Parser.                 */
+						in_data_size = in_buffer_count - (sizeof (no_carrier_p) - 0x00000001);
+						break;
+					}
+				}
+				else
+				{
+					no_carrier_count = 0x00;
+				}
+			}
+
+			/* Then, check for some bytes to be forwarded data to the AT        */
+			/* Parser.                                                          */
+			if (in_data_size > 0x00000000)
+			{
+
+				/* Forward data to the AT Parser. If any error occurred, then   */
+				/* proceed.                                                     */
+				if (atp_send_data ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).atp_id, \
+								   ATP_UART_GSM_PORT_NB, \
+								   (void *) ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p), \
+								   in_data_size, \
+								   &((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left)) != RV_OK)
+				{
+					rvf_send_trace ("  ATP-UART (handle_msg). Data transfer to the AT Parser failed ",
+									63,
+									NULL_PARAM,
+									RV_TRACE_LEVEL_WARNING,
+									ATP_USE_ID);
+				}
+
+				/* Update the number of bytes left.                             */
+				in_data_size   -= (gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left;
+				in_buffer_size -= in_data_size;
+
+				/* Check whether some bytes left to be sent to the AT Parser.   */
+				if ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left > 0x00000000)
+				{
+
+					/* Defrag.                                                  */
+					memcpy ((void *) ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p),
+							(void *) (&((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_data_size])),
+							in_buffer_size);
+
+					/* Update the number of bytes left.                         */
+					(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left = in_buffer_size;
+					continue;
+				}
+			}
+
+			/* At last, check whether an 'Ok' or a 'No Carrier' final result    */
+			/* code was detected.                                               */
+			if (result_code_nb != ATP_MAX_NB_OF_RESULT_CODES)
+			{
+
+				/* Turn the 'Data Carrier Detect' off before sending it to the  */
+				/* AT Parser (See ITU-T Recommendation V.250 ter page 37).      */
+				/* However, proceed whether any error occurred.                 */
+				if (((atp_set_signal ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).atp_id, \
+									  ATP_UART_GSM_PORT_NB, \
+									  ATP_DCD_0, \
+									  ATP_DCD_UNMASK) != RV_OK) || \
+					 (atp_send_cmd ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).atp_id, \
+									ATP_UART_GSM_PORT_NB, \
+									RESULT_CODE, \
+									result_code_nb, \
+									NULL) != RV_OK)))
+				{
+					rvf_send_trace ("  ATP-UART (handle_msg). Final result code not transmitted ",
+									59,
+									NULL_PARAM,
+									RV_TRACE_LEVEL_WARNING,
+									ATP_USE_ID);
+				}
+
+				/* Update the number of bytes left.                             */
+				in_buffer_size -= in_buffer_count;
+
+				/* Update the mode of the virtual modem port.                   */
+				(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).virtual_modem_port_mode = ATP_PORT_CMD_MODE;
+			}
+
+			/* If some data left, then defrag.                                  */
+			memcpy ((void *) ((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p),
+					(void *) (&((gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).in_buffer_p[in_buffer_count])),
+					in_buffer_size);
+
+			/* Update the number of bytes left.                                 */
+			(gbl_atp_uart_ctrl_blk_p->conn_ctrl_blk).nb_bytes_left = in_buffer_size;
+		}
+	}
+	return (RVM_OK);
+
+} /********************* End of atp_uart_handle_msg function ********************/