From ae4b36507f9fef30b8386cfadc0e89c0ce0425c0 Mon Sep 17 00:00:00 2001 From: winckel <winckel@eurecom.fr> Date: Thu, 5 Dec 2013 13:03:18 +0000 Subject: [PATCH] Added ITTI timer support for RTAI builds (timers must only be used from non RT tasks). Blocked message sending to ended tasks. Modified lte-softmodem to use ITTI signal handling. Uniformized lte-softmodem and oaisim with ITTI handling. git-svn-id: http://svn.eurecom.fr/openair4G/trunk@4607 818b1a75-f10b-46b9-bf7c-635c3b92a50f --- common/utils/itti/intertask_interface.c | 103 ++++++++++++++---------- common/utils/itti/signals.c | 25 ++++-- common/utils/itti/signals.h | 2 +- common/utils/itti/timer.c | 8 -- common/utils/itti/timer.h | 4 + openair2/ENB_APP/enb_app.c | 2 +- openair2/UTIL/LOG/vcd_signal_dumper.c | 6 +- targets/RTAI/USER/Makefile | 3 + targets/RTAI/USER/lte-softmodem.c | 65 ++++++++++++--- 9 files changed, 143 insertions(+), 75 deletions(-) diff --git a/common/utils/itti/intertask_interface.c b/common/utils/itti/intertask_interface.c index a221f30929..d39fda861b 100644 --- a/common/utils/itti/intertask_interface.c +++ b/common/utils/itti/intertask_interface.c @@ -339,54 +339,67 @@ int itti_send_msg_to_task(task_id_t destination_task_id, instance_t instance, Me vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_ITTI_ENQUEUE_MESSAGE, VCD_FUNCTION_IN); #endif - /* We cannot send a message if the task is not running */ - DevCheck(itti_desc.threads[destination_thread_id].task_state == TASK_STATE_READY, itti_desc.threads[destination_thread_id].task_state, - TASK_STATE_READY, destination_thread_id); + if (itti_desc.threads[destination_thread_id].task_state == TASK_STATE_ENDED) + { + ITTI_DEBUG("Message %s, number %lu with priority %d can not be sent from %s to queue (%u:%s), ended destination task!\n", + itti_desc.messages_info[message_id].name, + message_number, + priority, + itti_get_task_name(origin_task_id), + destination_task_id, + itti_get_task_name(destination_task_id)); + } + else + { + /* We cannot send a message if the task is not running */ + DevCheck(itti_desc.threads[destination_thread_id].task_state == TASK_STATE_READY, destination_thread_id, + itti_desc.threads[destination_thread_id].task_state, message_id); - /* Allocate new list element */ - new = (message_list_t *) malloc (sizeof(struct message_list_s)); - DevAssert(new != NULL); + /* Allocate new list element */ + new = (message_list_t *) malloc (sizeof(struct message_list_s)); + DevAssert(new != NULL); - /* Fill in members */ - new->msg = message; - new->message_number = message_number; - new->message_priority = priority; + /* Fill in members */ + new->msg = message; + new->message_number = message_number; + new->message_priority = priority; - /* Enqueue message in destination task queue */ - lfds611_queue_enqueue(itti_desc.tasks[destination_task_id].message_queue, new); + /* Enqueue message in destination task queue */ + lfds611_queue_enqueue(itti_desc.tasks[destination_task_id].message_queue, new); #if defined(OAI_EMU) || defined(RTAI) - vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_ITTI_ENQUEUE_MESSAGE, VCD_FUNCTION_OUT); + vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_ITTI_ENQUEUE_MESSAGE, VCD_FUNCTION_OUT); #endif #ifdef RTAI - if (itti_desc.threads[TASK_GET_THREAD_ID(origin_task_id)].real_time) - { - /* This is a RT task, increase destination task messages pending counter */ - __sync_fetch_and_add (&itti_desc.threads[destination_thread_id].messages_pending, 1); - } - else + if (itti_desc.threads[TASK_GET_THREAD_ID(origin_task_id)].real_time) + { + /* This is a RT task, increase destination task messages pending counter */ + __sync_fetch_and_add (&itti_desc.threads[destination_thread_id].messages_pending, 1); + } + else #endif - { - /* Only use event fd for tasks, subtasks will pool the queue */ - if (TASK_GET_PARENT_TASK_ID(destination_task_id) == TASK_UNKNOWN) { - ssize_t write_ret; - uint64_t sem_counter = 1; + /* Only use event fd for tasks, subtasks will pool the queue */ + if (TASK_GET_PARENT_TASK_ID(destination_task_id) == TASK_UNKNOWN) + { + ssize_t write_ret; + uint64_t sem_counter = 1; - /* Call to write for an event fd must be of 8 bytes */ - write_ret = write (itti_desc.threads[destination_thread_id].task_event_fd, &sem_counter, sizeof(sem_counter)); - DevCheck(write_ret == sizeof(sem_counter), write_ret, sem_counter, destination_thread_id); + /* Call to write for an event fd must be of 8 bytes */ + write_ret = write (itti_desc.threads[destination_thread_id].task_event_fd, &sem_counter, sizeof(sem_counter)); + DevCheck(write_ret == sizeof(sem_counter), write_ret, sem_counter, destination_thread_id); + } } - } - ITTI_DEBUG("Message %s, number %lu with priority %d successfully sent from %s to queue (%u:%s)\n", - itti_desc.messages_info[message_id].name, - message_number, - priority, - itti_get_task_name(origin_task_id), - destination_task_id, - itti_get_task_name(destination_task_id)); + ITTI_DEBUG("Message %s, number %lu with priority %d successfully sent from %s to queue (%u:%s)\n", + itti_desc.messages_info[message_id].name, + message_number, + priority, + itti_get_task_name(origin_task_id), + destination_task_id, + itti_get_task_name(destination_task_id)); + } } #if defined(OAI_EMU) || defined(RTAI) @@ -634,6 +647,15 @@ void itti_mark_task_ready(task_id_t task_id) } void itti_exit_task(void) { +#if defined(OAI_EMU) || defined(RTAI) + task_id_t task_id = itti_get_current_task_id(); + + if (task_id > TASK_UNKNOWN) + { + vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLE_ITTI_RECV_MSG, + __sync_and_and_fetch (&itti_desc.vcd_receive_msg, ~(1L << task_id))); + } +#endif pthread_exit (NULL); } @@ -692,12 +714,7 @@ int itti_init(task_id_t task_max, thread_id_t thread_max, MessagesIds messages_i ITTI_DEBUG("Init: %d tasks, %d threads, %d messages\n", task_max, thread_max, messages_id_max); -#if !defined(RTAI) - /* SR: disable signals module for RTAI (need to harmonize management - * between softmodem and oaisim). - */ - CHECK_INIT_RETURN(signal_init()); -#endif + CHECK_INIT_RETURN(signal_mask()); /* Saves threads and messages max values */ itti_desc.task_max = task_max; @@ -799,9 +816,7 @@ int itti_init(task_id_t task_max, thread_id_t thread_max, MessagesIds messages_i itti_dump_init (messages_definition_xml, dump_file_name); -#ifndef RTAI - CHECK_INIT_RETURN(timer_init ()); -#endif + CHECK_INIT_RETURN(timer_init ()); return 0; } diff --git a/common/utils/itti/signals.c b/common/utils/itti/signals.c index 0f5c4e7415..0d42579faf 100644 --- a/common/utils/itti/signals.c +++ b/common/utils/itti/signals.c @@ -48,10 +48,23 @@ #include "assertions.h" #include "signals.h" +#include "log.h" + +#if defined (LOG_D) && defined (LOG_E) +# define SIG_DEBUG(x, args...) LOG_D(EMU, x, ##args) +# define SIG_ERROR(x, args...) LOG_E(EMU, x, ##args) +#endif + +#ifndef SIG_DEBUG +# define SIG_DEBUG(x, args...) do { fprintf(stdout, "[SIG][D]"x, ##args); } while(0) +#endif +#ifndef SIG_ERROR +# define SIG_ERROR(x, args...) do { fprintf(stdout, "[SIG][E]"x, ##args); } while(0) +#endif static sigset_t set; -int signal_init(void) +int signal_mask(void) { /* We set the signal mask to avoid threads other than the main thread * to receive the timer signal. Note that threads created will inherit this @@ -73,8 +86,6 @@ int signal_init(void) return 0; } -extern int timer_handle_signal(siginfo_t *info); - int signal_handle(int *end) { int ret; @@ -113,24 +124,24 @@ int signal_handle(int *end) /* Dispatch the signal to sub-handlers */ switch(info.si_signo) { case SIGUSR1: - printf("Received SIGUSR1\n"); + SIG_DEBUG("Received SIGUSR1\n"); *end = 1; break; case SIGSEGV: /* Fall through */ case SIGABRT: - printf("Received SIGABORT\n"); + SIG_DEBUG("Received SIGABORT\n"); backtrace_handle_signal(&info); break; case SIGINT: - printf("Received SIGINT\n"); + SIG_DEBUG("Received SIGINT\n"); itti_send_terminate_message(TASK_UNKNOWN); *end = 1; break; default: - printf("Received unknown signal %d\n", info.si_signo); + SIG_ERROR("Received unknown signal %d\n", info.si_signo); break; } } diff --git a/common/utils/itti/signals.h b/common/utils/itti/signals.h index 6901c5f051..7a1f335e83 100644 --- a/common/utils/itti/signals.h +++ b/common/utils/itti/signals.h @@ -1,7 +1,7 @@ #ifndef SIGNALS_H_ #define SIGNALS_H_ -int signal_init(void); +int signal_mask(void); int signal_handle(int *end); diff --git a/common/utils/itti/timer.c b/common/utils/itti/timer.c index faee3908fc..72af0bdfc4 100644 --- a/common/utils/itti/timer.c +++ b/common/utils/itti/timer.c @@ -85,7 +85,6 @@ do { \ int timer_handle_signal(siginfo_t *info) { -#if !defined(RTAI) struct timer_elm_s *timer_p; MessageDef *message_p; timer_has_expired_t *timer_expired_p; @@ -127,7 +126,6 @@ int timer_handle_signal(siginfo_t *info) free(message_p); return -1; } -#endif return 0; } @@ -141,7 +139,6 @@ int timer_setup( void *timer_arg, long *timer_id) { -#if !defined(RTAI) struct sigevent se; struct itimerspec its; struct timer_elm_s *timer_p; @@ -209,9 +206,6 @@ int timer_setup( pthread_mutex_lock(&timer_desc.timer_list_mutex); STAILQ_INSERT_TAIL(&timer_desc.timer_queue, timer_p, entries); pthread_mutex_unlock(&timer_desc.timer_list_mutex); -#else - return -1; -#endif return 0; } @@ -219,7 +213,6 @@ int timer_setup( int timer_remove(long timer_id) { int rc = 0; -#if !defined(RTAI) struct timer_elm_s *timer_p; TMR_DEBUG("Removing timer 0x%lx\n", timer_id); @@ -243,7 +236,6 @@ int timer_remove(long timer_id) } free(timer_p); timer_p = NULL; -#endif return rc; } diff --git a/common/utils/itti/timer.h b/common/utils/itti/timer.h index 1a0e8eb6fb..e3b1ae2492 100644 --- a/common/utils/itti/timer.h +++ b/common/utils/itti/timer.h @@ -31,6 +31,8 @@ #ifndef TIMER_H_ #define TIMER_H_ +#include <signal.h> + #define SIGTIMER SIGRTMIN typedef enum timer_type_s { @@ -39,6 +41,8 @@ typedef enum timer_type_s { TIMER_TYPE_MAX, } timer_type_t; +int timer_handle_signal(siginfo_t *info); + /** \brief Request a new timer * \param interval_sec timer interval in seconds * \param interval_us timer interval in micro seconds diff --git a/openair2/ENB_APP/enb_app.c b/openair2/ENB_APP/enb_app.c index a7fae5e434..89102b7b25 100644 --- a/openair2/ENB_APP/enb_app.c +++ b/openair2/ENB_APP/enb_app.c @@ -215,7 +215,7 @@ void *eNB_app_task(void *args_p) if (timer_setup (ENB_REGISTER_RETRY_DELAY, 0, TASK_ENB_APP, INSTANCE_DEFAULT, TIMER_ONE_SHOT, NULL, &enb_register_retry_timer_id) < 0) { - LOG_E(ENB_APP, " Can not start eNB register retry timer!\n"); + LOG_E(ENB_APP, " Can not start eNB register retry timer, use \"usleep\" instead!\n"); usleep(ENB_REGISTER_RETRY_DELAY * 1000000); /* Restart the registration process */ diff --git a/openair2/UTIL/LOG/vcd_signal_dumper.c b/openair2/UTIL/LOG/vcd_signal_dumper.c index 1755b2fd9b..8cbc2f3e1c 100644 --- a/openair2/UTIL/LOG/vcd_signal_dumper.c +++ b/openair2/UTIL/LOG/vcd_signal_dumper.c @@ -51,6 +51,7 @@ #include <unistd.h> #include <assert.h> +#include "signals.h" #if defined(ENABLE_VCD_FIFO) # include "liblfds611.h" #endif @@ -266,6 +267,10 @@ void *vcd_dumper_thread_rt(void *args) vcd_queue_user_data_t *data; char binary_string[(sizeof (uint64_t) * BYTE_SIZE) + 1]; +# if defined(ENABLE_ITTI) + signal_mask(); +# endif + while(1) { if (lfds611_queue_dequeue(vcd_queue, (void **) &data) == 0) { /* No element -> sleep a while */ @@ -313,7 +318,6 @@ void *vcd_dumper_thread_rt(void *args) void vcd_signal_dumper_init(char *filename) { - if (ouput_vcd) { // char filename[] = "/tmp/openair_vcd_dump.vcd"; diff --git a/targets/RTAI/USER/Makefile b/targets/RTAI/USER/Makefile index 74dc1f6715..d2afb7d25a 100644 --- a/targets/RTAI/USER/Makefile +++ b/targets/RTAI/USER/Makefile @@ -155,6 +155,9 @@ RTAI_CFLAGS += $(shell rtai-config --lxrt-cflags) -DRTAI LDFLAGS += -lpthread -lm -lforms ifeq ($(RTAI),1) LDFLAGS += $(shell rtai-config --lxrt-ldflags) +ifdef ENABLE_ITTI +LDFLAGS += -lrt +endif else LDFLAGS += -lrt endif diff --git a/targets/RTAI/USER/lte-softmodem.c b/targets/RTAI/USER/lte-softmodem.c index de1919bb75..6238395d87 100644 --- a/targets/RTAI/USER/lte-softmodem.c +++ b/targets/RTAI/USER/lte-softmodem.c @@ -217,6 +217,7 @@ unsigned int build_rfdc(int dcoff_i_rxfe, int dcoff_q_rxfe) return (dcoff_i_rxfe + (dcoff_q_rxfe<<8)); } +#if !defined(ENABLE_ITTI) void signal_handler(int sig) { void *array[10]; @@ -235,6 +236,7 @@ void signal_handler(int sig) oai_exit=1; } } +#endif void exit_fun(const char* s) { @@ -480,17 +482,48 @@ void *l2l1_task(void *arg) free (message_p); } itti_receive_msg (TASK_L2L1, &message_p); + + switch (ITTI_MSG_ID(message_p)) { + case INITIALIZE_MESSAGE: + /* Start eNB thread */ + start_eNB = 1; + break; + + case TERMINATE_MESSAGE: + oai_exit=1; + itti_exit_task (); + break; + + default: + LOG_E(EMU, "Received unexpected message %s\n", ITTI_MSG_NAME(message_p)); + break; + } } while (ITTI_MSG_ID(message_p) != INITIALIZE_MESSAGE); free (message_p); - - /* Start eNB thread */ - start_eNB = 1; } - while (!oai_exit) - { - usleep(500000); - } + do { + // Wait for a message + itti_receive_msg (TASK_L2L1, &message_p); + + switch (ITTI_MSG_ID(message_p)) { + case TERMINATE_MESSAGE: + oai_exit=1; + itti_exit_task (); + break; + + case MESSAGE_TEST: + LOG_I(EMU, "Received %s\n", ITTI_MSG_NAME(message_p)); + break; + + default: + LOG_E(EMU, "Received unexpected message %s\n", ITTI_MSG_NAME(message_p)); + break; + } + + free (message_p); + } while(1); + return NULL; } #endif @@ -1270,26 +1303,23 @@ int main(int argc, char **argv) { else { log_set_instance_type (LOG_INSTANCE_ENB); } -#endif -#if defined(ENABLE_ITTI) itti_init(TASK_MAX, THREAD_MAX, MESSAGES_ID_MAX, tasks_info, messages_info, messages_definition_xml, itti_dump_file); if (create_tasks(UE_flag ? 0 : 1, UE_flag ? 1 : 0) < 0) { exit(-1); // need a softer mode } - - // Handle signals until all tasks are terminated -// itti_wait_tasks_end(); #endif #ifdef NAS_NETLINK netlink_init(); #endif +#if !defined(ENABLE_ITTI) // to make a graceful exit when ctrl-c is pressed signal(SIGSEGV, signal_handler); signal(SIGINT, signal_handler); +#endif #ifndef RTAI check_clock(); @@ -1463,11 +1493,15 @@ int main(int argc, char **argv) { g_log->log_component[OTG].flag = LOG_HIGH; g_log->log_component[RRC].level = LOG_INFO; g_log->log_component[RRC].flag = LOG_HIGH; -#if defined(ENABLE_ITTI) && defined(ENABLE_USE_MME) +#if defined(ENABLE_ITTI) + g_log->log_component[EMU].level = LOG_INFO; + g_log->log_component[EMU].flag = LOG_HIGH; +# if defined(ENABLE_USE_MME) g_log->log_component[S1AP].level = LOG_INFO; g_log->log_component[S1AP].flag = LOG_HIGH; g_log->log_component[SCTP].level = LOG_INFO; g_log->log_component[SCTP].flag = LOG_HIGH; +# endif #endif g_log->log_component[ENB_APP].level = LOG_INFO; g_log->log_component[ENB_APP].flag = LOG_HIGH; @@ -1864,8 +1898,13 @@ int main(int argc, char **argv) { // wait for end of program printf("TYPE <CTRL-C> TO TERMINATE\n"); //getchar(); + +#if defined(ENABLE_ITTI) + itti_wait_tasks_end(); +#else while (oai_exit==0) rt_sleep_ns(FRAME_PERIOD); +#endif // stop threads #ifdef XFORMS -- GitLab