| Matt Spinler | 711d51d | 2019-11-06 09:36:51 -0600 | [diff] [blame] | 1 | /** | 
 | 2 |  * Copyright © 2019 IBM Corporation | 
 | 3 |  * | 
 | 4 |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
 | 5 |  * you may not use this file except in compliance with the License. | 
 | 6 |  * You may obtain a copy of the License at | 
 | 7 |  * | 
 | 8 |  *     http://www.apache.org/licenses/LICENSE-2.0 | 
 | 9 |  * | 
 | 10 |  * Unless required by applicable law or agreed to in writing, software | 
 | 11 |  * distributed under the License is distributed on an "AS IS" BASIS, | 
 | 12 |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | 13 |  * See the License for the specific language governing permissions and | 
 | 14 |  * limitations under the License. | 
 | 15 |  */ | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 16 | #include "private_header.hpp" | 
 | 17 |  | 
| Harisuddin Mohamed Isa | 2ecb465 | 2020-01-10 10:51:02 +0800 | [diff] [blame] | 18 | #include "json_utils.hpp" | 
| Matt Spinler | 289aa47 | 2019-09-20 12:33:29 -0500 | [diff] [blame] | 19 | #include "log_id.hpp" | 
| Matt Spinler | 1a94cc3 | 2019-09-11 13:32:12 -0500 | [diff] [blame] | 20 | #include "pel_types.hpp" | 
| Aatir | c92b4eb | 2019-11-18 13:44:51 -0600 | [diff] [blame] | 21 | #include "pel_values.hpp" | 
| Matt Spinler | 1a94cc3 | 2019-09-11 13:32:12 -0500 | [diff] [blame] | 22 |  | 
| Arya K Padman | 5bc2653 | 2024-04-10 06:19:25 -0500 | [diff] [blame] | 23 | #include <phosphor-logging/lg2.hpp> | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 24 |  | 
 | 25 | namespace openpower | 
 | 26 | { | 
 | 27 | namespace pels | 
 | 28 | { | 
 | 29 |  | 
| Aatir | c92b4eb | 2019-11-18 13:44:51 -0600 | [diff] [blame] | 30 | namespace pv = openpower::pels::pel_values; | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 31 |  | 
| Matt Spinler | 289aa47 | 2019-09-20 12:33:29 -0500 | [diff] [blame] | 32 | PrivateHeader::PrivateHeader(uint16_t componentID, uint32_t obmcLogID, | 
 | 33 |                              uint64_t timestamp) | 
 | 34 | { | 
 | 35 |     _header.id = static_cast<uint16_t>(SectionID::privateHeader); | 
 | 36 |     _header.size = PrivateHeader::flattenedSize(); | 
 | 37 |     _header.version = privateHeaderVersion; | 
 | 38 |     _header.subType = 0; | 
 | 39 |     _header.componentID = componentID; | 
 | 40 |  | 
 | 41 |     _createTimestamp = getBCDTime(timestamp); | 
 | 42 |  | 
 | 43 |     auto now = std::chrono::system_clock::now(); | 
 | 44 |     _commitTimestamp = getBCDTime(now); | 
 | 45 |  | 
 | 46 |     _creatorID = static_cast<uint8_t>(CreatorID::openBMC); | 
 | 47 |  | 
 | 48 |     // Add support for reminder and telemetry log types here if | 
 | 49 |     // ever necessary. | 
 | 50 |     _logType = 0; | 
 | 51 |  | 
 | 52 |     _reservedByte = 0; | 
 | 53 |  | 
 | 54 |     // the final section count will be updated later | 
 | 55 |     _sectionCount = 1; | 
 | 56 |  | 
 | 57 |     _obmcLogID = obmcLogID; | 
 | 58 |  | 
 | 59 |     _id = generatePELID(); | 
 | 60 |  | 
 | 61 |     _plid = _id; | 
 | 62 |  | 
 | 63 |     // Leave _creatorVersion at 0 | 
 | 64 |  | 
 | 65 |     _valid = true; | 
 | 66 | } | 
 | 67 |  | 
| Matt Spinler | 31eed99 | 2019-10-09 14:07:52 -0500 | [diff] [blame] | 68 | PrivateHeader::PrivateHeader(Stream& pel) : | 
 | 69 |     _creatorID(0), _logType(0), _reservedByte(0), _sectionCount(0), | 
 | 70 |     _obmcLogID(0), _plid(0), _id(0) | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 71 | { | 
 | 72 |     try | 
 | 73 |     { | 
| Matt Spinler | cf5a8d0 | 2019-09-05 12:58:53 -0500 | [diff] [blame] | 74 |         unflatten(pel); | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 75 |         validate(); | 
 | 76 |     } | 
 | 77 |     catch (const std::exception& e) | 
 | 78 |     { | 
| Arya K Padman | 5bc2653 | 2024-04-10 06:19:25 -0500 | [diff] [blame] | 79 |         lg2::error("Cannot unflatten private header: {EXCEPTION}", "EXCEPTION", | 
 | 80 |                    e); | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 81 |         _valid = false; | 
 | 82 |     } | 
 | 83 | } | 
| Matt Spinler | b832aa5 | 2023-03-21 15:32:34 -0500 | [diff] [blame] | 84 | std::optional<std::string> PrivateHeader::getJSON(uint8_t creatorID) const | 
| Aatir | c92b4eb | 2019-11-18 13:44:51 -0600 | [diff] [blame] | 85 | { | 
 | 86 |     char tmpPhVal[50]; | 
| Harisuddin Mohamed Isa | 160c51c | 2020-02-13 23:42:20 +0800 | [diff] [blame] | 87 |     sprintf(tmpPhVal, "%02X/%02X/%02X%02X %02X:%02X:%02X", | 
| Aatir | c92b4eb | 2019-11-18 13:44:51 -0600 | [diff] [blame] | 88 |             _createTimestamp.month, _createTimestamp.day, | 
 | 89 |             _createTimestamp.yearMSB, _createTimestamp.yearLSB, | 
 | 90 |             _createTimestamp.hour, _createTimestamp.minutes, | 
 | 91 |             _createTimestamp.seconds); | 
 | 92 |     std::string phCreateTStr(tmpPhVal); | 
| Harisuddin Mohamed Isa | 160c51c | 2020-02-13 23:42:20 +0800 | [diff] [blame] | 93 |     sprintf(tmpPhVal, "%02X/%02X/%02X%02X %02X:%02X:%02X", | 
| Aatir | c92b4eb | 2019-11-18 13:44:51 -0600 | [diff] [blame] | 94 |             _commitTimestamp.month, _commitTimestamp.day, | 
| Sumit Kumar | 74348c6 | 2021-10-26 05:48:28 -0500 | [diff] [blame] | 95 |             _commitTimestamp.yearMSB, _commitTimestamp.yearLSB, | 
| Aatir | c92b4eb | 2019-11-18 13:44:51 -0600 | [diff] [blame] | 96 |             _commitTimestamp.hour, _commitTimestamp.minutes, | 
 | 97 |             _commitTimestamp.seconds); | 
 | 98 |     std::string phCommitTStr(tmpPhVal); | 
| Harisuddin Mohamed Isa | bebeb94 | 2020-03-12 17:12:24 +0800 | [diff] [blame] | 99 |     std::string creator = getNumberString("%c", _creatorID); | 
| Aatir | c92b4eb | 2019-11-18 13:44:51 -0600 | [diff] [blame] | 100 |     creator = pv::creatorIDs.count(creator) ? pv::creatorIDs.at(creator) | 
 | 101 |                                             : "Unknown CreatorID"; | 
 | 102 |     std::string phCreatorVersionStr = | 
 | 103 |         std::string(reinterpret_cast<const char*>(_creatorVersion.version)); | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 104 |  | 
| Harisuddin Mohamed Isa | 2ecb465 | 2020-01-10 10:51:02 +0800 | [diff] [blame] | 105 |     std::string ph; | 
| Harisuddin Mohamed Isa | bebeb94 | 2020-03-12 17:12:24 +0800 | [diff] [blame] | 106 |     jsonInsert(ph, pv::sectionVer, getNumberString("%d", privateHeaderVersion), | 
 | 107 |                1); | 
 | 108 |     jsonInsert(ph, pv::subSection, getNumberString("%d", _header.subType), 1); | 
| Matt Spinler | b832aa5 | 2023-03-21 15:32:34 -0500 | [diff] [blame] | 109 |     jsonInsert(ph, pv::createdBy, | 
 | 110 |                getComponentName(_header.componentID, creatorID), 1); | 
| Harisuddin Mohamed Isa | 2ecb465 | 2020-01-10 10:51:02 +0800 | [diff] [blame] | 111 |     jsonInsert(ph, "Created at", phCreateTStr, 1); | 
 | 112 |     jsonInsert(ph, "Committed at", phCommitTStr, 1); | 
 | 113 |     jsonInsert(ph, "Creator Subsystem", creator, 1); | 
 | 114 |     jsonInsert(ph, "CSSVER", phCreatorVersionStr, 1); | 
| Harisuddin Mohamed Isa | bebeb94 | 2020-03-12 17:12:24 +0800 | [diff] [blame] | 115 |     jsonInsert(ph, "Platform Log Id", getNumberString("0x%X", _plid), 1); | 
 | 116 |     jsonInsert(ph, "Entry Id", getNumberString("0x%X", _id), 1); | 
 | 117 |     jsonInsert(ph, "BMC Event Log Id", std::to_string(_obmcLogID), 1); | 
| Harisuddin Mohamed Isa | 2ecb465 | 2020-01-10 10:51:02 +0800 | [diff] [blame] | 118 |     ph.erase(ph.size() - 2); | 
| Aatir | c92b4eb | 2019-11-18 13:44:51 -0600 | [diff] [blame] | 119 |  | 
 | 120 |     return ph; | 
 | 121 | } | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 122 | void PrivateHeader::validate() | 
 | 123 | { | 
 | 124 |     bool failed = false; | 
 | 125 |  | 
| Matt Spinler | 1a94cc3 | 2019-09-11 13:32:12 -0500 | [diff] [blame] | 126 |     if (header().id != static_cast<uint16_t>(SectionID::privateHeader)) | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 127 |     { | 
| Arya K Padman | 5bc2653 | 2024-04-10 06:19:25 -0500 | [diff] [blame] | 128 |         lg2::error("Invalid private header section ID: {HEADER_ID}", | 
 | 129 |                    "HEADER_ID", lg2::hex, header().id); | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 130 |         failed = true; | 
 | 131 |     } | 
 | 132 |  | 
 | 133 |     if (header().version != privateHeaderVersion) | 
 | 134 |     { | 
| Arya K Padman | 5bc2653 | 2024-04-10 06:19:25 -0500 | [diff] [blame] | 135 |         lg2::error("Invalid private header version: {HEADER_VERSION}", | 
 | 136 |                    "HEADER_VERSION", lg2::hex, header().version); | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 137 |         failed = true; | 
 | 138 |     } | 
 | 139 |  | 
 | 140 |     if (_sectionCount < minSectionCount) | 
 | 141 |     { | 
| Arya K Padman | 5bc2653 | 2024-04-10 06:19:25 -0500 | [diff] [blame] | 142 |         lg2::error("Invalid section count in private header: {SECTION_COUNT}", | 
 | 143 |                    "SECTION_COUNT", lg2::hex, _sectionCount); | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 144 |         failed = true; | 
 | 145 |     } | 
 | 146 |  | 
 | 147 |     _valid = (failed) ? false : true; | 
 | 148 | } | 
 | 149 |  | 
| Matt Spinler | cf5a8d0 | 2019-09-05 12:58:53 -0500 | [diff] [blame] | 150 | void PrivateHeader::unflatten(Stream& stream) | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 151 | { | 
| Matt Spinler | cf5a8d0 | 2019-09-05 12:58:53 -0500 | [diff] [blame] | 152 |     stream >> _header >> _createTimestamp >> _commitTimestamp >> _creatorID >> | 
 | 153 |         _logType >> _reservedByte >> _sectionCount >> _obmcLogID >> | 
 | 154 |         _creatorVersion >> _plid >> _id; | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 155 | } | 
 | 156 |  | 
| Matt Spinler | 0688545 | 2019-11-06 10:35:42 -0600 | [diff] [blame] | 157 | void PrivateHeader::flatten(Stream& stream) const | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 158 | { | 
| Matt Spinler | cf5a8d0 | 2019-09-05 12:58:53 -0500 | [diff] [blame] | 159 |     stream << _header << _createTimestamp << _commitTimestamp << _creatorID | 
 | 160 |            << _logType << _reservedByte << _sectionCount << _obmcLogID | 
 | 161 |            << _creatorVersion << _plid << _id; | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 162 | } | 
 | 163 |  | 
 | 164 | Stream& operator>>(Stream& s, CreatorVersion& cv) | 
 | 165 | { | 
 | 166 |     for (size_t i = 0; i < sizeof(CreatorVersion); i++) | 
 | 167 |     { | 
 | 168 |         s >> cv.version[i]; | 
 | 169 |     } | 
 | 170 |     return s; | 
 | 171 | } | 
 | 172 |  | 
| Matt Spinler | 0688545 | 2019-11-06 10:35:42 -0600 | [diff] [blame] | 173 | Stream& operator<<(Stream& s, const CreatorVersion& cv) | 
| Matt Spinler | d3335df | 2019-07-10 11:04:21 -0500 | [diff] [blame] | 174 | { | 
 | 175 |     for (size_t i = 0; i < sizeof(CreatorVersion); i++) | 
 | 176 |     { | 
 | 177 |         s << cv.version[i]; | 
 | 178 |     } | 
 | 179 |     return s; | 
 | 180 | } | 
 | 181 |  | 
 | 182 | } // namespace pels | 
 | 183 | } // namespace openpower |