xref: /openbmc/pldm/pldmtool/pldm_fru_cmd.cpp (revision 16c2a0a03e5daac77e204eb99e00711490fb6e26)
1 #include "pldm_fru_cmd.hpp"
2 
3 #include "pldm_cmd_helper.hpp"
4 
5 #ifdef OEM_IBM
6 #include <libpldm/oem/ibm/fru.h>
7 #endif
8 
9 #include <endian.h>
10 
11 #include <functional>
12 #include <tuple>
13 
14 namespace pldmtool
15 {
16 
17 namespace fru
18 {
19 
20 namespace
21 {
22 
23 using namespace pldmtool::helper;
24 
25 std::vector<std::unique_ptr<CommandInterface>> commands;
26 
27 } // namespace
28 
29 class GetFruRecordTableMetadata : public CommandInterface
30 {
31   public:
32     ~GetFruRecordTableMetadata() = default;
33     GetFruRecordTableMetadata() = delete;
34     GetFruRecordTableMetadata(const GetFruRecordTableMetadata&) = delete;
35     GetFruRecordTableMetadata(GetFruRecordTableMetadata&&) = default;
36     GetFruRecordTableMetadata&
37         operator=(const GetFruRecordTableMetadata&) = delete;
38     GetFruRecordTableMetadata& operator=(GetFruRecordTableMetadata&&) = delete;
39 
40     using CommandInterface::CommandInterface;
41 
createRequestMsg()42     std::pair<int, std::vector<uint8_t>> createRequestMsg() override
43     {
44         std::vector<uint8_t> requestMsg(sizeof(pldm_msg_hdr));
45         auto request = reinterpret_cast<pldm_msg*>(requestMsg.data());
46 
47         auto rc = encode_get_fru_record_table_metadata_req(
48             instanceId, request, PLDM_GET_FRU_RECORD_TABLE_METADATA_REQ_BYTES);
49         return {rc, requestMsg};
50     }
51 
parseResponseMsg(pldm_msg * responsePtr,size_t payloadLength)52     void parseResponseMsg(pldm_msg* responsePtr, size_t payloadLength) override
53     {
54         uint8_t cc = 0;
55         uint8_t fru_data_major_version, fru_data_minor_version;
56         uint32_t fru_table_maximum_size, fru_table_length;
57         uint16_t total_record_set_identifiers, total_table_records;
58         uint32_t checksum;
59 
60         auto rc = decode_get_fru_record_table_metadata_resp(
61             responsePtr, payloadLength, &cc, &fru_data_major_version,
62             &fru_data_minor_version, &fru_table_maximum_size, &fru_table_length,
63             &total_record_set_identifiers, &total_table_records, &checksum);
64         if (rc != PLDM_SUCCESS || cc != PLDM_SUCCESS)
65         {
66             std::cerr << "Response Message Error: "
67                       << "rc=" << rc << ",cc=" << (int)cc << std::endl;
68             return;
69         }
70         ordered_json output;
71         output["FRUDATAMajorVersion"] =
72             static_cast<uint32_t>(fru_data_major_version);
73         output["FRUDATAMinorVersion"] =
74             static_cast<uint32_t>(fru_data_minor_version);
75         output["FRUTableMaximumSize"] = fru_table_maximum_size;
76         output["FRUTableLength"] = fru_table_length;
77         output["Total number of Record Set Identifiers in table"] =
78             total_record_set_identifiers;
79         output["Total number of records in table"] = total_table_records;
80         output["FRU DATAStructureTableIntegrityChecksum"] = checksum;
81         pldmtool::helper::DisplayInJson(output);
82     }
83 };
84 
85 class FRUTablePrint
86 {
87   public:
FRUTablePrint(const uint8_t * table,size_t table_size)88     explicit FRUTablePrint(const uint8_t* table, size_t table_size) :
89         table(table), table_size(table_size)
90     {}
91 
print()92     void print()
93     {
94         auto p = table;
95         ordered_json frutable;
96         ordered_json output;
97         while (!isTableEnd(p))
98         {
99             auto record =
100                 reinterpret_cast<const pldm_fru_record_data_format*>(p);
101             output["FRU Record Set Identifier"] =
102                 (int)le16toh(record->record_set_id);
103             output["FRU Record Type"] =
104                 typeToString(fruRecordTypes, record->record_type);
105             output["Number of FRU fields"] = (int)record->num_fru_fields;
106             output["Encoding Type for FRU fields"] =
107                 typeToString(fruEncodingType, record->encoding_type);
108 
109             p += sizeof(pldm_fru_record_data_format) -
110                  sizeof(pldm_fru_record_tlv);
111 
112             std::map<uint8_t, std::string> FruFieldTypeMap;
113             std::string fruFieldValue;
114 
115             ordered_json frudata;
116             ordered_json frufielddata;
117             frufielddata.emplace_back(output);
118             for (int i = 0; i < record->num_fru_fields; i++)
119             {
120                 auto tlv = reinterpret_cast<const pldm_fru_record_tlv*>(p);
121                 if (record->record_type == PLDM_FRU_RECORD_TYPE_GENERAL)
122                 {
123                     FruFieldTypeMap.insert(fruGeneralFieldTypes.begin(),
124                                            fruGeneralFieldTypes.end());
125                     if (tlv->type == PLDM_FRU_FIELD_TYPE_IANA)
126                     {
127                         fruFieldValue =
128                             fruFieldParserU32(tlv->value, tlv->length);
129                     }
130                     else if (tlv->type == PLDM_FRU_FIELD_TYPE_MANUFAC_DATE)
131                     {
132                         fruFieldValue =
133                             fruFieldParserTimestamp(tlv->value, tlv->length);
134                     }
135                     else
136                     {
137                         fruFieldValue =
138                             fruFieldValuestring(tlv->value, tlv->length);
139                     }
140 
141                     frudata["FRU Field Type"] =
142                         typeToString(FruFieldTypeMap, tlv->type);
143                     frudata["FRU Field Length"] = (int)(tlv->length);
144                     frudata["FRU Field Value"] = fruFieldValue;
145                     frufielddata.emplace_back(frudata);
146                 }
147                 else
148                 {
149 #ifdef OEM_IBM
150                     if (tlv->type == PLDM_OEM_FRU_FIELD_TYPE_RT)
151                     {
152                         auto oemIPZValue =
153                             fruFieldValuestring(tlv->value, tlv->length);
154 
155                         if (populateMaps.contains(oemIPZValue))
156                         {
157                             const std::map<uint8_t, std::string> IPZTypes =
158                                 populateMaps.at(oemIPZValue);
159                             FruFieldTypeMap.insert(IPZTypes.begin(),
160                                                    IPZTypes.end());
161                         }
162                     }
163                     else
164                     {
165                         FruFieldTypeMap.insert(fruOemFieldTypes.begin(),
166                                                fruOemFieldTypes.end());
167                     }
168                     if (tlv->type == PLDM_OEM_FRU_FIELD_TYPE_IANA)
169                     {
170                         fruFieldValue =
171                             fruFieldParserU32(tlv->value, tlv->length);
172                     }
173                     else if (tlv->type != 2)
174                     {
175                         fruFieldValue =
176                             fruFieldIPZParser(tlv->value, tlv->length);
177                     }
178                     else
179                     {
180                         fruFieldValue =
181                             fruFieldValuestring(tlv->value, tlv->length);
182                     }
183                     frudata["FRU Field Type"] =
184                         typeToString(FruFieldTypeMap, tlv->type);
185                     frudata["FRU Field Length"] = (int)(tlv->length);
186                     frudata["FRU Field Value"] = fruFieldValue;
187                     frufielddata.emplace_back(frudata);
188 
189 #endif
190                 }
191                 p += sizeof(pldm_fru_record_tlv) - 1 + tlv->length;
192             }
193             frutable.emplace_back(frufielddata);
194         }
195         pldmtool::helper::DisplayInJson(frutable);
196     }
197 
198   private:
199     const uint8_t* table;
200     size_t table_size;
201 
isTableEnd(const uint8_t * p)202     bool isTableEnd(const uint8_t* p)
203     {
204         auto offset = p - table;
205         return (table_size - offset) <= 7;
206     }
207 
208     static inline const std::map<uint8_t, std::string> fruEncodingType{
209         {PLDM_FRU_ENCODING_UNSPECIFIED, "Unspecified"},
210         {PLDM_FRU_ENCODING_ASCII, "ASCII"},
211         {PLDM_FRU_ENCODING_UTF8, "UTF8"},
212         {PLDM_FRU_ENCODING_UTF16, "UTF16"},
213         {PLDM_FRU_ENCODING_UTF16LE, "UTF16LE"},
214         {PLDM_FRU_ENCODING_UTF16BE, "UTF16BE"}};
215 
216     static inline const std::map<uint8_t, std::string> fruGeneralFieldTypes{
217         {PLDM_FRU_FIELD_TYPE_CHASSIS, "Chassis"},
218         {PLDM_FRU_FIELD_TYPE_MODEL, "Model"},
219         {PLDM_FRU_FIELD_TYPE_PN, "Part Number"},
220         {PLDM_FRU_FIELD_TYPE_SN, "Serial Number"},
221         {PLDM_FRU_FIELD_TYPE_MANUFAC, "Manufacturer"},
222         {PLDM_FRU_FIELD_TYPE_MANUFAC_DATE, "Manufacture Date"},
223         {PLDM_FRU_FIELD_TYPE_VENDOR, "Vendor"},
224         {PLDM_FRU_FIELD_TYPE_NAME, "Name"},
225         {PLDM_FRU_FIELD_TYPE_SKU, "SKU"},
226         {PLDM_FRU_FIELD_TYPE_VERSION, "Version"},
227         {PLDM_FRU_FIELD_TYPE_ASSET_TAG, "Asset Tag"},
228         {PLDM_FRU_FIELD_TYPE_DESC, "Description"},
229         {PLDM_FRU_FIELD_TYPE_EC_LVL, "Engineering Change Level"},
230         {PLDM_FRU_FIELD_TYPE_OTHER, "Other Information"},
231         {PLDM_FRU_FIELD_TYPE_IANA, "Vendor IANA"}};
232 
233     static inline const std::map<uint8_t, std::string> fruRecordTypes{
234         {PLDM_FRU_RECORD_TYPE_GENERAL, "General"},
235         {PLDM_FRU_RECORD_TYPE_OEM, "OEM"}};
236 
237 #ifdef OEM_IBM
238     static inline const std::map<uint8_t, std::string> fruOemFieldTypes{
239         {PLDM_OEM_FRU_FIELD_TYPE_IANA, "Vendor IANA"},
240         {PLDM_OEM_FRU_FIELD_TYPE_RT, "RT"},
241         {PLDM_OEM_FRU_FIELD_TYPE_LOCATION_CODE, "Location Code"}};
242 
243     static inline const std::map<uint8_t, std::string> VINIFieldTypes{
244         {2, "RT"},  {3, "B3"},  {4, "B4"},  {5, "B7"},  {6, "CC"},  {7, "CE"},
245         {8, "CT"},  {9, "DR"},  {10, "FG"}, {11, "FN"}, {12, "HE"}, {13, "HW"},
246         {14, "HX"}, {15, "PN"}, {16, "SN"}, {17, "TS"}, {18, "VZ"}};
247 
248     static inline const std::map<uint8_t, std::string> VSYSFieldTypes{
249         {2, "RT"},  {3, "BR"},  {4, "DR"},  {5, "FV"},  {6, "ID"},
250         {7, "MN"},  {8, "NN"},  {9, "RB"},  {10, "RG"}, {11, "SE"},
251         {12, "SG"}, {13, "SU"}, {14, "TM"}, {15, "TN"}, {16, "WN"}};
252 
253     static inline const std::map<uint8_t, std::string> LXR0FieldTypes{
254         {2, "RT"}, {3, "LX"}, {4, "VZ"}};
255 
256     static inline const std::map<uint8_t, std::string> VW10FieldTypes{
257         {2, "RT"}, {3, "DR"}, {4, "GD"}};
258 
259     static inline const std::map<uint8_t, std::string> VR10FieldTypes{
260         {2, "RT"}, {3, "DC"}, {4, "DR"}, {5, "FL"}, {6, "WA"}};
261 
262     static inline const std::map<std::string,
263                                  const std::map<uint8_t, std::string>>
264         populateMaps{{"VINI", VINIFieldTypes},
265                      {"VSYS", VSYSFieldTypes},
266                      {"LXR0", LXR0FieldTypes},
267                      {"VWX10", VW10FieldTypes},
268                      {"VR10", VR10FieldTypes}};
269 #endif
270 
typeToString(std::map<uint8_t,std::string> typeMap,uint8_t type)271     std::string typeToString(std::map<uint8_t, std::string> typeMap,
272                              uint8_t type)
273     {
274         auto typeString = std::to_string(type);
275         try
276         {
277             return std::string(typeMap.at(type)) + "(" + typeString + ")";
278         }
279         catch (const std::out_of_range& e)
280         {
281             return typeString;
282         }
283     }
284 
fruFieldValuestring(const uint8_t * value,uint8_t length)285     std::string fruFieldValuestring(const uint8_t* value, uint8_t length)
286     {
287         return std::string(reinterpret_cast<const char*>(value), length);
288     }
289 
fruFieldParserU32(const uint8_t * value,uint8_t length)290     static std::string fruFieldParserU32(const uint8_t* value, uint8_t length)
291     {
292         assert(length == 4);
293         uint32_t v;
294         std::memcpy(&v, value, length);
295         return std::to_string(le32toh(v));
296     }
297 
fruFieldParserTimestamp(const uint8_t *,uint8_t)298     static std::string fruFieldParserTimestamp(const uint8_t*, uint8_t)
299     {
300         return std::string("TODO");
301     }
302 
fruFieldIPZParser(const uint8_t * value,uint8_t length)303     static std::string fruFieldIPZParser(const uint8_t* value, uint8_t length)
304     {
305         std::ostringstream tempStream;
306         for (int i = 0; i < int(length); ++i)
307         {
308             tempStream << "0x" << std::setfill('0') << std::setw(2) << std::hex
309                        << (unsigned)value[i] << " ";
310         }
311         return tempStream.str();
312     }
313 };
314 
315 class GetFRURecordByOption : public CommandInterface
316 {
317   public:
318     ~GetFRURecordByOption() = default;
319     GetFRURecordByOption() = delete;
320     GetFRURecordByOption(const GetFRURecordByOption&) = delete;
321     GetFRURecordByOption(GetFruRecordTableMetadata&&) = delete;
322     GetFRURecordByOption& operator=(const GetFRURecordByOption&) = delete;
323     GetFRURecordByOption& operator=(GetFRURecordByOption&&) = delete;
324 
GetFRURecordByOption(const char * type,const char * name,CLI::App * app)325     explicit GetFRURecordByOption(const char* type, const char* name,
326                                   CLI::App* app) :
327         CommandInterface(type, name, app)
328     {
329         app->add_option("-i, --identifier", recordSetIdentifier,
330                         "Record Set Identifier\n"
331                         "Possible values: {All record sets = 0, Specific "
332                         "record set = 1 – 65535}")
333             ->required();
334         app->add_option("-r, --record", recordType,
335                         "Record Type\n"
336                         "Possible values: {All record types = 0, Specific "
337                         "record types = 1 – 255}")
338             ->required();
339         app->add_option("-f, --field", fieldType,
340                         "Field Type\n"
341                         "Possible values: {All record field types = 0, "
342                         "Specific field types = 1 – 15}")
343             ->required();
344     }
345 
createRequestMsg()346     std::pair<int, std::vector<uint8_t>> createRequestMsg() override
347     {
348         if (fieldType != 0 && recordType == 0)
349         {
350             throw std::invalid_argument("if field type is non-zero, the record "
351                                         "type shall also be non-zero");
352         }
353         if (recordType == 254 && (fieldType > 2 && fieldType < 254))
354         {
355             throw std::invalid_argument(
356                 "GetFRURecordByOption is not supported for recordType : 254 "
357                 "and fieldType > 2");
358         }
359 
360         auto payloadLength = sizeof(pldm_get_fru_record_by_option_req);
361 
362         std::vector<uint8_t> requestMsg(sizeof(pldm_msg_hdr) + payloadLength,
363                                         0);
364         auto reqMsg = reinterpret_cast<pldm_msg*>(requestMsg.data());
365 
366         auto rc = encode_get_fru_record_by_option_req(
367             instanceId, 0 /* DataTransferHandle */, 0 /* FRUTableHandle */,
368             recordSetIdentifier, recordType, fieldType, PLDM_GET_FIRSTPART,
369             reqMsg, payloadLength);
370 
371         return {rc, requestMsg};
372     }
373 
parseResponseMsg(pldm_msg * responsePtr,size_t payloadLength)374     void parseResponseMsg(pldm_msg* responsePtr, size_t payloadLength) override
375     {
376         uint8_t cc;
377         uint32_t dataTransferHandle;
378         uint8_t transferFlag;
379         variable_field fruData;
380 
381         auto rc = decode_get_fru_record_by_option_resp(
382             responsePtr, payloadLength, &cc, &dataTransferHandle, &transferFlag,
383             &fruData);
384 
385         if (rc != PLDM_SUCCESS || cc != PLDM_SUCCESS)
386         {
387             std::cerr << "Response Message Error: "
388                       << "rc=" << rc << ",cc=" << (int)cc << std::endl;
389             return;
390         }
391 
392         FRUTablePrint tablePrint(fruData.ptr, fruData.length);
393         tablePrint.print();
394     }
395 
396   private:
397     uint16_t recordSetIdentifier;
398     uint8_t recordType;
399     uint8_t fieldType;
400 };
401 
402 class GetFruRecordTable : public CommandInterface
403 {
404   public:
405     ~GetFruRecordTable() = default;
406     GetFruRecordTable() = delete;
407     GetFruRecordTable(const GetFruRecordTable&) = delete;
408     GetFruRecordTable(GetFruRecordTable&&) = default;
409     GetFruRecordTable& operator=(const GetFruRecordTable&) = delete;
410     GetFruRecordTable& operator=(GetFruRecordTable&&) = delete;
411 
412     using CommandInterface::CommandInterface;
createRequestMsg()413     std::pair<int, std::vector<uint8_t>> createRequestMsg() override
414     {
415         std::vector<uint8_t> requestMsg(
416             sizeof(pldm_msg_hdr) + PLDM_GET_FRU_RECORD_TABLE_REQ_BYTES);
417         auto request = reinterpret_cast<pldm_msg*>(requestMsg.data());
418 
419         auto rc = encode_get_fru_record_table_req(
420             instanceId, 0, PLDM_GET_FIRSTPART, request,
421             requestMsg.size() - sizeof(pldm_msg_hdr));
422         return {rc, requestMsg};
423     }
parseResponseMsg(pldm_msg * responsePtr,size_t payloadLength)424     void parseResponseMsg(pldm_msg* responsePtr, size_t payloadLength) override
425     {
426         uint8_t cc = 0;
427         uint32_t next_data_transfer_handle = 0;
428         uint8_t transfer_flag = 0;
429         size_t fru_record_table_length = 0;
430         std::vector<uint8_t> fru_record_table_data(payloadLength);
431 
432         auto rc = decode_get_fru_record_table_resp(
433             responsePtr, payloadLength, &cc, &next_data_transfer_handle,
434             &transfer_flag, fru_record_table_data.data(),
435             &fru_record_table_length);
436 
437         if (rc != PLDM_SUCCESS || cc != PLDM_SUCCESS)
438         {
439             std::cerr << "Response Message Error: "
440                       << "rc=" << rc << ",cc=" << (int)cc << std::endl;
441             return;
442         }
443 
444         FRUTablePrint tablePrint(fru_record_table_data.data(),
445                                  fru_record_table_length);
446         tablePrint.print();
447     }
448 };
449 
registerCommand(CLI::App & app)450 void registerCommand(CLI::App& app)
451 {
452     auto fru = app.add_subcommand("fru", "FRU type command");
453     fru->require_subcommand(1);
454     auto getFruRecordTableMetadata = fru->add_subcommand(
455         "GetFruRecordTableMetadata", "get FRU record table metadata");
456     commands.push_back(std::make_unique<GetFruRecordTableMetadata>(
457         "fru", "GetFruRecordTableMetadata", getFruRecordTableMetadata));
458 
459     auto getFRURecordByOption =
460         fru->add_subcommand("GetFRURecordByOption", "get FRU Record By Option");
461     commands.push_back(std::make_unique<GetFRURecordByOption>(
462         "fru", "GetFRURecordByOption", getFRURecordByOption));
463 
464     auto getFruRecordTable =
465         fru->add_subcommand("GetFruRecordTable", "get FRU Record Table");
466     commands.push_back(std::make_unique<GetFruRecordTable>(
467         "fru", "GetFruRecordTable", getFruRecordTable));
468 }
469 
470 } // namespace fru
471 
472 } // namespace pldmtool
473