blob: f6552a59cf2ce3b07a3f7ce2a95c1f3e80b9e33a [file] [log] [blame]
Patrick Ventureab296412020-12-30 13:39:37 -08001/*
2// Copyright (c) 2018 Intel 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*/
Brad Bishope45d8c72022-05-25 15:12:53 -040016/// \file fru_utils.hpp
Patrick Ventureab296412020-12-30 13:39:37 -080017
18#pragma once
Zev Weiss309c0b12022-02-25 01:44:12 +000019#include "fru_reader.hpp"
Ed Tanous3013fb42022-07-09 08:27:06 -070020
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053021#include <boost/container/flat_map.hpp>
Kumar Thangaveld79d0252022-08-24 14:26:01 +053022#include <sdbusplus/asio/object_server.hpp>
Patrick Ventureab296412020-12-30 13:39:37 -080023
24#include <cstdint>
25#include <functional>
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053026#include <regex>
Patrick Ventureab296412020-12-30 13:39:37 -080027#include <string>
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053028#include <utility>
Patrick Ventureab296412020-12-30 13:39:37 -080029#include <vector>
Patrick Ventureab296412020-12-30 13:39:37 -080030extern "C"
31{
32// Include for I2C_SMBUS_BLOCK_MAX
33#include <linux/i2c.h>
34}
35
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053036constexpr size_t fruBlockSize = 8;
37
Kumar Thangavelc74e7512022-02-03 22:53:05 +053038using DeviceMap = boost::container::flat_map<int, std::vector<uint8_t>>;
39using BusMap = boost::container::flat_map<int, std::shared_ptr<DeviceMap>>;
40
Ed Tanousfc171422024-04-04 17:18:16 -070041// NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
Kumar Thangavelc74e7512022-02-03 22:53:05 +053042inline BusMap busMap;
43
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053044enum class DecodeState
45{
46 ok,
47 end,
48 err,
49};
50
51enum class resCodes
52{
53 resOK,
54 resWarn,
55 resErr
56};
57
Patrick Ventureab296412020-12-30 13:39:37 -080058enum class fruAreas
59{
60 fruAreaInternal = 0,
61 fruAreaChassis,
62 fruAreaBoard,
63 fruAreaProduct,
64 fruAreaMultirecord
65};
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053066
Kumar Thangavelbdfc5ec2022-08-29 22:23:00 +053067struct FruArea
68{
69 size_t start; // Fru Area Start offset
70 size_t size; // Fru Area Size
71 size_t end; // Fru Area end offset
72 size_t updateFieldLoc; // Fru Area update Field Location
Kumar Thangavel51b557b2022-09-13 13:40:47 +053073 size_t restFieldsLoc; // Starting location of restFRUArea data
74 size_t restFieldsEnd; // Ending location of restFRUArea data
Kumar Thangavelbdfc5ec2022-08-29 22:23:00 +053075};
76
Ed Tanous07d467b2021-02-23 14:48:37 -080077const std::vector<std::string> fruAreaNames = {"INTERNAL", "CHASSIS", "BOARD",
78 "PRODUCT", "MULTIRECORD"};
79const std::regex nonAsciiRegex("[^\x01-\x7f]");
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053080
Ed Tanous07d467b2021-02-23 14:48:37 -080081const std::vector<std::string> chassisFruAreas = {"PART_NUMBER",
82 "SERIAL_NUMBER"};
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053083
Ed Tanous07d467b2021-02-23 14:48:37 -080084const std::vector<std::string> boardFruAreas = {"MANUFACTURER", "PRODUCT_NAME",
85 "SERIAL_NUMBER", "PART_NUMBER",
86 "FRU_VERSION_ID"};
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053087
Ed Tanous07d467b2021-02-23 14:48:37 -080088const std::vector<std::string> productFruAreas = {
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053089 "MANUFACTURER", "PRODUCT_NAME", "PART_NUMBER", "VERSION",
90 "SERIAL_NUMBER", "ASSET_TAG", "FRU_VERSION_ID"};
91
Ed Tanous07d467b2021-02-23 14:48:37 -080092const std::string fruCustomFieldName = "INFO_AM";
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +053093
Patrick Ventureab296412020-12-30 13:39:37 -080094inline fruAreas operator++(fruAreas& x)
95{
96 return x = static_cast<fruAreas>(std::underlying_type<fruAreas>::type(x) +
97 1);
98}
99
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +0530100inline const std::string& getFruAreaName(fruAreas area)
101{
Ed Tanous07d467b2021-02-23 14:48:37 -0800102 return fruAreaNames[static_cast<unsigned int>(area)];
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +0530103}
104
Delphine CC Chiua3ca14a2024-03-27 17:02:24 +0800105std::tm intelEpoch();
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +0530106
107char sixBitToChar(uint8_t val);
108
109/* 0xd - 0xf are reserved values, but not fatal; use a placeholder char. */
Ed Tanous3013fb42022-07-09 08:27:06 -0700110constexpr std::array<char, 6> bcdHighChars = {
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +0530111 ' ', '-', '.', 'X', 'X', 'X',
112};
113
114char bcdPlusToChar(uint8_t val);
115
116bool verifyOffset(const std::vector<uint8_t>& fruBytes, fruAreas currentArea,
117 uint8_t len);
118
119std::pair<DecodeState, std::string>
120 decodeFRUData(std::vector<uint8_t>::const_iterator& iter,
121 const std::vector<uint8_t>::const_iterator& end,
122 bool isLangEng);
123
124bool checkLangEng(uint8_t lang);
125
126resCodes
Michael Shen0961b112022-02-22 11:06:33 +0800127 formatIPMIFRU(const std::vector<uint8_t>& fruBytes,
Andrew Jefferyf8ae2ba2022-03-25 15:13:55 +1030128 boost::container::flat_map<std::string, std::string>& result);
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +0530129
krishnar4213ee212022-11-11 15:39:30 +0530130std::vector<uint8_t>& getFRUInfo(const uint16_t& bus, const uint8_t& address);
Kumar Thangavelc8dc4af2021-01-12 10:36:38 +0530131
132uint8_t calculateChecksum(std::vector<uint8_t>::const_iterator iter,
133 std::vector<uint8_t>::const_iterator end);
134
135uint8_t calculateChecksum(std::vector<uint8_t>& fruAreaData);
136
137unsigned int updateFRUAreaLenAndChecksum(std::vector<uint8_t>& fruData,
138 size_t fruAreaStart,
139 size_t fruAreaEndOfFieldsOffset,
140 size_t fruAreaEndOffset);
141
142ssize_t getFieldLength(uint8_t fruFieldTypeLenValue);
143
Oskar Senftbd4075f2021-10-05 23:42:43 -0400144/// \brief Find a FRU header.
Zev Weiss309c0b12022-02-25 01:44:12 +0000145/// \param reader the FRUReader to read via
Oskar Senftbd4075f2021-10-05 23:42:43 -0400146/// \param errorHelp and a helper string for failures
147/// \param blockData buffer to return the last read block
148/// \param baseOffset the offset to start the search at;
149/// set to 0 to perform search;
150/// returns the offset at which a header was found
151/// \return whether a header was found
Zev Weiss309c0b12022-02-25 01:44:12 +0000152bool findFRUHeader(FRUReader& reader, const std::string& errorHelp,
Oskar Senftbd4075f2021-10-05 23:42:43 -0400153 std::array<uint8_t, I2C_SMBUS_BLOCK_MAX>& blockData,
Zev Weiss1525e852022-03-22 22:27:43 +0000154 off_t& baseOffset);
Oskar Senftbd4075f2021-10-05 23:42:43 -0400155
Patrick Ventureab296412020-12-30 13:39:37 -0800156/// \brief Read and validate FRU contents.
Zev Weiss309c0b12022-02-25 01:44:12 +0000157/// \param reader the FRUReader to read via
Patrick Ventureab296412020-12-30 13:39:37 -0800158/// \param errorHelp and a helper string for failures
Marvin Drees2b3ed302023-04-14 16:35:14 +0200159/// \return the FRU contents from the file and bool indicating if the FRU Header
160/// was found
161std::pair<std::vector<uint8_t>, bool>
162 readFRUContents(FRUReader& reader, const std::string& errorHelp);
Patrick Ventureab296412020-12-30 13:39:37 -0800163
164/// \brief Validate an IPMI FRU common header
165/// \param blockData the bytes comprising the common header
166/// \return true if valid
167bool validateHeader(const std::array<uint8_t, I2C_SMBUS_BLOCK_MAX>& blockData);
168
169/// \brief Get offset for a common header area
170/// \param area - the area
171/// \return the field offset
172unsigned int getHeaderAreaFieldOffset(fruAreas area);
Kumar Thangavelbdfc5ec2022-08-29 22:23:00 +0530173
174/// \brief Iterate fruArea Names and find offset/location and fields and size of
175/// properties
176/// \param fruData - vector to store fru data
177/// \param propertyName - fru property Name
Kumar Thangavel51b557b2022-09-13 13:40:47 +0530178/// \param fruAreaParams - struct to have fru Area parameters like length,
179/// size.
180/// \return true if fru field is found, fruAreaParams like updateFieldLoc,
181/// Start, Size, End are updated with fruArea and field info.
Kumar Thangavelbdfc5ec2022-08-29 22:23:00 +0530182bool findFruAreaLocationAndField(std::vector<uint8_t>& fruData,
183 const std::string& propertyName,
Kumar Thangavel51b557b2022-09-13 13:40:47 +0530184 struct FruArea& fruAreaParams);
185
186/// \brief Copy the fru Area fields and properties into restFRUAreaFieldsData.
187/// restFRUAreaField is the rest of the fields in FRU area after the field that
188/// is being updated.
189/// \param fruData - vector to store fru data
190/// \param propertyName - fru property Name
191/// \param fruAreaParams - struct to have fru Area parameters like length
192/// \param restFRUAreaFieldsData - vector to store fru Area Fields and
193/// properties.
194/// \return true on success false on failure. restFieldLoc and restFieldEnd
195/// are updated.
Kumar Thangavel9f2162a2022-08-10 18:05:20 +0530196
Kumar Thangavel51b557b2022-09-13 13:40:47 +0530197bool copyRestFRUArea(std::vector<uint8_t>& fruData,
198 const std::string& propertyName,
199 struct FruArea& fruAreaParams,
200 std::vector<uint8_t>& restFRUAreaFieldsData);
Kumar Thangaveld79d0252022-08-24 14:26:01 +0530201
202/// \brief Get all device dbus path and match path with product name using
203/// regular expression and find the device index for all devices.
204/// \param dbusInterfaceMap - Map to store fru device dbus path and interface
205/// \param productName - fru device product name.
206/// \return optional<int> highest index for fru device on success, return
207/// nullopt on failure.
208std::optional<int> findIndexForFRU(
209 boost::container::flat_map<
210 std::pair<size_t, size_t>,
211 std::shared_ptr<sdbusplus::asio::dbus_interface>>& dbusInterfaceMap,
212 std::string& productName);
Kumar Thangavel9f2162a2022-08-10 18:05:20 +0530213
214/// \brief It does format fru data and find productName in the formatted
215/// fru data and return productName.
216/// \param device - vector that contains device list
217/// \param formattedFRU - map that contains formatted FRU data
218/// \param bus - bus number of the device
219/// \param address - address of the device
220/// \param unknownBusObjectCount - Unknown Bus object counter variable
221/// \return optional string. it returns productName or NULL
222
223std::optional<std::string> getProductName(
224 std::vector<uint8_t>& device,
225 boost::container::flat_map<std::string, std::string>& formattedFRU,
226 uint32_t bus, uint32_t address, size_t& unknownBusObjectCount);
Kumar Thangavel9d6f5902022-08-26 16:52:14 +0530227
228bool getFruData(std::vector<uint8_t>& fruData, uint32_t bus, uint32_t address);