#include "config.h" #include "elog_serialize.hpp" #include #include #include #include #include #include // Register class version // From cereal documentation; // "This macro should be placed at global scope" CEREAL_CLASS_VERSION(phosphor::logging::Entry, CLASS_VERSION) namespace phosphor { namespace logging { /** @brief Function required by Cereal to perform serialization. * @tparam Archive - Cereal archive type (binary in our case). * @param[in] a - reference to Cereal archive. * @param[in] e - const reference to error entry. * @param[in] version - Class version that enables handling * a serialized data across code levels */ template void save(Archive& a, const Entry& e, const std::uint32_t /*version*/) { a(e.id(), e.severity(), e.timestamp(), e.message(), e.additionalData(), e.associations(), e.resolved(), e.version(), e.updateTimestamp()); } /** @brief Function required by Cereal to perform deserialization. * @tparam Archive - Cereal archive type (binary in our case). * @param[in] a - reference to Cereal archive. * @param[in] e - reference to error entry. * @param[in] version - Class version that enables handling * a serialized data across code levels */ template void load(Archive& a, Entry& e, const std::uint32_t version) { using namespace sdbusplus::xyz::openbmc_project::Logging::server; uint32_t id{}; Entry::Level severity{}; uint64_t timestamp{}; std::string message{}; std::vector additionalData{}; bool resolved{}; AssociationList associations{}; std::string fwVersion{}; uint64_t updateTimestamp{}; if (version < std::stoul(FIRST_CEREAL_CLASS_VERSION_WITH_FWLEVEL)) { a(id, severity, timestamp, message, additionalData, associations, resolved); updateTimestamp = timestamp; } else if (version < std::stoul(FIRST_CEREAL_CLASS_VERSION_WITH_UPDATE_TS)) { a(id, severity, timestamp, message, additionalData, associations, resolved, fwVersion); updateTimestamp = timestamp; } else { a(id, severity, timestamp, message, additionalData, associations, resolved, fwVersion, updateTimestamp); } e.id(id); e.severity(severity); e.timestamp(timestamp); e.message(message); e.additionalData(additionalData); e.sdbusplus::xyz::openbmc_project::Logging::server::Entry::resolved( resolved); e.associations(associations); e.version(fwVersion); e.purpose(sdbusplus::xyz::openbmc_project::Software::server::Version:: VersionPurpose::BMC); e.updateTimestamp(updateTimestamp); } fs::path serialize(const Entry& e, const fs::path& dir) { auto path = dir / std::to_string(e.id()); std::ofstream os(path.c_str(), std::ios::binary); cereal::BinaryOutputArchive oarchive(os); oarchive(e); return path; } bool deserialize(const fs::path& path, Entry& e) { try { if (fs::exists(path)) { std::ifstream is(path.c_str(), std::ios::in | std::ios::binary); cereal::BinaryInputArchive iarchive(is); iarchive(e); return true; } return false; } catch (cereal::Exception& e) { log(e.what()); fs::remove(path); return false; } catch (const std::length_error& e) { // Running into: USCiLab/cereal#192 // This may be indicating some other issue in the // way vector may have been used inside the logging. // possibly associations ??. But handling it here for // now since we are anyway tossing the log // TODO: openbmc/phosphor-logging#8 log(e.what()); fs::remove(path); return false; } } } // namespace logging } // namespace phosphor