Add watchdog event monitor

This change add a watchdog event monitor
when watchdog enable/disable, it will add
a IPMI SEL record to journal.

teseted:
    - Successfully build and verified in ipmitool sel elist

Signed-off-by: Charles Hsu <Charles.Hsu@quantatw.com>
Change-Id: I3a7e408fdeced50d4a81acf95d7e35da3f0a6410
diff --git a/CMakeLists.txt b/CMakeLists.txt
index a0f428c..779093b 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -20,6 +20,13 @@
 )
 
 option (
+    SEL_LOGGER_MONITOR_WATCHDOG_EVENTS
+    "Enable SEL Logger to monitor and automatically
+     log SEL records for watchdog sensor events"
+    OFF
+)
+
+option (
     SEL_LOGGER_SEND_TO_LOGGING_SERVICE
     "Make SEL Logger to send sel to logging service instead of journal"
     OFF
@@ -29,6 +36,7 @@
     sel-logger PRIVATE
     $<$<BOOL:${SEL_LOGGER_MONITOR_THRESHOLD_EVENTS}>: -DSEL_LOGGER_MONITOR_THRESHOLD_EVENTS>
     $<$<BOOL:${REDFISH_LOG_MONITOR_PULSE_EVENTS}>: -DREDFISH_LOG_MONITOR_PULSE_EVENTS>
+    $<$<BOOL:${SEL_LOGGER_MONITOR_WATCHDOG_EVENTS}>: -DSEL_LOGGER_MONITOR_WATCHDOG_EVENTS>
     $<$<BOOL:${SEL_LOGGER_SEND_TO_LOGGING_SERVICE}>: -DSEL_LOGGER_SEND_TO_LOGGING_SERVICE>
 )
 
