blob: 1f1cb9e88c66b0293a577508ba539a7e9677311c [file] [log] [blame]
Patrick Venturec83c4dc2018-11-01 16:29:18 -07001#include "impl.hpp"
2
Patrick Williamsc78d8872023-05-10 07:50:56 -05003#include "vpdecc/vpdecc.h"
4
SunnySrivastava198419be6d32020-03-03 07:21:45 -06005#include "const.hpp"
Patrick Venturec83c4dc2018-11-01 16:29:18 -07006#include "defines.hpp"
Sunny Srivastava6c71c9d2021-04-15 04:43:54 -05007#include "ibm_vpd_utils.hpp"
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -05008#include "types.hpp"
SunnySrivastava19849d4f1122020-08-26 02:41:46 -05009#include "vpd_exceptions.hpp"
Patrick Venturec83c4dc2018-11-01 16:29:18 -070010
11#include <algorithm>
Deepak Kodihalli810c9de2016-11-22 11:42:51 -060012#include <exception>
Patrick Venturec83c4dc2018-11-01 16:29:18 -070013#include <iomanip>
Deepak Kodihalli810c9de2016-11-22 11:42:51 -060014#include <iostream>
15#include <iterator>
Patrick Venturec83c4dc2018-11-01 16:29:18 -070016#include <sstream>
Deepak Kodihalli4a475bd2016-11-24 07:08:20 -060017#include <tuple>
Patrick Venturec83c4dc2018-11-01 16:29:18 -070018#include <unordered_map>
Deepak Kodihalli810c9de2016-11-22 11:42:51 -060019
20namespace openpower
21{
22namespace vpd
23{
24namespace parser
25{
SunnySrivastava198419be6d32020-03-03 07:21:45 -060026using namespace openpower::vpd::constants;
SunnySrivastava19849d4f1122020-08-26 02:41:46 -050027using namespace openpower::vpd::exceptions;
Deepak Kodihalli810c9de2016-11-22 11:42:51 -060028
Patrick Venturec83c4dc2018-11-01 16:29:18 -070029static const std::unordered_map<std::string, Record> supportedRecords = {
30 {"VINI", Record::VINI}, {"OPFR", Record::OPFR}, {"OSYS", Record::OSYS}};
Deepak Kodihallia1143462016-11-24 06:28:45 -060031
Patrick Venturec83c4dc2018-11-01 16:29:18 -070032static const std::unordered_map<std::string, internal::KeywordInfo>
33 supportedKeywords = {
34 {"DR", std::make_tuple(record::Keyword::DR, keyword::Encoding::ASCII)},
35 {"PN", std::make_tuple(record::Keyword::PN, keyword::Encoding::ASCII)},
36 {"SN", std::make_tuple(record::Keyword::SN, keyword::Encoding::ASCII)},
37 {"CC", std::make_tuple(record::Keyword::CC, keyword::Encoding::ASCII)},
38 {"HW", std::make_tuple(record::Keyword::HW, keyword::Encoding::RAW)},
39 {"B1", std::make_tuple(record::Keyword::B1, keyword::Encoding::B1)},
40 {"VN", std::make_tuple(record::Keyword::VN, keyword::Encoding::ASCII)},
George Liuee79ca82019-07-12 11:05:33 +080041 {"MB", std::make_tuple(record::Keyword::MB, keyword::Encoding::MB)},
Patrick Venturec83c4dc2018-11-01 16:29:18 -070042 {"MM", std::make_tuple(record::Keyword::MM, keyword::Encoding::ASCII)},
43 {"UD", std::make_tuple(record::Keyword::UD, keyword::Encoding::UD)},
44 {"VP", std::make_tuple(record::Keyword::VP, keyword::Encoding::ASCII)},
45 {"VS", std::make_tuple(record::Keyword::VS, keyword::Encoding::ASCII)},
Deepak Kodihalli4a475bd2016-11-24 07:08:20 -060046};
47
Alpana Kumaric0aeac32019-11-28 05:20:10 -060048namespace
49{
50constexpr auto toHex(size_t c)
51{
52 constexpr auto map = "0123456789abcdef";
53 return map[c];
54}
55} // namespace
56
57/*readUInt16LE: Read 2 bytes LE data*/
58static LE2ByteData readUInt16LE(Binary::const_iterator iterator)
59{
60 LE2ByteData lowByte = *iterator;
61 LE2ByteData highByte = *(iterator + 1);
62 lowByte |= (highByte << 8);
63 return lowByte;
64}
65
66RecordOffset Impl::getVtocOffset() const
67{
68 auto vpdPtr = vpd.cbegin();
69 std::advance(vpdPtr, offsets::VTOC_PTR);
70 // Get VTOC Offset
71 auto vtocOffset = readUInt16LE(vpdPtr);
72
73 return vtocOffset;
74}
75
76#ifdef IPZ_PARSER
girik18bb9852022-11-16 05:48:13 -060077int Impl::vhdrEccCheck()
Alpana Kumaric0aeac32019-11-28 05:20:10 -060078{
79 int rc = eccStatus::SUCCESS;
80 auto vpdPtr = vpd.cbegin();
81
82 auto l_status =
83 vpdecc_check_data(const_cast<uint8_t*>(&vpdPtr[offsets::VHDR_RECORD]),
84 lengths::VHDR_RECORD_LENGTH,
85 const_cast<uint8_t*>(&vpdPtr[offsets::VHDR_ECC]),
86 lengths::VHDR_ECC_LENGTH);
girik18bb9852022-11-16 05:48:13 -060087 if (l_status == VPD_ECC_CORRECTABLE_DATA)
88 {
89 try
90 {
91 if (vpdFileStream.is_open())
92 {
93 vpdFileStream.seekp(vpdStartOffset + offsets::VHDR_RECORD,
94 std::ios::beg);
95 vpdFileStream.write(
96 reinterpret_cast<const char*>(&vpd[offsets::VHDR_RECORD]),
97 lengths::VHDR_RECORD_LENGTH);
98 }
99 else
100 {
101 std::cerr << "File not open";
102 rc = eccStatus::FAILED;
103 }
104 }
105 catch (const std::fstream::failure& e)
106 {
107 std::cout << "Error while operating on file with exception:"
108 << e.what();
109 rc = eccStatus::FAILED;
110 }
111 }
112 else if (l_status != VPD_ECC_OK)
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600113 {
114 rc = eccStatus::FAILED;
115 }
116
117 return rc;
118}
119
girik18bb9852022-11-16 05:48:13 -0600120int Impl::vtocEccCheck()
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600121{
122 int rc = eccStatus::SUCCESS;
123 // Use another pointer to get ECC information from VHDR,
124 // actual pointer is pointing to VTOC data
125
126 auto vpdPtr = vpd.cbegin();
127
128 // Get VTOC Offset
129 auto vtocOffset = getVtocOffset();
130
131 // Get the VTOC Length
132 std::advance(vpdPtr, offsets::VTOC_PTR + sizeof(RecordOffset));
133 auto vtocLength = readUInt16LE(vpdPtr);
134
135 // Get the ECC Offset
136 std::advance(vpdPtr, sizeof(RecordLength));
137 auto vtocECCOffset = readUInt16LE(vpdPtr);
138
139 // Get the ECC length
140 std::advance(vpdPtr, sizeof(ECCOffset));
141 auto vtocECCLength = readUInt16LE(vpdPtr);
142
143 // Reset pointer to start of the vpd,
144 // so that Offset will point to correct address
145 vpdPtr = vpd.cbegin();
146 auto l_status = vpdecc_check_data(
147 const_cast<uint8_t*>(&vpdPtr[vtocOffset]), vtocLength,
148 const_cast<uint8_t*>(&vpdPtr[vtocECCOffset]), vtocECCLength);
girik18bb9852022-11-16 05:48:13 -0600149 if (l_status == VPD_ECC_CORRECTABLE_DATA)
150 {
151 try
152 {
153 if (vpdFileStream.is_open())
154 {
155 vpdFileStream.seekp(vpdStartOffset + vtocOffset, std::ios::beg);
156 vpdFileStream.write(
157 reinterpret_cast<const char*>(&vpdPtr[vtocOffset]),
158 vtocLength);
159 }
160 else
161 {
162 std::cerr << "File not open";
163 rc = eccStatus::FAILED;
164 }
165 }
166 catch (const std::fstream::failure& e)
167 {
168 std::cout << "Error while operating on file with exception "
169 << e.what();
170 rc = eccStatus::FAILED;
171 }
172 }
173 else if (l_status != VPD_ECC_OK)
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600174 {
175 rc = eccStatus::FAILED;
176 }
177
178 return rc;
179}
180
girik18bb9852022-11-16 05:48:13 -0600181int Impl::recordEccCheck(Binary::const_iterator iterator)
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600182{
183 int rc = eccStatus::SUCCESS;
184
185 auto recordOffset = readUInt16LE(iterator);
186
187 std::advance(iterator, sizeof(RecordOffset));
188 auto recordLength = readUInt16LE(iterator);
189
190 std::advance(iterator, sizeof(RecordLength));
191 auto eccOffset = readUInt16LE(iterator);
192
193 std::advance(iterator, sizeof(ECCOffset));
194 auto eccLength = readUInt16LE(iterator);
195
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500196 if (eccLength == 0 || eccOffset == 0)
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600197 {
Santosh Puranikc04e1852022-12-20 11:13:22 +0530198 throw(VpdEccException(
199 "Could not find ECC's offset or Length for Record:"));
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500200 }
201
202 if (recordOffset == 0 || recordLength == 0)
203 {
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500204 throw(VpdDataException("Could not find VPD record offset or VPD record "
Santosh Puranikc04e1852022-12-20 11:13:22 +0530205 "length for Record:"));
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600206 }
207
208 auto vpdPtr = vpd.cbegin();
209
210 auto l_status = vpdecc_check_data(
211 const_cast<uint8_t*>(&vpdPtr[recordOffset]), recordLength,
212 const_cast<uint8_t*>(&vpdPtr[eccOffset]), eccLength);
girik18bb9852022-11-16 05:48:13 -0600213 if (l_status == VPD_ECC_CORRECTABLE_DATA)
214 {
215 try
216 {
217 if (vpdFileStream.is_open())
218 {
219 vpdFileStream.seekp(vpdStartOffset + recordOffset,
220 std::ios::beg);
221 vpdFileStream.write(
222 reinterpret_cast<const char*>(&vpdPtr[recordOffset]),
223 recordLength);
224 }
225 else
226 {
227 std::cerr << "File not open";
228 rc = eccStatus::FAILED;
229 }
230 }
231 catch (const std::fstream::failure& e)
232 {
233 std::cout << "Error while operating on file with exception "
234 << e.what();
235 rc = eccStatus::FAILED;
236 }
237 }
238 else if (l_status != VPD_ECC_OK)
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600239 {
240 rc = eccStatus::FAILED;
241 }
242
243 return rc;
244}
245#endif
246
girik18bb9852022-11-16 05:48:13 -0600247void Impl::checkHeader()
Deepak Kodihalli810c9de2016-11-22 11:42:51 -0600248{
249 if (vpd.empty() || (lengths::RECORD_MIN > vpd.size()))
250 {
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500251 throw(VpdDataException("Malformed VPD"));
Deepak Kodihalli810c9de2016-11-22 11:42:51 -0600252 }
253 else
254 {
255 auto iterator = vpd.cbegin();
256 std::advance(iterator, offsets::VHDR);
257 auto stop = std::next(iterator, lengths::RECORD_NAME);
258 std::string record(iterator, stop);
259 if ("VHDR" != record)
260 {
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500261 throw(VpdDataException("VHDR record not found"));
Deepak Kodihalli810c9de2016-11-22 11:42:51 -0600262 }
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600263
264#ifdef IPZ_PARSER
265 // Check ECC
266 int rc = eccStatus::FAILED;
267 rc = vhdrEccCheck();
268 if (rc != eccStatus::SUCCESS)
269 {
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500270 throw(VpdEccException("ERROR: VHDR ECC check Failed"));
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600271 }
272#endif
Deepak Kodihalli810c9de2016-11-22 11:42:51 -0600273 }
274}
275
girik18bb9852022-11-16 05:48:13 -0600276std::size_t Impl::readTOC(Binary::const_iterator& iterator)
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600277{
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600278 // The offset to VTOC could be 1 or 2 bytes long
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600279 RecordOffset vtocOffset = getVtocOffset();
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600280
281 // Got the offset to VTOC, skip past record header and keyword header
282 // to get to the record name.
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700283 std::advance(iterator, vtocOffset + sizeof(RecordId) + sizeof(RecordSize) +
284 // Skip past the RT keyword, which contains
285 // the record name.
286 lengths::KW_NAME + sizeof(KwSize));
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600287
288 auto stop = std::next(iterator, lengths::RECORD_NAME);
289 std::string record(iterator, stop);
290 if ("VTOC" != record)
291 {
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500292 throw(VpdDataException("VTOC record not found"));
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600293 }
294
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600295#ifdef IPZ_PARSER
296 // Check ECC
297 int rc = eccStatus::FAILED;
298 rc = vtocEccCheck();
299 if (rc != eccStatus::SUCCESS)
300 {
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500301 throw(VpdEccException("ERROR: VTOC ECC check Failed"));
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600302 }
303#endif
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600304 // VTOC record name is good, now read through the TOC, stored in the PT
305 // PT keyword; vpdBuffer is now pointing at the first character of the
306 // name 'VTOC', jump to PT data.
307 // Skip past record name and KW name, 'PT'
308 std::advance(iterator, lengths::RECORD_NAME + lengths::KW_NAME);
309 // Note size of PT
310 std::size_t ptLen = *iterator;
311 // Skip past PT size
312 std::advance(iterator, sizeof(KwSize));
313
SunnySrivastava1984f6d541e2020-02-04 12:50:40 -0600314 // length of PT keyword
315 return ptLen;
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600316}
317
318internal::OffsetList Impl::readPT(Binary::const_iterator iterator,
girik18bb9852022-11-16 05:48:13 -0600319 std::size_t ptLength)
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600320{
321 internal::OffsetList offsets{};
322
323 auto end = iterator;
324 std::advance(end, ptLength);
325
326 // Look at each entry in the PT keyword. In the entry,
327 // we care only about the record offset information.
328 while (iterator < end)
329 {
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500330#ifdef IPZ_PARSER
331 auto iteratorToRecName = iterator;
332#endif
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600333 // Skip record name and record type
334 std::advance(iterator, lengths::RECORD_NAME + sizeof(RecordType));
335
336 // Get record offset
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600337 auto offset = readUInt16LE(iterator);
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600338 offsets.push_back(offset);
339
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600340#ifdef IPZ_PARSER
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500341 std::string recordName(iteratorToRecName,
342 iteratorToRecName + lengths::RECORD_NAME);
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600343
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500344 try
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600345 {
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500346 // Verify the ECC for this Record
347 int rc = recordEccCheck(iterator);
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500348
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500349 if (rc != eccStatus::SUCCESS)
350 {
Patrick Williamsc78d8872023-05-10 07:50:56 -0500351 std::string errorMsg = std::string(
352 "ERROR: ECC check did not pass for the "
353 "Record:");
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500354 throw(VpdEccException(errorMsg));
355 }
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600356 }
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500357 catch (const VpdEccException& ex)
358 {
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500359 inventory::PelAdditionalData additionalData{};
Santosh Puranikc04e1852022-12-20 11:13:22 +0530360 additionalData.emplace("DESCRIPTION",
361 std::string{ex.what()} + recordName);
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500362 additionalData.emplace("CALLOUT_INVENTORY_PATH", inventoryPath);
363 createPEL(additionalData, PelSeverity::WARNING,
Sunny Srivastavaa2ddc962022-06-29 08:53:16 -0500364 errIntfForEccCheckFail, nullptr);
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500365 }
366 catch (const VpdDataException& ex)
367 {
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500368 inventory::PelAdditionalData additionalData{};
Santosh Puranikc04e1852022-12-20 11:13:22 +0530369 additionalData.emplace("DESCRIPTION",
370 std::string{ex.what()} + recordName);
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500371 additionalData.emplace("CALLOUT_INVENTORY_PATH", inventoryPath);
372 createPEL(additionalData, PelSeverity::WARNING,
Sunny Srivastavaa2ddc962022-06-29 08:53:16 -0500373 errIntfForInvalidVPD, nullptr);
Sunny Srivastavaf31a91b2022-06-09 08:11:29 -0500374 }
375
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600376#endif
377
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600378 // Jump record size, record length, ECC offset and ECC length
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600379 std::advance(iterator, sizeof(RecordOffset) + sizeof(RecordLength) +
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700380 sizeof(ECCOffset) + sizeof(ECCLength));
Deepak Kodihalli023112f2016-11-22 22:02:14 -0600381 }
382
383 return offsets;
384}
385
Deepak Kodihallia1143462016-11-24 06:28:45 -0600386void Impl::processRecord(std::size_t recordOffset)
387{
388 // Jump to record name
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700389 auto nameOffset = recordOffset + sizeof(RecordId) + sizeof(RecordSize) +
Deepak Kodihallia1143462016-11-24 06:28:45 -0600390 // Skip past the RT keyword, which contains
391 // the record name.
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700392 lengths::KW_NAME + sizeof(KwSize);
Deepak Kodihallia1143462016-11-24 06:28:45 -0600393 // Get record name
394 auto iterator = vpd.cbegin();
395 std::advance(iterator, nameOffset);
396
397 std::string name(iterator, iterator + lengths::RECORD_NAME);
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600398
Alpana Kumari26a74af2019-09-10 23:53:58 -0500399#ifndef IPZ_PARSER
Deepak Kodihallia1143462016-11-24 06:28:45 -0600400 if (supportedRecords.end() != supportedRecords.find(name))
401 {
Alpana Kumari26a74af2019-09-10 23:53:58 -0500402#endif
Deepak Kodihallia1143462016-11-24 06:28:45 -0600403 // If it's a record we're interested in, proceed to find
404 // contained keywords and their values.
405 std::advance(iterator, lengths::RECORD_NAME);
Alpana Kumari26a74af2019-09-10 23:53:58 -0500406
407#ifdef IPZ_PARSER
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600408
Alpana Kumari26a74af2019-09-10 23:53:58 -0500409 // Reverse back to RT Kw, in ipz vpd, to Read RT KW & value
410 std::advance(iterator, -(lengths::KW_NAME + sizeof(KwSize) +
411 lengths::RECORD_NAME));
412#endif
Deepak Kodihallia1143462016-11-24 06:28:45 -0600413 auto kwMap = readKeywords(iterator);
414 // Add entry for this record (and contained keyword:value pairs)
415 // to the parsed vpd output.
416 out.emplace(std::move(name), std::move(kwMap));
Alpana Kumaric0aeac32019-11-28 05:20:10 -0600417
Alpana Kumari26a74af2019-09-10 23:53:58 -0500418#ifndef IPZ_PARSER
Deepak Kodihallia1143462016-11-24 06:28:45 -0600419 }
Alpana Kumari26a74af2019-09-10 23:53:58 -0500420#endif
Deepak Kodihallia1143462016-11-24 06:28:45 -0600421}
422
Deepak Kodihalli4a475bd2016-11-24 07:08:20 -0600423std::string Impl::readKwData(const internal::KeywordInfo& keyword,
424 std::size_t dataLength,
425 Binary::const_iterator iterator)
426{
Santosh Puranikbd011b22020-01-23 04:05:25 -0600427 using namespace openpower::vpd;
Deepak Kodihalli4a475bd2016-11-24 07:08:20 -0600428 switch (std::get<keyword::Encoding>(keyword))
429 {
430 case keyword::Encoding::ASCII:
431 {
432 auto stop = std::next(iterator, dataLength);
433 return std::string(iterator, stop);
434 }
435
436 case keyword::Encoding::RAW:
437 {
438 auto stop = std::next(iterator, dataLength);
439 std::string data(iterator, stop);
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700440 std::string result{};
441 std::for_each(data.cbegin(), data.cend(), [&result](size_t c) {
442 result += toHex(c >> 4);
443 result += toHex(c & 0x0F);
444 });
Deepak Kodihalli4a475bd2016-11-24 07:08:20 -0600445 return result;
446 }
447
George Liuee79ca82019-07-12 11:05:33 +0800448 case keyword::Encoding::MB:
449 {
450 // MB is BuildDate, represent as
451 // 1997-01-01-08:30:00
452 // <year>-<month>-<day>-<hour>:<min>:<sec>
453 auto stop = std::next(iterator, MB_LEN_BYTES);
454 std::string data(iterator, stop);
455 std::string result;
456 result.reserve(MB_LEN_BYTES);
457 auto strItr = data.cbegin();
458 std::advance(strItr, 1);
459 std::for_each(strItr, data.cend(), [&result](size_t c) {
460 result += toHex(c >> 4);
461 result += toHex(c & 0x0F);
462 });
463
464 result.insert(MB_YEAR_END, 1, '-');
465 result.insert(MB_MONTH_END, 1, '-');
466 result.insert(MB_DAY_END, 1, '-');
467 result.insert(MB_HOUR_END, 1, ':');
468 result.insert(MB_MIN_END, 1, ':');
469
470 return result;
471 }
472
Deepak Kodihalli4a475bd2016-11-24 07:08:20 -0600473 case keyword::Encoding::B1:
474 {
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700475 // B1 is MAC address, represent as AA:BB:CC:DD:EE:FF
Deepak Kodihalli4a475bd2016-11-24 07:08:20 -0600476 auto stop = std::next(iterator, MAC_ADDRESS_LEN_BYTES);
477 std::string data(iterator, stop);
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700478 std::string result{};
Deepak Kodihalli4a475bd2016-11-24 07:08:20 -0600479 auto strItr = data.cbegin();
480 size_t firstDigit = *strItr;
481 result += toHex(firstDigit >> 4);
482 result += toHex(firstDigit & 0x0F);
483 std::advance(strItr, 1);
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700484 std::for_each(strItr, data.cend(), [&result](size_t c) {
485 result += ":";
486 result += toHex(c >> 4);
487 result += toHex(c & 0x0F);
488 });
Deepak Kodihalli4a475bd2016-11-24 07:08:20 -0600489 return result;
490 }
491
Dinesh Chinaric576b482017-07-17 16:34:10 -0500492 case keyword::Encoding::UD:
493 {
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700494 // UD, the UUID info, represented as
495 // 123e4567-e89b-12d3-a456-426655440000
Dinesh Chinaric576b482017-07-17 16:34:10 -0500496 //<time_low>-<time_mid>-<time hi and version>
497 //-<clock_seq_hi_and_res clock_seq_low>-<48 bits node id>
498 auto stop = std::next(iterator, UUID_LEN_BYTES);
499 std::string data(iterator, stop);
500 std::string result{};
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700501 std::for_each(data.cbegin(), data.cend(), [&result](size_t c) {
502 result += toHex(c >> 4);
503 result += toHex(c & 0x0F);
504 });
Dinesh Chinaric576b482017-07-17 16:34:10 -0500505 result.insert(UUID_TIME_LOW_END, 1, '-');
506 result.insert(UUID_TIME_MID_END, 1, '-');
507 result.insert(UUID_TIME_HIGH_END, 1, '-');
508 result.insert(UUID_CLK_SEQ_END, 1, '-');
509
510 return result;
511 }
Deepak Kodihalli4a475bd2016-11-24 07:08:20 -0600512 default:
513 break;
514 }
515
516 return {};
517}
518
Deepak Kodihalli683bf722016-11-24 06:50:43 -0600519internal::KeywordMap Impl::readKeywords(Binary::const_iterator iterator)
520{
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700521 internal::KeywordMap map{};
Deepak Kodihalli683bf722016-11-24 06:50:43 -0600522 while (true)
523 {
524 // Note keyword name
525 std::string kw(iterator, iterator + lengths::KW_NAME);
526 if (LAST_KW == kw)
527 {
528 // We're done
529 break;
530 }
Alpana Kumari26a74af2019-09-10 23:53:58 -0500531 // Check if the Keyword is '#kw'
532 char kwNameStart = *iterator;
533
Deepak Kodihalli683bf722016-11-24 06:50:43 -0600534 // Jump past keyword name
535 std::advance(iterator, lengths::KW_NAME);
Alpana Kumari26a74af2019-09-10 23:53:58 -0500536
537 std::size_t length;
538 std::size_t lengthHighByte;
539 if (POUND_KW == kwNameStart)
540 {
541 // Note keyword data length
542 length = *iterator;
543 lengthHighByte = *(iterator + 1);
544 length |= (lengthHighByte << 8);
545
546 // Jump past 2Byte keyword length
547 std::advance(iterator, sizeof(PoundKwSize));
548 }
549 else
550 {
551 // Note keyword data length
552 length = *iterator;
553
554 // Jump past keyword length
555 std::advance(iterator, sizeof(KwSize));
556 }
557
Deepak Kodihalli683bf722016-11-24 06:50:43 -0600558 // Pointing to keyword data now
Alpana Kumari26a74af2019-09-10 23:53:58 -0500559#ifndef IPZ_PARSER
Deepak Kodihalli683bf722016-11-24 06:50:43 -0600560 if (supportedKeywords.end() != supportedKeywords.find(kw))
561 {
562 // Keyword is of interest to us
Patrick Venturec83c4dc2018-11-01 16:29:18 -0700563 std::string data = readKwData((supportedKeywords.find(kw))->second,
564 length, iterator);
Deepak Kodihalli683bf722016-11-24 06:50:43 -0600565 map.emplace(std::move(kw), std::move(data));
566 }
Alpana Kumari26a74af2019-09-10 23:53:58 -0500567
568#else
569 // support all the Keywords
570 auto stop = std::next(iterator, length);
571 std::string kwdata(iterator, stop);
572 map.emplace(std::move(kw), std::move(kwdata));
573
574#endif
Deepak Kodihalli683bf722016-11-24 06:50:43 -0600575 // Jump past keyword data length
576 std::advance(iterator, length);
577 }
578
579 return map;
580}
581
Deepak Kodihalli174caf62016-11-25 05:41:19 -0600582Store Impl::run()
583{
SunnySrivastava1984a20be8e2020-08-26 02:00:50 -0500584 // Check if the VHDR record is present
585 checkHeader();
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500586
SunnySrivastava1984a20be8e2020-08-26 02:00:50 -0500587 auto iterator = vpd.cbegin();
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500588
SunnySrivastava1984a20be8e2020-08-26 02:00:50 -0500589 // Read the table of contents record
590 std::size_t ptLen = readTOC(iterator);
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500591
SunnySrivastava1984a20be8e2020-08-26 02:00:50 -0500592 // Read the table of contents record, to get offsets
593 // to other records.
594 auto offsets = readPT(iterator, ptLen);
595 for (const auto& offset : offsets)
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500596 {
SunnySrivastava1984a20be8e2020-08-26 02:00:50 -0500597 processRecord(offset);
SunnySrivastava19849d4f1122020-08-26 02:41:46 -0500598 }
SunnySrivastava1984a20be8e2020-08-26 02:00:50 -0500599 // Return a Store object, which has interfaces to
600 // access parsed VPD by record:keyword
601 return Store(std::move(out));
Deepak Kodihalli174caf62016-11-25 05:41:19 -0600602}
603
SunnySrivastava1984f6d541e2020-02-04 12:50:40 -0600604void Impl::checkVPDHeader()
605{
606 // Check if the VHDR record is present and is valid
607 checkHeader();
608}
609
Priyanga Ramasamy28079c82021-10-07 16:30:51 -0500610std::string Impl::readKwFromHw(const std::string& record,
611 const std::string& keyword)
612{
613 // Check if the VHDR record is present
614 checkHeader();
615
616 auto iterator = vpd.cbegin();
617
618 // Read the table of contents record
619 std::size_t ptLen = readTOC(iterator);
620
621 // Read the table of contents record, to get offsets
622 // to other records.
623 auto offsets = readPT(iterator, ptLen);
624 for (const auto& offset : offsets)
625 {
626 // Jump to record name
627 auto nameOffset = offset + sizeof(RecordId) + sizeof(RecordSize) +
628 // Skip past the RT keyword, which contains
629 // the record name.
630 lengths::KW_NAME + sizeof(KwSize);
631 // Get record name
632 auto iterator = vpd.cbegin();
633 std::advance(iterator, nameOffset);
634
635 std::string name(iterator, iterator + lengths::RECORD_NAME);
636 if (name != record)
637 {
638 continue;
639 }
640 else
641 {
642 processRecord(offset);
643 const auto& itr = out.find(record);
644 if (itr != out.end())
645 {
646 const auto& kwValItr = (itr->second).find(keyword);
647 if (kwValItr != (itr->second).end())
648 {
649 return kwValItr->second;
650 }
651 else
652 {
653 return "";
654 }
655 }
656 }
657 }
658 return "";
659}
660
Deepak Kodihalli810c9de2016-11-22 11:42:51 -0600661} // namespace parser
662} // namespace vpd
Patrick Williamsc78d8872023-05-10 07:50:56 -0500663} // namespace openpower