LCOV - code coverage report
Current view: top level - daemon - dlt-daemon.c (source / functions) Hit Total Coverage
Test: dlt_final_coverage.info Lines: 606 1152 52.6 %
Date: 2025-01-09 05:30:37 Functions: 31 42 73.8 %

          Line data    Source code
       1             : /*
       2             :  * SPDX license identifier: MPL-2.0
       3             :  *
       4             :  * Copyright (C) 2011-2015, BMW AG
       5             :  *
       6             :  * This file is part of COVESA Project DLT - Diagnostic Log and Trace.
       7             :  *
       8             :  * This Source Code Form is subject to the terms of the
       9             :  * Mozilla Public License (MPL), v. 2.0.
      10             :  * If a copy of the MPL was not distributed with this file,
      11             :  * You can obtain one at http://mozilla.org/MPL/2.0/.
      12             :  *
      13             :  * For further information see http://www.covesa.org/.
      14             :  */
      15             : 
      16             : /*!
      17             :  * \author
      18             :  * Alexander Wenzel <alexander.aw.wenzel@bmw.de>
      19             :  * Markus Klein <Markus.Klein@esk.fraunhofer.de>
      20             :  * Mikko Rapeli <mikko.rapeli@bmw.de>
      21             :  *
      22             :  * \copyright Copyright © 2011-2015 BMW AG. \n
      23             :  * License MPL-2.0: Mozilla Public License version 2.0 http://mozilla.org/MPL/2.0/.
      24             :  *
      25             :  * \file dlt-daemon.c
      26             :  */
      27             : 
      28             : #include <netdb.h>
      29             : #include <ctype.h>
      30             : #include <stdio.h>      /* for printf() and fprintf() */
      31             : #include <sys/socket.h> /* for socket(), connect(), (), and recv() */
      32             : #include <sys/un.h>
      33             : #include <arpa/inet.h>  /* for sockaddr_in and inet_addr() */
      34             : #include <stdlib.h>     /* for atoi() and exit() */
      35             : #include <string.h>     /* for memset() */
      36             : #include <unistd.h>     /* for close() and access */
      37             : #include <fcntl.h>
      38             : #include <signal.h>
      39             : #include <syslog.h>
      40             : #include <errno.h>
      41             : #include <pthread.h>
      42             : #include <grp.h>
      43             : 
      44             : #ifdef linux
      45             : #   include <sys/timerfd.h>
      46             : #endif
      47             : #include <sys/stat.h>
      48             : #include <sys/time.h>
      49             : #include <libgen.h>
      50             : 
      51             : #if defined(linux) && defined(__NR_statx)
      52             : #   include <linux/stat.h>
      53             : #endif
      54             : 
      55             : #ifdef DLT_DAEMON_VSOCK_IPC_ENABLE
      56             : #   ifdef linux
      57             : #       include <linux/vm_sockets.h>
      58             : #   endif
      59             : #   ifdef __QNX__
      60             : #       include <vm_sockets.h>
      61             : #   endif
      62             : #endif
      63             : 
      64             : #include "dlt_types.h"
      65             : #include "dlt-daemon.h"
      66             : #include "dlt-daemon_cfg.h"
      67             : #include "dlt_daemon_common_cfg.h"
      68             : 
      69             : #include "dlt_daemon_socket.h"
      70             : #include "dlt_daemon_unix_socket.h"
      71             : #include "dlt_daemon_serial.h"
      72             : 
      73             : #include "dlt_daemon_client.h"
      74             : #include "dlt_daemon_connection.h"
      75             : #include "dlt_daemon_event_handler.h"
      76             : #include "dlt_daemon_offline_logstorage.h"
      77             : #include "dlt_gateway.h"
      78             : 
      79             : #ifdef UDP_CONNECTION_SUPPORT
      80             : #   include "dlt_daemon_udp_socket.h"
      81             : #endif
      82             : #if defined(DLT_SYSTEMD_WATCHDOG_ENABLE) || defined(DLT_SYSTEMD_ENABLE)
      83             : #   include "sd-daemon.h"
      84             : #endif
      85             : 
      86             : /**
      87             :  * \defgroup daemon DLT Daemon
      88             :  * \addtogroup daemon
      89             :  \{
      90             :  */
      91             : 
      92             : #define DLT_DAEMON_APP_ID "DLTD"
      93             : #define DLT_DAEMON_CTX_ID "INTM"
      94             : 
      95             : 
      96             : static int dlt_daemon_log_internal(DltDaemon *daemon,
      97             :                                    DltDaemonLocal *daemon_local, char *str,
      98             :                                    DltLogLevelType level, const char *app_id,
      99             :                                    const char *ctx_id, int verbose);
     100             : 
     101             : static int dlt_daemon_check_numeric_setting(char *token,
     102             :                                             char *value,
     103             :                                             unsigned long *data);
     104             : 
     105             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
     106             : 
     107             : struct DltTraceLoadLogParams {
     108             :     DltDaemon *daemon;
     109             :     DltDaemonLocal *daemon_local;
     110             :     int verbose;
     111             :     char *app_id;
     112             : };
     113             : 
     114             : static DltReturnValue dlt_daemon_output_internal_msg(DltLogLevelType loglevel, const char *text, void *params);
     115             : 
     116             : pthread_rwlock_t trace_load_rw_lock;
     117             : #endif
     118             : 
     119             : /* used in main event loop and signal handler */
     120             : int g_exit = 0;
     121             : 
     122             : int g_signo = 0;
     123             : 
     124             : /* used for value from conf file */
     125             : static int value_length = 1024;
     126             : 
     127             : static char dlt_timer_conn_types[DLT_TIMER_UNKNOWN + 1] = {
     128             :     [DLT_TIMER_PACKET] = DLT_CONNECTION_ONE_S_TIMER,
     129             :     [DLT_TIMER_ECU] = DLT_CONNECTION_SIXTY_S_TIMER,
     130             : #ifdef DLT_SYSTEMD_WATCHDOG_ENABLE
     131             :     [DLT_TIMER_SYSTEMD] = DLT_CONNECTION_SYSTEMD_TIMER,
     132             : #endif
     133             :     [DLT_TIMER_GATEWAY] = DLT_CONNECTION_GATEWAY_TIMER,
     134             :     [DLT_TIMER_UNKNOWN] = DLT_CONNECTION_TYPE_MAX
     135             : };
     136             : 
     137             : static char dlt_timer_names[DLT_TIMER_UNKNOWN + 1][32] = {
     138             :     [DLT_TIMER_PACKET] = "Timing packet",
     139             :     [DLT_TIMER_ECU] = "ECU version",
     140             : #ifdef DLT_SYSTEMD_WATCHDOG_ENABLE
     141             :     [DLT_TIMER_SYSTEMD] = "Systemd watchdog",
     142             : #endif
     143             :     [DLT_TIMER_GATEWAY] = "Gateway",
     144             :     [DLT_TIMER_UNKNOWN] = "Unknown timer"
     145             : };
     146             : 
     147             : #ifdef __QNX__
     148             : static int dlt_timer_pipes[DLT_TIMER_UNKNOWN][2] = {
     149             :     /* [timer_id] = {read_pipe, write_pipe} */
     150             :     [DLT_TIMER_PACKET] = {DLT_FD_INIT, DLT_FD_INIT},
     151             :     [DLT_TIMER_ECU] = {DLT_FD_INIT, DLT_FD_INIT},
     152             : #ifdef DLT_SYSTEMD_WATCHDOG_ENABLE
     153             :     [DLT_TIMER_SYSTEMD] = {DLT_FD_INIT, DLT_FD_INIT},
     154             : #endif
     155             :     [DLT_TIMER_GATEWAY] = {DLT_FD_INIT, DLT_FD_INIT}
     156             : };
     157             : 
     158             : static pthread_t timer_threads[DLT_TIMER_UNKNOWN] = {
     159             :     [DLT_TIMER_PACKET] = 0,
     160             :     [DLT_TIMER_ECU] = 0,
     161             : #ifdef DLT_SYSTEMD_WATCHDOG_ENABLE
     162             :     [DLT_TIMER_SYSTEMD] = 0,
     163             : #endif
     164             :     [DLT_TIMER_GATEWAY] = 0
     165             : };
     166             : 
     167             : static DltDaemonPeriodicData *timer_data[DLT_TIMER_UNKNOWN] = {
     168             :     [DLT_TIMER_PACKET] = NULL,
     169             :     [DLT_TIMER_ECU] = NULL,
     170             : #ifdef DLT_SYSTEMD_WATCHDOG_ENABLE
     171             :     [DLT_TIMER_SYSTEMD] = NULL,
     172             : #endif
     173             :     [DLT_TIMER_GATEWAY] = NULL
     174             : };
     175             : 
     176             : void close_pipes(int fds[2])
     177             : {
     178             :     if (fds[0] > 0) {
     179             :         close(fds[0]);
     180             :         fds[0] = DLT_FD_INIT;
     181             :     }
     182             : 
     183             :     if (fds[1] > 0) {
     184             :         close(fds[1]);
     185             :         fds[1] = DLT_FD_INIT;
     186             :     }
     187             : }
     188             : 
     189             : #endif // __QNX__
     190             : 
     191             : /**
     192             :  * Print usage information of tool.
     193             :  */
     194           0 : void usage()
     195             : {
     196             :     char version[DLT_DAEMON_TEXTBUFSIZE];
     197           0 :     dlt_get_version(version, DLT_DAEMON_TEXTBUFSIZE);
     198             : 
     199             :     /*printf("DLT logging daemon %s %s\n", _DLT_PACKAGE_VERSION, _DLT_PACKAGE_VERSION_STATE); */
     200             :     /*printf("Compile options: %s %s %s %s",_DLT_SYSTEMD_ENABLE, _DLT_SYSTEMD_WATCHDOG_ENABLE, _DLT_TEST_ENABLE, _DLT_SHM_ENABLE); */
     201             :     printf("%s", version);
     202             :     printf("Usage: dlt-daemon [options]\n");
     203             :     printf("Options:\n");
     204             :     printf("  -d            Daemonize\n");
     205             :     printf("  -h            Usage\n");
     206             :     printf("  -c filename   DLT daemon configuration file (Default: " CONFIGURATION_FILES_DIR "/dlt.conf)\n");
     207             : 
     208             : #ifdef DLT_DAEMON_USE_FIFO_IPC
     209             :     printf("  -t directory  Directory for local fifo and user-pipes (Default: /tmp)\n");
     210             :     printf("                (Applications wanting to connect to a daemon using a\n");
     211             :     printf("                custom directory need to be started with the environment \n");
     212             :     printf("                variable DLT_PIPE_DIR set appropriately)\n");
     213             : #endif
     214             : 
     215             : #ifdef DLT_SHM_ENABLE
     216             :     printf("  -s filename   The file name to create the share memory (Default: /dlt-shm)\n");
     217             :     printf("                (Applications wanting to connect to a daemon using a\n");
     218             :     printf("                custom shm name need to be started with the environment \n");
     219             :     printf("                variable DLT_SHM_NAME set appropriately)\n");
     220             : #endif
     221             :     printf("  -p port       port to monitor for incoming requests (Default: 3490)\n");
     222             :     printf("                (Applications wanting to connect to a daemon using a custom\n");
     223             :     printf("                port need to be started with the environment variable\n");
     224             :     printf("                DLT_DAEMON_TCP_PORT set appropriately)\n");
     225             : #ifdef DLT_LOG_LEVEL_APP_CONFIG
     226             :     printf("  -a filename   The filename for load default app id log levels (Default: " CONFIGURATION_FILES_DIR "/dlt-log-levels.conf)\n");
     227             : #endif
     228             : 
     229             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
     230             :     printf("  -l filename   The filename for load limits (Default: " CONFIGURATION_FILES_DIR "/dlt-trace-load.conf)\n");
     231             : #endif
     232             : 
     233             : #
     234           0 : } /* usage() */
     235             : 
     236             : /**
     237             :  * Option handling
     238             :  */
     239           9 : int option_handling(DltDaemonLocal *daemon_local, int argc, char *argv[])
     240             : {
     241             :     int c;
     242             :     char options[255];
     243             :     memset(options, 0, sizeof options);
     244             :     const char *const default_options = "hdc:t:p:";
     245             :     strcpy(options, default_options);
     246             : 
     247           9 :     if (daemon_local == 0) {
     248           0 :         fprintf (stderr, "Invalid parameter passed to option_handling()\n");
     249           0 :         return -1;
     250             :     }
     251             : 
     252             :     /* Initialize flags */
     253             :     memset(daemon_local, 0, sizeof(DltDaemonLocal));
     254             : 
     255             :     /* default values */
     256           9 :     daemon_local->flags.port = DLT_DAEMON_TCP_PORT;
     257             : 
     258             : #ifdef DLT_DAEMON_USE_FIFO_IPC
     259           9 :     dlt_log_set_fifo_basedir(DLT_USER_IPC_PATH);
     260             : #endif
     261             : 
     262             : #ifdef DLT_SHM_ENABLE
     263             :     strncpy(dltShmName, "/dlt-shm", NAME_MAX);
     264             : #endif
     265             : 
     266           9 :     opterr = 0;
     267             : 
     268             : #ifdef DLT_SHM_ENABLE
     269             :     strcpy(options + strlen(options), "s:");
     270             : #endif
     271             : #ifdef DLT_LOG_LEVEL_APP_CONFIG
     272             :     strcpy(options + strlen(options), "a:");
     273             : #endif
     274             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
     275             :     strcpy(options + strlen(options), "l:");
     276             : #endif
     277          20 :     while ((c = getopt(argc, argv, options)) != -1)
     278          11 :         switch (c) {
     279           2 :         case 'd':
     280             :         {
     281           2 :             daemon_local->flags.dflag = 1;
     282           2 :             break;
     283             :         }
     284           8 :         case 'c':
     285             :         {
     286           8 :             strncpy(daemon_local->flags.cvalue, optarg, NAME_MAX);
     287             :             break;
     288             :         }
     289             : #ifdef DLT_LOG_LEVEL_APP_CONFIG
     290             :         case 'a':
     291             :         {
     292             :             strncpy(daemon_local->flags.avalue, optarg, NAME_MAX);
     293             :             break;
     294             :         }
     295             : #endif
     296             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
     297             :         case 'l':
     298             :         {
     299             :             strncpy(daemon_local->flags.lvalue, optarg, NAME_MAX);
     300             :             break;
     301             :         }
     302             : #endif
     303             : #ifdef DLT_DAEMON_USE_FIFO_IPC
     304           0 :         case 't':
     305             :         {
     306           0 :             dlt_log_set_fifo_basedir(optarg);
     307           0 :             break;
     308             :         }
     309             : #endif
     310             : 
     311             : #ifdef DLT_SHM_ENABLE
     312             :         case 's':
     313             :         {
     314             :             strncpy(dltShmName, optarg, NAME_MAX);
     315             :             break;
     316             :         }
     317             : #endif
     318           1 :         case 'p':
     319             :         {
     320           1 :             daemon_local->flags.port = (unsigned int) atoi(optarg);
     321             : 
     322           1 :             if (daemon_local->flags.port == 0) {
     323           0 :                 fprintf (stderr, "Invalid port `%s' specified.\n", optarg);
     324           0 :                 return -1;
     325             :             }
     326             : 
     327             :             break;
     328             :         }
     329           0 :         case 'h':
     330             :         {
     331           0 :             usage();
     332           0 :             return -2; /* return no error */
     333             :         }
     334             : 
     335           0 :         case '?':
     336             :         {
     337           0 :             if ((optopt == 'c') || (optopt == 't') || (optopt == 'p')
     338             :     #ifdef DLT_LOG_LEVEL_APP_CONFIG
     339             :                   || (optopt == 'a')
     340             :     #endif
     341             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
     342             :                 || (optopt == 'l')
     343             : #endif
     344             :           )
     345           0 :                 fprintf (stderr, "Option -%c requires an argument.\n", optopt);
     346           0 :             else if (isprint (optopt))
     347           0 :                 fprintf (stderr, "Unknown option `-%c'.\n", optopt);
     348             :             else
     349           0 :                 fprintf (stderr, "Unknown option character `\\x%x'.\n", (uint32_t)optopt);
     350             : 
     351             :             /* unknown or wrong option used, show usage information and terminate */
     352           0 :             usage();
     353           0 :             return -1;
     354             :         }
     355           0 :         default:
     356             :         {
     357           0 :             fprintf (stderr, "Invalid option, this should never occur!\n");
     358           0 :             return -1;
     359             :         }
     360             :         }
     361             : 
     362             :     /* switch() */
     363             : 
     364             : #ifdef DLT_DAEMON_USE_FIFO_IPC
     365           9 :     snprintf(daemon_local->flags.userPipesDir, DLT_PATH_MAX,
     366             :              "%s/dltpipes", dltFifoBaseDir);
     367           9 :     snprintf(daemon_local->flags.daemonFifoName, DLT_PATH_MAX,
     368             :              "%s/dlt", dltFifoBaseDir);
     369             : #endif
     370             : 
     371             : #ifdef DLT_SHM_ENABLE
     372             :     strncpy(daemon_local->flags.dltShmName, dltShmName, NAME_MAX);
     373             : #endif
     374             : 
     375           9 :     return 0;
     376             : 
     377             : }  /* option_handling() */
     378             : 
     379             : /**
     380             :  * Option file parser
     381             :  */
     382           9 : int option_file_parser(DltDaemonLocal *daemon_local)
     383           9 : {
     384             :     FILE *pFile;
     385           9 :     char line[value_length - 1];
     386           9 :     char token[value_length];
     387           9 :     char value[value_length];
     388             :     char *pch;
     389             :     const char *filename;
     390             :     ssize_t n;
     391             : 
     392             :     /* set default values for configuration */
     393           9 :     daemon_local->flags.sharedMemorySize = DLT_SHM_SIZE;
     394           9 :     daemon_local->flags.sendMessageTime = 0;
     395           9 :     daemon_local->flags.offlineTraceDirectory[0] = 0;
     396           9 :     daemon_local->flags.offlineTraceFileSize = 1000000;
     397           9 :     daemon_local->flags.offlineTraceMaxSize = 4000000;
     398           9 :     daemon_local->flags.offlineTraceFilenameTimestampBased = true;
     399           9 :     daemon_local->flags.loggingMode = DLT_LOG_TO_CONSOLE;
     400           9 :     daemon_local->flags.loggingLevel = LOG_INFO;
     401             : 
     402             : #ifdef DLT_DAEMON_USE_UNIX_SOCKET_IPC
     403             :     n = snprintf(daemon_local->flags.loggingFilename,
     404             :                  sizeof(daemon_local->flags.loggingFilename),
     405             :                  "%s/dlt.log", DLT_USER_IPC_PATH);
     406             : #else /* DLT_DAEMON_USE_FIFO_IPC */
     407           9 :     n = snprintf(daemon_local->flags.loggingFilename,
     408             :                  sizeof(daemon_local->flags.loggingFilename),
     409             :                  "%s/dlt.log", dltFifoBaseDir);
     410             : #endif
     411             : 
     412           9 :     if (n < 0 || (size_t)n > sizeof(daemon_local->flags.loggingFilename)) {
     413           0 :         dlt_vlog(LOG_WARNING, "%s: snprintf truncation/error(%ld) %s\n",
     414             :                 __func__, n, daemon_local->flags.loggingFilename);
     415             :     }
     416           9 :     daemon_local->flags.enableLoggingFileLimit = false;
     417           9 :     daemon_local->flags.loggingFileSize = 250000;
     418           9 :     daemon_local->flags.loggingFileMaxSize = 1000000;
     419             : 
     420           9 :     daemon_local->timeoutOnSend = 4;
     421           9 :     daemon_local->RingbufferMinSize = DLT_DAEMON_RINGBUFFER_MIN_SIZE;
     422           9 :     daemon_local->RingbufferMaxSize = DLT_DAEMON_RINGBUFFER_MAX_SIZE;
     423           9 :     daemon_local->RingbufferStepSize = DLT_DAEMON_RINGBUFFER_STEP_SIZE;
     424           9 :     daemon_local->daemonFifoSize = 0;
     425           9 :     daemon_local->flags.sendECUSoftwareVersion = 0;
     426           9 :     memset(daemon_local->flags.pathToECUSoftwareVersion, 0, sizeof(daemon_local->flags.pathToECUSoftwareVersion));
     427           9 :     memset(daemon_local->flags.ecuSoftwareVersionFileField, 0, sizeof(daemon_local->flags.ecuSoftwareVersionFileField));
     428           9 :     daemon_local->flags.sendTimezone = 0;
     429           9 :     daemon_local->flags.offlineLogstorageMaxDevices = 0;
     430           9 :     daemon_local->flags.offlineLogstorageDirPath[0] = 0;
     431           9 :     daemon_local->flags.offlineLogstorageTimestamp = 1;
     432           9 :     daemon_local->flags.offlineLogstorageDelimiter = '_';
     433           9 :     daemon_local->flags.offlineLogstorageMaxCounter = UINT_MAX;
     434           9 :     daemon_local->flags.offlineLogstorageMaxCounterIdx = 0;
     435           9 :     daemon_local->flags.offlineLogstorageOptionalCounter = false;
     436           9 :     daemon_local->flags.offlineLogstorageCacheSize = 30000; /* 30MB */
     437           9 :     dlt_daemon_logstorage_set_logstorage_cache_size(
     438             :         daemon_local->flags.offlineLogstorageCacheSize);
     439           9 :     strncpy(daemon_local->flags.ctrlSockPath,
     440             :             DLT_DAEMON_DEFAULT_CTRL_SOCK_PATH,
     441             :             sizeof(daemon_local->flags.ctrlSockPath));
     442             : #ifdef DLT_DAEMON_USE_UNIX_SOCKET_IPC
     443             :     snprintf(daemon_local->flags.appSockPath, DLT_IPC_PATH_MAX, "%s/dlt", DLT_USER_IPC_PATH);
     444             : 
     445             :     if (strlen(DLT_USER_IPC_PATH) > DLT_IPC_PATH_MAX)
     446             :         fprintf(stderr, "Provided path too long...trimming it to path[%s]\n",
     447             :                 daemon_local->flags.appSockPath);
     448             : 
     449             : #else /* DLT_DAEMON_USE_FIFO_IPC */
     450           9 :     memset(daemon_local->flags.daemonFifoGroup, 0, sizeof(daemon_local->flags.daemonFifoGroup));
     451             : #endif
     452           9 :     daemon_local->flags.gatewayMode = 0;
     453           9 :     strncpy(daemon_local->flags.gatewayConfigFile,
     454             :             DLT_GATEWAY_CONFIG_PATH,
     455             :             DLT_DAEMON_FLAG_MAX);
     456           9 :     daemon_local->flags.autoResponseGetLogInfoOption = 7;
     457           9 :     daemon_local->flags.contextLogLevel = DLT_LOG_INFO;
     458           9 :     daemon_local->flags.contextTraceStatus = DLT_TRACE_STATUS_OFF;
     459           9 :     daemon_local->flags.enforceContextLLAndTS = 0; /* default is off */
     460             : #ifdef UDP_CONNECTION_SUPPORT
     461             :     daemon_local->UDPConnectionSetup = MULTICAST_CONNECTION_ENABLED;
     462             :     strncpy(daemon_local->UDPMulticastIPAddress, MULTICASTIPADDRESS, MULTICASTIP_MAX_SIZE - 1);
     463             :     daemon_local->UDPMulticastIPPort = MULTICASTIPPORT;
     464             : #endif
     465           9 :     daemon_local->flags.ipNodes = NULL;
     466           9 :     daemon_local->flags.injectionMode = 1;
     467             : 
     468             :     /* open configuration file */
     469           9 :     if (daemon_local->flags.cvalue[0])
     470           8 :         filename = daemon_local->flags.cvalue;
     471             :     else
     472             :         filename = CONFIGURATION_FILES_DIR "/dlt.conf";
     473             : 
     474             :     /*printf("Load configuration from file: %s\n",filename); */
     475           9 :     pFile = fopen (filename, "r");
     476             : 
     477           9 :     if (pFile != NULL) {
     478             :         while (1) {
     479             :             /* fetch line from configuration file */
     480         112 :             if (fgets (line, value_length - 1, pFile) != NULL) {
     481          49 :                 pch = strtok (line, " =\r\n");
     482          49 :                 token[0] = 0;
     483          49 :                 value[0] = 0;
     484             : 
     485          92 :                 while (pch != NULL) {
     486          86 :                     if (strcmp(pch, "#") == 0)
     487             :                         break;
     488             : 
     489          86 :                     if (token[0] == 0) {
     490          43 :                         strncpy(token, pch, sizeof(token) - 1);
     491          43 :                         token[sizeof(token) - 1] = 0;
     492             :                     }
     493             :                     else {
     494          43 :                         strncpy(value, pch, sizeof(value) - 1);
     495          43 :                         value[sizeof(value) - 1] = 0;
     496          43 :                         break;
     497             :                     }
     498             : 
     499          43 :                     pch = strtok (NULL, " =\r\n");
     500             :                 }
     501             : 
     502          49 :                 if (token[0] && value[0]) {
     503             :                     /* parse arguments here */
     504          43 :                     if (strcmp(token, "Verbose") == 0) {
     505           0 :                         daemon_local->flags.vflag = atoi(value);
     506             :                         /*printf("Option: %s=%s\n",token,value); */
     507             :                     }
     508          43 :                     else if (strcmp(token, "PrintASCII") == 0)
     509             :                     {
     510           0 :                         daemon_local->flags.aflag = atoi(value);
     511             :                         /*printf("Option: %s=%s\n",token,value); */
     512             :                     }
     513          43 :                     else if (strcmp(token, "PrintHex") == 0)
     514             :                     {
     515           0 :                         daemon_local->flags.xflag = atoi(value);
     516             :                         /*printf("Option: %s=%s\n",token,value); */
     517             :                     }
     518          43 :                     else if (strcmp(token, "PrintHeadersOnly") == 0)
     519             :                     {
     520           0 :                         daemon_local->flags.sflag = atoi(value);
     521             :                         /*printf("Option: %s=%s\n",token,value); */
     522             :                     }
     523          43 :                     else if (strcmp(token, "SendSerialHeader") == 0)
     524             :                     {
     525           0 :                         daemon_local->flags.lflag = atoi(value);
     526             :                         /*printf("Option: %s=%s\n",token,value); */
     527             :                     }
     528          43 :                     else if (strcmp(token, "SendContextRegistration") == 0)
     529             :                     {
     530           1 :                         daemon_local->flags.rflag = atoi(value);
     531             :                         /*printf("Option: %s=%s\n",token,value); */
     532             :                     }
     533          42 :                     else if (strcmp(token, "SendContextRegistrationOption") == 0)
     534             :                     {
     535           0 :                         daemon_local->flags.autoResponseGetLogInfoOption = atoi(value);
     536             :                         /*printf("Option: %s=%s\n",token,value); */
     537             :                     }
     538          42 :                     else if (strcmp(token, "SendMessageTime") == 0)
     539             :                     {
     540           0 :                         daemon_local->flags.sendMessageTime = atoi(value);
     541             :                         /*printf("Option: %s=%s\n",token,value); */
     542             :                     }
     543          42 :                     else if (strcmp(token, "RS232SyncSerialHeader") == 0)
     544             :                     {
     545           0 :                         daemon_local->flags.mflag = atoi(value);
     546             :                         /*printf("Option: %s=%s\n",token,value); */
     547             :                     }
     548          42 :                     else if (strcmp(token, "TCPSyncSerialHeader") == 0)
     549             :                     {
     550           0 :                         daemon_local->flags.nflag = atoi(value);
     551             :                         /*printf("Option: %s=%s\n",token,value); */
     552             :                     }
     553          42 :                     else if (strcmp(token, "RS232DeviceName") == 0)
     554             :                     {
     555           0 :                         strncpy(daemon_local->flags.yvalue, value, NAME_MAX);
     556           0 :                         daemon_local->flags.yvalue[NAME_MAX] = 0;
     557             :                         /*printf("Option: %s=%s\n",token,value); */
     558             :                     }
     559          42 :                     else if (strcmp(token, "RS232Baudrate") == 0)
     560             :                     {
     561           0 :                         strncpy(daemon_local->flags.bvalue, value, NAME_MAX);
     562           0 :                         daemon_local->flags.bvalue[NAME_MAX] = 0;
     563             :                         /*printf("Option: %s=%s\n",token,value); */
     564             :                     }
     565          42 :                     else if (strcmp(token, "ECUId") == 0)
     566             :                     {
     567           1 :                         strncpy(daemon_local->flags.evalue, value, NAME_MAX);
     568           1 :                         daemon_local->flags.evalue[NAME_MAX] = 0;
     569             :                         /*printf("Option: %s=%s\n",token,value); */
     570             :                     }
     571          41 :                     else if (strcmp(token, "PersistanceStoragePath") == 0)
     572             :                     {
     573           0 :                         strncpy(daemon_local->flags.ivalue, value, NAME_MAX);
     574           0 :                         daemon_local->flags.ivalue[NAME_MAX] = 0;
     575             :                         /*printf("Option: %s=%s\n",token,value); */
     576             :                     }
     577          41 :                     else if (strcmp(token, "LoggingMode") == 0)
     578             :                     {
     579           1 :                         daemon_local->flags.loggingMode = (DltLoggingMode)atoi(value);
     580             :                         /*printf("Option: %s=%s\n",token,value); */
     581             :                     }
     582          40 :                     else if (strcmp(token, "LoggingLevel") == 0)
     583             :                     {
     584           1 :                         daemon_local->flags.loggingLevel = atoi(value);
     585             :                         /*printf("Option: %s=%s\n",token,value); */
     586             :                     }
     587          39 :                     else if (strcmp(token, "LoggingFilename") == 0)
     588             :                     {
     589             :                         strncpy(daemon_local->flags.loggingFilename,
     590             :                                 value,
     591             :                                 sizeof(daemon_local->flags.loggingFilename) - 1);
     592           1 :                         daemon_local->flags.loggingFilename[sizeof(daemon_local->flags.loggingFilename) - 1] = 0;
     593             :                         /*printf("Option: %s=%s\n",token,value); */
     594             :                     }
     595          38 :                     else if (strcmp(token, "EnableLoggingFileLimit") == 0)
     596             :                     {
     597           0 :                         daemon_local->flags.enableLoggingFileLimit = (bool)atoi(value);
     598             :                     }
     599          38 :                     else if (strcmp(token, "LoggingFileSize") == 0)
     600             :                     {
     601           0 :                         daemon_local->flags.loggingFileSize = atoi(value);
     602             :                     }
     603          38 :                     else if (strcmp(token, "LoggingFileMaxSize") == 0)
     604             :                     {
     605           0 :                         daemon_local->flags.loggingFileMaxSize = atoi(value);
     606             :                     }
     607          38 :                     else if (strcmp(token, "TimeOutOnSend") == 0)
     608             :                     {
     609           1 :                         daemon_local->timeoutOnSend = atoi(value);
     610             :                         /*printf("Option: %s=%s\n",token,value); */
     611             :                     }
     612          37 :                     else if (strcmp(token, "RingbufferMinSize") == 0)
     613             :                     {
     614           1 :                         if (dlt_daemon_check_numeric_setting(token,
     615             :                                 value, &(daemon_local->RingbufferMinSize)) < 0) {
     616           0 :                             fclose (pFile);
     617           0 :                             return -1;
     618             :                         }
     619             :                     }
     620          36 :                     else if (strcmp(token, "RingbufferMaxSize") == 0)
     621             :                     {
     622           1 :                         if (dlt_daemon_check_numeric_setting(token,
     623             :                                 value, &(daemon_local->RingbufferMaxSize)) < 0) {
     624           0 :                             fclose (pFile);
     625           0 :                             return -1;
     626             :                         }
     627             :                     }
     628          35 :                     else if (strcmp(token, "RingbufferStepSize") == 0)
     629             :                     {
     630           1 :                         if (dlt_daemon_check_numeric_setting(token,
     631             :                                 value, &(daemon_local->RingbufferStepSize)) < 0) {
     632           0 :                             fclose (pFile);
     633           0 :                             return -1;
     634             :                         }
     635             :                     }
     636          34 :                     else if (strcmp(token, "SharedMemorySize") == 0)
     637             :                     {
     638           1 :                         daemon_local->flags.sharedMemorySize = atoi(value);
     639             :                         /*printf("Option: %s=%s\n",token,value); */
     640             :                     }
     641          33 :                     else if (strcmp(token, "OfflineTraceDirectory") == 0)
     642             :                     {
     643           0 :                         strncpy(daemon_local->flags.offlineTraceDirectory, value,
     644             :                                 sizeof(daemon_local->flags.offlineTraceDirectory) - 1);
     645             :                         daemon_local->flags.offlineTraceDirectory[sizeof(daemon_local->flags.offlineTraceDirectory) -
     646           0 :                                                                   1] = 0;
     647             :                         /*printf("Option: %s=%s\n",token,value); */
     648             :                     }
     649          33 :                     else if (strcmp(token, "OfflineTraceFileSize") == 0)
     650             :                     {
     651           0 :                         daemon_local->flags.offlineTraceFileSize = atoi(value);
     652             :                         /*printf("Option: %s=%s\n",token,value); */
     653             :                     }
     654          33 :                     else if (strcmp(token, "OfflineTraceMaxSize") == 0)
     655             :                     {
     656           0 :                         daemon_local->flags.offlineTraceMaxSize = atoi(value);
     657             :                         /*printf("Option: %s=%s\n",token,value); */
     658             :                     }
     659          33 :                     else if (strcmp(token, "OfflineTraceFileNameTimestampBased") == 0)
     660             :                     {
     661           0 :                         daemon_local->flags.offlineTraceFilenameTimestampBased = (bool)atoi(value);
     662             :                         /*printf("Option: %s=%s\n",token,value); */
     663             :                     }
     664          33 :                     else if (strcmp(token, "SendECUSoftwareVersion") == 0)
     665             :                     {
     666           0 :                         daemon_local->flags.sendECUSoftwareVersion = atoi(value);
     667             :                         /*printf("Option: %s=%s\n",token,value); */
     668             :                     }
     669          33 :                     else if (strcmp(token, "PathToECUSoftwareVersion") == 0)
     670             :                     {
     671             :                         strncpy(daemon_local->flags.pathToECUSoftwareVersion, value,
     672             :                                 sizeof(daemon_local->flags.pathToECUSoftwareVersion) - 1);
     673             :                         daemon_local->flags.pathToECUSoftwareVersion[sizeof(daemon_local->flags.pathToECUSoftwareVersion)
     674           0 :                                                                      - 1] = 0;
     675             :                         /*printf("Option: %s=%s\n",token,value); */
     676             :                     }
     677          33 :                     else if (strcmp(token, "ECUSoftwareVersionFileField") == 0) {
     678             :                         strncpy(daemon_local->flags.ecuSoftwareVersionFileField, value,
     679             :                                 sizeof(daemon_local->flags.ecuSoftwareVersionFileField) - 1);
     680             :                         daemon_local->flags.ecuSoftwareVersionFileField[sizeof(daemon_local->flags.ecuSoftwareVersionFileField)
     681           0 :                                                                      - 1] = 0;
     682             :                         /*printf("Option: %s=%s\n",token,value); */
     683             :                     }
     684          33 :                     else if (strcmp(token, "SendTimezone") == 0)
     685             :                     {
     686           0 :                         daemon_local->flags.sendTimezone = atoi(value);
     687             :                         /*printf("Option: %s=%s\n",token,value); */
     688             :                     }
     689          33 :                     else if (strcmp(token, "OfflineLogstorageMaxDevices") == 0)
     690             :                     {
     691           6 :                         daemon_local->flags.offlineLogstorageMaxDevices = (uint32_t) atoi(value);
     692             :                     }
     693          27 :                     else if (strcmp(token, "OfflineLogstorageDirPath") == 0)
     694             :                     {
     695           6 :                         strncpy(daemon_local->flags.offlineLogstorageDirPath,
     696             :                                 value,
     697             :                                 sizeof(daemon_local->flags.offlineLogstorageDirPath) - 1);
     698             :                     }
     699          21 :                     else if (strcmp(token, "OfflineLogstorageTimestamp") == 0)
     700             :                     {
     701             :                         /* Check if set to 0, default otherwise */
     702           6 :                         if (atoi(value) == 0)
     703           6 :                             daemon_local->flags.offlineLogstorageTimestamp = 0;
     704             :                     }
     705          15 :                     else if (strcmp(token, "OfflineLogstorageDelimiter") == 0)
     706             :                     {
     707             :                         /* Check if valid punctuation, default otherwise*/
     708           0 :                         if (ispunct((char)value[0]))
     709           0 :                             daemon_local->flags.offlineLogstorageDelimiter = (char)value[0];
     710             :                     }
     711          15 :                     else if (strcmp(token, "OfflineLogstorageMaxCounter") == 0)
     712             :                     {
     713           0 :                         daemon_local->flags.offlineLogstorageMaxCounter = (unsigned int) atoi(value);
     714           0 :                         daemon_local->flags.offlineLogstorageMaxCounterIdx = (unsigned int) strlen(value);
     715          15 :                     } else if (strcmp(token, "OfflineLogstorageOptionalIndex") == 0) {
     716           6 :                         daemon_local->flags.offlineLogstorageOptionalCounter = atoi(value);
     717             :                     }
     718           9 :                     else if (strcmp(token, "OfflineLogstorageCacheSize") == 0)
     719             :                     {
     720           0 :                         daemon_local->flags.offlineLogstorageCacheSize =
     721           0 :                             (unsigned int)atoi(value);
     722           0 :                         dlt_daemon_logstorage_set_logstorage_cache_size(
     723             :                             daemon_local->flags.offlineLogstorageCacheSize);
     724             :                     }
     725           9 :                     else if (strcmp(token, "ControlSocketPath") == 0)
     726             :                     {
     727             :                         memset(
     728             :                             daemon_local->flags.ctrlSockPath,
     729             :                             0,
     730             :                             DLT_DAEMON_FLAG_MAX);
     731             :                         strncpy(
     732             :                             daemon_local->flags.ctrlSockPath,
     733             :                             value,
     734             :                             DLT_DAEMON_FLAG_MAX - 1);
     735             :                     }
     736           2 :                     else if (strcmp(token, "GatewayMode") == 0)
     737             :                     {
     738           1 :                         daemon_local->flags.gatewayMode = atoi(value);
     739             :                         /*printf("Option: %s=%s\n",token,value); */
     740             :                     }
     741           1 :                     else if (strcmp(token, "GatewayConfigFile") == 0)
     742             :                     {
     743             :                         memset(
     744             :                             daemon_local->flags.gatewayConfigFile,
     745             :                             0,
     746             :                             DLT_DAEMON_FLAG_MAX);
     747             :                         strncpy(
     748             :                             daemon_local->flags.gatewayConfigFile,
     749             :                             value,
     750             :                             DLT_DAEMON_FLAG_MAX - 1);
     751             :                     }
     752           0 :                     else if (strcmp(token, "ContextLogLevel") == 0)
     753             :                     {
     754             :                         int const intval = atoi(value);
     755             : 
     756           0 :                         if ((intval >= DLT_LOG_OFF) && (intval <= DLT_LOG_VERBOSE)) {
     757           0 :                             daemon_local->flags.contextLogLevel = intval;
     758             :                             printf("Option: %s=%s\n", token, value);
     759             :                         }
     760             :                         else {
     761           0 :                             fprintf(stderr,
     762             :                                     "Invalid value for ContextLogLevel: %i. Must be in range [%i..%i]\n",
     763             :                                     intval,
     764             :                                     DLT_LOG_OFF,
     765             :                                     DLT_LOG_VERBOSE);
     766             :                         }
     767             :                     }
     768           0 :                     else if (strcmp(token, "ContextTraceStatus") == 0)
     769             :                     {
     770             :                         int const intval = atoi(value);
     771             : 
     772           0 :                         if ((intval >= DLT_TRACE_STATUS_OFF) && (intval <= DLT_TRACE_STATUS_ON)) {
     773           0 :                             daemon_local->flags.contextTraceStatus = intval;
     774             :                             printf("Option: %s=%s\n", token, value);
     775             :                         }
     776             :                         else {
     777           0 :                             fprintf(stderr,
     778             :                                     "Invalid value for ContextTraceStatus: %i. Must be in range [%i..%i]\n",
     779             :                                     intval,
     780             :                                     DLT_TRACE_STATUS_OFF,
     781             :                                     DLT_TRACE_STATUS_ON);
     782             :                         }
     783             :                     }
     784           0 :                     else if (strcmp(token, "ForceContextLogLevelAndTraceStatus") == 0)
     785             :                     {
     786             :                         int const intval = atoi(value);
     787             : 
     788           0 :                         if ((intval >= 0) && (intval <= 1)) {
     789           0 :                             daemon_local->flags.enforceContextLLAndTS = intval;
     790             :                             printf("Option: %s=%s\n", token, value);
     791             :                         }
     792             :                         else {
     793           0 :                             fprintf(stderr,
     794             :                                     "Invalid value for ForceContextLogLevelAndTraceStatus: %i. Must be 0, 1\n",
     795             :                                     intval);
     796             :                         }
     797             :                     }
     798             : 
     799             : #ifdef DLT_DAEMON_USE_FIFO_IPC
     800           0 :                     else if (strcmp(token, "DaemonFIFOSize") == 0)
     801             :                     {
     802           0 :                         if (dlt_daemon_check_numeric_setting(token,
     803             :                                 value, &(daemon_local->daemonFifoSize)) < 0) {
     804           0 :                             fclose (pFile);
     805           0 :                             return -1;
     806             :                         }
     807             : #ifndef __linux__
     808             :                             printf("Option DaemonFIFOSize is set but only supported on Linux. Ignored.\n");
     809             : #endif
     810             :                     }
     811           0 :                     else if (strcmp(token, "DaemonFifoGroup") == 0)
     812             :                     {
     813             :                         strncpy(daemon_local->flags.daemonFifoGroup, value, NAME_MAX);
     814           0 :                         daemon_local->flags.daemonFifoGroup[NAME_MAX] = 0;
     815             :                     }
     816             : #endif
     817             : #ifdef UDP_CONNECTION_SUPPORT
     818             :                     else if (strcmp(token, "UDPConnectionSetup") == 0)
     819             :                     {
     820             :                         const long longval = strtol(value, NULL, 10);
     821             : 
     822             :                         if ((longval == MULTICAST_CONNECTION_DISABLED)
     823             :                             || (longval == MULTICAST_CONNECTION_ENABLED)) {
     824             :                             daemon_local->UDPConnectionSetup = longval;
     825             :                             printf("Option: %s=%s\n", token, value);
     826             :                         }
     827             :                         else {
     828             :                             daemon_local->UDPConnectionSetup = MULTICAST_CONNECTION_DISABLED;
     829             :                             fprintf(stderr,
     830             :                                     "Invalid value for UDPConnectionSetup set to default %ld\n",
     831             :                                     longval);
     832             :                         }
     833             :                     }
     834             :                     else if (strcmp(token, "UDPMulticastIPAddress") == 0)
     835             :                     {
     836             :                         strncpy(daemon_local->UDPMulticastIPAddress, value,
     837             :                                 MULTICASTIP_MAX_SIZE - 1);
     838             :                     }
     839             :                     else if (strcmp(token, "UDPMulticastIPPort") == 0)
     840             :                     {
     841             :                         daemon_local->UDPMulticastIPPort = strtol(value, NULL, 10);
     842             :                     }
     843             : #endif
     844           0 :                     else if (strcmp(token, "BindAddress") == 0)
     845             :                     {
     846             :                         DltBindAddress_t *newNode = NULL;
     847             :                         DltBindAddress_t *temp = NULL;
     848             : 
     849           0 :                         char *tok = strtok(value, ",;");
     850             : 
     851           0 :                         if (tok != NULL) {
     852           0 :                             daemon_local->flags.ipNodes = calloc(1, sizeof(DltBindAddress_t));
     853             : 
     854           0 :                             if (daemon_local->flags.ipNodes == NULL) {
     855           0 :                                 dlt_vlog(LOG_ERR, "Could not allocate for IP list\n");
     856           0 :                                 fclose(pFile);
     857           0 :                                 return -1;
     858             :                             }
     859             :                             else {
     860           0 :                                 strncpy(daemon_local->flags.ipNodes->ip,
     861             :                                         tok,
     862             :                                         sizeof(daemon_local->flags.ipNodes->ip) - 1);
     863           0 :                                 daemon_local->flags.ipNodes->next = NULL;
     864             :                                 temp = daemon_local->flags.ipNodes;
     865             : 
     866           0 :                                 tok = strtok(NULL, ",;");
     867             : 
     868           0 :                                 while (tok != NULL) {
     869           0 :                                     newNode = calloc(1, sizeof(DltBindAddress_t));
     870             : 
     871           0 :                                     if (newNode == NULL) {
     872           0 :                                         dlt_vlog(LOG_ERR, "Could not allocate for IP list\n");
     873           0 :                                         fclose(pFile);
     874           0 :                                         return -1;
     875             :                                     }
     876             :                                     else {
     877           0 :                                         strncpy(newNode->ip, tok, sizeof(newNode->ip) - 1);
     878             :                                     }
     879             : 
     880           0 :                                     temp->next = newNode;
     881             :                                     temp = temp->next;
     882           0 :                                     tok = strtok(NULL, ",;");
     883             :                                 }
     884             :                             }
     885             :                         }
     886             :                         else {
     887           0 :                             dlt_vlog(LOG_WARNING, "BindAddress option is empty\n");
     888             :                         }
     889             :                     }
     890           0 :                     else if (strcmp(token, "InjectionMode") == 0) {
     891           0 :                         daemon_local->flags.injectionMode = atoi(value);
     892             :                     }
     893             :                     else {
     894           0 :                         fprintf(stderr, "Unknown option: %s=%s\n", token, value);
     895             :                     }
     896             :                 }
     897             :             }
     898             :             else {
     899             :                 break;
     900             :             }
     901             :         }
     902             : 
     903           7 :         fclose (pFile);
     904             :     }
     905             :     else {
     906           2 :         fprintf(stderr, "Cannot open configuration file: %s\n", filename);
     907             :     }
     908             : 
     909             :     return 0;
     910             : }
     911             : 
     912             : #ifdef DLT_LOG_LEVEL_APP_CONFIG
     913             : /**
     914             :  * Load configuration file parser
     915             :  */
     916             : 
     917             : static int compare_app_id_conf(const void *lhs, const void *rhs)
     918             : {
     919             :     return strncmp(((DltDaemonContextLogSettings *)lhs)->apid,
     920             :                    ((DltDaemonContextLogSettings *)rhs)->apid, DLT_ID_SIZE);
     921             : }
     922             : 
     923             : int app_id_default_log_level_config_parser(DltDaemon *daemon,
     924             :                                            DltDaemonLocal *daemon_local) {
     925             :     FILE *pFile;
     926             :     char line[value_length - 1];
     927             :     char app_id_value[value_length];
     928             :     char ctx_id_value[value_length];
     929             :     DltLogLevelType log_level_value;
     930             : 
     931             :     char *pch;
     932             :     const char *filename;
     933             : 
     934             :     /* open configuration file */
     935             :     filename = daemon_local->flags.avalue[0]
     936             :                    ? daemon_local->flags.avalue
     937             :                    : CONFIGURATION_FILES_DIR "/dlt-log-levels.conf";
     938             : 
     939             :     pFile = fopen(filename, "r");
     940             :     if (pFile == NULL) {
     941             :         dlt_vlog(LOG_WARNING, "Cannot open app log level configuration%s\n", filename);
     942             :         return -errno;
     943             :     }
     944             : 
     945             :     /* fetch lines from configuration file */
     946             :     while (fgets(line, value_length - 1, pFile) != NULL) {
     947             :         pch = strtok(line, " \t");
     948             :         memset(app_id_value, 0, value_length);
     949             :         memset(ctx_id_value, 0, value_length);
     950             :         log_level_value = DLT_LOG_MAX;
     951             : 
     952             :         /* ignore comments and new lines*/
     953             :         if (strncmp(pch, "#", 1) == 0 || strncmp(pch, "\n", 1) == 0
     954             :             || strlen(pch) < 1)
     955             :             continue;
     956             : 
     957             :         strncpy(app_id_value, pch, sizeof(app_id_value) - 1);
     958             :         app_id_value[sizeof(app_id_value) - 1] = 0;
     959             :         if (strlen(app_id_value) == 0 || strlen(app_id_value) > DLT_ID_SIZE) {
     960             :             if (app_id_value[strlen(app_id_value) - 1] == '\n') {
     961             :                 dlt_vlog(LOG_WARNING, "Missing log level for apid %s in log settings\n",
     962             :                          app_id_value);
     963             :             } else {
     964             :                 dlt_vlog(LOG_WARNING,
     965             :                          "Invalid apid for log settings settings: app id: %s\n",
     966             :                          app_id_value);
     967             :             }
     968             : 
     969             :             continue;
     970             :         }
     971             : 
     972             :         char *pch_next1 = strtok(NULL, " \t");
     973             :         char *pch_next2 = strtok(NULL, " \t");
     974             :         char *log_level;
     975             :         /* no context id given, log level is next token */
     976             :         if (pch_next2 == NULL || pch_next2[0] == '#') {
     977             :             log_level = pch_next1;
     978             :         } else {
     979             :             /* context id is given, log level is second to next token */
     980             :             log_level = pch_next2;
     981             : 
     982             :             /* next token is token id */
     983             :             strncpy(ctx_id_value, pch_next1, sizeof(ctx_id_value) - 1);
     984             :             if (strlen(ctx_id_value) == 0 || strlen(app_id_value) > DLT_ID_SIZE) {
     985             :                 dlt_vlog(LOG_WARNING,
     986             :                          "Invalid ctxid for log settings: app id: %s "
     987             :                          "(skipping line)\n",
     988             :                          app_id_value);
     989             :                 continue;
     990             :             }
     991             : 
     992             :             ctx_id_value[sizeof(ctx_id_value) - 1] = 0;
     993             :         }
     994             : 
     995             :         errno = 0;
     996             :         log_level_value = strtol(log_level, NULL, 10);
     997             :         if (errno != 0 || log_level_value >= DLT_LOG_MAX ||
     998             :             log_level_value <= DLT_LOG_DEFAULT) {
     999             :             dlt_vlog(LOG_WARNING,
    1000             :                      "Invalid log level (%i), app id %s, conversion error: %s\n",
    1001             :                      log_level_value, app_id_value, strerror(errno));
    1002             :             continue;
    1003             :         }
    1004             : 
    1005             :         DltDaemonContextLogSettings *settings =
    1006             :             dlt_daemon_find_configured_app_id_ctx_id_settings(
    1007             :                 daemon, app_id_value, NULL);
    1008             : 
    1009             :         if (settings != NULL &&
    1010             :             strncmp(settings->ctid, ctx_id_value, DLT_ID_SIZE) == 0) {
    1011             :             if (strlen(ctx_id_value) > 0) {
    1012             :                 dlt_vlog(LOG_WARNING,
    1013             :                          "Appid %s with ctxid %s is already configured, skipping "
    1014             :                          "duplicated entry\n",
    1015             :                          app_id_value, ctx_id_value);
    1016             :             } else {
    1017             :                 dlt_vlog(LOG_WARNING,
    1018             :                          "Appid %s is already configured, skipping duplicated entry\n",
    1019             :                          app_id_value);
    1020             :             }
    1021             : 
    1022             :             continue;
    1023             :         }
    1024             : 
    1025             :         /* allocate one more element in the trace load settings */
    1026             :         DltDaemonContextLogSettings *tmp =
    1027             :             realloc(daemon->app_id_log_level_settings,
    1028             :                     (++daemon->num_app_id_log_level_settings) *
    1029             :                         sizeof(DltDaemonContextLogSettings));
    1030             : 
    1031             :         if (tmp == NULL) {
    1032             :             dlt_log(LOG_CRIT, "Failed to allocate memory for app load settings\n");
    1033             :             continue;
    1034             :         }
    1035             : 
    1036             :         daemon->app_id_log_level_settings = tmp;
    1037             : 
    1038             :         /* update newly created entry */
    1039             :         settings = &daemon->app_id_log_level_settings
    1040             :                         [daemon->num_app_id_log_level_settings -1];
    1041             : 
    1042             :         memset(settings, 0, sizeof(DltDaemonContextLogSettings));
    1043             :         memcpy(settings->apid, app_id_value, DLT_ID_SIZE);
    1044             :         memcpy(settings->ctid, ctx_id_value, DLT_ID_SIZE);
    1045             :         settings->log_level = log_level_value;
    1046             : 
    1047             :         /* make sure ids are 0 terminated for printing */
    1048             :         char apid_buf[DLT_ID_SIZE + 1] = {0};
    1049             :         char ctid_buf[DLT_ID_SIZE + 1] = {0};
    1050             :         memcpy(apid_buf, app_id_value, DLT_ID_SIZE);
    1051             :         memcpy(ctid_buf, ctx_id_value, DLT_ID_SIZE);
    1052             : 
    1053             :         /* log with or without context id */
    1054             :         if (strlen(ctid_buf) > 0) {
    1055             :             dlt_vlog(
    1056             :                 LOG_INFO,
    1057             :                 "Configured trace limits for app id %s, context id %s, level %u\n",
    1058             :                 apid_buf, ctid_buf, log_level_value);
    1059             :         } else {
    1060             :             dlt_vlog(LOG_INFO, "Configured trace limits for app id %s, level %u\n",
    1061             :                      apid_buf, log_level_value);
    1062             :         }
    1063             : 
    1064             :     } /* while */
    1065             :     fclose(pFile);
    1066             : 
    1067             :     /* list must be sorted to speed up dlt_daemon_find_configured_app_id_ctx_id_settings */
    1068             :     qsort(daemon->app_id_log_level_settings,
    1069             :             daemon->num_app_id_log_level_settings,
    1070             :             sizeof(DltDaemonContextLogSettings), compare_app_id_conf);
    1071             : 
    1072             :     return 0;
    1073             : }
    1074             : #endif
    1075             : 
    1076             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
    1077             : static bool is_ascii_only(const char *str) {
    1078             :     while (*str) {
    1079             :         if ((unsigned char)*str > 127) {
    1080             :             return false;
    1081             :         }
    1082             :         str++;
    1083             :     }
    1084             :     return true;
    1085             : }
    1086             : 
    1087             : /**
    1088             :  * Load configuration file parser
    1089             :  */
    1090             : int trace_load_config_file_parser(DltDaemon *daemon, DltDaemonLocal *daemon_local)
    1091             : {
    1092             :     FILE *pFile;
    1093             :     const int max_tokens = 4;
    1094             :     const int min_tokens = 3;
    1095             :     char tokens[max_tokens][value_length];
    1096             :     char line[value_length - 1];
    1097             :     char app_id_value[value_length];
    1098             :     char ctx_id_value[value_length];
    1099             :     char soft_limit_value[value_length];
    1100             :     char hard_limit_value[value_length];
    1101             :     int i;
    1102             :     uint32_t soft_limit;
    1103             :     uint32_t hard_limit;
    1104             : 
    1105             :     char *pch;
    1106             :     const char *filename;
    1107             : 
    1108             :     bool skipped;
    1109             : 
    1110             :     if (daemon->preconfigured_trace_load_settings != NULL) {
    1111             :         free(daemon->preconfigured_trace_load_settings);
    1112             :         daemon->preconfigured_trace_load_settings = NULL;
    1113             :         daemon->preconfigured_trace_load_settings_count = 0;
    1114             :     }
    1115             :     daemon->preconfigured_trace_load_settings = malloc(sizeof(DltTraceLoadSettings));
    1116             :     if (daemon->preconfigured_trace_load_settings == NULL) {
    1117             :         dlt_log(LOG_CRIT, "Failed to allocate memory for trace load settings\n");
    1118             :         return DLT_RETURN_ERROR;
    1119             :     }
    1120             : 
    1121             :     /* open configuration file */
    1122             :     filename = daemon_local->flags.lvalue[0]
    1123             :                    ? daemon_local->flags.lvalue
    1124             :                    : CONFIGURATION_FILES_DIR "/dlt-trace-load.conf";
    1125             : 
    1126             :     pFile = fopen (filename, "r");
    1127             :     if (pFile == NULL) {
    1128             :         dlt_vlog(LOG_WARNING, "Cannot open trace load configuration file: %s\n",
    1129             :                  filename);
    1130             :         return -errno;
    1131             :     }
    1132             : 
    1133             :     while (1) {
    1134             :         /* fetch line from configuration file */
    1135             :         if (fgets(line, value_length - 1, pFile) == NULL) {
    1136             :             break;
    1137             :         }
    1138             : 
    1139             :         pch = strtok(line, " ");
    1140             :         app_id_value[0] = 0;
    1141             :         ctx_id_value[0] = 0;
    1142             :         soft_limit_value[0] = 0;
    1143             :         hard_limit_value[0] = 0;
    1144             :         soft_limit = 0U;
    1145             :         hard_limit = 0U;
    1146             :         memset(tokens, 0, sizeof(tokens));
    1147             :         i = 0;
    1148             : 
    1149             :         skipped = false;
    1150             :         while (pch != NULL && i < max_tokens) {
    1151             :             /* ignore comments, empty lines and new lines */
    1152             :             if (strncmp(pch, "#", 1) == 0 || strncmp(pch, "\n", 1) == 0 ||
    1153             :                 strncmp(pch, "\r", 1) == 0 || strncmp(pch, " ", 1) == 0) {
    1154             :                 skipped = true;
    1155             :                 break;
    1156             :             }
    1157             :             strncpy(tokens[i], pch, sizeof(tokens[i]) - 1);
    1158             :             pch = strtok(NULL, " ");
    1159             :             ++i;
    1160             :         }
    1161             : 
    1162             :         if (skipped && i < min_tokens)
    1163             :             continue;
    1164             : 
    1165             :         if (pch != NULL
    1166             :             && (pch[0] != '\n')
    1167             :             && (pch[0] != '\t')
    1168             :             && (pch[0] != ' ')
    1169             :             && (pch[0] != '#')) {
    1170             :             dlt_vlog(LOG_WARNING,
    1171             :                      "Invalid trace load settings: too many tokens in line '%s'\n", line);
    1172             :             continue;
    1173             :         }
    1174             : 
    1175             :         bool has_ctx_id = i == max_tokens;
    1176             :         int soft_limit_idx = has_ctx_id ? 2 : 1;
    1177             :         int hard_limit_idx = has_ctx_id ? 3 : 2;
    1178             : 
    1179             :         strncpy(app_id_value, tokens[0], sizeof(app_id_value) - 1);
    1180             :         if ((strlen(app_id_value) == 0)
    1181             :             || (strlen(app_id_value) > DLT_ID_SIZE)
    1182             :             || (!is_ascii_only(app_id_value))) {
    1183             :             dlt_vlog(LOG_WARNING,
    1184             :                      "Invalid apid for trace load settings: app id: '%s'\n", app_id_value);
    1185             :             continue;
    1186             :         }
    1187             : 
    1188             :         if (has_ctx_id) {
    1189             :             strncpy(ctx_id_value, tokens[1], sizeof(ctx_id_value) - 1);
    1190             :             if ((strlen(ctx_id_value) == 0)
    1191             :                 || (strlen(ctx_id_value) > DLT_ID_SIZE)
    1192             :                 || (!is_ascii_only(ctx_id_value))) {
    1193             :                 dlt_vlog(LOG_WARNING,
    1194             :                          "Invalid ctid for trace load settings: context id: '%s'\n", ctx_id_value);
    1195             :                 continue;
    1196             :             }
    1197             :         }
    1198             : 
    1199             :         if (strlen(tokens[soft_limit_idx]) == 0) {
    1200             :             dlt_vlog(LOG_WARNING,
    1201             :                      "Invalid soft_limit for trace load settings: app id: '%.4s', '%s'\n",
    1202             :                      app_id_value, tokens[soft_limit_idx]);
    1203             :             continue;
    1204             :         }
    1205             : 
    1206             :         if (strlen(tokens[hard_limit_idx]) == 0) {
    1207             :             dlt_vlog(LOG_WARNING,
    1208             :                      "Invalid hard_limit for trace load settings: app id: '%.4s', '%s'\n",
    1209             :                      app_id_value, tokens[hard_limit_idx]);
    1210             :             continue;
    1211             :         }
    1212             : 
    1213             :         strncpy(soft_limit_value, tokens[soft_limit_idx],
    1214             :                 sizeof(soft_limit_value) - 1);
    1215             :         strncpy(hard_limit_value, tokens[hard_limit_idx],
    1216             :                 sizeof(hard_limit_value) - 1);
    1217             : 
    1218             :         errno = 0;
    1219             :         char *endptr;
    1220             :         endptr = NULL;
    1221             :         soft_limit = strtoul(soft_limit_value, &endptr, 10);
    1222             :         if ((errno != 0)
    1223             :             || ((soft_limit == 0) && (soft_limit_value[0] != '0'))
    1224             :             || (soft_limit_value[0] == '-')
    1225             :             || ((*endptr != '\n') && (*endptr != '\0'))) {
    1226             :             dlt_vlog(LOG_WARNING,
    1227             :                      "Invalid soft_limit for trace load settings: app id: '%.4s', soft_limit '%s'\n",
    1228             :                      app_id_value, soft_limit_value);
    1229             :             continue;
    1230             :         }
    1231             : 
    1232             :         errno = 0;
    1233             :         endptr = NULL;
    1234             :         hard_limit = strtoul(hard_limit_value, &endptr, 10);
    1235             :         if ((errno != 0)
    1236             :             || ((hard_limit == 0) && (hard_limit_value[0] != '0'))
    1237             :             || (hard_limit_value[0] == '-')
    1238             :             || ((*endptr != '\n') && (*endptr != '\0'))) {
    1239             :             dlt_vlog(LOG_WARNING,
    1240             :                      "Invalid hard_limit for trace load settings: app id: '%.4s', hard_limit '%s'\n",
    1241             :                      app_id_value, hard_limit_value);
    1242             :             continue;
    1243             :         }
    1244             : 
    1245             :         if (soft_limit > hard_limit) {
    1246             :             dlt_vlog(LOG_WARNING,
    1247             :                      "Invalid trace load settings: app id: '%.4s', soft limit %u is greater than hard limit %u\n",
    1248             :                      app_id_value, soft_limit, hard_limit);
    1249             :             continue;
    1250             :         }
    1251             : 
    1252             :         DltTraceLoadSettings *settings = NULL;
    1253             :         int num_settings = 0;
    1254             :         DltReturnValue find_trace_settings_return_value = dlt_daemon_find_preconfigured_trace_load_settings(
    1255             :             daemon, app_id_value, ctx_id_value, &settings, &num_settings,
    1256             :             0);
    1257             :         if (find_trace_settings_return_value != DLT_RETURN_OK || num_settings != 0) {
    1258             :             dlt_vlog(LOG_WARNING,
    1259             :                      "App id '%.4s' is already configured, or an error occurred, skipping entry\n",
    1260             :                      app_id_value);
    1261             :             if (settings != NULL) {
    1262             :                 free(settings);
    1263             :             }
    1264             :             continue;
    1265             :         }
    1266             : 
    1267             :         /* allocate one more element in the trace load settings */
    1268             :         DltTraceLoadSettings *tmp =
    1269             :             realloc(daemon->preconfigured_trace_load_settings,
    1270             :                     (++daemon->preconfigured_trace_load_settings_count) *
    1271             :                         sizeof(DltTraceLoadSettings));
    1272             : 
    1273             :         if (tmp == NULL) {
    1274             :             dlt_log(LOG_CRIT,
    1275             :                     "Failed to allocate memory for trace load settings\n");
    1276             :             return DLT_RETURN_ERROR;
    1277             :         }
    1278             : 
    1279             :         daemon->preconfigured_trace_load_settings = tmp;
    1280             :         settings = &daemon->preconfigured_trace_load_settings
    1281             :                         [daemon->preconfigured_trace_load_settings_count - 1];
    1282             :         memset(settings, 0, sizeof(DltTraceLoadSettings));
    1283             :         settings->soft_limit = soft_limit;
    1284             :         settings->hard_limit = hard_limit;
    1285             : 
    1286             :         memcpy(settings->apid, app_id_value, DLT_ID_SIZE);
    1287             :         if (has_ctx_id) {
    1288             :             memcpy(settings->ctid, ctx_id_value, DLT_ID_SIZE);
    1289             :             dlt_vlog(LOG_INFO,
    1290             :                      "Configured trace limits for app id '%.4s', ctx id '%.4s', soft limit: %u, hard_limit: %u\n",
    1291             :                      app_id_value, ctx_id_value, soft_limit, hard_limit);
    1292             :         } else {
    1293             :             dlt_vlog(LOG_INFO,
    1294             :                      "Configured trace limits for app id '%.4s', soft limit: %u, hard_limit: %u\n",
    1295             :                      app_id_value, soft_limit, hard_limit);
    1296             :         }
    1297             : 
    1298             : 
    1299             : 
    1300             :     } /* while */
    1301             :     fclose(pFile);
    1302             : 
    1303             :     // sort limits to improve search performance
    1304             :     qsort(daemon->preconfigured_trace_load_settings, daemon->preconfigured_trace_load_settings_count,
    1305             :           sizeof(DltTraceLoadSettings),
    1306             :           dlt_daemon_compare_trace_load_settings);
    1307             :     return 0;
    1308             : }
    1309             : #endif
    1310             : 
    1311           9 : static int dlt_mkdir_recursive(const char *dir)
    1312             : {
    1313             :     int ret = 0;
    1314             :     char tmp[PATH_MAX + 1];
    1315             :     char *p = NULL;
    1316             :     char *end = NULL;
    1317             :     size_t len;
    1318             : 
    1319             :     strncpy(tmp, dir, PATH_MAX);
    1320           9 :     len = strlen(tmp);
    1321             : 
    1322           9 :     if (tmp[len - 1] == '/')
    1323           0 :         tmp[len - 1] = 0;
    1324             : 
    1325           9 :     end = tmp + len;
    1326             : 
    1327          36 :     for (p = tmp + 1; ((*p) && (ret == 0)) || ((ret == -1 && errno == EEXIST) && (p != end)); p++)
    1328          27 :         if (*p == '/') {
    1329           0 :             *p = 0;
    1330             : 
    1331           0 :             if (access(tmp, F_OK) != 0 && errno == ENOENT) {
    1332           0 :                 ret = mkdir(tmp,
    1333             :                 #ifdef DLT_DAEMON_USE_FIFO_IPC
    1334             :                                 S_IRWXU);
    1335             :                 #else
    1336             :                     S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IWGRP | S_IROTH  | S_IWOTH /*S_IRWXU*/);
    1337             :                 #endif
    1338             :             }
    1339             : 
    1340           0 :             *p = '/';
    1341             :         }
    1342             : 
    1343             : 
    1344             : 
    1345           9 :     if ((ret == 0) || ((ret == -1) && (errno == EEXIST)))
    1346           9 :         ret = mkdir(tmp,
    1347             :         #ifdef DLT_DAEMON_USE_FIFO_IPC
    1348             :                     S_IRWXU);
    1349             :         #else
    1350             :                     S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IWGRP | S_IROTH  | S_IWOTH /*S_IRWXU*/);
    1351             :         #endif
    1352             : 
    1353           9 :     if ((ret == -1) && (errno == EEXIST))
    1354             :         ret = 0;
    1355             : 
    1356           9 :     return ret;
    1357             : }
    1358             : 
    1359             : #ifdef DLT_DAEMON_USE_FIFO_IPC
    1360           9 : static DltReturnValue dlt_daemon_create_pipes_dir(char *dir)
    1361             : {
    1362             :     int ret = DLT_RETURN_OK;
    1363             : 
    1364           9 :     if (dir == NULL) {
    1365           0 :         dlt_vlog(LOG_ERR, "%s: Invalid parameter\n", __func__);
    1366           0 :         return DLT_RETURN_WRONG_PARAMETER;
    1367             :     }
    1368             : 
    1369             :     /* create dlt pipes directory */
    1370           9 :     ret = mkdir(dir,
    1371             :                 S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH | S_ISVTX);
    1372             : 
    1373           9 :     if ((ret == -1) && (errno != EEXIST)) {
    1374           0 :         dlt_vlog(LOG_ERR,
    1375             :                  "FIFO user dir %s cannot be created (%s)!\n",
    1376             :                  dir,
    1377             :                  strerror(errno));
    1378             : 
    1379           0 :         return DLT_RETURN_ERROR;
    1380             :     }
    1381             : 
    1382             :     /* S_ISGID cannot be set by mkdir, let's reassign right bits */
    1383           9 :     ret = chmod(dir,
    1384             :                 S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IWGRP | S_IXGRP | S_IROTH | S_IWOTH | S_IXOTH | S_ISGID |
    1385             :                 S_ISVTX);
    1386             : 
    1387           9 :     if (ret == -1) {
    1388           0 :         dlt_vlog(LOG_ERR,
    1389             :                  "FIFO user dir %s cannot be chmoded (%s)!\n",
    1390             :                  dir,
    1391           0 :                  strerror(errno));
    1392             : 
    1393           0 :         return DLT_RETURN_ERROR;
    1394             :     }
    1395             : 
    1396             :     return ret;
    1397             : }
    1398             : #endif
    1399             : // This will be defined when unit testing, so functions
    1400             : // from this file can be tested without defining main twice
    1401             : #ifndef DLT_DAEMON_UNIT_TESTS_NO_MAIN
    1402             : /**
    1403             :  * Main function of tool.
    1404             :  */
    1405           9 : int main(int argc, char *argv[])
    1406             : {
    1407             :     char version[DLT_DAEMON_TEXTBUFSIZE];
    1408             :     char local_str[DLT_DAEMON_TEXTBUFSIZE];
    1409             :     DltDaemonLocal daemon_local;
    1410             :     DltDaemon daemon;
    1411             :     int back = 0;
    1412             : 
    1413             :     memset(&daemon_local, 0, sizeof(DltDaemonLocal));
    1414             :     memset(&daemon, 0, sizeof(DltDaemon));
    1415             : 
    1416             :     /* Command line option handling */
    1417           9 :     if ((back = option_handling(&daemon_local, argc, argv)) < 0) {
    1418           0 :         if (back != -2)
    1419           0 :             fprintf (stderr, "option_handling() failed!\n");
    1420             : 
    1421           0 :         return -1;
    1422             :     }
    1423             : 
    1424             :     /* Configuration file option handling */
    1425           9 :     if ((back = option_file_parser(&daemon_local)) < 0) {
    1426           0 :         if (back != -2)
    1427           0 :             fprintf (stderr, "option_file_parser() failed!\n");
    1428             : 
    1429           0 :         return -1;
    1430             :     }
    1431             : 
    1432             :     /* Initialize internal logging facility */
    1433           9 :     dlt_log_set_filename(daemon_local.flags.loggingFilename);
    1434           9 :     dlt_log_set_level(daemon_local.flags.loggingLevel);
    1435             :     DltReturnValue log_init_result =
    1436           9 :             dlt_log_init_multiple_logfiles_support(daemon_local.flags.loggingMode,
    1437           9 :                                                    daemon_local.flags.enableLoggingFileLimit,
    1438             :                                                    daemon_local.flags.loggingFileSize,
    1439             :                                                    daemon_local.flags.loggingFileMaxSize);
    1440             : 
    1441           9 :     if (log_init_result != DLT_RETURN_OK) {
    1442           0 :       fprintf(stderr, "Failed to init internal logging\n");
    1443             : 
    1444             : #if WITH_DLT_FILE_LOGGING_SYSLOG_FALLBACK
    1445             :         if (daemon_local.flags.loggingMode == DLT_LOG_TO_FILE) {
    1446             :           fprintf(stderr, "Falling back to syslog mode\n");
    1447             : 
    1448             :           daemon_local.flags.loggingMode = DLT_LOG_TO_SYSLOG;
    1449             :           log_init_result = dlt_log_init(daemon_local.flags.loggingMode);
    1450             :           if (log_init_result != DLT_RETURN_OK) {
    1451             :             fprintf(stderr, "Failed to setup syslog logging, internal logs will "
    1452             :                             "not be available\n");
    1453             :           }
    1454             :       }
    1455             : #endif
    1456             :     }
    1457             : 
    1458             :     /* Print version information */
    1459           9 :     dlt_get_version(version, DLT_DAEMON_TEXTBUFSIZE);
    1460             : 
    1461           9 :     dlt_vlog(LOG_NOTICE, "Starting DLT Daemon; %s\n", version);
    1462             : 
    1463           9 :     PRINT_FUNCTION_VERBOSE(daemon_local.flags.vflag);
    1464             : 
    1465             : /* Make sure the parent user directory is created */
    1466             : #ifdef DLT_DAEMON_USE_FIFO_IPC
    1467             : 
    1468           9 :     if (dlt_mkdir_recursive(dltFifoBaseDir) != 0) {
    1469           0 :         dlt_vlog(LOG_ERR, "Base dir %s cannot be created!\n", dltFifoBaseDir);
    1470           0 :         return -1;
    1471             :   }
    1472             : 
    1473             : #else
    1474             :     if (dlt_mkdir_recursive(DLT_USER_IPC_PATH) != 0) {
    1475             :         dlt_vlog(LOG_ERR, "Base dir %s cannot be created!\n", daemon_local.flags.appSockPath);
    1476             :         return -1;
    1477             :     }
    1478             : 
    1479             : #endif
    1480             : 
    1481             :     /* --- Daemon init phase 1 begin --- */
    1482           9 :     if (dlt_daemon_local_init_p1(&daemon, &daemon_local, daemon_local.flags.vflag) == -1) {
    1483           0 :         dlt_log(LOG_CRIT, "Initialization of phase 1 failed!\n");
    1484           0 :         return -1;
    1485             :     }
    1486             : 
    1487             :     /* --- Daemon init phase 1 end --- */
    1488             : 
    1489           9 :     if (dlt_daemon_prepare_event_handling(&daemon_local.pEvent)) {
    1490             :         /* TODO: Perform clean-up */
    1491           0 :         dlt_log(LOG_CRIT, "Initialization of event handling failed!\n");
    1492           0 :         return -1;
    1493             :     }
    1494             : 
    1495             :     /* --- Daemon connection init begin */
    1496           9 :     if (dlt_daemon_local_connection_init(&daemon, &daemon_local, daemon_local.flags.vflag) == -1) {
    1497           0 :         dlt_log(LOG_CRIT, "Initialization of local connections failed!\n");
    1498           0 :         return -1;
    1499             :     }
    1500             : 
    1501             :     /* --- Daemon connection init end */
    1502             : 
    1503           9 :     if (dlt_daemon_init_runtime_configuration(&daemon, daemon_local.flags.ivalue, daemon_local.flags.vflag) == -1) {
    1504           0 :         dlt_log(LOG_ERR, "Could not load runtime config\n");
    1505           0 :         return -1;
    1506             :     }
    1507             : 
    1508             :     /*
    1509             :      * Load dlt-runtime.cfg if available.
    1510             :      * This must be loaded before offline setup
    1511             :      */
    1512           9 :     dlt_daemon_configuration_load(&daemon, daemon.runtime_configuration, daemon_local.flags.vflag);
    1513             : 
    1514             :     /* --- Daemon init phase 2 begin --- */
    1515           9 :     if (dlt_daemon_local_init_p2(&daemon, &daemon_local, daemon_local.flags.vflag) == -1) {
    1516           0 :         dlt_log(LOG_CRIT, "Initialization of phase 2 failed!\n");
    1517           0 :         return -1;
    1518             :     }
    1519             : 
    1520             : #ifdef DLT_LOG_LEVEL_APP_CONFIG
    1521             :     /* Load control app id level configuration file without setting `back` to
    1522             :      * prevent exit if file is missing */
    1523             :     if (app_id_default_log_level_config_parser(&daemon, &daemon_local) < 0) {
    1524             :         dlt_vlog(LOG_WARNING, "app_id_default_log_level_config_parser() failed, "
    1525             :                            "no app specific log levels will be configured\n");
    1526             :     }
    1527             : #endif
    1528             : 
    1529             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
    1530             :     /* Load control trace load configuration file without setting `back` to prevent exit if file is missing */
    1531             :     pthread_rwlock_init(&trace_load_rw_lock, NULL);
    1532             :     if (trace_load_config_file_parser(&daemon, &daemon_local) < 0) {
    1533             :         dlt_vlog(LOG_WARNING, "trace_load_config_file_parser() failed, using defaults for all app ids!\n");
    1534             :     }
    1535             : #endif
    1536             : 
    1537             :     /* --- Daemon init phase 2 end --- */
    1538             : 
    1539           9 :     if (daemon_local.flags.offlineLogstorageDirPath[0])
    1540           6 :         if (dlt_daemon_logstorage_setup_internal_storage(
    1541             :                 &daemon,
    1542             :                 &daemon_local,
    1543             :                 daemon_local.flags.offlineLogstorageDirPath,
    1544             :                 daemon_local.flags.vflag) == -1)
    1545           0 :             dlt_log(LOG_INFO,
    1546             :                     "Setting up internal offline log storage failed!\n");
    1547             : 
    1548             :     /* create fd for watchdog */
    1549             : #ifdef DLT_SYSTEMD_WATCHDOG_ENABLE
    1550             :     {
    1551             :         char *watchdogUSec = getenv("WATCHDOG_USEC");
    1552             :         int watchdogTimeoutSeconds = 0;
    1553             : 
    1554             :         dlt_log(LOG_DEBUG, "Systemd watchdog initialization\n");
    1555             : 
    1556             :         if (watchdogUSec)
    1557             :             watchdogTimeoutSeconds = atoi(watchdogUSec) / 2000000;
    1558             : 
    1559             :         daemon.watchdog_trigger_interval = watchdogTimeoutSeconds;
    1560             :         create_timer_fd(&daemon_local,
    1561             :                         watchdogTimeoutSeconds,
    1562             :                         watchdogTimeoutSeconds,
    1563             :                         DLT_TIMER_SYSTEMD);
    1564             :     }
    1565             : #endif
    1566             : 
    1567             :     /* create fd for timer timing packets */
    1568           9 :     create_timer_fd(&daemon_local, 1, 1, DLT_TIMER_PACKET);
    1569             : 
    1570             :     /* create fd for timer ecu version */
    1571           9 :     if ((daemon_local.flags.sendECUSoftwareVersion > 0) ||
    1572           9 :         (daemon_local.flags.sendTimezone > 0))
    1573           0 :         create_timer_fd(&daemon_local, 60, 60, DLT_TIMER_ECU);
    1574             : 
    1575             :     /* initiate gateway */
    1576           9 :     if (daemon_local.flags.gatewayMode == 1) {
    1577           1 :         if (dlt_gateway_init(&daemon_local, daemon_local.flags.vflag) == -1) {
    1578           0 :             dlt_log(LOG_CRIT, "Failed to create gateway\n");
    1579           0 :             return -1;
    1580             :         }
    1581             : 
    1582             :         /* create gateway timer */
    1583           1 :         create_timer_fd(&daemon_local,
    1584             :                         daemon_local.pGateway.interval,
    1585             :                         daemon_local.pGateway.interval,
    1586             :                         DLT_TIMER_GATEWAY);
    1587             :     }
    1588             : 
    1589             :     /* For offline tracing we still can use the same states */
    1590             :     /* as for socket sending. Using this trick we see the traces */
    1591             :     /* In the offline trace AND in the socket stream. */
    1592           9 :     if (daemon_local.flags.yvalue[0])
    1593           0 :         dlt_daemon_change_state(&daemon, DLT_DAEMON_STATE_SEND_DIRECT);
    1594             :     else
    1595           9 :         dlt_daemon_change_state(&daemon, DLT_DAEMON_STATE_BUFFER);
    1596             : 
    1597           9 :     dlt_daemon_init_user_information(&daemon,
    1598             :                                      &daemon_local.pGateway,
    1599             :                                      daemon_local.flags.gatewayMode,
    1600             :                                      daemon_local.flags.vflag);
    1601             : 
    1602             :     /*
    1603             :      * Check for app and ctx runtime cfg.
    1604             :      * These cfg must be loaded after ecuId and num_user_lists are available
    1605             :      */
    1606           9 :     if ((dlt_daemon_applications_load(&daemon, daemon.runtime_application_cfg,
    1607           0 :                                       daemon_local.flags.vflag) == 0) &&
    1608           0 :         (dlt_daemon_contexts_load(&daemon, daemon.runtime_context_cfg,
    1609             :                                   daemon_local.flags.vflag) == 0))
    1610           0 :         daemon.runtime_context_cfg_loaded = 1;
    1611             : 
    1612           9 :     dlt_daemon_log_internal(&daemon, &daemon_local,
    1613             :                             "Daemon launched. Starting to output traces...",
    1614             :                             DLT_LOG_INFO, DLT_DAEMON_APP_ID, DLT_DAEMON_CTX_ID,
    1615             :                             daemon_local.flags.vflag);
    1616             : 
    1617             :     /* Even handling loop. */
    1618        1550 :     while ((back >= 0) && (g_exit >= 0))
    1619        1541 :         back = dlt_daemon_handle_event(&daemon_local.pEvent,
    1620             :                                        &daemon,
    1621             :                                        &daemon_local);
    1622             : 
    1623           9 :     snprintf(local_str, DLT_DAEMON_TEXTBUFSIZE, "Exiting DLT daemon... [%d]",
    1624             :              g_signo);
    1625           9 :     dlt_daemon_log_internal(&daemon, &daemon_local, local_str, DLT_LOG_INFO,
    1626             :                             DLT_DAEMON_APP_ID, DLT_DAEMON_CTX_ID,
    1627             :                             daemon_local.flags.vflag);
    1628           9 :     dlt_vlog(LOG_NOTICE, "%s%s", local_str, "\n");
    1629             : 
    1630           9 :     dlt_daemon_local_cleanup(&daemon, &daemon_local, daemon_local.flags.vflag);
    1631             : 
    1632             : #ifdef UDP_CONNECTION_SUPPORT
    1633             :     dlt_daemon_udp_close_connection();
    1634             : #endif
    1635             : 
    1636           9 :     dlt_gateway_deinit(&daemon_local.pGateway, daemon_local.flags.vflag);
    1637             : 
    1638           9 :     dlt_daemon_free(&daemon, daemon_local.flags.vflag);
    1639             : 
    1640           9 :     dlt_log(LOG_NOTICE, "Leaving DLT daemon\n");
    1641             : 
    1642           9 :     dlt_log_free();
    1643             : 
    1644           9 :     return 0;
    1645             : 
    1646             : } /* main() */
    1647             : #endif
    1648             : 
    1649           9 : int dlt_daemon_local_init_p1(DltDaemon *daemon, DltDaemonLocal *daemon_local, int verbose)
    1650             : {
    1651           9 :     PRINT_FUNCTION_VERBOSE(verbose);
    1652             :     int ret = DLT_RETURN_OK;
    1653             : 
    1654           9 :     if ((daemon == 0) || (daemon_local == 0)) {
    1655           0 :         dlt_log(LOG_ERR, "Invalid function parameters used for function dlt_daemon_local_init_p1()\n");
    1656           0 :         return -1;
    1657             :     }
    1658             : 
    1659             : #if defined(DLT_SYSTEMD_WATCHDOG_ENABLE) || defined(DLT_SYSTEMD_ENABLE)
    1660             :     ret = sd_booted();
    1661             : 
    1662             :     if (ret == 0) {
    1663             :         dlt_log(LOG_CRIT, "System not booted with systemd!\n");
    1664             :     }
    1665             :     else if (ret < 0)
    1666             :     {
    1667             :         dlt_log(LOG_CRIT, "sd_booted failed!\n");
    1668             :         return -1;
    1669             :     }
    1670             :     else {
    1671             :         dlt_log(LOG_INFO, "System booted with systemd\n");
    1672             :     }
    1673             : 
    1674             : #endif
    1675             : 
    1676             : #ifdef DLT_DAEMON_USE_FIFO_IPC
    1677             : 
    1678           9 :     if (dlt_daemon_create_pipes_dir(daemon_local->flags.userPipesDir) == DLT_RETURN_ERROR)
    1679             :         return DLT_RETURN_ERROR;
    1680             : 
    1681             : #endif
    1682             : 
    1683             :     /* Check for daemon mode */
    1684           9 :     if (daemon_local->flags.dflag)
    1685           2 :         dlt_daemon_daemonize(daemon_local->flags.vflag);
    1686             : 
    1687             :     /* initialise structure to use DLT file */
    1688           9 :     ret = dlt_file_init(&(daemon_local->file), daemon_local->flags.vflag);
    1689             : 
    1690           9 :     if (ret == DLT_RETURN_ERROR) {
    1691           0 :         dlt_log(LOG_ERR, "Could not initialize file structure\n");
    1692             :         /* Return value ignored, dlt daemon will exit */
    1693           0 :         dlt_file_free(&(daemon_local->file), daemon_local->flags.vflag);
    1694           0 :         return ret;
    1695             :     }
    1696             : 
    1697           9 :     signal(SIGPIPE, SIG_IGN);
    1698             : 
    1699           9 :     signal(SIGTERM, dlt_daemon_signal_handler); /* software termination signal from kill */
    1700           9 :     signal(SIGHUP, dlt_daemon_signal_handler);  /* hangup signal */
    1701           9 :     signal(SIGQUIT, dlt_daemon_signal_handler);
    1702           9 :     signal(SIGINT, dlt_daemon_signal_handler);
    1703             : #ifdef __QNX__
    1704             :     signal(SIGUSR1, dlt_daemon_signal_handler); /* for timer threads */
    1705             : #endif
    1706             : 
    1707           9 :     return DLT_RETURN_OK;
    1708             : }
    1709             : 
    1710           9 : int dlt_daemon_local_init_p2(DltDaemon *daemon, DltDaemonLocal *daemon_local, int verbose)
    1711             : {
    1712           9 :     PRINT_FUNCTION_VERBOSE(verbose);
    1713             : 
    1714           9 :     if ((daemon == 0) || (daemon_local == 0)) {
    1715           0 :         dlt_log(LOG_ERR, "Invalid function parameters used for function dlt_daemon_local_init_p2()\n");
    1716           0 :         return -1;
    1717             :     }
    1718             : 
    1719             :     /* Daemon data */
    1720           9 :     if (dlt_daemon_init(daemon, daemon_local->RingbufferMinSize, daemon_local->RingbufferMaxSize,
    1721           9 :                         daemon_local->RingbufferStepSize, daemon_local->flags.ivalue,
    1722             :                         daemon_local->flags.contextLogLevel,
    1723             :                         daemon_local->flags.contextTraceStatus, daemon_local->flags.enforceContextLLAndTS,
    1724             :                         daemon_local->flags.vflag) == -1) {
    1725           0 :         dlt_log(LOG_ERR, "Could not initialize daemon data\n");
    1726           0 :         return -1;
    1727             :     }
    1728             : 
    1729             :     /* init offline trace */
    1730           9 :     if (((daemon->mode == DLT_USER_MODE_INTERNAL) || (daemon->mode == DLT_USER_MODE_BOTH)) &&
    1731           0 :         daemon_local->flags.offlineTraceDirectory[0]) {
    1732           0 :         if (multiple_files_buffer_init(&(daemon_local->offlineTrace),
    1733           0 :                                        daemon_local->flags.offlineTraceDirectory,
    1734             :                                        daemon_local->flags.offlineTraceFileSize,
    1735             :                                        daemon_local->flags.offlineTraceMaxSize,
    1736           0 :                                        daemon_local->flags.offlineTraceFilenameTimestampBased,
    1737             :                                        false,
    1738             :                                        DLT_OFFLINETRACE_FILENAME_BASE,
    1739             :                                        DLT_OFFLINETRACE_FILENAME_EXT) == -1) {
    1740           0 :             dlt_log(LOG_ERR, "Could not initialize offline trace\n");
    1741           0 :             return -1;
    1742             :         }
    1743             :     }
    1744             : 
    1745             :     /* Init offline logstorage for MAX devices */
    1746           9 :     if (daemon_local->flags.offlineLogstorageMaxDevices > 0) {
    1747           6 :         daemon->storage_handle = malloc(sizeof(DltLogStorage) * daemon_local->flags.offlineLogstorageMaxDevices);
    1748             : 
    1749           6 :         if (daemon->storage_handle == NULL) {
    1750           0 :             dlt_log(LOG_ERR, "Could not initialize offline logstorage\n");
    1751           0 :             return -1;
    1752             :         }
    1753             : 
    1754             :         memset(daemon->storage_handle, 0, (sizeof(DltLogStorage) * daemon_local->flags.offlineLogstorageMaxDevices));
    1755             :     }
    1756             : 
    1757             :     /* Set ECU id of daemon */
    1758           9 :     if (daemon_local->flags.evalue[0])
    1759           1 :         dlt_set_id(daemon->ecuid, daemon_local->flags.evalue);
    1760             :     else
    1761           8 :         dlt_set_id(daemon->ecuid, DLT_DAEMON_ECU_ID);
    1762             : 
    1763             :     /* Set flag for optional sending of serial header */
    1764           9 :     daemon->sendserialheader = daemon_local->flags.lflag;
    1765             : 
    1766             : #ifdef DLT_SHM_ENABLE
    1767             : 
    1768             :     /* init shared memory */
    1769             :     if (dlt_shm_init_server(&(daemon_local->dlt_shm), daemon_local->flags.dltShmName,
    1770             :                             daemon_local->flags.sharedMemorySize) == DLT_RETURN_ERROR) {
    1771             :         dlt_log(LOG_ERR, "Could not initialize shared memory\n");
    1772             :         return -1;
    1773             :     }
    1774             : 
    1775             :     daemon_local->recv_buf_shm = (unsigned char *)calloc(1, DLT_SHM_RCV_BUFFER_SIZE);
    1776             : 
    1777             :     if (NULL == daemon_local->recv_buf_shm) {
    1778             :         dlt_log(LOG_ERR, "failed to allocated the buffer to receive shm data\n");
    1779             :         return -1;
    1780             :     }
    1781             : 
    1782             : #endif
    1783             : 
    1784             :     /* prepare main loop */
    1785           9 :     if (dlt_message_init(&(daemon_local->msg), daemon_local->flags.vflag) == DLT_RETURN_ERROR) {
    1786           0 :         dlt_log(LOG_ERR, "Could not initialize message\n");
    1787           0 :         return -1;
    1788             :     }
    1789             : 
    1790             :     /* configure sending timing packets */
    1791           9 :     if (daemon_local->flags.sendMessageTime)
    1792           0 :         daemon->timingpackets = 1;
    1793             : 
    1794             :     /* Get ECU version info from a file. If it fails, use dlt_version as fallback. */
    1795           9 :     if (dlt_daemon_local_ecu_version_init(daemon, daemon_local, daemon_local->flags.vflag) < 0) {
    1796           9 :         daemon->ECUVersionString = malloc(DLT_DAEMON_TEXTBUFSIZE);
    1797             : 
    1798           9 :         if (daemon->ECUVersionString == 0) {
    1799           0 :             dlt_log(LOG_WARNING, "Could not allocate memory for version string\n");
    1800           0 :             return -1;
    1801             :         }
    1802             : 
    1803           9 :         dlt_get_version(daemon->ECUVersionString, DLT_DAEMON_TEXTBUFSIZE);
    1804             :     }
    1805             : 
    1806             :     /* Set to allows to maintain logstorage loglevel as default */
    1807           9 :     daemon->maintain_logstorage_loglevel = DLT_MAINTAIN_LOGSTORAGE_LOGLEVEL_ON;
    1808             : 
    1809           9 :     return 0;
    1810             : }
    1811             : 
    1812           9 : static int dlt_daemon_init_serial(DltDaemonLocal *daemon_local)
    1813             : {
    1814             :     /* create and open serial connection from/to client */
    1815             :     /* open serial connection */
    1816             :     int fd = -1;
    1817             : 
    1818           9 :     if (daemon_local->flags.yvalue[0] == '\0')
    1819             :         return 0;
    1820             : 
    1821           0 :     fd = open(daemon_local->flags.yvalue, O_RDWR);
    1822             : 
    1823           0 :     if (fd < 0) {
    1824           0 :         dlt_vlog(LOG_ERR, "Failed to open serial device %s\n",
    1825             :                  daemon_local->flags.yvalue);
    1826             : 
    1827           0 :         daemon_local->flags.yvalue[0] = 0;
    1828           0 :         return -1;
    1829             :     }
    1830             : 
    1831           0 :     if (isatty(fd)) {
    1832             :         int speed = DLT_DAEMON_SERIAL_DEFAULT_BAUDRATE;
    1833             : 
    1834           0 :         if (daemon_local->flags.bvalue[0])
    1835           0 :             speed = atoi(daemon_local->flags.bvalue);
    1836             : 
    1837           0 :         daemon_local->baudrate = dlt_convert_serial_speed(speed);
    1838             : 
    1839           0 :         if (dlt_setup_serial(fd, (speed_t) daemon_local->baudrate) < 0) {
    1840           0 :             close(fd);
    1841           0 :             daemon_local->flags.yvalue[0] = 0;
    1842             : 
    1843           0 :             dlt_vlog(LOG_ERR, "Failed to configure serial device %s (%s) \n",
    1844           0 :                      daemon_local->flags.yvalue, strerror(errno));
    1845             : 
    1846           0 :             return -1;
    1847             :         }
    1848             : 
    1849           0 :         if (daemon_local->flags.vflag)
    1850           0 :             dlt_log(LOG_DEBUG, "Serial init done\n");
    1851             :     }
    1852             :     else {
    1853           0 :         close(fd);
    1854           0 :         fprintf(stderr,
    1855             :                 "Device is not a serial device, device = %s (%s) \n",
    1856             :                 daemon_local->flags.yvalue,
    1857           0 :                 strerror(errno));
    1858           0 :         daemon_local->flags.yvalue[0] = 0;
    1859           0 :         return -1;
    1860             :     }
    1861             : 
    1862           0 :     return dlt_connection_create(daemon_local,
    1863             :                                  &daemon_local->pEvent,
    1864             :                                  fd,
    1865             :                                  POLLIN,
    1866             :                                  DLT_CONNECTION_CLIENT_MSG_SERIAL);
    1867             : }
    1868             : 
    1869             : #ifdef DLT_DAEMON_USE_FIFO_IPC
    1870           9 : static int dlt_daemon_init_fifo(DltDaemonLocal *daemon_local)
    1871             : {
    1872             :     int ret;
    1873             :     int fd = -1;
    1874             :     int fifo_size;
    1875             : 
    1876             :     /* open named pipe(FIFO) to receive DLT messages from users */
    1877           9 :     umask(0);
    1878             : 
    1879             :     /* Try to delete existing pipe, ignore result of unlink */
    1880           9 :     const char *tmpFifo = daemon_local->flags.daemonFifoName;
    1881           9 :     unlink(tmpFifo);
    1882             : 
    1883           9 :     ret = mkfifo(tmpFifo, S_IRUSR | S_IWUSR | S_IWGRP);
    1884             : 
    1885           9 :     if (ret == -1) {
    1886           0 :         dlt_vlog(LOG_WARNING, "FIFO user %s cannot be created (%s)!\n",
    1887           0 :                  tmpFifo, strerror(errno));
    1888           0 :         return -1;
    1889             :     } /* if */
    1890             : 
    1891             :     /* Set group of daemon FIFO */
    1892           9 :     if (daemon_local->flags.daemonFifoGroup[0] != 0) {
    1893           0 :         errno = 0;
    1894           0 :         struct group *group_dlt = getgrnam(daemon_local->flags.daemonFifoGroup);
    1895             : 
    1896           0 :         if (group_dlt) {
    1897           0 :             ret = chown(tmpFifo, -1, group_dlt->gr_gid);
    1898             : 
    1899           0 :             if (ret == -1)
    1900           0 :                 dlt_vlog(LOG_ERR, "FIFO user %s cannot be chowned to group %s (%s)\n",
    1901             :                          tmpFifo, daemon_local->flags.daemonFifoGroup,
    1902             :                          strerror(errno));
    1903             :         }
    1904           0 :         else if ((errno == 0) || (errno == ENOENT) || (errno == EBADF) || (errno == EPERM))
    1905             :         {
    1906           0 :             dlt_vlog(LOG_ERR, "Group name %s is not found (%s)\n",
    1907             :                      daemon_local->flags.daemonFifoGroup,
    1908             :                      strerror(errno));
    1909             :         }
    1910             :         else {
    1911           0 :             dlt_vlog(LOG_ERR, "Failed to get group id of %s (%s)\n",
    1912             :                      daemon_local->flags.daemonFifoGroup,
    1913             :                      strerror(errno));
    1914             :         }
    1915             :     }
    1916             : 
    1917             :     fd = open(tmpFifo, O_RDWR);
    1918             : 
    1919           9 :     if (fd == -1) {
    1920           0 :         dlt_vlog(LOG_WARNING, "FIFO user %s cannot be opened (%s)!\n",
    1921           0 :                  tmpFifo, strerror(errno));
    1922           0 :         return -1;
    1923             :     } /* if */
    1924             : 
    1925             : #ifdef __linux__
    1926             :     /* F_SETPIPE_SZ and F_GETPIPE_SZ are only supported for Linux.
    1927             :      * For other OSes it depends on its system e.g. pipe manager.
    1928             :      */
    1929           9 :     if (daemon_local->daemonFifoSize != 0) {
    1930             :         /* Set Daemon FIFO size */
    1931           0 :         if (fcntl(fd, F_SETPIPE_SZ, daemon_local->daemonFifoSize) == -1)
    1932           0 :             dlt_vlog(LOG_ERR, "set FIFO size error: %s\n", strerror(errno));
    1933             :     }
    1934             : 
    1935             :     /* Get Daemon FIFO size */
    1936           9 :     if ((fifo_size = fcntl(fd, F_GETPIPE_SZ, 0)) == -1)
    1937           0 :         dlt_vlog(LOG_ERR, "get FIFO size error: %s\n", strerror(errno));
    1938             :     else
    1939           9 :         dlt_vlog(LOG_INFO, "FIFO size: %d\n", fifo_size);
    1940             : #endif
    1941             : 
    1942             :     /* Early init, to be able to catch client (app) connections
    1943             :      * as soon as possible. This registration is automatically ignored
    1944             :      * during next execution.
    1945             :      */
    1946           9 :     return dlt_connection_create(daemon_local,
    1947             :                                  &daemon_local->pEvent,
    1948             :                                  fd,
    1949             :                                  POLLIN,
    1950             :                                  DLT_CONNECTION_APP_MSG);
    1951             : }
    1952             : #endif
    1953             : 
    1954             : #ifdef DLT_DAEMON_VSOCK_IPC_ENABLE
    1955             : static int dlt_daemon_init_vsock(DltDaemonLocal *daemon_local)
    1956             : {
    1957             :     int fd;
    1958             :     struct sockaddr_vm addr;
    1959             : 
    1960             :     fd = socket(AF_VSOCK, SOCK_STREAM, 0);
    1961             :     if (fd == -1) {
    1962             :         dlt_vlog(LOG_ERR, "Failed to create VSOCK socket: %s\n", strerror(errno));
    1963             :         return -1;
    1964             :     }
    1965             : 
    1966             :     memset(&addr, 0, sizeof(addr));
    1967             :     addr.svm_family = AF_VSOCK;
    1968             :     addr.svm_port = DLT_VSOCK_PORT;
    1969             :     addr.svm_cid = VMADDR_CID_ANY;
    1970             : 
    1971             :     if (bind(fd, (struct sockaddr *) &addr, sizeof(addr)) != 0) {
    1972             :         dlt_vlog(LOG_ERR, "Failed to bind VSOCK socket: %s\n", strerror(errno));
    1973             :         close(fd);
    1974             :         return -1;
    1975             :     }
    1976             : 
    1977             :     if (listen(fd, 1) != 0) {
    1978             :         dlt_vlog(LOG_ERR, "Failed to listen on VSOCK socket: %s\n", strerror(errno));
    1979             :         close(fd);
    1980             :         return -1;
    1981             :     }
    1982             : 
    1983             :     return dlt_connection_create(daemon_local,
    1984             :                                  &daemon_local->pEvent,
    1985             :                                  fd,
    1986             :                                  POLLIN,
    1987             :                                  DLT_CONNECTION_APP_CONNECT);
    1988             : }
    1989             : #endif
    1990             : 
    1991             : #ifdef DLT_DAEMON_USE_UNIX_SOCKET_IPC
    1992             : static DltReturnValue dlt_daemon_init_app_socket(DltDaemonLocal *daemon_local)
    1993             : {
    1994             :     /* socket access permission set to srw-rw-rw- (666) */
    1995             :     int mask = S_IXUSR | S_IXGRP | S_IXOTH;
    1996             :     DltReturnValue ret = DLT_RETURN_OK;
    1997             :     int fd = -1;
    1998             : 
    1999             :     if (daemon_local == NULL) {
    2000             :         dlt_vlog(LOG_ERR, "%s: Invalid function parameters\n", __func__);
    2001             :         return DLT_RETURN_ERROR;
    2002             :     }
    2003             : 
    2004             : #ifdef ANDROID
    2005             :     /* on android if we want to use security contexts on Unix sockets,
    2006             :      * they should be created by init (see dlt-daemon.rc in src/daemon)
    2007             :      * and recovered through the below API */
    2008             :     ret = dlt_daemon_unix_android_get_socket(&fd, daemon_local->flags.appSockPath);
    2009             :     if (ret < DLT_RETURN_OK) {
    2010             :         /* we failed to get app socket created by init.
    2011             :          * To avoid blocking users to launch dlt-daemon only through
    2012             :          * init on android (e.g: by hand for debugging purpose), try to
    2013             :          * create app socket by ourselves */
    2014             :         ret = dlt_daemon_unix_socket_open(&fd,
    2015             :                                           daemon_local->flags.appSockPath,
    2016             :                                           SOCK_STREAM,
    2017             :                                           mask);
    2018             :     }
    2019             : #else
    2020             :     ret = dlt_daemon_unix_socket_open(&fd,
    2021             :                                       daemon_local->flags.appSockPath,
    2022             :                                       SOCK_STREAM,
    2023             :                                       mask);
    2024             : #endif
    2025             :     if (ret == DLT_RETURN_OK) {
    2026             :         if (dlt_connection_create(daemon_local,
    2027             :                                   &daemon_local->pEvent,
    2028             :                                   fd,
    2029             :                                   POLLIN,
    2030             :                                   DLT_CONNECTION_APP_CONNECT)) {
    2031             :             dlt_log(LOG_CRIT, "Could not create connection for app socket.\n");
    2032             :             return DLT_RETURN_ERROR;
    2033             :         }
    2034             :     }
    2035             :     else {
    2036             :         dlt_log(LOG_CRIT, "Could not create and open app socket.\n");
    2037             :         return DLT_RETURN_ERROR;
    2038             :     }
    2039             : 
    2040             :     return ret;
    2041             : }
    2042             : #endif
    2043             : 
    2044           9 : static DltReturnValue dlt_daemon_initialize_control_socket(DltDaemonLocal *daemon_local)
    2045             : {
    2046             :     /* socket access permission set to srw-rw---- (660)  */
    2047             :     int mask = S_IXUSR | S_IXGRP | S_IROTH | S_IWOTH | S_IXOTH;
    2048             :     DltReturnValue ret = DLT_RETURN_OK;
    2049           9 :     int fd = -1;
    2050             : 
    2051           9 :     if (daemon_local == NULL) {
    2052           0 :         dlt_vlog(LOG_ERR, "%s: Invalid function parameters\n", __func__);
    2053           0 :         return -1;
    2054             :     }
    2055             : 
    2056             : #ifdef ANDROID
    2057             :     /* on android if we want to use security contexts on Unix sockets,
    2058             :      * they should be created by init (see dlt-daemon.rc in src/daemon)
    2059             :      * and recovered through the below API */
    2060             :     ret = dlt_daemon_unix_android_get_socket(&fd, daemon_local->flags.ctrlSockPath);
    2061             :     if (ret < DLT_RETURN_OK) {
    2062             :         /* we failed to get app socket created by init.
    2063             :          * To avoid blocking users to launch dlt-daemon only through
    2064             :          * init on android (e.g by hand for debugging purpose), try to
    2065             :          * create app socket by ourselves */
    2066             :         ret = dlt_daemon_unix_socket_open(&fd,
    2067             :                                           daemon_local->flags.ctrlSockPath,
    2068             :                                           SOCK_STREAM,
    2069             :                                           mask);
    2070             :     }
    2071             : #else
    2072           9 :     ret = dlt_daemon_unix_socket_open(&fd,
    2073           9 :                                       daemon_local->flags.ctrlSockPath,
    2074             :                                       SOCK_STREAM,
    2075             :                                       mask);
    2076             : #endif
    2077           9 :     if (ret == DLT_RETURN_OK) {
    2078           9 :         if (dlt_connection_create(daemon_local,
    2079             :                                   &daemon_local->pEvent,
    2080             :                                   fd,
    2081             :                                   POLLIN,
    2082             :                                   DLT_CONNECTION_CONTROL_CONNECT) < DLT_RETURN_OK) {
    2083           0 :             dlt_log(LOG_ERR, "Could not initialize control socket.\n");
    2084             :             ret = DLT_RETURN_ERROR;
    2085             :         }
    2086             :     }
    2087             : 
    2088             :     return ret;
    2089             : }
    2090             : 
    2091           9 : int dlt_daemon_local_connection_init(DltDaemon *daemon,
    2092             :                                      DltDaemonLocal *daemon_local,
    2093             :                                      int verbose)
    2094             : {
    2095           9 :     int fd = -1;
    2096             : 
    2097           9 :     PRINT_FUNCTION_VERBOSE(verbose);
    2098             : 
    2099           9 :     if ((daemon == NULL) || (daemon_local == NULL)) {
    2100           0 :         dlt_vlog(LOG_ERR, "%s: Invalid function parameters\n", __func__);
    2101           0 :         return -1;
    2102             :     }
    2103             : 
    2104           9 :     DltBindAddress_t *head = daemon_local->flags.ipNodes;
    2105             : 
    2106             : #ifdef DLT_DAEMON_USE_UNIX_SOCKET_IPC
    2107             :     /* create and open socket to receive incoming connections from user application */
    2108             :     if (dlt_daemon_init_app_socket(daemon_local) < DLT_RETURN_OK) {
    2109             :         dlt_log(LOG_ERR, "Unable to initialize app socket.\n");
    2110             :         return DLT_RETURN_ERROR;
    2111             :     }
    2112             : 
    2113             : #else /* DLT_DAEMON_USE_FIFO_IPC */
    2114             : 
    2115           9 :     if (dlt_daemon_init_fifo(daemon_local)) {
    2116           0 :         dlt_log(LOG_ERR, "Unable to initialize fifo.\n");
    2117           0 :         return DLT_RETURN_ERROR;
    2118             :     }
    2119             : 
    2120             : #endif
    2121             : 
    2122             : #ifdef DLT_DAEMON_VSOCK_IPC_ENABLE
    2123             :     if (dlt_daemon_init_vsock(daemon_local) != 0) {
    2124             :         dlt_log(LOG_ERR, "Unable to initialize app VSOCK socket.\n");
    2125             :         return DLT_RETURN_ERROR;
    2126             :     }
    2127             : #endif
    2128             : 
    2129             :     /* create and open socket to receive incoming connections from client */
    2130           9 :     daemon_local->client_connections = 0;
    2131             : 
    2132           9 :     if (head == NULL) { /* no IP set in BindAddress option, will use "0.0.0.0" as default */
    2133             : 
    2134           9 :         if (dlt_daemon_socket_open(&fd, daemon_local->flags.port, "0.0.0.0") == DLT_RETURN_OK) {
    2135           9 :             if (dlt_connection_create(daemon_local,
    2136             :                                       &daemon_local->pEvent,
    2137             :                                       fd,
    2138             :                                       POLLIN,
    2139             :                                       DLT_CONNECTION_CLIENT_CONNECT)) {
    2140           0 :                 dlt_log(LOG_ERR, "Could not initialize main socket.\n");
    2141           0 :                 return DLT_RETURN_ERROR;
    2142             :             }
    2143             :         }
    2144             :         else {
    2145           0 :             dlt_log(LOG_ERR, "Could not initialize main socket.\n");
    2146           0 :             return DLT_RETURN_ERROR;
    2147             :         }
    2148             :     }
    2149             :     else {
    2150             :         bool any_open = false;
    2151           0 :         while (head != NULL) { /* open socket for each IP in the bindAddress list */
    2152             : 
    2153           0 :             if (dlt_daemon_socket_open(&fd, daemon_local->flags.port, head->ip) == DLT_RETURN_OK) {
    2154           0 :                 if (dlt_connection_create(daemon_local,
    2155             :                                           &daemon_local->pEvent,
    2156             :                                           fd,
    2157             :                                           POLLIN,
    2158             :                                           DLT_CONNECTION_CLIENT_CONNECT)) {
    2159           0 :                     dlt_vlog(LOG_ERR, "Could not create connection, for binding %s\n", head->ip);
    2160             :                 } else {
    2161             :                     any_open = true;
    2162             :                 }
    2163             :             }
    2164             :             else {
    2165           0 :                 dlt_vlog(LOG_ERR, "Could not open main socket, for binding %s\n", head->ip);
    2166             :             }
    2167             : 
    2168           0 :             head = head->next;
    2169             :         }
    2170             : 
    2171           0 :         if (!any_open) {
    2172           0 :             dlt_vlog(LOG_ERR, "Failed create main socket for any configured binding\n");
    2173           0 :             return DLT_RETURN_ERROR;
    2174             :         }
    2175             :     }
    2176             : 
    2177             : #ifdef UDP_CONNECTION_SUPPORT
    2178             : 
    2179             :     if (daemon_local->UDPConnectionSetup == MULTICAST_CONNECTION_ENABLED) {
    2180             :         if (dlt_daemon_udp_connection_setup(daemon_local) < 0) {
    2181             :             dlt_log(LOG_ERR, "UDP fd creation failed\n");
    2182             :             return DLT_RETURN_ERROR;
    2183             :         }
    2184             :         else {
    2185             :             dlt_log(LOG_INFO, "UDP fd creation success\n");
    2186             :         }
    2187             :     }
    2188             : 
    2189             : #endif
    2190             : 
    2191             :     /* create and open unix socket to receive incoming connections from
    2192             :      * control application */
    2193           9 :     if (dlt_daemon_initialize_control_socket(daemon_local) < DLT_RETURN_OK) {
    2194           0 :         dlt_log(LOG_ERR, "Could not initialize control socket.\n");
    2195           0 :         return DLT_RETURN_ERROR;
    2196             :     }
    2197             : 
    2198             :     /* Init serial */
    2199           9 :     if (dlt_daemon_init_serial(daemon_local) < 0) {
    2200           0 :         dlt_log(LOG_ERR, "Could not initialize daemon data\n");
    2201           0 :         return DLT_RETURN_ERROR;
    2202             :     }
    2203             : 
    2204             :     return 0;
    2205             : }
    2206             : 
    2207           0 : static char* file_read_everything(FILE* const file, const size_t sizeLimit)
    2208             : {
    2209           0 :     if (!file) {
    2210             :         return NULL;
    2211             :     }
    2212             : 
    2213             :     /* Get the file size. Bail out if stat fails. */
    2214           0 :     const int fd = fileno(file);
    2215           0 :     struct stat s_buf = {0};
    2216           0 :     if (fstat(fd, &s_buf) < 0) {
    2217           0 :         dlt_log(LOG_WARNING, "failed to stat file size\n");
    2218           0 :         fclose(file);
    2219           0 :         return NULL;
    2220             :     }
    2221             : 
    2222             :     /* Size limit includes NULL terminator. */
    2223           0 :     const off_t size = s_buf.st_size;
    2224           0 :     if (size < 0 || (size_t)size >= sizeLimit) {
    2225           0 :         dlt_log(LOG_WARNING, "file size invalid\n");
    2226           0 :         fclose(file);
    2227           0 :         return NULL;
    2228             :     }
    2229             : 
    2230           0 :     char* const string = malloc((size_t)size + 1);
    2231           0 :     if (!string) {
    2232           0 :         dlt_log(LOG_WARNING, "failed to allocate string for file contents\n");
    2233           0 :         fclose(file);
    2234           0 :         return NULL;
    2235             :     }
    2236             : 
    2237             :     off_t offset = 0;
    2238           0 :     while (!feof(file)) {
    2239           0 :         offset += (off_t)fread(string + offset, 1, (size_t)size, file);
    2240             : 
    2241           0 :         if (ferror(file)) {
    2242           0 :             dlt_log(LOG_WARNING, "failed to read file\n");
    2243           0 :             free(string);
    2244           0 :             fclose(file);
    2245           0 :             return NULL;
    2246             :         }
    2247             : 
    2248           0 :         if (offset > size) {
    2249           0 :             dlt_log(LOG_WARNING, "file too long for buffer\n");
    2250           0 :             free(string);
    2251           0 :             fclose(file);
    2252           0 :             return NULL;
    2253             :         }
    2254             :     }
    2255             : 
    2256           0 :     string[offset] = '\0'; /* append null termination at end of string */
    2257             : 
    2258           0 :     return string;
    2259             : }
    2260             : 
    2261           0 : static char* file_read_field(FILE* const file, const char* const fieldName)
    2262             : {
    2263           0 :     if (!file) {
    2264             :         return NULL;
    2265             :     }
    2266             : 
    2267             :     const char* const kDelimiters = "\r\n\"\'=";
    2268           0 :     const size_t fieldNameLen = strlen(fieldName);
    2269             : 
    2270             :     char* result = NULL;
    2271             : 
    2272           0 :     char* buffer = NULL;
    2273           0 :     size_t bufferSize = 0;
    2274             : 
    2275             :     while (true) {
    2276             :         ssize_t lineSize = getline(&buffer, &bufferSize, file);
    2277           0 :         if (lineSize < 0 || !buffer) {
    2278             :             /* end of file */
    2279             :             break;
    2280             :         }
    2281             : 
    2282             :         char* line = buffer;
    2283             : 
    2284             :         /* trim trailing delimiters */
    2285           0 :         while (lineSize >= 1 && strchr(kDelimiters, line[lineSize - 1]) != NULL) {
    2286           0 :             line[lineSize - 1] = '\0';
    2287           0 :             --lineSize;
    2288             :         }
    2289             : 
    2290             :         /* check fieldName */
    2291           0 :         if (strncmp(line, fieldName, fieldNameLen) == 0 &&
    2292           0 :             (size_t)lineSize >= (fieldNameLen + 1) &&
    2293           0 :             strchr(kDelimiters, line[fieldNameLen]) != NULL) {
    2294             :             /* trim fieldName */
    2295             :             line += fieldNameLen;
    2296             : 
    2297             :             /* trim delimiter */
    2298           0 :             ++line;
    2299             : 
    2300             :             /* trim leading delimiters */
    2301           0 :             while (*line != '\0' && strchr(kDelimiters, *line) != NULL) {
    2302           0 :                 ++line;
    2303             :                 --lineSize;
    2304             :             }
    2305             : 
    2306           0 :             result = strdup(line);
    2307           0 :             break;
    2308             :         }
    2309             :     }
    2310             : 
    2311           0 :     free(buffer);
    2312             : 
    2313           0 :     return result;
    2314             : }
    2315             : 
    2316           9 : int dlt_daemon_local_ecu_version_init(DltDaemon *daemon, DltDaemonLocal *daemon_local, int verbose)
    2317             : {
    2318             :     FILE *f = NULL;
    2319             : 
    2320           9 :     PRINT_FUNCTION_VERBOSE(verbose);
    2321             : 
    2322             :     /* By default, version string is null. */
    2323           9 :     daemon->ECUVersionString = NULL;
    2324             : 
    2325             :     /* Open the file. Bail out if error occurs */
    2326           9 :     f = fopen(daemon_local->flags.pathToECUSoftwareVersion, "r");
    2327             : 
    2328           9 :     if (f == NULL) {
    2329             :         /* Error level notice, because this might be deliberate choice */
    2330           9 :         dlt_log(LOG_NOTICE, "Failed to open ECU Software version file.\n");
    2331           9 :         return -1;
    2332             :     }
    2333             : 
    2334           0 :     if (daemon_local->flags.ecuSoftwareVersionFileField[0] != '\0') {
    2335           0 :         daemon->ECUVersionString = file_read_field(f, daemon_local->flags.ecuSoftwareVersionFileField);
    2336             :     } else {
    2337           0 :         daemon->ECUVersionString = file_read_everything(f, DLT_DAEMON_TEXTBUFSIZE);
    2338             :     }
    2339             : 
    2340           0 :     fclose(f);
    2341             : 
    2342           0 :     return (daemon->ECUVersionString != NULL) ? 0 : -1;
    2343             : }
    2344             : 
    2345           9 : void dlt_daemon_local_cleanup(DltDaemon *daemon, DltDaemonLocal *daemon_local, int verbose)
    2346             : {
    2347           9 :     PRINT_FUNCTION_VERBOSE(verbose);
    2348             : 
    2349           9 :     if ((daemon == 0) || (daemon_local == 0)) {
    2350           0 :         dlt_log(LOG_ERR, "Invalid function parameters used for function dlt_daemon_local_cleanup()\n");
    2351           0 :         return;
    2352             :     }
    2353             : 
    2354             :     /* Don't receive event anymore */
    2355           9 :     dlt_event_handler_cleanup_connections(&daemon_local->pEvent);
    2356             : 
    2357           9 :     dlt_message_free(&(daemon_local->msg), daemon_local->flags.vflag);
    2358             : 
    2359             :     /* free shared memory */
    2360           9 :     if (daemon_local->flags.offlineTraceDirectory[0])
    2361           0 :         multiple_files_buffer_free(&(daemon_local->offlineTrace));
    2362             : 
    2363             :     /* Ignore result */
    2364           9 :     dlt_file_free(&(daemon_local->file), daemon_local->flags.vflag);
    2365             : 
    2366             : #ifdef DLT_DAEMON_USE_FIFO_IPC
    2367             :     /* Try to delete existing pipe, ignore result of unlink() */
    2368           9 :     unlink(daemon_local->flags.daemonFifoName);
    2369             : #else /* DLT_DAEMON_USE_UNIX_SOCKET_IPC */
    2370             :     /* Try to delete existing pipe, ignore result of unlink() */
    2371             :     if (unlink(daemon_local->flags.appSockPath) != 0) {
    2372             :         dlt_vlog(LOG_WARNING, "%s: unlink() failed: %s\n",
    2373             :                 __func__, strerror(errno));
    2374             :     }
    2375             : #endif
    2376             : 
    2377             : #ifdef DLT_SHM_ENABLE
    2378             :     /* free shared memory */
    2379             :     dlt_shm_free_server(&(daemon_local->dlt_shm), daemon_local->flags.dltShmName);
    2380             :     free(daemon_local->recv_buf_shm);
    2381             :     daemon_local->recv_buf_shm = NULL;
    2382             : #endif
    2383             : 
    2384           9 :     if (daemon_local->flags.offlineLogstorageMaxDevices > 0) {
    2385             :         /* disconnect all logstorage devices */
    2386           6 :         dlt_daemon_logstorage_cleanup(daemon,
    2387             :                                       daemon_local,
    2388             :                                       daemon_local->flags.vflag);
    2389             : 
    2390           6 :         free(daemon->storage_handle);
    2391             :     }
    2392             : 
    2393           9 :     if (daemon->ECUVersionString != NULL)
    2394           9 :         free(daemon->ECUVersionString);
    2395             : 
    2396           9 :     if (unlink(daemon_local->flags.ctrlSockPath) != 0) {
    2397           1 :         dlt_vlog(LOG_WARNING, "%s: unlink() failed: %s\n",
    2398           1 :                 __func__, strerror(errno));
    2399             :     }
    2400             : 
    2401             :     /* free IP list */
    2402           9 :     free(daemon_local->flags.ipNodes);
    2403             : }
    2404             : 
    2405           9 : void dlt_daemon_exit_trigger()
    2406             : {
    2407             :     /* stop event loop */
    2408           9 :     g_exit = -1;
    2409             : 
    2410             : #ifdef DLT_DAEMON_USE_FIFO_IPC
    2411           9 :     char tmp[DLT_PATH_MAX] = { 0 };
    2412             : 
    2413             :     ssize_t n;
    2414           9 :     n = snprintf(tmp, DLT_PATH_MAX, "%s/dlt", dltFifoBaseDir);
    2415           9 :     if (n < 0 || (size_t)n > DLT_PATH_MAX) {
    2416           0 :         dlt_vlog(LOG_WARNING, "%s: snprintf truncation/error(%ld) %s\n",
    2417             :                 __func__, n, tmp);
    2418             :     }
    2419             : 
    2420           9 :     (void)unlink(tmp);
    2421             : #endif
    2422             : 
    2423             : #ifdef __QNX__
    2424             :     dlt_daemon_cleanup_timers();
    2425             : #endif
    2426             : 
    2427           9 : }
    2428             : 
    2429           9 : void dlt_daemon_signal_handler(int sig)
    2430             : {
    2431           9 :     g_signo = sig;
    2432             : 
    2433           9 :     switch (sig) {
    2434           9 :         case SIGHUP:
    2435             :         case SIGTERM:
    2436             :         case SIGINT:
    2437             :         case SIGQUIT:
    2438             :         {
    2439             :             /* finalize the server */
    2440           9 :             dlt_vlog(LOG_NOTICE, "Exiting DLT daemon due to signal: %s\n",
    2441             :                      strsignal(sig));
    2442           9 :             dlt_daemon_exit_trigger();
    2443           9 :             break;
    2444             :         }
    2445             :         default:
    2446             :         {
    2447             :             /* This case should never happen! */
    2448             :             break;
    2449             :         }
    2450             :     } /* switch */
    2451             : 
    2452           9 : } /* dlt_daemon_signal_handler() */
    2453             : 
    2454             : #ifdef __QNX__
    2455             : void dlt_daemon_cleanup_timers()
    2456             : {
    2457             :     int i = 0;
    2458             :     while (i < DLT_TIMER_UNKNOWN) {
    2459             :         /* Remove FIFO of every timer and kill timer thread */
    2460             :         if (0 != timer_threads[i]) {
    2461             :             pthread_kill(timer_threads[i], SIGUSR1);
    2462             :             pthread_join(timer_threads[i], NULL);
    2463             :             timer_threads[i] = 0;
    2464             : 
    2465             :             close_pipes(dlt_timer_pipes[i]);
    2466             : 
    2467             :             /* Free data of every timer */
    2468             :             if (NULL != timer_data[i]) {
    2469             :                 free(timer_data[i]);
    2470             :                 timer_data[i] = NULL;
    2471             :             }
    2472             :         }
    2473             :         i++;
    2474             :     }
    2475             : }
    2476             : #endif
    2477             : 
    2478           2 : void dlt_daemon_daemonize(int verbose)
    2479             : {
    2480             :     int i;
    2481             :     int fd;
    2482             : 
    2483           2 :     PRINT_FUNCTION_VERBOSE(verbose);
    2484             : 
    2485           2 :     dlt_log(LOG_NOTICE, "Daemon mode\n");
    2486             : 
    2487             :     /* Daemonize */
    2488           2 :     i = fork();
    2489             : 
    2490           4 :     if (i < 0) {
    2491           0 :         dlt_log(LOG_CRIT, "Unable to fork(), exiting DLT daemon\n");
    2492           0 :         exit(-1); /* fork error */
    2493             :     }
    2494             : 
    2495           4 :     if (i > 0)
    2496           2 :         exit(0); /* parent exits */
    2497             : 
    2498             :     /* child (daemon) continues */
    2499             : 
    2500             :     /* Process independency */
    2501             : 
    2502             :     /* obtain a new process group */
    2503           2 :     if (setsid() == -1) {
    2504           0 :         dlt_log(LOG_CRIT, "setsid() failed, exiting DLT daemon\n");
    2505           0 :         exit(-1); /* fork error */
    2506             :     }
    2507             : 
    2508             :     /* Open standard descriptors stdin, stdout, stderr */
    2509             :     fd = open("/dev/null", O_RDWR);
    2510             : 
    2511           2 :     if (fd != -1) {
    2512             :         /* Redirect STDOUT to /dev/null */
    2513           2 :         if (dup2(fd, STDOUT_FILENO) < 0)
    2514           0 :             dlt_vlog(LOG_WARNING, "Failed to direct stdout to /dev/null. Error: %s\n", strerror(errno));
    2515             : 
    2516             :         /* Redirect STDERR to /dev/null */
    2517           2 :         if (dup2(fd, STDERR_FILENO) < 0)
    2518           0 :             dlt_vlog(LOG_WARNING, "Failed to direct stderr to /dev/null. Error: %s\n", strerror(errno));
    2519             : 
    2520           2 :         close(fd);
    2521             :     }
    2522             :     else {
    2523           0 :         dlt_log(LOG_CRIT, "Error opening /dev/null, exiting DLT daemon\n");
    2524           0 :         exit(-1); /* fork error */
    2525             :     }
    2526             : 
    2527             :     /* Set umask */
    2528           2 :     umask(DLT_DAEMON_UMASK);
    2529             : 
    2530             :     /* Change to root directory */
    2531           2 :     if (chdir("/") < 0)
    2532           0 :         dlt_log(LOG_WARNING, "Failed to chdir to root\n");
    2533             : 
    2534             :     /* Catch signals */
    2535           2 :     signal(SIGCHLD, SIG_IGN); /* ignore child */
    2536           2 :     signal(SIGTSTP, SIG_IGN); /* ignore tty signals */
    2537           2 :     signal(SIGTTOU, SIG_IGN);
    2538           2 :     signal(SIGTTIN, SIG_IGN);
    2539             : 
    2540           2 : } /* dlt_daemon_daemonize() */
    2541             : 
    2542             : /* This function logs str to the configured output sink (socket, serial, offline trace).
    2543             :  * To avoid recursion this function must be called only from DLT highlevel functions.
    2544             :  * E. g. calling it to output a failure when the open of the offline trace file fails
    2545             :  * would cause an endless loop because dlt_daemon_log_internal() would itself again try
    2546             :  * to open the offline trace file.
    2547             :  * This is a dlt-daemon only function. The libdlt has no equivalent function available. */
    2548          40 : int dlt_daemon_log_internal(DltDaemon *daemon, DltDaemonLocal *daemon_local,
    2549             :                             char *str, DltLogLevelType level,
    2550             :                             const char *app_id, const char *ctx_id, int verbose)
    2551             : {
    2552          40 :     DltMessage msg = { 0 };
    2553             :     static uint8_t uiMsgCount = 0;
    2554             :     DltStandardHeaderExtra *pStandardExtra = NULL;
    2555             :     uint32_t uiType;
    2556             :     uint16_t uiSize;
    2557             :     uint32_t uiExtraSize;
    2558             : 
    2559          40 :     PRINT_FUNCTION_VERBOSE(verbose);
    2560             : 
    2561             :     /* Set storageheader */
    2562          40 :     msg.storageheader = (DltStorageHeader *)(msg.headerbuffer);
    2563          40 :     dlt_set_storageheader(msg.storageheader, daemon->ecuid);
    2564             : 
    2565             :     /* Set standardheader */
    2566          40 :     msg.standardheader = (DltStandardHeader *)(msg.headerbuffer + sizeof(DltStorageHeader));
    2567          40 :     msg.standardheader->htyp = DLT_HTYP_UEH | DLT_HTYP_WEID | DLT_HTYP_WSID | DLT_HTYP_WTMS |
    2568             :         DLT_HTYP_PROTOCOL_VERSION1;
    2569          40 :     msg.standardheader->mcnt = uiMsgCount++;
    2570             : 
    2571             :     uiExtraSize = (uint32_t) (DLT_STANDARD_HEADER_EXTRA_SIZE(msg.standardheader->htyp) +
    2572             :         (DLT_IS_HTYP_UEH(msg.standardheader->htyp) ? sizeof(DltExtendedHeader) : 0));
    2573          40 :     msg.headersize = (uint32_t) sizeof(DltStorageHeader) + (uint32_t) sizeof(DltStandardHeader) + uiExtraSize;
    2574             : 
    2575             :     /* Set extraheader */
    2576             :     pStandardExtra =
    2577             :         (DltStandardHeaderExtra *)(msg.headerbuffer + sizeof(DltStorageHeader) + sizeof(DltStandardHeader));
    2578          40 :     dlt_set_id(pStandardExtra->ecu, daemon->ecuid);
    2579          40 :     pStandardExtra->tmsp = DLT_HTOBE_32(dlt_uptime());
    2580          40 :     pStandardExtra->seid = (unsigned int) DLT_HTOBE_32(getpid());
    2581             : 
    2582             :     /* Set extendedheader */
    2583          40 :     msg.extendedheader =
    2584          40 :         (DltExtendedHeader *)(msg.headerbuffer + sizeof(DltStorageHeader) + sizeof(DltStandardHeader) +
    2585          40 :                               DLT_STANDARD_HEADER_EXTRA_SIZE(msg.standardheader->htyp));
    2586          40 :     msg.extendedheader->msin = DLT_MSIN_VERB | (DLT_TYPE_LOG << DLT_MSIN_MSTP_SHIFT) |
    2587          40 :         ((level << DLT_MSIN_MTIN_SHIFT) & DLT_MSIN_MTIN);
    2588          40 :     msg.extendedheader->noar = 1;
    2589          40 :     dlt_set_id(msg.extendedheader->apid, app_id);
    2590          40 :     dlt_set_id(msg.extendedheader->ctid, ctx_id);
    2591             : 
    2592             :     /* Set payload data... */
    2593          40 :     uiType = DLT_TYPE_INFO_STRG;
    2594          40 :     uiSize = (uint16_t) (strlen(str) + 1);
    2595          40 :     msg.datasize = (uint32_t) (sizeof(uint32_t) + sizeof(uint16_t) + uiSize);
    2596             : 
    2597          40 :     msg.databuffer = (uint8_t *)malloc((size_t) msg.datasize);
    2598          40 :     msg.databuffersize = msg.datasize;
    2599             : 
    2600          40 :     if (msg.databuffer == 0) {
    2601           0 :         dlt_log(LOG_WARNING, "Can't allocate buffer for get log info message\n");
    2602           0 :         return -1;
    2603             :     }
    2604             : 
    2605             :     msg.datasize = 0;
    2606             :     memcpy((uint8_t *)(msg.databuffer + msg.datasize), (uint8_t *)(&uiType), sizeof(uint32_t));
    2607          40 :     msg.datasize += (uint32_t) sizeof(uint32_t);
    2608          40 :     memcpy((uint8_t *)(msg.databuffer + msg.datasize), (uint8_t *)(&uiSize), sizeof(uint16_t));
    2609          40 :     msg.datasize += (uint32_t) sizeof(uint16_t);
    2610          40 :     memcpy((uint8_t *)(msg.databuffer + msg.datasize), str, uiSize);
    2611          40 :     msg.datasize += uiSize;
    2612             : 
    2613             :     /* Calc length */
    2614          40 :     msg.standardheader->len = DLT_HTOBE_16(msg.headersize - sizeof(DltStorageHeader) + msg.datasize);
    2615             : 
    2616          40 :     dlt_daemon_client_send(DLT_DAEMON_SEND_TO_ALL, daemon,daemon_local,
    2617             :                            msg.headerbuffer, sizeof(DltStorageHeader),
    2618             :                            msg.headerbuffer + sizeof(DltStorageHeader),
    2619          40 :                            (int) (msg.headersize - sizeof(DltStorageHeader)),
    2620          40 :                            msg.databuffer, (int) msg.datasize, verbose);
    2621             : 
    2622          40 :     free(msg.databuffer);
    2623             : 
    2624          40 :     return 0;
    2625             : }
    2626             : 
    2627           3 : int dlt_daemon_check_numeric_setting(char *token,
    2628             :                                     char *value,
    2629             :                                     unsigned long *data)
    2630           3 : {
    2631           3 :     char value_check[value_length];
    2632           3 :     value_check[0] = 0;
    2633           3 :     sscanf(value, "%lu%s", data, value_check);
    2634           3 :     if (value_check[0] || !isdigit(value[0])) {
    2635           0 :         fprintf(stderr, "Invalid input [%s] detected in option %s\n",
    2636             :                 value,
    2637             :                 token);
    2638           0 :         return -1;
    2639             :     }
    2640             :     return 0;
    2641             : }
    2642             : 
    2643           4 : int dlt_daemon_process_client_connect(DltDaemon *daemon,
    2644             :                                       DltDaemonLocal *daemon_local,
    2645             :                                       DltReceiver *receiver,
    2646             :                                       int verbose)
    2647             : {
    2648             :     socklen_t cli_size;
    2649             :     struct sockaddr_un cli;
    2650             : 
    2651             :     int in_sock = -1;
    2652           4 :     char local_str[DLT_DAEMON_TEXTBUFSIZE] = { '\0' };
    2653             : 
    2654           4 :     PRINT_FUNCTION_VERBOSE(verbose);
    2655             : 
    2656           4 :     if ((daemon == NULL) || (daemon_local == NULL) || (receiver == NULL)) {
    2657           0 :         dlt_log(LOG_ERR,
    2658             :                 "Invalid function parameters used for function "
    2659             :                 "dlt_daemon_process_client_connect()\n");
    2660           0 :         return -1;
    2661             :     }
    2662             : 
    2663             :     /* event from TCP server socket, new connection */
    2664           4 :     cli_size = sizeof(cli);
    2665             : 
    2666           4 :     if ((in_sock = accept(receiver->fd, (struct sockaddr *)&cli, &cli_size)) < 0) {
    2667           0 :         if (errno == ECONNABORTED) // Caused by nmap -v -p 3490 -Pn <IP of dlt-daemon>
    2668             :             return 0;
    2669           0 :         dlt_vlog(LOG_ERR, "accept() for socket %d failed: %s\n", receiver->fd, strerror(errno));
    2670           0 :         return -1;
    2671             :     }
    2672             : 
    2673             :     /* check if file file descriptor was already used, and make it invalid if it
    2674             :      * is reused. */
    2675             :     /* This prevents sending messages to wrong file descriptor */
    2676           4 :     dlt_daemon_applications_invalidate_fd(daemon, daemon->ecuid, in_sock, verbose);
    2677           4 :     dlt_daemon_contexts_invalidate_fd(daemon, daemon->ecuid, in_sock, verbose);
    2678             : 
    2679             :     /* Set socket timeout in reception */
    2680             :     struct timeval timeout_send;
    2681           4 :     timeout_send.tv_sec = daemon_local->timeoutOnSend;
    2682           4 :     timeout_send.tv_usec = 0;
    2683             : 
    2684           4 :     if (setsockopt (in_sock,
    2685             :                     SOL_SOCKET,
    2686             :                     SO_SNDTIMEO,
    2687             :                     (char *)&timeout_send,
    2688             :                     sizeof(timeout_send)) < 0)
    2689           0 :         dlt_log(LOG_WARNING, "setsockopt failed\n");
    2690             : 
    2691           4 :     if (dlt_connection_create(daemon_local,
    2692             :                               &daemon_local->pEvent,
    2693             :                               in_sock,
    2694             :                               POLLIN,
    2695             :                               DLT_CONNECTION_CLIENT_MSG_TCP)) {
    2696           0 :         dlt_log(LOG_ERR, "Failed to register new client. \n");
    2697           0 :         close(in_sock);
    2698           0 :         return -1;
    2699             :     }
    2700             : 
    2701             :     /* send connection info about connected */
    2702           4 :     dlt_daemon_control_message_connection_info(in_sock,
    2703             :                                                daemon,
    2704             :                                                daemon_local,
    2705             :                                                DLT_CONNECTION_STATUS_CONNECTED,
    2706             :                                                "",
    2707             :                                                verbose);
    2708             : 
    2709             :     /* send ecu version string */
    2710           4 :     if (daemon_local->flags.sendECUSoftwareVersion > 0) {
    2711             :         if (daemon_local->flags.sendECUSoftwareVersion > 0)
    2712           0 :             dlt_daemon_control_get_software_version(DLT_DAEMON_SEND_TO_ALL,
    2713             :                                                     daemon,
    2714             :                                                     daemon_local,
    2715             :                                                     daemon_local->flags.vflag);
    2716             : 
    2717           0 :         if (daemon_local->flags.sendTimezone > 0)
    2718           0 :             dlt_daemon_control_message_timezone(DLT_DAEMON_SEND_TO_ALL,
    2719             :                                                 daemon,
    2720             :                                                 daemon_local,
    2721             :                                                 daemon_local->flags.vflag);
    2722             :     }
    2723             : 
    2724           4 :     snprintf(local_str, DLT_DAEMON_TEXTBUFSIZE,
    2725             :              "New client connection #%d established, Total Clients : %d",
    2726             :              in_sock, daemon_local->client_connections);
    2727             : 
    2728           4 :     dlt_daemon_log_internal(daemon, daemon_local, local_str, DLT_LOG_INFO,
    2729             :                             DLT_DAEMON_APP_ID, DLT_DAEMON_CTX_ID,
    2730             :                             daemon_local->flags.vflag);
    2731           4 :     dlt_vlog(LOG_DEBUG, "%s%s", local_str, "\n");
    2732             : 
    2733           4 :     if (daemon_local->client_connections == 1) {
    2734           2 :         if (daemon_local->flags.vflag)
    2735           0 :             dlt_log(LOG_DEBUG, "Send ring-buffer to client\n");
    2736             : 
    2737           2 :         dlt_daemon_change_state(daemon, DLT_DAEMON_STATE_SEND_BUFFER);
    2738             : 
    2739           2 :         if (dlt_daemon_send_ringbuffer_to_client(daemon, daemon_local, verbose) == -1) {
    2740           0 :             dlt_log(LOG_WARNING, "Can't send contents of ringbuffer to clients\n");
    2741           0 :             close(in_sock);
    2742             :             in_sock = -1;
    2743           0 :             return -1;
    2744             :         }
    2745             : 
    2746             :         /* send new log state to all applications */
    2747           2 :         daemon->connectionState = 1;
    2748           2 :         dlt_daemon_user_send_all_log_state(daemon, verbose);
    2749             : 
    2750             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
    2751             :         /* Reset number of received bytes from FIFO */
    2752             :         daemon->bytes_recv = 0;
    2753             : #endif
    2754             :     }
    2755             : 
    2756             :     return 0;
    2757             : }
    2758             : 
    2759           7 : int dlt_daemon_process_client_messages(DltDaemon *daemon,
    2760             :                                        DltDaemonLocal *daemon_local,
    2761             :                                        DltReceiver *receiver,
    2762             :                                        int verbose)
    2763             : {
    2764             :     int bytes_to_be_removed = 0;
    2765             :     int must_close_socket = -1;
    2766             : 
    2767           7 :     PRINT_FUNCTION_VERBOSE(verbose);
    2768             : 
    2769           7 :     if ((daemon == NULL) || (daemon_local == NULL) || (receiver == NULL)) {
    2770           0 :         dlt_log(LOG_ERR,
    2771             :                 "Invalid function parameters used for function "
    2772             :                 "dlt_daemon_process_client_messages()\n");
    2773           0 :         return -1;
    2774             :     }
    2775             : 
    2776           7 :     must_close_socket = dlt_receiver_receive(receiver);
    2777             : 
    2778           7 :     if (must_close_socket < 0) {
    2779           0 :         dlt_daemon_close_socket(receiver->fd,
    2780             :                                 daemon,
    2781             :                                 daemon_local,
    2782             :                                 verbose);
    2783           0 :         return -1;
    2784             :     }
    2785             : 
    2786             :     /* Process all received messages */
    2787          17 :     while (dlt_message_read(&(daemon_local->msg),
    2788          17 :                             (uint8_t *)receiver->buf,
    2789          17 :                             (unsigned int) receiver->bytesRcvd,
    2790             :                             daemon_local->flags.nflag,
    2791          17 :                             daemon_local->flags.vflag) == DLT_MESSAGE_ERROR_OK) {
    2792             :         /* Check for control message */
    2793          10 :         if ((0 < receiver->fd) &&
    2794          10 :             DLT_MSG_IS_CONTROL_REQUEST(&(daemon_local->msg)))
    2795          10 :             dlt_daemon_client_process_control(receiver->fd,
    2796             :                                               daemon,
    2797             :                                               daemon_local,
    2798             :                                               &(daemon_local->msg),
    2799             :                                               daemon_local->flags.vflag);
    2800             : 
    2801          10 :         bytes_to_be_removed = (int) (daemon_local->msg.headersize +
    2802          10 :             daemon_local->msg.datasize -
    2803             :             sizeof(DltStorageHeader));
    2804             : 
    2805          10 :         if (daemon_local->msg.found_serialheader)
    2806             :             bytes_to_be_removed += (int) sizeof(dltSerialHeader);
    2807             : 
    2808          10 :         if (daemon_local->msg.resync_offset)
    2809           0 :             bytes_to_be_removed += daemon_local->msg.resync_offset;
    2810             : 
    2811          10 :         if (dlt_receiver_remove(receiver, bytes_to_be_removed) == -1) {
    2812           0 :             dlt_log(LOG_WARNING,
    2813             :                     "Can't remove bytes from receiver for sockets\n");
    2814           0 :             return -1;
    2815             :         }
    2816             :     } /* while */
    2817             : 
    2818           7 :     if (dlt_receiver_move_to_begin(receiver) == -1) {
    2819           0 :         dlt_log(LOG_WARNING,
    2820             :                 "Can't move bytes to beginning of receiver buffer for sockets\n");
    2821           0 :         return -1;
    2822             :     }
    2823             : 
    2824           7 :     if (must_close_socket == 0)
    2825             :         /* FIXME: Why the hell do we need to close the socket
    2826             :          * on control message reception ??
    2827             :          */
    2828           2 :         dlt_daemon_close_socket(receiver->fd,
    2829             :                                 daemon,
    2830             :                                 daemon_local,
    2831             :                                 verbose);
    2832             : 
    2833             :     return 0;
    2834             : }
    2835             : 
    2836           0 : int dlt_daemon_process_client_messages_serial(DltDaemon *daemon,
    2837             :                                               DltDaemonLocal *daemon_local,
    2838             :                                               DltReceiver *receiver,
    2839             :                                               int verbose)
    2840             : {
    2841             :     int bytes_to_be_removed = 0;
    2842             : 
    2843           0 :     PRINT_FUNCTION_VERBOSE(verbose);
    2844             : 
    2845           0 :     if ((daemon == NULL) || (daemon_local == NULL) || (receiver == NULL)) {
    2846           0 :         dlt_log(LOG_ERR,
    2847             :                 "Invalid function parameters used for function "
    2848             :                 "dlt_daemon_process_client_messages_serial()\n");
    2849           0 :         return -1;
    2850             :     }
    2851             : 
    2852           0 :     if (dlt_receiver_receive(receiver) <= 0) {
    2853           0 :         dlt_log(LOG_WARNING,
    2854             :                 "dlt_receiver_receive_fd() for messages from serial interface "
    2855             :                 "failed!\n");
    2856           0 :         return -1;
    2857             :     }
    2858             : 
    2859             :     /* Process all received messages */
    2860           0 :     while (dlt_message_read(&(daemon_local->msg),
    2861           0 :                             (uint8_t *)receiver->buf,
    2862           0 :                             (unsigned int) receiver->bytesRcvd,
    2863             :                             daemon_local->flags.mflag,
    2864           0 :                             daemon_local->flags.vflag) == DLT_MESSAGE_ERROR_OK) {
    2865             :         /* Check for control message */
    2866           0 :         if (DLT_MSG_IS_CONTROL_REQUEST(&(daemon_local->msg))) {
    2867           0 :             if (dlt_daemon_client_process_control(receiver->fd,
    2868             :                                                   daemon,
    2869             :                                                   daemon_local,
    2870             :                                                   &(daemon_local->msg),
    2871             :                                                   daemon_local->flags.vflag)
    2872             :                 == -1) {
    2873           0 :                 dlt_log(LOG_WARNING, "Can't process control messages\n");
    2874           0 :                 return -1;
    2875             :             }
    2876             :         }
    2877             : 
    2878           0 :         bytes_to_be_removed = (int) (daemon_local->msg.headersize +
    2879           0 :             daemon_local->msg.datasize -
    2880             :             sizeof(DltStorageHeader));
    2881             : 
    2882           0 :         if (daemon_local->msg.found_serialheader)
    2883             :             bytes_to_be_removed += (int) sizeof(dltSerialHeader);
    2884             : 
    2885           0 :         if (daemon_local->msg.resync_offset)
    2886           0 :             bytes_to_be_removed += daemon_local->msg.resync_offset;
    2887             : 
    2888           0 :         if (dlt_receiver_remove(receiver, bytes_to_be_removed) == -1) {
    2889           0 :             dlt_log(LOG_WARNING,
    2890             :                     "Can't remove bytes from receiver for serial connection\n");
    2891           0 :             return -1;
    2892             :         }
    2893             :     } /* while */
    2894             : 
    2895           0 :     if (dlt_receiver_move_to_begin(receiver) == -1) {
    2896           0 :         dlt_log(LOG_WARNING,
    2897             :                 "Can't move bytes to beginning of receiver buffer for serial "
    2898             :                 "connection\n");
    2899           0 :         return -1;
    2900             :     }
    2901             : 
    2902             :     return 0;
    2903             : }
    2904             : 
    2905           2 : int dlt_daemon_process_control_connect(
    2906             :     DltDaemon *daemon,
    2907             :     DltDaemonLocal *daemon_local,
    2908             :     DltReceiver *receiver,
    2909             :     int verbose)
    2910             : {
    2911             :     socklen_t ctrl_size;
    2912             :     struct sockaddr_un ctrl;
    2913             :     int in_sock = -1;
    2914             : 
    2915           2 :     PRINT_FUNCTION_VERBOSE(verbose);
    2916             : 
    2917           2 :     if ((daemon == NULL) || (daemon_local == NULL) || (receiver == NULL)) {
    2918           0 :         dlt_log(LOG_ERR,
    2919             :                 "Invalid function parameters used for function "
    2920             :                 "dlt_daemon_process_control_connect()\n");
    2921           0 :         return -1;
    2922             :     }
    2923             : 
    2924             :     /* event from UNIX server socket, new connection */
    2925           2 :     ctrl_size = sizeof(ctrl);
    2926             : 
    2927           2 :     if ((in_sock = accept(receiver->fd, (struct sockaddr *)&ctrl, &ctrl_size)) < 0) {
    2928           0 :         dlt_vlog(LOG_ERR, "accept() on UNIX control socket %d failed: %s\n", receiver->fd, strerror(errno));
    2929           0 :         return -1;
    2930             :     }
    2931             : 
    2932             :     /* check if file file descriptor was already used, and make it invalid if it
    2933             :      *  is reused */
    2934             :     /* This prevents sending messages to wrong file descriptor */
    2935           2 :     dlt_daemon_applications_invalidate_fd(daemon, daemon->ecuid, in_sock, verbose);
    2936           2 :     dlt_daemon_contexts_invalidate_fd(daemon, daemon->ecuid, in_sock, verbose);
    2937             : 
    2938           2 :     if (dlt_connection_create(daemon_local,
    2939             :                               &daemon_local->pEvent,
    2940             :                               in_sock,
    2941             :                               POLLIN,
    2942             :                               DLT_CONNECTION_CONTROL_MSG)) {
    2943           0 :         dlt_log(LOG_ERR, "Failed to register new client. \n");
    2944             :         /* TODO: Perform clean-up */
    2945           0 :         return -1;
    2946             :     }
    2947             : 
    2948           2 :     if (verbose)
    2949           0 :         dlt_vlog(LOG_INFO, "New connection to control client established\n");
    2950             : 
    2951             :     return 0;
    2952             : }
    2953             : 
    2954             : #if defined DLT_DAEMON_USE_UNIX_SOCKET_IPC || defined DLT_DAEMON_VSOCK_IPC_ENABLE
    2955             : int dlt_daemon_process_app_connect(
    2956             :     DltDaemon *daemon,
    2957             :     DltDaemonLocal *daemon_local,
    2958             :     DltReceiver *receiver,
    2959             :     int verbose)
    2960             : {
    2961             :     int in_sock = -1;
    2962             : 
    2963             :     PRINT_FUNCTION_VERBOSE(verbose);
    2964             : 
    2965             :     if ((daemon == NULL) || (daemon_local == NULL) || (receiver == NULL)) {
    2966             :         dlt_vlog(LOG_ERR,
    2967             :                  "%s: Invalid parameters\n",
    2968             :                  __func__);
    2969             :         return DLT_RETURN_WRONG_PARAMETER;
    2970             :     }
    2971             : 
    2972             :     /* event from server socket, new connection */
    2973             : 
    2974             :     if ((in_sock = accept(receiver->fd, NULL, NULL)) < 0) {
    2975             :         dlt_vlog(LOG_ERR, "accept() on UNIX socket %d failed: %s\n", receiver->fd, strerror(errno));
    2976             :         return -1;
    2977             :     }
    2978             : 
    2979             :     /* check if file file descriptor was already used, and make it invalid if it
    2980             :      * is reused. This prevents sending messages to wrong file descriptor */
    2981             :     dlt_daemon_applications_invalidate_fd(daemon, daemon->ecuid, in_sock, verbose);
    2982             :     dlt_daemon_contexts_invalidate_fd(daemon, daemon->ecuid, in_sock, verbose);
    2983             : 
    2984             :     if (dlt_connection_create(daemon_local,
    2985             :                               &daemon_local->pEvent,
    2986             :                               in_sock,
    2987             :                               POLLIN,
    2988             :                               DLT_CONNECTION_APP_MSG)) {
    2989             :         dlt_log(LOG_ERR, "Failed to register new application. \n");
    2990             :         close(in_sock);
    2991             :         return -1;
    2992             :     }
    2993             : 
    2994             :     if (verbose)
    2995             :         dlt_vlog(LOG_INFO, "New connection to application established\n");
    2996             : 
    2997             :     return 0;
    2998             : }
    2999             : #endif
    3000             : 
    3001           4 : int dlt_daemon_process_control_messages(
    3002             :     DltDaemon *daemon,
    3003             :     DltDaemonLocal *daemon_local,
    3004             :     DltReceiver *receiver,
    3005             :     int verbose)
    3006             : {
    3007             :     int bytes_to_be_removed = 0;
    3008             : 
    3009           4 :     PRINT_FUNCTION_VERBOSE(verbose);
    3010             : 
    3011           4 :     if ((daemon == NULL) || (daemon_local == NULL) || (receiver == NULL)) {
    3012           0 :         dlt_log(LOG_ERR,
    3013             :                 "Invalid function parameters used for function "
    3014             :                 "dlt_daemon_process_control_messages()\n");
    3015           0 :         return -1;
    3016             :     }
    3017             : 
    3018           4 :     if (dlt_receiver_receive(receiver) <= 0) {
    3019           2 :         dlt_daemon_close_socket(receiver->fd,
    3020             :                                 daemon,
    3021             :                                 daemon_local,
    3022             :                                 verbose);
    3023             :         /* FIXME: Why the hell do we need to close the socket
    3024             :          * on control message reception ??
    3025             :          */
    3026           2 :         return 0;
    3027             :     }
    3028             : 
    3029             :     /* Process all received messages */
    3030           4 :     while (dlt_message_read(
    3031             :                &(daemon_local->msg),
    3032           4 :                (uint8_t *)receiver->buf,
    3033           4 :                (unsigned int) receiver->bytesRcvd,
    3034             :                daemon_local->flags.nflag,
    3035           4 :                daemon_local->flags.vflag) == DLT_MESSAGE_ERROR_OK) {
    3036             :         /* Check for control message */
    3037           2 :         if ((receiver->fd > 0) &&
    3038           2 :             DLT_MSG_IS_CONTROL_REQUEST(&(daemon_local->msg)))
    3039           2 :             dlt_daemon_client_process_control(receiver->fd,
    3040             :                                               daemon, daemon_local,
    3041             :                                               &(daemon_local->msg),
    3042             :                                               daemon_local->flags.vflag);
    3043             : 
    3044           2 :         bytes_to_be_removed = (int) (daemon_local->msg.headersize +
    3045           2 :             daemon_local->msg.datasize -
    3046             :             sizeof(DltStorageHeader));
    3047             : 
    3048           2 :         if (daemon_local->msg.found_serialheader)
    3049             :             bytes_to_be_removed += (int) sizeof(dltSerialHeader);
    3050             : 
    3051           2 :         if (daemon_local->msg.resync_offset)
    3052           0 :             bytes_to_be_removed += daemon_local->msg.resync_offset;
    3053             : 
    3054           2 :         if (dlt_receiver_remove(receiver, bytes_to_be_removed) == -1) {
    3055           0 :             dlt_log(LOG_WARNING,
    3056             :                     "Can't remove bytes from receiver for sockets\n");
    3057           0 :             return -1;
    3058             :         }
    3059             :     } /* while */
    3060             : 
    3061           2 :     if (dlt_receiver_move_to_begin(receiver) == -1) {
    3062           0 :         dlt_log(LOG_WARNING, "Can't move bytes to beginning of receiver buffer for sockets\n");
    3063           0 :         return -1;
    3064             :     }
    3065             : 
    3066             :     return 0;
    3067             : }
    3068             : 
    3069           0 : static int dlt_daemon_process_user_message_not_sup(DltDaemon *daemon,
    3070             :                                                    DltDaemonLocal *daemon_local,
    3071             :                                                    DltReceiver *receiver,
    3072             :                                                    int verbose)
    3073             : {
    3074           0 :     DltUserHeader *userheader = (DltUserHeader *)(receiver->buf);
    3075             :     (void)daemon;
    3076             :     (void)daemon_local;
    3077             : 
    3078           0 :     PRINT_FUNCTION_VERBOSE(verbose);
    3079             : 
    3080           0 :     dlt_vlog(LOG_ERR, "Invalid user message type received: %u!\n",
    3081             :              userheader->message);
    3082             : 
    3083             :     /* remove user header */
    3084           0 :     if (dlt_receiver_remove(receiver, sizeof(DltUserHeader)) == -1)
    3085           0 :         dlt_log(LOG_WARNING,
    3086             :                 "Can't remove bytes from receiver for user messages\n");
    3087             : 
    3088           0 :     return -1;
    3089             : }
    3090             : 
    3091             : static dlt_daemon_process_user_message_func process_user_func[DLT_USER_MESSAGE_NOT_SUPPORTED] = {
    3092             :     dlt_daemon_process_user_message_not_sup,
    3093             :     dlt_daemon_process_user_message_log,
    3094             :     dlt_daemon_process_user_message_register_application,
    3095             :     dlt_daemon_process_user_message_unregister_application,
    3096             :     dlt_daemon_process_user_message_register_context,
    3097             :     dlt_daemon_process_user_message_unregister_context,
    3098             :     dlt_daemon_process_user_message_not_sup,
    3099             :     dlt_daemon_process_user_message_not_sup,
    3100             :     dlt_daemon_process_user_message_overflow,
    3101             :     dlt_daemon_process_user_message_set_app_ll_ts,
    3102             :     dlt_daemon_process_user_message_not_sup,
    3103             :     dlt_daemon_process_user_message_not_sup,
    3104             :     dlt_daemon_process_user_message_not_sup,
    3105             :     dlt_daemon_process_user_message_marker,
    3106             :     dlt_daemon_process_user_message_not_sup,
    3107             :     dlt_daemon_process_user_message_not_sup
    3108             : };
    3109             : 
    3110        1501 : int dlt_daemon_process_user_messages(DltDaemon *daemon,
    3111             :                                      DltDaemonLocal *daemon_local,
    3112             :                                      DltReceiver *receiver,
    3113             :                                      int verbose)
    3114             : {
    3115             :     int offset = 0;
    3116             :     int run_loop = 1;
    3117             :     int32_t min_size = (int32_t) sizeof(DltUserHeader);
    3118             :     DltUserHeader *userheader;
    3119             :     int recv;
    3120             : 
    3121        1501 :     PRINT_FUNCTION_VERBOSE(verbose);
    3122             : 
    3123        1501 :     if ((daemon == NULL) || (daemon_local == NULL) || (receiver == NULL)) {
    3124           0 :         dlt_log(LOG_ERR,
    3125             :                 "Invalid function parameters used for function "
    3126             :                 "dlt_daemon_process_user_messages()\n");
    3127           0 :         return -1;
    3128             :     }
    3129             : 
    3130        1501 :     recv = dlt_receiver_receive(receiver);
    3131             : 
    3132        1501 :     if (recv <= 0 && receiver->type == DLT_RECEIVE_SOCKET) {
    3133           0 :         dlt_daemon_close_socket(receiver->fd,
    3134             :                                 daemon,
    3135             :                                 daemon_local,
    3136             :                                 verbose);
    3137           0 :         return 0;
    3138             :     }
    3139        1501 :     else if (recv < 0) {
    3140           0 :         dlt_log(LOG_WARNING,
    3141             :                 "dlt_receiver_receive_fd() for user messages failed!\n");
    3142           0 :         return -1;
    3143             :     }
    3144             : 
    3145             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
    3146             :     /* Count up number of received bytes from FIFO */
    3147             :     if (receiver->bytesRcvd > receiver->lastBytesRcvd)
    3148             :     {
    3149             :         daemon->bytes_recv += receiver->bytesRcvd - receiver->lastBytesRcvd;
    3150             :     }
    3151             : #endif
    3152             : 
    3153             :     /* look through buffer as long as data is in there */
    3154        7398 :     while ((receiver->bytesRcvd >= min_size) && run_loop) {
    3155             :         dlt_daemon_process_user_message_func func = NULL;
    3156             : 
    3157             :         offset = 0;
    3158        5897 :         userheader = (DltUserHeader *)(receiver->buf + offset);
    3159             : 
    3160        5897 :         while (!dlt_user_check_userheader(userheader) &&
    3161           0 :                (offset + min_size <= receiver->bytesRcvd)) {
    3162             :             /* resync if necessary */
    3163           0 :             offset++;
    3164           0 :             userheader = (DltUserHeader *)(receiver->buf + offset);
    3165             :         }
    3166             : 
    3167             :         /* Check for user header pattern */
    3168        5897 :         if (!dlt_user_check_userheader(userheader))
    3169             :             break;
    3170             : 
    3171             :         /* Set new start offset */
    3172        5897 :         if (offset > 0) {
    3173           0 :             if (dlt_receiver_remove(receiver, offset) == -1) {
    3174           0 :                 dlt_log(LOG_WARNING,
    3175             :                         "Can't remove offset from receiver\n");
    3176           0 :                 return -1;
    3177             :             }
    3178             :         }
    3179             : 
    3180        5897 :         if (userheader->message >= DLT_USER_MESSAGE_NOT_SUPPORTED)
    3181             :             func = dlt_daemon_process_user_message_not_sup;
    3182             :         else
    3183        5897 :             func = process_user_func[userheader->message];
    3184             : 
    3185        5897 :         if (func(daemon,
    3186             :                  daemon_local,
    3187             :                  receiver,
    3188             :                  daemon_local->flags.vflag) == -1)
    3189             :             run_loop = 0;
    3190             :     }
    3191             : 
    3192             :     /* keep not read data in buffer */
    3193        1501 :     if (dlt_receiver_move_to_begin(receiver) == -1) {
    3194           0 :         dlt_log(LOG_WARNING,
    3195             :                 "Can't move bytes to beginning of receiver buffer for user "
    3196             :                 "messages\n");
    3197           0 :         return -1;
    3198             :     }
    3199             : 
    3200             :     return 0;
    3201             : }
    3202             : 
    3203           0 : int dlt_daemon_process_user_message_overflow(DltDaemon *daemon,
    3204             :                                              DltDaemonLocal *daemon_local,
    3205             :                                              DltReceiver *rec,
    3206             :                                              int verbose)
    3207             : {
    3208             :     uint32_t len = sizeof(DltUserControlMsgBufferOverflow);
    3209             :     DltUserControlMsgBufferOverflow userpayload;
    3210             : 
    3211           0 :     PRINT_FUNCTION_VERBOSE(verbose);
    3212             : 
    3213           0 :     if ((daemon == NULL) || (daemon_local == NULL) || (rec == NULL)) {
    3214           0 :         dlt_vlog(LOG_ERR, "Invalid function parameters used for %s\n",
    3215             :                  __func__);
    3216           0 :         return -1;
    3217             :     }
    3218             : 
    3219           0 :     if (dlt_receiver_check_and_get(rec,
    3220             :                                    &userpayload,
    3221             :                                    len,
    3222             :                                    DLT_RCV_SKIP_HEADER | DLT_RCV_REMOVE) < 0)
    3223             :         /* Not enough bytes received */
    3224             :         return -1;
    3225             : 
    3226             :     /* Store in daemon, that a message buffer overflow has occured */
    3227             :     /* look if TCP connection to client is available or it least message can be put into buffer */
    3228           0 :     if (dlt_daemon_control_message_buffer_overflow(DLT_DAEMON_SEND_TO_ALL,
    3229             :                                                    daemon,
    3230             :                                                    daemon_local,
    3231             :                                                    userpayload.overflow_counter,
    3232             :                                                    userpayload.apid,
    3233             :                                                    verbose))
    3234             :         /* there was an error when storing message */
    3235             :         /* add the counter of lost messages to the daemon counter */
    3236           0 :         daemon->overflow_counter += userpayload.overflow_counter;
    3237             : 
    3238             :     return 0;
    3239             : }
    3240             : 
    3241           0 : int dlt_daemon_send_message_overflow(DltDaemon *daemon, DltDaemonLocal *daemon_local, int verbose)
    3242             : {
    3243             :     int ret;
    3244           0 :     PRINT_FUNCTION_VERBOSE(verbose);
    3245             : 
    3246           0 :     if ((daemon == 0) || (daemon_local == 0)) {
    3247           0 :         dlt_log(LOG_ERR, "Invalid function parameters used for function dlt_daemon_process_user_message_overflow()\n");
    3248           0 :         return DLT_DAEMON_ERROR_UNKNOWN;
    3249             :     }
    3250             : 
    3251             :     /* Store in daemon, that a message buffer overflow has occured */
    3252           0 :     if ((ret =
    3253           0 :              dlt_daemon_control_message_buffer_overflow(DLT_DAEMON_SEND_TO_ALL, daemon, daemon_local,
    3254             :                                                         daemon->overflow_counter,
    3255             :                                                         "", verbose)))
    3256           0 :         return ret;
    3257             : 
    3258             :     return DLT_DAEMON_ERROR_OK;
    3259             : }
    3260             : 
    3261           7 : int dlt_daemon_process_user_message_register_application(DltDaemon *daemon,
    3262             :                                                          DltDaemonLocal *daemon_local,
    3263             :                                                          DltReceiver *rec,
    3264             :                                                          int verbose)
    3265             : {
    3266             :     uint32_t len = sizeof(DltUserControlMsgRegisterApplication);
    3267             :     uint32_t to_remove = 0;
    3268             :     DltDaemonApplication *application = NULL;
    3269             :     DltDaemonApplication *old_application = NULL;
    3270             :     pid_t old_pid = 0;
    3271           7 :     char description[DLT_DAEMON_DESCSIZE + 1] = { '\0' };
    3272             :     DltUserControlMsgRegisterApplication userapp;
    3273             :     char *origin;
    3274             :     int fd = -1;
    3275             : 
    3276           7 :     PRINT_FUNCTION_VERBOSE(verbose);
    3277             : 
    3278           7 :     if ((daemon == NULL) || (daemon_local == NULL) || (rec == NULL)) {
    3279           0 :         dlt_vlog(LOG_ERR, "Invalid function parameters used for %s\n",
    3280             :                  __func__);
    3281           0 :         return -1;
    3282             :     }
    3283             : 
    3284             :     memset(&userapp, 0, sizeof(DltUserControlMsgRegisterApplication));
    3285           7 :     origin = rec->buf;
    3286             : 
    3287             :     /* Adding temp variable to check the return value */
    3288             :     int temp = 0;
    3289             : 
    3290             :     /* We shall not remove data before checking that everything is there. */
    3291           7 :     temp = dlt_receiver_check_and_get(rec,
    3292             :                                            &userapp,
    3293             :                                            len,
    3294             :                                            DLT_RCV_SKIP_HEADER);
    3295             : 
    3296           7 :     if (temp < 0)
    3297             :         /* Not enough bytes received */
    3298             :         return -1;
    3299             :     else {
    3300           7 :         to_remove = (uint32_t) temp;
    3301             :     }
    3302             : 
    3303           7 :     len = userapp.description_length;
    3304             : 
    3305           7 :     if (len > DLT_DAEMON_DESCSIZE) {
    3306             :         len = DLT_DAEMON_DESCSIZE;
    3307           0 :         dlt_log(LOG_WARNING, "Application description exceeds limit\n");
    3308             :     }
    3309             : 
    3310             :     /* adjust buffer pointer */
    3311           7 :     rec->buf += to_remove + sizeof(DltUserHeader);
    3312             : 
    3313           7 :     if (dlt_receiver_check_and_get(rec, description, len, DLT_RCV_NONE) < 0) {
    3314           0 :         dlt_log(LOG_ERR, "Unable to get application description\n");
    3315             :         /* in case description was not readable, set dummy description */
    3316             :         memcpy(description, "Unknown", sizeof("Unknown"));
    3317             : 
    3318             :         /* unknown len of original description, set to 0 to not remove in next
    3319             :          * step. Because message buffer is re-adjusted the corrupted description
    3320             :          * is ignored. */
    3321             :         len = 0;
    3322             :     }
    3323             : 
    3324             :     /* adjust to_remove */
    3325           7 :     to_remove += (uint32_t) sizeof(DltUserHeader) + len;
    3326             :     /* point to begin of message */
    3327           7 :     rec->buf = origin;
    3328             : 
    3329             :     /* We can now remove data. */
    3330           7 :     if (dlt_receiver_remove(rec, (int) to_remove) != DLT_RETURN_OK) {
    3331           0 :         dlt_log(LOG_WARNING, "Can't remove bytes from receiver\n");
    3332           0 :         return -1;
    3333             :     }
    3334             : 
    3335           7 :     old_application = dlt_daemon_application_find(daemon, userapp.apid, daemon->ecuid, verbose);
    3336             : 
    3337           7 :     if (old_application != NULL)
    3338           0 :         old_pid = old_application->pid;
    3339             : 
    3340           7 :     if (rec->type == DLT_RECEIVE_SOCKET)
    3341           0 :         fd = rec->fd; /* For sockets, an app specific fd has already been created with accept(). */
    3342             : 
    3343           7 :     application = dlt_daemon_application_add(daemon,
    3344             :                                              userapp.apid,
    3345             :                                              userapp.pid,
    3346             :                                              description,
    3347             :                                              fd,
    3348             :                                              daemon->ecuid,
    3349             :                                              verbose);
    3350             : 
    3351             :     /* send log state to new application */
    3352           7 :     dlt_daemon_user_send_log_state(daemon, application, verbose);
    3353             : 
    3354           7 :     if (application == NULL) {
    3355           0 :         dlt_vlog(LOG_WARNING, "Can't add ApplicationID '%.4s' for PID %d\n",
    3356             :                  userapp.apid, userapp.pid);
    3357           0 :         return -1;
    3358             :     }
    3359           7 :     else if (old_pid != application->pid)
    3360             :     {
    3361           7 :         char local_str[DLT_DAEMON_TEXTBUFSIZE] = { '\0' };
    3362             : 
    3363           7 :         snprintf(local_str,
    3364             :                  DLT_DAEMON_TEXTBUFSIZE,
    3365             :                  "ApplicationID '%.4s' registered for PID %d, Description=%s",
    3366           7 :                  application->apid,
    3367             :                  application->pid,
    3368             :                  application->application_description);
    3369           7 :         dlt_daemon_log_internal(daemon, daemon_local, local_str, DLT_LOG_INFO,
    3370             :                                 DLT_DAEMON_APP_ID, DLT_DAEMON_CTX_ID,
    3371             :                                 daemon_local->flags.vflag);
    3372           7 :         dlt_vlog(LOG_DEBUG, "%s%s", local_str, "\n");
    3373             :     }
    3374             : 
    3375             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
    3376             :     if (dlt_daemon_user_send_trace_load_config(daemon, application, verbose) != DLT_RETURN_OK)
    3377             :         dlt_vlog(LOG_WARNING, "Cannot send trace config to Apid: %.4s, PID: %d\n",
    3378             :                  application->apid, application->pid);
    3379             : #endif
    3380             : 
    3381             :     return 0;
    3382             : }
    3383             : 
    3384          38 : int dlt_daemon_process_user_message_register_context(DltDaemon *daemon,
    3385             :                                                      DltDaemonLocal *daemon_local,
    3386             :                                                      DltReceiver *rec,
    3387             :                                                      int verbose)
    3388             : {
    3389             :     uint32_t to_remove = 0;
    3390             :     uint32_t len = sizeof(DltUserControlMsgRegisterContext);
    3391             :     DltUserControlMsgRegisterContext userctxt;
    3392          38 :     char description[DLT_DAEMON_DESCSIZE + 1] = { '\0' };
    3393             :     DltDaemonApplication *application = NULL;
    3394             :     DltDaemonContext *context = NULL;
    3395             :     DltServiceGetLogInfoRequest *req = NULL;
    3396             :     char *origin;
    3397             : 
    3398             :     DltMessage msg;
    3399             : 
    3400          38 :     PRINT_FUNCTION_VERBOSE(verbose);
    3401             : 
    3402          38 :     if ((daemon == NULL) || (daemon_local == NULL) || (rec == NULL)) {
    3403           0 :         dlt_vlog(LOG_ERR, "Invalid function parameters used for %s\n",
    3404             :                  __func__);
    3405           0 :         return -1;
    3406             :     }
    3407             : 
    3408             :     memset(&userctxt, 0, sizeof(DltUserControlMsgRegisterContext));
    3409          38 :     origin = rec->buf;
    3410             : 
    3411             :     /* Adding temp variable to check the return value */
    3412             :     int temp = 0;
    3413             : 
    3414          38 :     temp = dlt_receiver_check_and_get(rec,
    3415             :                                            &userctxt,
    3416             :                                            len,
    3417             :                                            DLT_RCV_SKIP_HEADER);
    3418             : 
    3419          38 :     if (temp < 0)
    3420             :         /* Not enough bytes received */
    3421             :         return -1;
    3422             :     else {
    3423          38 :         to_remove = (uint32_t) temp;
    3424             :     }
    3425             : 
    3426          38 :     len = userctxt.description_length;
    3427             : 
    3428          38 :     if (len > DLT_DAEMON_DESCSIZE) {
    3429           0 :         dlt_vlog(LOG_WARNING, "Context description exceeds limit: %u\n", len);
    3430             :         len = DLT_DAEMON_DESCSIZE;
    3431             :     }
    3432             : 
    3433             :     /* adjust buffer pointer */
    3434          38 :     rec->buf += to_remove + sizeof(DltUserHeader);
    3435             : 
    3436          38 :     if (dlt_receiver_check_and_get(rec, description, len, DLT_RCV_NONE) < 0) {
    3437           0 :         dlt_log(LOG_ERR, "Unable to get context description\n");
    3438             :         /* in case description was not readable, set dummy description */
    3439             :         memcpy(description, "Unknown", sizeof("Unknown"));
    3440             : 
    3441             :         /* unknown len of original description, set to 0 to not remove in next
    3442             :          * step. Because message buffer is re-adjusted the corrupted description
    3443             :          * is ignored. */
    3444             :         len = 0;
    3445             :     }
    3446             : 
    3447             :     /* adjust to_remove */
    3448          38 :     to_remove += (uint32_t) sizeof(DltUserHeader) + len;
    3449             :     /* point to begin of message */
    3450          38 :     rec->buf = origin;
    3451             : 
    3452             :     /* We can now remove data. */
    3453          38 :     if (dlt_receiver_remove(rec, (int) to_remove) != DLT_RETURN_OK) {
    3454           0 :         dlt_log(LOG_WARNING, "Can't remove bytes from receiver\n");
    3455           0 :         return -1;
    3456             :     }
    3457             : 
    3458          38 :     application = dlt_daemon_application_find(daemon,
    3459             :                                               userctxt.apid,
    3460          38 :                                               daemon->ecuid,
    3461             :                                               verbose);
    3462             : 
    3463          38 :     if (application == 0) {
    3464           0 :         dlt_vlog(LOG_WARNING,
    3465             :                  "ApID '%.4s' not found for new ContextID '%.4s' in %s\n",
    3466             :                  userctxt.apid,
    3467             :                  userctxt.ctid,
    3468             :                  __func__);
    3469             : 
    3470           0 :         return 0;
    3471             :     }
    3472             : 
    3473             :     /* Set log level */
    3474          38 :     if (userctxt.log_level == DLT_USER_LOG_LEVEL_NOT_SET) {
    3475          38 :         userctxt.log_level = DLT_LOG_DEFAULT;
    3476             :     } else {
    3477             :         /* Plausibility check */
    3478           0 :         if ((userctxt.log_level < DLT_LOG_DEFAULT) ||
    3479             :                 (userctxt.log_level > DLT_LOG_VERBOSE)) {
    3480             :             return -1;
    3481             :         }
    3482             :     }
    3483             : 
    3484             :     /* Set trace status */
    3485          38 :     if (userctxt.trace_status == DLT_USER_TRACE_STATUS_NOT_SET) {
    3486          38 :         userctxt.trace_status = DLT_TRACE_STATUS_DEFAULT;
    3487             :     } else {
    3488             :         /* Plausibility check */
    3489           0 :         if ((userctxt.trace_status < DLT_TRACE_STATUS_DEFAULT) ||
    3490             :                 (userctxt.trace_status > DLT_TRACE_STATUS_ON)) {
    3491             :             return -1;
    3492             :         }
    3493             :     }
    3494             : 
    3495          38 :     context = dlt_daemon_context_add(daemon,
    3496             :                                      userctxt.apid,
    3497             :                                      userctxt.ctid,
    3498          38 :                                      userctxt.log_level,
    3499          38 :                                      userctxt.trace_status,
    3500             :                                      userctxt.log_level_pos,
    3501             :                                      application->user_handle,
    3502             :                                      description,
    3503             :                                      daemon->ecuid,
    3504             :                                      verbose);
    3505             : 
    3506          38 :     if (context == 0) {
    3507           0 :         dlt_vlog(LOG_WARNING,
    3508             :                  "Can't add ContextID '%.4s' for ApID '%.4s'\n in %s",
    3509             :                  userctxt.ctid, userctxt.apid, __func__);
    3510           0 :         return -1;
    3511             :     }
    3512             :     else {
    3513          38 :         char local_str[DLT_DAEMON_TEXTBUFSIZE] = { '\0' };
    3514             : 
    3515          38 :         snprintf(local_str,
    3516             :                  DLT_DAEMON_TEXTBUFSIZE,
    3517             :                  "ContextID '%.4s' registered for ApID '%.4s', Description=%s",
    3518          38 :                  context->ctid,
    3519          38 :                  context->apid,
    3520             :                  context->context_description);
    3521             : 
    3522          38 :         if (verbose)
    3523           0 :             dlt_daemon_log_internal(daemon, daemon_local, local_str,
    3524             :                                     DLT_LOG_INFO, DLT_DAEMON_APP_ID,
    3525             :                                     DLT_DAEMON_CTX_ID, verbose);
    3526             : 
    3527          38 :         dlt_vlog(LOG_DEBUG, "%s%s", local_str, "\n");
    3528             :     }
    3529             : 
    3530          38 :     if (daemon_local->flags.offlineLogstorageMaxDevices)
    3531             :         /* Store log level set for offline logstorage into context structure*/
    3532          33 :         context->storage_log_level =
    3533          33 :             (int8_t) dlt_daemon_logstorage_get_loglevel(daemon,
    3534          33 :                                                (int8_t) daemon_local->flags.offlineLogstorageMaxDevices,
    3535             :                                                userctxt.apid,
    3536             :                                                userctxt.ctid);
    3537             :     else
    3538           5 :         context->storage_log_level = DLT_LOG_DEFAULT;
    3539             : 
    3540             :     /* Create automatic get log info response for registered context */
    3541          38 :     if (daemon_local->flags.rflag) {
    3542             :         /* Prepare request for get log info with one application and one context */
    3543           0 :         if (dlt_message_init(&msg, verbose) == -1) {
    3544           0 :             dlt_log(LOG_WARNING, "Can't initialize message");
    3545           0 :             return -1;
    3546             :         }
    3547             : 
    3548           0 :         msg.datasize = sizeof(DltServiceGetLogInfoRequest);
    3549             : 
    3550           0 :         if (msg.databuffer && (msg.databuffersize < msg.datasize)) {
    3551           0 :             free(msg.databuffer);
    3552           0 :             msg.databuffer = 0;
    3553             :         }
    3554             : 
    3555           0 :         if (msg.databuffer == 0) {
    3556           0 :             msg.databuffer = (uint8_t *)malloc(msg.datasize);
    3557           0 :             msg.databuffersize = msg.datasize;
    3558             :         }
    3559             : 
    3560           0 :         if (msg.databuffer == 0) {
    3561           0 :             dlt_log(LOG_WARNING, "Can't allocate buffer for get log info message\n");
    3562           0 :             return -1;
    3563             :         }
    3564             : 
    3565             :         req = (DltServiceGetLogInfoRequest *)msg.databuffer;
    3566             : 
    3567           0 :         req->service_id = DLT_SERVICE_ID_GET_LOG_INFO;
    3568           0 :         req->options = (uint8_t) daemon_local->flags.autoResponseGetLogInfoOption;
    3569           0 :         dlt_set_id(req->apid, userctxt.apid);
    3570           0 :         dlt_set_id(req->ctid, userctxt.ctid);
    3571           0 :         dlt_set_id(req->com, "remo");
    3572             : 
    3573           0 :         dlt_daemon_control_get_log_info(DLT_DAEMON_SEND_TO_ALL, daemon, daemon_local, &msg, verbose);
    3574             : 
    3575           0 :         dlt_message_free(&msg, verbose);
    3576             :     }
    3577             : 
    3578          38 :     if (context->user_handle >= DLT_FD_MINIMUM) {
    3579          38 :         if ((userctxt.log_level == DLT_LOG_DEFAULT) || (userctxt.trace_status == DLT_TRACE_STATUS_DEFAULT)) {
    3580             :             /* This call also replaces the default values with the values defined for default */
    3581          38 :             if (dlt_daemon_user_send_log_level(daemon, context, verbose) == -1) {
    3582           0 :                 dlt_vlog(LOG_WARNING, "Can't send current log level as response to %s for (%.4s;%.4s)\n",
    3583             :                          __func__,
    3584             :                          context->apid,
    3585             :                          context->ctid);
    3586           0 :                 return -1;
    3587             :             }
    3588             :         }
    3589             :     }
    3590             : 
    3591             :     return 0;
    3592             : }
    3593             : 
    3594           6 : int dlt_daemon_process_user_message_unregister_application(DltDaemon *daemon,
    3595             :                                                            DltDaemonLocal *daemon_local,
    3596             :                                                            DltReceiver *rec,
    3597             :                                                            int verbose)
    3598             : {
    3599             :     uint32_t len = sizeof(DltUserControlMsgUnregisterApplication);
    3600             :     DltUserControlMsgUnregisterApplication userapp;
    3601             :     DltDaemonApplication *application = NULL;
    3602             :     DltDaemonContext *context;
    3603             :     int i, offset_base;
    3604             :     DltDaemonRegisteredUsers *user_list = NULL;
    3605             : 
    3606           6 :     PRINT_FUNCTION_VERBOSE(verbose);
    3607             : 
    3608           6 :     if ((daemon == NULL) || (daemon_local == NULL) || (rec == NULL)) {
    3609           0 :         dlt_vlog(LOG_ERR,
    3610             :                  "Invalid function parameters used for %s\n",
    3611             :                  __func__);
    3612           0 :         return -1;
    3613             :     }
    3614             : 
    3615           6 :     if (dlt_receiver_check_and_get(rec,
    3616             :                                    &userapp,
    3617             :                                    len,
    3618             :                                    DLT_RCV_SKIP_HEADER | DLT_RCV_REMOVE) < 0)
    3619             :         /* Not enough bytes received */
    3620             :         return -1;
    3621             : 
    3622           6 :     user_list = dlt_daemon_find_users_list(daemon, daemon->ecuid, verbose);
    3623             : 
    3624           6 :     if (user_list == NULL)
    3625             :         return -1;
    3626             : 
    3627           6 :     if (user_list->num_applications > 0) {
    3628             :         /* Delete this application and all corresponding contexts
    3629             :          * for this application from internal table.
    3630             :          */
    3631           6 :         application = dlt_daemon_application_find(daemon,
    3632             :                                                   userapp.apid,
    3633             :                                                   daemon->ecuid,
    3634             :                                                   verbose);
    3635             : 
    3636           6 :         if (application) {
    3637             :             /* Calculate start offset within contexts[] */
    3638             :             offset_base = 0;
    3639             : 
    3640           6 :             for (i = 0; i < (application - (user_list->applications)); i++)
    3641           0 :                 offset_base += user_list->applications[i].num_contexts;
    3642             : 
    3643           6 :             for (i = (application->num_contexts) - 1; i >= 0; i--) {
    3644           0 :                 context = &(user_list->contexts[offset_base + i]);
    3645             : 
    3646           0 :                 if (context) {
    3647             :                     /* Delete context */
    3648           0 :                     if (dlt_daemon_context_del(daemon,
    3649             :                                                context,
    3650             :                                                daemon->ecuid,
    3651             :                                                verbose) == -1) {
    3652           0 :                         dlt_vlog(LOG_WARNING,
    3653             :                                  "Can't delete CtID '%.4s' for ApID '%.4s' in %s\n",
    3654           0 :                                  context->ctid,
    3655           0 :                                  context->apid,
    3656             :                                  __func__);
    3657           0 :                         return -1;
    3658             :                     }
    3659             :                 }
    3660             :             }
    3661             : 
    3662             :             /* Delete this application entry from internal table*/
    3663           6 :             if (dlt_daemon_application_del(daemon,
    3664             :                                            application,
    3665             :                                            daemon->ecuid,
    3666             :                                            verbose) == -1) {
    3667           0 :                 dlt_vlog(LOG_WARNING,
    3668             :                          "Can't delete ApID '%.4s' in %s\n",
    3669           0 :                          application->apid,
    3670             :                          __func__);
    3671           0 :                 return -1;
    3672             :             }
    3673             :             else {
    3674           6 :                 char local_str[DLT_DAEMON_TEXTBUFSIZE] = { '\0' };
    3675             : 
    3676             :                 snprintf(local_str,
    3677             :                          DLT_DAEMON_TEXTBUFSIZE,
    3678             :                          "Unregistered ApID '%.4s'",
    3679             :                          userapp.apid);
    3680           6 :                 dlt_daemon_log_internal(daemon, daemon_local, local_str,
    3681             :                                         DLT_LOG_INFO, DLT_DAEMON_APP_ID,
    3682             :                                         DLT_DAEMON_CTX_ID, verbose);
    3683           6 :                 dlt_vlog(LOG_DEBUG, "%s%s", local_str, "\n");
    3684             :             }
    3685             :         }
    3686             :     }
    3687             : 
    3688             :     return 0;
    3689             : }
    3690             : 
    3691          37 : int dlt_daemon_process_user_message_unregister_context(DltDaemon *daemon,
    3692             :                                                        DltDaemonLocal *daemon_local,
    3693             :                                                        DltReceiver *rec,
    3694             :                                                        int verbose)
    3695             : {
    3696             :     uint32_t len = sizeof(DltUserControlMsgUnregisterContext);
    3697             :     DltUserControlMsgUnregisterContext userctxt;
    3698             :     DltDaemonContext *context;
    3699             : 
    3700          37 :     PRINT_FUNCTION_VERBOSE(verbose);
    3701             : 
    3702          37 :     if ((daemon == NULL) || (daemon_local == NULL) || (rec == NULL)) {
    3703           0 :         dlt_vlog(LOG_ERR,
    3704             :                  "Invalid function parameters used for %s\n",
    3705             :                  __func__);
    3706             : 
    3707           0 :         return -1;
    3708             :     }
    3709             : 
    3710          37 :     if (dlt_receiver_check_and_get(rec,
    3711             :                                    &userctxt,
    3712             :                                    len,
    3713             :                                    DLT_RCV_SKIP_HEADER | DLT_RCV_REMOVE) < 0)
    3714             :         /* Not enough bytes received */
    3715             :         return -1;
    3716             : 
    3717          37 :     context = dlt_daemon_context_find(daemon,
    3718             :                                       userctxt.apid,
    3719             :                                       userctxt.ctid,
    3720          37 :                                       daemon->ecuid,
    3721             :                                       verbose);
    3722             : 
    3723             :     /* In case the daemon is loaded with predefined contexts and its context
    3724             :      * unregisters, the context information will not be deleted from daemon's
    3725             :      * table until its parent application is unregistered.
    3726             :      */
    3727          37 :     if (context && (context->predefined == false)) {
    3728             :         /* Delete this connection entry from internal table*/
    3729          37 :         if (dlt_daemon_context_del(daemon, context, daemon->ecuid, verbose) == -1) {
    3730           0 :             dlt_vlog(LOG_WARNING,
    3731             :                      "Can't delete CtID '%.4s' for ApID '%.4s' in %s\n",
    3732             :                      userctxt.ctid,
    3733             :                      userctxt.apid,
    3734             :                      __func__);
    3735           0 :             return -1;
    3736             :         }
    3737             :         else {
    3738          37 :             char local_str[DLT_DAEMON_TEXTBUFSIZE] = { '\0' };
    3739             : 
    3740             :             snprintf(local_str,
    3741             :                      DLT_DAEMON_TEXTBUFSIZE,
    3742             :                      "Unregistered CtID '%.4s' for ApID '%.4s'",
    3743             :                      userctxt.ctid,
    3744             :                      userctxt.apid);
    3745             : 
    3746          37 :             if (verbose)
    3747           0 :                 dlt_daemon_log_internal(daemon, daemon_local, local_str,
    3748             :                                         DLT_LOG_INFO, DLT_DAEMON_APP_ID,
    3749             :                                         DLT_DAEMON_CTX_ID, verbose);
    3750             : 
    3751          37 :             dlt_vlog(LOG_DEBUG, "%s%s", local_str, "\n");
    3752             :         }
    3753             :     }
    3754             : 
    3755             :     /* Create automatic unregister context response for unregistered context */
    3756          37 :     if (daemon_local->flags.rflag)
    3757           0 :         dlt_daemon_control_message_unregister_context(DLT_DAEMON_SEND_TO_ALL,
    3758             :                                                       daemon,
    3759             :                                                       daemon_local,
    3760             :                                                       userctxt.apid,
    3761             :                                                       userctxt.ctid,
    3762             :                                                       "remo",
    3763             :                                                       verbose);
    3764             : 
    3765             :     return 0;
    3766             : }
    3767             : 
    3768        5809 : int dlt_daemon_process_user_message_log(DltDaemon *daemon,
    3769             :                                         DltDaemonLocal *daemon_local,
    3770             :                                         DltReceiver *rec,
    3771             :                                         int verbose)
    3772             : {
    3773             :     int ret = 0;
    3774             :     int size = 0;
    3775             : 
    3776        5809 :     PRINT_FUNCTION_VERBOSE(verbose);
    3777             : 
    3778        5809 :     if ((daemon == NULL) || (daemon_local == NULL) || (rec == NULL)) {
    3779           0 :         dlt_vlog(LOG_ERR, "%s: invalid function parameters.\n", __func__);
    3780           0 :         return DLT_DAEMON_ERROR_UNKNOWN;
    3781             :     }
    3782             : 
    3783             : #ifdef DLT_SYSTEMD_WATCHDOG_ENFORCE_MSG_RX_ENABLE
    3784             :     daemon->received_message_since_last_watchdog_interval = 1;
    3785             : #endif
    3786             : #ifdef DLT_SHM_ENABLE
    3787             : 
    3788             :     /** In case of SHM, the header still received via fifo/unix_socket receiver,
    3789             :      * so we need to remove header from the receiver.
    3790             :      */
    3791             :     if (dlt_receiver_remove(rec, sizeof(DltUserHeader)) < 0)
    3792             :         /* Not enough bytes received to remove*/
    3793             :         return DLT_DAEMON_ERROR_UNKNOWN;
    3794             : 
    3795             : #ifdef DLT_SYSTEMD_WATCHDOG_ENABLE
    3796             :     const unsigned int start_time = dlt_uptime();
    3797             : #endif
    3798             : 
    3799             :     while (1) {
    3800             : #ifdef DLT_SYSTEMD_WATCHDOG_ENABLE
    3801             :         const unsigned int uptime = dlt_uptime();
    3802             :         if ((uptime - start_time) / 10000 > daemon->watchdog_trigger_interval) {
    3803             :             dlt_vlog(LOG_WARNING,
    3804             :                      "spent already 1 watchdog trigger interval in %s, yielding to process other events.\n", __func__);
    3805             :             if (sd_notify(0, "WATCHDOG=1") < 0)
    3806             :                 dlt_vlog(LOG_CRIT, "Could not reset systemd watchdog from %s\n", __func__);
    3807             :             break;
    3808             :         }
    3809             : #endif
    3810             : 
    3811             :         /* get log message from SHM then store into receiver buffer */
    3812             :         size = dlt_shm_pull(&(daemon_local->dlt_shm),
    3813             :                             daemon_local->recv_buf_shm,
    3814             :                             DLT_SHM_RCV_BUFFER_SIZE);
    3815             : 
    3816             :         if (size <= 0)
    3817             :             break;
    3818             : 
    3819             :         ret = dlt_message_read(&(daemon_local->msg),
    3820             :                                daemon_local->recv_buf_shm, size, 0, verbose);
    3821             : 
    3822             :         if (DLT_MESSAGE_ERROR_OK != ret) {
    3823             :             dlt_shm_remove(&(daemon_local->dlt_shm));
    3824             :             dlt_log(LOG_WARNING, "failed to read messages from shm.\n");
    3825             :             return DLT_DAEMON_ERROR_UNKNOWN;
    3826             :         }
    3827             : 
    3828             : #if defined(DLT_LOG_LEVEL_APP_CONFIG) || defined(DLT_TRACE_LOAD_CTRL_ENABLE)
    3829             :         DltDaemonApplication *app = dlt_daemon_application_find(
    3830             :             daemon, daemon_local->msg.extendedheader->apid, daemon->ecuid, verbose);
    3831             : #endif
    3832             : 
    3833             :         /* discard non-allowed levels if enforcement is on */
    3834             :         bool keep_message = enforce_context_ll_and_ts_keep_message(
    3835             :             daemon_local
    3836             : #ifdef DLT_LOG_LEVEL_APP_CONFIG
    3837             :             , app
    3838             : #endif
    3839             :         );
    3840             : 
    3841             :         // check trace_load
    3842             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
    3843             :         keep_message &= trace_load_keep_message(app, size, daemon, daemon_local, verbose);
    3844             : #endif
    3845             : 
    3846             :         if (keep_message)
    3847             :           dlt_daemon_client_send_message_to_all_client(daemon, daemon_local, verbose);
    3848             : 
    3849             :         if (DLT_DAEMON_ERROR_OK != ret)
    3850             :             dlt_log(LOG_ERR, "failed to send message to client.\n");
    3851             :     }
    3852             : 
    3853             : #else
    3854        5809 :     ret = dlt_message_read(&(daemon_local->msg),
    3855        5809 :                            (unsigned char *)rec->buf + sizeof(DltUserHeader),
    3856        5809 :                            (unsigned int) ((unsigned int) rec->bytesRcvd - sizeof(DltUserHeader)),
    3857             :                            0,
    3858             :                            verbose);
    3859             : 
    3860        5809 :     if (ret != DLT_MESSAGE_ERROR_OK) {
    3861           0 :         if (ret != DLT_MESSAGE_ERROR_SIZE)
    3862             :             /* This is a normal usecase: The daemon reads the data in 10kb chunks.
    3863             :              * Thus the last trace in this chunk is probably not complete and will be completed
    3864             :              * with the next chunk read. This happens always when the FIFO is filled with more than 10kb before
    3865             :              * the daemon is able to read from the FIFO.
    3866             :              * Thus the loglevel of this message is set to DEBUG.
    3867             :              * A cleaner solution would be to check more in detail whether the message is not complete (normal usecase)
    3868             :              * or the headers are corrupted (error case). */
    3869           0 :             dlt_log(LOG_DEBUG, "Can't read messages from receiver\n");
    3870             : 
    3871           0 :         if (dlt_receiver_remove(rec, rec->bytesRcvd) != DLT_RETURN_OK) {
    3872             :             /* In certain rare scenarios where only a partial message has been received
    3873             :              * (Eg: kernel IPC buffer memory being full), we want to discard the message
    3874             :              * and not broadcast it forward to connected clients. Since the DLT library
    3875             :              * checks return value of the writev() call against the sent total message
    3876             :              * length, the partial message will be buffered and retransmitted again.
    3877             :              * This implicitly ensures that no message loss occurs.
    3878             :              */
    3879           0 :             dlt_log(LOG_WARNING, "failed to remove required bytes from receiver.\n");
    3880             :         }
    3881             : 
    3882           0 :         return DLT_DAEMON_ERROR_UNKNOWN;
    3883             :     }
    3884             : 
    3885             : #if defined(DLT_LOG_LEVEL_APP_CONFIG) || defined(DLT_TRACE_LOAD_CTRL_ENABLE)
    3886             :     DltDaemonApplication *app = dlt_daemon_application_find(
    3887             :         daemon, daemon_local->msg.extendedheader->apid, daemon->ecuid, verbose);
    3888             : #endif
    3889             : 
    3890             :     /* discard non-allowed levels if enforcement is on */
    3891           0 :     bool keep_message = enforce_context_ll_and_ts_keep_message(
    3892             :         daemon_local
    3893             : #ifdef DLT_LOG_LEVEL_APP_CONFIG
    3894             :         , app
    3895             : #endif
    3896             :     );
    3897             : 
    3898             :     // check trace_load
    3899             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
    3900             :     keep_message &=
    3901             :         trace_load_keep_message(app, size, daemon, daemon_local, verbose);
    3902             : #endif
    3903             : 
    3904           0 :     if (keep_message)
    3905        5809 :       dlt_daemon_client_send_message_to_all_client(daemon, daemon_local, verbose);
    3906             : 
    3907             :     /* keep not read data in buffer */
    3908        5809 :     size = (int) (daemon_local->msg.headersize +
    3909        5809 :         daemon_local->msg.datasize - sizeof(DltStorageHeader) +
    3910             :         sizeof(DltUserHeader));
    3911             : 
    3912        5809 :     if (daemon_local->msg.found_serialheader)
    3913             :         size += (int) sizeof(dltSerialHeader);
    3914             : 
    3915        5809 :     if (dlt_receiver_remove(rec, size) != DLT_RETURN_OK) {
    3916           0 :         dlt_log(LOG_WARNING, "failed to remove bytes from receiver.\n");
    3917           0 :         return DLT_DAEMON_ERROR_UNKNOWN;
    3918             :     }
    3919             : 
    3920             : #endif
    3921             : 
    3922             :     return DLT_DAEMON_ERROR_OK;
    3923             : }
    3924             : 
    3925           0 : bool enforce_context_ll_and_ts_keep_message(DltDaemonLocal *daemon_local
    3926             : #ifdef DLT_LOG_LEVEL_APP_CONFIG
    3927             :                                             , DltDaemonApplication *app
    3928             : #endif
    3929             : )
    3930             : {
    3931        5809 :     if (!daemon_local->flags.enforceContextLLAndTS ||
    3932           0 :         !daemon_local->msg.extendedheader) {
    3933             :         return true;
    3934             :     }
    3935             : 
    3936             : #ifdef DLT_LOG_LEVEL_APP_CONFIG
    3937             :     if (app == NULL) {
    3938             :         return true;
    3939             :     }
    3940             : #endif
    3941             : 
    3942           0 :     const int mtin = DLT_GET_MSIN_MTIN(daemon_local->msg.extendedheader->msin);
    3943             : #ifdef DLT_LOG_LEVEL_APP_CONFIG
    3944             :     if (app->num_context_log_level_settings > 0) {
    3945             :         DltDaemonContextLogSettings *log_settings =
    3946             :             dlt_daemon_find_app_log_level_config(app, daemon_local->msg.extendedheader->ctid);
    3947             : 
    3948             :         if (log_settings != NULL) {
    3949             :             return mtin <= log_settings->log_level;
    3950             :         }
    3951             :     }
    3952             : #endif
    3953           0 :     return mtin <= daemon_local->flags.contextLogLevel;
    3954             : }
    3955             : 
    3956             : 
    3957             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
    3958             : bool trace_load_keep_message(DltDaemonApplication *app,
    3959             :                              const int size, DltDaemon *const daemon,
    3960             :                              DltDaemonLocal *const daemon_local,
    3961             :                              int verbose)
    3962             : {
    3963             :     bool keep_message = true;
    3964             :     if (app == NULL || !daemon_local->msg.extendedheader) {
    3965             :         return keep_message;
    3966             :     }
    3967             : 
    3968             :     DltMessage* msg = &daemon_local->msg;
    3969             :     const int mtin = DLT_GET_MSIN_MTIN(msg->extendedheader->msin);
    3970             : 
    3971             :     struct DltTraceLoadLogParams params = {
    3972             :         daemon,
    3973             :         daemon_local,
    3974             :         verbose,
    3975             :         app->apid,
    3976             :     };
    3977             : 
    3978             :     DltTraceLoadSettings *trace_load_settings =
    3979             :         dlt_find_runtime_trace_load_settings(
    3980             :             app->trace_load_settings, app->trace_load_settings_count,
    3981             :             app->apid, msg->extendedheader->ctid);
    3982             : 
    3983             :     if (trace_load_settings != NULL) {
    3984             :         pthread_rwlock_wrlock(&trace_load_rw_lock);
    3985             :         keep_message = dlt_check_trace_load(
    3986             :             trace_load_settings, mtin, msg->headerextra.tmsp, size,
    3987             :             dlt_daemon_output_internal_msg, (void *)(&params));
    3988             :         pthread_rwlock_unlock(&trace_load_rw_lock);
    3989             :     }
    3990             :     else {
    3991             :         dlt_vlog(
    3992             :             LOG_ERR,
    3993             :             "Failed to lookup trace load limits for %s, "
    3994             :             "dropping message, likely app was not registered properly\n",
    3995             :             app->apid);
    3996             :         keep_message = false;
    3997             :     }
    3998             : 
    3999             :     return keep_message;
    4000             : }
    4001             : #endif
    4002             : 
    4003           0 : int dlt_daemon_process_user_message_set_app_ll_ts(DltDaemon *daemon,
    4004             :                                                   DltDaemonLocal *daemon_local,
    4005             :                                                   DltReceiver *rec,
    4006             :                                                   int verbose)
    4007             : {
    4008             :     uint32_t len = sizeof(DltUserControlMsgAppLogLevelTraceStatus);
    4009             :     DltUserControlMsgAppLogLevelTraceStatus userctxt;
    4010             :     DltDaemonApplication *application;
    4011             :     DltDaemonContext *context;
    4012             :     int i, offset_base;
    4013             :     int8_t old_log_level, old_trace_status;
    4014             :     DltDaemonRegisteredUsers *user_list = NULL;
    4015             : 
    4016           0 :     PRINT_FUNCTION_VERBOSE(verbose);
    4017             : 
    4018           0 :     if ((daemon == NULL) || (daemon_local == NULL) || (rec == NULL)) {
    4019           0 :         dlt_vlog(LOG_ERR,
    4020             :                  "Invalid function parameters used for %s\n",
    4021             :                  __func__);
    4022           0 :         return DLT_RETURN_ERROR;
    4023             :     }
    4024             : 
    4025           0 :     user_list = dlt_daemon_find_users_list(daemon, daemon->ecuid, verbose);
    4026             : 
    4027           0 :     if (user_list == NULL)
    4028             :         return DLT_RETURN_ERROR;
    4029             : 
    4030             :     memset(&userctxt, 0, len);
    4031             : 
    4032           0 :     if (dlt_receiver_check_and_get(rec,
    4033             :                                    &userctxt,
    4034             :                                    len,
    4035             :                                    DLT_RCV_SKIP_HEADER | DLT_RCV_REMOVE) < 0)
    4036             :         /* Not enough bytes received */
    4037             :         return DLT_RETURN_ERROR;
    4038             : 
    4039           0 :     if (user_list->num_applications > 0) {
    4040             :         /* Get all contexts with application id matching the received application id */
    4041           0 :         application = dlt_daemon_application_find(daemon,
    4042             :                                                   userctxt.apid,
    4043             :                                                   daemon->ecuid,
    4044             :                                                   verbose);
    4045             : 
    4046           0 :         if (application) {
    4047             :             /* Calculate start offset within contexts[] */
    4048             :             offset_base = 0;
    4049             : 
    4050           0 :             for (i = 0; i < (application - (user_list->applications)); i++)
    4051           0 :                 offset_base += user_list->applications[i].num_contexts;
    4052             : 
    4053           0 :             for (i = 0; i < application->num_contexts; i++) {
    4054           0 :                 context = &(user_list->contexts[offset_base + i]);
    4055             : 
    4056           0 :                 if (context) {
    4057           0 :                     old_log_level = context->log_level;
    4058           0 :                     context->log_level = (int8_t) userctxt.log_level; /* No endianess conversion necessary*/
    4059             : 
    4060           0 :                     old_trace_status = context->trace_status;
    4061           0 :                     context->trace_status = (int8_t) userctxt.trace_status;   /* No endianess conversion necessary */
    4062             : 
    4063             :                     /* The following function sends also the trace status */
    4064           0 :                     if ((context->user_handle >= DLT_FD_MINIMUM) &&
    4065           0 :                         (dlt_daemon_user_send_log_level(daemon,
    4066             :                                                         context,
    4067             :                                                         verbose) != 0)) {
    4068           0 :                         context->log_level = old_log_level;
    4069           0 :                         context->trace_status = old_trace_status;
    4070             :                     }
    4071             :                 }
    4072             :             }
    4073             :         }
    4074             :     }
    4075             : 
    4076             :     return DLT_RETURN_OK;
    4077             : }
    4078             : 
    4079           0 : int dlt_daemon_process_user_message_log_mode(DltDaemon *daemon,
    4080             :                                              DltDaemonLocal *daemon_local,
    4081             :                                              DltReceiver *rec,
    4082             :                                              int verbose)
    4083             : {
    4084             :     DltUserControlMsgLogMode userctxt;
    4085             :     uint32_t len = sizeof(DltUserControlMsgLogMode);
    4086             : 
    4087           0 :     PRINT_FUNCTION_VERBOSE(verbose);
    4088             : 
    4089           0 :     if ((daemon == 0) || (daemon_local == 0)) {
    4090           0 :         dlt_log(LOG_ERR, "Invalid function parameters used for function dlt_daemon_process_log_mode()\n");
    4091           0 :         return -1;
    4092             :     }
    4093             : 
    4094             :     memset(&userctxt, 0, len);
    4095             : 
    4096           0 :     if (dlt_receiver_check_and_get(rec,
    4097             :                                    &userctxt,
    4098             :                                    len,
    4099             :                                    DLT_RCV_SKIP_HEADER | DLT_RCV_REMOVE) < 0)
    4100             :         /* Not enough bytes received */
    4101             :         return -1;
    4102             : 
    4103             :     /* set the new log mode */
    4104           0 :     daemon->mode = userctxt.log_mode;
    4105             : 
    4106             :     /* write configuration persistantly */
    4107           0 :     dlt_daemon_configuration_save(daemon, daemon->runtime_configuration, verbose);
    4108             : 
    4109           0 :     return 0;
    4110             : }
    4111             : 
    4112           0 : int dlt_daemon_process_user_message_marker(DltDaemon *daemon,
    4113             :                                            DltDaemonLocal *daemon_local,
    4114             :                                            DltReceiver *rec,
    4115             :                                            int verbose)
    4116             : {
    4117             :     uint32_t len = sizeof(DltUserControlMsgLogMode);
    4118             :     DltUserControlMsgLogMode userctxt;
    4119           0 :     PRINT_FUNCTION_VERBOSE(verbose);
    4120             : 
    4121           0 :     if ((daemon == NULL) || (daemon_local == NULL) || (rec == NULL)) {
    4122           0 :         dlt_vlog(LOG_ERR, "Invalid function parameters used for %s\n",
    4123             :                  __func__);
    4124           0 :         return -1;
    4125             :     }
    4126             : 
    4127             :     memset(&userctxt, 0, len);
    4128             : 
    4129           0 :     if (dlt_receiver_check_and_get(rec,
    4130             :                                    &userctxt,
    4131             :                                    len,
    4132             :                                    DLT_RCV_SKIP_HEADER | DLT_RCV_REMOVE) < 0)
    4133             :         /* Not enough bytes received */
    4134             :         return -1;
    4135             : 
    4136             :     /* Create automatic unregister context response for unregistered context */
    4137           0 :     dlt_daemon_control_message_marker(DLT_DAEMON_SEND_TO_ALL, daemon, daemon_local, verbose);
    4138             : 
    4139           0 :     return 0;
    4140             : }
    4141             : 
    4142           2 : int dlt_daemon_send_ringbuffer_to_client(DltDaemon *daemon, DltDaemonLocal *daemon_local, int verbose)
    4143             : {
    4144             :     int ret;
    4145             :     static uint8_t data[DLT_DAEMON_RCVBUFSIZE];
    4146             :     int length;
    4147             : #ifdef DLT_SYSTEMD_WATCHDOG_ENABLE
    4148             :     uint32_t curr_time = 0U;
    4149             : #endif
    4150             : 
    4151           2 :     PRINT_FUNCTION_VERBOSE(verbose);
    4152             : 
    4153           2 :     if ((daemon == 0) || (daemon_local == 0)) {
    4154           0 :         dlt_log(LOG_ERR, "Invalid function parameters used for function dlt_daemon_send_ringbuffer_to_client()\n");
    4155           0 :         return DLT_DAEMON_ERROR_UNKNOWN;
    4156             :     }
    4157             : 
    4158           2 :     if (dlt_buffer_get_message_count(&(daemon->client_ringbuffer)) <= 0) {
    4159           0 :         dlt_daemon_change_state(daemon, DLT_DAEMON_STATE_SEND_DIRECT);
    4160           0 :         return DLT_DAEMON_ERROR_OK;
    4161             :     }
    4162             : 
    4163           4 :     while ((length = dlt_buffer_copy(&(daemon->client_ringbuffer), data, sizeof(data))) > 0) {
    4164             : #ifdef DLT_SYSTEMD_WATCHDOG_ENABLE
    4165             :         dlt_daemon_trigger_systemd_watchdog_if_necessary(&curr_time, daemon->watchdog_trigger_interval);
    4166             : #endif
    4167             : 
    4168           4 :         if ((ret =
    4169           4 :                  dlt_daemon_client_send(DLT_DAEMON_SEND_FORCE, daemon, daemon_local, 0, 0, data, length, 0, 0,
    4170             :                                         verbose)))
    4171           0 :             return ret;
    4172             : 
    4173           4 :         dlt_buffer_remove(&(daemon->client_ringbuffer));
    4174             : 
    4175           4 :         if (daemon->state != DLT_DAEMON_STATE_SEND_BUFFER)
    4176           0 :             dlt_daemon_change_state(daemon, DLT_DAEMON_STATE_SEND_BUFFER);
    4177             : 
    4178           4 :         if (dlt_buffer_get_message_count(&(daemon->client_ringbuffer)) <= 0) {
    4179           2 :             dlt_daemon_change_state(daemon, DLT_DAEMON_STATE_SEND_DIRECT);
    4180           2 :             return DLT_DAEMON_ERROR_OK;
    4181             :         }
    4182             :     }
    4183             : 
    4184             :     return DLT_DAEMON_ERROR_OK;
    4185             : }
    4186             : 
    4187             : #ifdef __QNX__
    4188             : static void *timer_thread(void *data)
    4189             : {
    4190             :     int pexit = 0;
    4191             :     unsigned int sleep_ret = 0;
    4192             : 
    4193             :     DltDaemonPeriodicData* timer_thread_data = (DltDaemonPeriodicData*) data;
    4194             : 
    4195             :     /* Timer will start in starts_in sec*/
    4196             :     if ((sleep_ret = sleep(timer_thread_data->starts_in))) {
    4197             :         dlt_vlog(LOG_NOTICE, "Sleep remains [%u] for starting!"
    4198             :                 "Stop thread of timer [%d]\n",
    4199             :                 sleep_ret, timer_thread_data->timer_id);
    4200             :          close_pipes(dlt_timer_pipes[timer_thread_data->timer_id]);
    4201             :          return NULL;
    4202             :     }
    4203             : 
    4204             :     while (1) {
    4205             :         if ((dlt_timer_pipes[timer_thread_data->timer_id][1] > 0) &&
    4206             :                 (0 > write(dlt_timer_pipes[timer_thread_data->timer_id][1], "1", 1))) {
    4207             :             dlt_vlog(LOG_ERR, "Failed to send notification for timer [%s]!\n",
    4208             :                     dlt_timer_names[timer_thread_data->timer_id]);
    4209             :             pexit = 1;
    4210             :         }
    4211             : 
    4212             :         if (pexit || g_exit) {
    4213             :             dlt_vlog(LOG_NOTICE, "Received signal!"
    4214             :                     "Stop thread of timer [%d]\n",
    4215             :                     timer_thread_data->timer_id);
    4216             :             close_pipes(dlt_timer_pipes[timer_thread_data->timer_id]);
    4217             :             return NULL;
    4218             :         }
    4219             : 
    4220             :         if ((sleep_ret = sleep(timer_thread_data->period_sec))) {
    4221             :             dlt_vlog(LOG_NOTICE, "Sleep remains [%u] for interval!"
    4222             :                     "Stop thread of timer [%d]\n",
    4223             :                     sleep_ret, timer_thread_data->timer_id);
    4224             :              close_pipes(dlt_timer_pipes[timer_thread_data->timer_id]);
    4225             :              return NULL;
    4226             :         }
    4227             :     }
    4228             : }
    4229             : #endif
    4230             : 
    4231          10 : int create_timer_fd(DltDaemonLocal *daemon_local,
    4232             :                     int period_sec,
    4233             :                     int starts_in,
    4234             :                     DltTimers timer_id)
    4235             : {
    4236             :     int local_fd = DLT_FD_INIT;
    4237             :     char *timer_name = NULL;
    4238             : 
    4239          10 :     if (timer_id >= DLT_TIMER_UNKNOWN) {
    4240           0 :         dlt_log(DLT_LOG_ERROR, "Unknown timer.");
    4241           0 :         return -1;
    4242             :     }
    4243             : 
    4244          10 :     timer_name = dlt_timer_names[timer_id];
    4245             : 
    4246          10 :     if (daemon_local == NULL) {
    4247           0 :         dlt_log(DLT_LOG_ERROR, "Daemon local structure is NULL");
    4248           0 :         return -1;
    4249             :     }
    4250             : 
    4251          10 :     if ((period_sec <= 0) || (starts_in <= 0)) {
    4252             :         /* timer not activated via the service file */
    4253           0 :         dlt_vlog(LOG_INFO, "<%s> not set: period=0\n", timer_name);
    4254             :         local_fd = DLT_FD_INIT;
    4255             :     }
    4256             :     else {
    4257             : #ifdef linux
    4258             :         struct itimerspec l_timer_spec;
    4259          10 :         local_fd = timerfd_create(CLOCK_MONOTONIC, 0);
    4260             : 
    4261          10 :         if (local_fd < 0)
    4262           0 :             dlt_vlog(LOG_WARNING, "<%s> timerfd_create failed: %s\n",
    4263           0 :                      timer_name, strerror(errno));
    4264             : 
    4265          10 :         l_timer_spec.it_interval.tv_sec = period_sec;
    4266          10 :         l_timer_spec.it_interval.tv_nsec = 0;
    4267          10 :         l_timer_spec.it_value.tv_sec = starts_in;
    4268          10 :         l_timer_spec.it_value.tv_nsec = 0;
    4269             : 
    4270          10 :         if (timerfd_settime(local_fd, 0, &l_timer_spec, NULL) < 0) {
    4271           0 :             dlt_vlog(LOG_WARNING, "<%s> timerfd_settime failed: %s\n",
    4272           0 :                      timer_name, strerror(errno));
    4273             :             local_fd = DLT_FD_INIT;
    4274             :         }
    4275             : #elif __QNX__
    4276             :         /*
    4277             :          * Since timerfd is not valid in QNX, new threads are introduced
    4278             :          * to manage timers and communicate with main thread when timer expires.
    4279             :          */
    4280             :         if(0 != pipe(dlt_timer_pipes[timer_id])) {
    4281             :             dlt_vlog(LOG_ERR, "Failed to create pipe for timer [%s]",
    4282             :                     dlt_timer_names[timer_id]);
    4283             :             return -1;
    4284             :         }
    4285             :         if (NULL == timer_data[timer_id]) {
    4286             :             timer_data[timer_id] = calloc(1, sizeof(DltDaemonPeriodicData));
    4287             :             if (NULL == timer_data[timer_id]) {
    4288             :                 dlt_vlog(LOG_ERR, "Failed to allocate memory for timer_data [%s]!\n",
    4289             :                          dlt_timer_names[timer_id]);
    4290             :                 close_pipes(dlt_timer_pipes[timer_id]);
    4291             :                 return -1;
    4292             :             }
    4293             :         }
    4294             : 
    4295             :         timer_data[timer_id]->timer_id = timer_id;
    4296             :         timer_data[timer_id]->period_sec = period_sec;
    4297             :         timer_data[timer_id]->starts_in = starts_in;
    4298             :         timer_data[timer_id]->wakeups_missed = 0;
    4299             : 
    4300             :         if (0 != pthread_create(&timer_threads[timer_id], NULL,
    4301             :                             &timer_thread, (void*)timer_data[timer_id])) {
    4302             :             dlt_vlog(LOG_ERR, "Failed to create new thread for timer [%s]!\n",
    4303             :                                      dlt_timer_names[timer_id]);
    4304             :             /* Clean up timer before returning */
    4305             :             close_pipes(dlt_timer_pipes[timer_id]);
    4306             :             free(timer_data[timer_id]);
    4307             :             timer_data[timer_id] = NULL;
    4308             : 
    4309             :             return -1;
    4310             :         }
    4311             :         local_fd = dlt_timer_pipes[timer_id][0];
    4312             : #endif
    4313             :     }
    4314             : 
    4315          10 :     return dlt_connection_create(daemon_local,
    4316             :                                  &daemon_local->pEvent,
    4317             :                                  local_fd,
    4318             :                                  POLLIN,
    4319          10 :                                  dlt_timer_conn_types[timer_id]);
    4320             : }
    4321             : 
    4322             : /* Close connection function */
    4323           5 : int dlt_daemon_close_socket(int sock, DltDaemon *daemon, DltDaemonLocal *daemon_local, int verbose)
    4324             : {
    4325           5 :     char local_str[DLT_DAEMON_TEXTBUFSIZE] = { '\0' };
    4326             : 
    4327           5 :     PRINT_FUNCTION_VERBOSE(verbose);
    4328             : 
    4329           5 :     if ((daemon_local == NULL) || (daemon == NULL)) {
    4330           0 :         dlt_log(LOG_ERR, "dlt_daemon_close_socket: Invalid input parmeters\n");
    4331           0 :         return -1;
    4332             :     }
    4333             : 
    4334             :     /* Closure is done while unregistering has for any connection */
    4335           5 :     dlt_event_handler_unregister_connection(&daemon_local->pEvent,
    4336             :                                             daemon_local,
    4337             :                                             sock);
    4338             : 
    4339           5 :     if (daemon_local->client_connections == 0) {
    4340             :         /* send new log state to all applications */
    4341           3 :         daemon->connectionState = 0;
    4342           3 :         dlt_daemon_user_send_all_log_state(daemon, verbose);
    4343             : 
    4344             :         /* For offline tracing we still can use the same states */
    4345             :         /* as for socket sending. Using this trick we see the traces */
    4346             :         /* In the offline trace AND in the socket stream. */
    4347           3 :         if (daemon_local->flags.yvalue[0] == 0)
    4348           3 :             dlt_daemon_change_state(daemon, DLT_DAEMON_STATE_BUFFER);
    4349             :     }
    4350             : 
    4351           5 :     dlt_daemon_control_message_connection_info(DLT_DAEMON_SEND_TO_ALL,
    4352             :                                                daemon,
    4353             :                                                daemon_local,
    4354             :                                                DLT_CONNECTION_STATUS_DISCONNECTED,
    4355             :                                                "",
    4356             :                                                verbose);
    4357             : 
    4358           5 :     snprintf(local_str, DLT_DAEMON_TEXTBUFSIZE,
    4359             :              "Client connection #%d closed. Total Clients : %d",
    4360             :              sock,
    4361             :              daemon_local->client_connections);
    4362           5 :     dlt_daemon_log_internal(daemon, daemon_local, local_str, DLT_LOG_INFO,
    4363             :                             DLT_DAEMON_APP_ID, DLT_DAEMON_CTX_ID,
    4364             :                             daemon_local->flags.vflag);
    4365           5 :     dlt_vlog(LOG_DEBUG, "%s%s", local_str, "\n");
    4366             : 
    4367           5 :     return 0;
    4368             : }
    4369             : 
    4370             : #ifdef DLT_TRACE_LOAD_CTRL_ENABLE
    4371             : 
    4372             : static DltReturnValue dlt_daemon_output_internal_msg(
    4373             :     const DltLogLevelType loglevel, const char *const text, void* const params) {
    4374             :     struct DltTraceLoadLogParams* log_params = (struct DltTraceLoadLogParams*)params;
    4375             :     return dlt_daemon_log_internal(
    4376             :         log_params->daemon, log_params->daemon_local, (char *)text, loglevel,
    4377             :         log_params->app_id, DLT_TRACE_LOAD_CONTEXT_ID, log_params->verbose);
    4378             : }
    4379             : #endif
    4380             : 
    4381             : 
    4382             : /**
    4383             :  \}
    4384             :  */

Generated by: LCOV version 1.14