diff --git a/include/watchdog_event_monitor.hpp b/include/watchdog_event_monitor.hpp
new file mode 100644
index 0000000..8b31118
--- /dev/null
+++ b/include/watchdog_event_monitor.hpp
@@ -0,0 +1,278 @@
+/*
+// Copyright (c) 2018 Intel Corporation
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//      http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+*/
+
+#pragma once
+#include <sel_logger.hpp>
+#include <sensorutils.hpp>
+
+#include <iostream>
+#include <string_view>
+#include <variant>
+
+enum class watchdogEventOffsets : uint8_t
+{
+    noAction = 0x00,
+    hardReset = 0x01,
+    powerDown = 0x02,
+    powerCycle = 0x03,
+};
+
+enum class watchdogTimerUseOffsets : uint8_t
+{
+    reserved = 0x00,
+    BIOSFRB2 = 0x01,
+    BIOSPOST = 0x02,
+    OSLoad = 0x03,
+    SMSOS = 0x04,
+    OEM = 0x05,
+    unspecified = 0x0f,
+};
+
+enum class watchdogInterruptTypeOffsets : uint8_t
+{
+    none = 0x00,
+    SMI = 0x01,
+    NMI = 0x02,
+    messageInterrupt = 0x03,
+    unspecified = 0x0f,
+};
+
+static constexpr const uint8_t wdtNologBit = (1 << 7);
+static constexpr int interruptTypeBits = 4;
+
+inline static sdbusplus::bus::match::match
+    startWatchdogEventMonitor(std::shared_ptr<sdbusplus::asio::connection> conn)
+{
+    auto watchdogEventMatcherCallback = [conn](
+                                            sdbusplus::message::message& msg) {
+        // This static set of std::pair<path, event> tracks asserted events to
+        // avoid duplicate logs or deasserts logged without an assert
+        static boost::container::flat_set<std::pair<std::string, std::string>>
+            assertedEvents;
+
+        // SEL event data is three bytes where 0xFF means unspecified
+        std::vector<uint8_t> eventData(selEvtDataMaxSize, 0xFF);
+
+        // Get the event type and assertion details from the message
+        std::string watchdogInterface;
+        boost::container::flat_map<std::string, std::variant<bool>>
+            propertiesChanged;
+
+        msg.read(watchdogInterface, propertiesChanged);
+
+        if (propertiesChanged.begin()->first != "Enabled")
+        {
+            return;
+        }
+
+        bool* pval = std::get_if<bool>(&propertiesChanged.begin()->second);
+
+        if (!pval)
+        {
+            std::cerr << "watchdog event direction has invalid type\n";
+            return;
+        }
+        bool assert = *pval;
+
+        sdbusplus::message::message getWatchdogStatus =
+            conn->new_method_call(msg.get_sender(), msg.get_path(),
+                                  "org.freedesktop.DBus.Properties", "GetAll");
+        getWatchdogStatus.append("xyz.openbmc_project.State.Watchdog");
+        boost::container::flat_map<std::string,
+                                   std::variant<std::string, uint64_t, bool>>
+            watchdogStatus;
+
+        try
+        {
+            sdbusplus::message::message getWatchdogStatusResp =
+                conn->call(getWatchdogStatus);
+            getWatchdogStatusResp.read(watchdogStatus);
+        }
+        catch (sdbusplus::exception_t&)
+        {
+            std::cerr << "error getting watchdog status from " << msg.get_path()
+                      << "\n";
+            return;
+        }
+
+        auto getExpireAction = watchdogStatus.find("ExpireAction");
+        std::string_view expireAction;
+        if (getExpireAction != watchdogStatus.end())
+        {
+            expireAction = std::get<std::string>(getExpireAction->second);
+            expireAction.remove_prefix(std::min(
+                expireAction.find_last_of(".") + 1, expireAction.size()));
+        }
+        if (expireAction == "HardReset")
+        {
+            eventData[0] =
+                static_cast<uint8_t>(watchdogEventOffsets::hardReset);
+        }
+        else if (expireAction == "PowerOff")
+        {
+            eventData[0] =
+                static_cast<uint8_t>(watchdogEventOffsets::powerDown);
+        }
+        else if (expireAction == "PowerCycle")
+        {
+            eventData[0] =
+                static_cast<uint8_t>(watchdogEventOffsets::powerCycle);
+        }
+        else if (expireAction == "None")
+        {
+            eventData[0] = static_cast<uint8_t>(watchdogEventOffsets::noAction);
+        }
+
+        auto getPreTimeoutInterrupt =
+            watchdogStatus.find("PreTimeoutInterrupt");
+        std::string_view preTimeoutInterrupt;
+        if (getPreTimeoutInterrupt != watchdogStatus.end())
+        {
+            preTimeoutInterrupt =
+                std::get<std::string>(getPreTimeoutInterrupt->second);
+            preTimeoutInterrupt.remove_prefix(
+                std::min(preTimeoutInterrupt.find_last_of(".") + 1,
+                         preTimeoutInterrupt.size()));
+        }
+        if (preTimeoutInterrupt == "None")
+        {
+            eventData[1] &=
+                (static_cast<uint8_t>(watchdogInterruptTypeOffsets::none)
+                 << interruptTypeBits);
+        }
+        else if (preTimeoutInterrupt == "SMI")
+        {
+            eventData[1] &=
+                (static_cast<uint8_t>(watchdogInterruptTypeOffsets::SMI)
+                 << interruptTypeBits);
+        }
+        else if (preTimeoutInterrupt == "NMI")
+        {
+            eventData[1] &=
+                (static_cast<uint8_t>(watchdogInterruptTypeOffsets::NMI)
+                 << interruptTypeBits);
+        }
+        else if (preTimeoutInterrupt == "MI")
+        {
+            eventData[1] &= (static_cast<uint8_t>(
+                                 watchdogInterruptTypeOffsets::messageInterrupt)
+                             << interruptTypeBits);
+        }
+
+        auto getCurrentTimerUse = watchdogStatus.find("CurrentTimerUse");
+        std::string_view currentTimerUse;
+        if (getCurrentTimerUse != watchdogStatus.end())
+        {
+            currentTimerUse = std::get<std::string>(getCurrentTimerUse->second);
+            currentTimerUse.remove_prefix(std::min(
+                currentTimerUse.find_last_of(".") + 1, currentTimerUse.size()));
+        }
+        if (currentTimerUse == "BIOSFRB2")
+        {
+            eventData[1] |=
+                static_cast<uint8_t>(watchdogTimerUseOffsets::BIOSFRB2);
+        }
+        else if (currentTimerUse == "BIOSPOST")
+        {
+            eventData[1] |=
+                static_cast<uint8_t>(watchdogTimerUseOffsets::BIOSPOST);
+        }
+        else if (currentTimerUse == "OSLoad")
+        {
+            eventData[1] |=
+                static_cast<uint8_t>(watchdogTimerUseOffsets::OSLoad);
+        }
+        else if (currentTimerUse == "SMSOS")
+        {
+            eventData[1] |=
+                static_cast<uint8_t>(watchdogTimerUseOffsets::SMSOS);
+        }
+        else if (currentTimerUse == "OEM")
+        {
+            eventData[1] |= static_cast<uint8_t>(watchdogTimerUseOffsets::OEM);
+        }
+        else
+        {
+            eventData[1] |=
+                static_cast<uint8_t>(watchdogTimerUseOffsets::unspecified);
+        }
+
+        auto getWatchdogInterval = watchdogStatus.find("Interval");
+        uint64_t watchdogInterval;
+        if (getWatchdogInterval != watchdogStatus.end())
+        {
+            watchdogInterval = std::get<uint64_t>(getWatchdogInterval->second);
+        }
+
+        // get watchdog status porperties
+        static bool wdt_nolog;
+        sdbusplus::bus::bus bus = sdbusplus::bus::new_default();
+        uint8_t netFn = 0x06;
+        uint8_t lun = 0x00;
+        uint8_t cmd = 0x25;
+        std::vector<uint8_t> commandData;
+        std::map<std::string, std::variant<int>> options;
+
+        auto ipmiCall = bus.new_method_call(
+            "xyz.openbmc_project.Ipmi.Host", "/xyz/openbmc_project/Ipmi",
+            "xyz.openbmc_project.Ipmi.Server", "execute");
+        ipmiCall.append(netFn, lun, cmd, commandData, options);
+        std::tuple<uint8_t, uint8_t, uint8_t, uint8_t, std::vector<uint8_t>>
+            rsp;
+        auto ipmiReply = bus.call(ipmiCall);
+        ipmiReply.read(rsp);
+        auto& [rnetFn, rlun, rcmd, cc, responseData] = rsp;
+
+        std::string direction;
+        std::string eventMessageArgs;
+        if (assert)
+        {
+            direction = " enable ";
+            eventMessageArgs = "Enabled";
+            wdt_nolog = responseData[0] & wdtNologBit;
+        }
+        else
+        {
+            direction = " disable ";
+            eventMessageArgs = "Disabled";
+        }
+
+        // Set Watchdog Timer byte1[7]-1b=don't log
+        if (!wdt_nolog)
+        {
+            // Construct a human-readable message of this event for the log
+            std::string journalMsg(
+                std::string(currentTimerUse) + std::string(direction) +
+                "watchdog countdown " +
+                std::to_string(watchdogInterval / 1000) + " seconds " +
+                std::string(expireAction) + " action");
+
+            std::string redfishMessageID = "OpenBMC.0.1.IPMIWatchdog";
+
+            selAddSystemRecord(
+                journalMsg, std::string(msg.get_path()), eventData, assert,
+                selBMCGenID, "REDFISH_MESSAGE_ID=%s", redfishMessageID.c_str(),
+                "REDFISH_MESSAGE_ARGS=%s", eventMessageArgs.c_str(), NULL);
+        }
+    };
+    sdbusplus::bus::match::match watchdogEventMatcher(
+        static_cast<sdbusplus::bus::bus&>(*conn),
+        "type='signal',interface='org.freedesktop.DBus.Properties',member='"
+        "PropertiesChanged',arg0namespace='xyz.openbmc_project.State."
+        "Watchdog'",
+        std::move(watchdogEventMatcherCallback));
+    return watchdogEventMatcher;
+}
diff --git a/src/sel_logger.cpp b/src/sel_logger.cpp
index fda9587..99527b1 100644
--- a/src/sel_logger.cpp
+++ b/src/sel_logger.cpp
@@ -23,6 +23,7 @@
 #include <sdbusplus/asio/object_server.hpp>
 #include <sel_logger.hpp>
 #include <threshold_event_monitor.hpp>
+#include <watchdog_event_monitor.hpp>
 
 #include <filesystem>
 #include <fstream>
@@ -243,6 +244,10 @@
         startPulseEventMonitor(conn);
 #endif
 
+#ifdef SEL_LOGGER_MONITOR_WATCHDOG_EVENTS
+    sdbusplus::bus::match::match watchdogEventMonitor =
+        startWatchdogEventMonitor(conn);
+#endif
     io.run();
 
     return 0;