summaryrefslogtreecommitdiff
path: root/src/system/dlt-system-watchdog.c
blob: a2b01de1936e7d53ed3dafc68ce146f19bc3974d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
/*
 * SPDX license identifier: MPL-2.0
 *
 * Copyright (C) 2011-2015, BMW AG
 *
 * This file is part of GENIVI 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.genivi.org/.
 */

/*!
 * \author
 * Lassi Marttala <lassi.lm.marttala@partner.bmw.de>
 * Alexander Wenzel <alexander.aw.wenzel@bmw.de>
 * Markus Klein <Markus.Klein@esk.fraunhofer.de>
 * Mikko Rapeli <mikko.rapeli@bmw.de>
 *
 * \copyright Copyright © 2011-2015 BMW AG. \n
 * License MPL-2.0: Mozilla Public License version 2.0 http://mozilla.org/MPL/2.0/.
 *
 * \file dlt-system-watchdog.c
 */

#if defined(DLT_SYSTEMD_WATCHDOG_ENABLE)
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/timerfd.h>
#include "dlt.h"
#include "dlt-system.h"
#include <poll.h>
#include <systemd/sd-daemon.h>

DLT_DECLARE_CONTEXT(watchdogContext)
DLT_IMPORT_CONTEXT(dltsystem)

int calculate_period(struct itimerspec *itval)
{
    unsigned int ns;
    unsigned int sec;
    char str[512];
    char *watchdogUSec;
    unsigned int watchdogTimeoutSeconds;
    unsigned int notifiyPeriodNSec;

    watchdogUSec = getenv("WATCHDOG_USEC");

    if (watchdogUSec == NULL) {
        DLT_LOG(watchdogContext, DLT_LOG_ERROR, DLT_STRING("systemd watchdog timeout (WATCHDOG_USEC) is null\n"));
        return -1;
    }
    
    DLT_LOG(watchdogContext, DLT_LOG_DEBUG, DLT_STRING("watchdogusec: "), DLT_STRING(watchdogUSec));
    watchdogTimeoutSeconds = atoi(watchdogUSec);

    if (watchdogTimeoutSeconds <= 0) {
        snprintf(str, 512, "systemd watchdog timeout incorrect: %u\n", watchdogTimeoutSeconds);
        DLT_LOG(watchdogContext, DLT_LOG_ERROR, DLT_STRING(str));
        return -1;
    }

    /* Calculate half of WATCHDOG_USEC in ns for timer tick */
    notifiyPeriodNSec = watchdogTimeoutSeconds / 2;

    snprintf(str,
            512,
            "systemd watchdog timeout: %u nsec - timer will be initialized: %u nsec\n",
            watchdogTimeoutSeconds,
            notifiyPeriodNSec);
    DLT_LOG(watchdogContext, DLT_LOG_DEBUG, DLT_STRING(str));

    /* Make the timer periodic */
    sec = notifiyPeriodNSec / 1000000;
    ns = (notifiyPeriodNSec - (sec * 1000000)) * 1000;
    itval->it_interval.tv_sec = sec;
    itval->it_interval.tv_nsec = ns;
    itval->it_value.tv_sec = sec;
    itval->it_value.tv_nsec = ns;

    return 0;
}

int register_watchdog_fd(struct pollfd *pollfd, int fdcnt)
{
    DLT_REGISTER_CONTEXT(watchdogContext, "DOG", "dlt system watchdog context.");
    struct itimerspec timerValue;
    if (calculate_period(&timerValue) != 0)
        return -1;

    int timerfd = timerfd_create(CLOCK_MONOTONIC, 0);
    if (timerfd < 0) {
        DLT_LOG(dltsystem, DLT_LOG_ERROR, DLT_STRING("Failed to create timer fd\n"));
        return -1;
    }
    pollfd[fdcnt].fd = timerfd;
    pollfd[fdcnt].events = POLLIN;

    if (timerfd_settime(timerfd, 0, &timerValue, NULL) < 0) {
        DLT_LOG(dltsystem, DLT_LOG_ERROR, DLT_STRING("Could not start timer\n"));
        return -1;
    }
    return 0;
}

void watchdog_fd_handler(int fd)
{
    long int timersElapsed = 0;
    int r = read(fd, &timersElapsed, 8);    // only needed to reset fd event
    if(r < 0)
        DLT_LOG(watchdogContext, DLT_LOG_ERROR, DLT_STRING("Could not reset systemd watchdog. Exit with: "), 
            DLT_STRING(strerror(r)));

    if (sd_notify(0, "WATCHDOG=1") < 0)
        DLT_LOG(watchdogContext, DLT_LOG_ERROR, DLT_STRING("Could not reset systemd watchdog\n"));

    DLT_LOG(watchdogContext, DLT_LOG_DEBUG, DLT_STRING("systemd watchdog waited periodic\n"));
}
#endif