/* * Licensed to the OpenAirInterface (OAI) Software Alliance under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The OpenAirInterface Software Alliance licenses this file to You under * the OAI Public License, Version 1.1 (the "License"); you may not use this file * except in compliance with the License. * You may obtain a copy of the License at * * http://www.openairinterface.org/?page_id=698 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. *------------------------------------------------------------------------------- * For more information about the OpenAirInterface (OAI) Software Alliance: * contact@openairinterface.org */ /*! \file log.c * \brief log implementaion * \author Navid Nikaein * \date 2009 - 2014 * \version 0.5 * @ingroup util */ #define _GNU_SOURCE /* required for pthread_getname_np */ //#define LOG_TEST 1 #define COMPONENT_LOG #define COMPONENT_LOG_IF #include <ctype.h> #define LOG_MAIN #include "log.h" #include "vcd_signal_dumper.h" #include "assertions.h" #if defined(ENABLE_ITTI) # include "intertask_interface.h" #endif # include <pthread.h> # include <string.h> #include <linux/prctl.h> #include "common/config/config_userapi.h" // main log variables log_mem_cnt_t log_mem_d[2]; int log_mem_flag=0; int log_mem_multi=1; volatile int log_mem_side=0; pthread_mutex_t log_mem_lock; pthread_cond_t log_mem_notify; pthread_t log_mem_thread; int log_mem_file_cnt=0; volatile int log_mem_write_flag=0; volatile int log_mem_write_side=0; char __log_mem_filename[1024]={0}; char * log_mem_filename = &__log_mem_filename[0]; char logmem_filename[1024] = {0}; mapping log_level_names[] = { {"error", OAILOG_ERR}, {"file", OAILOG_FILE}, {"warn", OAILOG_WARNING}, {"info", OAILOG_INFO}, {"debug", OAILOG_DEBUG}, {"trace", OAILOG_TRACE}, {NULL, -1} }; mapping log_options[] = { {"nocolor", FLAG_NOCOLOR }, {"level", FLAG_LEVEL }, {"thread", FLAG_THREAD }, {NULL,-1} }; mapping log_maskmap[] = { {"prach", DEBUG_PRACH}, {"RU", DEBUG_RU}, {"LTEESTIM", DEBUG_LTEESTIM}, {"ctrlsocket", DEBUG_CTRLSOCKET}, {"UE_PHYPROC", DEBUG_UE_PHYPROC}, {"UE_TIMING", UE_TIMING}, {NULL,-1} }; char *log_level_highlight_start[] = {LOG_RED, LOG_GREEN, LOG_ORANGE, "", LOG_BLUE, LOG_CYBL}; /*!< \brief Optional start-format strings for highlighting */ char *log_level_highlight_end[] = {LOG_RESET,LOG_RESET,LOG_RESET,LOG_RESET, LOG_RESET,LOG_RESET}; /*!< \brief Optional end-format strings for highlighting */ int write_file_matlab(const char *fname,const char *vname,void *data,int length,int dec,char format) { FILE *fp=NULL; int i; if (data == NULL) return -1; //printf("Writing %d elements of type %d to %s\n",length,format,fname); if (format == 10 || format ==11 || format == 12 || format == 13 || format == 14) { fp = fopen(fname,"a+"); } else if (format != 10 && format !=11 && format != 12 && format != 13 && format != 14) { fp = fopen(fname,"w+"); } if (fp== NULL) { printf("[OPENAIR][FILE OUTPUT] Cannot open file %s\n",fname); return(-1); } if (format != 10 && format !=11 && format != 12 && format != 13 && format != 14) fprintf(fp,"%s = [",vname); switch (format) { case 0: // real 16-bit for (i=0; i<length; i+=dec) { fprintf(fp,"%d\n",((short *)data)[i]); } break; case 1: // complex 16-bit case 13: case 14: case 15: for (i=0; i<length<<1; i+=(2*dec)) { fprintf(fp,"%d + j*(%d)\n",((short *)data)[i],((short *)data)[i+1]); } break; case 2: // real 32-bit for (i=0; i<length; i+=dec) { fprintf(fp,"%d\n",((int *)data)[i]); } break; case 3: // complex 32-bit for (i=0; i<length<<1; i+=(2*dec)) { fprintf(fp,"%d + j*(%d)\n",((int *)data)[i],((int *)data)[i+1]); } break; case 4: // real 8-bit for (i=0; i<length; i+=dec) { fprintf(fp,"%d\n",((char *)data)[i]); } break; case 5: // complex 8-bit for (i=0; i<length<<1; i+=(2*dec)) { fprintf(fp,"%d + j*(%d)\n",((char *)data)[i],((char *)data)[i+1]); } break; case 6: // real 64-bit for (i=0; i<length; i+=dec) { fprintf(fp,"%lld\n",((long long*)data)[i]); } break; case 7: // real double for (i=0; i<length; i+=dec) { fprintf(fp,"%g\n",((double *)data)[i]); } break; case 8: // complex double for (i=0; i<length<<1; i+=2*dec) { fprintf(fp,"%g + j*(%g)\n",((double *)data)[i], ((double *)data)[i+1]); } break; case 9: // real unsigned 8-bit for (i=0; i<length; i+=dec) { fprintf(fp,"%d\n",((unsigned char *)data)[i]); } break; case 10 : // case eren 16 bit complex : for (i=0; i<length<<1; i+=(2*dec)) { if((i < 2*(length-1)) && (i > 0)) fprintf(fp,"%d + j*(%d),",((short *)data)[i],((short *)data)[i+1]); else if (i == 2*(length-1)) fprintf(fp,"%d + j*(%d);",((short *)data)[i],((short *)data)[i+1]); else if (i == 0) fprintf(fp,"\n%d + j*(%d),",((short *)data)[i],((short *)data)[i+1]); } break; case 11 : //case eren 16 bit real for channel magnitudes: for (i=0; i<length; i+=dec) { if((i <(length-1))&& (i > 0)) fprintf(fp,"%d,",((short *)data)[i]); else if (i == (length-1)) fprintf(fp,"%d;",((short *)data)[i]); else if (i == 0) fprintf(fp,"\n%d,",((short *)data)[i]); } printf("\n eren: length :%d",length); break; case 12 : // case eren for log2_maxh real unsigned 8 bit fprintf(fp,"%d \n",((unsigned char *)&data)[0]); break; } if (format != 10 && format !=11 && format !=12 && format != 13 && format != 15) { fprintf(fp,"];\n"); fclose(fp); return(0); } else if (format == 10 || format ==11 || format == 12 || format == 13 || format == 15) { fclose(fp); return(0); } return 0; } /* get log parameters from configuration file */ void log_getconfig(log_t *g_log) { char *gloglevel = NULL; int level; paramdef_t logparams_defaults[] = LOG_GLOBALPARAMS_DESC; paramdef_t logparams_level[MAX_LOG_PREDEF_COMPONENTS]; paramdef_t logparams_logfile[MAX_LOG_PREDEF_COMPONENTS]; paramdef_t logparams_debug[sizeof(log_maskmap)/sizeof(mapping)]; paramdef_t logparams_matlab[sizeof(log_maskmap)/sizeof(mapping)]; int ret = config_get( logparams_defaults,sizeof(logparams_defaults)/sizeof(paramdef_t),CONFIG_STRING_LOG_PREFIX); if (ret <0) { fprintf(stderr,"[LOG] init aborted, configuration couldn't be performed"); return; } for(int i=0; i<logparams_defaults[LOG_OPTIONS_IDX].numelt ; i++) { for(int j=0; log_options[j].name != NULL ; j++) { if (strcmp(logparams_defaults[LOG_OPTIONS_IDX].strlistptr[i],log_options[j].name) == 0) { g_log->flag = g_log->flag | log_options[j].value; break; } else if (log_options[j+1].name == NULL){ fprintf(stderr,"Unknown log option: %s\n",logparams_defaults[LOG_OPTIONS_IDX].strlistptr[i]); exit(-1); } } } /* build the parameter array for setting per component log level and infile options */ memset(logparams_level, 0, sizeof(paramdef_t)*MAX_LOG_PREDEF_COMPONENTS); memset(logparams_logfile, 0, sizeof(paramdef_t)*MAX_LOG_PREDEF_COMPONENTS); for (int i=MIN_LOG_COMPONENTS; i < MAX_LOG_PREDEF_COMPONENTS; i++) { if(g_log->log_component[i].name == NULL) { g_log->log_component[i].name = malloc(16); sprintf((char *)g_log->log_component[i].name,"comp%i?",i); logparams_logfile[i].paramflags = PARAMFLAG_DONOTREAD; logparams_level[i].paramflags = PARAMFLAG_DONOTREAD; } sprintf(logparams_level[i].optname, LOG_CONFIG_LEVEL_FORMAT, g_log->log_component[i].name); sprintf(logparams_logfile[i].optname, LOG_CONFIG_LOGFILE_FORMAT, g_log->log_component[i].name); /* workaround: all log options in existing configuration files use lower case component names where component names include uppercase char in log.h.... */ for (int j=0 ; j<strlen(logparams_level[i].optname); j++) logparams_level[i].optname[j] = tolower(logparams_level[i].optname[j]); for (int j=0 ; j<strlen(logparams_level[i].optname); j++) logparams_logfile[i].optname[j] = tolower(logparams_logfile[i].optname[j]); /* */ logparams_level[i].defstrval = gloglevel; logparams_logfile[i].defuintval = 0; logparams_logfile[i].numelt = 0; logparams_level[i].numelt = 0; logparams_level[i].type = TYPE_STRING; logparams_logfile[i].type = TYPE_UINT; logparams_logfile[i].paramflags = logparams_logfile[i].paramflags|PARAMFLAG_BOOL; } /* read the per component parameters */ config_get( logparams_level, MAX_LOG_PREDEF_COMPONENTS,CONFIG_STRING_LOG_PREFIX); config_get( logparams_logfile, MAX_LOG_PREDEF_COMPONENTS,CONFIG_STRING_LOG_PREFIX); /* now set the log levels and infile option, according to what we read */ for (int i=MIN_LOG_COMPONENTS; i < MAX_LOG_PREDEF_COMPONENTS; i++) { level = map_str_to_int(log_level_names, *(logparams_level[i].strptr)); set_log(i, level,1); if (*(logparams_logfile[i].uptr) == 1) set_component_filelog(i); } /* build then read the debug and matlab parameter array */ for (int i=0;log_maskmap[i].name != NULL ; i++) { sprintf(logparams_debug[i].optname, LOG_CONFIG_DEBUG_FORMAT, log_maskmap[i].name); sprintf(logparams_matlab[i].optname, LOG_CONFIG_MATLAB_FORMAT, log_maskmap[i].name); logparams_debug[i].defuintval = 0; logparams_debug[i].type = TYPE_UINT; logparams_debug[i].paramflags = PARAMFLAG_BOOL; logparams_debug[i].uptr = NULL; logparams_debug[i].chkPptr = NULL; logparams_debug[i].numelt = 0; logparams_matlab[i].defuintval = 0; logparams_matlab[i].type = TYPE_UINT; logparams_matlab[i].paramflags = PARAMFLAG_BOOL; logparams_matlab[i].uptr = NULL; logparams_matlab[i].chkPptr = NULL; logparams_matlab[i].numelt = 0; } config_get( logparams_debug,(sizeof(log_maskmap)/sizeof(mapping)) - 1 ,CONFIG_STRING_LOG_PREFIX); config_get( logparams_matlab,(sizeof(log_maskmap)/sizeof(mapping)) - 1 ,CONFIG_STRING_LOG_PREFIX); /* set the debug mask according to the debug parameters values */ for (int i=0; log_maskmap[i].name != NULL ; i++) { if (*(logparams_debug[i].uptr) ) g_log->debug_mask = g_log->debug_mask | log_maskmap[i].value; if (*(logparams_matlab[i].uptr) ) g_log->matlab_mask = g_log->matlab_mask | log_maskmap[i].value; } } int register_log_component(char *name, char *fext, int compidx) { int computed_compidx=compidx; if (strlen(fext) > 3) { fext[3]=0; /* limit log file extension to 3 chars */ } if (compidx < 0) { /* this is not a pre-defined component */ for (int i = MAX_LOG_PREDEF_COMPONENTS; i< MAX_LOG_COMPONENTS; i++) { if (g_log->log_component[i].name == NULL) { computed_compidx=i; break; } } } if (computed_compidx >= 0 && computed_compidx <MAX_LOG_COMPONENTS) { g_log->log_component[computed_compidx].name = strdup(name); g_log->log_component[computed_compidx].level = LOG_ERR; g_log->log_component[computed_compidx].interval = 1; g_log->log_component[computed_compidx].stream = NULL; g_log->log_component[computed_compidx].filelog = 0; g_log->log_component[computed_compidx].filelog_name = malloc(strlen(name)+16);/* /tmp/<name>.%s rounded to ^2 */ sprintf(g_log->log_component[computed_compidx].filelog_name,"/tmp/%s.%s",name,fext); } else { fprintf(stderr,"{LOG} %s %d Couldn't register componemt %s\n",__FILE__,__LINE__,name); } return computed_compidx; } int logInit (void) { int i; g_log = calloc(1, sizeof(log_t)); if (g_log == NULL) { perror ("cannot allocated memory for log generation module \n"); exit(EXIT_FAILURE); } memset(g_log,0,sizeof(log_t)); register_log_component("PHY","log",PHY); register_log_component("MAC","log",MAC); register_log_component("OPT","log",OPT); register_log_component("RLC","log",RLC); register_log_component("PDCP","log",PDCP); register_log_component("RRC","log",RRC); register_log_component("OMG","csv",OMG); register_log_component("OTG","log",OTG); register_log_component("OTG_LATENCY","dat",OTG_LATENCY); register_log_component("OTG_LATENCY_BG","dat",OTG_LATENCY_BG); register_log_component("OTG_GP","dat",OTG_GP); register_log_component("OTG_GP_BG","dat",OTG_GP_BG); register_log_component("OTG_JITTER","dat",OTG_JITTER); register_log_component("OCG","",OCG); register_log_component("PERF","",PERF); register_log_component("OIP","",OIP); register_log_component("CLI","",CLI); register_log_component("MSC","log",MSC); register_log_component("OCM","log",OCM); register_log_component("HW","",HW); register_log_component("OSA","",OSA); register_log_component("eRAL","",RAL_ENB); register_log_component("mRAL","",RAL_UE); register_log_component("ENB_APP","log",ENB_APP); register_log_component("FLEXRAN_AGENT","log",FLEXRAN_AGENT); register_log_component("TMR","",TMR); register_log_component("USIM","txt",USIM); register_log_component("SIM","txt",SIM); /* following log component are used for the localization*/ register_log_component("LOCALIZE","log",LOCALIZE); register_log_component("NAS","log",NAS); register_log_component("UDP","",UDP_); register_log_component("GTPV1U","",GTPU); register_log_component("S1AP","",S1AP); register_log_component("SCTP","",SCTP); register_log_component("RRH","",RRH); g_log->level2string[OAILOG_ERR] = "E"; // ERROR g_log->level2string[OAILOG_WARNING] = "W"; // WARNING g_log->level2string[OAILOG_INFO] = "I"; //INFO g_log->level2string[OAILOG_DEBUG] = "D"; // DEBUG g_log->level2string[OAILOG_FILE] = "F"; // file g_log->level2string[OAILOG_TRACE] = "T"; // TRACE g_log->onlinelog = 1; //online log file g_log->filelog = 0; g_log->filelog_name = "/tmp/openair.log"; log_getconfig(g_log); // could put a loop here to check for all comps for (i=MIN_LOG_COMPONENTS; i < MAX_LOG_COMPONENTS; i++) { if (g_log->log_component[i].filelog == 1 ) { g_log->log_component[i].stream = fopen(g_log->log_component[i].filelog_name,"w"); g_log->log_component[i].fwrite = vfprintf; } else if (g_log->log_component[i].filelog == 1 ) { g_log->log_component[i].stream = fopen(g_log->filelog_name,"w"); g_log->log_component[i].fwrite = vfprintf; } else if (g_log->onlinelog == 1 ) { g_log->log_component[i].stream = stdout; g_log->log_component[i].fwrite = vfprintf; } } // set all unused component items to 0, they are for non predefined components for (i=MAX_LOG_PREDEF_COMPONENTS; i < MAX_LOG_COMPONENTS; i++) { memset(&(g_log->log_component[i]),0,sizeof(log_component_t)); } printf("log init done\n"); return 0; } char *log_getthreadname(char *threadname, int bufsize) { int rt = pthread_getname_np(pthread_self(), threadname,bufsize) ; if (rt == 0) { return threadname; } else { return "thread?"; } } void logRecord_mt(const char *file, const char *func, int line, int comp, int level, const char* format, ... ) { char threadname[PR_SET_NAME]; char log_buffer[MAX_LOG_TOTAL]; va_list args; va_start(args, format); if(log_mem_flag == 1) { log_output_memory(file,func,line,comp,level, format,args); } else { snprintf(log_buffer, MAX_LOG_TOTAL , "%s%s[%s]%s %s %s", log_level_highlight_end[level], ( (g_log->flag & FLAG_NOCOLOR)?"":log_level_highlight_start[level]), g_log->log_component[comp].name, ( (g_log->flag & FLAG_LEVEL)?g_log->level2string[level]:""), ( (g_log->flag & FLAG_THREAD)?log_getthreadname(threadname,PR_SET_NAME+1):""), format); g_log->log_component[comp].fwrite(g_log->log_component[comp].stream,log_buffer, args); } va_end(args); } int set_log(int component, int level, int interval) { /* Checking parameters */ DevCheck((component >= MIN_LOG_COMPONENTS) && (component < MAX_LOG_COMPONENTS), component, MIN_LOG_COMPONENTS, MAX_LOG_COMPONENTS); DevCheck((level < NUM_LOG_LEVEL) && (level >= OAILOG_ERR), level, NUM_LOG_LEVEL, OAILOG_ERR); DevCheck((interval >= 0) && (interval <= 0xFF), interval, 0, 0xFF); g_log->log_component[component].level = level; g_log->log_component[component].interval = interval; return 0; } void set_glog(int level) { for (int c=0; c< MAX_LOG_COMPONENTS; c++ ) { g_log->log_component[c].level = level; } } void set_glog_onlinelog(int enable) { g_log->onlinelog = enable; } void set_glog_filelog(int enable) { g_log->filelog = enable; } void set_component_filelog(int comp) { if (g_log->log_component[comp].filelog == 0) { g_log->log_component[comp].filelog = 1; if (g_log->log_component[comp].stream == NULL) { g_log->log_component[comp].stream = fopen(g_log->log_component[comp].filelog_name,"w"); } } } /* * for the two functions below, the passed array must have a final entry * with string value NULL */ /* map a string to an int. Takes a mapping array and a string as arg */ int map_str_to_int(mapping *map, const char *str) { while (1) { if (map->name == NULL) { return(-1); } if (!strcmp(map->name, str)) { return(map->value); } map++; } } /* map an int to a string. Takes a mapping array and a value */ char *map_int_to_str(mapping *map, int val) { while (1) { if (map->name == NULL) { return NULL; } if (map->value == val) { return map->name; } map++; } } int is_newline( char *str, int size) { int i; for ( i = 0; i < size; i++ ) { if ( str[i] == '\n' ) { return 1; } } /* if we get all the way to here, there must not have been a newline! */ return 0; } void logClean (void) { int i; LOG_I(PHY,"\n"); for (i=MIN_LOG_COMPONENTS; i < MAX_LOG_COMPONENTS; i++) { if (g_log->log_component[i].stream != NULL) { fclose(g_log->log_component[i].stream); } } } extern int oai_exit; void flush_mem_to_file(void) { int fp; char f_name[1024]; struct timespec slp_tm; slp_tm.tv_sec = 0; slp_tm.tv_nsec = 10000; pthread_setname_np( pthread_self(), "flush_mem_to_file"); while (!oai_exit) { pthread_mutex_lock(&log_mem_lock); log_mem_write_flag=0; pthread_cond_wait(&log_mem_notify, &log_mem_lock); log_mem_write_flag=1; pthread_mutex_unlock(&log_mem_lock); // write! if(log_mem_d[log_mem_write_side].enable_flag==0){ if(log_mem_file_cnt>1){ log_mem_file_cnt=1; printf("log over write!!!\n"); } snprintf(f_name,1024, "%s_%d.log",log_mem_filename,log_mem_file_cnt); fp=open(f_name, O_WRONLY | O_CREAT, 0666); int ret = write(fp, log_mem_d[log_mem_write_side].buf_p, log_mem_d[log_mem_write_side].buf_index); if ( ret < 0) { fprintf(stderr,"{LOG} %s %d Couldn't write in %s \n",__FILE__,__LINE__,f_name); exit(EXIT_FAILURE); } close(fp); log_mem_file_cnt++; log_mem_d[log_mem_write_side].buf_index=0; log_mem_d[log_mem_write_side].enable_flag=1; }else{ printf("If you'd like to write log, you should set enable flag to 0!!!\n"); nanosleep(&slp_tm,NULL); } } } void log_output_memory(const char *file, const char *func, int line, int comp, int level, const char* format,va_list args) { //logRecord_mt(file,func,line, pthread_self(), comp, level, format, ##args) int len = 0; log_component_t *c; char *log_start; char *log_end; /* The main difference with the version above is the use of this local log_buffer. * The other difference is the return value of snprintf which was not used * correctly. It was not a big problem because in practice MAX_LOG_TOTAL is * big enough so that the buffer is never full. */ char log_buffer[MAX_LOG_TOTAL]; /* for no gcc warnings */ (void)log_start; (void)log_end; c = &g_log->log_component[comp]; //VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_LOG_RECORD, VCD_FUNCTION_IN); // make sure that for log trace the extra info is only printed once, reset when the level changes if ((level == OAILOG_FILE) || (level == OAILOG_TRACE)) { log_start = log_buffer; len = vsnprintf(log_buffer, MAX_LOG_TOTAL, format, args); if (len > MAX_LOG_TOTAL) len = MAX_LOG_TOTAL; log_end = log_buffer + len; } else { if ( (g_log->flag & 0x001) || (c->flag & 0x001) ) { len += snprintf(&log_buffer[len], MAX_LOG_TOTAL - len, "%s", log_level_highlight_start[level]); if (len > MAX_LOG_TOTAL) len = MAX_LOG_TOTAL; } log_start = log_buffer + len; // if ( (g_log->flag & 0x004) || (c->flag & 0x004) ) { len += snprintf(&log_buffer[len], MAX_LOG_TOTAL - len, "[%s]", g_log->log_component[comp].name); if (len > MAX_LOG_TOTAL) len = MAX_LOG_TOTAL; // } // if ( (g_log->flag & FLAG_LEVEL) || (c->flag & FLAG_LEVEL) ) { len += snprintf(&log_buffer[len], MAX_LOG_TOTAL - len, "[%s]", g_log->level2string[level]); if (len > MAX_LOG_TOTAL) len = MAX_LOG_TOTAL; // } if ( (g_log->flag & FLAG_THREAD) || (c->flag & FLAG_THREAD) ) { # define THREAD_NAME_LEN 128 char threadname[THREAD_NAME_LEN]; if (pthread_getname_np(pthread_self(), threadname, THREAD_NAME_LEN) != 0) { perror("pthread_getname_np : "); } else { len += snprintf(&log_buffer[len], MAX_LOG_TOTAL - len, "[%s]", threadname); if (len > MAX_LOG_TOTAL) len = MAX_LOG_TOTAL; } # undef THREAD_NAME_LEN } if ( (g_log->flag & FLAG_FUNCT) || (c->flag & FLAG_FUNCT) ) { len += snprintf(&log_buffer[len], MAX_LOG_TOTAL - len, "[%s] ", func); if (len > MAX_LOG_TOTAL) len = MAX_LOG_TOTAL; } if ( (g_log->flag & FLAG_FILE_LINE) || (c->flag & FLAG_FILE_LINE) ) { len += snprintf(&log_buffer[len], MAX_LOG_TOTAL - len, "[%s:%d]", file, line); if (len > MAX_LOG_TOTAL) len = MAX_LOG_TOTAL; } //len += snprintf(&log_buffer[len], MAX_LOG_TOTAL - len, "[%08lx]", thread_id); //if (len > MAX_LOG_TOTAL) len = MAX_LOG_TOTAL; len += vsnprintf(&log_buffer[len], MAX_LOG_TOTAL - len, format, args); if (len > MAX_LOG_TOTAL) len = MAX_LOG_TOTAL; log_end = log_buffer + len; if ( (g_log->flag & FLAG_NOCOLOR) || (c->flag & FLAG_NOCOLOR) ) { len += snprintf(&log_buffer[len], MAX_LOG_TOTAL - len, "%s", log_level_highlight_end[level]); if (len > MAX_LOG_TOTAL) len = MAX_LOG_TOTAL; } } //va_end(args); // OAI printf compatibility if ((g_log->onlinelog == 1) && (level != OAILOG_FILE)) { if(log_mem_flag==1){ if(log_mem_d[log_mem_side].enable_flag==1){ int temp_index; temp_index=log_mem_d[log_mem_side].buf_index; if(temp_index+len+1 < LOG_MEM_SIZE){ log_mem_d[log_mem_side].buf_index+=len; memcpy(&log_mem_d[log_mem_side].buf_p[temp_index],log_buffer,len); }else{ log_mem_d[log_mem_side].enable_flag=0; if(log_mem_d[1-log_mem_side].enable_flag==1){ temp_index=log_mem_d[1-log_mem_side].buf_index; if(temp_index+len+1 < LOG_MEM_SIZE){ log_mem_d[1-log_mem_side].buf_index+=len; log_mem_side=1-log_mem_side; memcpy(&log_mem_d[log_mem_side].buf_p[temp_index],log_buffer,len); /* write down !*/ if (pthread_mutex_lock(&log_mem_lock) != 0) { return; } if(log_mem_write_flag==0){ log_mem_write_side=1-log_mem_side; if(pthread_cond_signal(&log_mem_notify) != 0) { } } if(pthread_mutex_unlock(&log_mem_lock) != 0) { return; } }else{ log_mem_d[1-log_mem_side].enable_flag=0; } } } } }else{ fwrite(log_buffer, len, 1, stdout); } } } int logInit_log_mem (void) { if(log_mem_flag==1){ if(log_mem_multi==1){ printf("log-mem multi!!!\n"); log_mem_d[0].buf_p = malloc(LOG_MEM_SIZE); log_mem_d[0].buf_index=0; log_mem_d[0].enable_flag=1; log_mem_d[1].buf_p = malloc(LOG_MEM_SIZE); log_mem_d[1].buf_index=0; log_mem_d[1].enable_flag=1; log_mem_side=0; if ((pthread_mutex_init (&log_mem_lock, NULL) != 0) || (pthread_cond_init (&log_mem_notify, NULL) != 0)) { log_mem_d[1].enable_flag=0; return -1; } pthread_create(&log_mem_thread, NULL, (void *(*)(void *))flush_mem_to_file, (void*)NULL); }else{ printf("log-mem single!!!\n"); log_mem_d[0].buf_p = malloc(LOG_MEM_SIZE); log_mem_d[0].buf_index=0; log_mem_d[0].enable_flag=1; log_mem_d[1].enable_flag=0; log_mem_side=0; } }else{ log_mem_d[0].buf_p=NULL; log_mem_d[1].buf_p=NULL; log_mem_d[0].enable_flag=0; log_mem_d[1].enable_flag=0; } printf("log init done\n"); return 0; } void close_log_mem(void){ int fp; char f_name[1024]; if(log_mem_flag==1){ log_mem_d[0].enable_flag=0; log_mem_d[1].enable_flag=0; usleep(10); // wait for log writing while(log_mem_write_flag==1){ usleep(100); } if(log_mem_multi==1){ snprintf(f_name,1024, "%s_%d.log",log_mem_filename,log_mem_file_cnt); fp=open(f_name, O_WRONLY | O_CREAT, 0666); int ret = write(fp, log_mem_d[0].buf_p, log_mem_d[0].buf_index); if ( ret < 0) { fprintf(stderr,"{LOG} %s %d Couldn't write in %s \n",__FILE__,__LINE__,f_name); exit(EXIT_FAILURE); } close(fp); free(log_mem_d[0].buf_p); snprintf(f_name,1024, "%s_%d.log",log_mem_filename,log_mem_file_cnt); fp=open(f_name, O_WRONLY | O_CREAT, 0666); ret = write(fp, log_mem_d[1].buf_p, log_mem_d[1].buf_index); if ( ret < 0) { fprintf(stderr,"{LOG} %s %d Couldn't write in %s \n",__FILE__,__LINE__,f_name); exit(EXIT_FAILURE); } close(fp); free(log_mem_d[1].buf_p); }else{ fp=open(log_mem_filename, O_WRONLY | O_CREAT, 0666); int ret = write(fp, log_mem_d[0].buf_p, log_mem_d[0].buf_index); if ( ret < 0) { fprintf(stderr,"{LOG} %s %d Couldn't write in %s \n",__FILE__,__LINE__,log_mem_filename); exit(EXIT_FAILURE); } close(fp); free(log_mem_d[0].buf_p); } } } #ifdef LOG_TEST int main(int argc, char *argv[]) { logInit(); test_log(); return 1; } int test_log(void) { LOG_ENTER(MAC); // because the default level is DEBUG LOG_I(EMU, "1 Starting OAI logs version %s Build date: %s on %s\n", BUILD_VERSION, BUILD_DATE, BUILD_HOST); LOG_D(MAC, "1 debug MAC \n"); LOG_W(MAC, "1 warning MAC \n"); set_log(EMU, OAILOG_INFO, FLAG_ONLINE); set_log(MAC, OAILOG_WARNING, 0); LOG_I(EMU, "2 Starting OAI logs version %s Build date: %s on %s\n", BUILD_VERSION, BUILD_DATE, BUILD_HOST); LOG_E(MAC, "2 error MAC\n"); LOG_D(MAC, "2 debug MAC \n"); LOG_W(MAC, "2 warning MAC \n"); LOG_I(MAC, "2 info MAC \n"); set_log(MAC, OAILOG_NOTICE, 1); LOG_ENTER(MAC); LOG_I(EMU, "3 Starting OAI logs version %s Build date: %s on %s\n", BUILD_VERSION, BUILD_DATE, BUILD_HOST); LOG_D(MAC, "3 debug MAC \n"); LOG_W(MAC, "3 warning MAC \n"); LOG_I(MAC, "3 info MAC \n"); set_log(MAC, LOG_DEBUG,1); set_log(EMU, LOG_DEBUG,1); LOG_ENTER(MAC); LOG_I(EMU, "4 Starting OAI logs version %s Build date: %s on %s\n", BUILD_VERSION, BUILD_DATE, BUILD_HOST); LOG_D(MAC, "4 debug MAC \n"); LOG_W(MAC, "4 warning MAC \n"); LOG_I(MAC, "4 info MAC \n"); set_log(MAC, LOG_DEBUG,0); set_log(EMU, LOG_DEBUG,0); LOG_I(LOG, "5 Starting OAI logs version %s Build date: %s on %s\n", BUILD_VERSION, BUILD_DATE, BUILD_HOST); LOG_D(MAC, "5 debug MAC \n"); LOG_W(MAC, "5 warning MAC \n"); LOG_I(MAC, "5 info MAC \n"); set_log(MAC, LOG_TRACE,0X07F); set_log(EMU, LOG_TRACE,0X07F); LOG_ENTER(MAC); LOG_I(LOG, "6 Starting OAI logs version %s Build date: %s on %s\n", BUILD_VERSION, BUILD_DATE, BUILD_HOST); LOG_D(MAC, "6 debug MAC \n"); LOG_W(MAC, "6 warning MAC \n"); LOG_I(MAC, "6 info MAC \n"); LOG_EXIT(MAC); return 0; } #endif