#include #include #include #include #include #include #include #include #include #include #include "elog-lookup.cpp" #include #include "config.h" #include "elog_entry.hpp" #include #include "log_manager.hpp" namespace phosphor { namespace logging { void Manager::commit(uint64_t transactionId, std::string errMsg) { constexpr const auto transactionIdVar = "TRANSACTION_ID"; // Length of 'TRANSACTION_ID' string. constexpr const auto transactionIdVarSize = strlen(transactionIdVar); // Length of 'TRANSACTION_ID=' string. constexpr const auto transactionIdVarOffset = transactionIdVarSize + 1; sd_journal *j = nullptr; int rc = sd_journal_open(&j, SD_JOURNAL_LOCAL_ONLY); if (rc < 0) { logging::log("Failed to open journal", logging::entry("DESCRIPTION=%s", strerror(-rc))); return; } std::string transactionIdStr = std::to_string(transactionId); std::set metalist(g_errMetaMap[errMsg].begin(), g_errMetaMap[errMsg].end()); const auto& metalistHostEvent = g_errMetaMapHostEvent[errMsg]; std::vector additionalData; // TODO Remove once host event error header file is auto-generated. // Also make metalist a const variable. // Tracking with issue openbmc/phosphor-logging#4 for (auto& metaVarStrHostEvent : metalistHostEvent) { metalist.insert(metaVarStrHostEvent); } // Read the journal from the end to get the most recent entry first. // The result from the sd_journal_get_data() is of the form VARIABLE=value. SD_JOURNAL_FOREACH_BACKWARDS(j) { const char *data = nullptr; size_t length = 0; // Look for the transaction id metadata variable rc = sd_journal_get_data(j, transactionIdVar, (const void **)&data, &length); if (rc < 0) { // This journal entry does not have the TRANSACTION_ID // metadata variable. continue; } // journald does not guarantee that sd_journal_get_data() returns NULL // terminated strings, so need to specify the size to use to compare, // use the returned length instead of anything that relies on NULL // terminators like strlen(). // The data variable is in the form of 'TRANSACTION_ID=1234'. Remove // the TRANSACTION_ID characters plus the (=) sign to do the comparison. // 'data + transactionIdVarOffset' will be in the form of '1234'. // 'length - transactionIdVarOffset' will be the length of '1234'. if ((length <= (transactionIdVarOffset)) || (transactionIdStr.compare(0, transactionIdStr.size(), data + transactionIdVarOffset, length - transactionIdVarOffset) != 0)) { // The value of the TRANSACTION_ID metadata is not the requested // transaction id number. continue; } // Search for all metadata variables in the current journal entry. for (auto i = metalist.cbegin(); i != metalist.cend();) { rc = sd_journal_get_data(j, (*i).c_str(), (const void **)&data, &length); if (rc < 0) { // Metadata variable not found, check next metadata variable. i++; continue; } // Metadata variable found, save it and remove it from the set. additionalData.emplace_back(data, length); i = metalist.erase(i); } if (metalist.empty()) { // All metadata variables found, break out of journal loop. break; } } if (!metalist.empty()) { // Not all the metadata variables were found in the journal. for (auto& metaVarStr : metalist) { logging::log("Failed to find metadata", logging::entry("META_FIELD=%s", metaVarStr.c_str())); } } sd_journal_close(j); // Create error Entry dbus object entryId++; auto ms = std::chrono::duration_cast( std::chrono::system_clock::now().time_since_epoch()).count(); auto objPath = std::string(OBJ_ENTRY) + '/' + std::to_string(entryId); AssociationList objects {}; entries.insert(std::make_pair(entryId, std::make_unique( busLog, objPath, entryId, ms, // Milliseconds since 1970 (Entry::Level)g_errLevelMap[errMsg], std::move(errMsg), std::move(additionalData), std::move(objects)))); return; } } // namespace logging } // namepsace phosphor