LCOV - code coverage report
Current view: top level - console - dlt-control-common.c (source / functions) Hit Total Coverage
Test: dlt_final_coverage.info Lines: 147 213 69.0 %
Date: 2025-01-09 05:30:37 Functions: 18 19 94.7 %

          Line data    Source code
       1             : /**
       2             :  * Copyright (C) 2015  Advanced Driver Information Technology.
       3             :  * This code is developed by Advanced Driver Information Technology.
       4             :  * Copyright of Advanced Driver Information Technology, Bosch and DENSO.
       5             :  *
       6             :  * This file is part of COVESA Project Dlt - Diagnostic Log and Trace console apps.
       7             :  *
       8             :  *
       9             :  * \copyright
      10             :  * This Source Code Form is subject to the terms of the
      11             :  * Mozilla Public License, v. 2.0. If a copy of the MPL was not distributed with
      12             :  * this file, You can obtain one at http://mozilla.org/MPL/2.0/.
      13             :  *
      14             :  *
      15             :  * \author Christoph Lipka <clipka@jp.adit-jv.com> ADIT 2015
      16             :  * \author Frederic Berat <fberat@de.adit-jv.com> ADIT 2015
      17             :  *
      18             :  * \file dlt-control-common.c
      19             :  * For further information see http://www.covesa.org/.
      20             :  */
      21             : 
      22             : /*******************************************************************************
      23             : **                                                                            **
      24             : **  SRC-MODULE: dlt-control-common.c                                          **
      25             : **                                                                            **
      26             : **  TARGET    : linux                                                         **
      27             : **                                                                            **
      28             : **  PROJECT   : DLT                                                           **
      29             : **                                                                            **
      30             : **  AUTHOR    : Christoph Lipka clipka@jp.adit-jv.com                         **
      31             : **  PURPOSE   :                                                               **
      32             : **                                                                            **
      33             : **  REMARKS   :                                                               **
      34             : **                                                                            **
      35             : **  PLATFORM DEPENDANT [yes/no]: yes                                          **
      36             : **                                                                            **
      37             : **  TO BE CHANGED BY USER [yes/no]: no                                        **
      38             : **                                                                            **
      39             : *******************************************************************************/
      40             : 
      41             : /*******************************************************************************
      42             : **                      Author Identity                                       **
      43             : ********************************************************************************
      44             : **                                                                            **
      45             : ** Initials     Name                       Company                            **
      46             : ** --------     -------------------------  ---------------------------------- **
      47             : **  cl          Christoph Lipka            ADIT                               **
      48             : **  fb          Frederic Berat             ADIT                               **
      49             : *******************************************************************************/
      50             : #define pr_fmt(fmt) "Common control: "fmt
      51             : 
      52             : #include <errno.h>
      53             : #include <dirent.h>
      54             : #include <stdio.h>
      55             : #include <stdlib.h>
      56             : #include <string.h>
      57             : #include <pthread.h>
      58             : #include <sys/types.h>
      59             : #include <sys/socket.h>
      60             : 
      61             : #include "dlt_common.h"
      62             : #include "dlt_protocol.h"
      63             : #include "dlt_client.h"
      64             : 
      65             : #include "dlt-control-common.h"
      66             : 
      67             : #ifdef EXTENDED_FILTERING
      68             :     #   if defined(__linux__) || defined(__ANDROID_API__)
      69             :     #      include <json-c/json.h> /* for json filter parsing on Linux and Android */
      70             :     #   endif
      71             :     #   ifdef __QNX__
      72             :     #      include <sys/json.h> /* for json filter parsing on QNX */
      73             :     #   endif
      74             : #endif
      75             : 
      76             : #define DLT_CTRL_APID    "DLTC"
      77             : #define DLT_CTRL_CTID    "DLTC"
      78             : 
      79             : /** @brief Analyze the daemon answer
      80             :  *
      81             :  * This function as to be provided by the user of the connection.
      82             :  *
      83             :  * @param answer  The textual answer of the daemon
      84             :  * @param payload The daemons answer payload
      85             :  * @param length  The daemons answer payload length
      86             :  *
      87             :  * @return User defined.
      88             :  */
      89             : static int (*response_analyzer_cb)(char *, void *, int);
      90             : 
      91             : static pthread_t daemon_connect_thread;
      92             : static DltClient g_client;
      93             : static int callback_return = -1;
      94             : static pthread_mutex_t answer_lock = PTHREAD_MUTEX_INITIALIZER;
      95             : static pthread_cond_t answer_cond = PTHREAD_COND_INITIALIZER;
      96             : 
      97             : static int local_verbose;
      98             : static char local_ecuid[DLT_CTRL_ECUID_LEN]; /* Name of ECU */
      99             : static int local_timeout;
     100             : static char local_filename[DLT_MOUNT_PATH_MAX]= {0}; /* Path to dlt.conf */
     101             : 
     102          18 : int get_verbosity(void)
     103             : {
     104          46 :     return local_verbose;
     105             : }
     106             : 
     107           0 : void set_verbosity(int v)
     108             : {
     109           2 :     local_verbose = !!v;
     110           0 : }
     111             : 
     112           4 : char *get_ecuid(void)
     113             : {
     114           4 :     return local_ecuid;
     115             : }
     116             : 
     117           2 : void set_ecuid(char *ecuid)
     118             : {
     119           2 :     char *ecuid_conf = NULL;
     120             : 
     121           2 :     if (local_ecuid != ecuid) {
     122             :         /* If user pass NULL, read ECUId from dlt.conf */
     123           0 :         if (ecuid == NULL) {
     124           0 :             if (dlt_parse_config_param("ECUId", &ecuid_conf) == 0) {
     125             :                 memset(local_ecuid, 0, DLT_CTRL_ECUID_LEN);
     126           0 :                 strncpy(local_ecuid, ecuid_conf, DLT_CTRL_ECUID_LEN);
     127           0 :                 local_ecuid[DLT_CTRL_ECUID_LEN -1] = '\0';
     128             :                 if (ecuid_conf !=NULL)
     129           0 :                     free(ecuid_conf);
     130             :             }
     131             :             else {
     132           0 :                 pr_error("Cannot read ECUid from dlt.conf\n");
     133             :             }
     134             :         }
     135             :         else {
     136             :             /* Set user passed ECUID */
     137             :             memset(local_ecuid, 0, DLT_CTRL_ECUID_LEN);
     138             :             strncpy(local_ecuid, ecuid, DLT_CTRL_ECUID_LEN);
     139           0 :             local_ecuid[DLT_CTRL_ECUID_LEN - 1] = '\0';
     140             :         }
     141             :     }
     142           2 : }
     143             : 
     144           2 : void set_conf(char *file_path)
     145             : {
     146           2 :     if (file_path != NULL) {
     147             :         memset(local_filename, 0, DLT_MOUNT_PATH_MAX);
     148             :         strncpy(local_filename, file_path, DLT_MOUNT_PATH_MAX);
     149           2 :         local_filename[DLT_MOUNT_PATH_MAX - 1] = '\0';
     150             :     }
     151             :     else {
     152           0 :         pr_error("Argument is NULL\n");
     153             :     }
     154           2 : }
     155             : 
     156           2 : int get_timeout(void)
     157             : {
     158           2 :     return local_timeout;
     159             : }
     160             : 
     161           2 : void set_timeout(int t)
     162             : {
     163           2 :     local_timeout = DLT_CTRL_TIMEOUT;
     164             : 
     165           2 :     if (t > 1)
     166           2 :         local_timeout = t;
     167             :     else
     168           0 :         pr_error("Timeout to small. Set to default: %d",
     169             :                  DLT_CTRL_TIMEOUT);
     170           2 : }
     171             : 
     172           2 : void set_send_serial_header(const int value)
     173             : {
     174           2 :     g_client.send_serial_header = value;
     175           2 : }
     176             : 
     177           2 : void set_resync_serial_header(const int value)
     178             : {
     179           2 :     g_client.resync_serial_header = value;
     180           2 : }
     181             : 
     182           2 : int dlt_parse_config_param(char *config_id, char **config_data)
     183             : {
     184             :     FILE *pFile = NULL;
     185             :     int value_length = DLT_LINE_LEN;
     186           2 :     char line[DLT_LINE_LEN - 1] = { 0 };
     187           2 :     char token[DLT_LINE_LEN] = { 0 };
     188           2 :     char value[DLT_LINE_LEN] = { 0 };
     189             :     char *pch = NULL;
     190             :     const char *filename = NULL;
     191             : 
     192           2 :     if (*config_data != NULL)
     193           0 :         *config_data = NULL;
     194             : 
     195             :     /* open configuration file */
     196           2 :     if (local_filename[0] != 0) {
     197             :         filename = local_filename;
     198             :     } else {
     199             :         filename = CONFIGURATION_FILES_DIR "/dlt.conf";
     200             :     }
     201           2 :     pFile = fopen(filename, "r");
     202             : 
     203           2 :     if (pFile != NULL) {
     204             :         while (1) {
     205             :             /* fetch line from configuration file */
     206          14 :             if (fgets(line, value_length - 1, pFile) != NULL) {
     207          12 :                 if (strncmp(line, config_id, strlen(config_id)) == 0) {
     208           2 :                     pch = strtok(line, " =\r\n");
     209           2 :                     token[0] = 0;
     210           2 :                     value[0] = 0;
     211             : 
     212           4 :                     while (pch != NULL) {
     213           4 :                         if (token[0] == 0) {
     214             :                             strncpy(token, pch, sizeof(token) - 1);
     215           2 :                             token[sizeof(token) - 1] = 0;
     216             :                         }
     217             :                         else {
     218             :                             strncpy(value, pch, sizeof(value) - 1);
     219           2 :                             value[sizeof(value) - 1] = 0;
     220           2 :                             break;
     221             :                         }
     222             : 
     223           2 :                         pch = strtok(NULL, " =\r\n");
     224             :                     }
     225             : 
     226           2 :                     if (token[0] && value[0]) {
     227           2 :                         if (strcmp(token, config_id) == 0) {
     228           2 :                             *(config_data) = (char *)
     229           2 :                                 calloc(DLT_DAEMON_FLAG_MAX, sizeof(char));
     230             :                             memcpy(*config_data,
     231             :                                    value,
     232             :                                    DLT_DAEMON_FLAG_MAX - 1);
     233             :                         }
     234             :                     }
     235             :                 }
     236             :             }
     237             :             else {
     238             :                 break;
     239             :             }
     240             :         }
     241             : 
     242           2 :         fclose (pFile);
     243             :     }
     244             :     else {
     245           0 :         fprintf(stderr, "Cannot open configuration file: %s\n", filename);
     246             :     }
     247             : 
     248           2 :     if (*config_data == NULL)
     249           0 :         return -1;
     250             : 
     251             :     return 0;
     252             : }
     253             : 
     254             : /** @brief Prepare the extra headers of a DLT message
     255             :  *
     256             :  * Modifies the extra headers of the message so that it can be sent.
     257             :  *
     258             :  * @param msg The message to be prepared
     259             :  * @param header The base header to be used.
     260             :  *
     261             :  * @return 0 on success, -1 otherwise.
     262             :  */
     263           2 : static int prepare_extra_headers(DltMessage *msg, uint8_t *header)
     264             : {
     265             :     uint32_t shift = 0;
     266             : 
     267           2 :     pr_verbose("Preparing extra headers.\n");
     268             : 
     269           2 :     if (!msg || !header)
     270             :         return -1;
     271             : 
     272           2 :     shift = (uint32_t) (sizeof(DltStorageHeader) +
     273             :         sizeof(DltStandardHeader) +
     274           2 :         DLT_STANDARD_HEADER_EXTRA_SIZE(msg->standardheader->htyp));
     275             : 
     276             :     /* Set header extra parameters */
     277           2 :     dlt_set_id(msg->headerextra.ecu, get_ecuid());
     278             : 
     279           2 :     msg->headerextra.tmsp = dlt_uptime();
     280             : 
     281             :     /* Copy header extra parameters to header buffer */
     282           2 :     if (dlt_message_set_extraparameters(msg, get_verbosity()) == -1) {
     283           0 :         pr_error("Cannot copy header extra parameter\n");
     284           0 :         return -1;
     285             :     }
     286             : 
     287             :     /* prepare extended header */
     288           2 :     msg->extendedheader = (DltExtendedHeader *)(header + shift);
     289             : 
     290           2 :     msg->extendedheader->msin = DLT_MSIN_CONTROL_REQUEST;
     291             : 
     292           2 :     msg->extendedheader->noar = 1; /* one payload packet */
     293             : 
     294             :     /* Dummy values have to be set */
     295           2 :     dlt_set_id(msg->extendedheader->apid, DLT_CTRL_APID);
     296           2 :     dlt_set_id(msg->extendedheader->ctid, DLT_CTRL_CTID);
     297             : 
     298           2 :     return 0;
     299             : }
     300             : 
     301             : /** @brief Prepare the headers of a DLT message
     302             :  *
     303             :  * Modifies the headers of the message so that it can be sent.
     304             :  *
     305             :  * @param msg The message to be prepared
     306             :  * @param header The base header to be used.
     307             :  *
     308             :  * @return 0 on success, -1 otherwise.
     309             :  */
     310           2 : static int prepare_headers(DltMessage *msg, uint8_t *header)
     311             : {
     312             :     uint32_t len = 0;
     313             : 
     314           2 :     pr_verbose("Preparing headers.\n");
     315             : 
     316           2 :     if (!msg || !header)
     317             :         return -1;
     318             : 
     319           2 :     msg->storageheader = (DltStorageHeader *)header;
     320             : 
     321           2 :     if (dlt_set_storageheader(msg->storageheader, "") == -1) {
     322           0 :         pr_error("Storage header initialization failed.\n");
     323           0 :         return -1;
     324             :     }
     325             : 
     326             :     /* prepare standard header */
     327           2 :     msg->standardheader =
     328           2 :         (DltStandardHeader *)(header + sizeof(DltStorageHeader));
     329             : 
     330           2 :     msg->standardheader->htyp = DLT_HTYP_WEID |
     331             :         DLT_HTYP_WTMS | DLT_HTYP_UEH | DLT_HTYP_PROTOCOL_VERSION1;
     332             : 
     333             : #if (BYTE_ORDER == BIG_ENDIAN)
     334             :     msg->standardheader->htyp = (msg->standardheader->htyp | DLT_HTYP_MSBF);
     335             : #endif
     336             : 
     337           2 :     msg->standardheader->mcnt = 0;
     338             : 
     339             :     /* prepare length information */
     340           2 :     msg->headersize = (uint32_t) (sizeof(DltStorageHeader) +
     341             :         sizeof(DltStandardHeader) +
     342             :         sizeof(DltExtendedHeader) +
     343             :         DLT_STANDARD_HEADER_EXTRA_SIZE(msg->standardheader->htyp));
     344             : 
     345           2 :     len = (uint32_t) (msg->headersize - sizeof(DltStorageHeader) + msg->datasize);
     346             : 
     347           2 :     if (len > UINT16_MAX) {
     348           0 :         pr_error("Message header is too long.\n");
     349           0 :         return -1;
     350             :     }
     351             : 
     352           2 :     msg->standardheader->len = DLT_HTOBE_16(len);
     353             : 
     354           2 :     return 0;
     355             : }
     356             : 
     357             : /** @brief Prepare a DLT message.
     358             :  *
     359             :  * The DLT message is built using the data given by the user.
     360             :  * The data is basically composed of a buffer and a size.
     361             :  *
     362             :  * @param data The message body to be used to build the DLT message.
     363             :  *
     364             :  * @return 0 on success, -1 otherwise.
     365             :  */
     366           2 : static DltMessage *dlt_control_prepare_message(DltControlMsgBody *data)
     367             : {
     368             :     DltMessage *msg = NULL;
     369             : 
     370           2 :     pr_verbose("Preparing message.\n");
     371             : 
     372           2 :     if (data == NULL) {
     373           0 :         pr_error("Data for message body is NULL\n");
     374           0 :         return NULL;
     375             :     }
     376             : 
     377           2 :     msg = calloc(1, sizeof(DltMessage));
     378             : 
     379           2 :     if (msg == NULL) {
     380           0 :         pr_error("Cannot allocate memory for Dlt Message\n");
     381           0 :         return NULL;
     382             :     }
     383             : 
     384           2 :     if (dlt_message_init(msg, get_verbosity()) == -1) {
     385           0 :         pr_error("Cannot initialize Dlt Message\n");
     386           0 :         free(msg);
     387           0 :         return NULL;
     388             :     }
     389             : 
     390             :     /* prepare payload of data */
     391           2 :     msg->databuffersize = msg->datasize = (uint32_t) data->size;
     392             : 
     393             :     /* Allocate memory for Dlt Message's buffer */
     394           2 :     msg->databuffer = (uint8_t *)calloc(1, data->size);
     395             : 
     396           2 :     if (msg->databuffer == NULL) {
     397           0 :         pr_error("Cannot allocate memory for data buffer\n");
     398           0 :         free(msg);
     399           0 :         return NULL;
     400             :     }
     401             : 
     402             :     /* copy data into message */
     403           2 :     memcpy(msg->databuffer, data->data, data->size);
     404             : 
     405             :     /* prepare storage header */
     406           2 :     if (prepare_headers(msg, msg->headerbuffer)) {
     407           0 :         dlt_message_free(msg, get_verbosity());
     408           0 :         free(msg);
     409           0 :         return NULL;
     410             :     }
     411             : 
     412             :     /* prepare extra headers */
     413           2 :     if (prepare_extra_headers(msg, msg->headerbuffer)) {
     414           0 :         dlt_message_free(msg, get_verbosity());
     415           0 :         free(msg);
     416           0 :         return NULL;
     417             :     }
     418             : 
     419             :     return msg;
     420             : }
     421             : 
     422             : /** @brief Initialize the connection with the daemon
     423             :  *
     424             :  * The connection is initialized using an internal callback. The user's
     425             :  * response analyzer will be finally executed by this callback.
     426             :  * The client pointer is used to established the connection.
     427             :  *
     428             :  * @param client A pointer to a valid client structure
     429             :  * @param cb The internal callback to be executed while receiving a new message
     430             :  *
     431             :  * @return 0 on success, -1 otherwise.
     432             :  */
     433           2 : static int dlt_control_init_connection(DltClient *client, void *cb)
     434             : {
     435           2 :     int (*callback)(DltMessage *message, void *data) = cb;
     436             : 
     437           2 :     if (!cb || !client) {
     438           0 :         pr_error("%s: Invalid parameters\n", __func__);
     439           0 :         return -1;
     440             :     }
     441             : 
     442           2 :     pr_verbose("Initializing the connection.\n");
     443             : 
     444           2 :     if (dlt_client_init(client, get_verbosity()) != 0) {
     445           0 :         pr_error("Failed to register callback (NULL)\n");
     446           0 :         return -1;
     447             :     }
     448             : 
     449           2 :     dlt_client_register_message_callback(callback);
     450             : 
     451           2 :     client->socketPath = NULL;
     452             : 
     453           2 :     if (dlt_parse_config_param("ControlSocketPath", &client->socketPath) != 0) {
     454             :         /* Failed to read from conf, copy default */
     455           0 :         if (dlt_client_set_socket_path(client, DLT_DAEMON_DEFAULT_CTRL_SOCK_PATH) == -1) {
     456           0 :             pr_error("set socket path didn't succeed\n");
     457           0 :             return -1;
     458             :         }
     459             :     }
     460             : 
     461           2 :     client->mode = DLT_CLIENT_MODE_UNIX;
     462             : 
     463           2 :     return dlt_client_connect(client, get_verbosity());
     464             : }
     465             : 
     466             : /** @brief Daemon listener function
     467             :  *
     468             :  * This function will continuously read on the DLT socket, until an error occurs
     469             :  * or the thread executing this function is canceled.
     470             :  *
     471             :  * @param data Thread parameter
     472             :  *
     473             :  * @return The thread parameter given as argument.
     474             :  */
     475           2 : static void *dlt_control_listen_to_daemon(void *data)
     476             : {
     477           2 :     pr_verbose("Ready to receive DLT answers.\n");
     478           2 :     dlt_client_main_loop(&g_client, NULL, get_verbosity());
     479           2 :     return data;
     480             : }
     481             : 
     482             : /** @brief Internal callback for DLT response
     483             :  *
     484             :  * This function is called by the dlt_client_main_loop once a response is read
     485             :  * from the DLT socket.
     486             :  * After some basic checks, the user's response analyzer is called. The return
     487             :  * value of the analyzer is then provided back to the dlt_control_send_message
     488             :  * function to be given back as a return value.
     489             :  * As this function is called in a dedicated thread, the return value is
     490             :  * provided using a global variable.
     491             :  * Access to this variable is controlled through a dedicated mutex.
     492             :  * New values are signaled using a dedicated condition variable.
     493             :  *
     494             :  * @param message The DLT answer
     495             :  * @param data Unused
     496             :  *
     497             :  * @return The analyzer return value or -1 on early errors.
     498             :  */
     499           2 : static int dlt_control_callback(DltMessage *message, void *data)
     500             : {
     501           2 :     char text[DLT_RECEIVE_BUFSIZE] = { 0 };
     502             :     (void)data;
     503             : 
     504           2 :     if (message == NULL) {
     505           0 :         pr_error("Received message is null\n");
     506           0 :         return -1;
     507             :     }
     508             : 
     509             :     /* prepare storage header */
     510           2 :     if (DLT_IS_HTYP_WEID(message->standardheader->htyp))
     511           2 :         dlt_set_storageheader(message->storageheader, message->headerextra.ecu);
     512             :     else
     513           0 :         dlt_set_storageheader(message->storageheader, "LCTL");
     514             : 
     515           2 :     dlt_message_header(message, text, DLT_RECEIVE_BUFSIZE, get_verbosity());
     516             : 
     517             :     /* Extracting payload */
     518           2 :     dlt_message_payload(message, text,
     519             :                         DLT_RECEIVE_BUFSIZE,
     520             :                         DLT_OUTPUT_ASCII,
     521             :                         get_verbosity());
     522             : 
     523             :     /*
     524             :      * Checking payload with the provided callback and return the result
     525             :      */
     526           2 :     pthread_mutex_lock(&answer_lock);
     527           4 :     callback_return = response_analyzer_cb(text,
     528           2 :                                            message->databuffer,
     529           2 :                                            (int) message->datasize);
     530           2 :     pthread_cond_signal(&answer_cond);
     531           2 :     pthread_mutex_unlock(&answer_lock);
     532             : 
     533           2 :     return callback_return;
     534             : }
     535             : 
     536             : /** @brief Send a message to the daemon and wait for the asynchronous answer.
     537             :  *
     538             :  * The answer is received and analyzed by a dedicated thread. Thus we need
     539             :  * to wait for the signal from this thread and then read the return value
     540             :  * to be provided to the user.
     541             :  * In case of timeout, this function fails.
     542             :  * The message provided by the user is formated in DLT format before sending.
     543             :  *
     544             :  * @param body The message provided by the user
     545             :  * @param timeout The time to wait before considering that no answer will come
     546             :  *
     547             :  * @return The user response analyzer return value, -1 in case of early error.
     548             :  */
     549           2 : int dlt_control_send_message(DltControlMsgBody *body, int timeout)
     550             : {
     551             :     struct timespec t;
     552             :     DltMessage *msg = NULL;
     553             : 
     554           2 :     if (!body) {
     555           0 :         pr_error("%s: Invalid input.\n", __func__);
     556           0 :         return -1;
     557             :     }
     558             : 
     559           2 :     if (clock_gettime(CLOCK_REALTIME, &t) == -1) {
     560           0 :         pr_error("Cannot read system time.\n");
     561           0 :         return -1;
     562             :     }
     563             : 
     564           2 :     t.tv_sec += timeout;
     565             : 
     566             :     /* send command to daemon here */
     567           2 :     msg = dlt_control_prepare_message(body);
     568             : 
     569           2 :     if (msg == NULL) {
     570           0 :         pr_error("Control message preparation failed\n");
     571           0 :         return -1;
     572             :     }
     573             : 
     574           2 :     pthread_mutex_lock(&answer_lock);
     575             : 
     576             :     /* Re-init the return value */
     577           2 :     callback_return = -1;
     578             : 
     579           2 :     if (dlt_client_send_message_to_socket(&g_client, msg) != DLT_RETURN_OK)
     580             :     {
     581           0 :         pr_error("Sending message to daemon failed\n");
     582           0 :         dlt_message_free(msg, get_verbosity());
     583           0 :         free(msg);
     584             : 
     585             :         /* make sure the mutex is unlocked to prevent deadlocks */
     586           0 :         pthread_mutex_unlock(&answer_lock);
     587           0 :         return -1;
     588             :     }
     589             : 
     590             :     /*
     591             :     * When a timeouts occurs, pthread_cond_timedwait()
     592             :     * shall nonetheless release and re-acquire the mutex referenced by mutex
     593             :     */
     594           2 :     pthread_cond_timedwait(&answer_cond, &answer_lock, &t);
     595           2 :     pthread_mutex_unlock(&answer_lock);
     596             : 
     597             :     /* Destroying the message */
     598           2 :     dlt_message_free(msg, get_verbosity());
     599           2 :     free(msg);
     600             : 
     601             :     /* At this point either the value is already correct, either it's still -1.
     602             :      * Then, we don't care to lock the access.
     603             :      */
     604           2 :     return callback_return;
     605             : }
     606             : 
     607             : /** @brief Control communication initialization
     608             :  *
     609             :  * This will prepare the DLT connection and the thread dedicated to the
     610             :  * response listening.
     611             :  *
     612             :  * @param response_analyzer User defined function used to analyze the response
     613             :  * @param ecuid The ECUID to provide to the daemon
     614             :  * @param verbosity The verbosity level
     615             :  *
     616             :  * @return 0 on success, -1 otherwise.
     617             :  */
     618           2 : int dlt_control_init(int (*response_analyzer)(char *, void *, int),
     619             :                      char *ecuid,
     620             :                      int verbosity)
     621             : {
     622           2 :     if (!response_analyzer || !ecuid) {
     623           0 :         pr_error("%s: Invalid input.\n", __func__);
     624           0 :         return -1;
     625             :     }
     626             : 
     627           2 :     response_analyzer_cb = response_analyzer;
     628           2 :     set_ecuid(ecuid);
     629             :     set_verbosity(verbosity);
     630             : 
     631           2 :     if (dlt_control_init_connection(&g_client, dlt_control_callback) != 0) {
     632           0 :         pr_error("Connection initialization failed\n");
     633           0 :         dlt_client_cleanup(&g_client, get_verbosity());
     634           0 :         return -1;
     635             :     }
     636             : 
     637             :     /* Contact DLT daemon */
     638           2 :     if (pthread_create(&daemon_connect_thread,
     639             :                        NULL,
     640             :                        dlt_control_listen_to_daemon,
     641             :                        NULL) != 0) {
     642           0 :         pr_error("Cannot create thread to communicate with DLT daemon.\n");
     643           0 :         return -1;
     644             :     }
     645             : 
     646             :     return 0;
     647             : }
     648             : 
     649             : /** @brief Control communication clean-up
     650             :  *
     651             :  * Cancels the listener thread and clean=up the dlt client structure.
     652             :  *
     653             :  * @return 0 on success, -1 otherwise.
     654             :  */
     655           2 : int dlt_control_deinit(void)
     656             : {
     657             :     /* At this stage, we want to stop sending/receiving
     658             :      * from dlt-daemon. So in order to avoid cancellation
     659             :      * at recv(), shutdown and close the socket
     660             :      */
     661           2 :     if (g_client.receiver.fd) {
     662           2 :         shutdown(g_client.receiver.fd, SHUT_RDWR);
     663           2 :         close(g_client.receiver.fd);
     664           2 :         g_client.receiver.fd = -1;
     665             :     }
     666             : 
     667             :         /* Stopping the listener thread */
     668           2 :     if (pthread_cancel(daemon_connect_thread)) {
     669           0 :         pr_error("Unable to cancel the thread with ERRNO=%s\n", strerror(errno));
     670             :     }
     671             :     else {
     672           2 :         if (pthread_join(daemon_connect_thread, NULL)) {
     673           0 :             pr_error("Unable to join the thread with ERRNO=%s\n", strerror(errno));
     674             :         }
     675             :     }
     676             : 
     677             :     /* Closing the socket */
     678           2 :     return dlt_client_cleanup(&g_client, get_verbosity());
     679             : }
     680             : 
     681             : 
     682             : #ifdef EXTENDED_FILTERING /* EXTENDED_FILTERING */
     683             : #   if defined(__linux__) || defined(__ANDROID_API__)
     684             : DltReturnValue dlt_json_filter_load(DltFilter *filter, const char *filename, int verbose)
     685             : {
     686             :     if ((filter == NULL) || (filename == NULL))
     687             :         return DLT_RETURN_WRONG_PARAMETER;
     688             : 
     689             :     if(verbose)
     690             :         pr_verbose("dlt_json_filter_load()\n");
     691             : 
     692             :     FILE *handle;
     693             :     char buffer[JSON_FILTER_SIZE*DLT_FILTER_MAX];
     694             :     struct json_object *j_parsed_json;
     695             :     struct json_object *j_app_id;
     696             :     struct json_object *j_context_id;
     697             :     struct json_object *j_log_level;
     698             :     struct json_object *j_payload_min;
     699             :     struct json_object *j_payload_max;
     700             :     enum json_tokener_error jerr;
     701             : 
     702             :     char app_id[DLT_ID_SIZE + 1] = "";
     703             :     char context_id[DLT_ID_SIZE + 1] = "";
     704             :     int32_t log_level = 0;
     705             :     int32_t payload_max = INT32_MAX;
     706             :     int32_t payload_min = 0;
     707             : 
     708             :     handle = fopen(filename, "r");
     709             : 
     710             :     if (handle == NULL) {
     711             :         pr_error("Filter file %s cannot be opened!\n", filename);
     712             :         return DLT_RETURN_ERROR;
     713             :     }
     714             : 
     715             :     if (fread(buffer, sizeof(buffer), 1, handle) != 0) {
     716             :         if (!feof(handle)) {
     717             :             pr_error("Filter file %s is to big for reading it with current buffer!\n", filename);
     718             :             return DLT_RETURN_ERROR;
     719             :         }
     720             :     }
     721             : 
     722             :     j_parsed_json = json_tokener_parse_verbose(buffer, &jerr);
     723             : 
     724             :     if (jerr != json_tokener_success) {
     725             :         pr_error("Faild to parse given filter %s: %s\n", filename, json_tokener_error_desc(jerr));
     726             :         return DLT_RETURN_ERROR;
     727             :     }
     728             : 
     729             :     printf("The following filter(s) are applied: \n");
     730             :     pr_verbose("The following filter(s) are applied: \n");
     731             :     int iterator = 0;
     732             :     json_object_object_foreach(j_parsed_json, key, val)
     733             :     {
     734             :         if (iterator >= DLT_FILTER_MAX) {
     735             :             pr_error("Maximum number (%d) of allowed filters reached, ignoring rest of filters!\n",
     736             :                      DLT_FILTER_MAX);
     737             :             break;
     738             :         }
     739             : 
     740             :         printf("%s:\n", key);
     741             :         pr_verbose("%s:\n", key);
     742             : 
     743             :         if (json_object_object_get_ex(val, "AppId", &j_app_id))
     744             :             strncpy(app_id, json_object_get_string(j_app_id), DLT_ID_SIZE);
     745             :         else
     746             :             dlt_set_id(app_id, "");
     747             : 
     748             :         if (json_object_object_get_ex(val, "ContextId", &j_context_id))
     749             :             strncpy(context_id, json_object_get_string(j_context_id), DLT_ID_SIZE);
     750             :         else
     751             :             dlt_set_id(context_id, "");
     752             : 
     753             :         if (json_object_object_get_ex(val, "LogLevel", &j_log_level))
     754             :             log_level = json_object_get_int(j_log_level);
     755             :         else
     756             :             log_level = 0;
     757             : 
     758             :         if (json_object_object_get_ex(val, "PayloadMin", &j_payload_min))
     759             :             payload_min = json_object_get_int(j_payload_min);
     760             :         else
     761             :             payload_min = 0;
     762             : 
     763             :         if (json_object_object_get_ex(val, "PayloadMax", &j_payload_max))
     764             :             payload_max = json_object_get_int(j_payload_max);
     765             :         else
     766             :             payload_max = INT32_MAX;
     767             : 
     768             :         dlt_filter_add(filter, app_id, context_id, log_level, payload_min, payload_max, verbose);
     769             : 
     770             :         printf("\tAppId: %.*s\n", DLT_ID_SIZE, app_id);
     771             :         pr_verbose("\tAppId: %.*s\n", DLT_ID_SIZE, app_id);
     772             :         printf("\tConextId: %.*s\n", DLT_ID_SIZE, context_id);
     773             :         pr_verbose("\tConextId: %.*s\n", DLT_ID_SIZE, context_id);
     774             :         printf("\tLogLevel: %i\n", log_level);
     775             :         pr_verbose("\tLogLevel: %i\n", log_level);
     776             :         printf("\tPayloadMin: %i\n", payload_min);
     777             :         pr_verbose("\tPayloadMin: %i\n", payload_min);
     778             :         printf("\tPayloadMax: %i\n", payload_max);
     779             :         pr_verbose("\tPayloadMax: %i\n", payload_max);
     780             : 
     781             :         iterator++;
     782             :     }
     783             : 
     784             :     fclose(handle);
     785             : 
     786             :     return DLT_RETURN_OK;
     787             : }
     788             : #   endif /* __Linux__ */
     789             : 
     790             : #   ifdef __QNX__
     791             : DltReturnValue dlt_json_filter_load(DltFilter *filter, const char *filename, int verbose)
     792             : {
     793             :     if ((filter == NULL) || (filename == NULL))
     794             :         return DLT_RETURN_WRONG_PARAMETER;
     795             : 
     796             :     if(verbose)
     797             :         pr_verbose("dlt_json_filter_load()\n");
     798             : 
     799             :     json_decoder_t *j_decoder = json_decoder_create();
     800             : 
     801             :     const char *s_app_id;
     802             :     const char *s_context_id;
     803             :     int32_t log_level = 0;
     804             :     int32_t payload_max = INT32_MAX;
     805             :     int32_t payload_min = 0;
     806             : 
     807             :     json_decoder_error_t ret = json_decoder_parse_file(j_decoder, filename);
     808             : 
     809             :     if (ret != JSON_DECODER_OK) {
     810             :         pr_error("Faild to parse given filter %s: json_decoder_error_t is %i\n", filename, ret);
     811             :         return DLT_RETURN_ERROR;
     812             :     }
     813             : 
     814             :     json_decoder_push_object(j_decoder, NULL, true);
     815             : 
     816             :     int iterator = 0;
     817             :     bool end_of_json = false;
     818             : 
     819             :     while (!end_of_json) {
     820             :         if (iterator >= DLT_FILTER_MAX) {
     821             :             pr_error("Maximum number (%d) of allowed filters reached, ignoring rest of filters!\n",
     822             :                      DLT_FILTER_MAX);
     823             :             break;
     824             :         }
     825             : 
     826             :         if (json_decoder_next(j_decoder) == JSON_DECODER_NOT_FOUND)
     827             :             end_of_json = true;
     828             : 
     829             :         json_decoder_previous(j_decoder);
     830             : 
     831             :         printf("%s:\n", json_decoder_name(j_decoder));
     832             :         json_decoder_push_object(j_decoder, NULL, true);
     833             : 
     834             :         if (json_decoder_get_string(j_decoder, "AppId", &s_app_id, true) != JSON_DECODER_OK)
     835             :             s_app_id = "";
     836             : 
     837             :         if (json_decoder_get_string(j_decoder, "ContextId", &s_context_id, true) != JSON_DECODER_OK)
     838             :             s_context_id = "";
     839             : 
     840             :         if (json_decoder_get_int(j_decoder, "LogLevel", &log_level, true) != JSON_DECODER_OK)
     841             :             log_level = 0;
     842             : 
     843             :         if (json_decoder_get_int(j_decoder, "PayloadMin", &payload_min, true) != JSON_DECODER_OK)
     844             :             payload_min = 0;
     845             : 
     846             :         if (json_decoder_get_int(j_decoder, "PayloadMax", &payload_max, true) != JSON_DECODER_OK)
     847             :             payload_max = INT32_MAX;
     848             : 
     849             :         char app_id[DLT_ID_SIZE];
     850             :         char context_id[DLT_ID_SIZE];
     851             : 
     852             :         #pragma GCC diagnostic push
     853             :         #pragma GCC diagnostic ignored "-Wstringop-truncation"
     854             :         strncpy(app_id, s_app_id, DLT_ID_SIZE);
     855             :         strncpy(context_id, s_context_id, DLT_ID_SIZE);
     856             :         #pragma GCC diagnostic pop
     857             : 
     858             :         dlt_filter_add(filter, app_id, context_id, log_level, payload_min, payload_max, verbose);
     859             : 
     860             :         printf("\tAppId: %.*s\n", DLT_ID_SIZE, app_id);
     861             :         printf("\tConextId: %.*s\n", DLT_ID_SIZE, context_id);
     862             :         printf("\tLogLevel: %i\n", log_level);
     863             :         printf("\tPayloadMin: %i\n", payload_min);
     864             :         printf("\tPayloadMax: %i\n", payload_max);
     865             : 
     866             :         json_decoder_pop(j_decoder);
     867             : 
     868             :         iterator++;
     869             :     }
     870             : 
     871             :     json_decoder_destroy(j_decoder);
     872             : 
     873             :     return DLT_RETURN_OK;
     874             : }
     875             : #   endif /* __QNX__ */
     876             : #endif /* EXTENDED_FILTERING */
     877             : 
     878             : #ifdef EXTENDED_FILTERING /* EXTENDED_FILTERING */
     879             : #   if defined(__linux__) || defined(__ANDROID_API__)
     880             : DltReturnValue dlt_json_filter_save(DltFilter *filter, const char *filename, int verbose)
     881             : {
     882             :     if ((filter == NULL) || (filename == NULL))
     883             :         return DLT_RETURN_WRONG_PARAMETER;
     884             : 
     885             :     if(verbose)
     886             :         pr_verbose("dlt_json_filter_save()\n");
     887             : 
     888             :     struct json_object *json_filter_obj = json_object_new_object();
     889             : 
     890             :     for (int num = 0; num < filter->counter; num++) {
     891             :         struct json_object *tmp_json_obj = json_object_new_object();
     892             :         char filter_name[JSON_FILTER_NAME_SIZE];
     893             :         sprintf(filter_name, "filter%i", num);
     894             : 
     895             :         if (filter->apid[num][DLT_ID_SIZE - 1] != 0)
     896             :             json_object_object_add(tmp_json_obj, "AppId", json_object_new_string_len(filter->apid[num], DLT_ID_SIZE));
     897             :         else
     898             :             json_object_object_add(tmp_json_obj, "AppId", json_object_new_string(filter->apid[num]));
     899             : 
     900             :         if (filter->ctid[num][DLT_ID_SIZE - 1] != 0)
     901             :             json_object_object_add(tmp_json_obj, "ContextId",
     902             :                                    json_object_new_string_len(filter->ctid[num], DLT_ID_SIZE));
     903             :         else
     904             :             json_object_object_add(tmp_json_obj, "ContextId", json_object_new_string(filter->ctid[num]));
     905             : 
     906             :         json_object_object_add(tmp_json_obj, "LogLevel", json_object_new_int(filter->log_level[num]));
     907             :         json_object_object_add(tmp_json_obj, "PayloadMin", json_object_new_int(filter->payload_min[num]));
     908             :         json_object_object_add(tmp_json_obj, "PayloadMax", json_object_new_int(filter->payload_max[num]));
     909             : 
     910             :         json_object_object_add(json_filter_obj, filter_name, tmp_json_obj);
     911             :     }
     912             : 
     913             :     printf("Saving current filter into '%s'\n", filename);
     914             :     json_object_to_file((char*)filename, json_filter_obj);
     915             : 
     916             :     return DLT_RETURN_OK;
     917             : }
     918             : #   endif /* __Linux__ */
     919             : 
     920             : #   ifdef __QNX__
     921             : DltReturnValue dlt_json_filter_save(DltFilter *filter, const char *filename, int verbose)
     922             : {
     923             :     if ((filter == NULL) || (filename == NULL))
     924             :         return DLT_RETURN_WRONG_PARAMETER;
     925             : 
     926             :     if(verbose)
     927             :         pr_verbose("dlt_json_filter_save()\n");
     928             : 
     929             :     char s_app_id[DLT_ID_SIZE + 1];
     930             :     char s_context_id[DLT_ID_SIZE + 1];
     931             : 
     932             :     json_encoder_t *j_encoder = json_encoder_create();
     933             :     json_encoder_start_object(j_encoder, NULL);
     934             : 
     935             :     for (int num = 0; num < filter->counter; num++) {
     936             :         char filter_name[JSON_FILTER_NAME_SIZE];
     937             :         sprintf(filter_name, "filter%i", num);
     938             :         json_encoder_start_object(j_encoder, filter_name);
     939             : 
     940             :         strncpy(s_app_id, filter->apid[num], DLT_ID_SIZE);
     941             : 
     942             :         if (filter->apid[num][DLT_ID_SIZE - 1] != 0)
     943             :             s_app_id[DLT_ID_SIZE] = '\0';
     944             : 
     945             :         strncpy(s_context_id, filter->ctid[num], DLT_ID_SIZE);
     946             : 
     947             :         if (filter->ctid[num][DLT_ID_SIZE - 1] != 0)
     948             :             s_context_id[DLT_ID_SIZE] = '\0';
     949             : 
     950             :         json_encoder_add_string(j_encoder, "AppId", s_app_id);
     951             :         json_encoder_add_string(j_encoder, "ContextId", s_context_id);
     952             :         json_encoder_add_int(j_encoder, "LogLevel", filter->log_level[num]);
     953             :         json_encoder_add_int(j_encoder, "PayloadMin", filter->payload_min[num]);
     954             :         json_encoder_add_int(j_encoder, "PayloadMax", filter->payload_max[num]);
     955             : 
     956             :         json_encoder_end_object(j_encoder);
     957             :     }
     958             : 
     959             :     json_encoder_end_object(j_encoder);
     960             : 
     961             :     printf("Saving current filter into '%s'\n", filename);
     962             :     FILE *handle = fopen(filename, "w");
     963             :     int filter_buffer_size = 100 * (filter->counter);
     964             :     char filter_buffer[filter_buffer_size];
     965             :     snprintf(filter_buffer, filter_buffer_size, json_encoder_buffer(j_encoder));
     966             :     fprintf(handle, filter_buffer);
     967             : 
     968             :     fclose(handle);
     969             :     json_encoder_destroy(j_encoder);
     970             : 
     971             :     return DLT_RETURN_OK;
     972             : }
     973             : #   endif /* __QNX__ */
     974             : #endif /* EXTENDED_FILTERING */

Generated by: LCOV version 1.14