SPDX license identifier: MPL-2.0
Copyright (C) 2011-2015, BMW AG
This file is part of COVESA Project DLT - Diagnostic Log and Trace.
This Source Code Form is subject to the terms of the Mozilla Public License (MPL), v. 2.0. If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/
For further information see http://www.covesa.org/
Alexander Wenzel [email protected] 0.0.1, 2012/10/10: Initial version
Luong Hong Duy Khanh [email protected] 0.0.2, 2020/04/10: Update and convert to markdown format
This document specifies the usage of the DLT daemon v2 and also the internal functionality of the DLT daemon v2. The DLT daemon v2 is a complete rework of the DLT daemon v1, which is part of the COVESA 1.0 release.
The DLT daemon component is based on the AUTOSAR 4.0 standard DLT (the current latest version is R19-11).
The DLT daemon is the central component in COVESA, where logs and traces are gathered from the DLT user and different applications, stored optionally temporarily or permanently in a file in the ECU. The DLT daemon forwards all logs and traces to a connected DLT client application, which can run directly on the COVESA system or more likely on an external tester device.
The DLT client can send control messages to the daemon, e.g. to set individual log levels of applications and contexts or get the list of applications and contexts registered in the DLT daemon.
In DLT, a log message contains debug information like state changes or value changes and a trace message contains information, which has passed via the Virtual Function Bus (VFB).
The DLT daemon is the central component between the DLT clients and one or more applications using the DLT user library.
The DLT daemon is a standalone application which is running as the center of management and this is implemented as a single-thread process. The DLT daemon communicates with the DLT clients over TCP/IP connections or over a serial line with the applications using the DLT user library over named pipes (FIFOs), UNIX sockets or VSOCK sockets based on compile time configuration.
The message format is specified in the DLT AUTOSAR Standard. More details concerning the exchanged user messages and their content can be found in chapter 6.
The main time, the DLT daemon executes the main loop, in which file and socket descriptors are watched (via poll()). If a message is received, the DLT daemon reacts.
Here is a rough schematic, how the DLT daemon is structured:
During initialization of the DLT daemon, the following steps occur:
- Option handling.
- Option file parsed. The most important configuration files is dlt.conf
- Initialization of logging facility for the DLT daemon application. The DLT daemon application by itself prints to screen, if not in daemon mode. In daemon mode, it uses the syslog daemon for log messages of the DLT daemon application.
- Then the daemon enters initialization phase 1:
- Enter daemon mode, if started as daemon
- Initialize output of messages to the local file
- Parse filter file, if specified; then set filters for file storage
- Setup signal handler
- Open DLT output file, if specified
- After phase 1, the daemon initializes the connection handling:
- Delete, create and open its own named FIFO /tmp/dlt or UNIX_SOCKET
- If VSOCK support is enabled in the daemon, open, bind and listen to VSOCK socket for incoming application connections.
- Open, bind and listen to TCP socket for incoming connections
- Setup and open serial device, if specified
- Then the daemon enters initialization phase 2:
- Initialize daemon data structures
- Initialize ring buffer for client connection
- Check if a runtime configuration exists, if yes load all stored application and context ids, as well as all log levels and trace statuses. Store this information to a local array.
- Prepare the main loop
- Initialize receiver objects for socket and serial connection
- Initialize binary semaphore
- Create and start thread(s) for timing messages, sending of these messages is disabled at default.
- After phase 2:
- Initialize the offline logstorage, if the directory path is specified.
- Prepare timer for system watchdog, if specified.
- Prepare timer to check incoming events.
- Prepare timer to send ECU version, if specified.
- Initialize the gateway mode, if specified, then prepare the timer for gateway.
- Now, the initialization is finished, and the DLT daemon enters the main loop.
In the main loop, the following things occur:
- Wait for event(s) on the incoming named pipe or socket, the TCP connections (the one connection for new connections, and all TCP connections from clients), and possibly on the serial device, via poll() call.
- Distinguish from which socket/file descriptor the connection came, and handle them:
- Event from TCP server socket (New DLT client to DLT daemon):
- Create a new TCP connection
- If the newly created connection is the first TCP connection, send all log messages already stored in ring buffer to DLT client.
- Event from incoming named pipe (FIFO) or UNIX socket and, if enabled, VSOCK
socket (from DLT user library to DLT daemon):
- Use dlt receiver to read data
- As long as there are DLT messages available in received data, the handle user messages type can be found at src/shared/dlt_user_shared_cfg.h
- Move rest of data in front of buffer for next read attempt
- Event from serial device (DLT client to DLT daemon via serial device)
- Use other dlt receiver to read data
- As long as there are DLT messages available in received data:
- Check for DLT message type control request, and if yes process the request
- Move rest of data in front of buffer for next read attempt
- Event from TCP socket (DLT client to DLT daemon via TCP connection)
- Use other dlt receiver to read data, and check for lost connection
- As long as there are DLT messages available in received data:
- Check for DLT message type control request, and if yes process the request
- Move rest of data in front of buffer for next read attempt
- Event from TCP server socket (New DLT client to DLT daemon):
How the received user messages and control messages are handled, is described in the Appendix.
Before exiting from DLT daemon, the following cleanup procedures occur:
- Local cleanup
- Destroy the connection list
- Free the used memory by the organising structure of file
- Uninitialise the offline trace
- Free the used memory by the organising structure of file
- Delete existing named pipes (FIFOs) or UNIX_SOCKET
- Cleanup dlt logstorage
- Close UDP connection, if specified
- De-initialize the gateway, if specified. All internal data will be freed
- De-Initialise the dlt daemon structure
- Clear all contexts in internal context management of specific ecu
- Clear all applications in internal application management of specific ecu
- Release and free memory used by dynamic ringbuffer
Now, the cleanup is finished, and the DLT daemon is terminated.
The DLT user library is linked to each application that wants to use DLT. It encapsulates the communication with the DLT daemon and provides two interfaces: the DLT user macro interface and the DLT user functional interface. All macros from the DLT user macro interface are mapped to functions in the DLT user functional interface. The DLT user library has one additional thread, called housekeeper thread, responsible for receiving and sending messages from/to the DLT daemon.
The DLT user library can be written/read from multiple threads (the threads of applications which are using DLT APIs, and DLT housekeeper thread which will send log messages to the daemon). To prevent the concurrent access to the DLT buffer, parts of functions accessible by the user interface are protected by a semaphore as well as parts of the function which is called within the thread.
During initialization, the following things are done:
- Setup internal structure including Application ID and Description (textual) of application
- Get value from environment variable "DLT_LOCAL_PRINT_MODE". This variable can
be used to control the local printing mode of the DLT user library. If enabled
(either via "AUTOMATIC" or via "FORCE_ON"), all messages, which are prepared to
be sent to the DLT daemon will also be print out locally (as ASCII string). If
the environment variable is set, this value overrides the local print mode which
can be set with the API function dlt_enable_local_print() or with dlt_disable_local_print()
from within the application using the DLT user library. The following values
are allowed:
- "AUTOMATIC": Local printing is enabled, if NO DLT daemon is running.
- "FORCE_ON": Local printing is always enabled.
- "FORCE_OFF": Local printing is always disabled.
- Clear internal context array (dynamically growing in step size
DLT_USER_CONTEXT_ALLOC_SIZE, typically 500). The internal context array is NOT
be kept sorted, as the DLT daemon stores for each registered context the offset
position within this array, and sends this offset position for faster access of
a context within this internal context array. The internal context array
contains one entry for each context:
- Context ID
- Log level for this context
- Trace status for this context
- Initialize table (dynamically growing in step-size 1) with function pointers for callback functions used for injection messages
- Description (textual) of context
- Initialize ringbuffer for local storage of not sent messages
- Setup signal handler and atexit handler
- Ignore all pipe signals
- Create and open own named pipe (with the name /tmp/dlt, where is the process id of the application using the DLT user library) or UNIX/VSOCK socket IPC to the DLT daemon
- Open local file for storage, if specified
- Initialize receiver object
- Start housekeeper thread for receiving messages
During de-initialization, the following things are done:
- De-register application (and all contexts belonging to this application) from DLT daemon
- Stop housekeeper thread
- Close and remove own named pipe to the DLT daemon, if FIFO is used
- Close and remove Unix socket if socket IPC is used
- De-Initialize receiver object
- De-Initialize ringbuffer
During register of the application, the following things occur:
- Auto-initialize DLT user library, if necessary
- Store application id to internal structure
- Store application description to internal structure
- Send message DLT_REGISTER_APPLICATION to DLT daemon
During register of a context of the application, the following things occur:
- Auto-initialize DLT user library, if necessary
- Check, if context was already registered, if not, dynamically increment if required, the context array in step size DLT_USER_CONTEXT_ALLOC_SIZE, typically 500. Then store one entry for the new context to internal context array.
- Send message DLT_REGISTER_CONTEXT to DLT daemon
During unregister of context, the following things occur:
- Delete context from internal structures
- Send message DLT_UNREGISTER_CONTEXT to DLT daemon
During unregister of application, the following things occur:
- Delete application from internal structures
- Send message DLT_UNREGISTER_APPLICATION to DLT daemon
During housekeeper thread within the DLT user library checks for newly received messages from the DLT daemon, and handles them in the following way:
- DLT_USER_MESSAGE_LOG_LEVEL
- Store received log level and trace status for the received context to the context array.
- DLT_USER_MESSAGE_INJECTION
- Check all registered callbacks for this context:
- Compare service id of registered callback with received service id, if
they matches:
- Call registered callback function.
- Compare service id of registered callback with received service id, if
they matches:
- Check all registered callbacks for this context:
If the named pipe/socket of the DLT daemon is full, an overflow flag is set and the message stored in a ring buffer. The next time, a message could be sent to the DLT daemon, an overflow message is sent first, then the contents of the ring buffer. If sending of this message was possible, the overflow flag is reset.
During sending of a log message, the following things occur:
- Auto-initialize DLT user library, if necessary
- Initialize DLT log structure
- Store log level of log message in DLT log structure
- Check if log level is smaller than or equal than the stored log level of the context, under which the message should be sent. If yes continue, else don't send this log message. This is a kind of filtering on the DLT user library side.
- In non-verbose mode, insert message id
- Add values (int, string, raw, …) to DLT log structure:
- Set argument type (only in verbose mode)
- Copy content of argument to message
- Set length information of argument
- Set number of arguments and total length of message
- Create new message with the help of the DLT log structure and handle this message:
- Initialize new message
- Add headers (standard header, extended header, storage header, ...) to this message
- If logging to a file is enabled, write the log message to file
- Print message locally, if requested by environment variable
- Check if connection to DLT daemon was lost and try to reattach to daemon.
- Check for overflow flag and try to send overflow message.
- Try to send message to DLT daemon, and check for return values. This can be:
- data could not be written
- handle not open or pipe/socket error
- other error condition
- no error, all right
- If sending failed, put this message in the ring buffer for later sending.
- Handle these error conditions.
During sending of a network trace message, the following things occur:
- Auto-initialize DLT user library, if necessary
- Check for trace status of network trace message to be sent.
- If Trace status equals on:
- Initialize DLT log structure
- Set trace status in DLT log structure to network trace type of message
- Copy length of network message header to DLT log structure
- Copy data of network message header to DLT log structure
- Copy length of network trace payload to DLT log structure
- Copy data of network trace payload to DLT log structure
- Create new message with the help of the DLT log structure and handle this message
During registration of a callback function for an injection message, then following steps are executed:
- Check if the environment variable DLT_DISABLE_INJECTION_MSG_AT_USER is set.
- If yes:
- libdlt will ignore all data/messages from dlt-daemon, including: loglevel change, custom injection messages ...
- If no (default):
- libdlt will handle all data/messages from dlt-daemon as normal.
- If yes:
- For the specified context, check if service id is already in the table of the
registered callbacks (this table is dynamically growing in steps of one entry).
- If yes:
- Stored service id is set to service id to be registered
- Stored callback function pointer is set to callback function pointer to be registered
- If no:
- Increase the number of callbacks for this context
- Store service id in callback table
- Store function pointer in callback table
- If yes:
On Android, pthread_cancel
is not available in bionic. So current
implementation uses SIGUSR1 and pthread_kill
to terminate housekeeper
thread. Due to this, application which is linked to DLT library should not
define SIGUSR1.
The communication mechanism (IPC) used between DLT daemon and DLT user library are named pipes (FIFOs), UNIX sockets or VSOCK sockets, based on compile time configuration).
During the startup of the DLT daemon, the DLT daemon creates and opens the IPC (FIFOs or UNIX_SOCKET) on the configured path called /tmp/dlt. If a DLT user application using the DLT user library wants to send log messages to the DLT daemon, this IPC is used.
During the startup of the DLT user application using the DLT user library, it creates and opens the same IPC type of DLT daemon. If the DLT daemon wants to send the control messages to the DLT user application, this IPC is used.
- When the FIFOs IPC is used, a named pipe called /tmp/dltpipes/dlt, where is the process id of the DLT user application. Each DLT user application creates its own named pipe to communicate with the DLT daemon.
- When the UNIX_SOCKETS IPC is used, the DLT user application just needs to connect to the opened socket port of DLT daemon. All DLT user applications use the same UNIX_SOCKETS port to communicate with the DLT daemon.
The exchanged messages are described in the chapter 7.1.
In a system with a single process tree where PIDs are unique, the FIFOs IPC shall work well.
In a system with the multiple nested process trees, each process tree maintains an entirely isolated set of processes. Because of that, PIDs are not unique anymore and a DLT User application running in the parent PID namespace might have the same PID as another DLT User application running in the child PID namespace. Since a DLT User application creates its own FIFO based on its PID (getpid()), the DLT Daemon has to deal with multiple applications registering itself with the same PID, which is impossible. To solve this problem, the UNIX_SOCKETS based communication is used. The UNIX_SOCKETS IPC is also supported in the Non-linux platforms (eg: QNX).
If the daemon is built with VSOCK socket support, it can also receive log messages from processes running in virtual machines. The communication mechanism between the daemon and the DLT user library on the host is still FIFOs or UNIX sockets. But the DLT user library for the system running in the virtual machine can be built to use VSOCK sockets for sending log messages to the daemon. See "man vsock" for more information about VSOCK sockets. This is an alternative to using a multinode setup for receiving DLT log messages from processes in a virtualized environment. No passive daemon(s) are required and it works without having a network configured between the guest and the host.
The following table shows, where the certain types of DLT messages are created. The message types are described in more detail in the AUTOSAR Specification for Diagnostic Log and Trace.
Type of message | DLT client | DLT daemon | DLT library |
---|---|---|---|
Control request | X | X | |
Control response | X | ||
Log message | X | ||
Trace message | X |
This table shows, that:
- DLT messages of message type control request are created in the DLT client, and then sent to the DLT daemon.
- DLT messages of message type control response (to a control request) are created in the DLT daemon, and then sent to the DLT client.
- DLT messages of type log and of type trace are created in the DLT user library, then passed (via the named pipe of the DLT daemon) to the DLT daemon, which forwards them to the connected DLT clients.
The following figure shows the overall flow of messages.
There are several user messages (each has its own message identifier DLT_USER_MESSAGE_*) which will are exchanged between DLT daemon and DLT user library, and will be described in the following sub-chapters.
From DLT user library to DLT daemon:
- DLT_USER_MESSAGE_REGISTER_APPLICATION
- DLT_USER_MESSAGE_UNREGISTER_APPLICATION
- DLT_USER_MESSAGE_REGISTER_CONTEXT
- DLT_USER_MESSAGE_UNREGISTER_CONTEXT
- DLT_USER_MESSAGE_LOG
- DLT_USER_MESSAGE_OVERFLOW
- DLT_USER_MESSAGE_APP_LL_TS
From DLT daemon to DLT user library:
- DLT_USER_MESSAGE_LOG_LEVEL
- DLT_USER_MESSAGE_INJECTION
Each of the following messages has a message header with the following information:
- Pattern: DUH0x01
- Message identifier
This message is sent by the DLT user library once per application to register the application to the DLT daemon. It contains the following information:
- The application id of the application to be registered
- The process id of the process using the DLT user library. This information is required, if the DLT daemon wants to send something to the DLT user library.
- The length of the following description.
- The application description.
This message is sent by the DLT user library once per application to unregister the application from the DLT daemon. It contains the following information:
- The application id of the application to be unregistered
- The process id of the process using the DLT user library.
This message is sent by the DLT user library once for each context which should be registered to the DLT daemon. It contains the following information:
- The application id of the application to be registered
- The context id of the application to be registered
- Each created context is stored with its associated information in a dynamically growing array in the DLT user library. The index in this array is sent.
- The process id of the process using the DLT user library. This information is required, if the DLT daemon wants to send something to the DLT user library.
- The initial log level of the context
- The initial trace status of the context
- The length of the following description.
- The context description.
This message is sent by the DLT user library once for each context which should be unregistered from the DLT daemon. It contains the following information:
- The application id of the application to be unregistered
- The context id of the application to be unregistered
- The process id of the process using the DLT user library.
This is the standard log/trace message send by the DLT user library to the DLT daemon. It contains the following information:
- A standard DLT message header as specified in the AUTOSAR R4.0 DLT standard.
- An extended DLT message header as specified in the AUTOSAR R4.0 DLT standard.
- The payload of the DLT message
This message is sent from the DLT user library to the DLT daemon, if there was an overflow during writing to the DLT daemon named pipe or socket IPC. It contains no further information.
This message is sent from the DLT user library to the DLT daemon, when the overall Log Level and Trace Status for the whole DLT application should be set from within the DLT application. It contains the following information:
- The application id
- The Log Level to be set for the whole application
- The Trace Status to be set for the whole application
If the log level or trace status is changed, or initialized, this message is sent from the DLT daemon to the DLT user library to store the current log level and trace status for filtering in the DLT user library.
It contains the following information:
- The new set log level
- New set trace status.
- Each created context is stored with its associated information in a dynamically growing array in the DLT user library. The index in this array is sent.
This message is sent from the DLT daemon to the DLT user library, if an injection message was received by the DLT daemon from a DLT client. Via the context, the appropriate application and its named pipe can be identified. The injection message is then passed to this named pipe.
It contains the following information:
- Each created context is stored with its associated information in a dynamically growing array in the DLT user library. The index in this array is sent.
- Service ID of the injection message
- Length of the following injection message
- The contents of the injection message
Following things occur for the received DLT user messages:
- DLT_USER_MESSAGE_REGISTER_APPLICATION
- Add all information about application to a local application array (dynamically growing in a predefined step size (DLT_DAEMON_APPL_ALLOC_SIZE, typically 500)). The array is always being kept sorted (via qsort()). Finding entries in this array is done by a binary search (via bsearch()).
- Open named pipe (or Unix socket IPC if specified) to DLT application using the DLT user library with process id . The name of the pipe (or Unix socket IPC) is /tmp/dlt
- DLT_USER_MESSAGE_UNREGISTER_APPLICATION
- Remove all information about this application from local application array, and remove all information of all contexts belonging to this application from local context array. The arrays are always being kept sorted, without empty entries in between. Dynamically shrinking of this array is NOT implemented.
- DLT_USER_MESSAGE_REGISTER_CONTEXT
- Add all information about context to local context array (dynamically growing in a predefined step size (DLT_DAEMON_CONTEXT_ALLOC_SIZE, typically 1000)). The array is always being kept sorted (via qsort()). Finding entries in this array is done by a binary search (via bsearch()).
- Send log level and trace status to DLT user library for this context. Therefore, the DLT_USER_MESSAGE_LOG_LEVEL is used.
- DLT_USER_MESSAGE_UNREGISTER_CONTEXT
- Remove all information about this context from local context array. The array is always being kept sorted, without empty entries in between. Dynamically shrinking of this array is NOT implemented.
- DLT_USER_MESSAGE_OVERFLOW
- Set internal flag for overflow.
- Create and send DLT control message response overflow to all connected TCP connections, and optionally to serial device, if connected. If the message was sent, reset the internal flag for overflow.
- DLT_USER_MESSAGE_LOG
- Overwrite ECU id, if requested
- Set storage header
- Check for filters of message, if no filter is set, or filter is matching:
- Possibly display message as specified in the options.
- Whole message display in hex
- Whole message display in ASCII
- Show message headers only
- Store message to output file
- Set serial header in front of message, if specified.
- Create and try to send DLT message to all DLT clients connected via TCP connection, and optionally via serial device.
- If the message could not be sent, store the message to a local ring buffer. The ring buffer internally uses a variable length for the buffered elements, and therefore uses the memory available for the buffer the best way possible. If the buffer is full, the oldest messages are silently discarded, until there is enough space for the message to be stored in the ring buffer.
- Possibly display message as specified in the options.
- DLT_USER_MESSAGE_APP_LL_TS
- For all contexts belonging to the specified application:
- Set specified log level
- Set specified trace status
- Send specified log level and trace status to DLT client library
- For all contexts belonging to the specified application:
If the DLT daemon receives a control message request from a DLT client, it handles it in the following way. First the service id of the message is detected, and if it is no injection message, the following things occur:
- DLT_SERVICE_ID_SET_LOG_LEVEL
- Check if received log level is other then already set log level. If yes:
- Store new log level to local array.
- Send new log level to DLT user library for this context. Therefore, the DLT_USER_MESSAGE_LOG_LEVEL is used. Note that the DLT would take the log level of Logstorage as the highest priority. So even if log level is changed from client, it will be overwritten by Logstorage definition.
- Send DLT control response to DLT client, with status of operation.
- Check if received log level is other then already set log level. If yes:
- DLT_SERVICE_ID_SET_TRACE_STATUS
- Check if received trace status is other then already set trace status. If yes:
- Store new trace status to local array.
- Send new trace status to DLT user library for this context. Therefore, the DLT_USER_MESSAGE_LOG_LEVEL is used.
- Send DLT control response to DLT client, with status of operation.
- Check if received trace status is other then already set trace status. If yes:
- DLT_SERVICE_ID_GET_LOG_INFO
- Create answer message for DLT control response. All kinds of requests are supported:
- Application/Context:
- All applications and all contexts
- One application and all of its contexts
- One application with one context.
- Request type (as specified in the AUTOSAR DLT Standard, 1 and 2 are not specified):
- 3: Application ID, Context ID
- 4: Application ID, Context ID, Log Level
- 5: Application ID, Context ID, Trace Status
- 6: Application ID, Context ID, Log Level, Trace Status
- 7: Application ID, Context ID, Log Level, Trace Status, Description
- Application/Context:
- Answer is of the corresponding type 3-7, or of type 8 (no matching context ids found), or of type 2 (error).
- Send DLT control response answer to DLT client.
- Create answer message for DLT control response. All kinds of requests are supported:
- DLT_SERVICE_ID_GET_DEFAULT_LOG_LEVEL
- Send DLT control response with default log level to DLT client.
- DLT_SERVICE_ID_STORE_CONFIG
- Store configuration files with currently registered application ids and context ids as well as currently set log levels and trace statuses to configuration files. If these files exists, they will be read in when the DLT daemon is started next time.
- Send DLT control response to DLT client.
- DLT_SERVICE_ID_RESET_TO_FACTORY_DEFAULT
- Delete the stored configuration files, if they exists.
- Set default log level and trace status to initial values, and inform all applications using the default log level and trace status about the new defaults.
- Send DLT control response to DLT client.
- DLT_SERVICE_ID_SET_COM_INTERFACE_STATUS
- Send DLT control response "Not supported" to DLT client
- DLT_SERVICE_ID_SET_COM_INTERFACE_MAX_BANDWIDTH
- Send DLT control response "Not supported" to DLT client
- DLT_SERVICE_ID_SET_VERBOSE_MODE
- Send DLT control response "Not supported" to DLT client
- DLT_SERVICE_ID_SET_MESSAGE_FILTERING
- Send DLT control response "Not supported" to DLT client
- DLT_SERVICE_ID_SET_TIMING_PACKETS
- Set flag to start/stop sending of timing messages in thread.
- Send DLT control response to DLT client.
- DLT_SERVICE_ID_GET_LOCAL_TIME
- Send DLT control response with local time to DLT client.
- DLT_SERVICE_ID_USE_ECU_ID
- Send DLT control response "Not supported" to DLT client
- DLT_SERVICE_ID_USE_SESSION_ID
- Send DLT control response "Not supported" to DLT client
- DLT_SERVICE_ID_USE_TIMESTAMP
- Send DLT control response "Not supported" to DLT client
- DLT_SERVICE_ID_USE_EXTENDED_HEADER
- Send DLT control response "Not supported" to DLT client
- DLT_SERVICE_ID_SET_DEFAULT_LOG_LEVEL
- Check if received default log level is other than already stored default
log level. If yes:
- Store new default log level to internal structure.
- Send a DLT_USER_MESSAGE_LOG_LEVEL to all DLT clients containing the new default log level, which uses a context which is set to default log level.
- Send DLT control response to DLT client, with status of operation.
- Check if received default log level is other than already stored default
log level. If yes:
- DLT_SERVICE_ID_SET_DEFAULT_TRACE_STATUS
- Check if received default trace status is other than already stored default
trace status. If yes:
- Store new default trace status to internal structure.
- Send a DLT_USER_MESSAGE_LOG_LEVEL to all DLT clients containing the new default trace status, which uses a context which is set to default trace status.
- Send DLT control response to DLT client, with status of operation.
- Check if received default trace status is other than already stored default
trace status. If yes:
- DLT_SERVICE_ID_GET_SOFTWARE_VERSION
- Send DLT control response containing a software version string to DLT Client.
- The software version string contains:
- the package version of the package dltv2, e.g. "2.0.0"
- the package status of the package dltv2, e.g. "alpha, beta, final"
- the overall subversion revision number for the package dltv2, e.g. "2300"
- DLT_SERVICE_ID_MESSAGE_BUFFER_OVERFLOW
- Try to send DLT control response containing the status of the internal flag for overflow. If the message could be send, reset the internal flag for overflow.
For handling of the injection message, the following steps occur:
- Create new user message DLT_USER_MESSAGE_INJECTION from received DLT control request message, and send this user message to the DLT application using the DLT user library.
Here is a description, how the whole project is structured and which files exists. The following table shows a top level view of the available Git repositories:
Directory | Description |
---|---|
dlt | DLT Daemon and Library implementation; command line utilities, examples and test programs |
dlt_viewer | DLT Client GUI (DLT Viewer): QT based implementation |
As this document has the focus on the DLT Daemon and the DLT user library, only the "dlt" directory is introduced in more detail:
Directory | Description |
---|---|
doc | Documentation |
include | Include files, installed on target |
src | Source Code |
src/shared | Shared source code (between DLT daemon and DLT user library) |
src/adaptor | Adaptors to DLT daemon:dlt-adaptor-stdin (for connection over stdin) dlt-adaptor-udp (for connection over UDP) |
src/console | Console utilities: logstorage, dlt-control, dlt-convert, dlt-passive-node-ctrl, dlt-receive, and dlt-sortbytimestamp |
src/daemon | DLT Daemon |
src/examples | Examples for usage of the DLT user library:dlt-example-user (Macro IF) anddlt-example-user-func (Function IF) andwintestclient (MS Windows based test client) |
src/lib | DLT library functions |
src/tests | Test programs:dlt-test-client and dlt-test-user for automatic tests, dlt-test-stress for stress tests, dlt-test-internal for internal tests |
testscripts | Several supporting scripts |
The DLT daemon implementation uses the following files, besides DLT functions from files from the shared directory:
File | Description |
---|---|
dlt-daemon.c | DLT daemon implementation |
dlt-daemon.h | Header file for dlt-daemon.c |
dlt-daemon_cfg.h | Compile time configuration for DLT daemon, Part1 |
dlt_daemon_common.c | Supporting functions for a DLT daemon implementation |
dlt_daemon_common.h | Header file for dlt_daemon_common.c |
dlt_daemon_common_cfg.h | Compile time configuration for DLT daemon, Part2 |
The DLT user library contains the following files:
File | Description |
---|---|
dlt_user.c | Implementation of functions, available via the DLT user library interface. |
dlt_user_cfg.h | Compile time configuration for dlt_user.c |
dlt_client.c | Functions required for DLT Client implementations |
dlt_client_cfg.h | Compile time configuration for dlt_client.c |
The shared directory contains the following files:
File | Description |
---|---|
dlt_common.c | Common helper functions, such as: Print functions for DLT messages, Functions for handling DLT Ids, Filter functions, DLT message handling functions, Functions for handling DLT files, DLT receiver functions, Log handling, Ringbuffer functions, Setting and checking of storage header |
dlt_common_cfg.h | Compile time configuration for dlt_common.c |
dlt_user_shared.c | Shared functions, required by the DLT daemon and the DLT user library, such as: Setting and checking the user header, Sending DLT messages over named pipes (FIFOs) or UNIX/VSOCK sockets |
dlt_user_shared.h | Header file for dlt_user_shared.c |
dlt_user_shared_cfg.h | Compile time configuration for dlt_user_shared.c |
The public available include directory contains the following header files:
File | Description |
---|---|
dlt.h | Overall include file, includes dlt_common.h and dlt_user.h |
dlt_common.h | Include file for dlt_common.c |
dlt_user.h | Include file for dlt_user.c, contains the "User API" functions |
dlt_user_macros.h | Include file for dlt_user.c, contains the "User API" macros |
dlt_client.h | Include file for dlt_client.c |
dlt_protocol.h | DLT protocol specfic definitions and macros |
dlt_types.h | Definition of types, must be adapted to the target architecture and compiler toolchain. |
The following important structures are used in the DLT Daemon and DLT User Library:
Structure | Description |
---|---|
DltDaemonFlags | Flags and values, set over command line during start-up of DLT daemon |
DltDaemonLocal | Global variables of the DLT Daemon, grouped together in a struct |
DltDaemon | Parameters of the DLT daemon |
DltDaemonApplication | Parameters of an application, used within DLT daemon |
DltDaemonContext | Parameters of a context, used within DLT daemon |
DltUserControlMsg... | User control messages |
DltContext | Each context is represented with this structure |
DltContextData | Temorary used structure for constructing a log message |
DltUserInjectionCallback | One entry in dynamic callback table for callback function |
dlt_ll_ts_type | Table managing all contexts (and loglevels and trce statuses) within DLT user library, contains dynamicly growing array with "DltUserInjectionCallback" entries |
DltUser | Parameters for DLT user library application, contains dynamically growing array with "dlt_ll_ts_type" entries |
DltStorageHeader | Storage header |
DltStandardHeader | Structure for standard header of a DLT message |
DltStandardHeaderExtra | Structure for standard header extra fields of a DLT message |
DltExtendedHeader | Structure for extended header of a DLT message |
DltService... | Structures for control messages (requests and responses) |
DltFilter | Structure to store filter parameters |
DltMessage | Structure to organize a DLT message, includes pointers to DltStorageHeader, DltStandardHeader, DltStandardHeaderExtra, and DltExtendedHeader, data of message is stored within a private buffer within this structure |
DltFile | Structure to organize access to a DLT File |
DltReceiver | Structure to organize the receiving of data |
DltRingbuffer | Structure to organize a ring buffer |
- The following preconditions were given prior to implementation:
- C-only implementation for the DLT daemon and the DLT user library
- Implementation of common functions, which can be used in a command line utility as well as in a Graphical UI
- Implementation of C++ like classes in C, see dlt_common.c and dlt_common.h
- The current implementation of the DLT daemon and DLT user library is tested with gcc under Ubuntu 16.04 on Intel HW; Android and QNX on IMX, RCAR, Qualcomm ARM target boards.
- It is assumed that packed structs are always stored in memory in the order specified within the packed struct.
- The implementation is multithread safe.
- Initialize DLT application and contexts, then forking and using the forked process, will lead to unclear behavior (or in the worst case to a crash), as the forked process uses another process id as the parent process. Instead, initialize the DLT application and contexts in the forked process.
- Calling of DLT logging and tracing functions within a callback function for injections are not supported, and will lead to an unclear behavior.