blob: bfe549cfd0c7539aae52c68dfbd7d908d465e231 [file] [log] [blame]
/*
// Copyright (c) 2019 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.
*/
#include <systemd/sd-journal.h>
#include <boost/algorithm/string/predicate.hpp>
#include <boost/asio/io_context.hpp>
#include <boost/asio/steady_timer.hpp>
#include <boost/container/flat_map.hpp>
#include <cstdint>
#include <iostream>
#include <sdbusplus/asio/connection.hpp>
#include <sdbusplus/bus.hpp>
#include <sdbusplus/bus/match.hpp>
#include <string>
#include <variant>
#include <vector>
using GetSubTreeType = std::vector<
std::pair<std::string,
std::vector<std::pair<std::string, std::vector<std::string>>>>>;
using BasicVariantType =
std::variant<std::vector<std::string>, std::string, int64_t, uint64_t,
double, int32_t, uint32_t, int16_t, uint16_t, uint8_t, bool>;
using Association = std::tuple<std::string, std::string, std::string>;
constexpr const char* assetTag =
"xyz.openbmc_project.Inventory.Decorator.Asset";
namespace mapper
{
constexpr const char* busName = "xyz.openbmc_project.ObjectMapper";
constexpr const char* path = "/xyz/openbmc_project/object_mapper";
constexpr const char* interface = "xyz.openbmc_project.ObjectMapper";
constexpr const char* subtree = "GetSubTree";
} // namespace mapper
namespace entityManager
{
constexpr const char* busName = "xyz.openbmc_project.EntityManager";
} // namespace entityManager
namespace inventory
{
constexpr const char* interface = "xyz.openbmc_project.Inventory.Item";
} // namespace inventory
namespace ledGroup
{
constexpr const char* interface = "xyz.openbmc_project.Led.Group";
constexpr const char* asserted = "Asserted";
} // namespace ledGroup
namespace properties
{
constexpr const char* interface = "org.freedesktop.DBus.Properties";
constexpr const char* get = "Get";
} // namespace properties
namespace power
{
const static constexpr char* busname = "xyz.openbmc_project.State.Host";
const static constexpr char* interface = "xyz.openbmc_project.State.Host";
const static constexpr char* path = "/xyz/openbmc_project/state/host0";
const static constexpr char* property = "CurrentHostState";
} // namespace power
namespace association
{
const static constexpr char* interface =
"xyz.openbmc_project.Association.Definitions";
} // namespace association
namespace hsbp
{
enum class registers : uint8_t
{
fpgaIdH = 0x0,
fpgaIdL = 0x1,
typeId = 0x2,
bootVer = 0x3,
fpgaVer = 0x4,
securityRev = 0x5,
funSupported = 0x6,
numDisks = 0x7,
presence = 0x8,
ssdIFDET = 0x9,
ifdetPart = 0xA,
statusLocate = 0xB,
statusFail = 0xC,
statusRebuild = 0xD,
ledOverride = 0xE,
ledStatus = 0xF,
ledPattern0 = 0x10,
ledPattern1 = 0x11,
ledPattern2 = 0x12,
ledPattern3 = 0x13,
ledPattern4 = 0x14,
ledPattern5 = 0x15,
ledPattern6 = 0x16,
ledPattern7 = 0x17,
};
} // namespace hsbp
static std::unique_ptr<sdbusplus::bus::match_t> powerMatch = nullptr;
static bool powerStatusOn = false;
bool isPowerOn(void)
{
if (!powerMatch)
{
throw std::runtime_error("Power Match Not Created");
}
return powerStatusOn;
}
void setupPowerMatch(const std::shared_ptr<sdbusplus::asio::connection>& conn)
{
static boost::asio::steady_timer timer(conn->get_io_context());
// create a match for powergood changes, first time do a method call to
// cache the correct value
if (powerMatch)
{
return;
}
powerMatch = std::make_unique<sdbusplus::bus::match_t>(
static_cast<sdbusplus::bus_t&>(*conn),
"type='signal',interface='" + std::string(properties::interface) +
"',path='" + std::string(power::path) + "',arg0='" +
std::string(power::interface) + "'",
[](sdbusplus::message_t& message) {
std::string objectName;
boost::container::flat_map<std::string, std::variant<std::string>>
values;
message.read(objectName, values);
auto findState = values.find(power::property);
if (findState != values.end())
{
bool on = boost::ends_with(
std::get<std::string>(findState->second), "Running");
if (!on)
{
timer.cancel();
powerStatusOn = false;
return;
}
// on comes too quickly
timer.expires_after(std::chrono::seconds(10));
timer.async_wait([](boost::system::error_code ec) {
if (ec == boost::asio::error::operation_aborted)
{
return;
}
else if (ec)
{
std::cerr << "Timer error " << ec.message() << "\n";
return;
}
powerStatusOn = true;
});
}
});
conn->async_method_call(
[](boost::system::error_code ec,
const std::variant<std::string>& state) {
if (ec)
{
// we commonly come up before power control, we'll capture the
// property change later
return;
}
powerStatusOn =
boost::ends_with(std::get<std::string>(state), "Running");
},
power::busname, power::path, properties::interface, properties::get,
power::interface, power::property);
}
inline void logDeviceAdded(const std::string& model, const std::string& type,
const std::string& sn)
{
sd_journal_send("MESSAGE=%s", "Inventory Added", "PRIORITY=%i", LOG_ERR,
"REDFISH_MESSAGE_ID=%s", "OpenBMC.0.1.InventoryAdded",
"REDFISH_MESSAGE_ARGS=%s,%s,%s", model.c_str(),
type.c_str(), sn.c_str(), NULL);
}
inline void logDeviceRemoved(const std::string& model, const std::string& type,
const std::string& sn)
{
sd_journal_send("MESSAGE=%s", "Inventory Removed", "PRIORITY=%i", LOG_ERR,
"REDFISH_MESSAGE_ID=%s", "OpenBMC.0.1.InventoryRemoved",
"REDFISH_MESSAGE_ARGS=%s,%s,%s", model.c_str(),
type.c_str(), sn.c_str(), NULL);
}
inline void logDriveError(const std::string& name)
{
sd_journal_send("MESSAGE=%s", "Drive Error", "PRIORITY=%i", LOG_ERR,
"REDFISH_MESSAGE_ID=%s", "OpenBMC.0.1.DriveError",
"REDFISH_MESSAGE_ARGS=%s", name.c_str(), NULL);
}