blob: 1b46bdf0cd0a42a806d1c9f1d91c1e64a2ac93bd [file] [log] [blame]
/**
* Copyright © 2020 IBM 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.
*/
#include "error_reporter.hpp"
#include "get_power_state.hpp"
#include "logging.hpp"
#include "psensor.hpp"
#include "utility.hpp"
#include <fmt/format.h>
#include <unistd.h>
#include <phosphor-logging/log.hpp>
#include <xyz/openbmc_project/Logging/Create/server.hpp>
#include <xyz/openbmc_project/Logging/Entry/server.hpp>
namespace phosphor::fan::presence
{
using json = nlohmann::json;
using namespace phosphor::logging;
using namespace sdbusplus::bus::match;
using namespace std::literals::string_literals;
using namespace std::chrono;
namespace fs = std::filesystem;
const auto itemIface = "xyz.openbmc_project.Inventory.Item"s;
const auto invPrefix = "/xyz/openbmc_project/inventory"s;
const auto loggingPath = "/xyz/openbmc_project/logging";
const auto loggingCreateIface = "xyz.openbmc_project.Logging.Create";
ErrorReporter::ErrorReporter(
sdbusplus::bus_t& bus,
const std::vector<
std::tuple<Fan, std::vector<std::unique_ptr<PresenceSensor>>>>& fans) :
_bus(bus),
_event(sdeventplus::Event::get_default()),
_powerState(getPowerStateObject())
{
_powerState->addCallback("errorReporter",
std::bind(&ErrorReporter::powerStateChanged, this,
std::placeholders::_1));
for (const auto& fan : fans)
{
const auto& fanData = std::get<0>(fan);
// Only deal with fans that have an error time defined.
if (std::get<std::optional<size_t>>(fanData))
{
auto path = invPrefix + std::get<1>(fanData);
// Register for fan presence changes, get their initial states,
// and create the fan missing timers.
_matches.emplace_back(
_bus, rules::propertiesChanged(path, itemIface),
std::bind(std::mem_fn(&ErrorReporter::presenceChanged), this,
std::placeholders::_1));
_fanStates.emplace(path, getPresence(fanData));
auto timer = std::make_unique<
sdeventplus::utility::Timer<sdeventplus::ClockId::Monotonic>>(
_event,
std::bind(std::mem_fn(&ErrorReporter::fanMissingTimerExpired),
this, path));
seconds errorTime{std::get<std::optional<size_t>>(fanData).value()};
_fanMissingTimers.emplace(
path, std::make_tuple(std::move(timer), std::move(errorTime)));
}
}
// If power is already on, check for currently missing fans.
if (_powerState->isPowerOn())
{
powerStateChanged(true);
}
}
void ErrorReporter::presenceChanged(sdbusplus::message_t& msg)
{
bool present;
auto fanPath = msg.get_path();
std::string interface;
std::map<std::string, std::variant<bool>> properties;
msg.read(interface, properties);
auto presentProp = properties.find("Present");
if (presentProp != properties.end())
{
present = std::get<bool>(presentProp->second);
if (_fanStates[fanPath] != present)
{
getLogger().log(fmt::format("Fan {} presence state change to {}",
fanPath, present));
_fanStates[fanPath] = present;
checkFan(fanPath);
}
}
}
void ErrorReporter::checkFan(const std::string& fanPath)
{
auto& timer = std::get<0>(_fanMissingTimers[fanPath]);
if (!_fanStates[fanPath])
{
// Fan is missing. If power is on, start the timer.
// If power is off, stop a running timer.
if (_powerState->isPowerOn())
{
timer->restartOnce(std::get<seconds>(_fanMissingTimers[fanPath]));
}
else if (timer->isEnabled())
{
timer->setEnabled(false);
}
}
else
{
// Fan is present. Stop a running timer.
if (timer->isEnabled())
{
timer->setEnabled(false);
}
}
}
void ErrorReporter::fanMissingTimerExpired(const std::string& fanPath)
{
getLogger().log(
fmt::format("Creating event log for missing fan {}", fanPath),
Logger::error);
std::map<std::string, std::string> additionalData;
additionalData.emplace("_PID", std::to_string(getpid()));
additionalData.emplace("CALLOUT_INVENTORY_PATH", fanPath);
auto severity =
sdbusplus::xyz::openbmc_project::Logging::server::convertForMessage(
sdbusplus::xyz::openbmc_project::Logging::server::Entry::Level::
Error);
// Save our logs to a temp file and get the file descriptor
// so it can be passed in as FFDC data.
auto logFile = getLogger().saveToTempFile();
util::FileDescriptor fd{-1};
fd.open(logFile, O_RDONLY);
std::vector<std::tuple<
sdbusplus::xyz::openbmc_project::Logging::server::Create::FFDCFormat,
uint8_t, uint8_t, sdbusplus::message::unix_fd>>
ffdc;
ffdc.emplace_back(sdbusplus::xyz::openbmc_project::Logging::server::Create::
FFDCFormat::Text,
0x01, 0x01, fd());
try
{
util::SDBusPlus::lookupAndCallMethod(
loggingPath, loggingCreateIface, "CreateWithFFDCFiles",
"xyz.openbmc_project.Fan.Error.Missing", severity, additionalData,
ffdc);
}
catch (const util::DBusError& e)
{
getLogger().log(
fmt::format(
"Call to create an error log for missing fan {} failed: {}",
fanPath, e.what()),
Logger::error);
fs::remove(logFile);
throw;
}
fs::remove(logFile);
}
void ErrorReporter::powerStateChanged(bool powerState)
{
if (powerState)
{
// If there are fans already missing, log it.
auto missing = std::count_if(_fanStates.begin(), _fanStates.end(),
[](const auto& fanState) {
return fanState.second == false;
});
if (missing)
{
getLogger().log(
fmt::format("At power on, there are {} missing fans", missing));
}
}
std::for_each(_fanStates.begin(), _fanStates.end(),
[this](const auto& fanState) {
this->checkFan(fanState.first);
});
}
} // namespace phosphor::fan::presence